tenant_router.lua 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. local redis = require "resty.redis"
  2. -- 定义Redis连接参数
  3. local REDIS_HOST = "redis"
  4. local REDIS_PORT = 6379
  5. local REDIS_PASSWORD = "secret_redis"
  6. local REDIS_TIMEOUT = 1000 -- 1秒超时
  7. local _M = {}
  8. -- 设置后端upstream
  9. function set_upstream(tenant, env)
  10. if env == "prod" then
  11. ngx.var.upstream = "backend_prod"
  12. elseif env == "dev" then
  13. ngx.var.upstream = "backend_dev"
  14. else
  15. ngx.log(ngx.ERR, "Upstream not found for env: " .. tostring(env))
  16. return ngx.exit(ngx.HTTP_BAD_REQUEST) -- 使用标准HTTP状态码
  17. end
  18. ngx.log(ngx.NOTICE, "Backend routing: ", "tenant=", tenant, ", env=", env, ", upstream=", ngx.var.upstream)
  19. end
  20. -- 设置前端upstream
  21. function set_frontend_upstream(tenant, env)
  22. if env == "prod" then
  23. ngx.var.frontend_upstream = "frontend_prod"
  24. elseif env == "dev" then
  25. ngx.var.frontend_upstream = "frontend_dev"
  26. else
  27. ngx.log(ngx.ERR, "Frontend upstream not found for env: " .. tostring(env))
  28. return ngx.exit(ngx.HTTP_BAD_REQUEST) -- 使用标准HTTP状态码
  29. end
  30. ngx.log(ngx.NOTICE, "Frontend routing: ", "tenant=", tenant, ", env=", env, ", frontend_upstream=", ngx.var.frontend_upstream)
  31. end
  32. -- 根据租户信息获取环境变量
  33. function get_env(tenant)
  34. local cache_key = "tenant_identification_location_" .. tenant
  35. local red = redis:new()
  36. red:set_timeouts(REDIS_TIMEOUT, REDIS_TIMEOUT, REDIS_TIMEOUT)
  37. -- 连接到Redis
  38. local ok, connect_err = red:connect(REDIS_HOST, REDIS_PORT)
  39. if not ok then
  40. ngx.log(ngx.ERR, "Redis connect failed: ", connect_err)
  41. return ngx.exit(ngx.HTTP_INTERNAL_SERVER_ERROR) -- 使用标准HTTP状态码
  42. end
  43. -- 认证Redis
  44. local ok, auth_err = red:auth(REDIS_PASSWORD)
  45. if not ok then
  46. ngx.log(ngx.ERR, "Failed to authenticate Redis: ", auth_err)
  47. return ngx.exit(ngx.HTTP_INTERNAL_SERVER_ERROR)
  48. end
  49. -- 选择数据库 0(默认数据库)
  50. local ok, err = red:select(0)
  51. if not ok then
  52. ngx.log(ngx.ERR, "Failed to select Redis database: ", err)
  53. return ngx.exit(ngx.HTTP_INTERNAL_SERVER_ERROR)
  54. end
  55. -- 从Redis中获取环境变量
  56. local env = red:get(cache_key)
  57. if not env then
  58. ngx.log(ngx.ERR, "No environment found for tenant key: ", cache_key)
  59. red:set_keepalive(10000, 100) -- 归还连接到连接池
  60. return ngx.exit(ngx.HTTP_INTERNAL_SERVER_ERROR) -- 使用标准HTTP状态码
  61. end
  62. red:set_keepalive(10000, 100)
  63. return env
  64. end
  65. -- 处理API请求的路由逻辑
  66. function _M.route_by_tenant_for_api()
  67. local tenant = ngx.req.get_headers()["X-Tenant"] or ngx.req.get_uri_args()["tenant"]
  68. if not tenant or tenant == "" then
  69. ngx.log(ngx.ERR, "Missing tenant identifier")
  70. return ngx.exit(ngx.HTTP_BAD_REQUEST) -- 使用标准HTTP状态码
  71. end
  72. local env = get_env(tenant)
  73. set_upstream(tenant, env)
  74. end
  75. -- 处理前端请求的路由逻辑
  76. function _M.route_by_tenant_for_frontend()
  77. local tenant = ngx.req.get_headers()["X-Tenant"] or ngx.req.get_uri_args()["tenant"]
  78. if not tenant or tenant == "" then
  79. ngx.log(ngx.ERR, "Missing tenant identifier")
  80. return ngx.exit(ngx.HTTP_BAD_REQUEST) -- 使用标准HTTP状态码
  81. end
  82. local env = get_env(tenant)
  83. set_frontend_upstream(env)
  84. end
  85. return _M