blob: 7eccf7cd7b1ea7923f6c9522d7dc8fd6db544268 [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 {
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_NS::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);
std::string 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().to_string());
ASSERT_EQ("text/plain", http_message.header().content_type());
}
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_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\naccePT: blahblah\r\nFoo: Bar\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\naccePT: blahblah\r\nuser-AGENT: myUA\r\nFoo: Bar\r\nHost: MyHost: 4321\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\naccePT: blahblah\r\nuser-AGENT: myUA\r\nauthorization: myAuthString\r\nFoo: Bar\r\nHost: MyHost: 4321\r\n\r\ndata", request);
// GET does not serialize content
header.set_method(brpc::HTTP_METHOD_GET);
MakeRawHttpRequest(&request, &header, ep, &content);
ASSERT_EQ("GET / HTTP/1.1\r\naccePT: blahblah\r\nuser-AGENT: myUA\r\nauthorization: myAuthString\r\nFoo: Bar\r\nHost: MyHost: 4321\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);
// content is cleared.
CHECK(content.empty());
// user-set content-length is ignored.
content.append("data2");
header.SetHeader("Content-Length", "100");
MakeRawHttpResponse(&response, &header, &content);
ASSERT_EQ("HTTP/1.1 200 OK\r\nContent-Length: 5\r\nFoo: Bar\r\n\r\ndata2", response);
// null content
MakeRawHttpResponse(&response, &header, NULL);
ASSERT_EQ("HTTP/1.1 200 OK\r\nFoo: Bar\r\n\r\n", response);
}
} //namespace