You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
245 lines
6.4 KiB
245 lines
6.4 KiB
/*
|
|
Copyright (c) 2019 Sogou, Inc.
|
|
|
|
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.
|
|
|
|
Authors: Xie Han (xiehan@sogou-inc.com)
|
|
Wu Jiaxu (wujiaxu@sogou-inc.com)
|
|
*/
|
|
|
|
#ifndef _WFSERVER_H_
|
|
#define _WFSERVER_H_
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <errno.h>
|
|
#include <functional>
|
|
#include <atomic>
|
|
#include <mutex>
|
|
#include <condition_variable>
|
|
#include <openssl/ssl.h>
|
|
#include "EndpointParams.h"
|
|
#include "WFTaskFactory.h"
|
|
|
|
struct WFServerParams
|
|
{
|
|
enum TransportType transport_type;
|
|
size_t max_connections;
|
|
int peer_response_timeout; /* timeout of each read or write operation */
|
|
int receive_timeout; /* timeout of receiving the whole message */
|
|
int keep_alive_timeout;
|
|
size_t request_size_limit;
|
|
int ssl_accept_timeout; /* if not ssl, this will be ignored */
|
|
};
|
|
|
|
static constexpr struct WFServerParams SERVER_PARAMS_DEFAULT =
|
|
{
|
|
.transport_type = TT_TCP,
|
|
.max_connections = 2000,
|
|
.peer_response_timeout = 10 * 1000,
|
|
.receive_timeout = -1,
|
|
.keep_alive_timeout = 60 * 1000,
|
|
.request_size_limit = (size_t)-1,
|
|
.ssl_accept_timeout = 10 * 1000,
|
|
};
|
|
|
|
class WFServerBase : protected CommService
|
|
{
|
|
public:
|
|
WFServerBase(const struct WFServerParams *params) :
|
|
conn_count(0)
|
|
{
|
|
this->params = *params;
|
|
this->unbind_finish = false;
|
|
this->listen_fd = -1;
|
|
}
|
|
|
|
public:
|
|
/* To start a TCP server */
|
|
|
|
/* Start on port with IPv4. */
|
|
int start(unsigned short port)
|
|
{
|
|
return start(AF_INET, NULL, port, NULL, NULL);
|
|
}
|
|
|
|
/* Start with family. AF_INET or AF_INET6. */
|
|
int start(int family, unsigned short port)
|
|
{
|
|
return start(family, NULL, port, NULL, NULL);
|
|
}
|
|
|
|
/* Start with hostname and port. */
|
|
int start(const char *host, unsigned short port)
|
|
{
|
|
return start(AF_INET, host, port, NULL, NULL);
|
|
}
|
|
|
|
/* Start with family, hostname and port. */
|
|
int start(int family, const char *host, unsigned short port)
|
|
{
|
|
return start(family, host, port, NULL, NULL);
|
|
}
|
|
|
|
/* Start with binding address. */
|
|
int start(const struct sockaddr *bind_addr, socklen_t addrlen)
|
|
{
|
|
return start(bind_addr, addrlen, NULL, NULL);
|
|
}
|
|
|
|
/* To start an SSL server. */
|
|
|
|
int start(unsigned short port, const char *cert_file, const char *key_file)
|
|
{
|
|
return start(AF_INET, NULL, port, cert_file, key_file);
|
|
}
|
|
|
|
int start(int family, unsigned short port,
|
|
const char *cert_file, const char *key_file)
|
|
{
|
|
return start(family, NULL, port, cert_file, key_file);
|
|
}
|
|
|
|
int start(const char *host, unsigned short port,
|
|
const char *cert_file, const char *key_file)
|
|
{
|
|
return start(AF_INET, host, port, cert_file, key_file);
|
|
}
|
|
|
|
int start(int family, const char *host, unsigned short port,
|
|
const char *cert_file, const char *key_file);
|
|
|
|
/* This is the only necessary start function. */
|
|
int start(const struct sockaddr *bind_addr, socklen_t addrlen,
|
|
const char *cert_file, const char *key_file);
|
|
|
|
/* To start with a specified fd. For graceful restart or SCTP server. */
|
|
int serve(int listen_fd)
|
|
{
|
|
return serve(listen_fd, NULL, NULL);
|
|
}
|
|
|
|
int serve(int listen_fd, const char *cert_file, const char *key_file);
|
|
|
|
/* stop() is a blocking operation. */
|
|
void stop()
|
|
{
|
|
this->shutdown();
|
|
this->wait_finish();
|
|
}
|
|
|
|
/* Nonblocking terminating the server. For stopping multiple servers.
|
|
* Typically, call shutdown() and then wait_finish().
|
|
* But indeed wait_finish() can be called before shutdown(), even before
|
|
* start() in another thread. */
|
|
void shutdown();
|
|
void wait_finish();
|
|
|
|
public:
|
|
size_t get_conn_count() const { return this->conn_count; }
|
|
|
|
/* Get the listening address. This is often used after starting
|
|
* server on a random port (start() with port == 0). */
|
|
int get_listen_addr(struct sockaddr *addr, socklen_t *addrlen) const
|
|
{
|
|
if (this->listen_fd >= 0)
|
|
return getsockname(this->listen_fd, addr, addrlen);
|
|
|
|
errno = ENOTCONN;
|
|
return -1;
|
|
}
|
|
|
|
const struct WFServerParams *get_params() const { return &this->params; }
|
|
|
|
protected:
|
|
/* Override this function to create the initial SSL CTX of the server */
|
|
virtual SSL_CTX *new_ssl_ctx(const char *cert_file, const char *key_file);
|
|
|
|
/* Override this function to implement server that supports TLS SNI.
|
|
* "servername" will be NULL if client does not set a host name.
|
|
* Returning NULL to indicate that servername is not supported. */
|
|
virtual SSL_CTX *get_server_ssl_ctx(const char *servername)
|
|
{
|
|
return this->get_ssl_ctx();
|
|
}
|
|
|
|
/* This can be used by the implementation of 'new_ssl_ctx'. */
|
|
static int ssl_ctx_callback(SSL *ssl, int *al, void *arg);
|
|
|
|
protected:
|
|
WFServerParams params;
|
|
|
|
protected:
|
|
virtual int create_listen_fd();
|
|
virtual WFConnection *new_connection(int accept_fd);
|
|
void delete_connection(WFConnection *conn);
|
|
|
|
private:
|
|
int init(const struct sockaddr *bind_addr, socklen_t addrlen,
|
|
const char *cert_file, const char *key_file);
|
|
virtual void handle_unbound();
|
|
|
|
protected:
|
|
std::atomic<size_t> conn_count;
|
|
|
|
private:
|
|
int listen_fd;
|
|
bool unbind_finish;
|
|
|
|
std::mutex mutex;
|
|
std::condition_variable cond;
|
|
|
|
class CommScheduler *scheduler;
|
|
};
|
|
|
|
template<class REQ, class RESP>
|
|
class WFServer : public WFServerBase
|
|
{
|
|
public:
|
|
WFServer(const struct WFServerParams *params,
|
|
std::function<void (WFNetworkTask<REQ, RESP> *)> proc) :
|
|
WFServerBase(params),
|
|
process(std::move(proc))
|
|
{
|
|
}
|
|
|
|
WFServer(std::function<void (WFNetworkTask<REQ, RESP> *)> proc) :
|
|
WFServerBase(&SERVER_PARAMS_DEFAULT),
|
|
process(std::move(proc))
|
|
{
|
|
}
|
|
|
|
protected:
|
|
virtual CommSession *new_session(long long seq, CommConnection *conn);
|
|
|
|
protected:
|
|
std::function<void (WFNetworkTask<REQ, RESP> *)> process;
|
|
};
|
|
|
|
template<class REQ, class RESP>
|
|
CommSession *WFServer<REQ, RESP>::new_session(long long seq, CommConnection *conn)
|
|
{
|
|
using factory = WFNetworkTaskFactory<REQ, RESP>;
|
|
WFNetworkTask<REQ, RESP> *task;
|
|
|
|
task = factory::create_server_task(this, this->process);
|
|
task->set_keep_alive(this->params.keep_alive_timeout);
|
|
task->set_receive_timeout(this->params.receive_timeout);
|
|
task->get_req()->set_size_limit(this->params.request_size_limit);
|
|
|
|
return task;
|
|
}
|
|
|
|
#endif
|
|
|