#node.js #docker
#node.js #docker
Вопрос:
У меня есть следующее приложение express / node:
FROM node:12
WORKDIR /usr/share/jtapp
COPY package*.json ./
RUN apt-get update
RUN apt-get install -y net-tools
RUN npm install
RUN npm install pm2@latest -g
#RUN npm ci --only=production
COPY . .
EXPOSE 3000
RUN pm2 start server.js
Кажется, что сборка выполняется нормально. Но когда я запускаю изображение и прикрепляю через bash, процессы pm2 не выполняются.
Вот результаты сборки:
Sending build context to Docker daemon 319.5kB Step 1/10 : FROM node:12 ---> cfcf3e70099d Step 2/10 : WORKDIR /usr/share/jtapp ---> Using cache ---> 2c2e2e1e92f2 Step 3/10 : COPY package*.json ./ ---> Using cache ---> 57829fd5e9d7 Step 4/10 : RUN apt-get update ---> Using cache ---> c0bbfed43ca8 Step 5/10 : RUN apt-get install -y net-tools ---> Using cache ---> 0b34759d8298 Step 6/10 : RUN npm install ---> Using cache ---> 6abc42a9f8a3 Step 7/10 : RUN npm install pm2@latest -g ---> Running in 2fafacd302bf /usr/local/bin/pm2 -> /usr/local/lib/node_modules/pm2/bin/pm2 /usr/local/bin/pm2-dev -> /usr/local/lib/node_modules/pm2/bin/pm2-dev /usr/local/bin/pm2-docker -> /usr/local/lib/node_modules/pm2/bin/pm2-docker /usr/local/bin/pm2-runtime -> /usr/local/lib/node_modules/pm2/bin/pm2-runtime
npm WARN optional SKIPPING OPTIONAL DEPENDENCY: fsevents@~2.1.2 (node_modules/pm2/node_modules/chokidar/node_modules/fsevents):
npm WARN notsup SKIPPING OPTIONAL DEPENDENCY: Unsupported platform for fsevents@2.1.3: wanted {"os":"darwin","arch":"any"} (current: {"os":"linux","arch":"x64"})
pm2@4.4.1
added 185 packages from 191 contributors in 10.165s
Removing intermediate container 2fafacd302bf
---> 11d15b993ca2
Step 8/10 : COPY . .
---> efab8283bbd9
Step 9/10 : EXPOSE 3000
---> Running in aae343a75727
Removing intermediate container aae343a75727
---> b97825863317
Step 10/10 : RUN pm2 start server.js
---> Running in f0d14c61a3a6
[PM2] Spawning PM2 daemon with pm2_home=/root/.pm2
[PM2] PM2 Successfully daemonized
[PM2] Starting /usr/share/jtapp/server.js in fork_mode (1 instance)
[PM2] Done.
┌─────┬───────────┬─────────────┬─────────┬─────────┬──────────┬────────┬──────┬───────────┬──────────┬──────────┬──────────┬──────────┐
│ id │ name │ namespace │ version │ mode │ pid │ uptime │ ↺ │ status │ cpu │ mem │ user │ watching │
├─────┼───────────┼─────────────┼─────────┼─────────┼──────────┼────────┼──────┼───────────┼──────────┼──────────┼──────────┼──────────┤
│ 0 │ server │ default │ 1.0.0 │ fork │ 28 │ 0s │ 0 │ online │ 0% │ 28.2mb │ root │ disabled │
└─────┴───────────┴─────────────┴─────────┴─────────┴──────────┴────────┴──────┴───────────┴──────────┴──────────┴──────────┴──────────┘
Removing intermediate container f0d14c61a3a6
---> 859f8479f3a6
Successfully built 859f8479f3a6
Затем вот как я запускаю контейнер и подключаюсь к нему:
lab:/var/lib/git/jtapp# docker run -d -it --name jtapp -p 3000:3000 jtapp bash
d62927716ab593376c4e38f7d4d05007223ce2adfcb4997e722b81ad7260f114
lab:/var/lib/git/jtapp # docker exec -it jtapp bash
root@d62927716ab5:/usr/share/jtapp # pm2
usage: pm2 [options] <command>
pm2 -h, --help all available commands and options
pm2 examples display pm2 usage examples
pm2 <command> -h help on a specific command
Access pm2 files in ~/.pm2
root@d62927716ab5:/usr/share/jtapp #
Но я могу запустить все вручную, как только окажусь внутри контейнера, вот так:
root@d62927716ab5:/usr/share/jtapp# pm2 start server.js
[PM2] Spawning PM2 daemon with pm2_home=/root/.pm2
[PM2] PM2 Successfully daemonized
[PM2] Starting /usr/share/jtapp/server.js in fork_mode (1 instance)
[PM2] Done.
┌─────┬───────────┬─────────────┬─────────┬─────────┬──────────┬────────┬──────┬───────────┬──────────┬──────────┬──────────┬──────────┐
│ id │ name │ namespace │ version │ mode │ pid │ uptime │ ↺ │ status │ cpu │ mem │ user │ watching │
├─────┼───────────┼─────────────┼─────────┼─────────┼──────────┼────────┼──────┼───────────┼──────────┼──────────┼──────────┼──────────┤
│ 0 │ server │ default │ 1.0.0 │ fork │ 37 │ 0s │ 0 │ online │ 0% │ 28.4mb │ root │ disabled │
└─────┴───────────┴─────────────┴─────────┴─────────┴──────────┴────────┴──────┴───────────┴──────────┴──────────┴──────────┴──────────┘
РЕДАКТИРОВАТЬ 1
Изменено:
RUN pm2 start server.js
Для
CMD ["pm2", "start", "server.js"]
А затем попытался перестроить / запустить контейнер.
когда я выполняю «pm2 ls», он по-прежнему возвращается пустым.
root@a8b7eb21d307:/usr/share/jtapp# pm2 ls
[PM2] Spawning PM2 daemon with pm2_home=/root/.pm2
[PM2] PM2 Successfully daemonized
┌────┬────────────────────┬──────────┬──────┬───────────┬──────────┬──────────┐
│ id │ name │ mode │ ↺ │ status │ cpu │ memory │
└────┴────────────────────┴──────────┴──────┴───────────┴──────────┴──────────┘
EDIT 2
When i remove the «-d» option in my run command:
lab:/var/lib/git/jtappd# docker run -it --name jtapp -p 3000:3000 jtapp bash
root@3cf4855372ec:/usr/share/jtapp# pm2 ls
[PM2] Spawning PM2 daemon with pm2_home=/root/.pm2
[PM2] PM2 Successfully daemonized
┌─────┬───────────┬─────────────┬─────────┬─────────┬──────────┬────────┬──────┬───────────┬──────────┬──────────┬──────────┬──────────┐
│ id │ name │ namespace │ version │ mode │ pid │ uptime │ ↺ │ status │ cpu │ mem │ user │ watching │
└─────┴───────────┴─────────────┴─────────┴─────────┴──────────┴────────┴──────┴───────────┴──────────┴──────────┴──────────┴──────────┘
root@3cf4855372ec:/usr/share/jtapp#
ПРАВКА 3
Итак, файл Dockerfile теперь выглядит так:
FROM node:12
WORKDIR /usr/share/jtapp
COPY package*.json ./
RUN apt-get update
RUN apt-get install -y net-tools
RUN npm install
RUN npm install pm2@latest -g
COPY . .
EXPOSE 3000
CMD ["npm", "run", "start:pm2"]
ecosystem.config.js выглядит так:
lab:/var/lib/git/jtapp# cat ecosystem.config.js
module.exports = {
apps: [{
name: 'jtapp',
script: './server.js', //entrypoint
instances: 1,
autorestart: true, // pm2 to restart if this app fails/stops
max_memory_restart: '1G'
}]
}
и раздел скрипта package.json:
1 {
2 "name": "nodeserver",
3 "version": "1.0.0",
4 "description": "simple node server",
5 "main": "index.js",
6 "scripts": {
7 "test": "echo "Error: no test specified" amp;amp; exit 1",
8 "start": "nodemon server.js",
9 "start:pm2": "pm2 start ecosystem.config.js --no-daemon"
10 },
Файл Dockerfile создается без ошибок. Но когда я запускаю контейнер… server.js не выполняется. pm2 установлен.
Но нет проблем с запуском его вручную с использованием файла экосистемы, как только я войду в контейнер, как вы можете видеть ниже:
root@d51117d757a3:/usr/share/jtapp# pm2 start ecosystem.config.js
[PM2][WARN] Applications jtapp not running, starting...
[PM2] App [jtapp ] launched (1 instances)
┌─────┬──────────┬─────────────┬─────────┬─────────┬──────────┬────────┬──────┬───────────┬──────────┬──────────┬──────────┬──────────┐
│ id │ name │ namespace │ version │ mode │ pid │ uptime │ ↺ │ status │ cpu │ mem │ user │ watching │
├─────┼──────────┼─────────────┼─────────┼─────────┼──────────┼────────┼──────┼───────────┼──────────┼──────────┼──────────┼──────────┤
│ 0 │ jtapp │ default │ 1.0.0 │ cluster │ 36 │ 0s │ 0 │ online │ 0% │ 32.6mb │ root │ disabled │
└─────┴──────────┴─────────────┴─────────┴─────────┴──────────┴────────┴──────┴───────────┴──────────┴──────────┴──────────┴──────────┘
Ответ №1:
Лучшим подходом было бы использование ecosystem.config.js
. Поместите этот файл в корневой каталог вашего проекта:
module.exports = {
apps: [{
name: 'app',
script: 'src/index.js', // Your entry point
instances: 1,
autorestart: true, // THIS is the important part, this will tell PM2 to restart your app if it falls over
max_memory_restart: '1G'
}]
}
затем установите pm2 в качестве модуля npm в вашем проекте и добавьте этот скрипт в свой package.json
...
scripts: {
...
"start:pm2":"pm2 start ecosystem.config.js --no-daemon"
}
и, наконец, используйте этот CMD в вашем dockerfile
CMD [ "npm", "run", "start:pm2" ]
РЕДАКТИРОВАТЬ 1
здесь вы можете найти пример проекта с оригинальным файлом dockerfile:
https://github.com/golkhandani/pm2_docker/tree/master
ПРАВКА 2
Я только что протестировал репозиторий с другим Manjaro
Linux, и вот команды
git clone https://github.com/golkhandani/pm2_docker.git
docker build pm2_docker/ --tag pm2docker:test
docker run -p 3000:3000 -t pm2docker:test
-p 3000:3000
аргумент должен быть добавлен для локального доступа к вашему серверу.
это дополнительные скриншоты, показывающие команду и результаты:
Комментарии:
1. действительно ли нужен «path»? Я не вижу, чтобы оно использовалось? Внедрю это и посмотрю, работает ли это. Спасибо.
2. извините, я обновлю свой ответ. на самом деле он будет использоваться для настройки скрипта, но вы можете создать путь к скрипту самостоятельно
3. итак, я попробовал это, и он по-прежнему не запускает pm2. обновляю свой пост с подробностями. Проверьте «РЕДАКТИРОВАТЬ 3» через несколько минут.
4. Я обновил свой ответ примером проекта
5. Удалите глобальную установку pm2 и предоставьте порт при запуске docker, предоставив мне работающий сервер