|
|
|
//
|
|
|
|
//
|
|
|
|
// Copyright 2017 gRPC authors.
|
|
|
|
//
|
|
|
|
// Licensed 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.
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
#include <sys/socket.h>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include <grpc/event_engine/slice_buffer.h>
|
|
|
|
#include <grpc/grpc.h>
|
|
|
|
#include <grpcpp/server.h>
|
|
|
|
#include <grpcpp/server_builder.h>
|
|
|
|
#include <grpcpp/support/config.h>
|
|
|
|
|
|
|
|
#include "src/core/lib/gprpp/notification.h"
|
|
|
|
#include "src/proto/grpc/testing/echo.grpc.pb.h"
|
|
|
|
#include "test/core/event_engine/event_engine_test_utils.h"
|
|
|
|
#include "test/core/test_util/port.h"
|
|
|
|
#include "test/core/test_util/test_config.h"
|
|
|
|
|
|
|
|
namespace grpc {
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
testing::EchoTestService::Service g_service;
|
|
|
|
|
|
|
|
std::string MakePort() {
|
|
|
|
std::ostringstream s;
|
|
|
|
int p = grpc_pick_unused_port_or_die();
|
|
|
|
s << "localhost:" << p;
|
|
|
|
return s.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& GetPort() {
|
|
|
|
static std::string g_port = MakePort();
|
|
|
|
return g_port;
|
|
|
|
}
|
|
|
|
|
|
|
|
class ServerBuilderTest : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
static void SetUpTestSuite() { grpc_init(); }
|
|
|
|
|
|
|
|
static void TearDownTestSuite() { grpc_shutdown(); }
|
|
|
|
};
|
|
|
|
TEST_F(ServerBuilderTest, NoOp) { ServerBuilder b; }
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, CreateServerNoPorts) {
|
|
|
|
ServerBuilder().RegisterService(&g_service).BuildAndStart()->Shutdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, CreateServerOnePort) {
|
|
|
|
ServerBuilder()
|
|
|
|
.RegisterService(&g_service)
|
|
|
|
.AddListeningPort(GetPort(), InsecureServerCredentials())
|
|
|
|
.BuildAndStart()
|
|
|
|
->Shutdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, CreateServerRepeatedPort) {
|
|
|
|
ServerBuilder()
|
|
|
|
.RegisterService(&g_service)
|
|
|
|
.AddListeningPort(GetPort(), InsecureServerCredentials())
|
|
|
|
.AddListeningPort(GetPort(), InsecureServerCredentials())
|
|
|
|
.BuildAndStart()
|
|
|
|
->Shutdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, CreateServerRepeatedPortWithDisallowedReusePort) {
|
|
|
|
EXPECT_EQ(ServerBuilder()
|
|
|
|
.RegisterService(&g_service)
|
|
|
|
.AddListeningPort(GetPort(), InsecureServerCredentials())
|
|
|
|
.AddListeningPort(GetPort(), InsecureServerCredentials())
|
|
|
|
.AddChannelArgument(GRPC_ARG_ALLOW_REUSEPORT, 0)
|
|
|
|
.BuildAndStart(),
|
|
|
|
nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, AddPassiveListener) {
|
|
|
|
std::unique_ptr<experimental::PassiveListener> passive_listener;
|
|
|
|
auto server =
|
|
|
|
ServerBuilder()
|
|
|
|
.experimental()
|
|
|
|
.AddPassiveListener(InsecureServerCredentials(), passive_listener)
|
|
|
|
.BuildAndStart();
|
|
|
|
server->Shutdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, PassiveListenerAcceptConnectedFd) {
|
|
|
|
std::unique_ptr<experimental::PassiveListener> passive_listener;
|
|
|
|
ServerBuilder builder;
|
|
|
|
auto cq = builder.AddCompletionQueue();
|
|
|
|
// TODO(hork): why is the service necessary? Queue isn't drained otherwise.
|
|
|
|
auto server =
|
|
|
|
builder.RegisterService(&g_service)
|
|
|
|
.experimental()
|
|
|
|
.AddPassiveListener(InsecureServerCredentials(), passive_listener)
|
|
|
|
.BuildAndStart();
|
|
|
|
ASSERT_NE(server.get(), nullptr);
|
|
|
|
#ifdef GPR_SUPPORT_CHANNELS_FROM_FD
|
|
|
|
int fd = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
auto accept_status = passive_listener->AcceptConnectedFd(fd);
|
|
|
|
ASSERT_TRUE(accept_status.ok()) << accept_status;
|
|
|
|
#else
|
|
|
|
int fd = -1;
|
|
|
|
auto accept_status = passive_listener->AcceptConnectedFd(fd);
|
|
|
|
ASSERT_FALSE(accept_status.ok()) << accept_status;
|
|
|
|
#endif
|
|
|
|
server->Shutdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ServerBuilderTest, PassiveListenerAcceptConnectedEndpoint) {
|
|
|
|
std::unique_ptr<experimental::PassiveListener> passive_listener;
|
|
|
|
auto server =
|
|
|
|
ServerBuilder()
|
|
|
|
.experimental()
|
|
|
|
.AddPassiveListener(InsecureServerCredentials(), passive_listener)
|
|
|
|
.BuildAndStart();
|
|
|
|
grpc_core::Notification endpoint_destroyed;
|
|
|
|
auto success = passive_listener->AcceptConnectedEndpoint(
|
|
|
|
std::make_unique<grpc_event_engine::experimental::ThreadedNoopEndpoint>(
|
|
|
|
&endpoint_destroyed));
|
|
|
|
ASSERT_TRUE(success.ok())
|
|
|
|
<< "AcceptConnectedEndpoint failure: " << success.ToString();
|
|
|
|
endpoint_destroyed.WaitForNotification();
|
|
|
|
server->Shutdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
} // namespace grpc
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
grpc::testing::TestEnvironment env(&argc, argv);
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
int ret = RUN_ALL_TESTS();
|
|
|
|
return ret;
|
|
|
|
}
|