Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(plugins): grpc-web, grpc-gateway: TE trailers #14042

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
message: "**grpc-web** and **grpc-gateway**: Fixed a bug where the `TE` (transfer-encoding) header would not be sent to the upstream gRPC servers when `grpc-web` or `grpc-gateweay` are in use."
type: bugfix
scope: Plugin
18 changes: 14 additions & 4 deletions kong/pdk/service/request.lua
Original file line number Diff line number Diff line change
Expand Up @@ -313,14 +313,24 @@ local function new(self)
-- kong.service.request.set_header("X-Foo", "value")
request.set_header = function(header, value)
check_phase(access_rewrite_balancer)

validate_header(header, value)

if string_lower(header) == "host" then
local header_lower = string_lower(header)

if header_lower == "host" then
ngx_var.upstream_host = value
end
return

elseif header_lower == "te" then
if (ngx_var.upstream_scheme == "grpc" or
ngx_var.upstream_scheme == "grpcs") and value ~= "trailers" then
return nil, "grpc requires TE to be set to trailers"
end

ngx.var.upstream_te = value
return

if string_lower(header) == ":authority" then
elseif header_lower == ":authority" then
if ngx_var.upstream_scheme == "grpc" or
ngx_var.upstream_scheme == "grpcs"
then
Expand Down
73 changes: 71 additions & 2 deletions spec/03-plugins/28-grpc-gateway/01-proxy_spec.lua
Original file line number Diff line number Diff line change
Expand Up @@ -48,10 +48,50 @@ for _, strategy in helpers.each_strategy() do
},
})

assert(helpers.start_kong {
local mock_grpc_service = assert(bp.services:insert {
name = "mock_grpc_service",
url = "http://localhost:8765",
})

local mock_grpc_route = assert(bp.routes:insert {
protocols = { "http" },
hosts = { "grpc_mock.example" },
service = mock_grpc_service,
preserve_host = true,
})

assert(bp.plugins:insert {
route = mock_grpc_route,
name = "grpc-gateway",
config = {
proto = "./spec/fixtures/grpc/targetservice.proto",
},
})

local fixtures = {
http_mock = {}
}
fixtures.http_mock.my_server_block = [[
server {
server_name myserver;
listen 8765;

location ~ / {
content_by_lua_block {
local headers = ngx.req.get_headers()
ngx.header.content_type = "application/grpc"
ngx.header.received_host = headers["Host"]
ngx.header.received_te = headers["te"]
}
}
}
]]

assert(helpers.start_kong({
database = strategy,
plugins = "bundled,grpc-gateway",
})
nginx_conf = "spec/fixtures/custom_nginx.template",
}, nil, nil, fixtures))
end)

before_each(function()
Expand All @@ -63,6 +103,35 @@ for _, strategy in helpers.each_strategy() do
helpers.stop_grpc_target()
end)

test("Sets 'TE: trailers'", function()
local res, err = proxy_client:post("/v1/echo", {
headers = {
["Host"] = "grpc_mock.example",
["Content-Type"] = "application/json",
},
})

assert.equal("trailers", res.headers["received-te"])
assert.is_nil(err)
end)

test("Ignores user-agent TE", function()
-- in grpc-gateway, kong acts as a grpc client on behalf of the client
-- (which generally is a web-browser); as such, the Te header must be
-- set by kong, which will append trailers to the response body
local res, err = proxy_client:post("/v1/echo", {
headers = {
["Host"] = "grpc_mock.example",
["Content-Type"] = "application/json",
["TE"] = "chunked",
},
})

assert.equal("trailers", res.headers["received-te"])
assert.is_nil(err)
end)


test("main entrypoint", function()
local res, err = proxy_client:get("/v1/messages/john_doe")

Expand Down
70 changes: 68 additions & 2 deletions spec/03-plugins/32-grpc-web/01-proxy_spec.lua
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,25 @@ for _, strategy in helpers.each_strategy() do
service = service1,
})

local mock_grpc_service = assert(bp.services:insert {
name = "mock_grpc_service",
url = "http://localhost:8765",
})

local mock_grpc_route = assert(bp.routes:insert {
protocols = { "http" },
hosts = { "grpc_mock.example" },
service = mock_grpc_service,
preserve_host = true,
})

assert(bp.plugins:insert {
route = mock_grpc_route,
name = "grpc-web",
config = {
},
})

assert(bp.plugins:insert {
route = route1,
name = "grpc-web",
Expand All @@ -66,10 +85,30 @@ for _, strategy in helpers.each_strategy() do
},
})

assert(helpers.start_kong {
local fixtures = {
http_mock = {}
}
fixtures.http_mock.my_server_block = [[
server {
server_name myserver;
listen 8765;

location ~ / {
content_by_lua_block {
local headers = ngx.req.get_headers()
ngx.header.content_type = "application/grpc"
ngx.header.received_host = headers["Host"]
ngx.header.received_te = headers["te"]
}
}
}
]]

assert(helpers.start_kong({
database = strategy,
plugins = "bundled,grpc-web",
})
nginx_conf = "spec/fixtures/custom_nginx.template",
}, nil, nil, fixtures))
end)

before_each(function()
Expand All @@ -81,6 +120,33 @@ for _, strategy in helpers.each_strategy() do
helpers.stop_kong()
end)

test("Sets 'TE: trailers'", function()
local res, err = proxy_client:post("/", {
headers = {
["Host"] = "grpc_mock.example",
["Content-Type"] = "application/grpc-web-text",
},
})

assert.equal("trailers", res.headers["received-te"])
assert.is_nil(err)
end)

test("Ignores user-agent TE", function()
-- in grpc-web, kong acts as a grpc client on behalf of the client
-- (which generally is a web-browser); as such, the Te header must be
-- set by kong, which will append trailers to the response body
local res, err = proxy_client:post("/", {
headers = {
["Host"] = "grpc_mock.example",
["Content-Type"] = "application/grpc-web-text",
["TE"] = "chunked",
},
})

assert.equal("trailers", res.headers["received-te"])
assert.is_nil(err)
end)

test("Call gRCP-base64 via HTTP", function()
local res, err = proxy_client:post("/hello.HelloService/SayHello", {
Expand Down
Loading