2019-10-23 16:07:23 +00:00
|
|
|
prom = require('prom-client')
|
|
|
|
registry = require('prom-client').register
|
2019-10-28 12:40:12 +00:00
|
|
|
metrics = new Map()
|
2019-10-23 16:07:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
optsKey = (opts) ->
|
|
|
|
keys = Object.keys(opts)
|
|
|
|
return '' if keys.length == 0
|
|
|
|
|
|
|
|
keys = keys.sort()
|
|
|
|
|
|
|
|
hash = '';
|
|
|
|
for key in keys
|
|
|
|
hash += "," if hash.length
|
|
|
|
hash += "#{key}:#{opts[key]}"
|
|
|
|
|
|
|
|
return hash
|
|
|
|
|
|
|
|
extendOpts = (opts, labelNames) ->
|
|
|
|
for label in labelNames
|
|
|
|
opts[label] ||= ''
|
|
|
|
return opts
|
|
|
|
|
|
|
|
optsAsArgs = (opts, labelNames) ->
|
|
|
|
args = []
|
|
|
|
for label in labelNames
|
|
|
|
args.push(opts[label] || '')
|
|
|
|
return args
|
|
|
|
|
|
|
|
|
|
|
|
PromWrapper =
|
|
|
|
ttlInMinutes: 0
|
|
|
|
registry: registry
|
|
|
|
|
|
|
|
metric: (type, name) ->
|
2019-10-28 12:40:12 +00:00
|
|
|
metrics.get(name) || new MetricWrapper(type, name)
|
2019-10-23 16:07:23 +00:00
|
|
|
|
|
|
|
collectDefaultMetrics: prom.collectDefaultMetrics
|
|
|
|
|
|
|
|
|
|
|
|
class MetricWrapper
|
|
|
|
constructor: (type, name) ->
|
2019-10-28 12:40:12 +00:00
|
|
|
metrics.set(name, this)
|
2019-10-23 16:07:23 +00:00
|
|
|
@name = name
|
2019-10-28 12:40:12 +00:00
|
|
|
@instances = new Map()
|
2019-10-23 16:07:23 +00:00
|
|
|
@lastAccess = new Date()
|
|
|
|
@metric = switch type
|
|
|
|
when "counter"
|
|
|
|
new prom.Counter({
|
|
|
|
name: name,
|
|
|
|
help: name,
|
|
|
|
labelNames: ['app','host','status','method', 'path']
|
|
|
|
})
|
|
|
|
when "summary"
|
|
|
|
new prom.Summary({
|
|
|
|
name: name,
|
|
|
|
help: name,
|
2020-03-10 15:00:10 +00:00
|
|
|
maxAgeSeconds: 60,
|
2019-10-23 16:07:23 +00:00
|
|
|
ageBuckets: 10,
|
|
|
|
labelNames: ['app', 'host', 'path', 'status_code', 'method', 'collection', 'query']
|
|
|
|
})
|
|
|
|
when "gauge"
|
2019-12-16 11:32:47 +00:00
|
|
|
new prom.Gauge({
|
2019-10-23 16:07:23 +00:00
|
|
|
name: name,
|
|
|
|
help: name,
|
|
|
|
labelNames: ['app','host', 'status']
|
|
|
|
})
|
|
|
|
|
|
|
|
inc: (opts, value) ->
|
|
|
|
@_execMethod 'inc', opts, value
|
|
|
|
|
|
|
|
observe: (opts, value) ->
|
|
|
|
@_execMethod 'observe', opts, value
|
|
|
|
|
|
|
|
set: (opts, value) ->
|
|
|
|
@_execMethod 'set', opts, value
|
|
|
|
|
|
|
|
sweep: () ->
|
|
|
|
thresh = new Date(Date.now() - 1000 * 60 * PromWrapper.ttlInMinutes)
|
2019-10-28 12:40:12 +00:00
|
|
|
@instances.forEach (instance, key) =>
|
|
|
|
if thresh > instance.time
|
2019-10-23 16:07:23 +00:00
|
|
|
if process.env['DEBUG_METRICS']
|
2019-10-28 12:40:12 +00:00
|
|
|
console.log("Sweeping stale metric instance", @name, opts: instance.opts, key)
|
|
|
|
@metric.remove(optsAsArgs(instance.opts, @metric.labelNames)...)
|
2019-10-23 16:07:23 +00:00
|
|
|
|
|
|
|
if thresh > @lastAccess
|
|
|
|
if process.env['DEBUG_METRICS']
|
2019-10-28 12:40:12 +00:00
|
|
|
console.log("Sweeping stale metric", @name, thresh, @lastAccess)
|
|
|
|
metrics.delete(@name)
|
2019-10-23 16:07:23 +00:00
|
|
|
registry.removeSingleMetric(@name)
|
|
|
|
|
|
|
|
_execMethod: (method, opts, value) ->
|
|
|
|
opts = extendOpts(opts, @metric.labelNames)
|
|
|
|
key = optsKey(opts)
|
2019-10-28 12:40:12 +00:00
|
|
|
@instances.set(key, { time: new Date(), opts }) unless key == ''
|
2019-10-23 16:07:23 +00:00
|
|
|
@lastAccess = new Date()
|
|
|
|
@metric[method](opts, value)
|
|
|
|
|
|
|
|
|
|
|
|
unless PromWrapper.sweepRegistered
|
|
|
|
if process.env['DEBUG_METRICS']
|
|
|
|
console.log("Registering sweep method")
|
|
|
|
PromWrapper.sweepRegistered = true
|
|
|
|
setInterval(
|
|
|
|
() ->
|
|
|
|
if PromWrapper.ttlInMinutes
|
|
|
|
if process.env['DEBUG_METRICS']
|
|
|
|
console.log("Sweeping metrics")
|
2019-10-28 12:40:12 +00:00
|
|
|
metrics.forEach (metric, key) =>
|
|
|
|
metric.sweep()
|
2019-10-23 16:07:23 +00:00
|
|
|
60000)
|
|
|
|
|
|
|
|
|
|
|
|
module.exports = PromWrapper
|