| # |
| # Licensed to the Apache Software Foundation (ASF) under one or more |
| # contributor license agreements. See the NOTICE file distributed with |
| # this work for additional information regarding copyright ownership. |
| # The ASF licenses this file to You under the Apache License, Version 2.0 |
| # (the "License"); you may not use this file except in compliance with |
| # the License. You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| # |
| use t::APISIX 'no_plan'; |
| |
| repeat_each(1); |
| no_long_string(); |
| no_root_location(); |
| no_shuffle(); |
| log_level("info"); |
| |
| run_tests; |
| |
| __DATA__ |
| |
| === TEST 1: set upstream(id: 1) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local etcd = require("apisix.core.etcd") |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "set" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| |
| local res = assert(etcd.get('/upstreams/1')) |
| local create_time = res.body.node.value.create_time |
| assert(create_time ~= nil, "create_time is nil") |
| local update_time = res.body.node.value.update_time |
| assert(update_time ~= nil, "update_time is nil") |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 2: get upstream(id: 1) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_GET, |
| nil, |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "get" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 3: delete upstream(id: 1) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, message = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_DELETE, |
| nil, |
| [[{ |
| "action": "delete" |
| }]] |
| ) |
| ngx.say("[delete] code: ", code, " message: ", message) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| [delete] code: 200 message: passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 4: delete upstream(id: not_found) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code = t('/apisix/admin/upstreams/not_found', |
| ngx.HTTP_DELETE, |
| nil, |
| [[{ |
| "action": "delete" |
| }]] |
| ) |
| |
| ngx.say("[delete] code: ", code) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| [delete] code: 404 |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 5: push upstream + delete |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local etcd = require("apisix.core.etcd") |
| local code, message, res = t('/apisix/admin/upstreams', |
| ngx.HTTP_POST, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| } |
| }, |
| "action": "create" |
| }]] |
| ) |
| |
| if code ~= 200 then |
| ngx.status = code |
| ngx.say(message) |
| return |
| end |
| |
| ngx.say("[push] code: ", code, " message: ", message) |
| |
| local id = string.sub(res.node.key, #"/apisix/upstreams/" + 1) |
| local res = assert(etcd.get('/upstreams/' .. id)) |
| local create_time = res.body.node.value.create_time |
| assert(create_time ~= nil, "create_time is nil") |
| local update_time = res.body.node.value.update_time |
| assert(update_time ~= nil, "update_time is nil") |
| |
| code, message = t('/apisix/admin/upstreams/' .. id, |
| ngx.HTTP_DELETE, |
| nil, |
| [[{ |
| "action": "delete" |
| }]] |
| ) |
| ngx.say("[delete] code: ", code, " message: ", message) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| [push] code: 200 message: passed |
| [delete] code: 200 message: passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 6: empty nodes |
| --- config |
| location /t { |
| content_by_lua_block { |
| local core = require("apisix.core") |
| local t = require("lib.test_admin").test |
| local code, message, res = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": {}, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| if code ~= 200 then |
| ngx.status = code |
| ngx.print(message) |
| return |
| end |
| |
| ngx.say("[push] code: ", code, " message: ", message) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"nodes\" validation failed: object matches none of the requireds"} |
| |
| |
| |
| === TEST 7: invalid upstream id in uri |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/invalid_id$', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| ngx.exit(code) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 8: different id |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": 3, |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"wrong upstream id"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 9: id in the rule |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": "1", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "set" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 10: integer id less than 1 |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": -100, |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"id\" validation failed: object matches none of the requireds"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 11: invalid upstream id: string value |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": "invalid_id$", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"id\" validation failed: object matches none of the requireds"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 12: no additional properties is valid |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": 1, |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "invalid_property": "/index.html" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: additional properties forbidden, found invalid_property"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 13: set upstream(type: chash) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "remote_addr", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "key": "remote_addr", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "set" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 14: invalid type |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": 1, |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "invalid_type" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"type\" validation failed: matches non of the enum values"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 15: invalid weight of node |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": 1, |
| "nodes": { |
| "127.0.0.1:8080": "1" |
| }, |
| "type": "chash" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"nodes\" validation failed: object matches none of the requireds"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 16: invalid weight of node |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_PUT, |
| [[{ |
| "id": 1, |
| "nodes": { |
| "127.0.0.1:8080": -100 |
| }, |
| "type": "chash" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"nodes\" validation failed: object matches none of the requireds"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 17: set upstream (missing key) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"missing key"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 18: wrong upstream id, do not need it |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_POST, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"wrong upstream id, do not need it"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 19: wrong upstream id, do not need it |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams', |
| ngx.HTTP_POST, |
| [[{ |
| "id": 1, |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"wrong upstream id, do not need it"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 20: patch upstream(whole) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local etcd = require("apisix.core.etcd") |
| |
| local id = 1 |
| local res = assert(etcd.get('/upstreams/' .. id)) |
| local prev_create_time = res.body.node.value.create_time |
| local prev_update_time = res.body.node.value.update_time |
| ngx.sleep(1) |
| |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| |
| local res = assert(etcd.get('/upstreams/' .. id)) |
| local create_time = res.body.node.value.create_time |
| assert(prev_create_time == create_time, "create_time mismatched") |
| local update_time = res.body.node.value.update_time |
| assert(prev_update_time ~= update_time, "update_time should be changed") |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 21: patch upstream(new desc) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "desc": "new 21 upstream" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new 21 upstream" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 22: patch upstream(new nodes) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8081": 3, |
| "127.0.0.1:8082": 4 |
| } |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1, |
| "127.0.0.1:8081": 3, |
| "127.0.0.1:8082": 4 |
| }, |
| "type": "roundrobin", |
| "desc": "new 21 upstream" |
| } |
| } |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 23: patch upstream(weight is 0) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8081": 3, |
| "127.0.0.1:8082": 0 |
| } |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8081": 3, |
| "127.0.0.1:8082": 0 |
| }, |
| "type": "roundrobin", |
| "desc": "new 21 upstream" |
| } |
| } |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 24: patch upstream(whole - sub path) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1/', |
| ngx.HTTP_PATCH, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream 24" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream 24" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 25: patch upstream(new desc - sub path) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1/desc', |
| ngx.HTTP_PATCH, |
| '"new 25 upstream"', |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new 25 upstream" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 26: patch upstream(new nodes) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1/nodes', |
| ngx.HTTP_PATCH, |
| [[{ |
| "127.0.0.6:8081": 3, |
| "127.0.0.7:8082": 4 |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.6:8081": 3, |
| "127.0.0.7:8082": 4 |
| }, |
| "type": "roundrobin", |
| "desc": "new 25 upstream" |
| } |
| } |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 27: patch upstream(weight is 0 - sub path) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1/nodes', |
| ngx.HTTP_PATCH, |
| [[{ |
| "127.0.0.7:8081": 0, |
| "127.0.0.8:8082": 4 |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.7:8081": 0, |
| "127.0.0.8:8082": 4 |
| }, |
| "type": "roundrobin", |
| "desc": "new 25 upstream" |
| } |
| } |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 28: set upstream(type: chash) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "server_name", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 29: wrong upstream key, hash_on default vars |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1, |
| "127.0.0.1:8081": 2 |
| }, |
| "type": "chash", |
| "key": "not_support", |
| "desc": "new upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: failed to match pattern \"^((uri|server_name|server_addr|request_uri|remote_port|remote_addr|query_string|host|hostname)|arg_[0-9a-zA-z_-]+)$\" with \"not_support\""} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 30: set upstream with args(type: chash) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "arg_device_id", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 31: set upstream(type: chash) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "server_name", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 32: wrong upstream key, hash_on default vars |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1, |
| "127.0.0.1:8081": 2 |
| }, |
| "type": "chash", |
| "key": "not_support", |
| "desc": "new upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: failed to match pattern \"^((uri|server_name|server_addr|request_uri|remote_port|remote_addr|query_string|host|hostname)|arg_[0-9a-zA-z_-]+)$\" with \"not_support\""} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 33: set upstream with args(type: chash) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "arg_device_id", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 34: type chash, hash_on: vars |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "arg_device_id", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "vars", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 35: type chash, hash_on: header, header name with '_', underscores_in_headers on |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "custom_header", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "header", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 36: type chash, hash_on: header, header name with invalid character |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "$#^@", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "header", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: failed to match pattern \"^[a-zA-Z0-9-_]+$\" with \"$#^@\""} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 37: type chash, hash_on: cookie |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "custom_cookie", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "cookie", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 38: type chash, hash_on: cookie, cookie name with invalid character |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "$#^@abc", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "cookie", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: failed to match pattern \"^[a-zA-Z0-9-_]+$\" with \"$#^@abc\""} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 39: type chash, hash_on: consumer, do not need upstream key |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "consumer", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 40: type chash, hash_on: consumer, set key but invalid |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "consumer", |
| "key": "invalid-key", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 41: type chash, invalid hash_on type |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "key": "dsadas", |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "chash", |
| "hash_on": "aabbcc", |
| "desc": "new chash upstream" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"hash_on\" validation failed: matches non of the enum values"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 42: set upstream(id: 1 + name: test name) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "name": "test upstream name" |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "name": "test upstream name" |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "set" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 43: string id |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/a-b-c-ABC_0123', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| if code >= 300 then |
| ngx.status = code |
| end |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 44: string id(delete) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/a-b-c-ABC_0123', |
| ngx.HTTP_DELETE |
| ) |
| if code >= 300 then |
| ngx.status = code |
| end |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 45: invalid string id |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/*invalid', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin" |
| }]] |
| ) |
| if code >= 300 then |
| ngx.status = code |
| end |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"id\" validation failed: object matches none of the requireds"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 46: retries is 0 |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/a-b-c-ABC_0123', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1, |
| "127.0.0.1:8090": 1 |
| }, |
| "retries": 0, |
| "type": "roundrobin" |
| }]] |
| ) |
| if code >= 300 then |
| ngx.status = code |
| end |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 47: retries is -1 (INVALID) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/a-b-c-ABC_0123', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1, |
| "127.0.0.1:8090": 1 |
| }, |
| "retries": -1, |
| "type": "roundrobin" |
| }]] |
| ) |
| if code >= 300 then |
| ngx.status = code |
| end |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"retries\" validation failed: expected -1 to be greater than 0"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 48: invalid route: multi nodes with `node` mode to pass host |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "httpbin.org:8080": 1, |
| "test.com:8080": 1 |
| }, |
| "type": "roundrobin", |
| "pass_host": "node" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 49: invalid route: empty `upstream_host` when `pass_host` is `rewrite` |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "httpbin.org:8080": 1, |
| "test.com:8080": 1 |
| }, |
| "type": "roundrobin", |
| "pass_host": "rewrite", |
| "upstream_host": "" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 50: set upstream(with labels) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "labels": { |
| "build":"16", |
| "env":"production", |
| "version":"v2" |
| } |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "labels": { |
| "build":"16", |
| "env":"production", |
| "version":"v2" |
| } |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "set" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 51: get upstream(with labels) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_GET, |
| nil, |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "labels": { |
| "version":"v2", |
| "build":"16", |
| "env":"production" |
| } |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "get" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 52: patch upstream(only labels) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "labels": { |
| "build": "17" |
| } |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "labels": { |
| "version":"v2", |
| "build":"17", |
| "env":"production" |
| } |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 53: invalid format of label value: set upstream |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "labels": { |
| "env": ["production", "release"] |
| } |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.print(body) |
| } |
| } |
| --- request |
| GET /t |
| --- error_code: 400 |
| --- response_body |
| {"error_msg":"invalid configuration: property \"labels\" validation failed: failed to validate env (matching \".*\"): wrong type: expected string, got table"} |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 54: patch upstream(whole, create_time) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local etcd = require("apisix.core.etcd") |
| |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream", |
| "create_time": 1705252779 |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream", |
| "create_time": 1705252779 |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| |
| if code >= 300 then |
| return |
| end |
| |
| local res = assert(etcd.get('/upstreams/1')) |
| local create_time = res.body.node.value.create_time |
| assert(create_time == 1705252779, "create_time mismatched") |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 55: patch upstream(whole, update_time) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local etcd = require("apisix.core.etcd") |
| |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PATCH, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream", |
| "update_time": 1705252779 |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "desc": "new upstream", |
| "create_time": 1705252779 |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "compareAndSwap" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| |
| if code >= 300 then |
| return |
| end |
| |
| local res = assert(etcd.get('/upstreams/1')) |
| local update_time = res.body.node.value.update_time |
| assert(update_time == 1705252779, "update_time mismatched") |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 56: create upstream with create_time and update_time(id: 1) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, body = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_PUT, |
| [[{ |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "create_time": 1602883670, |
| "update_time": 1602893670 |
| }]], |
| [[{ |
| "node": { |
| "value": { |
| "nodes": { |
| "127.0.0.1:8080": 1 |
| }, |
| "type": "roundrobin", |
| "create_time": 1602883670, |
| "update_time": 1602893670 |
| }, |
| "key": "/apisix/upstreams/1" |
| }, |
| "action": "set" |
| }]] |
| ) |
| |
| ngx.status = code |
| ngx.say(body) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| passed |
| --- no_error_log |
| [error] |
| |
| |
| |
| === TEST 57: delete test upstream(id: 1) |
| --- config |
| location /t { |
| content_by_lua_block { |
| local t = require("lib.test_admin").test |
| local code, message = t('/apisix/admin/upstreams/1', |
| ngx.HTTP_DELETE, |
| nil, |
| [[{ |
| "action": "delete" |
| }]] |
| ) |
| ngx.say("[delete] code: ", code, " message: ", message) |
| } |
| } |
| --- request |
| GET /t |
| --- response_body |
| [delete] code: 200 message: passed |
| --- no_error_log |
| [error] |