diff --git a/kong-1.3.0-0.rockspec b/kong-1.3.0-0.rockspec index 823b417fd743..fbf0313db596 100644 --- a/kong-1.3.0-0.rockspec +++ b/kong-1.3.0-0.rockspec @@ -45,6 +45,7 @@ dependencies = { "kong-proxy-cache-plugin ~> 1.2", "kong-plugin-request-transformer ~> 1.2", "kong-plugin-session ~> 2.1", + "kong-plugin-aws-lambda ~> 3.0", } build = { type = "builtin", @@ -344,10 +345,6 @@ build = { ["kong.plugins.bot-detection.schema"] = "kong/plugins/bot-detection/schema.lua", ["kong.plugins.bot-detection.rules"] = "kong/plugins/bot-detection/rules.lua", - ["kong.plugins.aws-lambda.handler"] = "kong/plugins/aws-lambda/handler.lua", - ["kong.plugins.aws-lambda.schema"] = "kong/plugins/aws-lambda/schema.lua", - ["kong.plugins.aws-lambda.v4"] = "kong/plugins/aws-lambda/v4.lua", - ["kong.plugins.request-termination.handler"] = "kong/plugins/request-termination/handler.lua", ["kong.plugins.request-termination.schema"] = "kong/plugins/request-termination/schema.lua", } diff --git a/kong/plugins/aws-lambda/handler.lua b/kong/plugins/aws-lambda/handler.lua deleted file mode 100644 index 79b0205f286d..000000000000 --- a/kong/plugins/aws-lambda/handler.lua +++ /dev/null @@ -1,248 +0,0 @@ --- Copyright (C) Kong Inc. -local aws_v4 = require "kong.plugins.aws-lambda.v4" -local http = require "resty.http" -local cjson = require "cjson.safe" -local meta = require "kong.meta" -local constants = require "kong.constants" - - -local VIA_HEADER = constants.HEADERS.VIA -local VIA_HEADER_VALUE = meta._NAME .. "/" .. meta._VERSION - - -local tostring = tostring -local tonumber = tonumber -local type = type -local fmt = string.format -local ngx_encode_base64 = ngx.encode_base64 - - -local raw_content_types = { - ["text/plain"] = true, - ["text/html"] = true, - ["application/xml"] = true, - ["text/xml"] = true, - ["application/soap+xml"] = true, -} - - -local AWS_PORT = 443 - - ---[[ - Response format should be - { - "statusCode": httpStatusCode, - "headers": { "headerName": "headerValue", ... }, - "body": "..." - } ---]] -local function validate_custom_response(response) - if type(response.statusCode) ~= "number" then - return nil, "statusCode must be a number" - end - - if response.headers ~= nil and type(response.headers) ~= "table" then - return nil, "headers must be a table" - end - - if response.body ~= nil and type(response.body) ~= "string" then - return nil, "body must be a string" - end - - return true -end - - -local function extract_proxy_response(content) - local serialized_content, err = cjson.decode(content) - if not serialized_content then - return nil, err - end - - local ok, err = validate_custom_response(serialized_content) - if not ok then - return nil, err - end - - local headers = serialized_content.headers or {} - local body = serialized_content.body or "" - headers["Content-Length"] = #body - - return { - status_code = tonumber(serialized_content.statusCode), - body = body, - headers = headers, - } -end - - -local AWSLambdaHandler = {} - - -function AWSLambdaHandler:access(conf) - local upstream_body = kong.table.new(0, 6) - local var = ngx.var - - if conf.forward_request_body or conf.forward_request_headers - or conf.forward_request_method or conf.forward_request_uri - then - -- new behavior to forward request method, body, uri and their args - if conf.forward_request_method then - upstream_body.request_method = kong.request.get_method() - end - - if conf.forward_request_headers then - upstream_body.request_headers = kong.request.get_headers() - end - - if conf.forward_request_uri then - local path = kong.request.get_path() - local query = kong.request.get_raw_query() - if query ~= "" then - upstream_body.request_uri = path .. "?" .. query - else - upstream_body.request_uri = path - end - upstream_body.request_uri_args = kong.request.get_query() - end - - if conf.forward_request_body then - local content_type = kong.request.get_header("content-type") - local body_raw = kong.request.get_raw_body() - local body_args, err = kong.request.get_body() - if err and err:match("content type") then - body_args = {} - if not raw_content_types[content_type] then - -- don't know what this body MIME type is, base64 it just in case - body_raw = ngx_encode_base64(body_raw) - upstream_body.request_body_base64 = true - end - end - - upstream_body.request_body = body_raw - upstream_body.request_body_args = body_args - end - - else - -- backwards compatible upstream body for configurations not specifying - -- `forward_request_*` values - local body_args = kong.request.get_body() - upstream_body = kong.table.merge(kong.request.get_query(), body_args) - end - - local upstream_body_json, err = cjson.encode(upstream_body) - if not upstream_body_json then - kong.log.err("could not JSON encode upstream body", - " to forward request values: ", err) - end - - local host = fmt("lambda.%s.amazonaws.com", conf.aws_region) - local path = fmt("/2015-03-31/functions/%s/invocations", - conf.function_name) - local port = conf.port or AWS_PORT - - local opts = { - region = conf.aws_region, - service = "lambda", - method = "POST", - headers = { - ["X-Amz-Target"] = "invoke", - ["X-Amz-Invocation-Type"] = conf.invocation_type, - ["X-Amz-Log-Type"] = conf.log_type, - ["Content-Type"] = "application/x-amz-json-1.1", - ["Content-Length"] = upstream_body_json and tostring(#upstream_body_json), - }, - body = upstream_body_json, - path = path, - host = host, - port = port, - access_key = conf.aws_key, - secret_key = conf.aws_secret, - query = conf.qualifier and "Qualifier=" .. conf.qualifier - } - - local request, err = aws_v4(opts) - if err then - kong.log.err(err) - return kong.response.exit(500, { message = "An unexpected error occurred" }) - end - - -- Trigger request - local client = http.new() - client:set_timeout(conf.timeout) - client:connect(host, port) - local ok, err = client:ssl_handshake() - if not ok then - kong.log.err(err) - return kong.response.exit(500, { message = "An unexpected error occurred" }) - end - - local res, err = client:request { - method = "POST", - path = request.url, - body = request.body, - headers = request.headers - } - if not res then - kong.log.err(err) - return kong.response.exit(500, { message = "An unexpected error occurred" }) - end - - local content = res:read_body() - local headers = res.headers - - if var.http2 then - headers["Connection"] = nil - headers["Keep-Alive"] = nil - headers["Proxy-Connection"] = nil - headers["Upgrade"] = nil - headers["Transfer-Encoding"] = nil - end - - local ok, err = client:set_keepalive(conf.keepalive) - if not ok then - kong.log.err(err) - return kong.response.exit(500, { message = "An unexpected error occurred" }) - end - - local status - - if conf.is_proxy_integration then - local proxy_response, err = extract_proxy_response(content) - if not proxy_response then - kong.log.err(err) - return kong.response.exit(502, { message = "Bad Gateway", - error = "could not JSON decode Lambda " .. - "function response: " .. err }) - end - - status = proxy_response.status_code - headers = kong.table.merge(headers, proxy_response.headers) - content = proxy_response.body - end - - if not status then - if conf.unhandled_status - and headers["X-Amz-Function-Error"] == "Unhandled" - then - status = conf.unhandled_status - - else - status = res.status - end - end - - if kong.configuration.enabled_headers[VIA_HEADER] then - headers[VIA_HEADER] = VIA_HEADER_VALUE - end - - return kong.response.exit(status, content, headers) -end - - -AWSLambdaHandler.PRIORITY = 750 -AWSLambdaHandler.VERSION = "2.0.0" - - -return AWSLambdaHandler diff --git a/kong/plugins/aws-lambda/schema.lua b/kong/plugins/aws-lambda/schema.lua deleted file mode 100644 index 5c186b728bc8..000000000000 --- a/kong/plugins/aws-lambda/schema.lua +++ /dev/null @@ -1,97 +0,0 @@ -local typedefs = require "kong.db.schema.typedefs" - -local REGIONS = { - "ap-northeast-1", "ap-northeast-2", - "ap-south-1", - "ap-southeast-1", "ap-southeast-2", - "ca-central-1", - "cn-north-1", - "cn-northwest-1", - "eu-central-1", - "eu-west-1", "eu-west-2", - "sa-east-1", - "us-east-1", "us-east-2", - "us-gov-west-1", - "us-west-1", "us-west-2", -} - - -return { - name = "aws-lambda", - fields = { - { run_on = typedefs.run_on_first }, - { protocols = typedefs.protocols_http }, - { config = { - type = "record", - fields = { - { timeout = { - type = "number", - required = true, - default = 60000 - } }, - { keepalive = { - type = "number", - required = true, - default = 60000 - } }, - { aws_key = { - type = "string", - required = true - } }, - { aws_secret = { - type = "string", - required = true - } }, - { aws_region = { - type = "string", - required = true, - one_of = REGIONS - } }, - { function_name = { - type = "string", - required = true - } }, - { qualifier = { - type = "string" - } }, - { invocation_type = { - type = "string", - required = true, - default = "RequestResponse", - one_of = { "RequestResponse", "Event", "DryRun" }, - } }, - { log_type = { - type = "string", - required = true, - default = "Tail", - one_of = { "Tail", "None" }, - } }, - { port = typedefs.port { default = 443 }, }, - { unhandled_status = { - type = "integer", - between = { 100, 999 }, - } }, - { forward_request_method = { - type = "boolean", - default = false - } }, - { forward_request_uri = { - type = "boolean", - default = false - } }, - { forward_request_headers = { - type = "boolean", - default = false - } }, - { forward_request_body = { - type = "boolean", - default = false - } }, - { is_proxy_integration = { - type = "boolean", - default = false - } }, - } - } }, - }, -} diff --git a/kong/plugins/aws-lambda/v4.lua b/kong/plugins/aws-lambda/v4.lua deleted file mode 100644 index d7c40de54600..000000000000 --- a/kong/plugins/aws-lambda/v4.lua +++ /dev/null @@ -1,227 +0,0 @@ --- Performs AWSv4 Signing --- http://docs.aws.amazon.com/general/latest/gr/sigv4_signing.html - -local resty_sha256 = require "resty.sha256" -local pl_string = require "pl.stringx" -local openssl_hmac = require "openssl.hmac" - -local ALGORITHM = "AWS4-HMAC-SHA256" - -local CHAR_TO_HEX = {}; -for i = 0, 255 do - local char = string.char(i) - local hex = string.format("%02x", i) - CHAR_TO_HEX[char] = hex -end - -local function hmac(secret, data) - return openssl_hmac.new(secret, "sha256"):final(data) -end - -local function hash(str) - local sha256 = resty_sha256:new() - sha256:update(str) - return sha256:final() -end - -local function hex_encode(str) -- From prosody's util.hex - return (str:gsub(".", CHAR_TO_HEX)) -end - -local function percent_encode(char) - return string.format("%%%02X", string.byte(char)) -end - -local function canonicalise_path(path) - local segments = {} - for segment in path:gmatch("/([^/]*)") do - if segment == "" or segment == "." then - segments = segments -- do nothing and avoid lint - elseif segment == " .. " then - -- intentionally discards components at top level - segments[#segments] = nil - else - segments[#segments+1] = ngx.unescape_uri(segment):gsub("[^%w%-%._~]", - percent_encode) - end - end - local len = #segments - if len == 0 then - return "/" - end - -- If there was a slash on the end, keep it there. - if path:sub(-1, -1) == "/" then - len = len + 1 - segments[len] = "" - end - segments[0] = "" - segments = table.concat(segments, "/", 0, len) - return segments -end - -local function canonicalise_query_string(query) - local q = {} - for key, val in query:gmatch("([^&=]+)=?([^&]*)") do - key = ngx.unescape_uri(key):gsub("[^%w%-%._~]", percent_encode) - val = ngx.unescape_uri(val):gsub("[^%w%-%._~]", percent_encode) - q[#q+1] = key .. "=" .. val - end - table.sort(q) - return table.concat(q, "&") -end - -local function derive_signing_key(kSecret, date, region, service) - local kDate = hmac("AWS4" .. kSecret, date) - local kRegion = hmac(kDate, region) - local kService = hmac(kRegion, service) - local kSigning = hmac(kService, "aws4_request") - return kSigning -end - -local function prepare_awsv4_request(tbl) - local domain = tbl.domain or "amazonaws.com" - local region = tbl.region - local service = tbl.service - local request_method = tbl.method - local canonicalURI = tbl.canonicalURI - local path = tbl.path - if path and not canonicalURI then - canonicalURI = canonicalise_path(path) - elseif canonicalURI == nil or canonicalURI == "" then - canonicalURI = "/" - end - local canonical_querystring = tbl.canonical_querystring - local query = tbl.query - if query and not canonical_querystring then - canonical_querystring = canonicalise_query_string(query) - end - local req_headers = tbl.headers or {} - local req_payload = tbl.body - local access_key = tbl.access_key - local signing_key = tbl.signing_key - local secret_key - if not signing_key then - secret_key = tbl.secret_key - if secret_key == nil then - return nil, "either 'signing_key' or 'secret_key' must be provided" - end - end - local tls = tbl.tls - if tls == nil then - tls = true - end - local port = tbl.port or (tls and 443 or 80) - local timestamp = tbl.timestamp or ngx.time() - local req_date = os.date("!%Y%m%dT%H%M%SZ", timestamp) - local date = os.date("!%Y%m%d", timestamp) - - local host = service .. "." .. region .. "." .. domain - local host_header do -- If the "standard" port is not in use, the port should be added to the Host header - local with_port - if tls then - with_port = port ~= 443 - else - with_port = port ~= 80 - end - if with_port then - host_header = string.format("%s:%d", host, port) - else - host_header = host - end - end - - local headers = { - ["X-Amz-Date"] = req_date; - Host = host_header; - } - local add_auth_header = true - for k, v in pairs(req_headers) do - k = k:gsub("%f[^%z-]%w", string.upper) -- convert to standard header title case - if k == "Authorization" then - add_auth_header = false - elseif v == false then -- don't allow a default value for this header - v = nil - end - headers[k] = v - end - - -- Task 1: Create a Canonical Request For Signature Version 4 - -- http://docs.aws.amazon.com/general/latest/gr/sigv4-create-canonical-request.html - local canonical_headers, signed_headers do - -- We structure this code in a way so that we only have to sort once. - canonical_headers, signed_headers = {}, {} - local i = 0 - for name, value in pairs(headers) do - if value then -- ignore headers with 'false', they are used to override defaults - i = i + 1 - local name_lower = name:lower() - signed_headers[i] = name_lower - if canonical_headers[name_lower] ~= nil then - return nil, "header collision" - end - canonical_headers[name_lower] = pl_string.strip(value) - end - end - table.sort(signed_headers) - for j=1, i do - local name = signed_headers[j] - local value = canonical_headers[name] - canonical_headers[j] = name .. ":" .. value .. "\n" - end - signed_headers = table.concat(signed_headers, ";", 1, i) - canonical_headers = table.concat(canonical_headers, nil, 1, i) - end - local canonical_request = - request_method .. '\n' .. - canonicalURI .. '\n' .. - (canonical_querystring or "") .. '\n' .. - canonical_headers .. '\n' .. - signed_headers .. '\n' .. - hex_encode(hash(req_payload or "")) - - local hashed_canonical_request = hex_encode(hash(canonical_request)) - -- Task 2: Create a String to Sign for Signature Version 4 - -- http://docs.aws.amazon.com/general/latest/gr/sigv4-create-string-to-sign.html - local credential_scope = date .. "/" .. region .. "/" .. service .. "/aws4_request" - local string_to_sign = - ALGORITHM .. '\n' .. - req_date .. '\n' .. - credential_scope .. '\n' .. - hashed_canonical_request - - -- Task 3: Calculate the AWS Signature Version 4 - -- http://docs.aws.amazon.com/general/latest/gr/sigv4-calculate-signature.html - if signing_key == nil then - signing_key = derive_signing_key(secret_key, date, region, service) - end - local signature = hex_encode(hmac(signing_key, string_to_sign)) - -- Task 4: Add the Signing Information to the Request - -- http://docs.aws.amazon.com/general/latest/gr/sigv4-add-signature-to-request.html - local authorization = ALGORITHM - .. " Credential=" .. access_key .. "/" .. credential_scope - .. ", SignedHeaders=" .. signed_headers - .. ", Signature=" .. signature - if add_auth_header then - headers.Authorization = authorization - end - - local target = path or canonicalURI - if query or canonical_querystring then - target = target .. "?" .. (query or canonical_querystring) - end - local scheme = tls and "https" or "http" - local url = scheme .. "://" .. host_header .. target - - return { - url = url, - host = host, - port = port, - tls = tls, - method = request_method, - target = target, - headers = headers, - body = req_payload, - } -end - -return prepare_awsv4_request diff --git a/spec/03-plugins/22-aws-lambda/01-access_spec.lua b/spec/03-plugins/22-aws-lambda/01-access_spec.lua deleted file mode 100644 index 597aa6a0db75..000000000000 --- a/spec/03-plugins/22-aws-lambda/01-access_spec.lua +++ /dev/null @@ -1,882 +0,0 @@ -local cjson = require "cjson" -local helpers = require "spec.helpers" -local meta = require "kong.meta" - - -local server_tokens = meta._SERVER_TOKENS - - -local fixtures = { - http_mock = { - lambda_plugin = [[ - - server { - server_name mock_aws_lambda; - listen 10001 ssl; - - ssl_certificate ${{SSL_CERT}}; - ssl_certificate_key ${{SSL_CERT_KEY}}; - ssl_protocols TLSv1.1 TLSv1.2 TLSv1.3; - - location ~ "/2015-03-31/functions/(?:[^/])*/invocations" { - content_by_lua_block { - local function x() - local function say(res, status) - ngx.header["x-amzn-RequestId"] = "foo" - - if string.match(ngx.var.uri, "functionWithUnhandledError") then - ngx.header["X-Amz-Function-Error"] = "Unhandled" - end - - ngx.status = status - - if string.match(ngx.var.uri, "functionWithBadJSON") then - local badRes = "{\"foo\":\"bar\"" - ngx.header["Content-Length"] = #badRes + 1 - ngx.say(badRes) - - elseif string.match(ngx.var.uri, "functionWithNoResponse") then - ngx.header["Content-Length"] = 0 - - elseif type(res) == 'string' then - ngx.header["Content-Length"] = #res + 1 - ngx.say(res) - - else - ngx.req.discard_body() - ngx.header['Content-Length'] = 0 - end - - ngx.exit(0) - end - - ngx.sleep(.2) -- mock some network latency - - local invocation_type = ngx.var.http_x_amz_invocation_type - if invocation_type == 'Event' then - say(nil, 202) - - elseif invocation_type == 'DryRun' then - say(nil, 204) - end - - local qargs = ngx.req.get_uri_args() - ngx.req.read_body() - local args = require("cjson").decode(ngx.req.get_body_data()) - - say(ngx.req.get_body_data(), 200) - end - local ok, err = pcall(x) - if not ok then - ngx.log(ngx.ERR, "Mock error: ", err) - end - } - } - } - - ]] - }, -} - - -for _, strategy in helpers.each_strategy() do - describe("Plugin: AWS Lambda (access) [#" .. strategy .. "]", function() - local proxy_client - local admin_client - - lazy_setup(function() - local bp = helpers.get_db_utils(strategy, { - "routes", - "services", - "plugins", - }) - - local route1 = bp.routes:insert { - hosts = { "lambda.com" }, - } - - local route2 = bp.routes:insert { - hosts = { "lambda2.com" }, - } - - local route3 = bp.routes:insert { - hosts = { "lambda3.com" }, - } - - local route4 = bp.routes:insert { - hosts = { "lambda4.com" }, - } - - local route5 = bp.routes:insert { - hosts = { "lambda5.com" }, - } - - local route6 = bp.routes:insert { - hosts = { "lambda6.com" }, - } - - local route7 = bp.routes:insert { - hosts = { "lambda7.com" }, - } - - local route8 = bp.routes:insert { - hosts = { "lambda8.com" }, - } - - local route9 = bp.routes:insert { - hosts = { "lambda9.com" }, - protocols = { "http", "https" }, - service = bp.services:insert({ - protocol = "http", - host = "httpbin.org", - port = 80, - }) - } - - local service10 = bp.services:insert({ - protocol = "http", - host = "httpbin.org", - port = 80, - }) - - local route10 = bp.routes:insert { - hosts = { "lambda10.com" }, - protocols = { "http", "https" }, - service = service10 - } - - local service11 = bp.services:insert({ - protocol = "http", - host = "httpbin.org", - port = 80, - }) - - local route11 = bp.routes:insert { - hosts = { "lambda11.com" }, - protocols = { "http", "https" }, - service = service11 - } - - local service12 = bp.services:insert({ - protocol = "http", - host = "httpbin.org", - port = 80, - }) - - local route12 = bp.routes:insert { - hosts = { "lambda12.com" }, - protocols = { "http", "https" }, - service = service12 - } - - local route13 = bp.routes:insert { - hosts = { "lambda13.com" }, - protocols = { "http", "https" }, - service = service12, - } - - local route14 = bp.routes:insert { - hosts = { "lambda14.com" }, - protocols = { "http", "https" }, - service = ngx.null, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route1.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route2.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - invocation_type = "Event", - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route3.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - invocation_type = "DryRun", - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route4.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - timeout = 100, - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route5.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "functionWithUnhandledError", - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route6.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "functionWithUnhandledError", - invocation_type = "Event", - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route7.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "functionWithUnhandledError", - invocation_type = "DryRun", - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route8.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "functionWithUnhandledError", - unhandled_status = 412, - }, - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route9.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - forward_request_method = true, - forward_request_uri = true, - forward_request_headers = true, - forward_request_body = true, - } - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route10.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - forward_request_method = true, - forward_request_uri = false, - forward_request_headers = true, - forward_request_body = true, - } - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route11.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - is_proxy_integration = true, - } - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route12.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "functionWithBadJSON", - is_proxy_integration = true, - } - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route13.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "functionWithNoResponse", - is_proxy_integration = true, - } - } - - bp.plugins:insert { - name = "aws-lambda", - route = { id = route14.id }, - config = { - port = 10001, - aws_key = "mock-key", - aws_secret = "mock-secret", - aws_region = "us-east-1", - function_name = "kongLambdaTest", - }, - } - - assert(helpers.start_kong({ - database = strategy, - nginx_conf = "spec/fixtures/custom_nginx.template", - }, nil, nil, fixtures)) - end) - - before_each(function() - proxy_client = helpers.proxy_client() - admin_client = helpers.admin_client() - end) - - after_each(function () - proxy_client:close() - admin_client:close() - end) - - lazy_teardown(function() - helpers.stop_kong() - end) - - it("invokes a Lambda function with GET", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda.com" - } - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - assert.equal("some_value1", body.key1) - assert.is_nil(res.headers["X-Amz-Function-Error"]) - end) - it("invokes a Lambda function with POST params", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda.com", - ["Content-Type"] = "application/x-www-form-urlencoded" - }, - body = { - key1 = "some_value_post1", - key2 = "some_value_post2", - key3 = "some_value_post3" - } - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - assert.equal("some_value_post1", body.key1) - end) - it("invokes a Lambda function with POST json body", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda.com", - ["Content-Type"] = "application/json" - }, - body = { - key1 = "some_value_json1", - key2 = "some_value_json2", - key3 = "some_value_json3" - } - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - assert.equal("some_value_json1", body.key1) - end) - it("passes empty json arrays unmodified", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda.com", - ["Content-Type"] = "application/json" - }, - body = '[{}, []]' - }) - assert.res_status(200, res) - assert.equal('[{},[]]', string.gsub(res:read_body(), "\n","")) - end) - it("invokes a Lambda function with POST and both querystring and body params", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post?key1=from_querystring", - headers = { - ["Host"] = "lambda.com", - ["Content-Type"] = "application/x-www-form-urlencoded" - }, - body = { - key2 = "some_value_post2", - key3 = "some_value_post3" - } - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - assert.equal("from_querystring", body.key1) - end) - it("invokes a Lambda function with POST and xml payload, custom header and query partameter", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post?key1=from_querystring", - headers = { - ["Host"] = "lambda9.com", - ["Content-Type"] = "application/xml", - ["custom-header"] = "someheader" - }, - body = "" - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - - -- request_method - assert.equal("POST", body.request_method) - - -- request_uri - assert.equal("/post?key1=from_querystring", body.request_uri) - assert.is_table(body.request_uri_args) - - -- request_headers - assert.equal("someheader", body.request_headers["custom-header"]) - assert.equal("lambda9.com", body.request_headers.host) - - -- request_body - assert.equal("", body.request_body) - assert.is_table(body.request_body_args) - end) - it("invokes a Lambda function with POST and json payload, custom header and query partameter", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post?key1=from_querystring", - headers = { - ["Host"] = "lambda10.com", - ["Content-Type"] = "application/json", - ["custom-header"] = "someheader" - }, - body = { key2 = "some_value" } - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - - -- request_method - assert.equal("POST", body.request_method) - - -- no request_uri - assert.is_nil(body.request_uri) - assert.is_nil(body.request_uri_args) - - -- request_headers - assert.equal("lambda10.com", body.request_headers.host) - assert.equal("someheader", body.request_headers["custom-header"]) - - -- request_body - assert.equal("some_value", body.request_body_args.key2) - assert.is_table(body.request_body_args) - end) - it("invokes a Lambda function with POST and txt payload, custom header and query partameter", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post?key1=from_querystring", - headers = { - ["Host"] = "lambda9.com", - ["Content-Type"] = "text/plain", - ["custom-header"] = "someheader" - }, - body = "some text" - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - - -- request_method - assert.equal("POST", body.request_method) - - -- request_uri - assert.equal("/post?key1=from_querystring", body.request_uri) - assert.is_table(body.request_uri_args) - - -- request_headers - assert.equal("someheader", body.request_headers["custom-header"]) - assert.equal("lambda9.com", body.request_headers.host) - - -- request_body - assert.equal("some text", body.request_body) - assert.is_nil(body.request_body_base64) - assert.is_table(body.request_body_args) - end) - it("invokes a Lambda function with POST and binary payload and custom header", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post?key1=from_querystring", - headers = { - ["Host"] = "lambda9.com", - ["Content-Type"] = "application/octet-stream", - ["custom-header"] = "someheader" - }, - body = "01234" - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - - -- request_method - assert.equal("POST", body.request_method) - - -- request_uri - assert.equal("/post?key1=from_querystring", body.request_uri) - assert.is_table(body.request_uri_args) - - -- request_headers - assert.equal("lambda9.com", body.request_headers.host) - assert.equal("someheader", body.request_headers["custom-header"]) - - -- request_body - assert.equal(ngx.encode_base64('01234'), body.request_body) - assert.is_true(body.request_body_base64) - assert.is_table(body.request_body_args) - end) - it("invokes a Lambda function with POST params and Event invocation_type", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda2.com", - ["Content-Type"] = "application/x-www-form-urlencoded" - }, - body = { - key1 = "some_value_post1", - key2 = "some_value_post2", - key3 = "some_value_post3" - } - }) - assert.res_status(202, res) - assert.is_string(res.headers["x-amzn-RequestId"]) - end) - it("invokes a Lambda function with POST params and DryRun invocation_type", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda3.com", - ["Content-Type"] = "application/x-www-form-urlencoded" - }, - body = { - key1 = "some_value_post1", - key2 = "some_value_post2", - key3 = "some_value_post3" - } - }) - assert.res_status(204, res) - assert.is_string(res.headers["x-amzn-RequestId"]) - end) - it("errors on connection timeout", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda4.com", - } - }) - assert.res_status(500, res) - end) - - it("invokes a Lambda function with an unhandled function error (and no unhandled_status set)", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda5.com" - } - }) - assert.res_status(200, res) - assert.equal("Unhandled", res.headers["X-Amz-Function-Error"]) - end) - it("invokes a Lambda function with an unhandled function error with Event invocation type", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda6.com" - } - }) - assert.res_status(202, res) - assert.equal("Unhandled", res.headers["X-Amz-Function-Error"]) - end) - it("invokes a Lambda function with an unhandled function error with DryRun invocation type", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda7.com" - } - }) - assert.res_status(204, res) - assert.equal("Unhandled", res.headers["X-Amz-Function-Error"]) - end) - it("invokes a Lambda function with an unhandled function error", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda8.com" - } - }) - assert.res_status(412, res) - assert.equal("Unhandled", res.headers["X-Amz-Function-Error"]) - end) - - it("returns server tokens with Via header", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda.com" - } - }) - - assert.equal(server_tokens, res.headers["Via"]) - end) - - it("returns Content-Length header", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda.com" - } - }) - - assert.equal(65, tonumber(res.headers["Content-Length"])) - end) - - - describe("config.is_proxy_integration = true", function() - it("sets proper status code on custom response from Lambda", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda11.com", - ["Content-Type"] = "application/json" - }, - body = { - statusCode = 201, - } - }) - local body = assert.res_status(201, res) - assert.equal(0, tonumber(res.headers["Content-Length"])) - assert.equal(nil, res.headers["X-Custom-Header"]) - assert.equal("", body) - end) - - it("sets proper status code/headers/body on custom response from Lambda", function() - -- the lambda function must return a string - -- for the custom response "body" property - local body = cjson.encode({ - key1 = "some_value_post1", - key2 = "some_value_post2", - key3 = "some_value_post3", - }) - - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda11.com", - ["Content-Type"] = "application/json", - }, - body = { - statusCode = 201, - body = body, - headers = { - ["X-Custom-Header"] = "Hello world!" - } - } - }) - - local res_body = assert.res_status(201, res) - assert.equal(79, tonumber(res.headers["Content-Length"])) - assert.equal("Hello world!", res.headers["X-Custom-Header"]) - assert.equal(body, res_body) - end) - - it("override duplicated headers with value from the custom response from Lambda", function() - -- the default "x-amzn-RequestId" returned is "foo" - -- let's check it is overriden with a custom value - local headers = { - ["x-amzn-RequestId"] = "bar", - } - - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda11.com", - ["Content-Type"] = "application/json", - }, - body = { - statusCode = 201, - headers = headers, - } - }) - - assert.res_status(201, res) - assert.equal("bar", res.headers["x-amzn-RequestId"]) - end) - - it("returns HTTP 502 when 'status' property of custom response is not a number", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda11.com", - ["Content-Type"] = "application/json", - }, - body = { - statusCode = "hello", - } - }) - - assert.res_status(502, res) - local b = assert.response(res).has.jsonbody() - assert.equal("Bad Gateway", b.message) - end) - - it("returns HTTP 502 when 'headers' property of custom response is not a table", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda11.com", - ["Content-Type"] = "application/json", - }, - body = { - headers = "hello", - } - }) - - assert.res_status(502, res) - local b = assert.response(res).has.jsonbody() - assert.equal("Bad Gateway", b.message) - end) - - it("returns HTTP 502 when 'body' property of custom response is not a string", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda11.com", - ["Content-Type"] = "application/json", - }, - body = { - statusCode = 201, - body = 1234, - } - }) - - assert.res_status(502, res) - local b = assert.response(res).has.jsonbody() - assert.equal("Bad Gateway", b.message) - end) - - it("returns HTTP 502 with when response from lambda is not valid JSON", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda12.com", - } - }) - - assert.res_status(502, res) - local b = assert.response(res).has.jsonbody() - assert.equal("Bad Gateway", b.message) - end) - - it("returns HTTP 502 on empty response from Lambda", function() - local res = assert(proxy_client:send { - method = "POST", - path = "/post", - headers = { - ["Host"] = "lambda13.com", - } - }) - - assert.res_status(502, res) - local b = assert.response(res).has.jsonbody() - assert.equal("Bad Gateway", b.message) - end) - - it("invokes a Lambda function with GET using serviceless route", function() - local res = assert(proxy_client:send { - method = "GET", - path = "/get?key1=some_value1&key2=some_value2&key3=some_value3", - headers = { - ["Host"] = "lambda14.com" - } - }) - assert.res_status(200, res) - local body = assert.response(res).has.jsonbody() - assert.is_string(res.headers["x-amzn-RequestId"]) - assert.equal("some_value1", body.key1) - assert.is_nil(res.headers["X-Amz-Function-Error"]) - end) - end) - end) -end diff --git a/spec/03-plugins/22-aws-lambda/02-schema_spec.lua b/spec/03-plugins/22-aws-lambda/02-schema_spec.lua deleted file mode 100644 index bbda7062ba34..000000000000 --- a/spec/03-plugins/22-aws-lambda/02-schema_spec.lua +++ /dev/null @@ -1,55 +0,0 @@ -local schema_def = require "kong.plugins.aws-lambda.schema" -local validate_plugin_config_schema = require("spec.helpers").validate_plugin_config_schema - - -local kong = { - table = require("kong.pdk.table").new() -} - - -local DEFAULTS = { - timeout = 60000, - keepalive = 60000, - aws_key = "my-key", - aws_secret = "my-secret", - aws_region = "us-east-1", - function_name = "my-function", - invocation_type = "RequestResponse", - log_type = "Tail", - port = 443, -} - - -local function v(config) - return validate_plugin_config_schema( - kong.table.merge(DEFAULTS, config), - schema_def - ) -end - - -describe("Plugin: AWS Lambda (schema)", function() - it("accepts nil Unhandled Response Status Code", function() - local ok, err = v({ unhandled_status = nil }) - assert.truthy(ok) - assert.is_nil(err) - end) - - it("accepts correct Unhandled Response Status Code", function() - local ok, err = v({ unhandled_status = 412 }) - assert.truthy(ok) - assert.is_nil(err) - end) - - it("errors with Unhandled Response Status Code less than 100", function() - local ok, err = v({ unhandled_status = 99 }) - assert.falsy(ok) - assert.equal("value should be between 100 and 999", err.config.unhandled_status) - end) - - it("errors with Unhandled Response Status Code greater than 999", function() - local ok, err = v({ unhandled_status = 1000 }) - assert.falsy(ok) - assert.equal("value should be between 100 and 999", err.config.unhandled_status) - end) -end)