Примечание
Статья не новая, да и node.js развивается очень активно, поэтому некоторые фич описаных в статье устарели/deprecated.
Более подробно читайте:
комментарии rumkin:
habrahabr.ru/company/mailru/blog/283228/#comment_8890604
комментарии Aiditz:
habrahabr.ru/company/mailru/blog/283228/#comment_8890476
комментарии Suvitruf:
habrahabr.ru/company/mailru/blog/283228/#comment_8890430
Идея этой публикации была навеяна серией книг Кайла Симпсона «Вы не знаете JavaScript». Они являются хорошим началом для изучения основ этого языка. А Node — это практически тот же JavaScript, за исключением небольших отличий, о которых я расскажу в этой статье. Весь код, приведённый ниже, вы можете скачать из репозитория, из папки
code
.Зачем вообще переживать насчёт Node? Node — это JavaScript, а JavaScript используется почти везде! Мир был бы лучше, если бы большинство разработчиков в совершенстве владели Node. Чем лучше приложения, тем лучше жизнь!
Эта статья представляет собой реалистичный взгляд на наиболее интересные основные возможности Node. Ключевые моменты статьи:
- Цикл событий: освежаем ключевую концепцию, позволяющую реализовать неблокирующие операции ввода/вывода.
- Глобальный объект и процесс: как получить больше информации.
- Эмиттеры событий: интенсивное введение в событийную модель (event-based pattern)
- Stream’ы и буферы: эффективный способ работы с данными
- Кластеры: форкай процессы как профессионал
- Обработка асинхронный ошибок: AsyncWrap, Domain и uncaughtException
- Аддоны на C++: внесение своих наработок в ядро и написание собственных аддонов на С++
Цикл событий
Начнём с цикла событий, лежащего в основе Node.
Неблокирующие операции ввода/вывода в Node.js
Цикл позволяет нам работать с другими задачами параллельно с выполнение операций ввода/вывода. Сравните Nginx и Apache. Именно благодаря циклу событий Node работает очень быстро и эффективно, поскольку блокирующие операции ввода/вывода недёшевы!
Взгляните на этот простой пример отложенной функции
println
в Java:System.out.println("Step: 1");
System.out.println("Step: 2");
Thread.sleep(1000);
System.out.println("Step: 3");
Это сравнимо (хотя и не совсем) с кодом Node:
console.log('Step: 1')
setTimeout(function () {
console.log('Step: 3')
}, 1000)
console.log('Step: 2')
Это не то же самое. Начните думать с точки зрения асинхронной работы. Выходные данные Node-скрипта — 1, 2, 3; но если бы после “Step 2” у нас было больше выражений, то сначала выполнялись бы они, а уже потом callback функции
setTimeout
. Взгляните на этот фрагмент:console.log('Step: 1')
setTimeout(function () {
console.log('Step: 3')
console.log('Step 5')
}, 1000);
console.log('Step: 2')
console.log('Step 4')
Результатом его работы будет очерёдность 1, 2, 4, 3, 5. Причина в том, что
setTimeout
помещает свой callback в будущие периоды цикла событий.Можно воспринимать цикл событий как бесконечный цикл наподобие
for … while
. Он останавливается только тогда, когда выполнять больше нечего, ни сейчас, ни в будущем.Блокирующие операции ввода/вывода: многопоточная Java
Цикл событий позволяет системе работать более эффективно, приложение может сделать что-то еще пока ждет завершение дорогих операций ввода/вывода.
Неблокирующие операции ввода/вывода в Node.js
Это контрастирует с более распространённой сегодня моделью параллельной обработки (concurrency model), при которой задействуются thread’ы операционной системы. Сетевая потоковая модель (thread-based networking) достаточно неэффективна и очень трудна в использовании. Более того, пользователи Node могут не опасаться полного блокирования процессов — здесь отсутствуют lock’и.
К слову, в Node.js всё же можно написать блокирующий код. Присмотритесь к этому простому фрагменту:
console.log('Step: 1')
var start = Date.now()
for (var i = 1; i<1000000000; i++) {
// This will take 100-1000ms depending on your machine
}
var end = Date.now()
console.log('Step: 2')
console.log(end-start)
Конечно, обычно в нашем коде отсутствуют пустые циклы. При использовании чужих модулей может быть труднее выявлять синхронный, а значит блокирующий код. К примеру, основной модуль
fs
(файловая система) идёт с двумя наборами методов. Каждая пара делает одно и то же, но разными способами. Блокирующие методы в модуле fs
имеют в названиях слово Sync
:var fs = require('fs')
var contents = fs.readFileSync('accounts.txt','utf8')
console.log(contents)
console.log('Hello Ruby\n')
var contents = fs.readFileSync('ips.txt','utf8')
console.log(contents)
console.log('Hello Node!')
Результат выполнения этого кода совершенно предсказуем даже для новичков в Node/JavaScript:
data1->Hello Ruby->data2->Hello NODE!
Но всё меняется, когда мы переходим на асинхронные методы. Вот пример неблокирующего кода:
var fs = require('fs');
var contents = fs.readFile('accounts.txt','utf8', function(err,contents){
console.log(contents);
});
console.log('Hello Python\n');
var contents = fs.readFile('ips.txt','utf8', function(err,contents){
console.log(contents);
});
console.log("Hello Node!");
contents
выводятся на экран последними, потому что их выполнение занимает какое-то время, они же находятся в callback'ах. Цикл событий перейдёт к ним по окончании чтения файла: Hello Python->Hello Node->data1->data2
В общем, цикл событий и неблокирующие операции ввода/вывода — вещи очень мощные, но вам придётся писать асинхронный код, к чему многие не привыкли.
Глобальный объект
Когда разработчики переходят с браузерного JavaScript или иного языка на Node.js, то у них возникают следующие вопросы:
- Где хранить пароли?
- Как создавать глобальные переменные (в Node нет
window
)? - Как обращаться к входным данным CLI, ОС, платформе, памяти, версиям и т.д.?
Для этого существует глобальный объект, у которого есть определённые свойства. Вот некоторые из них:
global.process
: процесс, система, информация об окружении (вы можете обратиться к входным данным CLI, к переменным окружения с паролями, к памяти т.д.)global.__filename
: имя файла и путь к выполняемому в данный момент скрипту, в котором находится это выражение.global.__dirname
: полный путь к выполняемому в данный момент скрипту.global.module
: объект для экспорта кода, создающего модуль из этого файла.global.require()
: метод для импорта модулей, JSON-файлов и папок.
Также есть обычные подозреваемые — методы из браузерного JavaScript:
global.console()
global.setInterval()
global.setTimeout()
К каждому из глобальных свойств можно обратиться как с помощью набранного заглавными буквами имени
GLOBAL
, так и вообще без имени, просто написав process
вместо global.process
.Процесс
Объект процесса заслуживает отдельной главы, потому что содержит массу информации. Вот некоторые из его свойств:
process.pid
: ID процесса этого экземпляра Node.process.versions
: разные версии Node, V8 и других компонентовprocess.arch
: архитектура системыprocess.argv
: аргументы CLIprocess.env
: переменные окружения
Некоторые методы:
process.uptime()
: получает время работыprocess.memoryUsage()
: получает объём потребляемой памятиprocess.cwd()
: получает текущую рабочую папку. Не путатьс __dirname
, не зависящим от места, из которого был запущен процесс.process.exit()
: выходит из текущего процесс. К примеру, можно передать код 0 или 1.process.on()
: прикрепляет на событие, например, `on(‘uncaughtException’)
Трудный вопрос: кому нравятся и кто понимает суть callback'ов?
Кто-то так сильно в них «влюблён», что создал http://callbackhell.com. Если этот термин вам не знаком, то вот иллюстрация:
fs.readdir(source, function (err, files) {
if (err) {
console.log('Error finding files: ' + err)
} else {
files.forEach(function (filename, fileIndex) {
console.log(filename)
gm(source + filename).size(function (err, values) {
if (err) {
console.log('Error identifying file size: ' + err)
} else {
console.log(filename + ' : ' + values)
aspect = (values.width / values.height)
widths.forEach(function (width, widthIndex) {
height = Math.round(width / aspect)
console.log('resizing ' + filename + 'to ' + height + 'x' + height)
this.resize(width, height).write(dest + 'w' + width + '_' + filename, function(err) {
if (err) console.log('Error writing file: ' + err)
})
}.bind(this))
}
})
})
}
})
«Ад callback’ов» труден для чтения, и здесь легко можно допустить ошибки. Так как же нам разделять на модули и организовывать асинхронный код, если не с помощью callback’ов, которые не слишком-то удобны для масштабирования с точки зрения разработки.
Эмиттеры событий
Чтобы справиться с адом callback’ов, или пирамидой погибели (pyramid of doom), применяются эмиттеры событий. С их помощью можно реализовать асинхронный код с использованием событий.
Если кратко, то эмиттер событий — это триггер для события, которое может прослушать кто угодно. В Node.js за каждым событием закреплено строковое имя, на которое эмиттером может быть повешен callback.
Для чего нужны эмиттеры:
- В Node события обрабатываются с использованием шаблона “observer”.
- Событие (или субъект) отслеживает все связанные с ним функции.
- Эти функции — observer’ы— исполняются при активизации данного события.
Для использования эмиттеров нужно импортировать модуль и создать экземпляр объекта:
var events = require('events')
var emitter = new events.EventEmitter()
Далее можно прикрепить получателей событий и активировать/передавать события:
emitter.on('knock', function() {
console.log('Who\'s there?')
})
emitter.on('knock', function() {
console.log('Go away!')
})
emitter.emit('knock')
Давайте с помощью
EventEmitter
сделаем что-нибудь полезное, унаследовав это от него. Допустим, вам регулярно нужно реализовывать какой-то класс — ежемесячно, еженедельно или каждый день. Этот класс должен быть достаточно гибким, чтобы другие разработчики могли кастомизировать финальный результат. Иными словами, по окончании вашей работы любой человек должен иметь возможность поместить в класс какую-то свою логику.На этой схеме показано, как мы с помощью наследования от модуля событий создаём
Job
, а затем используем получателя событий done
для изменения поведения класса Job
:Эмиттеры событий в Node.js: шаблон “observer”
Класс
Job
сохранит свои свойства, но в то же время получит и события. По окончании процесса нам нужно лишь запустить событие done
:// job.js
var util = require('util')
var Job = function Job() {
var job = this
// ...
job.process = function() {
// ...
job.emit('done', { completedOn: new Date() })
}
}
util.inherits(Job, require('events').EventEmitter)
module.exports = Job
В завершение изменим поведение
Job
. Раз он передаёт done
, значит мы можем прикрепить получателя событий:// weekly.js
var Job = require('./job.js')
var job = new Job()
job.on('done', function(details){
console.log('Job was completed at', details.completedOn)
job.removeAllListeners()
})
job.process()
У эмиттеров есть и другие возможности:
emitter.listeners(eventName)
: формирует список всех получателей для данного события.emitter.once(eventName, listener)
: прикрепляет одноразового получателя событий.emitter.removeListener(eventName, listener)
: удаляет получателя событий.
В Node везде используется событийный шаблон, особенно в основных модулях. Так что если вы будете грамотно использовать события, то сэкономите кучу времени.
Stream’ы
При работе с большими объёмами данных в Node возникает несколько проблем. Производительность может быть низкой, а размер буфера ограничен примерно 1 Гб. Кроме того, как работать в условиях бесконечного ресурса, который создавался из расчёта, что он никогда не закончится? В этих ситуациях нам помогут stream’ы.
Stream’ы в Node являются абстракцией, обозначающей непрерывное разбиение данных на фрагменты. Иными словами, вам не нужно ждать полной загрузки ресурса. На диаграмме показан стандартный подход к буферизации:
Подход к буферизации в Node.js
Прежде чем начать обработку данных и/или выводить их, нам приходится ждать полной загрузки буфера. А теперь сравните это со схемой работы stream’ов. В этом случае мы можем сразу начать обрабатывать данные и/или выводить их, как только получим первый чанк:
Поточный подход в Node.js
В Node есть четыре типа stream’ов:
- Читаемые (readable): из них можно читать.
- Записываемые (writable): в них можно писать.
- Дуплексные (duplex): можно и писать, и читать.
- Stream’ы преобразования (transform): их можно использовать для преобразования данных.
Виртуально stream’ы используются в Node повсеместно. Наиболее востребованные реализации stream’ов:
- HTTP-запросы и отклики.
- Стандартные операции ввода/вывода.
- Чтение из файлов и запись в них.
Для обеспечения шаблона “observer”, stream’ы — события — наследуют от объекта «эмиттер событий». Мы можем использовать это для реализации stream’ов.
Пример читаемого stream’а
В качестве примера можно привести
process.stdin
, являющийся стандартным stream’ом ввода. В нём содержатся данные, которые поступают в приложение. Обычно это информация с клавиатуры, используемая для запуска процесса.Для считывания данных из
stdin
применяются события data
и end
. Callback события data
в качестве аргумента будет иметь chunk
:process.stdin.resume()
process.stdin.setEncoding('utf8')
process.stdin.on('data', function (chunk) {
console.log('chunk: ', chunk)
})
process.stdin.on('end', function () {
console.log('--- END ---')
})
Далее
chunk
скармливается программе в качестве входных данных. Это событие может быть активировано несколько раз, в зависимости от общего объёма входящей информации. О завершении stream’а необходимо сигнализировать с помощью события end
.Обратите внимание:
stdin
по умолчанию ставится на паузу, из которой его надо вывести прежде, чем считывать из него данные. У читаемых stream’ов есть синхронно работающий интерфейс
read()
. По завершении stream’а он возвращает chunk
или null
. Мы можем воспользоваться этим поведением, положив в условие while
конструкцию null !== (chunk = readable.read())
:var readable = getReadableStreamSomehow()
readable.on('readable', () => {
var chunk
while (null !== (chunk = readable.read())) {
console.log('got %d bytes of data', chunk.length)
}
})
В идеале, нам хотелось бы как можно чаще писать в Node асинхронный код, чтобы избегать блокирования thread’а. Но благодаря маленькому размеру чанков можно не волноваться насчёт того, что синхронный
readable.read()
заблокирует thread.Пример записываемого stream’а
В качестве примера можно привести
process.stdout
, являющийся стандартным stream’ом вывода. В нём содержатся данные, которые покидают приложение. Записывать в stream можно с помощью операции write
.process.stdout.write('A simple message\n')
Записанные в стандартный stream вывода данные отображаются в командной строке, как если бы мы воспользовались
console.log()
. Pipe
В Node есть альтернатива вышеописанным событиям — метод
pipe()
. В следующем примере происходит чтение данных из файла, сжатие с помощью GZip и запись результата в файл:var r = fs.createReadStream('file.txt')
var z = zlib.createGzip()
var w = fs.createWriteStream('file.txt.gz')
r.pipe(z).pipe(w)
Readable.pipe()
берёт поток данных и пропускает через все stream’ы, поэтому мы можем создавать цепочки из методов pipe()
.Так что при использовании stream’ов вы можете применять события или pipe.
HTTP-stream’ы
Большинство из нас используют Node для создания веб-приложений: традиционных (серверных) или на основе REST API (клиентских). А что насчёт HTTP-запросов? Их можно стримить? Однозначно!
Запросы и отклики представляют из себя читаемые и записываемые stream’ы, наследуемые от эмиттеров событий. Можно прикрепить получателя событий
data
и принимать chunk
в его callback'е, который можно сразу преобразовывать, не дожидаясь получения всего отклика. В следующем примере мы конкатенируем body
и парсим его в callback события end
:const http = require('http')
var server = http.createServer( (req, res) => {
var body = ''
req.setEncoding('utf8')
req.on('data', (chunk) => {
body += chunk
})
req.on('end', () => {
var data = JSON.parse(body)
res.write(typeof data)
res.end()
})
})
server.listen(1337)
Обратите внимание: согласно ES6,
()=>{}
является новым синтаксисом для анонимных функций, а const
— новый оператор. Если вы ещё не знакомы с особенностями и синтаксисом ES6/ES2015, то можете изучить статью Top 10 свойств ES6, о которых должен знать каждый занятой JavaScript-разработчик.Давайте теперь воспользуемся Express.js, чтобы наш сервер стал менее оторванным от реальной жизни. Возьмём огромное изображение (около 8 Мб) и два набора Express routes
/stream
и /non-stream
.server-stream.js:
app.get('/non-stream', function(req, res) {
var file = fs.readFile(largeImagePath, function(error, data){
res.end(data)
})
})
app.get('/stream', function(req, res) {
var stream = fs.createReadStream(largeImagePath)
stream.pipe(res)
})
Также у меня есть альтернативная реализация
/stream2
с событиями, и синхронная реализация /non-stream2
. Они делают всё то же самое, но в них использован другой синтаксис и стиль. В данном случае синхронные методы работают быстрее, потому что мы отправляем лишь один запрос, а не несколько конкурирующих.Запустить это код можно через терминал:
$ node server-stream
Теперь откройте в Chrome http://localhost:3000/stream и http://localhost:3000/non-stream. Обратите внимание на заголовки во вкладке Сеть в инструментах разработчика, сравнив
X-Response-Time
. В моём случае /stream
и /stream2
различались на порядки: 300 мсек. и 3–5 сек. У вас могут быть другие значения, но идея понятна: в случае со /stream
пользователи/клиенты раньше начнут получать данные. Stream’ы в Node — очень мощный инструмент! Вы можете научиться хорошо управлять ресурсами stream’ов, став в вашей команде экспертом в этой области.С помощью npm можете установить себе Stream Handbook и stream-adventure:
$ sudo npm install -g stream-adventure
$ stream-adventure
Буферы
Какие типы мы можем использовать для бинарных данных? Если помните, в браузере JavaScript нет бинарного типа данных, а в Node есть. Это называется буфером. Он представляет собой глобальный объект, поэтому нет нужды импортировать его в виде модуля.
Можно использовать одно из этих выражений для создания бинарного типа:
new Buffer(size)
new Buffer(array)
new Buffer(buffer)
new Buffer(str[, encoding])
Полный список методов и кодировок доступен в документации по буферу. Чаще всего используется кодировка
utf8
.Обычно содержимое буфера выглядит абракадаброй, поэтому, чтобы его можно было прочитать человеку, нужно сначала конвертировать его в строковое представление с помощью
toString()
. Создадим буфер с алфавитом с помощью цикла for
:let buf = new Buffer(26)
for (var i = 0 ; i < 26 ; i++) {
buf[i] = i + 97 // 97 is ASCII a
}
Если не конвертировать буфер в строковое представление, то он будет выглядеть как массив чисел:
console.log(buf) // <Buffer 61 62 63 64 65 66 67 68 69 6a 6b 6c 6d 6e 6f 70 71 72 73 74 75 76 77 78 79 7a>
Осуществим конвертацию:
buf.toString('utf8') // outputs: abcdefghijklmnopqrstuvwxyz
buf.toString('ascii') // outputs: abcdefghijklmnopqrstuvwxyz
Если нам нужно лишь часть строки (sub string), то метод берёт начальное число и конечную позицию нужного отрезка:
buf.toString('ascii', 0, 5) // outputs: abcde
buf.toString('utf8', 0, 5) // outputs: abcde
buf.toString(undefined, 0, 5) // encoding defaults to 'utf8', outputs abcde
Помните
fs
? По умолчанию значение data
тоже является буфером:fs.readFile('/etc/passwd', function (err, data) {
if (err) return console.error(err)
console.log(data)
});
data
выполняет роль буфера при работе с файлами.Кластеры
Противники Node часто приводят аргумент, что он может масштабироваться, поскольку обладает лишь одним thread’ом. Однако с помощью основного модуля
cluster
(вам не нужно его устанавливать, это часть платформы) мы можем использовать все ресурсы процессора на любой машине. Иными словами, благодаря кластерам мы можем вертикально масштабировать Node-приложения.Код очень прост: импортируем модуль, создаём одного мастера и несколько работников (worker). Обычно создают по одному процессу на каждый ЦПУ, но это не является незыблемым правилом. Вы можете наделать столько процессов, сколько пожелаете, но с определённого момента прирост производительности прекратится, согласно закону убывания доходности.
Код мастера и работника находится в одном файле. Работник может прослушивать тот же порт, отправляя мастеру сообщения посредством событий. Мастер может слушать события, и в случае необходимости перезапускать кластеры. Для мастера используется
cluster.isMaster()
, для работника — cluster.isWorker()
. Большая часть серверного кода будет расположена в работнике (isWorker()
). // cluster.js
var cluster = require('cluster')
if (cluster.isMaster) {
for (var i = 0; i < numCPUs; i++) {
cluster.fork()
}
} else if (cluster.isWorker) {
// ваш серверный код
})
В этом примере мой сервер выдаёт ID процессов, поэтому можно наблюдать, как разные работники обрабатывают разные запросы. Похоже на балансировщика нагрузки, но это лишь впечатление, потому что нагрузка не будет распределяться равномерно. Например, по PID вы увидите, как один из процессов может обрабатывать гораздо больше запросов.
Чтобы посмотреть, как разные работники обслуживают разные запросы, воспользуйтесь нагрузочным тестовым инструментом
loadtest
на базе Node:- Установите
loadtest
с помощьюnpm: $ npm install -g loadtest
- Запустите
code/cluster.js
с помощью node ($ node cluster.js
); пусть сервер продолжает работать. - Запустите в новом окне нагрузочное тестирование:
$ loadtest http://localhost:3000 -t 20 -c 10
. - Проанализируйте результаты из серверного терминала и из терминала
loadtest
. - По окончании тестирования в серверном терминале нажмите Ctrl+C. Вы увидите разные PID.
В команде
loadtest
кусок -t 20 -c 10
означает, что будет выполнено 10 конкурентных запросов в течение не более чем 20 секунд.Кластер — это часть ядра, и это практически единственное его преимущество. Когда ваш проект будет готов к развёртыванию, вам может понадобиться воспользоваться более продвинутым диспетчером процессов. Хорошим выбором могут быть:
strong-cluster-control
(https://github.com/strongloop/strong-cluster-control) или$ slc run
pm2
(https://github.com/Unitech/pm2)
pm2
Давайте рассмотрим инструмент
pm2
, поскольку это один из лучших способов вертикального масштабирования вашего Node-приложения. Кроме того, pm2
повышает производительность на стадии production.Преимущества
pm2
:- Балансировщик нагрузок и ряд других полезных возможностей.
- Отсутствие даунтайма при перезагрузке, то есть он работает постоянно.
- Хорошее покрытие тестами.
Документацию можно найти в https://github.com/Unitech/pm2 и http://pm2.keymetrics.io.
В качестве примера использования pm2 можно привести экспресс-сервер
server.js
. Хорошо, что здесь нет шаблонного кода isMaster()
, потому что вам не придётся модифицировать исходный код, как мы это делали в случае с cluster
. Здесь достаточно лишь логгировать pid
и сохранять по ним статистику.var express = require('express')
var port = 3000
global.stats = {}
console.log('worker (%s) is now listening to http://localhost:%s',
process.pid, port)
var app = express()
app.get('*', function(req, res) {
if (!global.stats[process.pid]) global.stats[process.pid] = 1
else global.stats[process.pid] += 1;
var l ='cluser '
+ process.pid
+ ' responded \n';
console.log(l, global.stats)
res.status(200).send(l)
})
app.listen(port)
Для запуска этого примера введите
pm2 start server.js
. Можете размножить экземпляры/процессы, введя необходимое количество (-i 0
означает, что процессов должно быть столько же, сколько ЦПУ, в моём случае их 4). Для сохранения лога в файл используйте опцию -l log.txt
:$ pm2 start server.js -i 0 -l ./log.txt
Приятно то, что pm2 работает в фоновом режиме. Для просмотра исполняемых в данный момент процессов введите:
$ pm2 list
Теперь воспользуйтесь
loadtest
, как мы это делали в примере с cluster
. В новом окне выполните:$ loadtest http://localhost:3000 -t 20 -c 10
У вас могут быть другие результаты, но у меня в
log.txt
распределение получилось более-менее равномерным:cluser 67415 responded
{ '67415': 4078 }
cluser 67430 responded
{ '67430': 4155 }
cluser 67404 responded
{ '67404': 4075 }
cluser 67403 responded
{ '67403': 4054 }
Сравниваем Spawn, Fork и Exec
В примере
cluter.js
для создания нового экземпляра Node-сервера был использован fork()
. Но нужно сказать, что запустить внешний процесс из Node.js можно тремя способами: spawn()
, fork()
и exec()
. Причём все они берутся из основного модуля child_process
. Разница между этими тремя способами заключается в следующем:require('child_process').spawn()
: используется для больших объёмов данных; поддерживает stream’ы; может применяться с любыми командами; не создаёт новый экземпляр V8.require('child_process').fork()
: создаёт новый экземпляр V8 и экземпляры работников; работает только со скриптами Node.js (командаnode
).require('child_process').exec()
: неудобен для больших объёмов данных, потому что использует буфер; работает асинхронно, чтобы одновременно получать все данные в callback’е; может применяться с любой командой, не толькоnode
.
Посмотрите на следующий пример: мы исполняем
node program.js
, но можно запустить и любые другие команды или скрипты — bash, Python, Ruby и т.д. Если нужно передать команде дополнительные аргументы, просто положите их в виде аргументов массива, который является параметром spawn()
. Данные попадут в обработчик события data
в виде stream’а:var fs = require('fs')
var process = require('child_process')
var p = process.spawn('node', 'program.js')
p.stdout.on('data', function(data)) {
console.log('stdout: ' + data)
})
С точки зрения команды
node program.js
, data
является стандартным способом вывода, например, в терминал.Синтаксис
fork()
почти аналогичен синтаксису метода spawn()
, за одним исключением: здесь не нужна команда, потому что fork()
предполагает, что все процессы относятся к Node.js:var fs = require('fs')
var process = require('child_process')
var p = process.fork('program.js')
p.stdout.on('data', function(data)) {
console.log('stdout: ' + data)
})
Наконец,
exec()
. Этот метод отличается от предыдущих тем, что использует не шаблон событий, а одиночный callback. Внутри него есть error, standard output и параметры стандартной ошибки:var fs = require('fs')
var process = require('child_process')
var p = process.exec('node program.js', function (error, stdout, stderr) {
if (error) console.log(error.code)
})
Разница между
error
и stderr
заключается в том, что первый мы получаем из exec()
(скажем, program.js
отказано в доступе), а второй — из вывода ошибки запущенной вами команды (например, program.js
не смог подключиться к базе данных). Обработка асинхронных ошибок
Для обработки ошибок в Node.js и почти всех остальных языках программирования используется
try/catch
. С синхронными ошибками эта конструкция работает замечательно.try {
throw new Error('Fail!')
} catch (e) {
console.log('Custom Error: ' + e.message)
}
Модули и функции кидают ошибки, а мы их потом ловим. Так работает Java и синхронный Node. Но в Node.js лучше всего писать асинхронный код, чтобы не блокировать thread.
Благодаря циклу событий система может делегировать и применять расписание для кода, который должен быть выполнен в будущем, по завершении ресурсозатратных задач ввода/вывода. Но тут у нас возникает проблема с асинхронными ошибками, потому что контекст ошибки утрачивается.
К примеру,
setTimeout()
работает асинхронно, откладывая на будущее вызов callback’а. Аналогично поведёт себя и асинхронная функция, делающая HTTP-запрос, читающая из БД или пишущая в файл:try {
setTimeout(function () {
throw new Error('Fail!')
}, Math.round(Math.random()*100))
} catch (e) {
console.log('Custom Error: ' + e.message)
}
Когда выполняется callback и приложение падает, то здесь уже нет
try/catch
. Конечно, можно поместить в callback другую конструкцию try/catch
, которая поймает ошибку, но это плохое решение. Все эти назойливые асинхронные ошибки труднее вылавливать и отлаживать. Для асинхронного кода try/catch
не лучший вариант.Итак, приложение упало. Что нам с этим делать? Вы уже видели, что в большинстве callback’ов есть аргумент
error
. В каждом случае вам нужно проверять и вытаскивать его: отвергнуть цепочку callback'а или вывести пользователю сообщение об ошибке. if (error) return callback(error)
// or
if (error) return console.error(error)
Другой хороший способ обработки асинхронных ошибок заключается в следующем:
- Прослушиваем все сообщения об ошибках (on error).
- Прослушиваем
uncaughtException
. - Используем
domain
(несколько устарел) или AsyncWrap. - Логгируем и отслеживаем.
- Уведомляем (по желанию).
- Выходим и перезапускаем процесс.
on(‘error’)
Прослушивайте события
on('error')
, генерируемые большинством основных объектов в Node.js, в особенности http
. Также error
генерирует всё, наследуется от или создаёт экземпляры Express.js, LoopBack, Sails, Hapi и т.д., потому что эти фреймворки расширяют http
.js
server.on('error', function (err) {
console.error(err)
console.error(err)
process.exit(1)
})
uncaughtException
Всегда прослушивайте
uncaughtException
в объекте process
! uncaughtException
— это очень грубый механизм обработки ошибок. Если ошибка не обработана, то ваше приложение — а значит и Node.js — находится в неопределённом состоянии. An unhandled exception means your application – and by extension Node.js itself – is in an undefined state. Если возобновить работу вслепую, то может произойти что угодно.
process.on('uncaughtException', function (err) {
console.error('uncaughtException: ', err.message)
console.error(err.stack)
process.exit(1)
})
или
process.addListener('uncaughtException', function (err) {
console.error('uncaughtException: ', err.message)
console.error(err.stack)
process.exit(1)
Domain
У
domain
нет ничего общего с сетевыми доменами в браузере. Это основной модуль Node.js для обработки асинхронных ошибок. Он сохраняет контекст, в котором реализован асинхронный код. Стандартный подход: создать копию domain
и поместить код с ошибкой внутрь callback'а run()
:var domain = require('domain').create()
domain.on('error', function(error){
console.log(error)
})
domain.run(function(){
throw new Error('Failed!')
})
Начиная с версии 4.0
domain
считается устаревшим, поэтому разработчики Node наверняка отделят его от платформы. Но на сегодняшний день в ядре Node нет альтернативы domain
. Кроме того, благодаря серьёзной поддержке и широкому использованию, domain
будет существовать в виде отдельного npm-модуля, поэтому вы сможете легко переключаться с основного модуля на npm. Так что с domain
мы не прощаемся.Давайте сделаем асинхронную ошибку с помощью того же
setTimeout()
:// domain-async.js:
var d = require('domain').create()
d.on('error', function(e) {
console.log('Custom Error: ' + e)
})
d.run(function() {
setTimeout(function () {
throw new Error('Failed!')
}, Math.round(Math.random()*100))
});
Код не упадёт! От принадлежащего
domain
обработчика событий error
мы получим красивое сообщение “Custom Error”, а не типичную для Node трассировку стека.Аддоны на C++
Популярность Node среди разработчиков железа, IoT, дронов, роботов и умных гаджетов заключается в том, что он позволяет забавляться с более низкоуровневым кодом на С/С++. Как же можно писать свои С/С++ биндинги?
Это последняя из основных возможностей в этой статье. Большинство новичков в Node даже не представляют о том, что здесь можно писать собственные аддоны на С++! Это настолько просто, что мы прямо сейчас напишем аддон с нуля.
Сначала создадим файл
hello.cc
, в начале которого поместим несколько шаблонных импортов. Затем определим метод, возвращающий строковое и экспортирующий себя. #include <node.h>
namespace demo {
using v8::FunctionCallbackInfo;
using v8::HandleScope;
using v8::Isolate;
using v8::Local;
using v8::Object;
using v8::String;
using v8::Value;
void Method(const FunctionCallbackInfo<Value>& args) {
Isolate* isolate = args.GetIsolate();
args.GetReturnValue().Set(String::NewFromUtf8(isolate, "capital one")); // String
}
void init(Local<Object> exports) {
NODE_SET_METHOD(exports, "hello", Method); // Exporting
}
NODE_MODULE(addon, init)
}
Даже если вы не знаток С, всё равно можете легко понять, что здесь происходит, ведь синтаксис не слишком сильно отличается от JavaScript. Строковым будет
capital one
:args.GetReturnValue().Set(String::NewFromUtf8(isolate, "capital one"));
А после экспортирования будет присвоено имя
hello
:void init(Local<Object> exports) {
NODE_SET_METHOD(exports, "hello", Method);
}
Файл
hello.cc
готов, идём дальше. Создадим binding.gyp
, содержащий имя файла с исходным кодом и именем аддона:{
"targets": [
{
"target_name": "addon",
"sources": [ "hello.cc" ]
}
]
}
Сохраним
binding.gyp
в одной папке с hello.cc
, а затем установим node-gyp:$ npm install -g node-gyp
Теперь в той же папке, где лежат
hello.cc
и binding.gyp
, выполните эти команды конфигурирования и сборки:$ node-gyp configure
$ node-gyp build
Будет создана папка
build
. Зайдите в build/Release/
и проверьте наличие файлов .node
. Наконец, напишите на Node.js создающий скрипт hello.js
, включив в него аддон на C++:var addon = require('./build/Release/addon')
console.log(addon.hello()) // 'capital one'
Чтобы запустить скрипт и увидеть на экране содержимое
capital one
, введите:$ node hello.js
Другие примеры аддонов на C++ можно найти здесь: https://github.com/nodejs/node-addon-examples.
Заключение
Весь вышеприведённый код можно скачать с GitHub. Если вас заинтересовали шаблоны Node.js, callback’и и Node-соглашения, можете почитать мою статью Шаблоны Node: от callback’ов до observer’а.
Краткое содержание текущей статьи:
- Цикл событий: механизм, лежащий в основе неблокирующих операций ввода/вывода в Node.
- Глобал и процесс: глобальные объекты и системная информация.
- Эмиттеры событий: шаблон “observer” в Node.js.
- Потоки: шаблон работы с большими объёмами данных.
- Буферы: тип двоичных данных.
- Кластеры: вертикальное масштабирование.
- Domain: обработка асинхронных ошибок.
- Аддоны на C++: низкоуровневые аддоны.
Большая часть Node представляет собой тот же JavaScript, за исключением некоторых основных возможностей, по большей части относящихся к системному доступу, глобальным объектам, внешним процессам и низкоуровневому коду. Если вы разобрались со всеми этими концепциями, то вскоре сможете стать настоящим сэнсэем Node.js.
Комментарии (30)
forgotten
11.05.2016 12:13+8Под громким заголовком «Вы не знаете Node» скрывается набор тривиальных фактов, щедро приправленный фактическими ошибками. Мде.
Кстати, а в чем разница между Apache 2.4 и nginx с точки зрения event loop, не расскажете?
Aiditz
11.05.2016 12:16+1> Начиная с версии 4.0 domain считается устаревшим, поэтому разработчики Node наверняка отделят его от платформы. Но на сегодняшний день в ядре Node нет альтернативы domain.
Только это не повод использовать устаревший модуль. К тому же, разве Promises не позволяют достаточно изящно обрабатывать асинхронные ошибки?dedokOne
11.05.2016 12:29-1Статья не новая, но мне показалась достаточно интересная.
Да промисы позволяют решить проблему. Но, точно знаю, что есть кучу кода который написан в old-style, не известно будут ли там внедрены промисы или нет, да и возможно ли такое внедрение не сломав API — вопрос. По этому domain все же имеет право на жизнь.
PieceOfMeat
11.05.2016 12:19Чтобы справиться с адом callback’ов, или пирамидой погибели (pyramid of doom), применяются эмиттеры событий. С их помощью можно реализовать асинхронный код с использованием событий.
nodejs.org/api/events.html#events_events
When the EventEmitter object emits an event, all of the Functions attached to that specific event are called synchronously.
То есть какой-то неправильный способ борьбы с callback hell предлагается, разве нет?dedokOne
11.05.2016 12:33Почему же? Что мешает сделать асинхронные действия в invoked function? Или я вас не понял :)
PieceOfMeat
11.05.2016 12:44+2Прошу прощения, разметку съел парсер почему-то.
Ничего не мешает сделать обработчики событий асинхронными, но в коде примеров в статье этого ведь не сделано.
Сказано что с помощью событий можно реализовать асинхронный код, а в примерах вместо этого написан синхронный (тут наверное уместно было бы подколоть автора что дескать сам не знает node.js — а все туда же :) )
rumkin
11.05.2016 12:50+5Статья явно не первой свежести. Конструкторы Buffer получили статус deprecated. EventEmitter вовсе не способ борьбы с callback hell, это место прочно заняли Promise. А вместо console.error пора начинать использовать process.emitWarning.
Кто-то явно не знает Node.js, остается вопрос – кто?dedokOne
11.05.2016 12:55-1Это статья не только про это, и что статья не новая было сказано. В ней есть много чего интересного, к примеру: как через pipe сделать chain of responsibility, явно не сказано но на мысли наводит; event drive…
А к деталям можно придраться в любом материале. ИМХОrumkin
11.05.2016 13:37+3Можно во всем искать только хорошее. Материал более чем на половину устарел или не соответствует действительности, а заголовок прямо указывает на обратное (отдает желтизной). Это может ввести новичков в заблуждение. Я бы на вашем месте сделал ремарку во вступительном абзаце, до того, как кто-то начнет применять эти советы на практике.
TimsTims
11.05.2016 14:491) Определенно в закладки
2) Если бы я был новичком и только-только открыл вашу статью, то уж точно ничего не понял бы. Статья вроде бы подразумевает, что читатели должны быть знакомы с Node, в то же время всё начинается весьма поверхностно, но не до конца разжевано. Например:
> Сравните Nginx и Apache. Именно благодаря циклу событий Node работает очень быстро
Читатель может быть не знаком с тем, что Nginx изначально асинхронный, а апач использует отдельные процессы. И далее сразу идет «именно поэтому Node работает очень быстро». Т.е. человек должен как-то сравнить два веб-сервера и прийти к выводу, что Node быстр. Что-то из разряда: «Сравните двс в Ладе и BMW. Именно поэтому самолеты такие быстрые!»
> Результат выполнения этого кода совершенно предсказуем даже для новичков в Node/JavaScript: data1->Hello Ruby->data2->Hello NODE!
Может я придираюсь, но прочитав ваш код — мне совершенно не очевидно, что там лежало в файлах, которые код должен был выдать. Там могло лежать «data1», а могла быть и «Война и мир»
И опять-же в следующем примере про асинхронное чтение файла — не факт, что первый файл прочитается быстрее, чем второй. Это зависит от FS и от размера файлов. Я думаю, это очень важно и про это стоит добавить, т.к. в этом и смысл callback'ов, что они могут вести себя весьма непредсказуемо — либо первый либо второй, зависит от разных причин. И это надо держать в голове.
Вроде примеры весьма простые, кажется что для новичков, но очень многое приходится додумывать, и будь на месте действительно новичок — ему будет многое непонятно. А для более-менее опытных — здесь мало нового. Я считаю, что если уж начали разжевывать, так разжуйте.
Как для читателя, соблазненным Node, но так и не освоившим на полную — подчеркну, что здесь приведен весьма интересный для меня материал. Однако столкнувшись с вашей статьёй на несколько лет раньше — не понял бы сути.
Еще вопрос:
Раскройте пожалуйста, что произойдет, если на один эмитер два раза повесить одно и то же событие 'knock':
emitter.on('knock', function(){})
Эмитер с вызовом функции срабоет дважды или единожды?napa3um
12.05.2016 15:41Если у вас что-то типа:
emitter.on('knock', aFun);
emitter.on('knock', bFun);
то при наступлении события knock будет вызваны, естественно, и слушатель aFun, и слушатель bFun (в порядке установки этих слушателей). Или вы о другом спрашивали?
norlin
11.05.2016 15:33+1Для избавления от callback hell эвент-эммиттеры не сильно помогут, разве что – косвенно.
А вот настоящее счастье-радость наступает с использованием ES6-генераторов скрещённых с Promise: co.js
console.log('Hello'); co(function* () { let data1 = yield asyncCall1(); // wait for async call console.log(data1); let data2 = yield asyncCall2(); // wait for async call console.log(data2); }); console.log('World!');
Выведет следующее:
Hello World! [data1] [data2]
Причём, data1 и data2 будут строго в этом порядке.ChALkeRx
11.05.2016 17:46+2Если честно, «co» нужен только как временная альтернатива пока нет нативного async/await.
Причём остался ли он нужен сегодня, когда Babel переводит вам код с async/await в код на генераторах — тоже вопрос.
Я не говорю, что «co» плохой или что его не надо использовать, но я сомневаюсь, что это то, с чего надо сейчас начинать.
Есть ненулевой шанс, что его задепрекейтят в ближайший год.
RSATom
12.05.2016 09:54> Аддоны на C++
Хочется поделиться ссылкой на проект который может оказаться весьма полезным для желающих написать свой С++ Addon для Node.js (впрочем как и для NW.js и Electron): cmake-js — данный проект позволяет использовать CMake в качестве системы сборки вместо node-gyp. Проблема в том что для большинства сторонних библиотек .gyp файлы придется писать самостоятельно (а зачем еще может понадобиться написание C/C++ addon кроме как для подключения сторонней библиотеки?), в то время как с CMake шансы избавить себя от дополнительной работы весьма высоки.
RSATom
12.05.2016 09:56Ну а если у вас все же возникнут проблемы в этом вопросе, то как говориться «Хочешь написать node.js addon — спроси меня как». Мне волей не волей пришлось съесть собаку на этой теме :)
dedokOne
12.05.2016 11:38Прикольная тулзовина!
— (а зачем еще может понадобиться написание C/C++ addon кроме как для подключения сторонней библиотеки?)
Для оптимизации некоторых build-in функций — бывало такое ;)RSATom
12.05.2016 12:07возможно вы этом случае было правильней слать патч разрабочикам Node.js, но в общем то согласен, такой вариант тоже имеет место быть.
egoroof
12.05.2016 12:45+1> Если помните, в браузере JavaScript нет бинарного типа данных
В браузерах давно есть ArrayBuffer, который, кстати в ноде делится памятью с Buffer.
iShatokhin
12.05.2016 12:46Какие типы мы можем использовать для бинарных данных? Если помните, в браузере JavaScript нет бинарного типа данных, а в Node есть. Это называется буфером.
В браузерах появился TypedArray (ES2015). В последних версиях Node «Buffer» является классом над стандартным Uint8Array API.
ChALkeRx
12.05.2016 13:13+1А вы точно «знаете Node.js»?
По пунктам:
> К слову, в Node.js всё же можно написать блокирующий код.
Вы это подаёте так: «в Node.js по умолчанию всё асинхронное, но можно использовать *Sync методы и будет как вы привыкли», не упоминая про то, что использование синхронных методов на сервере в процессе работы — выстрел себе в ноги из гранатомёта. В первую очередь в статье для новичков стоило бы на этом акцентировать внимание, а то пойдут же *Sync вызовы втыкать.
> Чтобы справиться с адом callback’ов, или пирамидой погибели (pyramid of doom), применяются эмиттеры событий.
Как уже сказали — Promises, async/await. Не надо городить ад на эмиттерах =).
> Буферы
Очень важный момент (не упомянутый в этой статье) — Buffer(number) фактически выделяет кусок памяти, в которым могут быть чувствительные данные. Именно поэтому в новой версии его разделили на Buffer.from(val) и Buffer.alloc(number)/Buffer.allocUnsafe(number) — чтобы случайно не вызвать Buffer(number) при передаче числа там, где предполагался вызов Buffer(val). Buffer.alloc(number), кстати, теперь зануляет выделяемую память (allocUnsafe — нет, но его стоит использовать очень осторожно).
> Где хранить пароли?
> Для этого существует глобальный объект,
Нет, глобальный объект существует не для этого. Вынесите ваш конфиг в модуль/файл.
> global.__filename, global.__dirname, global.module, global.require
Они все — undefined, в вашей статье ошибка.
Эти переменные определяются не на глобальном объекте, а как аргументы функции-обёртки require.
> pm2
В прошлый раз, когда я про это упоминал, местной аудитории это очень не понравилось (вплоть до ограничения на комментирование), но я всё равно повторю: не стоит думать, что pm2/forever в вакууме — надёжный способ перезапуска приложения в продакшне. Чтобы гарантировать перезапуск, надо построить цепочку ватчеров вплоть до системного инита (и до аппаратного watchdog), и никак иначе. pm2/forever тоже могут упасть (и делают это). А если вы используете pm2/forever _исключительно_ для перезапуска — тогда он вам вообще не нужен, скорее всего.
> uncaughtException
Ещё стоит упомянуть unhandledRejection.
> Обработка асинхронных ошибок
> Прослушиваем uncaughtException.
Нет, единственное, что стоит делать при возникновении uncaughtException — закрывать всё и завершать процесс, вы же сами это упоминали.
А для красивой обработки асинхронных ошибок есть Promises, можно в сочетании с async/await.
> Аддоны на C++
Первое, на что стоит смотреть для написания нативных аддонов — nan. Если вы будете дёргать v8 руками — пострадает совместимость с версиями Node.js, и при обновлении вам с большей вероятностью придётся что-то переписывать. nan берёт это на себя.
Suvitruf
Странно, что заикнулись про callback hell, но ни слова не сказали, про Promises. Тот же Bluebird.
napa3um
К слову, node уже достаточно хорошо покрывает ES2015 без полифилов, в том числе и Promise: http://node.green/.
gribnoysup
Да, но bluebird все еще быстрее, чем нативные промисы
ChALkeRx
Так-то оно так, но я бы до сих пор рекомендовал использовать Bluebird.
Во-первых, у нативных промисов всё ещё есть некоторые проблемы, связанные со скоростью работы и занимаемой памятью.
Во-вторых, пока нативной поддержки API на промисах нет — без promisifyAll весьма тяжело.