From a68132fd17dc2966feed6282a432e7b0bbf7c230 Mon Sep 17 00:00:00 2001 From: Murillo Paula Date: Thu, 20 May 2021 09:21:40 -0300 Subject: [PATCH 1/6] perf(client) reduce amount of timers on init_worker --- src/resty/dns/client.lua | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/resty/dns/client.lua b/src/resty/dns/client.lua index d684147..fe86368 100644 --- a/src/resty/dns/client.lua +++ b/src/resty/dns/client.lua @@ -822,6 +822,10 @@ local function syncQuery(qname, r_opts, try_list, count) try_status(try_list, "in progress (sync)") end + if ngx.get_phase() == "init" or ngx.get_phase() == "init_worker" then + return item, nil, try_list + end + -- block and wait for the async query to complete local ok, err = item.semaphore:wait(poolMaxWait) if ok and item.result then From 411e6fb5e57fbd680196717843ae2f2714f9ce6e Mon Sep 17 00:00:00 2001 From: Murillo Paula Date: Wed, 26 May 2021 09:45:58 -0300 Subject: [PATCH 2/6] tests(*) add prove tests for the client and adjust ci for it --- .gitignore | 2 +- .travis.yml | 5 +++ src/resty/dns/client.lua | 18 +++++++++- t/001-init.t | 75 ++++++++++++++++++++++++++++++++++++++++ 4 files changed, 98 insertions(+), 2 deletions(-) create mode 100644 t/001-init.t diff --git a/.gitignore b/.gitignore index c87f550..3a63889 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,3 @@ .DS_store *.rock - +t/servroot/ diff --git a/.travis.yml b/.travis.yml index ffe5cab..b6a95fa 100644 --- a/.travis.yml +++ b/.travis.yml @@ -25,6 +25,10 @@ install: - if [ -z "$OPENRESTY_VER" ]; then export OPENRESTY_VER=1.15.8.3; fi - if [ ! -f download-cache/openresty-$OPENRESTY_VER.tar.gz ]; then wget -O download-cache/openresty-$OPENRESTY_VER.tar.gz http://openresty.org/download/openresty-$OPENRESTY_VER.tar.gz; fi - if [ ! -f download-cache/luarocks-$LUAROCKS_VER.tar.gz ]; then wget -O download-cache/luarocks-$LUAROCKS_VER.tar.gz https://luarocks.github.io/luarocks/releases/luarocks-$LUAROCKS_VER.tar.gz; fi + - if [ ! -f download-cache/cpanm ]; then wget -O download-cache/cpanm https://cpanmin.us/; fi + - chmod +x download-cache/cpanm + - download-cache/cpanm --notest Test::Nginx >build.log 2>&1 || (cat build.log && exit 1) + - download-cache/cpanm --notest --local-lib=$TRAVIS_BUILD_DIR/perl5 local::lib && eval $(perl -I $TRAVIS_BUILD_DIR/perl5/lib/perl5/ -Mlocal::lib) - tar -zxf download-cache/openresty-$OPENRESTY_VER.tar.gz - tar -zxf download-cache/luarocks-$LUAROCKS_VER.tar.gz - pushd openresty-$OPENRESTY_VER @@ -49,3 +53,4 @@ install: script: - luacheck ./src - busted + - TEST_NGINX_RANDOMIZE=1 prove -v -r t diff --git a/src/resty/dns/client.lua b/src/resty/dns/client.lua index fe86368..7d48bda 100644 --- a/src/resty/dns/client.lua +++ b/src/resty/dns/client.lua @@ -33,6 +33,7 @@ local WARN = ngx.WARN local DEBUG = ngx.DEBUG local PREFIX = "[dns-client] " local timer_at = ngx.timer.at +local get_phase = ngx.get_phase local math_min = math.min local math_max = math.max @@ -822,7 +823,22 @@ local function syncQuery(qname, r_opts, try_list, count) try_status(try_list, "in progress (sync)") end - if ngx.get_phase() == "init" or ngx.get_phase() == "init_worker" then + local supported_semaphore_wait_phases = { + rewrite = true, + access = true, + content = true, + timer = true, + } + + local ngx_phase = get_phase() + + if not supported_semaphore_wait_phases[ngx_phase] then + -- phase not supported by `semaphore:wait` + -- return existing query (item) + -- + -- this will avoid: + -- "dns lookup pool exceeded retries" (second try and subsequent retries) + -- "API disabled in the context of init_worker_by_lua" (first try) return item, nil, try_list end diff --git a/t/001-init.t b/t/001-init.t new file mode 100644 index 0000000..ae9de39 --- /dev/null +++ b/t/001-init.t @@ -0,0 +1,75 @@ +use Test::Nginx::Socket 'no_plan'; + +run_tests(); + +__DATA__ + +=== TEST 1: init_worker: phase not supported by `semaphore:wait` +--- http_config eval +qq { + init_worker_by_lua_block { + local client = require("resty.dns.client") + assert(client.init()) + local host = "httpbin.org" + local typ = client.TYPE_A + local answers, err = client.resolve(host, { qtype = typ }) + ngx.log(ngx.ERR, "answers: ", answers) + ngx.log(ngx.ERR, err) + } +} +--- config + location = /t { + echo ok; + } +--- request +GET /t +--- response_body +ok +--- error_log +answers: nil +error: 101 empty record received +--- no_error_log +dns lookup pool exceeded retries +API disabled in the context of init_worker_by_lua + + + +=== TEST 2: access: phase supported by `semaphore:wait` +--- config + location = /t { + access_by_lua_block { + local client = require("resty.dns.client") + assert(client.init()) + local host = "httpbin.org" + local typ = client.TYPE_A + local answers, err = client.resolve(host, { qtype = typ }) + ngx.say("address name: ", answers[1].name) + } + } +--- request +GET /t +--- response_body +address name: httpbin.org + + + +=== TEST 3: init_worker: phase not supported by lua-resty-dns `new` API +--- http_config eval +qq { + init_worker_by_lua_block { + local resolver = require "resty.dns.resolver" + resolver:new({ nameservers = "8.8.8.8" }) + } +} +--- config + location = /t { + echo ok; + } +--- request +GET /t +--- response_body +ok +--- error_log +API disabled in the context of init_worker_by_lua +in function 'udp' +in function 'new' From 4df9b899c56ad31b90a39785ae9ca744f76d0afe Mon Sep 17 00:00:00 2001 From: Murillo Paula Date: Thu, 27 May 2021 16:23:12 -0300 Subject: [PATCH 3/6] tests(*) reusage of timers, independent on # of workers --- t/001-init.t | 106 +++++++++++++++++++++++++++++++++++++++++++++------ 1 file changed, 95 insertions(+), 11 deletions(-) diff --git a/t/001-init.t b/t/001-init.t index ae9de39..f833a08 100644 --- a/t/001-init.t +++ b/t/001-init.t @@ -1,10 +1,85 @@ -use Test::Nginx::Socket 'no_plan'; +use Test::Nginx::Socket; +plan tests => repeat_each() * (blocks() * 3) + 8; + +workers(6); + +no_shuffle(); run_tests(); __DATA__ -=== TEST 1: init_worker: phase not supported by `semaphore:wait` +=== TEST 1: reuse timers for queries of same name, independent on # of workers +--- http_config eval +qq { + init_worker_by_lua_block { + local client = require("resty.dns.client") + assert(client.init({ + nameservers = { "8.8.8.8" }, + hosts = {}, -- empty tables to parse to prevent defaulting to /etc/hosts + resolvConf = {}, -- and resolv.conf files + order = { "A" }, + })) + local host = "httpbin.org" + local typ = client.TYPE_A + for i = 1, 10 do + client.resolve(host, { qtype = typ }) + end + + local host = "mockbin.org" + for i = 1, 10 do + client.resolve(host, { qtype = typ }) + end + + workers = ngx.worker.count() + timers = ngx.timer.pending_count() + } +} +--- config + location = /t { + access_by_lua_block { + local client = require("resty.dns.client") + assert(client.init()) + local host = "httpbin.org" + local typ = client.TYPE_A + local answers, err = client.resolve(host, { qtype = typ }) + + if not answers then + ngx.say("failed to resolve: ", err) + end + + ngx.say("first address name: ", answers[1].name) + + host = "mockbin.org" + answers, err = client.resolve(host, { qtype = typ }) + + if not answers then + ngx.say("failed to resolve: ", err) + end + + ngx.say("second address name: ", answers[1].name) + + ngx.say("workers: ", workers) + + -- should be 2 timers maximum (1 for each hostname) + ngx.say("timers: ", timers) + } + } +--- request +GET /t +--- response_body +first address name: httpbin.org +second address name: mockbin.org +workers: 6 +timers: 2 +--- no_error_log +[error] +dns lookup pool exceeded retries +API disabled in the context of init_worker_by_lua + + + +=== TEST 2: init_worker: phase not supported by `semaphore:wait` --- http_config eval qq { init_worker_by_lua_block { @@ -12,29 +87,29 @@ qq { assert(client.init()) local host = "httpbin.org" local typ = client.TYPE_A - local answers, err = client.resolve(host, { qtype = typ }) - ngx.log(ngx.ERR, "answers: ", answers) - ngx.log(ngx.ERR, err) + answers, err = client.resolve(host, { qtype = typ }) } } --- config location = /t { - echo ok; + access_by_lua_block { + ngx.say("answers: ", answers) + ngx.say("err: ", err) + } } --- request GET /t --- response_body -ok ---- error_log answers: nil -error: 101 empty record received +err: dns client error: 101 empty record received --- no_error_log +[error] dns lookup pool exceeded retries API disabled in the context of init_worker_by_lua -=== TEST 2: access: phase supported by `semaphore:wait` +=== TEST 3: access: phase supported by `semaphore:wait` --- config location = /t { access_by_lua_block { @@ -43,6 +118,11 @@ API disabled in the context of init_worker_by_lua local host = "httpbin.org" local typ = client.TYPE_A local answers, err = client.resolve(host, { qtype = typ }) + + if not answers then + ngx.say("failed to resolve: ", err) + end + ngx.say("address name: ", answers[1].name) } } @@ -50,10 +130,14 @@ API disabled in the context of init_worker_by_lua GET /t --- response_body address name: httpbin.org +--- no_error_log +[error] +dns lookup pool exceeded retries +API disabled in the context of init_worker_by_lua -=== TEST 3: init_worker: phase not supported by lua-resty-dns `new` API +=== TEST 4: init_worker: phase not supported by lua-resty-dns `new` API --- http_config eval qq { init_worker_by_lua_block { From 6e494b67aa095eeabc1f78decfd988ad241c0e06 Mon Sep 17 00:00:00 2001 From: Vinicius Mignot Date: Thu, 10 Jun 2021 16:27:26 -0300 Subject: [PATCH 4/6] tests(*) simple refactor --- t/00-sanity.t | 27 ++++++++++ t/01-phases.t | 66 ++++++++++++++++++++++++ t/{001-init.t => 02-timer-usage.t} | 83 +----------------------------- 3 files changed, 94 insertions(+), 82 deletions(-) create mode 100644 t/00-sanity.t create mode 100644 t/01-phases.t rename t/{001-init.t => 02-timer-usage.t} (51%) diff --git a/t/00-sanity.t b/t/00-sanity.t new file mode 100644 index 0000000..b4a7be5 --- /dev/null +++ b/t/00-sanity.t @@ -0,0 +1,27 @@ +use strict; +use warnings FATAL => 'all'; +use Test::Nginx::Socket::Lua; + +plan tests => 2; + +run_tests(); + +__DATA__ + +=== TEST 1: load lua-resty-dns-client +--- config + location = /t { + access_by_lua_block { + local client = require("resty.dns.client") + assert(client.init()) + local host = "localhost" + local typ = client.TYPE_A + local answers, err = assert(client.resolve(host, { qtype = typ })) + ngx.say(answers[1].address) + } + } +--- request +GET /t +--- response_body +127.0.0.1 +--- no_error_log diff --git a/t/01-phases.t b/t/01-phases.t new file mode 100644 index 0000000..da23c9e --- /dev/null +++ b/t/01-phases.t @@ -0,0 +1,66 @@ +use Test::Nginx::Socket; + +plan tests => repeat_each() * (blocks() * 5); + +workers(6); + +no_shuffle(); +run_tests(); + +__DATA__ + +=== TEST 1: client supports access phase +--- config + location = /t { + access_by_lua_block { + local client = require("resty.dns.client") + assert(client.init()) + local host = "localhost" + local typ = client.TYPE_A + local answers, err = client.resolve(host, { qtype = typ }) + + if not answers then + ngx.say("failed to resolve: ", err) + end + + ngx.say("address name: ", answers[1].name) + } + } +--- request +GET /t +--- response_body +address name: localhost +--- no_error_log +[error] +dns lookup pool exceeded retries +API disabled in the context of init_worker_by_lua + + + +=== TEST 2: client does not support init_worker phase +--- http_config eval +qq { + init_worker_by_lua_block { + local client = require("resty.dns.client") + assert(client.init()) + local host = "konghq.com" + local typ = client.TYPE_A + answers, err = client.resolve(host, { qtype = typ }) + } +} +--- config + location = /t { + access_by_lua_block { + ngx.say("answers: ", answers) + ngx.say("err: ", err) + } + } +--- request +GET /t +--- response_body +answers: nil +err: dns client error: 101 empty record received +--- no_error_log +[error] +dns lookup pool exceeded retries +API disabled in the context of init_worker_by_lua diff --git a/t/001-init.t b/t/02-timer-usage.t similarity index 51% rename from t/001-init.t rename to t/02-timer-usage.t index f833a08..b4d3acf 100644 --- a/t/001-init.t +++ b/t/02-timer-usage.t @@ -1,6 +1,6 @@ use Test::Nginx::Socket; -plan tests => repeat_each() * (blocks() * 3) + 8; +plan tests => repeat_each() * (blocks() * 5); workers(6); @@ -76,84 +76,3 @@ timers: 2 [error] dns lookup pool exceeded retries API disabled in the context of init_worker_by_lua - - - -=== TEST 2: init_worker: phase not supported by `semaphore:wait` ---- http_config eval -qq { - init_worker_by_lua_block { - local client = require("resty.dns.client") - assert(client.init()) - local host = "httpbin.org" - local typ = client.TYPE_A - answers, err = client.resolve(host, { qtype = typ }) - } -} ---- config - location = /t { - access_by_lua_block { - ngx.say("answers: ", answers) - ngx.say("err: ", err) - } - } ---- request -GET /t ---- response_body -answers: nil -err: dns client error: 101 empty record received ---- no_error_log -[error] -dns lookup pool exceeded retries -API disabled in the context of init_worker_by_lua - - - -=== TEST 3: access: phase supported by `semaphore:wait` ---- config - location = /t { - access_by_lua_block { - local client = require("resty.dns.client") - assert(client.init()) - local host = "httpbin.org" - local typ = client.TYPE_A - local answers, err = client.resolve(host, { qtype = typ }) - - if not answers then - ngx.say("failed to resolve: ", err) - end - - ngx.say("address name: ", answers[1].name) - } - } ---- request -GET /t ---- response_body -address name: httpbin.org ---- no_error_log -[error] -dns lookup pool exceeded retries -API disabled in the context of init_worker_by_lua - - - -=== TEST 4: init_worker: phase not supported by lua-resty-dns `new` API ---- http_config eval -qq { - init_worker_by_lua_block { - local resolver = require "resty.dns.resolver" - resolver:new({ nameservers = "8.8.8.8" }) - } -} ---- config - location = /t { - echo ok; - } ---- request -GET /t ---- response_body -ok ---- error_log -API disabled in the context of init_worker_by_lua -in function 'udp' -in function 'new' From aa1a65e41d1ade692c49417449b440c694a63cf3 Mon Sep 17 00:00:00 2001 From: Murillo Paula Date: Thu, 10 Jun 2021 17:53:24 -0300 Subject: [PATCH 5/6] docs(readme) reduce amount of timers on init_worker --- README.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/README.md b/README.md index 2fb5ad9..ba125e5 100644 --- a/README.md +++ b/README.md @@ -50,6 +50,10 @@ Release process: 4. commit and tag the release 5. upload rock to LuaRocks +### Unreleased + +- Performance: reduce amount of timers on init_worker + ### 6.0.0 (05-Apr-2021) - BREAKING: the round-robin balancing algorithm is now implemented in From ab9b1a0fcd155b953dd3ea2f12b8736fcd6b0132 Mon Sep 17 00:00:00 2001 From: Murillo Paula Date: Thu, 10 Jun 2021 17:54:51 -0300 Subject: [PATCH 6/6] docs(readme) add PR link --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index ba125e5..7df6ce6 100644 --- a/README.md +++ b/README.md @@ -52,7 +52,7 @@ Release process: ### Unreleased -- Performance: reduce amount of timers on init_worker +- Performance: reduce amount of timers on init_worker. [PR 130](https://github.com/Kong/lua-resty-dns-client/pull/130) ### 6.0.0 (05-Apr-2021)