-
Notifications
You must be signed in to change notification settings - Fork 231
/
Copy pathprometheus.lua
991 lines (899 loc) · 30.9 KB
/
prometheus.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
--- @module Prometheus
--
-- vim: ts=2:sw=2:sts=2:expandtab:textwidth=80
-- This module uses a single dictionary shared between Nginx workers to keep
-- all metrics. Each metric is stored as a separate entry in that dictionary.
--
-- In addition, each worker process has a separate set of counters within
-- its lua runtime that are used to track increments to counte metrics, and
-- are regularly flushed into the main shared dictionary. This is a performance
-- optimization that allows counters to be incremented without locking the
-- shared dictionary. It also means that counter increments are "eventually
-- consistent"; it can take up to a single counter sync interval (which
-- defaults to 1 second) for counter values to be visible for collection.
--
-- Prometheus requires that (a) all samples for a given metric are presented
-- as one uninterrupted group, and (b) buckets of a histogram appear in
-- increasing numerical order. We satisfy that by carefully constructing full
-- metric names (i.e. metric name along with all labels) so that they meet
-- those requirements while being sorted alphabetically. In particular:
--
-- * all labels for a given metric are presented in reproducible order (the one
-- used when labels were declared). "le" label for histogram metrics always
-- goes last;
-- * bucket boundaries (which are exposed as values of the "le" label) are
-- stored as floating point numbers with leading and trailing zeroes,
-- and those zeros would be removed just before we expose the metrics;
-- * internally "+Inf" bucket is stored as "Inf" (to make it appear after
-- all numeric buckets), and gets replaced by "+Inf" just before we
-- expose the metrics.
--
-- For example, if you define your bucket boundaries as {0.00005, 10, 1000}
-- then we will keep the following samples for a metric `m1` with label
-- `site` set to `site1`:
--
-- m1_bucket{site="site1",le="0000.00005"}
-- m1_bucket{site="site1",le="0010.00000"}
-- m1_bucket{site="site1",le="1000.00000"}
-- m1_bucket{site="site1",le="Inf"}
-- m1_count{site="site1"}
-- m1_sum{site="site1"}
--
-- And when exposing the metrics, their names will be changed to:
--
-- m1_bucket{site="site1",le="0.00005"}
-- m1_bucket{site="site1",le="10"}
-- m1_bucket{site="site1",le="1000"}
-- m1_bucket{site="site1",le="+Inf"}
-- m1_count{site="site1"}
-- m1_sum{site="site1"}
--
-- You can find the latest version and documentation at
-- https://github.com/knyar/nginx-lua-prometheus
-- Released under MIT license.
-- This library provides per-worker counters used to store counter metric
-- increments. Copied from https://github.com/Kong/lua-resty-counter
local resty_counter_lib = require("prometheus_resty_counter")
local key_index_lib = require("prometheus_keys")
local ngx = ngx
local ngx_re_match = ngx.re.match
local ngx_re_gsub = ngx.re.gsub
local error = error
local type = type
local get_phase = ngx.get_phase
local ngx_sleep = ngx.sleep
local select = select
local YIELD_ITERATIONS = 200
local Prometheus = {}
local mt = { __index = Prometheus }
local TYPE_COUNTER = 0x1
local TYPE_GAUGE = 0x2
local TYPE_HISTOGRAM = 0x4
local TYPE_LITERAL = {
[TYPE_COUNTER] = "counter",
[TYPE_GAUGE] = "gauge",
[TYPE_HISTOGRAM] = "histogram",
}
local can_yield_phases = {
rewrite = true,
access = true,
content = true,
timer = true
}
-- Default name for error metric incremented by this library.
local DEFAULT_ERROR_METRIC_NAME = "nginx_metric_errors_total"
-- Default value for per-worker counter sync interval (seconds).
local DEFAULT_SYNC_INTERVAL = 1
-- Default set of latency buckets, 5ms to 10s:
local DEFAULT_BUCKETS = {0.005, 0.01, 0.02, 0.03, 0.05, 0.075, 0.1, 0.2, 0.3,
0.4, 0.5, 0.75, 1, 1.5, 2, 3, 4, 5, 10}
-- Prefix for internal shared dictionary items.
local KEY_INDEX_PREFIX = "__ngx_prom__"
local METRICS_KEY_REGEX = [[(.*[,{]le=")(.*)(".*)]]
local ERR_MSG_COUNTER_NOT_INITIALIZED = "counter not initialized! " ..
"Have you called Prometheus:init() from the " ..
"init_worker_by_lua_block nginx phase?"
-- Error message that gets logged when the shared dictionary gets full.
local ERR_MSG_LRU_EVICTION = "Shared dictionary used for prometheus metrics " ..
"is full. REPORTED METRIC DATA MIGHT BE INCOMPLETE. Please increase the " ..
"size of the dictionary or reduce metric cardinality."
-- Accepted range of byte values for tailing bytes of utf8 strings.
-- This is defined outside of the validate_utf8_string function as a const
-- variable to avoid creating and destroying table frequently.
-- Values in this table (and in validate_utf8_string) are from table 3-7 of
-- www.unicode.org/versions/Unicode6.2.0/UnicodeStandard-6.2.pdf
local accept_range = {
{lo = 0x80, hi = 0xBF},
{lo = 0xA0, hi = 0xBF},
{lo = 0x80, hi = 0x9F},
{lo = 0x90, hi = 0xBF},
{lo = 0x80, hi = 0x8F}
}
-- Validate utf8 string for label values.
--
-- Args:
-- str: string
--
-- Returns:
-- (bool) whether the input string is a valid utf8 string.
-- (number) position of the first invalid byte.
local function validate_utf8_string(str)
local i, n = 1, #str
local first, byte, left_size, range_idx
while i <= n do
first = string.byte(str, i)
if first >= 0x80 then
range_idx = 1
if first >= 0xC2 and first <= 0xDF then -- 2 bytes
left_size = 1
elseif first >= 0xE0 and first <= 0xEF then -- 3 bytes
left_size = 2
if first == 0xE0 then
range_idx = 2
elseif first == 0xED then
range_idx = 3
end
elseif first >= 0xF0 and first <= 0xF4 then -- 4 bytes
left_size = 3
if first == 0xF0 then
range_idx = 4
elseif first == 0xF4 then
range_idx = 5
end
else
return false, i
end
if i + left_size > n then
return false, i
end
for j = 1, left_size do
byte = string.byte(str, i + j)
if byte < accept_range[range_idx].lo or byte > accept_range[range_idx].hi then
return false, i
end
range_idx = 1
end
i = i + left_size
end
i = i + 1
end
return true
end
-- copy form https://github.com/apache/apisix/blob/release/2.13/apisix/core/table.lua#L50-L58
local function table_insert_tail(tab, ...)
local idx = #tab
for i = 1, select('#', ...) do
idx = idx + 1
tab[idx] = select(i, ...)
end
return idx
end
-- Return table as a string. Nested tables are not expanded.
local function table_to_string(t)
if t == nil then
return "nil"
end
local buf = "<"
for i = 1, #t do
buf = buf .. tostring(t[i])
if i < #t then
buf = buf .. ","
end
end
return buf .. ">"
end
-- Export to allow testing.
function Prometheus._table_to_string(t) return table_to_string(t) end
-- Generate full metric name that includes all labels.
--
-- Args:
-- name: string
-- label_names: (array) a list of label keys.
-- label_values: (array) a list of label values.
--
-- Returns:
-- (string) full metric name.
local function full_metric_name(name, label_names, label_values)
if not label_names then
return name
end
local label_parts = {}
for idx, key in ipairs(label_names) do
local label_value
if type(label_values[idx]) == "string" then
local valid, pos = validate_utf8_string(label_values[idx])
if not valid then
label_value = string.sub(label_values[idx], 1, pos - 1)
:gsub("\\", "\\\\")
:gsub('"', '\\"')
:gsub("\n", "\\n")
else
label_value = label_values[idx]
:gsub("\\", "\\\\")
:gsub('"', '\\"')
:gsub("\n", "\\n")
end
else
label_value = tostring(label_values[idx])
end
table_insert_tail(label_parts, key .. '="' .. label_value .. '"')
end
return name .. "{" .. table.concat(label_parts, ",") .. "}"
end
-- Extract short metric name from the full one.
--
-- This function is only used by Prometheus:metric_data.
--
-- Args:
-- full_name: (string) full metric name that can include labels.
--
-- Returns:
-- (string) short metric name with no labels. For a `*_bucket` metric of
-- histogram the _bucket suffix will be removed.
local function short_metric_name(full_name)
local labels_start, _ = full_name:find("{")
if not labels_start then
return full_name
end
-- Try to detect if this is a histogram metric. We only check for the
-- `_bucket` suffix here, since it alphabetically goes before other
-- histogram suffixes (`_count` and `_sum`).
local suffix_idx, _ = full_name:find("_bucket{")
if suffix_idx and full_name:find("le=") then
-- this is a histogram metric
return full_name:sub(1, suffix_idx - 1)
end
-- this is not a histogram metric
return full_name:sub(1, labels_start - 1)
end
-- Check metric name and label names for correctness.
--
-- Regular expressions to validate metric and label names are
-- documented in https://prometheus.io/docs/concepts/data_model/
--
-- Args:
-- metric_name: (string) metric name.
-- label_names: label names (array of strings).
--
-- Returns:
-- Either an error string, or nil of no errors were found.
local function check_metric_and_label_names(metric_name, label_names)
if not metric_name:match("^[a-zA-Z_:][a-zA-Z0-9_:]*$") then
return "Metric name '" .. metric_name .. "' is invalid"
end
if metric_name:find(KEY_INDEX_PREFIX) == 1 then
return "Prefix '" .. KEY_INDEX_PREFIX .. "' is reserved for internal purposes"
end
for _, label_name in ipairs(label_names or {}) do
if label_name == "le" then
return "Invalid label name 'le' in " .. metric_name
end
if not label_name:match("^[a-zA-Z_][a-zA-Z0-9_]*$") then
return "Metric '" .. metric_name .. "' label name '" .. label_name ..
"' is invalid"
end
end
end
-- Construct bucket format for a list of buckets.
--
-- This receives a list of buckets and returns a sprintf template that should
-- be used for bucket boundaries to make them come in increasing order when
-- sorted alphabetically.
--
-- To re-phrase, this is where we detect how many leading and trailing zeros we
-- need.
--
-- Args:
-- buckets: a list of buckets
--
-- Returns:
-- (string) a sprintf template.
local function construct_bucket_format(buckets)
local max_order = 1
local max_precision = 1
for _, bucket in ipairs(buckets) do
assert(type(bucket) == "number", "bucket boundaries should be numeric")
-- floating point number with all trailing zeros removed
local bucket_str = string.format("%f", bucket)
local as_string = ngx_re_gsub(bucket_str, "0*$", "", "jo")
local dot_idx = as_string:find(".", 1, true)
max_order = math.max(max_order, dot_idx - 1)
max_precision = math.max(max_precision, as_string:len() - dot_idx)
end
return "%0" .. (max_order + max_precision + 1) .. "." .. max_precision .. "f"
end
-- Format bucket format when exposing metrics.
--
-- This function removes leading and trailing zeroes from `le` label values.
--
-- Args:
-- key: the metric key
--
-- Returns:
-- (string) the formatted key
local function fix_histogram_bucket_labels(key)
local match, err = ngx_re_match(key, METRICS_KEY_REGEX, "jo")
if err then
ngx.log(ngx.ERR, "failed to match regex: ", err)
return
end
if not match then
return key
end
if match[2] == "Inf" then
return table.concat({match[1], "+Inf", match[3]})
else
return table.concat({match[1], tostring(tonumber(match[2])), match[3]})
end
end
-- Return a full metric name for a given metric+label combination.
--
-- This function calculates a full metric name (or, in case of a histogram
-- metric, several metric names) for a given combination of label values. It
-- stores the result in a tree of tables used as a cache (self.lookup) and
-- uses that cache to return results faster.
--
-- Args:
-- self: a `metric` object, created by register().
-- label_values: a list of label values.
--
-- Returns:
-- - If `self` is a counter or a gauge: full metric name as a string.
-- - If `self` is a histogram metric: a list of strings:
-- [0]: full name of the _count histogram metric;
-- [1]: full name of the _sum histogram metric;
-- [...]: full names of each _bucket metrics.
local function lookup_or_create(self, label_values)
local cnt = label_values and #label_values or 0
if cnt ~= self.label_count then
return nil, string.format(
"incorrect label count for metric %s, expected %d, got %d (%s)",
self.name, self.label_count, cnt, table_to_string(label_values))
end
local t = self.lookup
if label_values then
-- Don't use ipairs here to avoid inner loop generates trace first
-- Otherwise the inner for loop below is likely to get JIT compiled before
-- the outer loop which include `lookup_or_create`, in this case the trace
-- for outer loop will be aborted. By not using ipairs, we will be able to
-- compile longer traces as possible.
local label
for i=1, self.label_count do
label = label_values[i]
if label == nil then
return nil, string.format("unexpected nil value for label %s of %s",
self.label_names[i], self.name)
end
if not t[label] then
t[label] = {}
end
t = t[label]
end
end
local LEAF_KEY = mt -- key used to store full metric names in leaf tables.
local full_name = t[LEAF_KEY]
if full_name then
return full_name
end
if self.typ == TYPE_HISTOGRAM then
-- Pass empty metric name to full_metric_name to just get the formatted
-- labels ({key1="value1",key2="value2",...}).
local labels = full_metric_name("", self.label_names, label_values)
full_name = {
self.name .. "_count" .. labels,
self.name .. "_sum" .. labels,
}
local bucket_pref
if self.label_count > 0 then
-- strip last }
bucket_pref = self.name .. "_bucket" .. string.sub(labels, 1, #labels-1) .. ","
else
bucket_pref = self.name .. "_bucket{"
end
for i, buc in ipairs(self.buckets) do
full_name[i+2] = string.format("%sle=\"%s\"}", bucket_pref, self.bucket_format:format(buc))
end
-- Last bucket. Note, that the label value is "Inf" rather than "+Inf"
-- required by Prometheus. This is necessary for this bucket to be the last
-- one when all metrics are lexicographically sorted. "Inf" will get replaced
-- by "+Inf" in Prometheus:metric_data().
full_name[self.bucket_count+3] = string.format("%sle=\"Inf\"}", bucket_pref)
else
full_name = full_metric_name(self.name, self.label_names, label_values)
end
t[LEAF_KEY] = full_name
local err = self._key_index:add(full_name, ERR_MSG_LRU_EVICTION)
if err then
return nil, err
end
return full_name
end
-- Increment a gauge metric.
--
-- Gauges are incremented in the dictionary directly to provide strong ordering
-- of inc() and set() operations.
--
-- Args:
-- self: a `metric` object, created by register().
-- value: numeric value to increment by. Can be negative.
-- label_values: a list of label values, in the same order as label keys.
local function inc_gauge(self, value, label_values)
local k, err, _, forcible
k, err = lookup_or_create(self, label_values)
if err then
self._log_error(err)
return
end
_, err, forcible = self._dict:incr(k, value, 0)
if err or forcible then
self._log_error_kv(k, value, err or ERR_MSG_LRU_EVICTION)
end
end
-- Increment a counter metric.
--
-- Counters are incremented in the per-worker counter, which will eventually get
-- flushed into the global shared dictionary.
--
-- Args:
-- self: a `metric` object, created by register().
-- value: numeric value to increment by. Can't be negative.
-- label_values: a list of label values, in the same order as label keys.
local function inc_counter(self, value, label_values)
-- counter is not allowed to decrease
if value and value < 0 then
self._log_error_kv(self.name, value, "Value should not be negative")
return
end
local k, err
k, err = lookup_or_create(self, label_values)
if err then
self._log_error(err)
return
end
local c = self._counter
if not c then
c = self.parent._counter
if not c then
self._log_error(ERR_MSG_COUNTER_NOT_INITIALIZED)
return
end
self._counter = c
end
c:incr(k, value)
end
-- Delete a counter or a gauge metric.
--
-- Args:
-- self: a `metric` object, created by register().
-- label_values: a list of label values, in the same order as label keys.
local function del(self, label_values)
local k, _, err
k, err = lookup_or_create(self, label_values)
if err then
self._log_error(err)
return
end
-- `del` might be called immediately after a configuration change that stops a
-- given metric from being used, so we cannot guarantee that other workers
-- don't have unflushed counter values for a metric that is about to be
-- deleted. We wait for `sync_interval` here to ensure that those values are
-- synced (and deleted from worker-local counters) before a given metric is
-- removed.
-- Gauge metrics don't use per-worker counters, so for gauges we don't need to
-- wait for the counter to sync.
if self.typ ~= TYPE_GAUGE then
ngx.log(ngx.INFO, "waiting ", self.parent.sync_interval, "s for counter to sync")
ngx.sleep(self.parent.sync_interval)
end
err = self._key_index:remove(k, ERR_MSG_LRU_EVICTION)
if err then
self._log_error(err)
end
_, err = self._dict:delete(k)
if err then
self._log_error("Error deleting key: ".. k .. ": " .. err)
end
-- Clean up the full metric name lookup table as well.
self.lookup = {}
end
-- Set the value of a gauge metric.
--
-- Args:
-- self: a `metric` object, created by register().
-- value: numeric value.
-- label_values: a list of label values, in the same order as label keys.
local function set(self, value, label_values)
if not value then
self._log_error("No value passed for " .. self.name)
return
end
local k, _, err, forcible
k, err = lookup_or_create(self, label_values)
if err then
self._log_error(err)
return
end
_, err, forcible = self._dict:set(k, value)
if err or forcible then
self._log_error_kv(k, value, err or ERR_MSG_LRU_EVICTION)
end
end
-- Record a given value in a histogram.
--
-- Args:
-- self: a `metric` object, created by register().
-- value: numeric value to record. Should be defined.
-- label_values: a list of label values, in the same order as label keys.
local function observe(self, value, label_values)
if not value then
self._log_error("No value passed for " .. self.name)
return
end
local keys, err = lookup_or_create(self, label_values)
if err then
self._log_error(err)
return
end
local c = self._counter
if not c then
c = self.parent._counter
if not c then
self._log_error(ERR_MSG_COUNTER_NOT_INITIALIZED)
return
end
self._counter = c
end
-- _count metric.
c:incr(keys[1], 1)
-- _sum metric.
c:incr(keys[2], value)
-- the last bucket (le="Inf").
c:incr(keys[self.bucket_count+3], 1)
local seen = false
-- check in reverse order, otherwise we will always
-- need to traverse the whole table.
for i=self.bucket_count, 1, -1 do
if value <= self.buckets[i] then
c:incr(keys[2+i], 1)
seen = true
elseif seen then
break
end
end
end
-- Delete all metrics for a given gauge, counter or a histogram.
--
-- This is like `del`, but will delete all time series for all previously
-- recorded label values.
--
-- Args:
-- self: a `metric` object, created by register().
local function reset(self)
-- Wait for other worker threads to sync their counters before removing the
-- metric (please see `del` for a more detailed comment).
-- Gauge metrics don't use per-worker counters, so for gauges we don't need to
-- wait for the counter to sync.
if self.typ ~= TYPE_GAUGE then
ngx.log(ngx.INFO, "waiting ", self.parent.sync_interval, "s for counter to sync")
ngx.sleep(self.parent.sync_interval)
end
local keys = self._key_index:list()
local name_prefixes = {}
local name_prefix_length_base = #self.name
if self.typ == TYPE_HISTOGRAM then
if self.label_count == 0 then
name_prefixes[self.name .. "_count"] = name_prefix_length_base + 6
name_prefixes[self.name .. "_sum"] = name_prefix_length_base + 4
else
name_prefixes[self.name .. "_count{"] = name_prefix_length_base + 7
name_prefixes[self.name .. "_sum{"] = name_prefix_length_base + 5
end
name_prefixes[self.name .. "_bucket{"] = name_prefix_length_base + 8
else
name_prefixes[self.name .. "{"] = name_prefix_length_base + 1
end
for _, key in ipairs(keys) do
local value, key_err = self._dict:get(key)
if value then
-- For a metric to be deleted its name should either match exactly, or
-- have a prefix listed in `name_prefixes` (which ensures deletion of
-- metrics with label values).
local remove = key == self.name
if not remove then
for name_prefix, name_prefix_length in pairs(name_prefixes) do
if name_prefix == string.sub(key, 1, name_prefix_length) then
remove = true
break
end
end
end
if remove then
local _, err
err = self._key_index:remove(key, ERR_MSG_LRU_EVICTION)
if err then
self._log_error(err)
end
_, err = self._dict:set(key, nil)
if err then
self._log_error("Error resetting '", key, "': ", err)
end
end
else
if type(key_err) == "string" then
self._log_error("Error getting '", key, "': ", key_err)
end
end
end
-- Clean up the full metric name lookup table as well.
self.lookup = {}
end
-- Initialize the module.
--
-- This should be called once from the `init_by_lua` section in nginx
-- configuration.
--
-- Args:
-- dict_name: (string) name of the nginx shared dictionary which will be
-- used to store all metrics
-- prefix: (optional string) if supplied, prefix is added to all
-- metric names on output
--
-- Returns:
-- an object that should be used to register metrics.
function Prometheus.init(dict_name, options_or_prefix)
if ngx.get_phase() ~= 'init' and ngx.get_phase() ~= 'init_worker' then
error('Prometheus.init can only be called from ' ..
'init_by_lua_block or init_worker_by_lua_block', 2)
end
local self = setmetatable({}, mt)
dict_name = dict_name or "prometheus_metrics"
self.dict_name = dict_name
self.dict = ngx.shared[dict_name]
if self.dict == nil then
error("Dictionary '" .. dict_name .. "' does not seem to exist. " ..
"Please define the dictionary using `lua_shared_dict`.", 2)
end
if type(options_or_prefix) == "table" then
self.prefix = options_or_prefix.prefix or ''
self.error_metric_name = options_or_prefix.error_metric_name or
DEFAULT_ERROR_METRIC_NAME
self.sync_interval = options_or_prefix.sync_interval or
DEFAULT_SYNC_INTERVAL
else
self.prefix = options_or_prefix or ''
self.error_metric_name = DEFAULT_ERROR_METRIC_NAME
self.sync_interval = DEFAULT_SYNC_INTERVAL
end
self.registry = {}
self.key_index = key_index_lib.new(self.dict, KEY_INDEX_PREFIX, function(metric_key)
-- When another worker calls reset or del on a metric, reset that
-- metric's local lookup table.
local metric_name = ngx_re_gsub(metric_key, "{.*", "", "jo")
if self.registry[metric_name] then
local m = self.registry[metric_name]
m.lookup = {}
return
end
metric_name = ngx_re_gsub(metric_name, "_sum$", "", "jo")
local m = self.registry[metric_name]
if m and m.typ == TYPE_HISTOGRAM then
m.lookup = {}
end
end)
self.initialized = true
self:counter(self.error_metric_name, "Number of nginx-lua-prometheus errors")
self.dict:set(self.error_metric_name, 0)
local err = self.key_index:add(self.error_metric_name, ERR_MSG_LRU_EVICTION)
if err then
self:log_error(err)
end
if ngx.get_phase() == 'init_worker' then
self:init_worker(self.sync_interval)
end
return self
end
-- Initialize the worker counter.
--
-- You can call this function from the `init_worker_by_lua` if you are calling
-- Prometheus.init() from `init_by_lua`, but this is deprecated. Instead, just
-- call Prometheus.init() from `init_worker_by_lua_block` and pass sync_interval
-- as part of the `options` argument if you need.
--
-- Args:
-- sync_interval: per-worker counter sync interval (in seconds).
function Prometheus:init_worker(sync_interval)
if ngx.get_phase() ~= 'init_worker' then
error('Prometheus:init_worker can only be called in ' ..
'init_worker_by_lua_block', 2)
end
if self._counter then
ngx.log(ngx.WARN, 'init_worker() has been called twice. ' ..
'Please do not explicitly call init_worker. ' ..
'Instead, call Prometheus:init() in the init_worker_by_lua_block')
return
end
self.sync_interval = sync_interval or DEFAULT_SYNC_INTERVAL
local counter_instance, err = resty_counter_lib.new(
self.dict_name, self.sync_interval, self.error_metric_name)
if err then
error(err, 2)
end
self._counter = counter_instance
ngx.timer.every(self.sync_interval, function (_)
self.key_index:sync()
end)
end
-- Register a new metric.
--
-- Args:
-- self: a Prometheus object.
-- name: (string) name of the metric. Required.
-- help: (string) description of the metric. Will be used for the HELP
-- comment on the metrics page. Optional.
-- label_names: array of strings, defining a list of metrics. Optional.
-- buckets: array if numbers, defining bucket boundaries. Only used for
-- histogram metrics.
-- typ: metric type (one of the TYPE_* constants).
--
-- Returns:
-- a new metric object.
local function register(self, name, help, label_names, buckets, typ)
if not self.initialized then
ngx.log(ngx.ERR, "Prometheus module has not been initialized")
return
end
local err = check_metric_and_label_names(name, label_names)
if err then
self:log_error(err)
return
end
local gsub_a = ngx_re_gsub(name, "_bucket$", "", "jo")
local gsub_b = ngx_re_gsub(gsub_a, "_count$", "", "jo")
local name_maybe_historgram = ngx_re_gsub(gsub_b, "_sum$", "", "jo")
if (typ ~= TYPE_HISTOGRAM and (
self.registry[name] or self.registry[name_maybe_historgram]
)) or
(typ == TYPE_HISTOGRAM and (
self.registry[name] or
self.registry[name .. "_count"] or
self.registry[name .. "_sum"] or self.registry[name .. "_bucket"]
)) then
self:log_error("Duplicate metric " .. name)
return
end
local metric = {
name = name,
help = help,
typ = typ,
label_names = label_names,
label_count = label_names and #label_names or 0,
-- Lookup is a tree of lua tables that contain label values, with leaf
-- tables containing full metric names. For example, given a metric
-- `http_count` and labels `host` and `status`, it might contain the
-- following values:
-- ['me.com']['200'][LEAF_KEY] = 'http_count{host="me.com",status="200"}'
-- ['me.com']['500'][LEAF_KEY] = 'http_count{host="me.com",status="500"}'
-- ['my.net']['200'][LEAF_KEY] = 'http_count{host="my.net",status="200"}'
-- ['my.net']['500'][LEAF_KEY] = 'http_count{host="my.net",status="500"}'
lookup = {},
parent = self,
-- Store a reference for logging functions for faster lookup.
_log_error = function(...) self:log_error(...) end,
_log_error_kv = function(...) self:log_error_kv(...) end,
_key_index = self.key_index,
_dict = self.dict,
reset = reset,
}
if typ < TYPE_HISTOGRAM then
if typ == TYPE_GAUGE then
metric.set = set
metric.inc = inc_gauge
else
metric.inc = inc_counter
end
metric.del = del
else
metric.observe = observe
metric.buckets = buckets or DEFAULT_BUCKETS
metric.bucket_count = #metric.buckets
metric.bucket_format = construct_bucket_format(metric.buckets)
end
self.registry[name] = metric
return metric
end
-- inspired by https://github.com/Kong/kong/blob/2.8.1/kong/tools/utils.lua#L1430-L1446
-- limit to work only in rewrite, access, content and timer
local yield
do
local counter = 0
yield = function()
if not can_yield_phases[get_phase()] then
return
end
counter = counter + 1
if counter % YIELD_ITERATIONS ~= 0 then
return
end
counter = 0
ngx_sleep(0)
end
end
-- Public function to register a counter.
function Prometheus:counter(name, help, label_names)
return register(self, name, help, label_names, nil, TYPE_COUNTER)
end
-- Public function to register a gauge.
function Prometheus:gauge(name, help, label_names)
return register(self, name, help, label_names, nil, TYPE_GAUGE)
end
-- Public function to register a histogram.
function Prometheus:histogram(name, help, label_names, buckets)
return register(self, name, help, label_names, buckets, TYPE_HISTOGRAM)
end
-- Prometheus compatible metric data as an array of strings.
--
-- Returns:
-- Array of strings with all metrics in a text format compatible with
-- Prometheus.
function Prometheus:metric_data()
if not self.initialized then
ngx.log(ngx.ERR, "Prometheus module has not been initialized")
return
end
-- Force a manual sync of counter local state (mostly to make tests work).
self._counter:sync()
local keys = self.key_index:list()
-- Prometheus server expects buckets of a histogram to appear in increasing
-- numerical order of their label values.
table.sort(keys)
local seen_metrics = {}
local output = {}
for _, key in ipairs(keys) do
yield()
local value, err = self.dict:get(key)
if value then
local short_name = short_metric_name(key)
if not seen_metrics[short_name] then
local m = self.registry[short_name]
if m then
if m.help then
table_insert_tail(output, string.format("# HELP %s%s %s\n",
self.prefix, short_name, m.help))
end
if m.typ then
table_insert_tail(output, string.format("# TYPE %s%s %s\n",
self.prefix, short_name, TYPE_LITERAL[m.typ]))
end
end
seen_metrics[short_name] = true
end
key = fix_histogram_bucket_labels(key)
table_insert_tail(output, string.format("%s%s %s\n", self.prefix, key, value))
else
if type(err) == "string" then
self:log_error("Error getting '", key, "': ", err)
end
end
end
return output
end
-- Present all metrics in a text format compatible with Prometheus.
--
-- This function should be used to expose the metrics on a separate HTTP page.
-- It will get the metrics from the dictionary, sort them, and expose them
-- aling with TYPE and HELP comments.
function Prometheus:collect()
ngx.header.content_type = "text/plain"
ngx.print(self:metric_data())
end
-- Log an error, incrementing the error counter.
function Prometheus:log_error(...)
ngx.log(ngx.ERR, ...)
self.dict:incr(self.error_metric_name, 1, 0)
end
-- Log an error that happened while setting up a dictionary key.
function Prometheus:log_error_kv(key, value, err)
self:log_error(
"Error while setting '", key, "' to '", value, "': '", err, "'")
end
return Prometheus