gelf.js 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. "use strict";
  2. var zlib = require('zlib');
  3. var layouts = require('../layouts');
  4. var levels = require('../levels');
  5. var dgram = require('dgram');
  6. var util = require('util');
  7. var debug = require('../debug')('GELF Appender');
  8. var LOG_EMERG=0; // system is unusable
  9. var LOG_ALERT=1; // action must be taken immediately
  10. var LOG_CRIT=2; // critical conditions
  11. var LOG_ERR=3; // error conditions
  12. var LOG_ERROR=3; // because people WILL typo
  13. var LOG_WARNING=4; // warning conditions
  14. var LOG_NOTICE=5; // normal, but significant, condition
  15. var LOG_INFO=6; // informational message
  16. var LOG_DEBUG=7; // debug-level message
  17. var levelMapping = {};
  18. levelMapping[levels.ALL] = LOG_DEBUG;
  19. levelMapping[levels.TRACE] = LOG_DEBUG;
  20. levelMapping[levels.DEBUG] = LOG_DEBUG;
  21. levelMapping[levels.INFO] = LOG_INFO;
  22. levelMapping[levels.WARN] = LOG_WARNING;
  23. levelMapping[levels.ERROR] = LOG_ERR;
  24. levelMapping[levels.FATAL] = LOG_CRIT;
  25. /**
  26. * GELF appender that supports sending UDP packets to a GELF compatible server such as Graylog
  27. *
  28. * @param layout a function that takes a logevent and returns a string (defaults to none).
  29. * @param host - host to which to send logs (default:localhost)
  30. * @param port - port at which to send logs to (default:12201)
  31. * @param hostname - hostname of the current host (default:os hostname)
  32. * @param facility - facility to log to (default:nodejs-server)
  33. */
  34. function gelfAppender (layout, host, port, hostname, facility) {
  35. var config, customFields;
  36. if (typeof(host) === 'object') {
  37. config = host;
  38. host = config.host;
  39. port = config.port;
  40. hostname = config.hostname;
  41. facility = config.facility;
  42. customFields = config.customFields;
  43. }
  44. host = host || 'localhost';
  45. port = port || 12201;
  46. hostname = hostname || require('os').hostname();
  47. facility = facility || 'nodejs-server';
  48. layout = layout || layouts.messagePassThroughLayout;
  49. var defaultCustomFields = customFields || {};
  50. var client = dgram.createSocket("udp4");
  51. process.on('exit', function() {
  52. if (client) client.close();
  53. });
  54. /**
  55. * Add custom fields (start with underscore )
  56. * - if the first object passed to the logger contains 'GELF' field,
  57. * copy the underscore fields to the message
  58. * @param loggingEvent
  59. * @param msg
  60. */
  61. function addCustomFields(loggingEvent, msg){
  62. /* append defaultCustomFields firsts */
  63. Object.keys(defaultCustomFields).forEach(function(key) {
  64. // skip _id field for graylog2, skip keys not starts with UNDERSCORE
  65. if (key.match(/^_/) && key !== "_id") {
  66. msg[key] = defaultCustomFields[key];
  67. }
  68. });
  69. /* append custom fields per message */
  70. var data = loggingEvent.data;
  71. if (!Array.isArray(data) || data.length === 0) return;
  72. var firstData = data[0];
  73. if (!firstData.GELF) return; // identify with GELF field defined
  74. Object.keys(firstData).forEach(function(key) {
  75. // skip _id field for graylog2, skip keys not starts with UNDERSCORE
  76. if (key.match(/^_/) || key !== "_id") {
  77. msg[key] = firstData[key];
  78. }
  79. });
  80. /* the custom field object should be removed, so it will not be looged by the later appenders */
  81. loggingEvent.data.shift();
  82. }
  83. function preparePacket(loggingEvent) {
  84. var msg = {};
  85. addCustomFields(loggingEvent, msg);
  86. msg.full_message = layout(loggingEvent);
  87. msg.short_message = msg.full_message;
  88. msg.version="1.0";
  89. msg.timestamp = msg.timestamp || new Date().getTime() / 1000 >> 0;
  90. msg.host = hostname;
  91. msg.level = levelMapping[loggingEvent.level || levels.DEBUG];
  92. msg.facility = facility;
  93. return msg;
  94. }
  95. function sendPacket(packet) {
  96. try {
  97. client.send(packet, 0, packet.length, port, host);
  98. } catch(e) {}
  99. }
  100. return function(loggingEvent) {
  101. var message = preparePacket(loggingEvent);
  102. zlib.gzip(new Buffer(JSON.stringify(message)), function(err, packet) {
  103. if (err) {
  104. console.error(err.stack);
  105. } else {
  106. if (packet.length > 8192) {
  107. debug("Message packet length (" + packet.length + ") is larger than 8k. Not sending");
  108. } else {
  109. sendPacket(packet);
  110. }
  111. }
  112. });
  113. };
  114. }
  115. function configure(config) {
  116. var layout;
  117. if (config.layout) {
  118. layout = layouts.layout(config.layout.type, config.layout);
  119. }
  120. return gelfAppender(layout, config);
  121. }
  122. exports.appender = gelfAppender;
  123. exports.configure = configure;