1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.default = void 0;
|
7 |
|
8 | var _autoBind = _interopRequireDefault(require("auto-bind"));
|
9 |
|
10 | var _System = _interopRequireDefault(require("./System"));
|
11 |
|
12 | var _Module = _interopRequireDefault(require("./Module"));
|
13 |
|
14 | var _asyncUtils = require("./asyncUtils");
|
15 |
|
16 | var _consts = require("./consts");
|
17 |
|
18 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
19 |
|
20 | function _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 |
|
22 | const {
|
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 |
|
33 | class 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 |
|
168 | exports.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 |