diff --git a/lib/resty/wasmx.lua b/lib/resty/wasmx.lua index 3fddad748..bfe0543b5 100644 --- a/lib/resty/wasmx.lua +++ b/lib/resty/wasmx.lua @@ -17,6 +17,9 @@ ffi.cdef [[ typedef unsigned char u_char; typedef intptr_t ngx_int_t; + typedef uintptr_t ngx_uint_t; + typedef ngx_uint_t ngx_msec_t; + typedef struct ngx_log_s ngx_log_t; typedef struct ngx_wavm_t ngx_wasm_vm_t; typedef struct ngx_wasm_ops_plan_t ngx_wasm_plan_t; diff --git a/lib/resty/wasmx/shm.lua b/lib/resty/wasmx/shm.lua new file mode 100644 index 000000000..32fe5c070 --- /dev/null +++ b/lib/resty/wasmx/shm.lua @@ -0,0 +1,505 @@ +-- vim:set ts=4 sw=4 sts=4 et: + +local ffi = require "ffi" +local wasmx = require "resty.wasmx" + + +local C = ffi.C +local error = error +local type = type +local table_new = table.new +local str_fmt = string.format +local ffi_cast = ffi.cast +local ffi_fill = ffi.fill +local ffi_new = ffi.new +local ffi_str = ffi.string +local ngx_log = ngx.log +local ngx_debug = ngx.DEBUG +local ngx_warn = ngx.WARN +local ngx_sleep = ngx.sleep +local FFI_ABORT = wasmx.FFI_ABORT +local FFI_DECLINED = wasmx.FFI_DECLINED +local FFI_ERROR = wasmx.FFI_ERROR +local FFI_OK = wasmx.FFI_OK + + +ffi.cdef [[ + typedef struct ngx_slab_pool_s ngx_slab_pool_t; + + typedef enum { + NGX_WA_SHM_TYPE_KV, + NGX_WA_SHM_TYPE_QUEUE, + NGX_WA_SHM_TYPE_METRICS, + } ngx_wa_shm_type_e; + + typedef enum { + NGX_WA_SHM_EVICTION_LRU, + NGX_WA_SHM_EVICTION_SLRU, + NGX_WA_SHM_EVICTION_NONE, + } ngx_wa_shm_eviction_e; + + typedef struct { + ngx_wa_shm_type_e type; + ngx_wa_shm_eviction_e eviction; + ngx_str_t name; + ngx_log_t *log; + ngx_slab_pool_t *shpool; + void *data; + } ngx_wa_shm_t; + + typedef enum { + NGX_WA_METRIC_COUNTER, + NGX_WA_METRIC_GAUGE, + NGX_WA_METRIC_HISTOGRAM, + } ngx_wa_metric_type_e; + + typedef struct { + ngx_uint_t value; + ngx_msec_t last_update; + } ngx_wa_metrics_gauge_t; + + typedef struct { + uint32_t upper_bound; + uint32_t count; + } ngx_wa_metrics_bin_t; + + typedef struct { + uint8_t n_bins; + ngx_wa_metrics_bin_t bins[]; + } ngx_wa_metrics_histogram_t; + + typedef union { + ngx_uint_t counter; + ngx_wa_metrics_gauge_t gauge; + ngx_wa_metrics_histogram_t *histogram; + } ngx_wa_metric_val_t; + + typedef struct { + ngx_wa_metric_type_e metric_type; + ngx_wa_metric_val_t slots[]; + } ngx_wa_metric_t; + + + typedef void (*ngx_wa_ffi_shm_setup_zones_handler)(ngx_wa_shm_t *shm); + + + ngx_int_t ngx_wa_ffi_shm_setup_zones(ngx_wa_ffi_shm_setup_zones_handler handler); + ngx_int_t ngx_wa_ffi_shm_iterate_keys(ngx_wa_shm_t *shm, + ngx_uint_t page, + ngx_uint_t page_size, + ngx_str_t **keys, + ngx_uint_t *total); + void ngx_wa_ffi_shm_lock(ngx_wa_shm_t *shm); + void ngx_wa_ffi_shm_unlock(ngx_wa_shm_t *shm); + + ngx_int_t ngx_wa_ffi_shm_get_kv_value(ngx_wa_shm_t *shm, + ngx_str_t *k, + ngx_str_t **v, + uint32_t *cas); + ngx_int_t ngx_wa_ffi_shm_set_kv_value(ngx_wa_shm_t *shm, + ngx_str_t *k, + ngx_str_t *v, + uint32_t cas, + ngx_int_t *written); + + ngx_int_t ngx_wa_ffi_shm_define_metric(ngx_str_t *name, + ngx_wa_metric_type_e type, + uint32_t *metric_id); + ngx_int_t ngx_wa_ffi_shm_record_metric(uint32_t metric_id, + ngx_uint_t value); + ngx_int_t ngx_wa_ffi_shm_increment_metric(uint32_t metric_id, + ngx_uint_t value); + ngx_int_t ngx_wa_ffi_shm_get_metric(uint32_t metric_id, + ngx_str_t *name, + u_char *mbuf, size_t mbs, + u_char *hbuf, size_t hbs); + + ngx_int_t ngx_wa_ffi_shm_one_slot_metric_size(); + ngx_int_t ngx_wa_ffi_shm_max_histogram_size(); +]] + + +local WASM_SHM_KEY = {} +local DEFAULT_KEY_BATCH_SIZE = 500 + + +local _M = setmetatable({}, { + __index = function(_, k) + error("resty.wasmx.shm: no \"" .. k .. "\" shm configured", 2) + end, +}) + + +local _types = { + ffi_shm = { + SHM_TYPE_KV = 0, + SHM_TYPE_QUEUE = 1, + SHM_TYPE_METRICS = 2, + }, + ffi_metric = { + COUNTER = 0, + GAUGE = 1, + HISTOGRAM = 2, + } +} + +local _metric_type_set = { + [_types.ffi_metric.COUNTER] = true, + [_types.ffi_metric.GAUGE] = true, + [_types.ffi_metric.HISTOGRAM] = true, +} + +local _initialized = false +local _kbuf = ffi_new("ngx_str_t *[?]", DEFAULT_KEY_BATCH_SIZE) +local _mbs = C.ngx_wa_ffi_shm_one_slot_metric_size() +local _mbuf = ffi_new("u_char [?]", _mbs) +local _hbs = C.ngx_wa_ffi_shm_max_histogram_size() +local _hbuf = ffi_new("u_char [?]", _hbs) + + +local function lock_shm(zone) + C.ngx_wa_ffi_shm_lock(zone[WASM_SHM_KEY]) +end + + +local function unlock_shm(zone) + C.ngx_wa_ffi_shm_unlock(zone[WASM_SHM_KEY]) +end + + +local function key_iterator(ctx) + if ctx.i < ctx.page_total then + local ckey = ctx.ckeys[ctx.i] + ctx.i = ctx.i + 1 + + return ffi_str(ckey.data, ckey.len) + + else + ctx.cpage_total[0] = 0 + + local rc = C.ngx_wa_ffi_shm_iterate_keys(ctx.shm, + ctx.page, ctx.page_size, + ctx.ckeys, ctx.cpage_total) + if rc == FFI_ABORT then + local zone_name = ffi_str(ctx.shm.name.data, ctx.shm.name.len) + local err = "attempt to iterate over the keys of an unlocked shm zone." + .. " please call resty.wasmx.shm.%s:lock() before calling" + .. " iterate_keys() and resty.wasmx.shm.%s:unlock() after" + + error(str_fmt(err, zone_name, zone_name), 2) + end + + if rc == FFI_DECLINED then + return + end + + ngx_sleep(0) + + ctx.page_total = tonumber(ctx.cpage_total[0]) + ctx.page = ctx.page + ctx.page_total + ctx.i = 1 + + return ffi_str(ctx.ckeys[0].data, ctx.ckeys[0].len) + end +end + + +local function iterate_keys(zone, page_size) + if page_size ~= nil and type(page_size) ~= "number" then + error("page_size must be a number", 2) + end + + local ctx = { + shm = zone[WASM_SHM_KEY], + i = 0, + page = 0, + page_size = page_size and page_size or DEFAULT_KEY_BATCH_SIZE, + page_total = 0, + cpage_total = ffi_new("ngx_uint_t [1]"), + ckeys = page_size and ffi_new("ngx_str_t *[?]", page_size) or _kbuf + } + + return key_iterator, ctx +end + + +local function get_keys(zone, max) + if max ~= nil and type(max) ~= "number" then + error("max must be number", 2) + end + + local shm = zone[WASM_SHM_KEY] + local nkeys + local keys + local start = 0 + local ctotal = ffi_new("ngx_uint_t [1]") + + lock_shm(zone) + + if max == 0 then + local rc = C.ngx_wa_ffi_shm_iterate_keys(shm, start, 0, nil, ctotal) + nkeys = tonumber(ctotal[0]) + + if rc == FFI_DECLINED then + unlock_shm(zone) + return {} + end + + else + nkeys = max and max or DEFAULT_KEY_BATCH_SIZE + end + + local ckeys = ffi_new("ngx_str_t *[?]", nkeys) + + C.ngx_wa_ffi_shm_iterate_keys(shm, start, nkeys, ckeys, ctotal) + + unlock_shm(zone) + + local total = tonumber(ctotal[0]) + local keys = table_new(0, total) + + for i = 1, total do + local ckey = ckeys[i - 1] + keys[i] = ffi_str(ckey.data, ckey.len) + end + + return keys +end + + +local function get_kv_value(zone, key) + if type(key) ~= "string" then + error("key must be string", 2) + end + + local shm = zone[WASM_SHM_KEY] + local cname = ffi_new("ngx_str_t", { data = key, len = #key }) + local cvalue = ffi_new("ngx_str_t *[1]") + local ccas = ffi_new("uint32_t [1]") + + local rc = C.ngx_wa_ffi_shm_get_kv_value(shm, cname, cvalue, ccas) + if rc == FFI_DECLINED then + return nil, nil, "key not found" + end + + return ffi_str(cvalue[0].data, cvalue[0].len), tonumber(ccas[0]) +end + + +local function set_kv_value(zone, key, value, cas) + if type(key) ~= "string" then + error("key must be a string", 2) + end + + if type(value) ~= "string" then + error("value must be a string", 2) + end + + if type(cas) ~= "number" then + error("cas must be a number", 2) + end + + local shm = zone[WASM_SHM_KEY] + local cname = ffi_new("ngx_str_t", { data = key, len = #key }) + local cvalue = ffi_new("ngx_str_t", { data = value, len = #value }) + local written = ffi_new("uint32_t [1]") + + local rc = C.ngx_wa_ffi_shm_set_kv_value(shm, cname, cvalue, cas, written) + if rc == FFI_ERR then + return nil, "failed setting kv value, no memory" + end + + return tonumber(written[0]) +end + + +local function parse_cmetric(cmetric) + if cmetric.metric_type == _types.ffi_metric.COUNTER then + return { type = "counter", value = tonumber(cmetric.slots[0].counter) } + + elseif cmetric.metric_type == _types.ffi_metric.GAUGE then + return { type = "gauge", value = tonumber(cmetric.slots[0].gauge.value) } + + elseif cmetric.metric_type == _types.ffi_metric.HISTOGRAM then + local hbuf = cmetric.slots[0].histogram + local ch = ffi_cast("ngx_wa_metrics_histogram_t *", hbuf) + local h = { type = "histogram", value = {} } + + for i = 0, ch.n_bins do + local cb = ch.bins[i] + + if cb.upper_bound == 0 then + break + end + + h.value[#(h.value) + 1] = { ub = cb.upper_bound, count = cb.count } + end + + return h + end +end + + +local function get_metric(zone, metric_id) + if type(metric_id) ~= "number" then + error("metric_id must be a number", 2) + end + + ffi_fill(_mbuf, _mbs) + ffi_fill(_hbuf, _hbs) + + local rc = C.ngx_wa_ffi_shm_get_metric(metric_id, nil, + _mbuf, _mbs, _hbuf, _hbs) + if rc == FFI_DECLINED then + return nil, "metric not found" + end + + return parse_cmetric(ffi_cast("ngx_wa_metric_t *", _mbuf)) +end + + +--- +-- ngx_wasm_module internally prefixes metric names according to where they +-- have been defined, e.g. pw.filter.metric, lua.metric or wa.metric. +-- +-- get_metric_by_name assumes that it's retrieving a lua land metric and +-- will by default prefix name with `lua.` +-- +-- This behavior can be disabled by passing `opts.prefix` as false. +local function get_metric_by_name(zone, name, opts) + if type(name) ~= "string" or name == "" then + error("name must be a non-empty string", 2) + end + + if opts ~= nil then + if type(opts) ~= "table" then + error("opts must be a table", 2) + end + + if opts.prefix ~= nil and type(opts.prefix) ~= "boolean" then + error("opts.prefix must be a boolean", 2) + end + end + + name = (opts and opts.prefix == false) and name or "lua." .. name + local cname = ffi_new("ngx_str_t", { data = name, len = #name }) + + ffi_fill(_mbuf, _mbs) + ffi_fill(_hbuf, _hbs) + + local rc = C.ngx_wa_ffi_shm_get_metric(0, cname, _mbuf, _mbs, _hbuf, _hbs) + if rc == FFI_DECLINED then + return nil, "metric not found" + end + + return parse_cmetric(ffi_cast("ngx_wa_metric_t *", _mbuf)) +end + + +local function define_metric(zone, name, metric_type) + if type(name) ~= "string" or name == "" then + error("name must be a non-empty string", 2) + end + + if _metric_type_set[metric_type] == nil then + local err = "metric_type must be either" .. + " resty.wasmx.shm.metrics.COUNTER," .. + " resty.wasmx.shm.metrics.GAUGE, or" .. + " resty.wasmx.shm.metrics.HISTOGRAM" + + error(err, 2) + end + + local cname = ffi_new("ngx_str_t", { data = name, len = #name }) + local m_id = ffi_new("uint32_t [1]") + + local rc = C.ngx_wa_ffi_shm_define_metric(cname, metric_type, m_id) + if rc == FFI_ABORT then + return nil, "failed defining metric, name too long" + end + + if rc == FFI_ERROR then + return nil, "failed defining metric, no memory" + end + + return tonumber(m_id[0]) +end + + +local function record_metric(zone, metric_id, value) + if type(metric_id) ~= "number" then + error("metric_id must be a number", 2) + end + + if type(value) ~= "number" then + error("value must be a number", 2) + end + + local rc = C.ngx_wa_ffi_shm_record_metric(metric_id, value) + if rc == FFI_DECLINED then + return nil, "metric not found" + end + + return true +end + + +local function increment_metric(zone, metric_id, value) + if type(metric_id) ~= "number" then + error("metric_id must be a number", 2) + end + + if value ~= nil and (type(value) ~= "number" or value < 1) then + error("value must be a number greater than zero", 2) + end + + value = value and value or 1 + + local rc = C.ngx_wa_ffi_shm_increment_metric(metric_id, value) + if rc == FFI_DECLINED then + return nil, "metric not found" + end + + return true +end + + +local _setup_zones_handler = ffi_cast("ngx_wa_ffi_shm_setup_zones_handler", +function(shm) + local zone_name = ffi_str(shm.name.data, shm.name.len) + _M[zone_name] = { + [WASM_SHM_KEY] = shm, + lock = lock_shm, + unlock = unlock_shm, + } + + if shm.type == _types.ffi_shm.SHM_TYPE_KV then + _M[zone_name].get_keys = get_keys + _M[zone_name].iterate_keys = iterate_keys + _M[zone_name].get = get_kv_value + _M[zone_name].set = set_kv_value + + elseif shm.type == _types.ffi_shm.SHM_TYPE_QUEUE then + -- NYI + + elseif shm.type == _types.ffi_shm.SHM_TYPE_METRICS then + _M[zone_name].get_keys = get_keys + _M[zone_name].iterate_keys = iterate_keys + _M[zone_name].get = get_metric + _M[zone_name].get_by_name = get_metric_by_name + _M[zone_name].define = define_metric + _M[zone_name].increment = increment_metric + _M[zone_name].record = record_metric + _M[zone_name].COUNTER = _types.ffi_metric.COUNTER + _M[zone_name].GAUGE = _types.ffi_metric.GAUGE + _M[zone_name].HISTOGRAM = _types.ffi_metric.HISTOGRAM + end +end) + + +if C.ngx_wa_ffi_shm_setup_zones(_setup_zones_handler) == FFI_ABORT then + ngx_log(ngx.DEBUG, "no shm zones found for resty.wasmx.shm interface") +end + + +return _M diff --git a/src/common/lua/ngx_wasm_lua_ffi.c b/src/common/lua/ngx_wasm_lua_ffi.c index 1c24ae600..5bf2936f8 100644 --- a/src/common/lua/ngx_wasm_lua_ffi.c +++ b/src/common/lua/ngx_wasm_lua_ffi.c @@ -4,6 +4,7 @@ #include "ddebug.h" #include +#include #include @@ -257,3 +258,218 @@ ngx_http_wasm_ffi_set_host_properties_handlers(ngx_http_request_t *r, return ngx_proxy_wasm_properties_set_ffi_handlers(pwctx, getter, setter, r); } #endif + + +void +ngx_wa_ffi_shm_lock(ngx_wa_shm_t *shm) +{ + ngx_wa_assert(shm); + + ngx_wa_shm_lock(shm); +} + + +void +ngx_wa_ffi_shm_unlock(ngx_wa_shm_t *shm) +{ + ngx_wa_assert(shm); + + ngx_wa_shm_unlock(shm); +} + + +ngx_int_t +ngx_wa_ffi_shm_setup_zones(ngx_wa_ffi_shm_setup_zones_handler_pt setup_handler) +{ + ngx_uint_t i; + ngx_array_t *shms = ngx_wasmx_shms((ngx_cycle_t *) ngx_cycle); + ngx_wa_shm_t *shm; + ngx_wa_shm_mapping_t *mappings; + + ngx_wa_assert(setup_handler); + + if (shms == NULL) { + return NGX_ABORT; + } + + mappings = shms->elts; + + for (i = 0; i < shms->nelts; i++) { + shm = mappings[i].zone->data; + setup_handler(shm); + } + + return NGX_OK; +} + + +static void +shm_kv_retrieve_keys(ngx_rbtree_node_t *node, ngx_rbtree_node_t *sentinel, + ngx_uint_t start, ngx_uint_t limit, + ngx_str_t **keys, ngx_uint_t *total) +{ + ngx_wa_shm_kv_node_t *n = (ngx_wa_shm_kv_node_t *) node; + + if (limit > 0 && (*total - start) == limit) { + return; + } + + if (keys && *total >= start) { + keys[(*total - start)] = &n->key.str; + } + + (*total)++; + + if (node->left != sentinel) { + shm_kv_retrieve_keys(node->left, sentinel, start, limit, keys, total); + } + + if (node->right != sentinel) { + shm_kv_retrieve_keys(node->right, sentinel, start, limit, keys, total); + } +} + + +ngx_int_t +ngx_wa_ffi_shm_iterate_keys(ngx_wa_shm_t *shm, + ngx_uint_t page, ngx_uint_t page_size, + ngx_str_t **keys, ngx_uint_t *total) +{ + ngx_wa_shm_kv_t *kv; + + ngx_wa_assert(shm && shm->type != NGX_WA_SHM_TYPE_QUEUE); + + if (!ngx_wa_shm_locked(shm)) { + return NGX_ABORT; + } + + kv = shm->data; + + if (page >= kv->nelts) { + return NGX_DECLINED; + } + + if (kv->rbtree.root != kv->rbtree.sentinel) { + shm_kv_retrieve_keys(kv->rbtree.root, kv->rbtree.sentinel, + page, page_size, keys, total); + } + + *total = *total - page; + + return NGX_OK; +} + + +ngx_int_t +ngx_wa_ffi_shm_get_kv_value(ngx_wa_shm_t *shm, + ngx_str_t *k, ngx_str_t **v, uint32_t *cas) +{ + unsigned unlock = 0; + ngx_int_t rc; + + ngx_wa_assert(shm && k && v && cas); + + if (!ngx_wa_shm_locked(shm)) { + ngx_wa_shm_lock(shm); + unlock = 1; + } + + rc = ngx_wa_shm_kv_get_locked(shm, k, NULL, v, cas); + + if (unlock) { + ngx_wa_shm_unlock(shm); + } + + return rc; +} + + +ngx_int_t +ngx_wa_ffi_shm_set_kv_value(ngx_wa_shm_t *shm, + ngx_str_t *k, ngx_str_t *v, uint32_t cas, + ngx_int_t *written) +{ + unsigned unlock = 0; + ngx_int_t rc; + + ngx_wa_assert(shm && k && v && written); + + if (!ngx_wa_shm_locked(shm)) { + ngx_wa_shm_lock(shm); + unlock = 1; + } + + rc = ngx_wa_shm_kv_set_locked(shm, k, v, cas, written); + + if (unlock) { + ngx_wa_shm_unlock(shm); + } + + return rc; +} + + +ngx_int_t +ngx_wa_ffi_shm_define_metric(ngx_str_t *name, ngx_wa_metric_type_e type, + uint32_t *metric_id) +{ + ngx_int_t rc; + ngx_str_t prefixed_name; + ngx_wa_metrics_t *metrics = ngx_wasmx_metrics((ngx_cycle_t *) ngx_cycle); + u_char buf[NGX_MAX_ERROR_STR]; + + ngx_wa_assert(metrics && name); + + prefixed_name.data = buf; + prefixed_name.len = ngx_sprintf(buf, "lua.%V", name) - buf; + + rc = ngx_wa_metrics_define(metrics, &prefixed_name, type, metric_id); + if (rc != NGX_OK) { + return rc; + } + + return NGX_OK; +} + + +ngx_int_t +ngx_wa_ffi_shm_increment_metric(uint32_t metric_id, ngx_uint_t value) +{ + ngx_wa_metrics_t *metrics = ngx_wasmx_metrics((ngx_cycle_t *) ngx_cycle); + + ngx_wa_assert(metrics); + + return ngx_wa_metrics_increment(metrics, metric_id, value); +} + + +ngx_int_t +ngx_wa_ffi_shm_record_metric(uint32_t metric_id, ngx_uint_t value) +{ + ngx_wa_metrics_t *metrics = ngx_wasmx_metrics((ngx_cycle_t *) ngx_cycle); + + ngx_wa_assert(metrics); + + return ngx_wa_metrics_record(metrics, metric_id, value); +} + + +ngx_int_t +ngx_wa_ffi_shm_get_metric(uint32_t metric_id, ngx_str_t *name, + u_char *m_buf, size_t mbs, u_char *h_buf, size_t hbs) +{ + ngx_wa_metrics_t *metrics = ngx_wasmx_metrics((ngx_cycle_t *) ngx_cycle); + ngx_wa_metric_t *m; + + ngx_wa_assert(metrics && (name || metric_id) && m_buf && h_buf); + + m = (ngx_wa_metric_t *) m_buf; + ngx_wa_metrics_histogram_set_buffer(m, h_buf, hbs); + + if (metric_id) { + return ngx_wa_metrics_get(metrics, metric_id, m); + } + + return ngx_wa_metrics_get(metrics, + ngx_crc32_long(name->data, name->len), m); +} diff --git a/src/common/lua/ngx_wasm_lua_ffi.h b/src/common/lua/ngx_wasm_lua_ffi.h index 0ed226f86..50bcab114 100644 --- a/src/common/lua/ngx_wasm_lua_ffi.h +++ b/src/common/lua/ngx_wasm_lua_ffi.h @@ -24,6 +24,9 @@ typedef struct { } ngx_wasm_ffi_filter_t; +typedef void (*ngx_wa_ffi_shm_setup_zones_handler_pt)(ngx_wa_shm_t *shm); + + ngx_int_t ngx_http_wasm_ffi_plan_new(ngx_wavm_t *vm, ngx_wasm_ffi_filter_t *filters, size_t n_filters, ngx_wasm_ops_plan_t **out, u_char *err, size_t *errlen); @@ -43,4 +46,38 @@ ngx_int_t ngx_http_wasm_ffi_set_host_properties_handlers(ngx_http_request_t *r, #endif +ngx_int_t ngx_wa_ffi_shm_setup_zones( + ngx_wa_ffi_shm_setup_zones_handler_pt handler); +ngx_int_t ngx_wa_ffi_shm_iterate_keys(ngx_wa_shm_t *shm, ngx_uint_t page, + ngx_uint_t page_size, ngx_str_t **keys, ngx_uint_t *total); +void ngx_wa_ffi_shm_lock(ngx_wa_shm_t *shm); +void ngx_wa_ffi_shm_unlock(ngx_wa_shm_t *shm); + +ngx_int_t ngx_wa_ffi_shm_get_kv_value(ngx_wa_shm_t *shm, ngx_str_t *k, + ngx_str_t **v, uint32_t *cas); +ngx_int_t ngx_wa_ffi_shm_set_kv_value(ngx_wa_shm_t *shm, ngx_str_t *k, + ngx_str_t *v, uint32_t cas, ngx_int_t *written); + +ngx_int_t ngx_wa_ffi_shm_define_metric(ngx_str_t *name, + ngx_wa_metric_type_e type, uint32_t *metric_id); +ngx_int_t ngx_wa_ffi_shm_increment_metric(uint32_t metric_id, ngx_uint_t value); +ngx_int_t ngx_wa_ffi_shm_record_metric(uint32_t metric_id, ngx_uint_t value); +ngx_int_t ngx_wa_ffi_shm_get_metric(uint32_t metric_id, ngx_str_t *name, + u_char *m_buf, size_t mbs, u_char *h_buf, size_t hbs); + + +ngx_int_t +ngx_wa_ffi_shm_one_slot_metric_size() +{ + return NGX_WA_METRICS_ONE_SLOT_METRIC_SIZE; +} + + +ngx_int_t +ngx_wa_ffi_shm_max_histogram_size() +{ + return NGX_WA_METRICS_MAX_HISTOGRAM_SIZE; +} + + #endif /* _NGX_WASM_LUA_FFI_H_INCLUDED_ */ diff --git a/src/common/shm/ngx_wa_shm.h b/src/common/shm/ngx_wa_shm.h index 671f79656..1e21fe862 100644 --- a/src/common/shm/ngx_wa_shm.h +++ b/src/common/shm/ngx_wa_shm.h @@ -59,4 +59,11 @@ ngx_wa_shm_unlock(ngx_wa_shm_t *shm) } +static ngx_inline unsigned +ngx_wa_shm_locked(ngx_wa_shm_t *shm) +{ + return (ngx_pid_t) *shm->shpool->mutex.lock == ngx_pid; +} + + #endif /* _NGX_WA_SHM_H_INCLUDED_ */ diff --git a/t/04-openresty/ffi/shm/001-setup_zones.t b/t/04-openresty/ffi/shm/001-setup_zones.t new file mode 100644 index 000000000..5e881cf91 --- /dev/null +++ b/t/04-openresty/ffi/shm/001-setup_zones.t @@ -0,0 +1,74 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(4); +run_tests(); + +__DATA__ + +=== TEST 1: shm - setup_zones() sanity +setup_zones() is silently called when resty.wasmx.shm module is loaded. +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + shm_queue q1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + assert(shm.kv1) + assert(shm.q1) + assert(shm.metrics) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] + + + +=== TEST 2: shm - setup_zones() no zones +--- valgrind +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm - bad zone indexing +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + print(shm.bad) + } + } +--- error_code: 500 +--- response_body_like: 500 Internal Server Error +--- error_log eval +qr/runtime error: access_by_lua\(nginx\.conf:\d+\):4: resty\.wasmx\.shm: no "bad" shm configured/ +--- no_error_log +[crit] diff --git a/t/04-openresty/ffi/shm/002-iterate_keys.t b/t/04-openresty/ffi/shm/002-iterate_keys.t new file mode 100644 index 000000000..a1932914a --- /dev/null +++ b/t/04-openresty/ffi/shm/002-iterate_keys.t @@ -0,0 +1,182 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm - iterate_keys() +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local data = { k1 = "v1", k2 = "v2" } + local retrieved_keys = {} + + for k, v in pairs(data) do + shm.kv1:set(k, v, 0) + end + + shm.kv1:lock() + + for k in shm.kv1:iterate_keys() do + retrieved_keys[k] = true + end + + shm.kv1:unlock() + + assert(retrieved_keys.k1) + assert(retrieved_keys.k2) + assert(retrieved_keys.inexistent_key == nil) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm - iterate_keys(), with batch size +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local data = { k1 = "v1", k2 = "v2" } + local retrieved_keys = {} + local batch_size = 1 + + for k, v in pairs(data) do + shm.kv1:set(k, v, 0) + end + + shm.kv1:lock() + + for k in shm.kv1:iterate_keys(batch_size) do + retrieved_keys[k] = true + end + + shm.kv1:unlock() + + assert(retrieved_keys.k1) + assert(retrieved_keys.k2) + assert(retrieved_keys.inexistent_key == nil) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 3: shm - iterate_keys(), no keys +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv1:lock() + + for k in shm.kv1:iterate_keys() do + ngx.say("fail") + end + + shm.kv1:unlock() + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 4: shm - iterate_keys(), unlocked +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + for k in shm.kv1:iterate_keys() do + end + } + } +--- error_code: 500 +--- error_log eval +qr/\[error\] .*? attempt to iterate over the keys of an unlocked shm zone. please call resty.wasmx.shm.kv1:lock\(\) before calling iterate_keys\(\) and resty.wasmx.shm.kv1:unlock\(\) after/ +--- no_error_log +[crit] +[emerg] +[alert] +[stub] + + + +=== TEST 5: shm - iterate_keys(), queue +--- valgrind +--- main_config + wasm { + shm_queue q1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + for k in shm.q1:iterate_keys() do + end + } + } +--- error_code: 500 +--- error_log eval +qr/\[error\] .*? attempt to call method 'iterate_keys' \(a nil value\)/ +--- no_error_log +[crit] +[emerg] +[alert] +[stub] diff --git a/t/04-openresty/ffi/shm/003-get_keys.t b/t/04-openresty/ffi/shm/003-get_keys.t new file mode 100644 index 000000000..b5a16684f --- /dev/null +++ b/t/04-openresty/ffi/shm/003-get_keys.t @@ -0,0 +1,143 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm - get_keys() +--- valgrind +--- wasm_modules: hostcalls +--- shm_kv: kv1 16k +--- config + location /t { + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv1/k1 \ + value=hello1'; + + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv1/k2 \ + value=hello2'; + + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local retrieved_keys = {} + + for _, k in ipairs(shm.kv1:get_keys()) do + retrieved_keys[k] = true + end + + assert(retrieved_keys["kv1/k1"]) + assert(retrieved_keys["kv1/k2"]) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm - get_keys(), with limit +--- valgrind +--- wasm_modules: hostcalls +--- shm_kv: kv1 16k +--- config + location /t { + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv1/k1 \ + value=hello1'; + + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv1/k2 \ + value=hello2'; + + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local retrieved_keys = {} + local limit = 1 + + for _, k in ipairs(shm.kv1:get_keys(limit)) do + retrieved_keys[k] = true + end + + assert(retrieved_keys["kv1/k1"]) + assert(retrieved_keys["kv1/k2"] == nil) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 3: shm - get_keys(), no keys +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + assert(#shm.kv1:get_keys() == 0) + assert(#shm.kv1:get_keys(0) == 0) + assert(#shm.kv1:get_keys(1) == 0) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 4: shm - get_keys(), queue +--- valgrind +--- main_config + wasm { + shm_queue q1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.q1:get_keys() + } + } +--- error_code: 500 +--- error_log eval +qr/\[error\] .*? attempt to call method 'get_keys' \(a nil value\)/ +--- no_error_log +[crit] +[emerg] +[alert] +[stub] diff --git a/t/04-openresty/ffi/shm/101-kv_get.t b/t/04-openresty/ffi/shm/101-kv_get.t new file mode 100644 index 000000000..3e1b35656 --- /dev/null +++ b/t/04-openresty/ffi/shm/101-kv_get.t @@ -0,0 +1,90 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm_kv - get() +--- valgrind +--- wasm_modules: hostcalls +--- shm_kv: kv1 16k +--- config + location /t { + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv1/k1 \ + value=v1'; + + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv1/k2 \ + value=v2'; + + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local expected = { ["kv1/k1"] = "v1", ["kv1/k2"] = "v2" } + + shm.kv1:lock() + + for k in shm.kv1:iterate_keys() do + local v, cas, err = shm.kv1:get(k) + assert(v == expected[k]) + assert(cas == 1) + end + + shm.kv1:unlock() + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm_kv - get(), unlocked +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local initial_cas = 0 + local k = "k1" + local expected_v = "v1" + + local written, err = shm.kv1:set(k, expected_v, initial_cas) + assert(written == 1) + assert(err == nil) + + local v, cas, err = shm.kv1:get(k) + assert(v == expected_v) + assert(cas == initial_cas + 1) + assert(err == nil) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] diff --git a/t/04-openresty/ffi/shm/102-kv_set.t b/t/04-openresty/ffi/shm/102-kv_set.t new file mode 100644 index 000000000..b440bb2f4 --- /dev/null +++ b/t/04-openresty/ffi/shm/102-kv_set.t @@ -0,0 +1,102 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm_kv - set() +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local values = { k1 = "v1", k2 = "v2" } + local cas = 0 + + for k, v in pairs(values) do + local written, err = shm.kv1:set(k, v, cas) + assert(written == 1) + assert(err == nil) + end + + for k, expected_v in pairs(values) do + local v, cas, err = shm.kv1:get(k) + assert(v == expected_v) + assert(cas == 1) + assert(err == nil) + end + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm_kv - set(), locked +--- valgrind +--- main_config + wasm { + shm_kv kv1 16k; + } +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local initial_values = { k1 = "v1", k2 = "v2" } + local expected_values = { k1 = "nv1", k2 = "nv2" } + local cas = 0 + + for k, v in pairs(initial_values) do + local written, err = shm.kv1:set(k, v, cas) + assert(written == 1) + assert(err == nil) + end + + shm.kv1:lock() + + for k in shm.kv1:iterate_keys() do + local written, err = shm.kv1:set(k, expected_values[k], cas + 1) + assert(written == 1) + assert(err == nil) + end + + shm.kv1:unlock() + + for k, expected_v in pairs(expected_values) do + local v, cas, err = shm.kv1:get(k) + assert(v == expected_v) + assert(cas == 2) + assert(err == nil) + end + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] diff --git a/t/04-openresty/ffi/shm/201-metrics_get.t b/t/04-openresty/ffi/shm/201-metrics_get.t new file mode 100644 index 000000000..4badfa365 --- /dev/null +++ b/t/04-openresty/ffi/shm/201-metrics_get.t @@ -0,0 +1,137 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm_metrics - get, by name +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local pretty = require "pl.pretty" + local shm = require "resty.wasmx.shm" + + local metrics = { + c1 = shm.metrics:define("c1", shm.metrics.COUNTER), + g1 = shm.metrics:define("g1", shm.metrics.GAUGE), + h1 = shm.metrics:define("h1", shm.metrics.HISTOGRAM), + } + local values = {} + + shm.metrics:increment(metrics.c1) + shm.metrics:record(metrics.g1, 10) + shm.metrics:record(metrics.h1, 100) + + for name, _ in pairs(metrics) do + values[name] = shm.metrics:get_by_name(name) + end + + ngx.say("c1: " .. pretty.write(values.c1, "")) + ngx.say("g1: " .. pretty.write(values.g1, "")) + ngx.say("h1: " .. pretty.write(values.h1, "")) + } + } +--- response_body +c1: {type="counter",value=1} +g1: {type="gauge",value=10} +h1: {type="histogram",value={{count=1,ub=128},{count=0,ub=4294967295}}} +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm_metrics - get, by name (prefix = false) +--- valgrind +--- wasm_modules: hostcalls +--- config + location /t { + proxy_wasm hostcalls 'on_configure=define_metrics \ + on=request_headers \ + n_increments=13 \ + test=/t/metrics/increment_counters \ + metrics=c1'; + + proxy_wasm hostcalls 'on_configure=define_metrics \ + on=request_headers \ + test=/t/metrics/toggle_gauges \ + metrics=g1'; + + proxy_wasm hostcalls 'on_configure=define_metrics \ + on=request_headers \ + test=/t/metrics/record_histograms \ + metrics=h1 \ + value=10'; + + proxy_wasm hostcalls 'on_configure=define_metrics \ + on=request_headers \ + test=/t/metrics/record_histograms \ + metrics=h1 \ + value=100'; + + access_by_lua_block { + local pretty = require "pl.pretty" + local shm = require "resty.wasmx.shm" + + shm.metrics:lock() + + for name in shm.metrics:iterate_keys() do + local m, err = shm.metrics:get_by_name(name, { prefix = false }) + + assert(err == nil) + ngx.say(name .. ": " .. pretty.write(m, "")) + end + + shm.metrics:unlock() + } + } +--- response_body +pw.hostcalls.c1: {type="counter",value=13} +pw.hostcalls.g1: {type="gauge",value=1} +pw.hostcalls.h1: {type="histogram",value={{count=1,ub=16},{count=1,ub=128},{count=0,ub=4294967295}}} +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 3: shm_metrics - get, by name (invalid name) +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local pretty = require "pl.pretty" + local shm = require "resty.wasmx.shm" + + local m, err = shm.metrics:get_by_name("invalid_name") + + assert(m == nil) + assert(err == "metric not found") + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] diff --git a/t/04-openresty/ffi/shm/202-metrics_define.t b/t/04-openresty/ffi/shm/202-metrics_define.t new file mode 100644 index 000000000..e6050f584 --- /dev/null +++ b/t/04-openresty/ffi/shm/202-metrics_define.t @@ -0,0 +1,106 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(9); +run_tests(); + +__DATA__ + +=== TEST 1: shm_metrics - define() +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local metrics = { + c1 = shm.metrics:define("c1", shm.metrics.COUNTER), + g1 = shm.metrics:define("g1", shm.metrics.GAUGE), + h1 = shm.metrics:define("h1", shm.metrics.HISTOGRAM), + } + + for name, id in pairs(metrics) do + assert(type(id) == "number" and id > 0) + end + + ngx.say("ok") + } + } +--- response_body +ok +--- error_log eval +[ + qr/.*? \[info\] .*? defined counter "lua.c1" with id \d+/, + qr/.*? \[info\] .*? defined gauge "lua.g1" with id \d+/, + qr/.*? \[info\] .*? defined histogram "lua.h1" with id \d+/, +] +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm_metrics - define(), name too long +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local name_too_long = string.rep("x", 500) + local mid, err = shm.metrics:define(name_too_long, shm.metrics.HISTOGRAM) + + assert(mid == nil) + assert(err == "failed defining metric, name too long") + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] +[emerg] +[alert] +defined histogram +[stub] +[stub] + + + +=== TEST 3: shm_metrics - define(), invalid metric type +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.metrics:define("c1", 10) + } + } +--- error_code: 500 +--- error_log eval +qr/\[error\] .*? metric_type must be either resty.wasmx.shm.metrics.COUNTER, resty.wasmx.shm.metrics.GAUGE, or resty.wasmx.shm.metrics.HISTOGRAM/ +--- no_error_log +[crit] +[emerg] +[alert] +defined counter +[stub] +[stub] +[stub] diff --git a/t/04-openresty/ffi/shm/203-metrics_record.t b/t/04-openresty/ffi/shm/203-metrics_record.t new file mode 100644 index 000000000..16ee2b331 --- /dev/null +++ b/t/04-openresty/ffi/shm/203-metrics_record.t @@ -0,0 +1,71 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm_metrics - record() +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local pretty = require "pl.pretty" + local shm = require "resty.wasmx.shm" + + local g1 = shm.metrics:define("g1", shm.metrics.GAUGE) + local h1 = shm.metrics:define("h1", shm.metrics.HISTOGRAM) + + shm.metrics:record(g1, 10) + shm.metrics:record(h1, 100) + + ngx.say("g1: " .. pretty.write(shm.metrics:get(g1), "")) + ngx.say("h1: " .. pretty.write(shm.metrics:get(h1), "")) + } + } +--- response_body +g1: {type="gauge",value=10} +h1: {type="histogram",value={{count=1,ub=128},{count=0,ub=4294967295}}} +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm_metrics - record(), inexistent metric +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local ok, err = shm.metrics:record(1, 10) + + if not ok then + ngx.say(err) + return + end + + ngx.say("fail") + } + } +--- response_body +metric not found +--- no_error_log +[error] +[crit] +[emerg] +[alert] diff --git a/t/04-openresty/ffi/shm/204-metrics_increment.t b/t/04-openresty/ffi/shm/204-metrics_increment.t new file mode 100644 index 000000000..fdd413a9b --- /dev/null +++ b/t/04-openresty/ffi/shm/204-metrics_increment.t @@ -0,0 +1,68 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +plan_tests(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm_metrics - increment() +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local pretty = require "pl.pretty" + local shm = require "resty.wasmx.shm" + + local c1 = shm.metrics:define("c1", shm.metrics.COUNTER) + + shm.metrics:increment(c1) + shm.metrics:increment(c1, 10) + + ngx.say("c1: " .. pretty.write(shm.metrics:get(c1), "")) + } + } +--- response_body +c1: {type="counter",value=11} +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 2: shm_metrics - increment(), inexistent metric +--- valgrind +--- main_config + wasm {} +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local ok, err = shm.metrics:increment(1, 10) + + if not ok then + ngx.say(err) + return + end + + ngx.say("fail") + } + } +--- response_body +metric not found +--- no_error_log +[error] +[crit] +[emerg] +[alert]