UNPKG

13.7 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.MemoryLeakDetector = exports.FunctionCpuUsage = exports.FunctionStatsMap = exports.FactoryMap = void 0;
4const provider_1 = require("./provider");
5const shared_1 = require("./shared");
6class FactoryMap extends Map {
7 constructor(factory) {
8 super();
9 this.factory = factory;
10 }
11 getOrCreate(key) {
12 let val = this.get(key);
13 if (!val) {
14 val = this.factory(key);
15 this.set(key, val);
16 }
17 return val;
18 }
19}
20exports.FactoryMap = FactoryMap;
21class FunctionStatsMap {
22 constructor() {
23 this.fIncremental = new FactoryMap(() => new provider_1.FunctionStats());
24 this.fAggregate = new FactoryMap(() => new provider_1.FunctionStats());
25 this.aggregate = new provider_1.FunctionStats();
26 }
27 update(fn, key, value) {
28 this.fIncremental.getOrCreate(fn)[key].update(value);
29 this.fAggregate.getOrCreate(fn)[key].update(value);
30 this.aggregate[key].update(value);
31 }
32 incr(fn, key, n = 1) {
33 this.fIncremental.getOrCreate(fn)[key] += n;
34 this.fAggregate.getOrCreate(fn)[key] += n;
35 this.aggregate[key] += n;
36 }
37 resetIncremental() {
38 this.fIncremental.clear();
39 }
40 toString() {
41 return [...this.fAggregate].map(([key, value]) => `[${key}] ${value}`).join("\n");
42 }
43 clear() {
44 this.fIncremental.clear();
45 this.fAggregate.clear();
46 }
47}
48exports.FunctionStatsMap = FunctionStatsMap;
49class FunctionCpuUsage {
50 constructor() {
51 this.utime = new shared_1.Statistics();
52 this.stime = new shared_1.Statistics();
53 this.cpuTime = new shared_1.Statistics();
54 this.smallest = new shared_1.SmallestN(100);
55 }
56}
57exports.FunctionCpuUsage = FunctionCpuUsage;
58class FunctionMemoryStats {
59 constructor() {
60 this.rss = new shared_1.Statistics();
61 this.heapTotal = new shared_1.Statistics();
62 this.heapUsed = new shared_1.Statistics();
63 this.external = new shared_1.Statistics();
64 }
65}
66class FunctionMemoryCounters {
67 constructor() {
68 this.heapUsedGrowth = 0;
69 this.externalGrowth = 0;
70 }
71}
72class MemoryLeakDetector {
73 constructor(memorySize) {
74 this.instances = new FactoryMap(() => new FunctionMemoryStats());
75 this.counters = new FactoryMap(() => new FunctionMemoryCounters());
76 this.warned = new Set();
77 this.memorySize = memorySize || 100;
78 }
79 detectedNewLeak(fn, instanceId, memoryUsage) {
80 if (this.warned.has(fn)) {
81 return false;
82 }
83 const { rss, heapTotal, heapUsed, external } = memoryUsage;
84 const instanceStats = this.instances.getOrCreate(instanceId);
85 const counters = this.counters.getOrCreate(instanceId);
86 if (heapUsed > instanceStats.heapUsed.max) {
87 counters.heapUsedGrowth++;
88 }
89 else {
90 counters.heapUsedGrowth = 0;
91 }
92 if (external > instanceStats.external.max) {
93 counters.externalGrowth++;
94 }
95 else {
96 counters.externalGrowth = 0;
97 }
98 instanceStats.rss.update(rss);
99 instanceStats.heapTotal.update(heapTotal);
100 instanceStats.heapUsed.update(heapUsed);
101 instanceStats.external.update(external);
102 if (heapUsed > this.memorySize * 0.8 * 2 ** 20 ||
103 external > this.memorySize * 0.8 * 2 ** 20) {
104 if (counters.heapUsedGrowth > 4 || counters.externalGrowth > 4) {
105 this.warned.add(fn);
106 return true;
107 }
108 }
109 return false;
110 }
111 clear() {
112 this.instances.clear();
113 this.counters.clear();
114 this.warned.clear();
115 }
116}
117exports.MemoryLeakDetector = MemoryLeakDetector;
118//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"metrics.js","sourceRoot":"","sources":["../../src/metrics.ts"],"names":[],"mappings":"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","sourcesContent":["import { FunctionStats } from \"./provider\";\nimport { SmallestN, Statistics } from \"./shared\";\nimport { PropertiesOfType } from \"./types\";\n\nexport class FactoryMap<K = string, V = {}> extends Map<K, V> {\n    constructor(readonly factory: (key: K) => V) {\n        super();\n    }\n\n    getOrCreate(key: K) {\n        let val = this.get(key);\n        if (!val) {\n            val = this.factory(key);\n            this.set(key, val);\n        }\n        return val;\n    }\n}\n\nexport class FunctionStatsMap {\n    fIncremental = new FactoryMap(() => new FunctionStats());\n    fAggregate = new FactoryMap(() => new FunctionStats());\n    aggregate = new FunctionStats();\n\n    update(\n        fn: string,\n        key: keyof PropertiesOfType<FunctionStats, Statistics>,\n        value: number\n    ) {\n        this.fIncremental.getOrCreate(fn)[key].update(value);\n        this.fAggregate.getOrCreate(fn)[key].update(value);\n        this.aggregate[key].update(value);\n    }\n\n    incr(fn: string, key: keyof PropertiesOfType<FunctionStats, number>, n: number = 1) {\n        this.fIncremental.getOrCreate(fn)[key] += n;\n        this.fAggregate.getOrCreate(fn)[key] += n;\n        this.aggregate[key] += n;\n    }\n\n    resetIncremental() {\n        this.fIncremental.clear();\n    }\n\n    toString() {\n        return [...this.fAggregate].map(([key, value]) => `[${key}] ${value}`).join(\"\\n\");\n    }\n\n    clear() {\n        this.fIncremental.clear();\n        this.fAggregate.clear();\n    }\n}\n\nexport class FunctionCpuUsage {\n    utime = new Statistics();\n    stime = new Statistics();\n    cpuTime = new Statistics();\n    smallest = new SmallestN(100);\n}\n\nclass FunctionMemoryStats {\n    rss = new Statistics();\n    heapTotal = new Statistics();\n    heapUsed = new Statistics();\n    external = new Statistics();\n}\n\nclass FunctionMemoryCounters {\n    heapUsedGrowth = 0;\n    externalGrowth = 0;\n}\n\nexport class MemoryLeakDetector {\n    private instances = new FactoryMap(() => new FunctionMemoryStats());\n    private counters = new FactoryMap(() => new FunctionMemoryCounters());\n    private warned = new Set<string>();\n    private memorySize: number;\n\n    constructor(memorySize?: number) {\n        this.memorySize = memorySize || 100;\n    }\n\n    detectedNewLeak(fn: string, instanceId: string, memoryUsage: NodeJS.MemoryUsage) {\n        if (this.warned.has(fn)) {\n            return false;\n        }\n        const { rss, heapTotal, heapUsed, external } = memoryUsage;\n        const instanceStats = this.instances.getOrCreate(instanceId);\n        const counters = this.counters.getOrCreate(instanceId);\n        if (heapUsed > instanceStats.heapUsed.max) {\n            counters.heapUsedGrowth++;\n        } else {\n            counters.heapUsedGrowth = 0;\n        }\n        if (external > instanceStats.external.max) {\n            counters.externalGrowth++;\n        } else {\n            counters.externalGrowth = 0;\n        }\n        instanceStats.rss.update(rss);\n        instanceStats.heapTotal.update(heapTotal);\n        instanceStats.heapUsed.update(heapUsed);\n        instanceStats.external.update(external);\n\n        if (\n            heapUsed > this.memorySize * 0.8 * 2 ** 20 ||\n            external > this.memorySize * 0.8 * 2 ** 20\n        ) {\n            if (counters.heapUsedGrowth > 4 || counters.externalGrowth > 4) {\n                this.warned.add(fn);\n                return true;\n            }\n        }\n        return false;\n    }\n\n    clear() {\n        this.instances.clear();\n        this.counters.clear();\n        this.warned.clear();\n    }\n}\n"]}
\No newline at end of file