Что такое Windows-эквивалент process.on ('SIGINT') в node.js?


85

Я следую приведенным здесь инструкциям (прослушивание SIGINTсобытий), чтобы корректно завершить работу моего приложения node.js, размещенного в Windows 8, в ответ на Ctrl+ Cили завершение работы сервера.

Но в Windows этого нет SIGINT. Я тоже пробовал process.on('exit'), но, похоже, уже поздно делать что-то продуктивное.

В Windows этот код дает мне: Ошибка: такого модуля нет

process.on( 'SIGINT', function() {
  console.log( "\ngracefully shutting down from  SIGINT (Crtl-C)" )
  // wish this worked on Windows
  process.exit( )
})

В Windows этот код работает, но уже слишком поздно, чтобы сделать что-нибудь изящное :

process.on( 'exit', function() {
  console.log( "never see this log message" )
})

Есть ли SIGINTаналогичное событие в Windows?


Эта проблема случайно возникла у меня сегодня, и я думаю, что она как-то связана с самим модулем readline. Я не мог провести никакого тестирования, но с тех пор, как я добавил этот модуль, у меня начались проблемы.
Sv443 01

Ответы:


151

Вы должны использовать модуль readline и прослушивать событие SIGINT:

http://nodejs.org/api/readline.html#readline_event_sigint

if (process.platform === "win32") {
  var rl = require("readline").createInterface({
    input: process.stdin,
    output: process.stdout
  });

  rl.on("SIGINT", function () {
    process.emit("SIGINT");
  });
}

process.on("SIGINT", function () {
  //graceful shutdown
  process.exit();
});

отличное обновление, спасибо! установка этого в качестве ответа, поскольку предыдущий ответ (прослушивание нажатия клавиш) больше не работает.
pappadog

45
Это нелепо. Почему это не обрабатывается ядром узла?
Balupton

3
Потому что, когда вы слушаете stdin, процесс никогда не завершается, пока вы явно не отправите сигнал SIGINT.
Gabriel Llamas

2
Следовательно, вам необходимо отправить случайное сообщение от родителя к ребенку, например: «SIGINT».
Габриэль Лламас

8
Похоже, что это было решено довольно давно: github.com/nodejs/node-v0.x-archive/issues/5054
SimonSimCity

23

Я не уверен, когда именно, но на узле 8.x и в Windows 10 исходный код вопроса теперь просто работает.

process.on( "SIGINT", function() {
  console.log( "\ngracefully shutting down from SIGINT (Crtl-C)" );
  process.exit();
} );

process.on( "exit", function() {
  console.log( "never see this log message" );
} );

setInterval( () => console.log( "tick" ), 2500 );

введите описание изображения здесь

также работает с командной строкой Windows.


Я тоже это заметил. Windows 7 является для меня серьезной проблемой,
Грегори Новаковски

Этот метод работает у меня с Node 8.11.1 в Windows 7, но я запускал его из оболочки git bash. Подумал, что сначала попробую простой способ, и это сработало.
user944849

7

Если вам не нужен импорт «строки чтения» для других задач, я бы предложил импортировать «строку чтения» после того, как программа проверит, что она работает в Windows. Кроме того, для тех, кто может не знать - это работает как в 32-битных, так и в 64-битных системах Windows (что вернет ключевое слово «win32»). Спасибо за это решение, Габриэль.

if (process.platform === "win32") {
  require("readline")
    .createInterface({
      input: process.stdin,
      output: process.stdout
    })
    .on("SIGINT", function () {
      process.emit("SIGINT");
    });
}

process.on("SIGINT", function () {
  // graceful shutdown
  process.exit();
});

5

В настоящее время он работает на всех платформах , включая Windows.

Следующий код записывает, а затем корректно завершает работу в Windows 10:

process.on('SIGINT', () => {
    console.log("Terminating...");
    process.exit(0);
});

2
У меня это НЕ работает под cygwin. Используется последняя версия Windows 10 (автоматические обновления), версия узла 8.11.4. Под «это НЕ работает» я имею в виду, что 1) процесс завершается, но 2) сообщение не записывается в консоль и 3) созданные HTTP-соединения не завершаются. Однако я должен добавить, что я пробовал это в PowerShell, и там он работает, как ожидалось. Но я перешел на cygwin, потому что в PowerShell плохая команда curl. Штопать!
Джон Дейган

2
Он перестал работать у меня только сегодня без видимой причины (вообще без обновлений), и я до сих пор не знаю почему. Принятый ответ исправил это.
Sv443 01

4

В настоящее время в узле по-прежнему нет поддержки для захвата событий управления консоли Windows, поэтому нет эквивалентов сигналам POSIX:

https://github.com/joyent/node/issues/1553

Однако документация модуля tty дает пример механизма для захвата нажатий клавиш, чтобы инициировать плавное завершение работы, но тогда это работает только для ctrl+c .

var tty = require('tty');

process.stdin.resume();
tty.setRawMode(true);

process.stdin.on('keypress', function(char, key) {
  if (key && key.ctrl && key.name == 'c') {
    console.log('graceful exit of process %d', process.pid);
    process.exit();
  }
});

Спасибо, я искал эту информацию, для меня это приемлемая замена, если я могу реализовать ее на сервере для CTRL + C. +1 (Но .. Есть идеи, повлияет ли это на производительность путем добавления слушателей событий в процесс?)
Кори Гросс

Я пробовал это сделать, но когда на моем сервере запущен игровой цикл, stdin кажется недоступным, CTRL + C не работает, когда я использую вышеуказанное.
Кори Гросс

0

Начиная с node.js 0.8 keypressсобытие больше не существует. Однако существует пакет npm, называемый keypress, который повторно реализует событие.

Установите с помощью npm install keypress, затем выполните что-то вроде:

// Windows doesn't use POSIX signals
if (process.platform === "win32") {
    const keypress = require("keypress");
    keypress(process.stdin);
    process.stdin.resume();
    process.stdin.setRawMode(true);
    process.stdin.setEncoding("utf8");
    process.stdin.on("keypress", function(char, key) {
        if (key && key.ctrl && key.name == "c") {
            // Behave like a SIGUSR2
            process.emit("SIGUSR2");
        } else if (key && key.ctrl && key.name == "r") {
            // Behave like a SIGHUP
            process.emit("SIGHUP");
        }
    });
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.