From 982e5b5561a46d0fa5f85d2f979cc41c359fad73 Mon Sep 17 00:00:00 2001 From: Thibault Charbonnier Date: Fri, 13 Sep 2024 09:25:50 -0700 Subject: [PATCH] wip(*) merge edits --- lib/resty/wasmx/shm.lua | 452 ++++++++++-------- src/common/debug/ngx_wasm_debug_module.c | 2 +- src/common/lua/ngx_wasm_lua_ffi.c | 142 +++--- src/common/lua/ngx_wasm_lua_ffi.h | 41 +- src/common/metrics/ngx_wa_histogram.c | 2 +- src/common/metrics/ngx_wa_metrics.c | 8 +- src/common/metrics/ngx_wa_metrics.h | 9 +- src/common/proxy_wasm/ngx_proxy_wasm_host.c | 12 +- src/ngx_wasmx.c | 1 + t/04-openresty/ffi/shm/000-setup_zones.t | 137 ++++++ t/04-openresty/ffi/shm/001-kv_get.t | 90 ++++ t/04-openresty/ffi/shm/001-setup_zones.t | 74 --- t/04-openresty/ffi/shm/002-iterate_keys.t | 182 ------- t/04-openresty/ffi/shm/002-kv_set.t | 103 ++++ t/04-openresty/ffi/shm/003-get_keys.t | 143 ------ t/04-openresty/ffi/shm/010-iterate_keys.t | 192 ++++++++ t/04-openresty/ffi/shm/011-get_keys.t | 114 +++++ t/04-openresty/ffi/shm/020-metrics_define.t | 128 +++++ .../ffi/shm/021-metrics_increment.t | 83 ++++ t/04-openresty/ffi/shm/022-metrics_record.t | 82 ++++ t/04-openresty/ffi/shm/023-metrics_get.t | 82 ++++ .../ffi/shm/024-metrics_get_by_name.t | 136 ++++++ t/04-openresty/ffi/shm/101-kv_get.t | 90 ---- t/04-openresty/ffi/shm/102-kv_set.t | 102 ---- t/04-openresty/ffi/shm/201-metrics_get.t | 137 ------ t/04-openresty/ffi/shm/202-metrics_define.t | 106 ---- t/04-openresty/ffi/shm/203-metrics_record.t | 71 --- .../ffi/shm/204-metrics_increment.t | 68 --- .../hostcalls/src/tests/mod.rs | 47 +- .../hostcalls/src/types/test_http.rs | 4 +- util/_lib.sh | 1 + 31 files changed, 1559 insertions(+), 1282 deletions(-) create mode 100644 t/04-openresty/ffi/shm/000-setup_zones.t create mode 100644 t/04-openresty/ffi/shm/001-kv_get.t delete mode 100644 t/04-openresty/ffi/shm/001-setup_zones.t delete mode 100644 t/04-openresty/ffi/shm/002-iterate_keys.t create mode 100644 t/04-openresty/ffi/shm/002-kv_set.t delete mode 100644 t/04-openresty/ffi/shm/003-get_keys.t create mode 100644 t/04-openresty/ffi/shm/010-iterate_keys.t create mode 100644 t/04-openresty/ffi/shm/011-get_keys.t create mode 100644 t/04-openresty/ffi/shm/020-metrics_define.t create mode 100644 t/04-openresty/ffi/shm/021-metrics_increment.t create mode 100644 t/04-openresty/ffi/shm/022-metrics_record.t create mode 100644 t/04-openresty/ffi/shm/023-metrics_get.t create mode 100644 t/04-openresty/ffi/shm/024-metrics_get_by_name.t delete mode 100644 t/04-openresty/ffi/shm/101-kv_get.t delete mode 100644 t/04-openresty/ffi/shm/102-kv_set.t delete mode 100644 t/04-openresty/ffi/shm/201-metrics_get.t delete mode 100644 t/04-openresty/ffi/shm/202-metrics_define.t delete mode 100644 t/04-openresty/ffi/shm/203-metrics_record.t delete mode 100644 t/04-openresty/ffi/shm/204-metrics_increment.t diff --git a/lib/resty/wasmx/shm.lua b/lib/resty/wasmx/shm.lua index 32fe5c070..6e4ebc865 100644 --- a/lib/resty/wasmx/shm.lua +++ b/lib/resty/wasmx/shm.lua @@ -5,22 +5,26 @@ local wasmx = require "resty.wasmx" local C = ffi.C +local assert = assert local error = error local type = type -local table_new = table.new +local tonumber = tonumber +local min = math.min +local new_tab = table.new +local insert = table.insert 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 ngx_log = ngx.log +local DEBUG = ngx.DEBUG +local FFI_OK = wasmx.FFI_OK +local FFI_ERROR = wasmx.FFI_ERROR +local FFI_DONE = wasmx.FFI_DONE 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 [[ @@ -85,42 +89,44 @@ ffi.cdef [[ 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_uint_t *clast_index, + ngx_uint_t *cur_idx, + ngx_str_t **keys); - ngx_int_t ngx_wa_ffi_shm_get_kv_value(ngx_wa_shm_t *shm, + ngx_uint_t ngx_wa_ffi_shm_kv_nelts(ngx_wa_shm_t *shm); + ngx_int_t ngx_wa_ffi_shm_kv_get_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_int_t ngx_wa_ffi_shm_kv_set_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_int_t ngx_wa_ffi_shm_metric_define(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_int_t ngx_wa_ffi_shm_metric_increment(uint32_t metric_id, ngx_uint_t value); - ngx_int_t ngx_wa_ffi_shm_get_metric(uint32_t metric_id, + ngx_int_t ngx_wa_ffi_shm_metric_record(uint32_t metric_id, + ngx_uint_t value); + ngx_int_t ngx_wa_ffi_shm_metric_get(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(); + 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_metrics_one_slot_size(); + ngx_int_t ngx_wa_ffi_shm_metrics_histogram_max_size(); ]] local WASM_SHM_KEY = {} -local DEFAULT_KEY_BATCH_SIZE = 500 +local DEFAULT_KEYS_PAGE_SIZE = 500 local _M = setmetatable({}, { @@ -149,144 +155,163 @@ local _metric_type_set = { [_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 _mbs = C.ngx_wa_ffi_shm_metrics_one_slot_size() +local _hbs = C.ngx_wa_ffi_shm_metrics_histogram_max_size() +local _mbuf = ffi_new("u_char[?]", _mbs) +local _hbuf = ffi_new("u_char[?]", _hbs) +local _kbuf = ffi_new("ngx_str_t *[?]", DEFAULT_KEYS_PAGE_SIZE) -local function lock_shm(zone) +local function shm_lock(zone) C.ngx_wa_ffi_shm_lock(zone[WASM_SHM_KEY]) end -local function unlock_shm(zone) +local function shm_unlock(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 + if ctx.i == tonumber(ctx.ccur_index[0]) then + ngx_sleep(0) -- TODO: support non-yielding phases - return ffi_str(ckey.data, ckey.len) + ctx.i = 0 + ctx.ccur_index[0] = 0 - else - ctx.cpage_total[0] = 0 + local rc = C.ngx_wa_ffi_shm_iterate_keys(ctx.shm, ctx.page_size, + ctx.clast_index, ctx.ccur_index, ctx.ckeys) - 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 + -- users must manage locking themselves (e.g. break condition in the for loop) 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" + local err = "attempt to call %s:iterate_keys() but the " .. + "shm zone is not locked; invoke :lock() before " .. + "and :unlock() after." - error(str_fmt(err, zone_name, zone_name), 2) + error(str_fmt(err, zone_name), 2) end - if rc == FFI_DECLINED then + if rc == FFI_DONE then return end - ngx_sleep(0) - - ctx.page_total = tonumber(ctx.cpage_total[0]) - ctx.page = ctx.page + ctx.page_total - ctx.i = 1 + if rc ~= FFI_OK then + error("unknown error") + end - return ffi_str(ctx.ckeys[0].data, ctx.ckeys[0].len) + ngx_log(DEBUG, "iterate_keys fetched a new page") end + + local ckey = ctx.ckeys[ctx.i] + local key = ffi_str(ctx.ckeys[ctx.i].data, ctx.ckeys[ctx.i].len) + + ctx.i = ctx.i + 1 + + return key 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) +local function shm_iterate_keys(zone, opts) + if opts ~= nil then + if type(opts) ~= "table" then + error("opts must be a table", 2) + end + + if opts.page_size ~= nil then + if type(opts.page_size) ~= "number" then + error("opts.page_size must be a number", 2) + end + + if opts.page_size < 1 then + error("opts.page_size must be > 0", 2) + end + end end local ctx = { + zone = zone, shm = zone[WASM_SHM_KEY], + page_size = opts and opts.page_size + and opts.page_size + or DEFAULT_KEYS_PAGE_SIZE, 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 + clast_index = ffi_new("ngx_uint_t[1]"), + ccur_index = 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 function shm_get_keys(zone, max_count) + if max_count == nil then + max_count = DEFAULT_KEYS_PAGE_SIZE - local shm = zone[WASM_SHM_KEY] - local nkeys - local keys - local start = 0 - local ctotal = ffi_new("ngx_uint_t [1]") + elseif type(max_count) ~= "number" then + error("max_count must be a number", 2) - lock_shm(zone) + elseif max_count < 0 then + error("max_count must be >= 0", 2) + end - if max == 0 then - local rc = C.ngx_wa_ffi_shm_iterate_keys(shm, start, 0, nil, ctotal) - nkeys = tonumber(ctotal[0]) + shm_lock(zone) - if rc == FFI_DECLINED then - unlock_shm(zone) - return {} - end + local shm = zone[WASM_SHM_KEY] + local nkeys = tonumber(C.ngx_wa_ffi_shm_kv_nelts(shm)) + if nkeys == 0 then + shm_unlock(zone) + return {} + end - else - nkeys = max and max or DEFAULT_KEY_BATCH_SIZE + if max_count > 0 then + nkeys = min(nkeys, max_count) end + local ctotal = ffi_new("ngx_uint_t[1]") local ckeys = ffi_new("ngx_str_t *[?]", nkeys) - C.ngx_wa_ffi_shm_iterate_keys(shm, start, nkeys, ckeys, ctotal) + C.ngx_wa_ffi_shm_iterate_keys(shm, nkeys, nil, ctotal, ckeys) - unlock_shm(zone) + shm_unlock(zone) local total = tonumber(ctotal[0]) - local keys = table_new(0, total) + local keys = new_tab(0, total) + assert(total == nkeys) for i = 1, total do - local ckey = ckeys[i - 1] - keys[i] = ffi_str(ckey.data, ckey.len) + keys[i] = ffi_str(ckeys[i - 1].data, ckeys[i - 1].len) end return keys end -local function get_kv_value(zone, key) +local function shm_kv_get(zone, key) if type(key) ~= "string" then - error("key must be string", 2) + error("key must be a 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 ccas = ffi_new("uint32_t[1]") - local rc = C.ngx_wa_ffi_shm_get_kv_value(shm, cname, cvalue, ccas) + local rc = C.ngx_wa_ffi_shm_kv_get_value(shm, cname, cvalue, ccas) if rc == FFI_DECLINED then - return nil, nil, "key not found" + return nil + end + + if rc ~= FFI_OK then + return nil, "unknown error" end return ffi_str(cvalue[0].data, cvalue[0].len), tonumber(ccas[0]) end -local function set_kv_value(zone, key, value, cas) +local function shm_kv_set(zone, key, value, cas) if type(key) ~= "string" then error("key must be a string", 2) end @@ -295,52 +320,154 @@ local function set_kv_value(zone, key, value, cas) error("value must be a string", 2) end - if type(cas) ~= "number" then + if cas == nil then + cas = 0 + + elseif 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 written = ffi_new("ngx_int_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" + local rc = C.ngx_wa_ffi_shm_kv_set_value(shm, cname, cvalue, cas, written) + if rc == FFI_ERROR then + return nil, "no memory" + end + + if rc ~= FFI_OK then + return nil, "unknown error" end return tonumber(written[0]) end +local function metrics_define(zone, name, metric_type) + if type(name) ~= "string" or name == "" then + error("name must be a non-empty string", 2) + end + + if not _metric_type_set[metric_type] 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 + + name = "lua." .. name + + 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_metric_define(cname, metric_type, m_id) + if rc == FFI_ERROR then + return nil, "no memory" + end + + if rc == FFI_ABORT then + return nil, "name too long" + end + + if rc ~= FFI_OK then + -- FFI_BUSY: unreachable + return nil, "unknown error" + end + + return tonumber(m_id[0]) +end + + +local function metrics_increment(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 > 0", 2) + end + + value = value and value or 1 + + local rc = C.ngx_wa_ffi_shm_metric_increment(metric_id, value) + if rc == FFI_DECLINED then + return nil, "metric not found" + end + + if rc ~= FFI_OK then + return nil, "unknown error" + end + + return true +end + + +local function metrics_record(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_metric_record(metric_id, value) + if rc == FFI_DECLINED then + return nil, "metric not found" + end + + if rc ~= FFI_OK then + return nil, "unknown error" + end + + return true +end + + local function parse_cmetric(cmetric) if cmetric.metric_type == _types.ffi_metric.COUNTER then - return { type = "counter", value = tonumber(cmetric.slots[0].counter) } + return { + type = "counter", + value = tonumber(cmetric.slots[0].counter), + } + end - elseif cmetric.metric_type == _types.ffi_metric.GAUGE then - return { type = "gauge", value = tonumber(cmetric.slots[0].gauge.value) } + if cmetric.metric_type == _types.ffi_metric.GAUGE then + return { + type = "gauge", + value = tonumber(cmetric.slots[0].gauge.value), + } + end - elseif cmetric.metric_type == _types.ffi_metric.HISTOGRAM then + if 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 } + insert(h.value, { + ub = cb.upper_bound, + count = cb.count, + }) end return h end + + assert(false, "unreachable") end -local function get_metric(zone, metric_id) +local function metrics_get_by_id(zone, metric_id) if type(metric_id) ~= "number" then error("metric_id must be a number", 2) end @@ -348,10 +475,15 @@ local function get_metric(zone, metric_id) ffi_fill(_mbuf, _mbs) ffi_fill(_hbuf, _hbs) - local rc = C.ngx_wa_ffi_shm_get_metric(metric_id, nil, - _mbuf, _mbs, _hbuf, _hbs) + local rc = C.ngx_wa_ffi_shm_metric_get(metric_id, nil, + _mbuf, _mbs, + _hbuf, _hbs) if rc == FFI_DECLINED then - return nil, "metric not found" + return nil + end + + if rc ~= FFI_OK then + return nil, "unknown error" end return parse_cmetric(ffi_cast("ngx_wa_metric_t *", _mbuf)) @@ -359,14 +491,15 @@ 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. +-- ngx_wasm_module internally prefixes metric names according to +-- where they have been defined, e.g. "pw.filter.*", "lua.*", or +-- "wa.*". -- --- get_metric_by_name assumes that it's retrieving a lua land metric and --- will by default prefix name with `lua.` +-- metrics_get_by_name assumes it is retrieving a Lua-defined metric +-- and will by default prefix the given name with `lua.` -- -- This behavior can be disabled by passing `opts.prefix` as false. -local function get_metric_by_name(zone, name, opts) +local function metrics_get_by_name(zone, name, opts) if type(name) ~= "string" or name == "" then error("name must be a non-empty string", 2) end @@ -381,86 +514,23 @@ local function get_metric_by_name(zone, name, opts) 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 + name = (opts and opts.prefix == false) and name or "lua." .. name 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) + local rc = C.ngx_wa_ffi_shm_metric_get(0, cname, _mbuf, _mbs, _hbuf, _hbs) if rc == FFI_DECLINED then - return nil, "metric not found" + return nil 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) + if rc ~= FFI_OK then + return nil, "unknown error" 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 + return parse_cmetric(ffi_cast("ngx_wa_metric_t *", _mbuf)) end @@ -469,27 +539,29 @@ 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, + lock = shm_lock, + unlock = shm_unlock, } 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 + _M[zone_name].iterate_keys = shm_iterate_keys + _M[zone_name].get_keys = shm_get_keys + _M[zone_name].get = shm_kv_get + _M[zone_name].set = shm_kv_set 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].get_keys = shm_get_keys + _M[zone_name].iterate_keys = shm_iterate_keys + + _M[zone_name].define = metrics_define + _M[zone_name].increment = metrics_increment + _M[zone_name].record = metrics_record + _M[zone_name].get = metrics_get_by_id + _M[zone_name].get_by_name = metrics_get_by_name + _M[zone_name].COUNTER = _types.ffi_metric.COUNTER _M[zone_name].GAUGE = _types.ffi_metric.GAUGE _M[zone_name].HISTOGRAM = _types.ffi_metric.HISTOGRAM @@ -498,7 +570,7 @@ 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") + ngx_log(DEBUG, "no shm zones found for resty.wasmx.shm interface") end diff --git a/src/common/debug/ngx_wasm_debug_module.c b/src/common/debug/ngx_wasm_debug_module.c index a73e50bdb..4c9043479 100644 --- a/src/common/debug/ngx_wasm_debug_module.c +++ b/src/common/debug/ngx_wasm_debug_module.c @@ -63,7 +63,7 @@ ngx_wasm_debug_init(ngx_cycle_t *cycle) ngx_wa_metrics_define(ngx_wasmx_metrics(cycle), &metric_name, 100, - &mid) == NGX_ABORT + &mid) == NGX_BUSY ); /* unknown metric type name */ diff --git a/src/common/lua/ngx_wasm_lua_ffi.c b/src/common/lua/ngx_wasm_lua_ffi.c index 5bf2936f8..8540e855a 100644 --- a/src/common/lua/ngx_wasm_lua_ffi.c +++ b/src/common/lua/ngx_wasm_lua_ffi.c @@ -5,7 +5,9 @@ #include #include +#ifdef NGX_WASM_HTTP #include +#endif ngx_wavm_t * @@ -260,24 +262,6 @@ ngx_http_wasm_ffi_set_host_properties_handlers(ngx_http_request_t *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) { @@ -304,40 +288,41 @@ ngx_wa_ffi_shm_setup_zones(ngx_wa_ffi_shm_setup_zones_handler_pt setup_handler) 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) +retrieve_keys(ngx_rbtree_node_t *node, ngx_rbtree_node_t *sentinel, + ngx_uint_t limit, ngx_uint_t start_idx, ngx_uint_t *cur_idx, + ngx_str_t **keys) { - ngx_wa_shm_kv_node_t *n = (ngx_wa_shm_kv_node_t *) node; + ngx_wa_shm_kv_node_t *n = (ngx_wa_shm_kv_node_t *) node; - if (limit > 0 && (*total - start) == limit) { + if ((*cur_idx - start_idx) == limit) { return; } - if (keys && *total >= start) { - keys[(*total - start)] = &n->key.str; + if (*cur_idx >= start_idx) { + /* append to result */ + keys[(*cur_idx - start_idx)] = &n->key.str; } - (*total)++; + (*cur_idx)++; if (node->left != sentinel) { - shm_kv_retrieve_keys(node->left, sentinel, start, limit, keys, total); + retrieve_keys(node->left, sentinel, limit, start_idx, cur_idx, keys); } if (node->right != sentinel) { - shm_kv_retrieve_keys(node->right, sentinel, start, limit, keys, total); + retrieve_keys(node->right, sentinel, limit, start_idx, cur_idx, keys); } } 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_ffi_shm_iterate_keys(ngx_wa_shm_t *shm, ngx_uint_t page_size, + ngx_uint_t *clast_idx, ngx_uint_t *cur_idx, ngx_str_t **keys) { + ngx_uint_t last_idx = clast_idx ? *clast_idx : 0; ngx_wa_shm_kv_t *kv; - ngx_wa_assert(shm && shm->type != NGX_WA_SHM_TYPE_QUEUE); + ngx_wa_assert(shm->type != NGX_WA_SHM_TYPE_QUEUE); if (!ngx_wa_shm_locked(shm)) { return NGX_ABORT; @@ -345,85 +330,84 @@ ngx_wa_ffi_shm_iterate_keys(ngx_wa_shm_t *shm, kv = shm->data; - if (page >= kv->nelts) { - return NGX_DECLINED; + if (last_idx >= kv->nelts) { + /* finished */ + return NGX_DONE; } - if (kv->rbtree.root != kv->rbtree.sentinel) { - shm_kv_retrieve_keys(kv->rbtree.root, kv->rbtree.sentinel, - page, page_size, keys, total); - } + /* cannot be empty here (kv->nelts > 0) */ + ngx_wa_assert(kv->rbtree.root != kv->rbtree.sentinel); - *total = *total - page; + retrieve_keys(kv->rbtree.root, kv->rbtree.sentinel, page_size, last_idx, + cur_idx, keys); + + *cur_idx -= last_idx; + + if (clast_idx) { + *clast_idx += *cur_idx; + } return NGX_OK; } +ngx_uint_t +ngx_wa_ffi_shm_kv_nelts(ngx_wa_shm_t *shm) +{ + ngx_wa_shm_kv_t *kv; + + ngx_wa_assert(shm->type != NGX_WA_SHM_TYPE_QUEUE); + + kv = shm->data; + + return kv->nelts; +} + + 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_wa_ffi_shm_kv_get_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); + ngx_wa_assert(shm->type == NGX_WA_SHM_TYPE_KV); - if (!ngx_wa_shm_locked(shm)) { - ngx_wa_shm_lock(shm); - unlock = 1; - } + ngx_wa_shm_lock(shm); rc = ngx_wa_shm_kv_get_locked(shm, k, NULL, v, cas); - if (unlock) { - ngx_wa_shm_unlock(shm); - } + 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) +ngx_wa_ffi_shm_kv_set_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); + ngx_wa_assert(shm->type == NGX_WA_SHM_TYPE_KV); - if (!ngx_wa_shm_locked(shm)) { - ngx_wa_shm_lock(shm); - unlock = 1; - } + ngx_wa_shm_lock(shm); rc = ngx_wa_shm_kv_set_locked(shm, k, v, cas, written); - if (unlock) { - ngx_wa_shm_unlock(shm); - } + 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_wa_ffi_shm_metric_define(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); + rc = ngx_wa_metrics_define(metrics, name, type, metric_id); if (rc != NGX_OK) { return rc; } @@ -433,36 +417,30 @@ ngx_wa_ffi_shm_define_metric(ngx_str_t *name, ngx_wa_metric_type_e type, ngx_int_t -ngx_wa_ffi_shm_increment_metric(uint32_t metric_id, ngx_uint_t value) +ngx_wa_ffi_shm_metric_increment(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_ffi_shm_metric_record(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_ffi_shm_metric_get(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); diff --git a/src/common/lua/ngx_wasm_lua_ffi.h b/src/common/lua/ngx_wasm_lua_ffi.h index 50bcab114..33f728e7a 100644 --- a/src/common/lua/ngx_wasm_lua_ffi.h +++ b/src/common/lua/ngx_wasm_lua_ffi.h @@ -48,35 +48,48 @@ ngx_int_t ngx_http_wasm_ffi_set_host_properties_handlers(ngx_http_request_t *r, 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_iterate_keys(ngx_wa_shm_t *shm, ngx_uint_t page_size, + ngx_uint_t *clast_idx, ngx_uint_t *cur_idx, ngx_str_t **keys); -ngx_int_t ngx_wa_ffi_shm_get_kv_value(ngx_wa_shm_t *shm, ngx_str_t *k, +ngx_uint_t ngx_wa_ffi_shm_kv_nelts(ngx_wa_shm_t *shm); +ngx_int_t ngx_wa_ffi_shm_kv_get_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_int_t ngx_wa_ffi_shm_kv_set_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_int_t ngx_wa_ffi_shm_metric_define(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, +ngx_int_t ngx_wa_ffi_shm_metric_increment(uint32_t metric_id, ngx_uint_t value); +ngx_int_t ngx_wa_ffi_shm_metric_record(uint32_t metric_id, ngx_uint_t value); +ngx_int_t ngx_wa_ffi_shm_metric_get(uint32_t metric_id, ngx_str_t *name, u_char *m_buf, size_t mbs, u_char *h_buf, size_t hbs); +void +ngx_wa_ffi_shm_lock(ngx_wa_shm_t *shm) +{ + ngx_wa_shm_lock(shm); +} + + +void +ngx_wa_ffi_shm_unlock(ngx_wa_shm_t *shm) +{ + ngx_wa_shm_unlock(shm); +} + + ngx_int_t -ngx_wa_ffi_shm_one_slot_metric_size() +ngx_wa_ffi_shm_metrics_one_slot_size() { - return NGX_WA_METRICS_ONE_SLOT_METRIC_SIZE; + return NGX_WA_METRICS_ONE_SLOT_SIZE; } ngx_int_t -ngx_wa_ffi_shm_max_histogram_size() +ngx_wa_ffi_shm_metrics_histogram_max_size() { - return NGX_WA_METRICS_MAX_HISTOGRAM_SIZE; + return NGX_WA_METRICS_HISTOGRAM_MAX_SIZE; } diff --git a/src/common/metrics/ngx_wa_histogram.c b/src/common/metrics/ngx_wa_histogram.c index d52e46cef..4aee8ddfa 100644 --- a/src/common/metrics/ngx_wa_histogram.c +++ b/src/common/metrics/ngx_wa_histogram.c @@ -129,7 +129,7 @@ histogram_log(ngx_wa_metrics_t *metrics, ngx_wa_metric_t *m, uint32_t mid) u_char *p; ngx_wa_metrics_bin_t *b; ngx_wa_metrics_histogram_t *h; - u_char h_buf[NGX_WA_METRICS_MAX_HISTOGRAM_SIZE]; + u_char h_buf[NGX_WA_METRICS_HISTOGRAM_MAX_SIZE]; u_char s_buf[NGX_MAX_ERROR_STR]; ngx_memzero(h_buf, sizeof(h_buf)); diff --git a/src/common/metrics/ngx_wa_metrics.c b/src/common/metrics/ngx_wa_metrics.c index e0579e693..ed4c53cab 100644 --- a/src/common/metrics/ngx_wa_metrics.c +++ b/src/common/metrics/ngx_wa_metrics.c @@ -289,7 +289,7 @@ ngx_wa_metrics_define(ngx_wa_metrics_t *metrics, ngx_str_t *name, && type != NGX_WA_METRIC_GAUGE && type != NGX_WA_METRIC_HISTOGRAM) { - return NGX_ABORT; + return NGX_BUSY; } if (name->len > metrics->config.max_metric_name_length) { @@ -337,9 +337,9 @@ ngx_wa_metrics_define(ngx_wa_metrics_t *metrics, ngx_str_t *name, ngx_wa_shm_unlock(metrics->shm); if (rc == NGX_OK) { - ngx_wasm_log_error(NGX_LOG_INFO, metrics->shm->log, 0, - "defined %V \"%V\" with id %uD", - ngx_wa_metric_type_name(type), name, mid); + ngx_log_debug3(NGX_LOG_DEBUG_WASM, metrics->shm->log, 0, + "defined %V \"%V\" with id %uD", + ngx_wa_metric_type_name(type), name, mid); } ngx_wa_assert(rc == NGX_OK || rc == NGX_ERROR); diff --git a/src/common/metrics/ngx_wa_metrics.h b/src/common/metrics/ngx_wa_metrics.h index 00942b45c..49ad6de25 100644 --- a/src/common/metrics/ngx_wa_metrics.h +++ b/src/common/metrics/ngx_wa_metrics.h @@ -12,11 +12,12 @@ #define NGX_WA_METRICS_BINS_INIT 5 #define NGX_WA_METRICS_BINS_MAX 18 #define NGX_WA_METRICS_BINS_INCREMENT 4 -#define NGX_WA_METRICS_MAX_HISTOGRAM_SIZE sizeof(ngx_wa_metrics_histogram_t)\ - + sizeof(ngx_wa_metrics_bin_t) \ - * NGX_WA_METRICS_BINS_MAX -#define NGX_WA_METRICS_ONE_SLOT_METRIC_SIZE sizeof(ngx_wa_metric_t) \ +#define NGX_WA_METRICS_ONE_SLOT_SIZE sizeof(ngx_wa_metric_t) \ + sizeof(ngx_wa_metric_val_t) +#define NGX_WA_METRICS_HISTOGRAM_MAX_SIZE \ + sizeof(ngx_wa_metrics_histogram_t) \ + + sizeof(ngx_wa_metrics_bin_t) \ + * NGX_WA_METRICS_BINS_MAX typedef struct ngx_wa_metrics_s ngx_wa_metrics_t; diff --git a/src/common/proxy_wasm/ngx_proxy_wasm_host.c b/src/common/proxy_wasm/ngx_proxy_wasm_host.c index cf0f6c3f7..ba5242104 100644 --- a/src/common/proxy_wasm/ngx_proxy_wasm_host.c +++ b/src/common/proxy_wasm/ngx_proxy_wasm_host.c @@ -1601,7 +1601,7 @@ ngx_proxy_wasm_hfuncs_define_metric(ngx_wavm_instance_t *instance, ngx_memzero(trapmsg, NGX_MAX_ERROR_STR); - switch(pw_type) { + switch (pw_type) { case NGX_PROXY_WASM_METRIC_COUNTER: type = NGX_WA_METRIC_COUNTER; break; @@ -1648,8 +1648,10 @@ ngx_proxy_wasm_hfuncs_define_metric(ngx_wavm_instance_t *instance, rets, NGX_WAVM_ERROR); default: - ngx_wa_assert(rc == NGX_OK); - break; + if (rc != NGX_OK) { + return ngx_proxy_wasm_result_trap(pwexec, "unknown error", + rets, NGX_WAVM_ERROR); + } } return ngx_proxy_wasm_result_ok(rets); @@ -1668,8 +1670,8 @@ ngx_proxy_wasm_hfuncs_get_metric(ngx_wavm_instance_t *instance, ngx_wa_metrics_t *metrics = ngx_wasmx_metrics(cycle); ngx_wa_metric_t *m; ngx_proxy_wasm_exec_t *pwexec = ngx_proxy_wasm_instance2pwexec(instance); - u_char m_buf[NGX_WA_METRICS_ONE_SLOT_METRIC_SIZE]; - u_char h_buf[NGX_WA_METRICS_MAX_HISTOGRAM_SIZE]; + u_char m_buf[NGX_WA_METRICS_ONE_SLOT_SIZE]; + u_char h_buf[NGX_WA_METRICS_HISTOGRAM_MAX_SIZE]; u_char trapmsg[NGX_MAX_ERROR_STR]; ngx_memzero(m_buf, sizeof(m_buf)); diff --git a/src/ngx_wasmx.c b/src/ngx_wasmx.c index 0260d4592..b4470c517 100644 --- a/src/ngx_wasmx.c +++ b/src/ngx_wasmx.c @@ -347,6 +347,7 @@ ngx_wasmx_metrics(ngx_cycle_t *cycle) ngx_wa_conf_t *wacf; if (cycle->conf_ctx == NULL) { + /* cycle->old_cycle */ return NULL; } diff --git a/t/04-openresty/ffi/shm/000-setup_zones.t b/t/04-openresty/ffi/shm/000-setup_zones.t new file mode 100644 index 000000000..a263afe1f --- /dev/null +++ b/t/04-openresty/ffi/shm/000-setup_zones.t @@ -0,0 +1,137 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +add_block_preprocessor(sub { + my $block = shift; + if (!defined $block->main_config) { + $block->set_value("main_config", <<_EOC_ + wasm { + shm_kv kv 16k; + shm_queue q 16k; + } +_EOC_ + ); + } +}); + +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 - setup_zones() 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] + + + +=== TEST 4: shm - calling functions on bad shm types +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(function() + shm.q:get() + end) + ngx.say(perr) + + _, perr = pcall(function() + shm.q:set() + end) + ngx.say(perr) + + _, perr = pcall(function() + shm.q:iterate_keys() + end) + ngx.say(perr) + + _, perr = pcall(function() + shm.q:get_keys() + end) + ngx.say(perr) + + _, perr = pcall(function() + shm.metrics:set() + end) + ngx.say(perr) + + _, perr = pcall(function() + shm.kv:define() + end) + ngx.say(perr) + } + } +--- response_body_like +.*? attempt to call method 'get' \(a nil value\) +.*? attempt to call method 'set' \(a nil value\) +.*? attempt to call method 'iterate_keys' \(a nil value\) +.*? attempt to call method 'get_keys' \(a nil value\) +.*? attempt to call method 'set' \(a nil value\) +.*? attempt to call method 'define' \(a nil value\) +--- no_error_log +[crit] +[emerg] diff --git a/t/04-openresty/ffi/shm/001-kv_get.t b/t/04-openresty/ffi/shm/001-kv_get.t new file mode 100644 index 000000000..c16067581 --- /dev/null +++ b/t/04-openresty/ffi/shm/001-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::Lua; + +skip_no_openresty(); + +plan_tests(4); +run_tests(); + +__DATA__ + +=== TEST 1: shm_kv - get() sanity +--- valgrind +--- wasm_modules: hostcalls +--- shm_kv: kv 16k +--- config + location /t { + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv/k1 \ + value=v1'; + + proxy_wasm hostcalls 'test=/t/shm/set_shared_data \ + key=kv/k2 \ + value=v2'; + + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local expected = { + ["kv/k1"] = "v1", + ["kv/k2"] = "v2", + } + + local v, cas = shm.kv:get("kv/k1") + assert(cas == 1) + ngx.say(v) + + v, cas = shm.kv:get("kv/k2") + assert(cas == 1) + ngx.say(v) + } + } +--- response_body +v1 +v2 +--- no_error_log +[error] +[crit] + + + +=== TEST 2: shm_kv - get() metric not found +--- valgrind +--- shm_kv: kv 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local v, cas = shm.kv:get("none") + assert(cas == nil) + ngx.say(v) + } + } +--- response_body +nil +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm_kv - get() bad args +--- shm_kv: kv 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.kv.get, {}, false) + ngx.say(perr) + } + } +--- response_body +key must be a string +--- no_error_log +[crit] +[emerg] diff --git a/t/04-openresty/ffi/shm/001-setup_zones.t b/t/04-openresty/ffi/shm/001-setup_zones.t deleted file mode 100644 index 5e881cf91..000000000 --- a/t/04-openresty/ffi/shm/001-setup_zones.t +++ /dev/null @@ -1,74 +0,0 @@ -# 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 deleted file mode 100644 index a1932914a..000000000 --- a/t/04-openresty/ffi/shm/002-iterate_keys.t +++ /dev/null @@ -1,182 +0,0 @@ -# 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/002-kv_set.t b/t/04-openresty/ffi/shm/002-kv_set.t new file mode 100644 index 000000000..e7f50cf49 --- /dev/null +++ b/t/04-openresty/ffi/shm/002-kv_set.t @@ -0,0 +1,103 @@ +# 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_kv - set() sanity +--- valgrind +--- wasm_modules: hostcalls +--- shm_kv: kv 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local written, err = shm.kv:set("kv/k1", "v1", 0) + assert(err == nil) + assert(written == 1) + + written, err = shm.kv:set("kv/k2", "v2", 0) + assert(err == nil) + assert(written == 1) + + ngx.say("ok") + } + + proxy_wasm hostcalls 'test=/t/shm/log_shared_data \ + on=log \ + key=kv/k1'; + + proxy_wasm hostcalls 'test=/t/shm/log_shared_data \ + on=log \ + key=kv/k2'; + } +--- ignore_response_body +--- error_log +kv/k1: "v1" 1 +kv/k2: "v2" 1 +--- no_error_log +[error] + + + +=== TEST 2: shm_kv - set() no memory +--- shm_kv: kv 16k eviction=none +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + for i = 1, 5 do + local _, err = shm.kv:set(tostring(i), string.rep(".", 2^10)) + if err then + ngx.log(ngx.ERR, err) + break + end + end + + ngx.say("ok") + } + } +--- ignore_response_body +--- error_log eval +[ + qr/\[crit\] .*? "kv" shm store: no memory; cannot allocate pair with key size 1 and value size 1024/, + qr/\[error\] .*? access_by_lua\(.*?\):\d+: no memory/, +] +--- no_error_log +[emerg] + + + +=== TEST 3: shm_kv - set() bad args +--- shm_kv: kv 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.kv.set, {}, false) + ngx.say(perr) + + _, perr = pcall(shm.kv.set, {}, "k1", false) + ngx.say(perr) + + _, perr = pcall(shm.kv.set, {}, "k1", "v1", false) + ngx.say(perr) + } + } +--- response_body +key must be a string +value must be a string +cas must be a number +--- no_error_log +[crit] +[emerg] diff --git a/t/04-openresty/ffi/shm/003-get_keys.t b/t/04-openresty/ffi/shm/003-get_keys.t deleted file mode 100644 index b5a16684f..000000000 --- a/t/04-openresty/ffi/shm/003-get_keys.t +++ /dev/null @@ -1,143 +0,0 @@ -# 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/010-iterate_keys.t b/t/04-openresty/ffi/shm/010-iterate_keys.t new file mode 100644 index 000000000..528e25d38 --- /dev/null +++ b/t/04-openresty/ffi/shm/010-iterate_keys.t @@ -0,0 +1,192 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +add_block_preprocessor(sub { + my $block = shift; + if (!defined $block->main_config) { + $block->set_value("main_config", <<_EOC_ + wasm { + shm_kv kv 16k; + shm_queue q 16k; + } +_EOC_ + ); + } +}); + +plan_tests(4); +run_tests(); + +__DATA__ + +=== TEST 1: shm - iterate_keys() with page_size > nelts +--- skip_no_debug +--- valgrind +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv:set("k1", "value") + shm.kv:set("k2", "value") + shm.kv:set("k3", "value") + + shm.kv:lock() + + for k in shm.kv:iterate_keys() do + ngx.say(k) + end + + shm.kv:unlock() + } + } +--- response_body +k3 +k2 +k1 +--- grep_error_log: iterate_keys fetched a new page +--- grep_error_log_out +iterate_keys fetched a new page +--- no_error_log +[error] + + + +=== TEST 2: shm - iterate_keys() with page_size < nelts +--- skip_no_debug +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv:set("k1", "value") + shm.kv:set("k2", "value") + shm.kv:set("k3", "value") + + shm.kv:lock() + + for k in shm.kv:iterate_keys({ page_size = 2 }) do + ngx.say(k) + end + + shm.kv:unlock() + } + } +--- response_body +k3 +k2 +k1 +--- grep_error_log: iterate_keys fetched a new page +--- grep_error_log_out +iterate_keys fetched a new page +iterate_keys fetched a new page +--- no_error_log +[error] + + + +=== TEST 3: shm - iterate_keys() with page_size == nelts +--- skip_no_debug +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv:set("k1", "value") + shm.kv:set("k2", "value") + shm.kv:set("k3", "value") + + shm.kv:lock() + + for k in shm.kv:iterate_keys({ page_size = 3 }) do + ngx.say(k) + end + + shm.kv:unlock() + } + } +--- response_body +k3 +k2 +k1 +--- grep_error_log: iterate_keys fetched a new page +--- grep_error_log_out +iterate_keys fetched a new page +--- no_error_log +[error] + + + +=== TEST 4: shm - iterate_keys() on empty shm +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv:lock() + + for k in shm.kv:iterate_keys() do + ngx.say("fail") + end + + shm.kv:unlock() + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] + + + +=== TEST 5: shm - iterate_keys() on unlocked shm +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + for k in shm.kv:iterate_keys() do + + end + } + } +--- error_code: 500 +--- error_log eval +qr/\[error\] .*? attempt to call kv:iterate_keys\(\) but the shm zone is not locked; invoke :lock\(\) before and :unlock\(\) after./ +--- no_error_log +[crit] +[emerg] + + + +=== TEST 6: shm - iterate_keys() bad args +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.kv.iterate_keys, {}, true) + ngx.say(perr) + + _, perr = pcall(shm.kv.iterate_keys, {}, { page_size = true }) + ngx.say(perr) + + _, perr = pcall(shm.kv.iterate_keys, {}, { page_size = 0 }) + ngx.say(perr) + } + } +--- response_body +opts must be a table +opts.page_size must be a number +opts.page_size must be > 0 +--- no_error_log +[crit] +[emerg] diff --git a/t/04-openresty/ffi/shm/011-get_keys.t b/t/04-openresty/ffi/shm/011-get_keys.t new file mode 100644 index 000000000..5396d6003 --- /dev/null +++ b/t/04-openresty/ffi/shm/011-get_keys.t @@ -0,0 +1,114 @@ +# vim:set ft= ts=4 sts=4 sw=4 et fdm=marker: + +use strict; +use lib '.'; +use t::TestWasmX::Lua; + +skip_no_openresty(); + +add_block_preprocessor(sub { + my $block = shift; + if (!defined $block->main_config) { + $block->set_value("main_config", <<_EOC_ + wasm { + shm_kv kv 16k; + shm_queue q 16k; + } +_EOC_ + ); + } +}); + +plan_tests(4); +run_tests(); + +__DATA__ + +=== TEST 1: shm - get_keys() default limit +--- valgrind +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv:set("k1", "value") + shm.kv:set("k2", "value") + + for _, k in ipairs(shm.kv:get_keys()) do + ngx.say(k) + end + } + } +--- response_body +k1 +k2 +--- no_error_log +[error] +[crit] + + + +=== TEST 2: shm - get_keys() with limit +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + shm.kv:set("k1", "value") + shm.kv:set("k2", "value") + + local limit = 1 + + for _, k in ipairs(shm.kv:get_keys(limit)) do + ngx.say(k) + end + } + } +--- response_body +k1 +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm - get_keys() on empty shm +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + ngx.say(#shm.kv:get_keys()) + ngx.say(#shm.kv:get_keys(0)) + ngx.say(#shm.kv:get_keys(1)) + } + } +--- response_body +0 +0 +0 +--- no_error_log +[error] +[crit] + + + +=== TEST 4: shm - get_keys() bad args +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.kv.get_keys, {}, false) + ngx.say(perr) + + _, perr = pcall(shm.kv.get_keys, {}, -1) + ngx.say(perr) + } + } +--- response_body +max_count must be a number +max_count must be >= 0 +--- no_error_log +[crit] +[emerg] diff --git a/t/04-openresty/ffi/shm/020-metrics_define.t b/t/04-openresty/ffi/shm/020-metrics_define.t new file mode 100644 index 000000000..f6675387d --- /dev/null +++ b/t/04-openresty/ffi/shm/020-metrics_define.t @@ -0,0 +1,128 @@ +# 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(6); +run_tests(); + +__DATA__ + +=== TEST 1: shm_metrics - define() sanity +--- skip_no_debug +--- valgrind +--- metrics: 16k +--- 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 _, id in pairs(metrics) do + assert(id > 0) + end + + ngx.say("ok") + } + } +--- response_body +ok +--- error_log eval +[ + qr/.*? \[debug\] .*? defined counter "lua.c1" with id \d+/, + qr/.*? \[debug\] .*? defined gauge "lua.g1" with id \d+/, + qr/.*? \[debug\] .*? defined histogram "lua.h1" with id \d+/, +] +--- no_error_log +[error] + + + +=== TEST 2: shm_metrics - define() name too long +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local mid, err = shm.metrics:define(string.rep(".", 500), shm.metrics.HISTOGRAM) + ngx.say("mid: ", mid) + ngx.say("err: ", err) + } + } +--- response_body +mid: nil +err: name too long +--- no_error_log +[error] +[crit] +[emerg] +[alert] + + + +=== TEST 3: shm_metrics - define() no memory +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + for i = 1, 100 do + local _, err = shm.metrics:define(string.rep(i, 100), shm.metrics.HISTOGRAM) + if err then + ngx.log(ngx.ERR, err) + break + end + end + + ngx.say("ok") + } + } +--- response_body +ok +--- error_log eval +[ + qr/\[crit\] .*? "metrics" shm store: no memory; cannot allocate pair with key size 204 and value size 24/, + qr/\[error\] .*? access_by_lua\(.*?\):\d+: no memory/, +] +--- no_error_log +[emerg] +[alert] + + + +=== TEST 4: shm_metrics - define() bad args +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.metrics.define, {}, false, 10) + ngx.say(perr) + + _, perr = pcall(shm.metrics.define, {}, "", 10) + ngx.say(perr) + + _, perr = pcall(shm.metrics.define, {}, "c1", 10) + ngx.say(perr) + } + } +--- response_body +name must be a non-empty string +name must be a non-empty string +metric_type must be either resty.wasmx.shm.metrics.COUNTER, resty.wasmx.shm.metrics.GAUGE, or resty.wasmx.shm.metrics.HISTOGRAM +--- no_error_log +[error] +[crit] +[emerg] +[alert] diff --git a/t/04-openresty/ffi/shm/021-metrics_increment.t b/t/04-openresty/ffi/shm/021-metrics_increment.t new file mode 100644 index 000000000..36cfd4eba --- /dev/null +++ b/t/04-openresty/ffi/shm/021-metrics_increment.t @@ -0,0 +1,83 @@ +# 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_metrics - increment() sanity +--- valgrind +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + local pretty = require "pl.pretty" + + local c1 = shm.metrics:define("c1", shm.metrics.COUNTER) + + shm.metrics:increment(c1) -- value defaults to 1 + 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] + + + +=== TEST 2: shm_metrics - increment() metric not found +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local ok, err = shm.metrics:increment(1, 10) + ngx.say("ok: ", ok) + ngx.say("err: ", err) + } + } +--- response_body +ok: nil +err: metric not found +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm_metrics - increment() bad args +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.metrics.increment, {}, false) + ngx.say(perr) + + _, perr = pcall(shm.metrics.increment, {}, 1, false) + ngx.say(perr) + + _, perr = pcall(shm.metrics.increment, {}, 1, 0) + ngx.say(perr) + } + } +--- response_body +metric_id must be a number +value must be > 0 +value must be > 0 +--- no_error_log +[error] +[crit] diff --git a/t/04-openresty/ffi/shm/022-metrics_record.t b/t/04-openresty/ffi/shm/022-metrics_record.t new file mode 100644 index 000000000..ce36bdd6d --- /dev/null +++ b/t/04-openresty/ffi/shm/022-metrics_record.t @@ -0,0 +1,82 @@ +# 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_metrics - record() sanity +--- valgrind +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + local pretty = require "pl.pretty" + + local g1 = shm.metrics:define("g1", shm.metrics.GAUGE) + local h1 = shm.metrics:define("h1", shm.metrics.HISTOGRAM) + + assert(shm.metrics:record(g1, 10)) + assert(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] + + + +=== TEST 2: shm_metrics - record() metric not found +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local ok, err = shm.metrics:record(1, 10) + ngx.say("ok: ", ok) + ngx.say("err: ", err) + } + } +--- response_body +ok: nil +err: metric not found +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm_metrics - record() bad args +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.metrics.record, {}, false) + ngx.say(perr) + + _, perr = pcall(shm.metrics.record, {}, 1, false) + ngx.say(perr) + } + } +--- response_body +metric_id must be a number +value must be a number +--- no_error_log +[error] +[crit] diff --git a/t/04-openresty/ffi/shm/023-metrics_get.t b/t/04-openresty/ffi/shm/023-metrics_get.t new file mode 100644 index 000000000..2af9e6469 --- /dev/null +++ b/t/04-openresty/ffi/shm/023-metrics_get.t @@ -0,0 +1,82 @@ +# 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_metrics - get() sanity +--- valgrind +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + local pretty = require "pl.pretty" + + local c1 = shm.metrics:define("c1", shm.metrics.COUNTER) + local g1 = shm.metrics:define("g1", shm.metrics.GAUGE) + local h1 = shm.metrics:define("h1", shm.metrics.HISTOGRAM) + + shm.metrics:increment(c1) + shm.metrics:record(g1, 10) + shm.metrics:record(h1, 100) + + ngx.say("c1: ", pretty.write(shm.metrics:get(c1), "")) + ngx.say("g1: ", pretty.write(shm.metrics:get(g1), "")) + ngx.say("h1: ", pretty.write(shm.metrics:get(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] + + + +=== TEST 2: shm_metrics - get() metric not found +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local m = shm.metrics:get(99) + assert(m == nil) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm_metrics - get() bad args +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.metrics.get, {}, false) + ngx.say(perr) + } + } +--- response_body +metric_id must be a number +--- no_error_log +[error] +[crit] diff --git a/t/04-openresty/ffi/shm/024-metrics_get_by_name.t b/t/04-openresty/ffi/shm/024-metrics_get_by_name.t new file mode 100644 index 000000000..af144097d --- /dev/null +++ b/t/04-openresty/ffi/shm/024-metrics_get_by_name.t @@ -0,0 +1,136 @@ +# 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_metrics - get_by_name() sanity FFI-defined metrics +prefix: lua.* +--- valgrind +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + local pretty = require "pl.pretty" + + local c1 = shm.metrics:define("c1", shm.metrics.COUNTER) + local g1 = shm.metrics:define("g1", shm.metrics.GAUGE) + local h1 = shm.metrics:define("h1", shm.metrics.HISTOGRAM) + + shm.metrics:increment(c1) + shm.metrics:record(g1, 10) + shm.metrics:record(h1, 100) + + ngx.say("c1: ", pretty.write(shm.metrics:get_by_name("c1"), "")) + ngx.say("g1: ", pretty.write(shm.metrics:get_by_name("g1"), "")) + ngx.say("h1: ", pretty.write(shm.metrics:get_by_name("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] + + + +=== TEST 2: shm_metrics - get_by_name() sanity non-FFI-defined metrics +prefix: pw.hostcalls.* +--- wasm_modules: hostcalls +--- config + location /t { + proxy_wasm hostcalls 'on_configure=define_metrics \ + on=request_headers \ + test=/t/metrics/increment_counters \ + metrics=c1 \ + n_increments=13'; + + 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=100'; + + content_by_lua_block { + local shm = require "resty.wasmx.shm" + local pretty = require "pl.pretty" + + ngx.say("c1: ", pretty.write(shm.metrics:get_by_name("pw.hostcalls.c1", { prefix = false }), "")) + ngx.say("g1: ", pretty.write(shm.metrics:get_by_name("pw.hostcalls.g1", { prefix = false }), "")) + ngx.say("h1: ", pretty.write(shm.metrics:get_by_name("pw.hostcalls.h1", { prefix = false }), "")) + } + } +--- response_body +c1: {type="counter",value=13} +g1: {type="gauge",value=1} +h1: {type="histogram",value={{count=1,ub=128},{count=0,ub=4294967295}}} +--- no_error_log +[error] +[crit] + + + +=== TEST 3: shm_metrics - get_by_name() metric not found +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local m = shm.metrics:get_by_name("no_such_metric") + assert(m == nil) + + ngx.say("ok") + } + } +--- response_body +ok +--- no_error_log +[error] +[crit] + + + +=== TEST 4: shm_metrics - get_by_name() bad args +--- metrics: 16k +--- config + location /t { + access_by_lua_block { + local shm = require "resty.wasmx.shm" + + local _, perr = pcall(shm.metrics.get_by_name, {}, false) + ngx.say(perr) + + _, perr = pcall(shm.metrics.get_by_name, {}, "") + ngx.say(perr) + + _, perr = pcall(shm.metrics.get_by_name, {}, "m", false) + ngx.say(perr) + + _, perr = pcall(shm.metrics.get_by_name, {}, "m", { prefix = 1 }) + ngx.say(perr) + } + } +--- response_body +name must be a non-empty string +name must be a non-empty string +opts must be a table +opts.prefix must be a boolean +--- no_error_log +[error] +[crit] diff --git a/t/04-openresty/ffi/shm/101-kv_get.t b/t/04-openresty/ffi/shm/101-kv_get.t deleted file mode 100644 index 3e1b35656..000000000 --- a/t/04-openresty/ffi/shm/101-kv_get.t +++ /dev/null @@ -1,90 +0,0 @@ -# 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 deleted file mode 100644 index b440bb2f4..000000000 --- a/t/04-openresty/ffi/shm/102-kv_set.t +++ /dev/null @@ -1,102 +0,0 @@ -# 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 deleted file mode 100644 index 4badfa365..000000000 --- a/t/04-openresty/ffi/shm/201-metrics_get.t +++ /dev/null @@ -1,137 +0,0 @@ -# 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 deleted file mode 100644 index e6050f584..000000000 --- a/t/04-openresty/ffi/shm/202-metrics_define.t +++ /dev/null @@ -1,106 +0,0 @@ -# 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 deleted file mode 100644 index 16ee2b331..000000000 --- a/t/04-openresty/ffi/shm/203-metrics_record.t +++ /dev/null @@ -1,71 +0,0 @@ -# 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 deleted file mode 100644 index fdd413a9b..000000000 --- a/t/04-openresty/ffi/shm/204-metrics_increment.t +++ /dev/null @@ -1,68 +0,0 @@ -# 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] diff --git a/t/lib/proxy-wasm-tests/hostcalls/src/tests/mod.rs b/t/lib/proxy-wasm-tests/hostcalls/src/tests/mod.rs index 17acfb3c3..82993ef99 100644 --- a/t/lib/proxy-wasm-tests/hostcalls/src/tests/mod.rs +++ b/t/lib/proxy-wasm-tests/hostcalls/src/tests/mod.rs @@ -125,7 +125,9 @@ pub(crate) fn test_log_properties(ctx: &(dyn TestContext + 'static), phase: Test pub(crate) fn test_log_metrics(ctx: &(dyn TestContext + 'static), phase: TestPhase) { for (n, id) in ctx.get_metrics_mapping() { - if n.starts_with('h') { continue } + if n.starts_with('h') { + continue; + } let value = get_metric(*id).unwrap(); info!("{}: {} at {:?}", n, value, phase) } @@ -393,6 +395,21 @@ pub(crate) fn test_get_shared_data(ctx: &TestHttp) { ); } +pub(crate) fn test_log_shared_data(ctx: &TestHttp) { + let key = ctx.config.get("key").unwrap(); + let (data, cas) = ctx.get_shared_data(key); + let cas_value = cas + .map(|x| format!("{x}")) + .unwrap_or_else(|| "0".to_string()); + + info!( + "{}: {:?} {}", + key, + std::str::from_utf8(data.as_deref().unwrap_or_default()).unwrap(), + cas_value.as_str() + ) +} + pub(crate) fn test_set_shared_data(ctx: &TestHttp) { let cas = ctx.config.get("cas").map(|x| x.parse::().unwrap()); @@ -478,7 +495,11 @@ pub(crate) fn test_define_metrics(ctx: &mut (dyn TestContext + 'static)) { } } -pub(crate) fn test_increment_counters(ctx: &(dyn TestContext + 'static), phase: TestPhase, skip_others: Option) { +pub(crate) fn test_increment_counters( + ctx: &(dyn TestContext + 'static), + phase: TestPhase, + skip_others: Option, +) { if !should_run_on_current_worker(ctx) { return; } @@ -488,7 +509,9 @@ pub(crate) fn test_increment_counters(ctx: &(dyn TestContext + 'static), phase: .map_or(1, |x| x.parse::().expect("bad n_increments value")); for (n, id) in ctx.get_metrics_mapping() { - if skip_others.unwrap_or(true) && !n.starts_with('c') { continue } + if skip_others.unwrap_or(true) && !n.starts_with('c') { + continue; + } for _ in 0..n_increments { increment_metric(*id, 1).unwrap(); @@ -500,13 +523,19 @@ pub(crate) fn test_increment_counters(ctx: &(dyn TestContext + 'static), phase: test_log_metrics(ctx, phase); } -pub(crate) fn test_toggle_gauges(ctx: &(dyn TestContext + 'static), phase: TestPhase, skip_others: Option) { +pub(crate) fn test_toggle_gauges( + ctx: &(dyn TestContext + 'static), + phase: TestPhase, + skip_others: Option, +) { if !should_run_on_current_worker(ctx) { return; } for (n, id) in ctx.get_metrics_mapping() { - if skip_others.unwrap_or(true) && !n.starts_with('g') { continue } + if skip_others.unwrap_or(true) && !n.starts_with('g') { + continue; + } let new_value = match get_metric(*id).unwrap() { 0 => 1, @@ -530,7 +559,9 @@ pub(crate) fn test_record_metric(ctx: &(dyn TestContext + 'static), phase: TestP .map_or(1, |x| x.parse::().expect("bad value")); for (n, id) in ctx.get_metrics_mapping() { - if n.starts_with('c') { continue } + if n.starts_with('c') { + continue; + } record_metric(*id, value).unwrap(); info!("record {} on {} at {:?}", value, n, phase); } @@ -540,7 +571,9 @@ pub(crate) fn test_record_metric(ctx: &(dyn TestContext + 'static), phase: TestP pub(crate) fn test_get_metrics(ctx: &TestHttp) { for (n, id) in ctx.get_metrics_mapping() { - if n.starts_with('h') { continue } + if n.starts_with('h') { + continue; + } let name = n.replace("_", "-"); let value = get_metric(*id).unwrap().to_string(); diff --git a/t/lib/proxy-wasm-tests/hostcalls/src/types/test_http.rs b/t/lib/proxy-wasm-tests/hostcalls/src/types/test_http.rs index a097f0415..5240f512e 100644 --- a/t/lib/proxy-wasm-tests/hostcalls/src/types/test_http.rs +++ b/t/lib/proxy-wasm-tests/hostcalls/src/types/test_http.rs @@ -118,6 +118,7 @@ impl TestHttp { /* shared memory */ "/t/shm/get_shared_data" => test_get_shared_data(self), + "/t/shm/log_shared_data" => test_log_shared_data(self), "/t/shm/set_shared_data" => test_set_shared_data(self), "/t/shm/set_shared_data_by_len" => test_set_shared_data_by_len(self), "/t/shm/enqueue" => test_shared_queue_enqueue(self), @@ -144,7 +145,8 @@ impl TestHttp { } "/t/metrics/get_histogram" => { info!("retrieving histogram in \"{:?}\"", cur_phase); - let h_id = define_metric(MetricType::Histogram, "h1").expect("cannot define new metric"); + let h_id = + define_metric(MetricType::Histogram, "h1").expect("cannot define new metric"); get_metric(h_id).unwrap(); } diff --git a/util/_lib.sh b/util/_lib.sh index 0f796c12e..e60bd069a 100644 --- a/util/_lib.sh +++ b/util/_lib.sh @@ -74,6 +74,7 @@ install_luarocks() { $BIN_LUAROCKS --tree=$DIR_LUAJIT --lua-version=5.1 install lua-resty-dns-client $BIN_LUAROCKS --tree=$DIR_LUAJIT --lua-version=5.1 install LuaFileSystem + $BIN_LUAROCKS --tree=$DIR_LUAJIT --lua-version=5.1 install penlight } remove_luarocks() {