Define _WIN32_WINNT using a cmake macro; support move semantics for parameter; remove address from HttpServer; add utility for dumping endpoints.

master
Adam Gu 8 years ago
parent 2793357da3
commit 2327edbd5d

@ -1,19 +1,32 @@
cmake_minimum_required(VERSION 3.0) cmake_minimum_required(VERSION 3.0)
project(csoap) project(csoap)
# TODO: Replace with log level. # See: https://stackoverflow.com/a/40217291
option(CSOAP_ENABLE_OUTPUT "Enable output for request & response?" OFF) if(WIN32)
macro(get_WIN32_WINNT version)
if(CSOAP_ENABLE_OUTPUT) if(CMAKE_SYSTEM_VERSION)
add_definitions(-DCSOAP_ENABLE_OUTPUT) set(ver ${CMAKE_SYSTEM_VERSION})
endif() string(REGEX MATCH "^([0-9]+).([0-9])" ver ${ver})
string(REGEX MATCH "^([0-9]+)" verMajor ${ver})
add_definitions(-DUNICODE -D_UNICODE) # Check for Windows 10, b/c we'll need to convert to hex 'A'.
if("${verMajor}" MATCHES "10")
if(MSVC) set(verMajor "A")
# Win7 (Boost.Asio needs this) string(REGEX REPLACE "^([0-9]+)" ${verMajor} ver ${ver})
add_definitions(-D_WIN32_WINNT=0x0601) endif("${verMajor}" MATCHES "10")
endif() # Remove all remaining '.' characters.
string(REPLACE "." "" ver ${ver})
# Prepend each digit with a zero.
string(REGEX REPLACE "([0-9A-Z])" "0\\1" ver ${ver})
set(${version} "0x${ver}")
endif(CMAKE_SYSTEM_VERSION)
endmacro(get_WIN32_WINNT)
get_WIN32_WINNT(ver)
# E.g., 0x0601 for Win7.
message(STATUS "_WIN32_WINNT=${ver}")
# Asio needs this!
add_definitions(-D_WIN32_WINNT=${ver})
endif(WIN32)
# Boost version: 1.66+ # Boost version: 1.66+
set(Boost_USE_STATIC_LIBS ON) set(Boost_USE_STATIC_LIBS ON)
@ -25,9 +38,4 @@ endif()
include_directories(${PROJECT_SOURCE_DIR}/src) include_directories(${PROJECT_SOURCE_DIR}/src)
if(MSVC)
add_definitions(-D_CRT_SECURE_NO_WARNINGS)
endif()
add_subdirectory(src) add_subdirectory(src)

@ -2,6 +2,12 @@ if(UNIX)
add_definitions(-D_GLIBCXX_USE_WCHAR_T -std=c++11) add_definitions(-D_GLIBCXX_USE_WCHAR_T -std=c++11)
endif() endif()
option(CSOAP_ENABLE_OUTPUT "Enable output for request & response?" OFF)
if(CSOAP_ENABLE_OUTPUT)
add_definitions(-DCSOAP_ENABLE_OUTPUT)
endif()
# Don't use any deprecated definitions (e.g., io_service). # Don't use any deprecated definitions (e.g., io_service).
add_definitions(-DBOOST_ASIO_NO_DEPRECATED) add_definitions(-DBOOST_ASIO_NO_DEPRECATED)

@ -71,6 +71,10 @@ Parameter::Parameter(const std::string& key, const std::string& value)
: key_(key), value_(value) { : key_(key), value_(value) {
} }
Parameter::Parameter(const std::string& key, std::string&& value)
: key_(key), value_(std::move(value)) {
}
Parameter::Parameter(const std::string& key, int value) Parameter::Parameter(const std::string& key, int value)
: key_(key) { : key_(key) {
value_ = boost::lexical_cast<std::string>(value); value_ = boost::lexical_cast<std::string>(value);

@ -84,8 +84,13 @@ extern const Namespace kSoapEnvNamespace;
// Parameter in the SOAP request envelope. // Parameter in the SOAP request envelope.
class Parameter { class Parameter {
public: public:
Parameter() = default;
Parameter(const Parameter& rhs) = default;
Parameter& operator=(const Parameter& rhs) = default;
Parameter(const std::string& key, const char* value); Parameter(const std::string& key, const char* value);
Parameter(const std::string& key, const std::string& value); Parameter(const std::string& key, const std::string& value);
Parameter(const std::string& key, std::string&& value);
Parameter(const std::string& key, int value); Parameter(const std::string& key, int value);
Parameter(const std::string& key, double value); Parameter(const std::string& key, double value);
Parameter(const std::string& key, bool value); Parameter(const std::string& key, bool value);

@ -2,7 +2,9 @@
#define CSOAP_HTTP_CLIENT_H_ #define CSOAP_HTTP_CLIENT_H_
#include <array> #include <array>
#include "boost/asio/io_context.hpp" #include "boost/asio/io_context.hpp"
#include "csoap/common.h" #include "csoap/common.h"
namespace csoap { namespace csoap {

@ -3,14 +3,15 @@
#include <signal.h> #include <signal.h>
#include "csoap/soap_service.h" #include "csoap/soap_service.h"
#include "csoap/utility.h"
using tcp = boost::asio::ip::tcp;
namespace csoap { namespace csoap {
HttpServer::HttpServer(const std::string& address, HttpServer::HttpServer(unsigned short port)
const std::string& port) : io_context_(1) // TODO: concurrency_hint (threads)
: io_context_(1) // TODO: concurrency_hint (threads) , signals_(io_context_) {
, signals_(io_context_)
, acceptor_(io_context_) {
// Register to handle the signals that indicate when the server should exit. // Register to handle the signals that indicate when the server should exit.
// It is safe to register for the same signal multiple times in a program, // It is safe to register for the same signal multiple times in a program,
@ -24,15 +25,15 @@ HttpServer::HttpServer(const std::string& address,
DoAwaitStop(); DoAwaitStop();
// Open the acceptor with the option to reuse the address (i.e. SO_REUSEADDR). // NOTE:
// TODO: What does SO_REUSEADDR mean? // "reuse_addr=true" means option SO_REUSEADDR will be set.
// TODO: Why need an address? // For more details about SO_REUSEADDR, see:
boost::asio::ip::tcp::resolver resolver(io_context_); // https://msdn.microsoft.com/en-us/library/windows/desktop/ms740621(v=vs.85).aspx
boost::asio::ip::tcp::endpoint endpoint = *resolver.resolve(address, port).begin(); // http://www.andy-pearce.com/blog/posts/2013/Feb/so_reuseaddr-on-windows/
acceptor_.open(endpoint.protocol()); // TODO: SO_EXCLUSIVEADDRUSE
acceptor_.set_option(boost::asio::ip::tcp::acceptor::reuse_address(true)); acceptor_.reset(new tcp::acceptor(io_context_,
acceptor_.bind(endpoint); tcp::endpoint(tcp::v4(), port),
acceptor_.listen(); true)); // reuse_addr
DoAccept(); DoAccept();
} }
@ -50,11 +51,11 @@ void HttpServer::Run() {
} }
void HttpServer::DoAccept() { void HttpServer::DoAccept() {
acceptor_.async_accept( acceptor_->async_accept(
[this](boost::system::error_code ec, boost::asio::ip::tcp::socket socket) { [this](boost::system::error_code ec, tcp::socket socket) {
// Check whether the server was stopped by a signal before this // Check whether the server was stopped by a signal before this
// completion handler had a chance to run. // completion handler had a chance to run.
if (!acceptor_.is_open()) { if (!acceptor_->is_open()) {
return; return;
} }
@ -75,7 +76,7 @@ void HttpServer::DoAwaitStop() {
// The server is stopped by cancelling all outstanding asynchronous // The server is stopped by cancelling all outstanding asynchronous
// operations. Once all operations have finished the io_context::run() // operations. Once all operations have finished the io_context::run()
// call will exit. // call will exit.
acceptor_.close(); acceptor_->close();
connection_manager_.StopAll(); connection_manager_.StopAll();
}); });
} }

@ -4,6 +4,7 @@
#include <string> #include <string>
#include <vector> #include <vector>
#include "boost/scoped_ptr.hpp"
#include "boost/asio/io_context.hpp" #include "boost/asio/io_context.hpp"
#include "boost/asio/signal_set.hpp" #include "boost/asio/signal_set.hpp"
#include "boost/asio/ip/tcp.hpp" #include "boost/asio/ip/tcp.hpp"
@ -14,16 +15,14 @@
namespace csoap { namespace csoap {
// The top-level class of the HTTP server. // HTTP server accepts TCP connections from TCP clients.
// NOTE: Only support IPv4.
class HttpServer { class HttpServer {
public: public:
HttpServer(const HttpServer&) = delete; HttpServer(const HttpServer&) = delete;
HttpServer& operator=(const HttpServer&) = delete; HttpServer& operator=(const HttpServer&) = delete;
// Construct the server to listen on the specified TCP address and port, and HttpServer(unsigned short port);
// serve up files from the given directory.
HttpServer(const std::string& address,
const std::string& port);
bool RegisterService(SoapServicePtr soap_service); bool RegisterService(SoapServicePtr soap_service);
@ -45,7 +44,7 @@ private:
boost::asio::signal_set signals_; boost::asio::signal_set signals_;
// Acceptor used to listen for incoming connections. // Acceptor used to listen for incoming connections.
boost::asio::ip::tcp::acceptor acceptor_; boost::scoped_ptr<boost::asio::ip::tcp::acceptor> acceptor_;
// The connection manager which owns all live connections. // The connection manager which owns all live connections.
ConnectionManager connection_manager_; ConnectionManager connection_manager_;

@ -11,13 +11,15 @@ void SoapRequest::AddParameter(Parameter&& parameter) {
parameters_.push_back(std::move(parameter)); parameters_.push_back(std::move(parameter));
} }
std::string SoapRequest::GetParameter(const std::string& key) const { const std::string& SoapRequest::GetParameter(const std::string& key) const {
for (const Parameter& p : parameters_) { for (const Parameter& p : parameters_) {
if (p.key() == key) { if (p.key() == key) {
return p.value(); return p.value();
} }
} }
return "";
static const std::string kEmptyValue;
return kEmptyValue;
} }
void SoapRequest::ToXmlBody(pugi::xml_node xbody) { void SoapRequest::ToXmlBody(pugi::xml_node xbody) {

@ -16,7 +16,7 @@ public:
void AddParameter(Parameter&& parameter); void AddParameter(Parameter&& parameter);
// Get parameter value by key. // Get parameter value by key.
std::string GetParameter(const std::string& key) const; const std::string& GetParameter(const std::string& key) const;
protected: protected:
void ToXmlBody(pugi::xml_node xbody) override; void ToXmlBody(pugi::xml_node xbody) override;

@ -23,6 +23,10 @@ public:
result_ = result; result_ = result;
} }
void set_result(std::string&& result) {
result_ = std::move(result);
}
protected: protected:
void ToXmlBody(pugi::xml_node xbody) override; void ToXmlBody(pugi::xml_node xbody) override;

@ -0,0 +1,28 @@
#include "csoap/utility.h"
#include <iostream>
using tcp = boost::asio::ip::tcp;
namespace csoap {
// Print the resolved endpoints.
// NOTE: Endpoint is one word, don't use "end point".
// TODO
void DumpEndpoints(tcp::resolver::results_type& endpoints) {
std::cout << "Endpoints: " << endpoints.size() << std::endl;
tcp::resolver::results_type::iterator it = endpoints.begin();
for (; it != endpoints.end(); ++it) {
std::cout << " - " << it->endpoint();
if (it->endpoint().protocol() == tcp::v4()) {
std::cout << ", v4";
} else if (it->endpoint().protocol() == tcp::v6()) {
std::cout << ", v6";
}
std::cout << std::endl;
}
}
} // namespace csoap

@ -0,0 +1,14 @@
#ifndef CSOAP_UTILITY_H_
#define CSOAP_UTILITY_H_
#include "boost/asio/ip/tcp.hpp"
namespace csoap {
// Print the resolved endpoints.
// NOTE: Endpoint is one word, don't use "end point".
void DumpEndpoints(boost::asio::ip::tcp::resolver::results_type& endpoints);
} // namespace csoap
#endif // CSOAP_UTILITY_H_

@ -2,7 +2,7 @@
#include "csoap/http_server.h" #include "csoap/http_server.h"
#include "calculator_service.h" #include "calculator_service.h"
static void PrintHelp(const char* argv0) { static void Help(const char* argv0) {
std::cout << "Usage: " << argv0 << " <port>" << std::endl; std::cout << "Usage: " << argv0 << " <port>" << std::endl;
std::cout << " E.g.," << std::endl; std::cout << " E.g.," << std::endl;
std::cout << " " << argv0 << " 8080" << std::endl; std::cout << " " << argv0 << " 8080" << std::endl;
@ -10,14 +10,14 @@ static void PrintHelp(const char* argv0) {
int main(int argc, char* argv[]) { int main(int argc, char* argv[]) {
if (argc != 2) { if (argc != 2) {
PrintHelp(argv[0]); Help(argv[0]);
return 1; return 1;
} }
const char* host = "0.0.0.0"; // TODO unsigned short port = std::atoi(argv[1]);
try { try {
csoap::HttpServer server(host, argv[1]); csoap::HttpServer server(port);
csoap::SoapServicePtr service(new CalculatorService); csoap::SoapServicePtr service(new CalculatorService);
server.RegisterService(service); server.RegisterService(service);

Loading…
Cancel
Save