/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved. miniob is licensed under Mulan PSL v2. You can use this software according to the terms and conditions of the Mulan PSL v2. You may obtain a copy of Mulan PSL v2 at: http://license.coscl.org.cn/MulanPSL2 THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. See the Mulan PSL v2 for more details. */ // // Created by Longda on 2021 // #include "net/server.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "common/lang/mutex.h" #include "common/log/log.h" #include "common/io/io.h" #include "common/seda/seda_config.h" #include "event/session_event.h" #include "session/session.h" #include "ini_setting.h" #include "net/communicator.h" #include using namespace common; static const std::string READ_SOCKET_METRIC_TAG = "SessionStage.readsocket"; static const std::string WRITE_SOCKET_METRIC_TAG = "SessionStage.writesocket"; Stage *Server::session_stage_ = nullptr; common::SimpleTimer *Server::read_socket_metric_ = nullptr; common::SimpleTimer *Server::write_socket_metric_ = nullptr; ServerParam::ServerParam() { listen_addr = INADDR_ANY; max_connection_num = MAX_CONNECTION_NUM_DEFAULT; port = PORT_DEFAULT; } Server::Server(ServerParam input_server_param) : server_param_(input_server_param) { started_ = false; server_socket_ = 0; event_base_ = nullptr; listen_ev_ = nullptr; } Server::~Server() { if (started_) { shutdown(); } } void Server::init() { session_stage_ = get_seda_config()->get_stage(SESSION_STAGE_NAME); MetricsRegistry &metricsRegistry = get_metrics_registry(); if (Server::read_socket_metric_ == nullptr) { Server::read_socket_metric_ = new SimpleTimer(); metricsRegistry.register_metric(READ_SOCKET_METRIC_TAG, Server::read_socket_metric_); } if (Server::write_socket_metric_ == nullptr) { Server::write_socket_metric_ = new SimpleTimer(); metricsRegistry.register_metric(WRITE_SOCKET_METRIC_TAG, Server::write_socket_metric_); } } int Server::set_non_block(int fd) { int flags = fcntl(fd, F_GETFL); if (flags == -1) { LOG_INFO("Failed to get flags of fd :%d. ", fd); return -1; } flags = fcntl(fd, F_SETFL, flags | O_NONBLOCK); if (flags == -1) { LOG_INFO("Failed to set non-block flags of fd :%d. ", fd); return -1; } return 0; } void Server::close_connection(Communicator *communicator) { LOG_INFO("Close connection of %s.", communicator->addr()); event_del(&communicator->read_event()); delete communicator; } void Server::recv(int fd, short ev, void *arg) { Communicator *comm = (Communicator *)arg; SessionEvent *event = nullptr; RC rc = comm->read_event(event); if (rc != RC::SUCCESS) { close_connection(comm); return; } if (event == nullptr) { LOG_WARN("event is null while read event return success"); return; } session_stage_->add_event(event); } #if 0 // 这个函数仅负责发送数据,至于是否是一个完整的消息,由调用者控制 int Server::send( *client, const char *buf, int data_len) { if (buf == nullptr || data_len == 0) { return 0; } TimerStat writeStat(*write_socket_metric_); int ret = common::writen(client->fd, buf, data_len); if (ret < 0) { LOG_ERROR("Failed to send data back to client. ret=%d, error=%s", ret, strerror(errno)); MUTEX_UNLOCK(&client->mutex); close_connection(client); return -STATUS_FAILED_NETWORK; } return 0; } #endif void Server::accept(int fd, short ev, void *arg) { Server *instance = (Server *)arg; struct sockaddr_in addr; socklen_t addrlen = sizeof(addr); int ret = 0; int client_fd = ::accept(fd, (struct sockaddr *)&addr, &addrlen); if (client_fd < 0) { LOG_ERROR("Failed to accept client's connection, %s", strerror(errno)); return; } char ip_addr[24]; if (inet_ntop(AF_INET, &addr.sin_addr, ip_addr, sizeof(ip_addr)) == nullptr) { LOG_ERROR("Failed to get ip address of client, %s", strerror(errno)); ::close(client_fd); return; } std::stringstream address; address << ip_addr << ":" << addr.sin_port; std::string addr_str = address.str(); ret = instance->set_non_block(client_fd); if (ret < 0) { LOG_ERROR("Failed to set socket of %s as non blocking, %s", addr_str.c_str(), strerror(errno)); ::close(client_fd); return; } if (!instance->server_param_.use_unix_socket) { // unix socket不支持设置NODELAY int yes = 1; ret = setsockopt(client_fd, IPPROTO_TCP, TCP_NODELAY, &yes, sizeof(yes)); if (ret < 0) { LOG_ERROR("Failed to set socket of %s option as : TCP_NODELAY %s\n", addr_str.c_str(), strerror(errno)); ::close(client_fd); return; } } Communicator *communicator = instance->communicator_factory_.create(instance->server_param_.protocol); RC rc = communicator->init(client_fd, new Session(Session::default_session()), addr_str); if (rc != RC::SUCCESS) { LOG_WARN("failed to init communicator. rc=%s", strrc(rc)); delete communicator; return; } event_set(&communicator->read_event(), client_fd, EV_READ | EV_PERSIST, recv, communicator); ret = event_base_set(instance->event_base_, &communicator->read_event()); if (ret < 0) { LOG_ERROR( "Failed to do event_base_set for read event of %s into libevent, %s", communicator->addr(), strerror(errno)); delete communicator; return; } ret = event_add(&communicator->read_event(), nullptr); if (ret < 0) { LOG_ERROR("Failed to event_add for read event of %s into libevent, %s", communicator->addr(), strerror(errno)); delete communicator; return; } LOG_INFO("Accepted connection from %s\n", communicator->addr()); } int Server::start() { if (server_param_.use_unix_socket) { return start_unix_socket_server(); } else { return start_tcp_server(); } } int Server::start_tcp_server() { int ret = 0; struct sockaddr_in sa; server_socket_ = socket(AF_INET, SOCK_STREAM, 0); if (server_socket_ < 0) { LOG_ERROR("socket(): can not create server socket: %s.", strerror(errno)); return -1; } int yes = 1; ret = setsockopt(server_socket_, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)); if (ret < 0) { LOG_ERROR("Failed to set socket option of reuse address: %s.", strerror(errno)); ::close(server_socket_); return -1; } ret = set_non_block(server_socket_); if (ret < 0) { LOG_ERROR("Failed to set socket option non-blocking:%s. ", strerror(errno)); ::close(server_socket_); return -1; } memset(&sa, 0, sizeof(sa)); sa.sin_family = AF_INET; sa.sin_port = htons(server_param_.port); sa.sin_addr.s_addr = htonl(server_param_.listen_addr); ret = bind(server_socket_, (struct sockaddr *)&sa, sizeof(sa)); if (ret < 0) { LOG_ERROR("bind(): can not bind server socket, %s", strerror(errno)); ::close(server_socket_); return -1; } ret = listen(server_socket_, server_param_.max_connection_num); if (ret < 0) { LOG_ERROR("listen(): can not listen server socket, %s", strerror(errno)); ::close(server_socket_); return -1; } LOG_INFO("Listen on port %d", server_param_.port); listen_ev_ = event_new(event_base_, server_socket_, EV_READ | EV_PERSIST, accept, this); if (listen_ev_ == nullptr) { LOG_ERROR("Failed to create listen event, %s.", strerror(errno)); ::close(server_socket_); return -1; } ret = event_add(listen_ev_, nullptr); if (ret < 0) { LOG_ERROR("event_add(): can not add accept event into libevent, %s", strerror(errno)); ::close(server_socket_); return -1; } started_ = true; LOG_INFO("Observer start success"); return 0; } int Server::start_unix_socket_server() { int ret = 0; server_socket_ = socket(PF_UNIX, SOCK_STREAM, 0); if (server_socket_ < 0) { LOG_ERROR("socket(): can not create unix socket: %s.", strerror(errno)); return -1; } ret = set_non_block(server_socket_); if (ret < 0) { LOG_ERROR("Failed to set socket option non-blocking:%s. ", strerror(errno)); ::close(server_socket_); return -1; } unlink(server_param_.unix_socket_path.c_str()); struct sockaddr_un sockaddr; memset(&sockaddr, 0, sizeof(sockaddr)); sockaddr.sun_family = PF_UNIX; snprintf(sockaddr.sun_path, sizeof(sockaddr.sun_path), "%s", server_param_.unix_socket_path.c_str()); ret = bind(server_socket_, (struct sockaddr *)&sockaddr, sizeof(sockaddr)); if (ret < 0) { LOG_ERROR("bind(): can not bind server socket(path=%s), %s", sockaddr.sun_path, strerror(errno)); ::close(server_socket_); return -1; } ret = listen(server_socket_, server_param_.max_connection_num); if (ret < 0) { LOG_ERROR("listen(): can not listen server socket, %s", strerror(errno)); ::close(server_socket_); return -1; } LOG_INFO("Listen on unix socket: %s", sockaddr.sun_path); listen_ev_ = event_new(event_base_, server_socket_, EV_READ | EV_PERSIST, accept, this); if (listen_ev_ == nullptr) { LOG_ERROR("Failed to create listen event, %s.", strerror(errno)); ::close(server_socket_); return -1; } ret = event_add(listen_ev_, nullptr); if (ret < 0) { LOG_ERROR("event_add(): can not add accept event into libevent, %s", strerror(errno)); ::close(server_socket_); return -1; } started_ = true; LOG_INFO("Observer start success"); return 0; } int Server::serve() { evthread_use_pthreads(); event_base_ = event_base_new(); if (event_base_ == nullptr) { LOG_ERROR("Failed to create event base, %s.", strerror(errno)); exit(-1); } int retval = start(); if (retval == -1) { LOG_PANIC("Failed to start network"); exit(-1); } event_base_dispatch(event_base_); if (listen_ev_ != nullptr) { event_del(listen_ev_); event_free(listen_ev_); listen_ev_ = nullptr; } if (event_base_ != nullptr) { event_base_free(event_base_); event_base_ = nullptr; } started_ = false; LOG_INFO("Server quit"); return 0; } void Server::shutdown() { LOG_INFO("Server shutting down"); // cleanup if (event_base_ != nullptr && started_) { started_ = false; event_base_loopexit(event_base_, nullptr); } }