@adalo/metrics 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,288 @@
1
+ 'use strict';
2
+
3
+ const client = require('prom-client');
4
+ const fs = require('fs');
5
+ const os = require('os');
6
+ const https = require('https');
7
+
8
+ /**
9
+ * MetricsClient handles Prometheus metrics collection and push.
10
+ * Supports gauges, counters, default metrics, and custom metrics.
11
+ */
12
+ class MetricsClient {
13
+ /**
14
+ * @param {Object} config
15
+ * @param {string} [config.appName] Name of the application
16
+ * @param {string} [config.dynoId] Dyno/instance ID
17
+ * @param {string} [config.processType] Process type (web, worker, etc.)
18
+ * @param {boolean} [config.enabled] Enable metrics collection
19
+ * @param {boolean} [config.logValues] Log metrics values to console
20
+ * @param {string} [config.pushgatewayUrl] PushGateway URL
21
+ * @param {string} [config.pushgatewayUser] PushGateway username
22
+ * @param {string} [config.pushgatewayPassword] PushGateway password
23
+ * @param {number} [config.intervalSec] Interval in seconds for pushing metrics
24
+ */
25
+ constructor(config = {}) {
26
+ this.appName = config.appName || process.env.METRICS_APP_NAME || 'unknown-app';
27
+ this.dynoId = config.dynoId || process.env.HOSTNAME || 'unknown-dyno';
28
+ this.processType = config.processType || process.env.BUILD_DYNO_PROCESS_TYPE || 'undefined_build_dyno_type';
29
+ this.enabled = config.enabled ?? process.env.METRICS_ENABLED === 'true';
30
+ this.logValues = config.logValues ?? process.env.METRICS_LOG_VALUES === 'true';
31
+ this.pushgatewayUrl = config.pushgatewayUrl || process.env.METRICS_PUSHGATEWAY_URL || '';
32
+ this.pushgatewayUser = config.pushgatewayUser || process.env.METRICS_PUSHGATEWAY_USER || '';
33
+ this.pushgatewayPassword = config.pushgatewayPassword || process.env.METRICS_PUSHGATEWAY_PASSWORD || '';
34
+ this.intervalSec = config.intervalSec || parseInt(process.env.METRICS_INTERVAL_SEC || '', 10) || 15;
35
+ this.prefixLogs = `[${this.processType}] [${this.appName}] [${this.dynoId}] [Monitoring]`;
36
+ this.registry = new client.Registry();
37
+ client.collectDefaultMetrics({
38
+ register: this.registry
39
+ });
40
+ this.defaultLabels = {
41
+ app: this.appName,
42
+ dyno_id: this.dynoId,
43
+ process_type: this.processType
44
+ };
45
+ const authToken = Buffer.from(`${this.pushgatewayUser}:${this.pushgatewayPassword}`).toString('base64');
46
+ this.gateway = new client.Pushgateway(this.pushgatewayUrl, {
47
+ headers: {
48
+ Authorization: `Basic ${authToken}`
49
+ },
50
+ agent: new https.Agent({
51
+ keepAlive: true
52
+ })
53
+ }, this.registry);
54
+ this.gauges = {};
55
+ this.counters = {};
56
+
57
+ /** @type {Object<string, function(): number | Promise<number>>} */
58
+ this.gaugeUpdaters = {};
59
+ /** @type {Object<string, function(data?: object, value?: number): void>} */
60
+ this.counterUpdaters = {};
61
+ this._lastUsageMicros = 0;
62
+ this._lastCheckTime = Date.now();
63
+ this._initDefaultMetrics();
64
+ }
65
+
66
+ /**
67
+ * Register default gauges and counters.
68
+ * @private
69
+ */
70
+ _initDefaultMetrics = () => {
71
+ this.createGauge('app_process_cpu_usage_percent', 'Current CPU usage of the Node.js process in percent', this.getCpuUsagePercent);
72
+ this.createGauge('app_available_cpu_count', 'How many CPU cores are available to this process', this.getAvailableCPUs);
73
+ this.createGauge('app_container_memory_usage_bytes', 'Current container RAM usage from cgroup', this.getContainerMemoryUsage);
74
+ this.createGauge('app_event_loop_lag_ms', 'Estimated event loop lag in milliseconds', this.measureLag);
75
+ this.createGauge('app_container_memory_limit_bytes', 'Max RAM available to container from cgroup (memory.max)', this.getContainerMemoryLimit);
76
+ this.createGauge('app_uptime_seconds', 'How long the process has been running', process.uptime);
77
+ this.createCounter('app_http_requests_total', 'Total number of HTTP requests handled by this process', [...Object.keys(this.defaultLabels), 'method', 'route', 'appId', 'databaseId', 'duration', 'requestSize', 'status_code']);
78
+ };
79
+
80
+ /**
81
+ * Create a gauge metric.
82
+ * @param {string} name
83
+ * @param {string} help
84
+ * @param {function(): number|Promise<number>} [updateFn] Optional function returning value
85
+ * @param {string[]} [labelNames]
86
+ * @returns {client.Gauge}
87
+ */
88
+ createGauge = (name, help, updateFn, labelNames = Object.keys(this.defaultLabels)) => {
89
+ if (this.gauges[name]) return this.gauges[name];
90
+ const g = new client.Gauge({
91
+ name,
92
+ help,
93
+ labelNames,
94
+ registers: [this.registry]
95
+ });
96
+ this.gauges[name] = g;
97
+ if (typeof updateFn === 'function') this.gaugeUpdaters[name] = updateFn;
98
+ return g;
99
+ };
100
+
101
+ /**
102
+ * Create a counter metric.
103
+ * Returns a trigger function to increment the counter manually.
104
+ * @param {string} name
105
+ * @param {string} help
106
+ * @param {string[]} [labelNames]
107
+ * @returns {function(data?: object, value?: number): void} triggerFn
108
+ */
109
+ createCounter(name, help, labelNames = Object.keys(this.defaultLabels)) {
110
+ if (this.counters[name]) return this.counters[name].triggerFn;
111
+ const c = new client.Counter({
112
+ name,
113
+ help,
114
+ labelNames,
115
+ registers: [this.registry]
116
+ });
117
+ this.counters[name] = c;
118
+ const triggerFn = (data = {}, value = 1) => {
119
+ c.inc({
120
+ ...this.defaultLabels,
121
+ ...data
122
+ }, value);
123
+ };
124
+ this.counters[name].triggerFn = triggerFn;
125
+ return triggerFn;
126
+ }
127
+
128
+ /**
129
+ * Get CPU usage percent (cgroup-aware)
130
+ * @returns {number}
131
+ */
132
+ getCpuUsagePercent = () => {
133
+ try {
134
+ const stat = fs.readFileSync('/sys/fs/cgroup/cpu.stat', 'utf-8');
135
+ const match = stat.match(/usage_usec (\d+)/);
136
+ if (!match) return 0;
137
+ const now = Date.now();
138
+ const currentUsage = parseInt(match[1], 10);
139
+ if (this._lastUsageMicros === 0) {
140
+ this._lastUsageMicros = currentUsage;
141
+ this._lastCheckTime = now;
142
+ return 0;
143
+ }
144
+ const deltaUsage = currentUsage - this._lastUsageMicros;
145
+ const deltaTime = now - this._lastCheckTime;
146
+ this._lastUsageMicros = currentUsage;
147
+ this._lastCheckTime = now;
148
+ return deltaUsage / (deltaTime * 1000) * 100;
149
+ } catch {
150
+ return 0;
151
+ }
152
+ };
153
+
154
+ /**
155
+ * Get available CPU cores.
156
+ * @returns {number}
157
+ */
158
+ getAvailableCPUs() {
159
+ try {
160
+ const cpuMaxPath = '/sys/fs/cgroup/cpu.max';
161
+ if (fs.existsSync(cpuMaxPath)) {
162
+ const [quotaStr, periodStr] = fs.readFileSync(cpuMaxPath, 'utf8').trim().split(' ');
163
+ if (quotaStr === 'max') return os.cpus().length;
164
+ return parseInt(quotaStr, 10) / parseInt(periodStr, 10);
165
+ }
166
+ return os.cpus().length;
167
+ } catch {
168
+ return 1;
169
+ }
170
+ }
171
+
172
+ /**
173
+ * Get container memory usage in bytes.
174
+ * @returns {number}
175
+ */
176
+ getContainerMemoryUsage() {
177
+ try {
178
+ return parseInt(fs.readFileSync('/sys/fs/cgroup/memory.current', 'utf-8').trim(), 10);
179
+ } catch {
180
+ return process.memoryUsage().rss;
181
+ }
182
+ }
183
+
184
+ /**
185
+ * Get container memory limit in bytes.
186
+ * @returns {number}
187
+ */
188
+ getContainerMemoryLimit() {
189
+ try {
190
+ const path = '/sys/fs/cgroup/memory.max';
191
+ if (fs.existsSync(path)) {
192
+ const val = fs.readFileSync(path, 'utf-8').trim();
193
+ if (val !== 'max') {
194
+ const parsed = parseInt(val, 10);
195
+ if (parsed && parsed < os.totalmem()) return parsed;
196
+ }
197
+ }
198
+ return os.totalmem();
199
+ } catch {
200
+ return os.totalmem();
201
+ }
202
+ }
203
+
204
+ /**
205
+ * Measure event loop lag in ms.
206
+ * @returns {Promise<number>}
207
+ */
208
+ measureLag() {
209
+ return new Promise(resolve => {
210
+ const start = Date.now();
211
+ setImmediate(() => resolve(Date.now() - start));
212
+ });
213
+ }
214
+
215
+ /**
216
+ * Express middleware to count HTTP requests.
217
+ * Increments the `app_http_requests_total` counter.
218
+ */
219
+ countHttpRequestMiddleware = (req, res, next) => {
220
+ const start = Date.now();
221
+ res.on('finish', () => {
222
+ const route = req.route?.path || req.path || 'unknown';
223
+ const appId = req.params?.appId || req.body?.appId || req.query?.appId || '';
224
+ const databaseId = req.params?.databaseId || req.body?.databaseId || req.query?.databaseId || '';
225
+ this.counters?.app_http_requests_total?.triggerFn({
226
+ method: req.method,
227
+ route,
228
+ status_code: res.statusCode,
229
+ appId,
230
+ databaseId,
231
+ duration: Date.now() - start,
232
+ requestSize: req.headers['content-length'] ? parseInt(req.headers['content-length'], 10) : 0
233
+ });
234
+ });
235
+ next();
236
+ };
237
+
238
+ /**
239
+ * Push all gauges and counters to PushGateway and optionally log.
240
+ */
241
+ pushMetrics = async () => {
242
+ try {
243
+ for (const [name, updateFn] of Object.entries(this.gaugeUpdaters)) {
244
+ try {
245
+ const result = updateFn();
246
+ const val = result instanceof Promise ? await result : result;
247
+ if (val !== undefined) this.gauges[name].set(this.defaultLabels, val);
248
+ } catch (err) {
249
+ console.error(`${this.prefixLogs} Failed to update gauge ${name}:`, err);
250
+ }
251
+ }
252
+ await this.gateway.push({
253
+ jobName: this.appName,
254
+ groupings: {
255
+ process_type: this.processType,
256
+ instance: this.dynoId
257
+ }
258
+ });
259
+ Object.values(this.counters).forEach(counter => counter.reset());
260
+ if (this.logValues) {
261
+ const metrics = await this.registry.getMetricsAsJSON();
262
+ console.log(`${this.prefixLogs} Metrics:\n`, JSON.stringify(metrics, null, 2));
263
+ }
264
+ } catch (err) {
265
+ console.error(`${this.prefixLogs} Failed to push metrics:`, err);
266
+ }
267
+ };
268
+
269
+ /**
270
+ * Start automatic periodic push of metrics.
271
+ * @param {number} [interval] Interval in seconds
272
+ */
273
+ startPush = (interval = this.intervalSec) => {
274
+ if (!this.enabled) {
275
+ return console.warn(`${this.prefixLogs} Metrics disabled`);
276
+ }
277
+ setInterval(() => {
278
+ this.pushMetrics().catch(err => {
279
+ console.error(`${this.prefixLogs} Failed to push metrics:`, err);
280
+ });
281
+ }, interval * 1000);
282
+ console.warn(`${this.prefixLogs} Metrics collection started.`);
283
+ };
284
+ }
285
+ module.exports = {
286
+ MetricsClient
287
+ };
288
+ //# sourceMappingURL=metricsClient.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metricsClient.js","names":["client","require","fs","os","https","MetricsClient","constructor","config","appName","process","env","METRICS_APP_NAME","dynoId","HOSTNAME","processType","BUILD_DYNO_PROCESS_TYPE","enabled","METRICS_ENABLED","logValues","METRICS_LOG_VALUES","pushgatewayUrl","METRICS_PUSHGATEWAY_URL","pushgatewayUser","METRICS_PUSHGATEWAY_USER","pushgatewayPassword","METRICS_PUSHGATEWAY_PASSWORD","intervalSec","parseInt","METRICS_INTERVAL_SEC","prefixLogs","registry","Registry","collectDefaultMetrics","register","defaultLabels","app","dyno_id","process_type","authToken","Buffer","from","toString","gateway","Pushgateway","headers","Authorization","agent","Agent","keepAlive","gauges","counters","gaugeUpdaters","counterUpdaters","_lastUsageMicros","_lastCheckTime","Date","now","_initDefaultMetrics","createGauge","getCpuUsagePercent","getAvailableCPUs","getContainerMemoryUsage","measureLag","getContainerMemoryLimit","uptime","createCounter","Object","keys","name","help","updateFn","labelNames","g","Gauge","registers","triggerFn","c","Counter","data","value","inc","stat","readFileSync","match","currentUsage","deltaUsage","deltaTime","cpuMaxPath","existsSync","quotaStr","periodStr","trim","split","cpus","length","memoryUsage","rss","path","val","parsed","totalmem","Promise","resolve","start","setImmediate","countHttpRequestMiddleware","req","res","next","on","route","appId","params","body","query","databaseId","app_http_requests_total","method","status_code","statusCode","duration","requestSize","pushMetrics","entries","result","undefined","set","err","console","error","push","jobName","groupings","instance","values","forEach","counter","reset","metrics","getMetricsAsJSON","log","JSON","stringify","startPush","interval","warn","setInterval","catch","module","exports"],"sources":["../src/metricsClient.js"],"sourcesContent":["'use strict'\n\nconst client = require('prom-client')\nconst fs = require('fs')\nconst os = require('os')\nconst https = require('https')\n\n/**\n * MetricsClient handles Prometheus metrics collection and push.\n * Supports gauges, counters, default metrics, and custom metrics.\n */\nclass MetricsClient {\n /**\n * @param {Object} config\n * @param {string} [config.appName] Name of the application\n * @param {string} [config.dynoId] Dyno/instance ID\n * @param {string} [config.processType] Process type (web, worker, etc.)\n * @param {boolean} [config.enabled] Enable metrics collection\n * @param {boolean} [config.logValues] Log metrics values to console\n * @param {string} [config.pushgatewayUrl] PushGateway URL\n * @param {string} [config.pushgatewayUser] PushGateway username\n * @param {string} [config.pushgatewayPassword] PushGateway password\n * @param {number} [config.intervalSec] Interval in seconds for pushing metrics\n */\n constructor(config = {}) {\n this.appName =\n config.appName || process.env.METRICS_APP_NAME || 'unknown-app'\n this.dynoId = config.dynoId || process.env.HOSTNAME || 'unknown-dyno'\n this.processType =\n config.processType ||\n process.env.BUILD_DYNO_PROCESS_TYPE ||\n 'undefined_build_dyno_type'\n this.enabled = config.enabled ?? process.env.METRICS_ENABLED === 'true'\n this.logValues =\n config.logValues ?? process.env.METRICS_LOG_VALUES === 'true'\n this.pushgatewayUrl =\n config.pushgatewayUrl || process.env.METRICS_PUSHGATEWAY_URL || ''\n this.pushgatewayUser =\n config.pushgatewayUser || process.env.METRICS_PUSHGATEWAY_USER || ''\n this.pushgatewayPassword =\n config.pushgatewayPassword ||\n process.env.METRICS_PUSHGATEWAY_PASSWORD ||\n ''\n this.intervalSec =\n config.intervalSec ||\n parseInt(process.env.METRICS_INTERVAL_SEC || '', 10) ||\n 15\n\n this.prefixLogs = `[${this.processType}] [${this.appName}] [${this.dynoId}] [Monitoring]`\n\n this.registry = new client.Registry()\n client.collectDefaultMetrics({ register: this.registry })\n\n this.defaultLabels = {\n app: this.appName,\n dyno_id: this.dynoId,\n process_type: this.processType,\n }\n\n const authToken = Buffer.from(\n `${this.pushgatewayUser}:${this.pushgatewayPassword}`\n ).toString('base64')\n this.gateway = new client.Pushgateway(\n this.pushgatewayUrl,\n {\n headers: { Authorization: `Basic ${authToken}` },\n agent: new https.Agent({ keepAlive: true }),\n },\n this.registry\n )\n\n this.gauges = {}\n this.counters = {}\n\n /** @type {Object<string, function(): number | Promise<number>>} */\n this.gaugeUpdaters = {}\n /** @type {Object<string, function(data?: object, value?: number): void>} */\n this.counterUpdaters = {}\n\n this._lastUsageMicros = 0\n this._lastCheckTime = Date.now()\n\n this._initDefaultMetrics()\n }\n\n /**\n * Register default gauges and counters.\n * @private\n */\n _initDefaultMetrics = () => {\n this.createGauge(\n 'app_process_cpu_usage_percent',\n 'Current CPU usage of the Node.js process in percent',\n this.getCpuUsagePercent\n )\n this.createGauge(\n 'app_available_cpu_count',\n 'How many CPU cores are available to this process',\n this.getAvailableCPUs\n )\n this.createGauge(\n 'app_container_memory_usage_bytes',\n 'Current container RAM usage from cgroup',\n this.getContainerMemoryUsage\n )\n this.createGauge(\n 'app_event_loop_lag_ms',\n 'Estimated event loop lag in milliseconds',\n this.measureLag\n )\n this.createGauge(\n 'app_container_memory_limit_bytes',\n 'Max RAM available to container from cgroup (memory.max)',\n this.getContainerMemoryLimit\n )\n this.createGauge(\n 'app_uptime_seconds',\n 'How long the process has been running',\n process.uptime\n )\n\n this.createCounter(\n 'app_http_requests_total',\n 'Total number of HTTP requests handled by this process',\n [\n ...Object.keys(this.defaultLabels),\n 'method',\n 'route',\n 'appId',\n 'databaseId',\n 'duration',\n 'requestSize',\n 'status_code',\n ]\n )\n }\n\n /**\n * Create a gauge metric.\n * @param {string} name\n * @param {string} help\n * @param {function(): number|Promise<number>} [updateFn] Optional function returning value\n * @param {string[]} [labelNames]\n * @returns {client.Gauge}\n */\n createGauge = (\n name,\n help,\n updateFn,\n labelNames = Object.keys(this.defaultLabels)\n ) => {\n if (this.gauges[name]) return this.gauges[name]\n\n const g = new client.Gauge({\n name,\n help,\n labelNames,\n registers: [this.registry],\n })\n this.gauges[name] = g\n\n if (typeof updateFn === 'function') this.gaugeUpdaters[name] = updateFn\n\n return g\n }\n\n /**\n * Create a counter metric.\n * Returns a trigger function to increment the counter manually.\n * @param {string} name\n * @param {string} help\n * @param {string[]} [labelNames]\n * @returns {function(data?: object, value?: number): void} triggerFn\n */\n createCounter(name, help, labelNames = Object.keys(this.defaultLabels)) {\n if (this.counters[name]) return this.counters[name].triggerFn\n\n const c = new client.Counter({\n name,\n help,\n labelNames,\n registers: [this.registry],\n })\n this.counters[name] = c\n\n const triggerFn = (data = {}, value = 1) => {\n c.inc({ ...this.defaultLabels, ...data }, value)\n }\n\n this.counters[name].triggerFn = triggerFn\n return triggerFn\n }\n\n /**\n * Get CPU usage percent (cgroup-aware)\n * @returns {number}\n */\n getCpuUsagePercent = () => {\n try {\n const stat = fs.readFileSync('/sys/fs/cgroup/cpu.stat', 'utf-8')\n const match = stat.match(/usage_usec (\\d+)/)\n if (!match) return 0\n\n const now = Date.now()\n const currentUsage = parseInt(match[1], 10)\n\n if (this._lastUsageMicros === 0) {\n this._lastUsageMicros = currentUsage\n this._lastCheckTime = now\n return 0\n }\n\n const deltaUsage = currentUsage - this._lastUsageMicros\n const deltaTime = now - this._lastCheckTime\n\n this._lastUsageMicros = currentUsage\n this._lastCheckTime = now\n\n return (deltaUsage / (deltaTime * 1000)) * 100\n } catch {\n return 0\n }\n }\n\n /**\n * Get available CPU cores.\n * @returns {number}\n */\n getAvailableCPUs() {\n try {\n const cpuMaxPath = '/sys/fs/cgroup/cpu.max'\n if (fs.existsSync(cpuMaxPath)) {\n const [quotaStr, periodStr] = fs\n .readFileSync(cpuMaxPath, 'utf8')\n .trim()\n .split(' ')\n if (quotaStr === 'max') return os.cpus().length\n return parseInt(quotaStr, 10) / parseInt(periodStr, 10)\n }\n return os.cpus().length\n } catch {\n return 1\n }\n }\n\n /**\n * Get container memory usage in bytes.\n * @returns {number}\n */\n getContainerMemoryUsage() {\n try {\n return parseInt(\n fs.readFileSync('/sys/fs/cgroup/memory.current', 'utf-8').trim(),\n 10\n )\n } catch {\n return process.memoryUsage().rss\n }\n }\n\n /**\n * Get container memory limit in bytes.\n * @returns {number}\n */\n getContainerMemoryLimit() {\n try {\n const path = '/sys/fs/cgroup/memory.max'\n if (fs.existsSync(path)) {\n const val = fs.readFileSync(path, 'utf-8').trim()\n if (val !== 'max') {\n const parsed = parseInt(val, 10)\n if (parsed && parsed < os.totalmem()) return parsed\n }\n }\n return os.totalmem()\n } catch {\n return os.totalmem()\n }\n }\n\n /**\n * Measure event loop lag in ms.\n * @returns {Promise<number>}\n */\n measureLag() {\n return new Promise(resolve => {\n const start = Date.now()\n setImmediate(() => resolve(Date.now() - start))\n })\n }\n\n /**\n * Express middleware to count HTTP requests.\n * Increments the `app_http_requests_total` counter.\n */\n countHttpRequestMiddleware = (req, res, next) => {\n const start = Date.now()\n res.on('finish', () => {\n const route = req.route?.path || req.path || 'unknown'\n const appId =\n req.params?.appId || req.body?.appId || req.query?.appId || ''\n const databaseId =\n req.params?.databaseId ||\n req.body?.databaseId ||\n req.query?.databaseId ||\n ''\n\n this.counters?.app_http_requests_total?.triggerFn({\n method: req.method,\n route,\n status_code: res.statusCode,\n appId,\n databaseId,\n duration: Date.now() - start,\n requestSize: req.headers['content-length']\n ? parseInt(req.headers['content-length'], 10)\n : 0,\n })\n })\n\n next()\n }\n\n /**\n * Push all gauges and counters to PushGateway and optionally log.\n */\n pushMetrics = async () => {\n try {\n for (const [name, updateFn] of Object.entries(this.gaugeUpdaters)) {\n try {\n const result = updateFn()\n const val = result instanceof Promise ? await result : result\n if (val !== undefined) this.gauges[name].set(this.defaultLabels, val)\n } catch (err) {\n console.error(\n `${this.prefixLogs} Failed to update gauge ${name}:`,\n err\n )\n }\n }\n\n await this.gateway.push({\n jobName: this.appName,\n groupings: { process_type: this.processType, instance: this.dynoId },\n })\n\n Object.values(this.counters).forEach(counter => counter.reset())\n\n if (this.logValues) {\n const metrics = await this.registry.getMetricsAsJSON()\n console.log(\n `${this.prefixLogs} Metrics:\\n`,\n JSON.stringify(metrics, null, 2)\n )\n }\n } catch (err) {\n console.error(`${this.prefixLogs} Failed to push metrics:`, err)\n }\n }\n\n /**\n * Start automatic periodic push of metrics.\n * @param {number} [interval] Interval in seconds\n */\n startPush = (interval = this.intervalSec) => {\n if (!this.enabled) {\n return console.warn(`${this.prefixLogs} Metrics disabled`)\n }\n\n setInterval(() => {\n this.pushMetrics().catch(err => {\n console.error(`${this.prefixLogs} Failed to push metrics:`, err)\n })\n }, interval * 1000)\n\n console.warn(`${this.prefixLogs} Metrics collection started.`)\n }\n}\n\nmodule.exports = { MetricsClient }\n"],"mappings":"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","ignoreList":[]}
package/package.json ADDED
@@ -0,0 +1,60 @@
1
+ {
2
+ "name": "@adalo/metrics",
3
+ "version": "0.1.0",
4
+ "description": "Reusable metrics utilities for Node.js and Laravel apps",
5
+ "main": "lib/index.js",
6
+ "types": "lib/index.d.ts",
7
+ "scripts": {
8
+ "build": "babel src --out-dir lib --source-maps --extensions .js,.ts && yarn build:types",
9
+ "build:types": "tsc --project ./tsconfig.types.json",
10
+ "watch": "yarn build -- --watch",
11
+ "start": "yarn watch",
12
+ "test": "jest",
13
+ "test:watch": "yarn test --watch",
14
+ "test:all": "yarn format:check && yarn test",
15
+ "types:check": "tsc --project ./tsconfig.json",
16
+ "prettier": "prettier --write 'src/**'",
17
+ "prettier:check": "prettier --check 'src/**'",
18
+ "lint": "eslint --fix src --ext .js,.ts",
19
+ "lint:check": "eslint src --ext .js,.ts",
20
+ "format": "yarn prettier && yarn lint",
21
+ "format:check": "yarn prettier:check && yarn lint:check"
22
+ },
23
+ "author": "@AdaloHQ",
24
+ "license": "ISC",
25
+ "dependencies": {
26
+ "dotenv": "^8.2.0",
27
+ "prom-client": "^15.1.3"
28
+ },
29
+ "devDependencies": {
30
+ "@babel/cli": "7.18.6",
31
+ "@babel/core": "7.18.6",
32
+ "@babel/preset-env": "7.18.6",
33
+ "@babel/preset-typescript": "7.18.6",
34
+ "@types/jest": "28.1.6",
35
+ "@types/node": "18.0.4",
36
+ "@typescript-eslint/eslint-plugin": "5.30.6",
37
+ "@typescript-eslint/parser": "5.30.6",
38
+ "babel-jest": "28.1.3",
39
+ "eslint": "~8.4.1",
40
+ "eslint-config-airbnb-base": "~15.0.0",
41
+ "eslint-config-airbnb-typescript": "17.0.0",
42
+ "eslint-config-prettier": "~8.3.0",
43
+ "eslint-import-resolver-typescript": "2.7.1",
44
+ "eslint-plugin-import": "~2.25.3",
45
+ "jest": "28.1.3",
46
+ "prettier": "2.5.1",
47
+ "typescript": "4.7.4"
48
+ },
49
+ "resolutions": {
50
+ "@types/istanbul-lib-report": "^3.0.0"
51
+ },
52
+ "jest": {
53
+ "testMatch": [
54
+ "**/__tests__/**/*.[jt]s?(x)"
55
+ ]
56
+ },
57
+ "engines": {
58
+ "node": ">=18.0.0"
59
+ }
60
+ }
package/src/index.ts ADDED
@@ -0,0 +1 @@
1
+ export * from './metricsClient'