Диагностический отчет
[Стабильно: 2 - Стабильно]
Стабильно: 2 Стабильность: 2 - Стабильно
[История]
Версия | Изменения |
---|---|
v23.3.0 | Добавлен параметр --report-exclude-env для исключения переменных среды из генерации отчета. |
v22.0.0, v20.13.0 | Добавлен параметр --report-exclude-network для исключения сетевых операций, которые в некоторых случаях могут замедлять генерацию отчета. |
Выдает сводку диагностики в формате JSON, записываемую в файл.
Отчет предназначен для использования в разработке, тестировании и промышленной эксплуатации для сбора и сохранения информации для определения проблем. Он включает в себя трассировки стека JavaScript и нативного кода, статистику кучи, информацию о платформе, использование ресурсов и т.д. При включенном параметре отчета диагностические отчеты могут быть инициированы при необработанных исключениях, фатальных ошибках и сигналах пользователя, в дополнение к программному запуску через вызовы API.
Ниже приведен пример полного отчета, сгенерированного при неперехваченном исключении, для справки.
{
"header": {
"reportVersion": 5,
"event": "exception",
"trigger": "Exception",
"filename": "report.20181221.005011.8974.0.001.json",
"dumpEventTime": "2018-12-21T00:50:11Z",
"dumpEventTimeStamp": "1545371411331",
"processId": 8974,
"cwd": "/home/nodeuser/project/node",
"commandLine": [
"/home/nodeuser/project/node/out/Release/node",
"--report-uncaught-exception",
"/home/nodeuser/project/node/test/report/test-exception.js",
"child"
],
"nodejsVersion": "v12.0.0-pre",
"glibcVersionRuntime": "2.17",
"glibcVersionCompiler": "2.17",
"wordSize": "64 bit",
"arch": "x64",
"platform": "linux",
"componentVersions": {
"node": "12.0.0-pre",
"v8": "7.1.302.28-node.5",
"uv": "1.24.1",
"zlib": "1.2.11",
"ares": "1.15.0",
"modules": "68",
"nghttp2": "1.34.0",
"napi": "3",
"llhttp": "1.0.1",
"openssl": "1.1.0j"
},
"release": {
"name": "node"
},
"osName": "Linux",
"osRelease": "3.10.0-862.el7.x86_64",
"osVersion": "#1 SMP Wed Mar 21 18:14:51 EDT 2018",
"osMachine": "x86_64",
"cpus": [
{
"model": "Intel(R) Core(TM) i7-6820HQ CPU @ 2.70GHz",
"speed": 2700,
"user": 88902660,
"nice": 0,
"sys": 50902570,
"idle": 241732220,
"irq": 0
},
{
"model": "Intel(R) Core(TM) i7-6820HQ CPU @ 2.70GHz",
"speed": 2700,
"user": 88902660,
"nice": 0,
"sys": 50902570,
"idle": 241732220,
"irq": 0
}
],
"networkInterfaces": [
{
"name": "en0",
"internal": false,
"mac": "13:10:de:ad:be:ef",
"address": "10.0.0.37",
"netmask": "255.255.255.0",
"family": "IPv4"
}
],
"host": "test_machine"
},
"javascriptStack": {
"message": "Error: *** test-exception.js: throwing uncaught Error",
"stack": [
"at myException (/home/nodeuser/project/node/test/report/test-exception.js:9:11)",
"at Object.<anonymous> (/home/nodeuser/project/node/test/report/test-exception.js:12:3)",
"at Module._compile (internal/modules/cjs/loader.js:718:30)",
"at Object.Module._extensions..js (internal/modules/cjs/loader.js:729:10)",
"at Module.load (internal/modules/cjs/loader.js:617:32)",
"at tryModuleLoad (internal/modules/cjs/loader.js:560:12)",
"at Function.Module._load (internal/modules/cjs/loader.js:552:3)",
"at Function.Module.runMain (internal/modules/cjs/loader.js:771:12)",
"at executeUserCode (internal/bootstrap/node.js:332:15)"
]
},
"nativeStack": [
{
"pc": "0x000055b57f07a9ef",
"symbol": "report::GetNodeReport(v8::Isolate*, node::Environment*, char const*, char const*, v8::Local<v8::String>, std::ostream&) [./node]"
},
{
"pc": "0x000055b57f07cf03",
"symbol": "report::GetReport(v8::FunctionCallbackInfo<v8::Value> const&) [./node]"
},
{
"pc": "0x000055b57f1bccfd",
"symbol": " [./node]"
},
{
"pc": "0x000055b57f1be048",
"symbol": "v8::internal::Builtin_HandleApiCall(int, v8::internal::Object**, v8::internal::Isolate*) [./node]"
},
{
"pc": "0x000055b57feeda0e",
"symbol": " [./node]"
}
],
"javascriptHeap": {
"totalMemory": 5660672,
"executableMemory": 524288,
"totalCommittedMemory": 5488640,
"availableMemory": 4341379928,
"totalGlobalHandlesMemory": 8192,
"usedGlobalHandlesMemory": 3136,
"usedMemory": 4816432,
"memoryLimit": 4345298944,
"mallocedMemory": 254128,
"externalMemory": 315644,
"peakMallocedMemory": 98752,
"nativeContextCount": 1,
"detachedContextCount": 0,
"doesZapGarbage": 0,
"heapSpaces": {
"read_only_space": {
"memorySize": 524288,
"committedMemory": 39208,
"capacity": 515584,
"used": 30504,
"available": 485080
},
"new_space": {
"memorySize": 2097152,
"committedMemory": 2019312,
"capacity": 1031168,
"used": 985496,
"available": 45672
},
"old_space": {
"memorySize": 2273280,
"committedMemory": 1769008,
"capacity": 1974640,
"used": 1725488,
"available": 249152
},
"code_space": {
"memorySize": 696320,
"committedMemory": 184896,
"capacity": 152128,
"used": 152128,
"available": 0
},
"map_space": {
"memorySize": 536576,
"committedMemory": 344928,
"capacity": 327520,
"used": 327520,
"available": 0
},
"large_object_space": {
"memorySize": 0,
"committedMemory": 0,
"capacity": 1520590336,
"used": 0,
"available": 1520590336
},
"new_large_object_space": {
"memorySize": 0,
"committedMemory": 0,
"capacity": 0,
"used": 0,
"available": 0
}
}
},
"resourceUsage": {
"rss": "35766272",
"free_memory": "1598337024",
"total_memory": "17179869184",
"available_memory": "1598337024",
"maxRss": "36624662528",
"constrained_memory": "36624662528",
"userCpuSeconds": 0.040072,
"kernelCpuSeconds": 0.016029,
"cpuConsumptionPercent": 5.6101,
"userCpuConsumptionPercent": 4.0072,
"kernelCpuConsumptionPercent": 1.6029,
"pageFaults": {
"IORequired": 0,
"IONotRequired": 4610
},
"fsActivity": {
"reads": 0,
"writes": 0
}
},
"uvthreadResourceUsage": {
"userCpuSeconds": 0.039843,
"kernelCpuSeconds": 0.015937,
"cpuConsumptionPercent": 5.578,
"userCpuConsumptionPercent": 3.9843,
"kernelCpuConsumptionPercent": 1.5937,
"fsActivity": {
"reads": 0,
"writes": 0
}
},
"libuv": [
{
"type": "async",
"is_active": true,
"is_referenced": false,
"address": "0x0000000102910900",
"details": ""
},
{
"type": "timer",
"is_active": false,
"is_referenced": false,
"address": "0x00007fff5fbfeab0",
"repeat": 0,
"firesInMsFromNow": 94403548320796,
"expired": true
},
{
"type": "check",
"is_active": true,
"is_referenced": false,
"address": "0x00007fff5fbfeb48"
},
{
"type": "idle",
"is_active": false,
"is_referenced": true,
"address": "0x00007fff5fbfebc0"
},
{
"type": "prepare",
"is_active": false,
"is_referenced": false,
"address": "0x00007fff5fbfec38"
},
{
"type": "check",
"is_active": false,
"is_referenced": false,
"address": "0x00007fff5fbfecb0"
},
{
"type": "async",
"is_active": true,
"is_referenced": false,
"address": "0x000000010188f2e0"
},
{
"type": "tty",
"is_active": false,
"is_referenced": true,
"address": "0x000055b581db0e18",
"width": 204,
"height": 55,
"fd": 17,
"writeQueueSize": 0,
"readable": true,
"writable": true
},
{
"type": "signal",
"is_active": true,
"is_referenced": false,
"address": "0x000055b581d80010",
"signum": 28,
"signal": "SIGWINCH"
},
{
"type": "tty",
"is_active": true,
"is_referenced": true,
"address": "0x000055b581df59f8",
"width": 204,
"height": 55,
"fd": 19,
"writeQueueSize": 0,
"readable": true,
"writable": true
},
{
"type": "loop",
"is_active": true,
"address": "0x000055fc7b2cb180",
"loopIdleTimeSeconds": 22644.8
},
{
"type": "tcp",
"is_active": true,
"is_referenced": true,
"address": "0x000055e70fcb85d8",
"localEndpoint": {
"host": "localhost",
"ip4": "127.0.0.1",
"port": 48986
},
"remoteEndpoint": {
"host": "localhost",
"ip4": "127.0.0.1",
"port": 38573
},
"sendBufferSize": 2626560,
"recvBufferSize": 131072,
"fd": 24,
"writeQueueSize": 0,
"readable": true,
"writable": true
}
],
"workers": [],
"environmentVariables": {
"REMOTEHOST": "REMOVED",
"MANPATH": "/opt/rh/devtoolset-3/root/usr/share/man:",
"XDG_SESSION_ID": "66126",
"HOSTNAME": "test_machine",
"HOST": "test_machine",
"TERM": "xterm-256color",
"SHELL": "/bin/csh",
"SSH_CLIENT": "REMOVED",
"PERL5LIB": "/opt/rh/devtoolset-3/root//usr/lib64/perl5/vendor_perl:/opt/rh/devtoolset-3/root/usr/lib/perl5:/opt/rh/devtoolset-3/root//usr/share/perl5/vendor_perl",
"OLDPWD": "/home/nodeuser/project/node/src",
"JAVACONFDIRS": "/opt/rh/devtoolset-3/root/etc/java:/etc/java",
"SSH_TTY": "/dev/pts/0",
"PCP_DIR": "/opt/rh/devtoolset-3/root",
"GROUP": "normaluser",
"USER": "nodeuser",
"LD_LIBRARY_PATH": "/opt/rh/devtoolset-3/root/usr/lib64:/opt/rh/devtoolset-3/root/usr/lib",
"HOSTTYPE": "x86_64-linux",
"XDG_CONFIG_DIRS": "/opt/rh/devtoolset-3/root/etc/xdg:/etc/xdg",
"MAIL": "/var/spool/mail/nodeuser",
"PATH": "/home/nodeuser/project/node:/opt/rh/devtoolset-3/root/usr/bin:/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin",
"PWD": "/home/nodeuser/project/node",
"LANG": "en_US.UTF-8",
"PS1": "\\u@\\h : \\[\\e[31m\\]\\w\\[\\e[m\\] > ",
"SHLVL": "2",
"HOME": "/home/nodeuser",
"OSTYPE": "linux",
"VENDOR": "unknown",
"PYTHONPATH": "/opt/rh/devtoolset-3/root/usr/lib64/python2.7/site-packages:/opt/rh/devtoolset-3/root/usr/lib/python2.7/site-packages",
"MACHTYPE": "x86_64",
"LOGNAME": "nodeuser",
"XDG_DATA_DIRS": "/opt/rh/devtoolset-3/root/usr/share:/usr/local/share:/usr/share",
"LESSOPEN": "||/usr/bin/lesspipe.sh %s",
"INFOPATH": "/opt/rh/devtoolset-3/root/usr/share/info",
"XDG_RUNTIME_DIR": "/run/user/50141",
"_": "./node"
},
"userLimits": {
"core_file_size_blocks": {
"soft": "",
"hard": "unlimited"
},
"data_seg_size_bytes": {
"soft": "unlimited",
"hard": "unlimited"
},
"file_size_blocks": {
"soft": "unlimited",
"hard": "unlimited"
},
"max_locked_memory_bytes": {
"soft": "unlimited",
"hard": 65536
},
"max_memory_size_bytes": {
"soft": "unlimited",
"hard": "unlimited"
},
"open_files": {
"soft": "unlimited",
"hard": 4096
},
"stack_size_bytes": {
"soft": "unlimited",
"hard": "unlimited"
},
"cpu_time_seconds": {
"soft": "unlimited",
"hard": "unlimited"
},
"max_user_processes": {
"soft": "unlimited",
"hard": 4127290
},
"virtual_memory_bytes": {
"soft": "unlimited",
"hard": "unlimited"
}
},
"sharedObjects": [
"/lib64/libdl.so.2",
"/lib64/librt.so.1",
"/lib64/libstdc++.so.6",
"/lib64/libm.so.6",
"/lib64/libgcc_s.so.1",
"/lib64/libpthread.so.0",
"/lib64/libc.so.6",
"/lib64/ld-linux-x86-64.so.2"
]
}
Использование
node --report-uncaught-exception --report-on-signal \
--report-on-fatalerror app.js
--report-uncaught-exception
Включает генерацию отчета при неперехваченных исключениях. Полезно при инспектировании стека JavaScript в сочетании со стеком нативного кода и другими данными среды выполнения.--report-on-signal
Включает генерацию отчета при получении указанного (или предопределенного) сигнала работающим процессом Node.js. (См. ниже, как изменить сигнал, который запускает отчет.) Сигнал по умолчанию —SIGUSR2
. Полезно, когда отчет необходимо сгенерировать из другой программы. Мониторы приложений могут использовать эту функцию для сбора отчетов через регулярные интервалы и отображения богатого набора внутренних данных среды выполнения в своих представлениях.
Генерация отчетов на основе сигналов не поддерживается в Windows.
В обычных условиях нет необходимости изменять сигнал, запускающий отчет. Однако, если SIGUSR2
уже используется для других целей, этот флаг помогает изменить сигнал для генерации отчета и сохранить исходное значение SIGUSR2
для указанных целей.
--report-on-fatalerror
Включает генерацию отчета при фатальных ошибках (внутренние ошибки в среде выполнения Node.js, такие как нехватка памяти), которые приводят к завершению работы приложения. Полезно для проверки различных элементов диагностических данных, таких как куча, стек, состояние цикла событий, потребление ресурсов и т. д., для анализа фатальной ошибки.--report-compact
Записывает отчеты в компактном формате, однострочном JSON, более удобном для обработки системами логирования, чем многострочный формат по умолчанию, предназначенный для чтения человеком.--report-directory
Расположение, в котором будет создан отчет.--report-filename
Имя файла, в который будет записан отчет.--report-signal
Устанавливает или сбрасывает сигнал для генерации отчета (не поддерживается в Windows). Сигнал по умолчанию —SIGUSR2
.--report-exclude-network
Исключаетheader.networkInterfaces
и отключает обратные DNS-запросы вlibuv.*.(remote|local)Endpoint.host
из диагностического отчета. По умолчанию этот параметр не установлен, и сетевые интерфейсы включены.--report-exclude-env
ИсключаетenvironmentVariables
из диагностического отчета. По умолчанию этот параметр не установлен, и переменные среды включены.
Отчет также может быть запущен с помощью вызова API из приложения JavaScript:
process.report.writeReport()
Эта функция принимает необязательный дополнительный аргумент filename
, который представляет собой имя файла, в который будет записан отчет.
process.report.writeReport('./foo.json')
Эта функция принимает необязательный дополнительный аргумент err
, который представляет собой объект Error
, который будет использоваться в качестве контекста для стека JavaScript, выводимого в отчете. При использовании отчета для обработки ошибок в обработчике обратного вызова или исключения это позволяет отчету включать местоположение исходной ошибки, а также место, где она была обработана.
try {
process.chdir('/non-existent-path')
} catch (err) {
process.report.writeReport(err)
}
// Любой другой код
Если в writeReport()
передаются как имя файла, так и объект ошибки, объект ошибки должен быть вторым параметром.
try {
process.chdir('/non-existent-path')
} catch (err) {
process.report.writeReport(filename, err)
}
// Любой другой код
Содержимое диагностического отчета может быть возвращено в виде объекта JavaScript с помощью вызова API из приложения JavaScript:
const report = process.report.getReport()
console.log(typeof report === 'object') // true
// Аналогично выводу process.report.writeReport()
console.log(JSON.stringify(report, null, 2))
Эта функция принимает необязательный дополнительный аргумент err
, который представляет собой объект Error
, который будет использоваться в качестве контекста для стека JavaScript, выводимого в отчете.
const report = process.report.getReport(new Error('custom error'))
console.log(typeof report === 'object') // true
Версии API полезны при проверке состояния среды выполнения изнутри приложения, в ожидании саморегулирования потребления ресурсов, балансировки нагрузки, мониторинга и т. д.
Содержимое отчета состоит из раздела заголовка, содержащего тип события, дату, время, PID и версию Node.js, разделов, содержащих трассировки стека JavaScript и нативного кода, раздела, содержащего информацию о куче V8, раздела, содержащего информацию о дескрипторах libuv
, и раздела информации о платформе ОС, показывающего использование ЦП и памяти, а также системные лимиты. Пример отчета может быть запущен с помощью REPL Node.js:
$ node
> process.report.writeReport();
Writing Node.js report to file: report.20181126.091102.8480.0.001.json
Node.js report completed
>
Когда отчет записывается, начальные и конечные сообщения выводятся в stderr, а имя файла отчета возвращается вызывающему объекту. Имя файла по умолчанию включает дату, время, PID и порядковый номер. Порядковый номер помогает связать дамп отчета с состоянием среды выполнения, если он генерируется несколько раз для одного и того же процесса Node.js.
Версия отчета
Диагностический отчет имеет связанный однозначный номер версии (report.header.reportVersion
), однозначно представляющий формат отчета. Номер версии увеличивается при добавлении или удалении нового ключа или изменении типа данных значения. Определения версий отчетов согласованы во всех LTS-релизах.
История версий
Версия 5
[История]
Версия | Изменения |
---|---|
v23.5.0 | Исправлены опечатки в единицах ограничения памяти. |
Замените ключи data_seg_size_kbytes
, max_memory_size_kbytes
и virtual_memory_kbytes
на data_seg_size_bytes
, max_memory_size_bytes
и virtual_memory_bytes
соответственно в разделе userLimits
, поскольку эти значения заданы в байтах.
{
"userLimits": {
// Пропуск некоторых ключей ...
"data_seg_size_bytes": {
// замена data_seg_size_kbytes
"soft": "unlimited",
"hard": "unlimited"
},
// ...
"max_memory_size_bytes": {
// замена max_memory_size_kbytes
"soft": "unlimited",
"hard": "unlimited"
},
// ...
"virtual_memory_bytes": {
// замена virtual_memory_kbytes
"soft": "unlimited",
"hard": "unlimited"
}
}
}
Версия 4
[История]
Версия | Изменения |
---|---|
v23.3.0 | Добавлен параметр --report-exclude-env для исключения переменных среды из генерации отчета. |
В конечные точки обработчиков libuv tcp
и udp
добавлены новые поля ipv4
и ipv6
. Примеры:
{
"libuv": [
{
"type": "tcp",
"is_active": true,
"is_referenced": true,
"address": "0x000055e70fcb85d8",
"localEndpoint": {
"host": "localhost",
"ip4": "127.0.0.1", // новый ключ
"port": 48986
},
"remoteEndpoint": {
"host": "localhost",
"ip4": "127.0.0.1", // новый ключ
"port": 38573
},
"sendBufferSize": 2626560,
"recvBufferSize": 131072,
"fd": 24,
"writeQueueSize": 0,
"readable": true,
"writable": true
},
{
"type": "tcp",
"is_active": true,
"is_referenced": true,
"address": "0x000055e70fcd68c8",
"localEndpoint": {
"host": "ip6-localhost",
"ip6": "::1", // новый ключ
"port": 52266
},
"remoteEndpoint": {
"host": "ip6-localhost",
"ip6": "::1", // новый ключ
"port": 38573
},
"sendBufferSize": 2626560,
"recvBufferSize": 131072,
"fd": 25,
"writeQueueSize": 0,
"readable": false,
"writable": false
}
]
}
Версия 3
[История]
Версия | Изменения |
---|---|
v19.1.0, v18.13.0 | Добавлена дополнительная информация о памяти. |
В раздел resourceUsage
добавлены следующие ключи использования памяти.
{
"resourceUsage": {
"rss": "35766272",
"free_memory": "1598337024",
"total_memory": "17179869184",
"available_memory": "1598337024",
"constrained_memory": "36624662528"
}
}
Версия 2
[История]
Версия | Изменения |
---|---|
v13.9.0, v12.16.2 | Рабочие потоки теперь включены в отчёт. |
Добавлена поддержка Worker
. Более подробную информацию см. в разделе Взаимодействие с рабочими потоками.
Версия 1
Это первая версия отчёта о диагностике.
Конфигурация
Дополнительная конфигурация времени выполнения генерации отчёта доступна через следующие свойства process.report
:
reportOnFatalError
запускает диагностический отчёт при фатальных ошибках, если значение true
. По умолчанию — false
.
reportOnSignal
запускает диагностический отчёт по сигналу, если значение true
. Не поддерживается в Windows. По умолчанию — false
.
reportOnUncaughtException
запускает диагностический отчёт при неперехваченном исключении, если значение true
. По умолчанию — false
.
signal
указывает идентификатор сигнала POSIX, который будет использоваться для перехвата внешних триггеров генерации отчёта. По умолчанию — 'SIGUSR2'
.
filename
указывает имя выходного файла в файловой системе. Особое значение имеют stdout
и stderr
. Их использование приведёт к записи отчёта в соответствующие стандартные потоки. В случаях использования стандартных потоков значение в directory
игнорируется. URL-адреса не поддерживаются. По умолчанию — составное имя файла, содержащее метку времени, PID и порядковый номер.
directory
указывает каталог файловой системы, куда будет записан отчёт. URL-адреса не поддерживаются. По умолчанию — текущий рабочий каталог процесса Node.js.
excludeNetwork
исключает header.networkInterfaces
из диагностического отчёта.
// Запуск отчёта только при неперехваченных исключениях.
process.report.reportOnFatalError = false
process.report.reportOnSignal = false
process.report.reportOnUncaughtException = true
// Запуск отчёта как для внутренних ошибок, так и для внешнего сигнала.
process.report.reportOnFatalError = true
process.report.reportOnSignal = true
process.report.reportOnUncaughtException = false
// Изменение сигнала по умолчанию на 'SIGQUIT' и его включение.
process.report.reportOnFatalError = false
process.report.reportOnUncaughtException = false
process.report.reportOnSignal = true
process.report.signal = 'SIGQUIT'
// Отключение отчётности о сетевых интерфейсах
process.report.excludeNetwork = true
Конфигурация при инициализации модуля также доступна через переменные среды:
NODE_OPTIONS="--report-uncaught-exception \
--report-on-fatalerror --report-on-signal \
--report-signal=SIGUSR2 --report-filename=./report.json \
--report-directory=/home/nodeuser"
Подробная документация по API находится в разделе документация по API процесса
.
Взаимодействие с воркерами
[История]
Версия | Изменения |
---|---|
v13.9.0, v12.16.2 | Воркеры теперь включены в отчет. |
Потоки Worker
могут создавать отчеты так же, как и главный поток.
Отчеты будут включать информацию о всех воркерах, являющихся дочерними для текущего потока, как часть секции workers
, при этом каждый воркер генерирует отчет в стандартном формате отчета.
Поток, генерирующий отчет, будет ожидать завершения отчетов от потоков воркеров. Однако задержка при этом обычно будет низкой, поскольку как выполнение JavaScript, так и цикл событий прерываются для генерации отчета.