UNPKG

15.8 kBJavaScriptView Raw
1"use strict";
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.default = void 0;
7
8var _autoBind = _interopRequireDefault(require("auto-bind"));
9
10var _System = _interopRequireDefault(require("./System"));
11
12var _Module = _interopRequireDefault(require("./Module"));
13
14var _asyncUtils = require("./asyncUtils");
15
16var _consts = require("./consts");
17
18function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
19
20function _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; }
21
22const {
23 MODULE_NAME,
24 log,
25 warn,
26 error,
27 noteGauge,
28 noteCount,
29 noteTimer,
30 trackOp
31} = new _Module.default(__filename); // eslint-disable-line no-unused-vars
32
33class Execution {
34 /**
35 * An async function that is called AFTER a termination has accured and infra resources were released,
36 * This handler is called after infra resources were flushed, so it should avoid using them.
37 */
38 constructor(options) {
39 _defineProperty(this, "_options", void 0);
40
41 _defineProperty(this, "_startTimeMS", void 0);
42
43 _defineProperty(this, "_endTimeMS", void 0);
44
45 _defineProperty(this, "_terminating", false);
46
47 _defineProperty(this, "_onTerminateHandlers", []);
48
49 _defineProperty(this, "terminationHandler", void 0);
50
51 this._options = options || {
52 dontTerminateOnCompletion: false
53 };
54 (0, _autoBind.default)(this);
55 }
56
57 async run(runFunc) {
58 try {
59 noteCount(`execution.start`); // Catches ctrl+c and PM2 shutdown
60
61 process.on('SIGINT', async () => await this._terminate(new Error('SIGINT')));
62
63 if (_System.default.getConfig().trackVitalsIntervalMS) {
64 this._trackVitals();
65 }
66
67 this._startTimeMS = Date.now();
68 log('Execution started');
69 const returnValue = await runFunc();
70
71 if (!this._options.dontTerminateOnCompletion) {
72 await this._terminate(null, returnValue);
73 }
74 } catch (err) {
75 await this._terminate(err);
76 }
77 }
78 /**
79 * Allows adding an async function that is called BEFORE a termination has accured.
80 * @param {*} terminationHandler
81 */
82
83
84 addOnTerminateHandler(onTerminateHandler) {
85 this._onTerminateHandlers.push(onTerminateHandler);
86 }
87
88 async _trackVitals() {
89 while (!this._terminating) {
90 try {
91 const used = process.memoryUsage();
92
93 for (const key in used) {
94 noteGauge(`memory.${key}MB`, Math.round(used[key] / 1024 / 1024 * 100) / 100);
95 }
96
97 await (0, _asyncUtils.sleep)(_System.default.getConfig().trackVitalsIntervalMS || 10 * _consts.SECOND_MS);
98 } catch (err) {
99 error('Failed to track vitals', {
100 err
101 });
102 await (0, _asyncUtils.sleep)(1 * _consts.MINUTE_MS);
103 }
104 }
105 }
106
107 async _terminate(terminationError, data) {
108 if (this._terminating) return;
109 this._terminating = true;
110 this._endTimeMS = Date.now(); // $FlowIgnore
111
112 const durationMS = this._endTimeMS - this._startTimeMS;
113
114 if (terminationError == null) {
115 await noteCount(`execution.succeed`);
116 await noteTimer('execution.succeed_duration', durationMS);
117 log('Execution ended successfully', {
118 durationMS
119 });
120 } else if (terminationError.message === 'SIGINT') {
121 await noteCount(`execution.terminate`);
122 await noteTimer('execution.terminate_duration', durationMS);
123 warn('Execution terminated by SIGINT signal, terminating...', {
124 durationMS
125 });
126 } else {
127 await noteCount(`execution.fail`);
128 await noteTimer('execution.fail_duration', durationMS);
129 error('Execution threw an unhandled exception, terminating...', {
130 durationMS,
131 err: terminationError
132 });
133 }
134
135 await noteCount('execution.end');
136 await noteTimer('execution.end_duration', durationMS); // onTerniate handlers
137
138 for (const onTerminateHandler of this._onTerminateHandlers) {
139 try {
140 await onTerminateHandler(terminationError, data);
141 } catch (err) {
142 error('one of the onTerminate handlers threw an error', {
143 err
144 });
145 }
146 }
147
148 await _System.default.flush(); // NOTE: logging or tracking behind this point are not guaranteed to be transmitted
149
150 if (!this.terminationHandler) {
151 process.exit(terminationError == null ? 0 : 1);
152 } else {
153 // terminationHandler
154 try {
155 await this.terminationHandler(terminationError, data);
156 } catch (err) {
157 error('terminationHandler threw an error', {
158 err
159 });
160 await _System.default.flush();
161 process.exit(1);
162 }
163 }
164 }
165
166}
167
168exports.default = Execution;
169//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Execution.js"],"names":["MODULE_NAME","log","warn","error","noteGauge","noteCount","noteTimer","trackOp","Module","__filename","Execution","constructor","options","_options","dontTerminateOnCompletion","run","runFunc","process","on","_terminate","Error","System","getConfig","trackVitalsIntervalMS","_trackVitals","_startTimeMS","Date","now","returnValue","err","addOnTerminateHandler","onTerminateHandler","_onTerminateHandlers","push","_terminating","used","memoryUsage","key","Math","round","SECOND_MS","MINUTE_MS","terminationError","data","_endTimeMS","durationMS","message","flush","terminationHandler","exit"],"mappings":"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","sourcesContent":["//@flow\n\nimport autoBind from 'auto-bind'\n\nimport System from './System'\nimport Module from './Module'\nimport { sleep } from './asyncUtils'\nimport { SECOND_MS, MINUTE_MS } from './consts'\n\nconst { MODULE_NAME, log, warn, error, noteGauge, noteCount, noteTimer, trackOp } = new Module(__filename) // eslint-disable-line no-unused-vars\n\ntype ExecutionOptions = {|\n  dontTerminateOnCompletion?: ?boolean, // Used for cases where code returns but some callbacks still keep the process alive (like web server)\n|}\n\nexport default class Execution<T> {\n\n  _options: ExecutionOptions\n  _startTimeMS: ?number\n  _endTimeMS: ?number\n  _terminating: boolean = false\n  _onTerminateHandlers: Array<(err: ?Error, data: ?T) => Promise<void>> = []\n\n  /**\n  * An async function that is called AFTER a termination has accured and infra resources were released,\n  * This handler is called after infra resources were flushed, so it should avoid using them.\n  */\n  terminationHandler: ?(err: ?Error, data: ?T) => Promise<void>\n\n\n  constructor(options?: ?ExecutionOptions) {\n    this._options = options || {\n      dontTerminateOnCompletion: false,\n    }\n\n    autoBind(this)\n  }\n\n\n  async run(runFunc: () => Promise<T>): Promise<void> {\n    try {\n      noteCount(`execution.start`)\n\n      // Catches ctrl+c and PM2 shutdown\n      process.on('SIGINT', async () => await this._terminate(new Error('SIGINT')))\n\n      if (System.getConfig().trackVitalsIntervalMS) {\n        this._trackVitals()\n      }\n\n      this._startTimeMS = Date.now()\n      log('Execution started')\n      const returnValue = await runFunc()\n\n      if (!this._options.dontTerminateOnCompletion) {\n        await this._terminate(null, returnValue)\n      }\n    } catch (err) {\n      await this._terminate(err)\n    } \n  }\n\n  /**\n  * Allows adding an async function that is called BEFORE a termination has accured.\n  * @param {*} terminationHandler\n  */\n  addOnTerminateHandler(onTerminateHandler: (err: ?Error, data: ?T) => Promise<void>): void {\n    this._onTerminateHandlers.push(onTerminateHandler)\n  }\n\n\n  async _trackVitals() {\n    while (!this._terminating) {\n      try {\n        const used = process.memoryUsage()\n        for (const key in used) {\n          noteGauge(`memory.${key}MB`, Math.round(used[key] / 1024 / 1024 * 100) / 100)\n        }\n        await sleep(System.getConfig().trackVitalsIntervalMS || 10 * SECOND_MS)\n      } catch (err) {\n        error('Failed to track vitals', { err })\n        await sleep(1 * MINUTE_MS)\n      }\n    }\n  }\n\n  async _terminate(terminationError: ?Error, data: ?T) {\n    if (this._terminating)\n      return\n    this._terminating = true\n    \n    this._endTimeMS = Date.now()\n    // $FlowIgnore\n    const durationMS = this._endTimeMS - this._startTimeMS\n\n    if (terminationError == null) {\n      await noteCount(`execution.succeed`)\n      await noteTimer('execution.succeed_duration', durationMS)\n      log('Execution ended successfully', { durationMS })\n    } else if (terminationError.message === 'SIGINT') {\n      await noteCount(`execution.terminate`)\n      await noteTimer('execution.terminate_duration', durationMS)\n      warn('Execution terminated by SIGINT signal, terminating...', { durationMS })\n    } else {\n      await noteCount(`execution.fail`)\n      await noteTimer('execution.fail_duration', durationMS)\n      error('Execution threw an unhandled exception, terminating...', { durationMS, err: terminationError })\n    }\n    await noteCount('execution.end')\n    await noteTimer('execution.end_duration', durationMS)\n\n    // onTerniate handlers\n    for (const onTerminateHandler of this._onTerminateHandlers) {\n      try {\n        await onTerminateHandler(terminationError, data)\n      } catch (err) {\n        error('one of the onTerminate handlers threw an error', { err })\n      }\n    }\n\n    await System.flush()\n    // NOTE: logging or tracking behind this point are not guaranteed to be transmitted\n\n    if (!this.terminationHandler) {\n      process.exit(terminationError == null ? 0 : 1)\n    } else {\n      // terminationHandler\n      try {\n        await this.terminationHandler(terminationError, data)\n      } catch (err) {\n        error('terminationHandler threw an error', { err })\n        await System.flush()\n        process.exit(1)\n      }\n    }\n  }\n\n}\n"]}
\No newline at end of file