UNPKG

13.4 kBJavaScriptView Raw
1"use strict";
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.default = void 0;
7
8var _jsonStringifySafe = _interopRequireDefault(require("json-stringify-safe"));
9
10var _lodash = _interopRequireDefault(require("lodash"));
11
12var _httpUtils = require("./httpUtils");
13
14var _stringUtils = require("./stringUtils");
15
16function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
17
18function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
19
20function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
21
22function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
23
24function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
25
26const LOG_FLUSH_INTERVAL_MS = 5 * 1000;
27
28class S3Logger {
29 constructor(options) {
30 _defineProperty(this, "_logObjects", []);
31
32 _defineProperty(this, "_logRecordNum", 0);
33
34 _defineProperty(this, "_options", void 0);
35
36 _defineProperty(this, "queueLogRecord", logObj => {
37 try {
38 const nowMS = Date.now();
39
40 let {
41 data,
42 message
43 } = logObj,
44 restLogObj = _objectWithoutProperties(logObj, ["data", "message"]);
45
46 const objToPush = _objectSpread({
47 event: message
48 }, restLogObj, {
49 record_num: ++this._logRecordNum,
50 time_ms: nowMS,
51 time_ts: new Date(nowMS).toISOString()
52 });
53
54 if (data != null) {
55 // make sure data is a object
56 if (typeof data !== 'object') {
57 data = {
58 value: data.toString()
59 };
60 } // promote data fields
61
62
63 if (this._options.dataFieldsToPromote) {
64 Object.entries(this._options.dataFieldsToPromote).forEach(([logProp, dataPath]) => {
65 // $FlowIgnore
66 const dataVal = _lodash.default.get(data, dataPath);
67
68 if (dataVal !== undefined) {
69 objToPush[logProp] = dataVal;
70 }
71 });
72 } // remove data fields
73
74
75 if (this._options.dataFieldsToRemove) {
76 this._options.dataFieldsToRemove.forEach(dataProp => {
77 delete data[dataProp];
78 });
79 } // serialize data
80
81
82 if (Object.keys(data).length > 0) objToPush.data = (0, _jsonStringifySafe.default)(data);
83 } // log to console
84
85
86 console.log(`${objToPush.time_ts.substr(11, 12)}: ${(0, _stringUtils.padRight)(objToPush.level, 5, ' ')} [${objToPush.module}] ${objToPush.event} ${objToPush.data || ''}`); // eslint-disable-line no-console
87
88 this._logObjects.push(objToPush);
89 } catch (err) {
90 console.error('Failed to queue a log record', logObj, err); // eslint-disable-line no-console
91 }
92 });
93
94 _defineProperty(this, "flushLogs", async () => {
95 try {
96 if (!this._logObjects.length || !this._options.host) return;
97 const logsToFlush = this._logObjects;
98 this._logObjects = [];
99 await (0, _httpUtils.httpReq)(`${this._options.noHTTPS ? 'http' : 'https'}://${this._options.host}/p1`, {
100 method: 'POST',
101 body: {
102 beacons: logsToFlush,
103 common: this._options.systemMeta,
104 client_sending_time_ms: Date.now()
105 },
106 timeout: 10000
107 });
108 } catch (err) {
109 console.error('Failed to flush logs', new Date().toUTCString(), err); // eslint-disable-line no-console
110 }
111 });
112
113 this._options = options;
114
115 if (this._options.host) {
116 setInterval(this.flushLogs, LOG_FLUSH_INTERVAL_MS);
117 }
118 }
119
120}
121
122exports.default = S3Logger;
123//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/S3Logger.js"],"names":["LOG_FLUSH_INTERVAL_MS","S3Logger","constructor","options","logObj","nowMS","Date","now","data","message","restLogObj","objToPush","event","record_num","_logRecordNum","time_ms","time_ts","toISOString","value","toString","_options","dataFieldsToPromote","Object","entries","forEach","logProp","dataPath","dataVal","_","get","undefined","dataFieldsToRemove","dataProp","keys","length","console","log","substr","level","module","_logObjects","push","err","error","host","logsToFlush","noHTTPS","method","body","beacons","common","systemMeta","client_sending_time_ms","timeout","toUTCString","setInterval","flushLogs"],"mappings":";;;;;;;AAIA;;AACA;;AAEA;;AACA;;;;;;;;;;;;AAEA,MAAMA,qBAAqB,GAAG,IAAI,IAAlC;;AAUe,MAAMC,QAAN,CAAe;AAO5BC,EAAAA,WAAW,CAACC,OAAD,EAAyB;AAAA,yCALP,EAKO;;AAAA,2CAJZ,CAIY;;AAAA;;AAAA,4CASlBC,MAAD,IAA0B;AACzC,UAAI;AACF,cAAMC,KAAK,GAAGC,IAAI,CAACC,GAAL,EAAd;;AAEA,YAAI;AAAEC,UAAAA,IAAF;AAAQC,UAAAA;AAAR,YAAmCL,MAAvC;AAAA,YAAwBM,UAAxB,4BAAuCN,MAAvC;;AAEA,cAAMO,SAAS;AACbC,UAAAA,KAAK,EAAEH;AADM,WAGVC,UAHU;AAKbG,UAAAA,UAAU,EAAE,EAAE,KAAKC,aALN;AAMbC,UAAAA,OAAO,EAAEV,KANI;AAObW,UAAAA,OAAO,EAAE,IAAIV,IAAJ,CAASD,KAAT,EAAgBY,WAAhB;AAPI,UAAf;;AAUA,YAAIT,IAAI,IAAI,IAAZ,EAAkB;AAEhB;AACA,cAAI,OAAOA,IAAP,KAAgB,QAApB,EAA8B;AAC5BA,YAAAA,IAAI,GAAG;AACLU,cAAAA,KAAK,EAAEV,IAAI,CAACW,QAAL;AADF,aAAP;AAGD,WAPe,CAShB;;;AACA,cAAI,KAAKC,QAAL,CAAcC,mBAAlB,EAAuC;AACrCC,YAAAA,MAAM,CAACC,OAAP,CAAe,KAAKH,QAAL,CAAcC,mBAA7B,EAAkDG,OAAlD,CAA0D,CAAC,CAACC,OAAD,EAAUC,QAAV,CAAD,KAAyB;AACjF;AACA,oBAAMC,OAAO,GAAGC,gBAAEC,GAAF,CAAMrB,IAAN,EAAYkB,QAAZ,CAAhB;;AACA,kBAAIC,OAAO,KAAKG,SAAhB,EAA2B;AACzBnB,gBAAAA,SAAS,CAACc,OAAD,CAAT,GAAqBE,OAArB;AACD;AACF,aAND;AAOD,WAlBe,CAoBhB;;;AACA,cAAI,KAAKP,QAAL,CAAcW,kBAAlB,EAAsC;AACpC,iBAAKX,QAAL,CAAcW,kBAAd,CAAiCP,OAAjC,CAAyCQ,QAAQ,IAAI;AACnD,qBAAOxB,IAAI,CAACwB,QAAD,CAAX;AACD,aAFD;AAGD,WAzBe,CA2BhB;;;AACA,cAAIV,MAAM,CAACW,IAAP,CAAYzB,IAAZ,EAAkB0B,MAAlB,GAA2B,CAA/B,EACEvB,SAAS,CAACH,IAAV,GAAiB,gCAAUA,IAAV,CAAjB;AAEH,SA9CC,CAgDF;;;AACA2B,QAAAA,OAAO,CAACC,GAAR,CAAa,GAAEzB,SAAS,CAACK,OAAV,CAAkBqB,MAAlB,CAAyB,EAAzB,EAA6B,EAA7B,CAAiC,KAAI,2BAAS1B,SAAS,CAAC2B,KAAnB,EAA0B,CAA1B,EAA6B,GAA7B,CAAkC,KAAI3B,SAAS,CAAC4B,MAAO,KAAI5B,SAAS,CAACC,KAAM,IAAGD,SAAS,CAACH,IAAV,IAAkB,EAAG,EAAvJ,EAjDE,CAiDwJ;;AAE1J,aAAKgC,WAAL,CAAiBC,IAAjB,CAAsB9B,SAAtB;AACD,OApDD,CAoDE,OAAO+B,GAAP,EAAY;AACZP,QAAAA,OAAO,CAACQ,KAAR,CAAc,8BAAd,EAA8CvC,MAA9C,EAAsDsC,GAAtD,EADY,CAC+C;AAC5D;AACF,KAjEmC;;AAAA,uCAoExB,YAA2B;AACrC,UAAI;AACF,YAAI,CAAC,KAAKF,WAAL,CAAiBN,MAAlB,IAA4B,CAAC,KAAKd,QAAL,CAAcwB,IAA/C,EACE;AAEF,cAAMC,WAAW,GAAG,KAAKL,WAAzB;AACA,aAAKA,WAAL,GAAmB,EAAnB;AAEA,cAAM,wBAAS,GAAE,KAAKpB,QAAL,CAAc0B,OAAd,GAAwB,MAAxB,GAAiC,OAAQ,MAAK,KAAK1B,QAAL,CAAcwB,IAAK,KAA5E,EAAkF;AACtFG,UAAAA,MAAM,EAAE,MAD8E;AAEtFC,UAAAA,IAAI,EAAE;AACJC,YAAAA,OAAO,EAAEJ,WADL;AAEJK,YAAAA,MAAM,EAAE,KAAK9B,QAAL,CAAc+B,UAFlB;AAGJC,YAAAA,sBAAsB,EAAE9C,IAAI,CAACC,GAAL;AAHpB,WAFgF;AAOtF8C,UAAAA,OAAO,EAAE;AAP6E,SAAlF,CAAN;AAUD,OAjBD,CAiBE,OAAOX,GAAP,EAAY;AACZP,QAAAA,OAAO,CAACQ,KAAR,CAAc,sBAAd,EAAuC,IAAIrC,IAAJ,EAAD,CAAagD,WAAb,EAAtC,EAAkEZ,GAAlE,EADY,CAC2D;AACxE;AACF,KAzFmC;;AAClC,SAAKtB,QAAL,GAAgBjB,OAAhB;;AAEA,QAAI,KAAKiB,QAAL,CAAcwB,IAAlB,EAAwB;AACtBW,MAAAA,WAAW,CAAC,KAAKC,SAAN,EAAiBxD,qBAAjB,CAAX;AACD;AACF;;AAb2B","sourcesContent":["// @flow\n\n// NOTE: don't use Module here, since Module uses this class\n\nimport stringify from 'json-stringify-safe'\nimport _ from 'lodash'\n\nimport { httpReq } from './httpUtils'\nimport { padRight } from './stringUtils'\n\nconst LOG_FLUSH_INTERVAL_MS = 5 * 1000\n\ntype LoggerOptions = {\n  noHTTPS?: boolean,\n  host?: ?string,\n  dataFieldsToPromote?: ?{ [id: string]: string },\n  dataFieldsToRemove?: ?Array<string>, \n  systemMeta?: ?{},\n}\n\nexport default class S3Logger {\n\n  _logObjects: Array<Object> = []\n  _logRecordNum: number = 0\n  _options: LoggerOptions\n\n\n  constructor(options: LoggerOptions) {\n    this._options = options\n\n    if (this._options.host) {\n      setInterval(this.flushLogs, LOG_FLUSH_INTERVAL_MS)\n    }\n  }\n\n\n  queueLogRecord = (logObj: Object): void => {\n    try {\n      const nowMS = Date.now()\n\n      let { data, message, ...restLogObj } = logObj\n\n      const objToPush = {\n        event: message,\n\n        ...restLogObj,\n\n        record_num: ++this._logRecordNum,\n        time_ms: nowMS,\n        time_ts: new Date(nowMS).toISOString(),\n      }\n\n      if (data != null) {\n\n        // make sure data is a object\n        if (typeof data !== 'object') {\n          data = {\n            value: data.toString()\n          }\n        }\n\n        // promote data fields\n        if (this._options.dataFieldsToPromote) {\n          Object.entries(this._options.dataFieldsToPromote).forEach(([logProp, dataPath]) => {\n            // $FlowIgnore\n            const dataVal = _.get(data, dataPath)\n            if (dataVal !== undefined) {\n              objToPush[logProp] = dataVal\n            }\n          })\n        }\n\n        // remove data fields\n        if (this._options.dataFieldsToRemove) {\n          this._options.dataFieldsToRemove.forEach(dataProp => {\n            delete data[dataProp]\n          })\n        }\n\n        // serialize data\n        if (Object.keys(data).length > 0)\n          objToPush.data = stringify(data)\n\n      }\n\n      // log to console\n      console.log(`${objToPush.time_ts.substr(11, 12)}: ${padRight(objToPush.level, 5, ' ')} [${objToPush.module}] ${objToPush.event} ${objToPush.data || ''}`) // eslint-disable-line no-console\n  \n      this._logObjects.push(objToPush)\n    } catch (err) {\n      console.error('Failed to queue a log record', logObj, err) // eslint-disable-line no-console\n    }\n  }\n\n\n  flushLogs = async (): Promise<void> => {\n    try {\n      if (!this._logObjects.length || !this._options.host)\n        return\n\n      const logsToFlush = this._logObjects\n      this._logObjects = []\n\n      await httpReq(`${this._options.noHTTPS ? 'http' : 'https'}://${this._options.host}/p1`, {\n        method: 'POST', \n        body: {\n          beacons: logsToFlush,\n          common: this._options.systemMeta,\n          client_sending_time_ms: Date.now(),\n        },\n        timeout: 10000,\n      })\n\n    } catch (err) {\n      console.error('Failed to flush logs', (new Date()).toUTCString(), err) // eslint-disable-line no-console\n    }\n  }\n}\n"]}
\No newline at end of file