blob: 9933a8d1bb39f057696b8cc840e7be4b07a6980b [file] [log] [blame]
// 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.
//
// Date 2014/10/24 16:44:30
#include <gtest/gtest.h>
#include <google/protobuf/descriptor.h>
#include "brpc/server.h"
#include "brpc/details/http_message.h"
#include "brpc/policy/http_rpc_protocol.h"
#include "echo.pb.h"
namespace brpc {
DECLARE_bool(allow_chunked_length);
DECLARE_bool(allow_http_1_1_request_without_host);
int main(int argc, char* argv[]) {
testing::InitGoogleTest(&argc, argv);
GFLAGS_NAMESPACE::ParseCommandLineFlags(&argc, &argv, true);
brpc::FLAGS_allow_http_1_1_request_without_host = true;
return RUN_ALL_TESTS();
}
namespace policy {
Server::MethodProperty*
FindMethodPropertyByURI(const std::string& uri_path, const Server* server,
std::string* unknown_method_str);
bool ParseHttpServerAddress(butil::EndPoint *point, const char *server_addr_and_port);
}}
namespace {
using brpc::policy::FindMethodPropertyByURI;
using brpc::policy::ParseHttpServerAddress;
TEST(HttpMessageTest, http_method) {
ASSERT_STREQ("DELETE", brpc::HttpMethod2Str(brpc::HTTP_METHOD_DELETE));
ASSERT_STREQ("GET", brpc::HttpMethod2Str(brpc::HTTP_METHOD_GET));
ASSERT_STREQ("POST", brpc::HttpMethod2Str(brpc::HTTP_METHOD_POST));
ASSERT_STREQ("PUT", brpc::HttpMethod2Str(brpc::HTTP_METHOD_PUT));
brpc::HttpMethod m;
ASSERT_TRUE(brpc::Str2HttpMethod("DELETE", &m));
ASSERT_EQ(brpc::HTTP_METHOD_DELETE, m);
ASSERT_TRUE(brpc::Str2HttpMethod("GET", &m));
ASSERT_EQ(brpc::HTTP_METHOD_GET, m);
ASSERT_TRUE(brpc::Str2HttpMethod("POST", &m));
ASSERT_EQ(brpc::HTTP_METHOD_POST, m);
ASSERT_TRUE(brpc::Str2HttpMethod("PUT", &m));
ASSERT_EQ(brpc::HTTP_METHOD_PUT, m);
// case-insensitive
ASSERT_TRUE(brpc::Str2HttpMethod("DeLeTe", &m));
ASSERT_EQ(brpc::HTTP_METHOD_DELETE, m);
ASSERT_TRUE(brpc::Str2HttpMethod("get", &m));
ASSERT_EQ(brpc::HTTP_METHOD_GET, m);
// non-existed
ASSERT_FALSE(brpc::Str2HttpMethod("DEL", &m));
ASSERT_FALSE(brpc::Str2HttpMethod("DELETE ", &m));
ASSERT_FALSE(brpc::Str2HttpMethod("GOT", &m));
}
TEST(HttpMessageTest, eof) {
GFLAGS_NAMESPACE::SetCommandLineOption("verbose", "100");
const char* http_request =
"GET /CloudApiControl/HttpServer/telematics/v3/weather?location=%E6%B5%B7%E5%8D%97%E7%9C%81%E7%9B%B4%E8%BE%96%E5%8E%BF%E7%BA%A7%E8%A1%8C%E6%94%BF%E5%8D%95%E4%BD%8D&output=json&ak=0l3FSP6qA0WbOzGRaafbmczS HTTP/1.1\r\n"
"X-Host: api.map.baidu.com\r\n"
"X-Forwarded-Proto: http\r\n"
"Host: api.map.baidu.com\r\n"
"User-Agent: IME/Android/4.4.2/N80.QHD.LT.X10.V3/N80.QHD.LT.X10.V3.20150812.031915\r\n"
"Accept: application/json\r\n"
"Accept-Charset: UTF-8,*;q=0.5\r\n"
"Accept-Encoding: deflate,sdch\r\n"
"Accept-Language: zh-CN,en-US;q=0.8,zh;q=0.6\r\n"
"Bfe-Atk: NORMAL_BROWSER\r\n"
"Bfe_logid: 8767802212038413243\r\n"
"Bfeip: 10.26.124.40\r\n"
"CLIENTIP: 119.29.102.26\r\n"
"CLIENTPORT: 59863\r\n"
"Cache-Control: max-age=0\r\n"
"Content-Type: application/json;charset=utf8\r\n"
"X-Forwarded-For: 119.29.102.26\r\n"
"X-Forwarded-Port: 59863\r\n"
"X-Ime-Imei: 35629601890905\r\n"
"X_BD_LOGID: 3959476981\r\n"
"X_BD_LOGID64: 16815814797661447369\r\n"
"X_BD_PRODUCT: map\r\n"
"X_BD_SUBSYS: apimap\r\n";
butil::IOBuf buf;
buf.append(http_request);
brpc::HttpMessage http_message;
ASSERT_EQ((ssize_t)buf.size(), http_message.ParseFromIOBuf(buf));
ASSERT_EQ(2, http_message.ParseFromArray("\r\n", 2));
ASSERT_TRUE(http_message.Completed());
}
TEST(HttpMessageTest, request_sanity) {
const char *http_request =
"POST /path/file.html?sdfsdf=sdfs&sldf1=sdf HTTP/12.34\r\n"
"From: someuser@jmarshall.com\r\n"
"User-Agent: HTTPTool/1.0 \r\n" // intended ending spaces
"Content-Type: json\r\n"
"Content-Length: 19\r\n"
"Log-ID: 456\r\n"
"Host: myhost\r\n"
"Correlation-ID: 123\r\n"
"Authorization: test\r\n"
"Accept: */*\r\n"
"\r\n"
"Message Body sdfsdf\r\n"
;
brpc::HttpMessage http_message;
ASSERT_EQ((ssize_t)strlen(http_request),
http_message.ParseFromArray(http_request, strlen(http_request)));
const brpc::HttpHeader& header = http_message.header();
// Check all keys
ASSERT_EQ("json", header.content_type());
ASSERT_TRUE(header.GetHeader("HOST"));
ASSERT_EQ("myhost", *header.GetHeader("host"));
ASSERT_TRUE(header.GetHeader("CORRELATION-ID"));
ASSERT_EQ("123", *header.GetHeader("CORRELATION-ID"));
ASSERT_TRUE(header.GetHeader("User-Agent"));
ASSERT_EQ("HTTPTool/1.0 ", *header.GetHeader("User-Agent"));
ASSERT_TRUE(header.GetHeader("Host"));
ASSERT_EQ("myhost", *header.GetHeader("Host"));
ASSERT_TRUE(header.GetHeader("Accept"));
ASSERT_EQ("*/*", *header.GetHeader("Accept"));
ASSERT_EQ(1, header.major_version());
ASSERT_EQ(34, header.minor_version());
ASSERT_EQ(brpc::HTTP_METHOD_POST, header.method());
ASSERT_EQ(brpc::HTTP_STATUS_OK, header.status_code());
ASSERT_STREQ("OK", header.reason_phrase());
ASSERT_TRUE(header.GetHeader("log-id"));
ASSERT_EQ("456", *header.GetHeader("log-id"));
ASSERT_TRUE(NULL != header.GetHeader("Authorization"));
ASSERT_EQ("test", *header.GetHeader("Authorization"));
}
TEST(HttpMessageTest, response_sanity) {
const char *http_response =
"HTTP/12.34 410 GoneBlah\r\n"
"From: someuser@jmarshall.com\r\n"
"User-Agent: HTTPTool/1.0 \r\n" // intended ending spaces
"Content-Type: json2\r\n"
"Content-Length: 19\r\n"
"Log-ID: 456\r\n"
"Host: myhost\r\n"
"Correlation-ID: 123\r\n"
"Authorization: test\r\n"
"Accept: */*\r\n"
"\r\n"
"Message Body sdfsdf\r\n"
;
brpc::HttpMessage http_message;
ASSERT_EQ((ssize_t)strlen(http_response),
http_message.ParseFromArray(http_response, strlen(http_response)));
// Check all keys
const brpc::HttpHeader& header = http_message.header();
ASSERT_EQ("json2", header.content_type());
ASSERT_TRUE(header.GetHeader("HOST"));
ASSERT_EQ("myhost", *header.GetHeader("host"));
ASSERT_TRUE(header.GetHeader("CORRELATION-ID"));
ASSERT_EQ("123", *header.GetHeader("CORRELATION-ID"));
ASSERT_TRUE(header.GetHeader("User-Agent"));
ASSERT_EQ("HTTPTool/1.0 ", *header.GetHeader("User-Agent"));
ASSERT_TRUE(header.GetHeader("Host"));
ASSERT_EQ("myhost", *header.GetHeader("Host"));
ASSERT_TRUE(header.GetHeader("Accept"));
ASSERT_EQ("*/*", *header.GetHeader("Accept"));
ASSERT_EQ(1, header.major_version());
ASSERT_EQ(34, header.minor_version());
// method is undefined for response, in our case, it's set to 0.
ASSERT_EQ(brpc::HTTP_METHOD_DELETE, header.method());
ASSERT_EQ(brpc::HTTP_STATUS_GONE, header.status_code());
ASSERT_STREQ(brpc::HttpReasonPhrase(header.status_code()), /*not GoneBlah*/
header.reason_phrase());
ASSERT_TRUE(header.GetHeader("log-id"));
ASSERT_EQ("456", *header.GetHeader("log-id"));
ASSERT_TRUE(header.GetHeader("Authorization"));
ASSERT_EQ("test", *header.GetHeader("Authorization"));
}
TEST(HttpMessageTest, bad_format) {
const char *http_request =
"slkdjflksdf skldjf\r\n";
brpc::HttpMessage http_message;
ASSERT_EQ(-1, http_message.ParseFromArray(http_request, strlen(http_request)));
}
TEST(HttpMessageTest, incompleted_request_line) {
const char *http_request = "GE" ;
brpc::HttpMessage http_message;
ASSERT_TRUE(http_message.ParseFromArray(http_request, strlen(http_request)) >= 0);
ASSERT_FALSE(http_message.Completed());
}
TEST(HttpMessageTest, parse_from_iobuf) {
const size_t content_length = 8192;
char header[1024];
snprintf(header, sizeof(header),
"GET /service/method?key1=value1&key2=value2&key3=value3 HTTP/1.1\r\n"
"Content-Type: text/plain\r\n"
"Content-Length: %lu\r\n"
"\r\n",
content_length);
butil::IOBuf content;
for (size_t i = 0; i < content_length; ++i) {
content.push_back('2');
}
butil::IOBuf request;
request.append(header);
request.append(content);
brpc::HttpMessage http_message;
ASSERT_TRUE(http_message.ParseFromIOBuf(request) >= 0);
ASSERT_TRUE(http_message.Completed());
ASSERT_EQ(content, http_message.body());
ASSERT_EQ(content, http_message.body().to_string());
ASSERT_EQ("text/plain", http_message.header().content_type());
}
TEST(HttpMessageTest, parse_http_head_response) {
char response1[1024] = "HTTP/1.1 200 OK\r\n"
"Content-Type: text/plain\r\n"
"Content-Length: 1024\r\n"
"\r\n";
butil::IOBuf request;
request.append(response1);
brpc::HttpMessage http_message(false, brpc::HTTP_METHOD_HEAD);
ASSERT_TRUE(http_message.ParseFromIOBuf(request) >= 0);
ASSERT_TRUE(http_message.Completed()) << http_message.stage();
ASSERT_EQ("text/plain", http_message.header().content_type());
const std::string* content_length = http_message.header().GetHeader("Content-Length");
ASSERT_NE(nullptr, content_length);
ASSERT_EQ("1024", *content_length);
char response2[1024] = "HTTP/1.1 200 OK\r\n"
"Content-Type: text/plain\r\n"
"Transfer-Encoding: chunked\r\n"
"\r\n";
butil::IOBuf request2;
request2.append(response2);
brpc::HttpMessage http_message2(false, brpc::HTTP_METHOD_HEAD);
ASSERT_TRUE(http_message2.ParseFromIOBuf(request2) >= 0);
ASSERT_TRUE(http_message2.Completed()) << http_message2.stage();
ASSERT_EQ("text/plain", http_message2.header().content_type());
const std::string* transfer_encoding = http_message2.header().GetHeader("Transfer-Encoding");
ASSERT_NE(nullptr, transfer_encoding);
ASSERT_EQ("chunked", *transfer_encoding);
}
TEST(HttpMessageTest, parse_http_cookie) {
const char* http_request =
"GET /CloudApiControl HTTP/1.1\r\n"
"Host: api.map.baidu.com\r\n"
"Accept: application/json\r\n"
"cookie: a=1\r\n"
"Cookie: b=2\r\n"
"\r\n";
butil::IOBuf buf;
buf.append(http_request);
brpc::HttpMessage http_message;
ASSERT_EQ((ssize_t)buf.size(), http_message.ParseFromIOBuf(buf));
ASSERT_TRUE(http_message.Completed());
const std::string* cookie
= http_message.header().GetHeader("cookie");
ASSERT_NE(nullptr, cookie);
ASSERT_EQ("a=1; b=2", *cookie);
}
TEST(HttpMessageTest, parse_http_set_cookie) {
char response[1024] = "HTTP/1.1 200 OK\r\n"
"Content-Type: text/plain\r\n"
"Content-Length: 1024\r\n"
"set-cookie: a=1\r\n"
"Set-Cookie: b=2\r\n"
"\r\n";
butil::IOBuf request;
request.append(response);
brpc::HttpMessage http_message(false, brpc::HTTP_METHOD_HEAD);
ASSERT_TRUE(http_message.ParseFromIOBuf(request) >= 0);
ASSERT_TRUE(http_message.Completed()) << http_message.stage();
const std::string* set_cookie = http_message.header().GetHeader("set-cookie");
ASSERT_NE(nullptr, set_cookie);
ASSERT_EQ("a=1", *set_cookie);
std::vector<const std::string*> all_set_cookie
= http_message.header().GetAllSetCookieHeader();
for (const std::string* sc : all_set_cookie) {
ASSERT_NE(nullptr, sc);
if (set_cookie == sc) {
ASSERT_EQ("a=1", *sc);
} else {
ASSERT_EQ("b=2", *sc);
}
if (http_message.header().IsSetCookie(*sc)) {
}
}
int set_cookie_value1_count = 0;
int set_cookie_value2_count = 0;
for (auto iter = http_message.header().HeaderBegin();
iter != http_message.header().HeaderEnd(); ++iter) {
if (!http_message.header().IsSetCookie(iter->first)) {
continue;
}
if (iter->second == "b=2") {
++set_cookie_value2_count;
} else if (iter->second == "a=1") {
++set_cookie_value1_count;
}
}
ASSERT_EQ(1, set_cookie_value1_count);
ASSERT_EQ(1, set_cookie_value2_count);
}
TEST(HttpMessageTest, cl_and_te) {
// https://datatracker.ietf.org/doc/html/rfc2616#section-14.41
// If multiple encodings have been applied to an entity, the transfer-
// codings MUST be listed in the order in which they were applied.
const char* request_buf1 = "POST /chunked_w_content_length HTTP/1.1\r\n"
"Content-Length: 10\r\n"
"Transfer-Encoding: gzip,chunked\r\n"
"\r\n"
"5; ilovew3;whattheluck=aretheseparametersfor\r\nhello\r\n"
"6; blahblah; blah\r\n world\r\n"
"0\r\n"
"\r\n";
butil::IOBuf request1;
request1.append(request_buf1);
const char* request_buf2 = "POST /chunked_w_content_length HTTP/1.1\r\n"
"Content-Length: 19\r\n"
"Transfer-Encoding: chunked,gzip\r\n"
"\r\n"
"Message Body sdfsdf";
butil::IOBuf request2;
request2.append(request_buf2);
const char* response_buf1 = "HTTP/1.1 200 OK\r\n"
"Content-Length: 10\r\n"
"Transfer-Encoding: gzip,chunked\r\n"
"\r\n"
"5; ilovew3;whattheluck=aretheseparametersfor\r\nhello\r\n"
"6; blahblah; blah\r\n world\r\n"
"0\r\n"
"\r\n";
butil::IOBuf response1;
response1.append(response_buf1);
const char* response_buf2 = "HTTP/1.1 200 OK\r\n"
"Content-Length: 19\r\n"
"Transfer-Encoding: chunked,gzip\r\n"
"\r\n"
"Message Body sdfsdf";
butil::IOBuf response2;
response2.append(response_buf2);
brpc::FLAGS_allow_chunked_length = false;
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(request1), -1)
<< http_message._parser;
}
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(request2), -1)
<< http_message._parser;
}
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(response1), -1)
<< http_message._parser;
}
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(response2), -1)
<< http_message._parser;
}
brpc::FLAGS_allow_chunked_length = true;
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(request1), request1.size())
<< http_message._parser;
}
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(request2), -1)
<< http_message._parser;
}
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(response1), response1.size())
<< http_message._parser;
}
{
brpc::HttpMessage http_message;
ASSERT_EQ(http_message.ParseFromIOBuf(response2), -1)
<< http_message._parser;
}
}
TEST(HttpMessageTest, find_method_property_by_uri) {
brpc::Server server;
ASSERT_EQ(0, server.AddService(new test::EchoService(),
brpc::SERVER_OWNS_SERVICE));
ASSERT_EQ(0, server.Start(9237, NULL));
std::string unknown_method;
brpc::Server::MethodProperty* mp = NULL;
mp = FindMethodPropertyByURI("", &server, NULL);
ASSERT_TRUE(mp);
ASSERT_EQ("index", mp->method->service()->name());
mp = FindMethodPropertyByURI("/", &server, NULL);
ASSERT_TRUE(mp);
ASSERT_EQ("index", mp->method->service()->name());
mp = FindMethodPropertyByURI("//", &server, NULL);
ASSERT_TRUE(mp);
ASSERT_EQ("index", mp->method->service()->name());
mp = FindMethodPropertyByURI("flags", &server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("flags", mp->method->service()->name());
mp = FindMethodPropertyByURI("/flags/port", &server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("flags", mp->method->service()->name());
ASSERT_EQ("port", unknown_method);
mp = FindMethodPropertyByURI("/flags/foo/bar", &server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("flags", mp->method->service()->name());
ASSERT_EQ("foo/bar", unknown_method);
mp = FindMethodPropertyByURI("/brpc.flags/$*",
&server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("flags", mp->method->service()->name());
ASSERT_EQ("$*", unknown_method);
mp = FindMethodPropertyByURI("EchoService/Echo", &server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("test.EchoService.Echo", mp->method->full_name());
mp = FindMethodPropertyByURI("/EchoService/Echo",
&server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("test.EchoService.Echo", mp->method->full_name());
mp = FindMethodPropertyByURI("/test.EchoService/Echo",
&server, &unknown_method);
ASSERT_TRUE(mp);
ASSERT_EQ("test.EchoService.Echo", mp->method->full_name());
mp = FindMethodPropertyByURI("/test.EchoService/no_such_method",
&server, &unknown_method);
ASSERT_FALSE(mp);
}
TEST(HttpMessageTest, http_header) {
brpc::HttpHeader header;
header.set_version(10, 100);
ASSERT_EQ(10, header.major_version());
ASSERT_EQ(100, header.minor_version());
ASSERT_TRUE(header.content_type().empty());
header.set_content_type("text/plain");
ASSERT_EQ("text/plain", header.content_type());
ASSERT_FALSE(header.GetHeader("content-type"));
header.set_content_type("application/json");
ASSERT_EQ("application/json", header.content_type());
ASSERT_FALSE(header.GetHeader("content-type"));
ASSERT_FALSE(header.GetHeader("key1"));
header.AppendHeader("key1", "value1");
const std::string* value = header.GetHeader("key1");
ASSERT_TRUE(value && *value == "value1");
header.AppendHeader("key1", "value2");
value = header.GetHeader("key1");
ASSERT_TRUE(value && *value == "value1,value2");
header.SetHeader("key1", "value3");
value = header.GetHeader("key1");
ASSERT_TRUE(value && *value == "value3");
header.RemoveHeader("key1");
ASSERT_FALSE(header.GetHeader("key1"));
ASSERT_FALSE(header.GetHeader(brpc::HttpHeader::COOKIE));
header.AppendHeader(brpc::HttpHeader::COOKIE, "value1=1");
value = header.GetHeader(brpc::HttpHeader::COOKIE);
ASSERT_TRUE(value && *value == "value1=1");
header.AppendHeader(brpc::HttpHeader::COOKIE, "value2=2");
value = header.GetHeader(brpc::HttpHeader::COOKIE);
ASSERT_TRUE(value && *value == "value1=1; value2=2");
header.SetHeader(brpc::HttpHeader::COOKIE, "value3");
value = header.GetHeader(brpc::HttpHeader::COOKIE);
ASSERT_TRUE(value && *value == "value3");
header.RemoveHeader(brpc::HttpHeader::COOKIE);
ASSERT_FALSE(header.GetHeader(brpc::HttpHeader::COOKIE));
std::string set_cookie_value1 = "a=1";
std::string set_cookie_value2 = "b=2";
std::string set_cookie_value3 = "c=3";
ASSERT_FALSE(header.GetHeader(brpc::HttpHeader::SET_COOKIE));
header.SetHeader(brpc::HttpHeader::SET_COOKIE, set_cookie_value1);
value = header.GetHeader(brpc::HttpHeader::SET_COOKIE);
ASSERT_TRUE(value && *value == set_cookie_value1);
header.AppendHeader(brpc::HttpHeader::SET_COOKIE, set_cookie_value2);
value = header.GetHeader(brpc::HttpHeader::SET_COOKIE);
ASSERT_TRUE(value && *value == set_cookie_value1);
header.SetHeader(brpc::HttpHeader::SET_COOKIE, set_cookie_value3);
value = header.GetHeader(brpc::HttpHeader::SET_COOKIE);
ASSERT_TRUE(value && *value == set_cookie_value3);
std::vector<const std::string*> all_set_cookie
= header.GetAllSetCookieHeader();
ASSERT_EQ(2u, all_set_cookie.size());
for (const std::string* sc : all_set_cookie) {
ASSERT_TRUE(sc);
ASSERT_TRUE(*sc == set_cookie_value2 || *sc == set_cookie_value3);
}
int set_cookie_value2_count = 0;
int set_cookie_value3_count = 0;
for (auto iter = header.HeaderBegin(); iter != header.HeaderEnd(); ++iter) {
if (!header.IsSetCookie(brpc::HttpHeader::SET_COOKIE)) {
continue;
}
if (iter->second == set_cookie_value2) {
++set_cookie_value2_count;
} else if (iter->second == set_cookie_value3) {
++set_cookie_value3_count;
}
}
ASSERT_EQ(1, set_cookie_value2_count);
ASSERT_EQ(1, set_cookie_value3_count);
header.RemoveHeader(brpc::HttpHeader::SET_COOKIE);
ASSERT_FALSE(header.GetHeader(brpc::HttpHeader::SET_COOKIE));
ASSERT_EQ(header._first_set_cookie, nullptr);
ASSERT_EQ(brpc::HTTP_METHOD_GET, header.method());
header.set_method(brpc::HTTP_METHOD_POST);
ASSERT_EQ(brpc::HTTP_METHOD_POST, header.method());
ASSERT_EQ(brpc::HTTP_STATUS_OK, header.status_code());
ASSERT_STREQ(brpc::HttpReasonPhrase(header.status_code()),
header.reason_phrase());
header.set_status_code(brpc::HTTP_STATUS_CONTINUE);
ASSERT_EQ(brpc::HTTP_STATUS_CONTINUE, header.status_code());
ASSERT_STREQ(brpc::HttpReasonPhrase(header.status_code()),
header.reason_phrase());
header.set_status_code(brpc::HTTP_STATUS_GONE);
ASSERT_EQ(brpc::HTTP_STATUS_GONE, header.status_code());
ASSERT_STREQ(brpc::HttpReasonPhrase(header.status_code()),
header.reason_phrase());
}
TEST(HttpMessageTest, empty_url) {
butil::EndPoint host;
ASSERT_FALSE(ParseHttpServerAddress(&host, ""));
}
TEST(HttpMessageTest, serialize_http_request) {
brpc::HttpHeader header;
ASSERT_EQ(0u, header.HeaderCount());
header.SetHeader("Foo", "Bar");
ASSERT_EQ(1u, header.HeaderCount());
header.set_method(brpc::HTTP_METHOD_POST);
butil::EndPoint ep;
ASSERT_EQ(0, butil::str2endpoint("127.0.0.1:1234", &ep));
butil::IOBuf request;
butil::IOBuf content;
content.append("data");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nContent-Length: 4\r\nHost: 127.0.0.1:1234\r\nFoo: Bar\r\nAccept: */*\r\nUser-Agent: brpc/1.0 curl/7.0\r\n\r\ndata", request);
// user-set content-length is ignored.
header.SetHeader("Content-Length", "100");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nContent-Length: 4\r\nHost: 127.0.0.1:1234\r\nFoo: Bar\r\nAccept: */*\r\nUser-Agent: brpc/1.0 curl/7.0\r\n\r\ndata", request);
// user-host overwrites passed-in remote_side
header.SetHeader("Host", "MyHost: 4321");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nContent-Length: 4\r\nFoo: Bar\r\nHost: MyHost: 4321\r\nAccept: */*\r\nUser-Agent: brpc/1.0 curl/7.0\r\n\r\ndata", request);
// user-set accept
header.SetHeader("accePT"/*intended uppercase*/, "blahblah");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nContent-Length: 4\r\nFoo: Bar\r\naccePT: blahblah\r\nHost: MyHost: 4321\r\nUser-Agent: brpc/1.0 curl/7.0\r\n\r\ndata", request);
// user-set UA
header.SetHeader("user-AGENT", "myUA");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nContent-Length: 4\r\nFoo: Bar\r\naccePT: blahblah\r\nHost: MyHost: 4321\r\nuser-AGENT: myUA\r\n\r\ndata", request);
// user-set Authorization
header.SetHeader("authorization", "myAuthString");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nContent-Length: 4\r\nFoo: Bar\r\naccePT: blahblah\r\nHost: MyHost: 4321\r\nuser-AGENT: myUA\r\nauthorization: myAuthString\r\n\r\ndata", request);
header.SetHeader("Transfer-Encoding", "chunked");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("POST / HTTP/1.1\r\nFoo: Bar\r\naccePT: blahblah\r\nTransfer-Encoding: chunked\r\nHost: MyHost: 4321\r\nuser-AGENT: myUA\r\nauthorization: myAuthString\r\n\r\ndata", request);
// GET does not serialize content and user-set content-length is ignored.
header.set_method(brpc::HTTP_METHOD_GET);
header.SetHeader("Content-Length", "100");
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("GET / HTTP/1.1\r\nFoo: Bar\r\naccePT: blahblah\r\nHost: MyHost: 4321\r\nuser-AGENT: myUA\r\nauthorization: myAuthString\r\n\r\n", request);
}
TEST(HttpMessageTest, serialize_http_response) {
brpc::HttpHeader header;
header.SetHeader("Foo", "Bar");
header.set_method(brpc::HTTP_METHOD_POST);
butil::IOBuf response;
butil::IOBuf content;
content.append("data");
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 200 OK\r\nContent-Length: 4\r\nFoo: Bar\r\n\r\ndata", response)
<< butil::ToPrintable(response);
// Content is cleared.
CHECK(content.empty());
// NULL content
header.SetHeader("Content-Length", "100");
MakeRawHttpResponse(&response, &header, NULL);
ASSERT_EQ("HTTP/1.1 200 OK\r\nFoo: Bar\r\nContent-Length: 100\r\n\r\n", response)
<< butil::ToPrintable(response);
header.SetHeader("Transfer-Encoding", "chunked");
MakeRawHttpResponse(&response, &header, NULL);
ASSERT_EQ("HTTP/1.1 200 OK\r\nFoo: Bar\r\nTransfer-Encoding: chunked\r\n\r\n", response)
<< butil::ToPrintable(response);
header.RemoveHeader("Transfer-Encoding");
// User-set content-length is ignored.
content.append("data2");
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 200 OK\r\nContent-Length: 5\r\nFoo: Bar\r\n\r\ndata2", response)
<< butil::ToPrintable(response);
header.SetHeader("Content-Length", "100");
header.SetHeader("Transfer-Encoding", "chunked");
MakeRawHttpResponse(&response, &header, NULL);
ASSERT_EQ("HTTP/1.1 200 OK\r\nFoo: Bar\r\nTransfer-Encoding: chunked\r\n\r\n", response)
<< butil::ToPrintable(response);
header.RemoveHeader("Transfer-Encoding");
// User-set content-length and transfer-encoding is ignored when status code is 204 or 1xx.
// 204 No Content.
header.SetHeader("Content-Length", "100");
header.SetHeader("Transfer-Encoding", "chunked");
header.set_status_code(brpc::HTTP_STATUS_NO_CONTENT);
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 204 No Content\r\nFoo: Bar\r\n\r\n", response);
// 101 Continue
header.SetHeader("Content-Length", "100");
header.SetHeader("Transfer-Encoding", "chunked");
header.set_status_code(brpc::HTTP_STATUS_CONTINUE);
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 100 Continue\r\nFoo: Bar\r\n\r\n", response)
<< butil::ToPrintable(response);
// when request method is HEAD:
// 1. There isn't user-set content-length, length of content is used.
header.set_method(brpc::HTTP_METHOD_HEAD);
header.set_status_code(brpc::HTTP_STATUS_OK);content.append("data2");
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 200 OK\r\nContent-Length: 5\r\nFoo: Bar\r\n\r\n", response)
<< butil::ToPrintable(response);
// 2. User-set content-length is not ignored .
header.SetHeader("Content-Length", "100");
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 200 OK\r\nFoo: Bar\r\nContent-Length: 100\r\n\r\n", response)
<< butil::ToPrintable(response);
}
TEST(HttpMessageTest, http_1_1_request_without_host) {
brpc::FLAGS_allow_http_1_1_request_without_host = false;
{
butil::IOBuf request;
request.append("GET /service/method HTTP/1.1\r\n"
"Content-Type: text/plain\r\n\r\n");
brpc::HttpMessage http_message;
ASSERT_TRUE(http_message.ParseFromIOBuf(request) < 0);
}
{
butil::IOBuf request;
request.append("GET http://baidu.com/service/method HTTP/1.1\r\n"
"Content-Type: text/plain\r\n\r\n");
brpc::HttpMessage http_message;
ASSERT_TRUE(http_message.ParseFromIOBuf(request) >= 0);
ASSERT_TRUE(http_message.Completed());
ASSERT_EQ("text/plain", http_message.header().content_type());
}
{
butil::IOBuf request;
request.append("GET /service/method HTTP/1.1\r\n"
"Content-Type: text/plain\r\n"
"Host: baidu.com\r\n\r\n");
brpc::HttpMessage http_message;
ASSERT_GE(http_message.ParseFromIOBuf(request), 0);
ASSERT_GE(http_message.ParseFromArray(NULL, 0), 0);
ASSERT_TRUE(http_message.Completed());
ASSERT_EQ("text/plain", http_message.header().content_type());
}
brpc::FLAGS_allow_http_1_1_request_without_host = true;
}
} //namespace