Skip to content

Latest commit

 

History

History
190 lines (153 loc) · 4.87 KB

README.md

File metadata and controls

190 lines (153 loc) · 4.87 KB

west

WEb Server Template is a bare-bones header-only web server for use with C++20 or later. It

  • Uses epoll to determine which socket is active

  • Is currently single-threaded. Simultaneous requests is a future topic.

  • Has a socket expire check (fires 20 s after no socket activity)

  • Limits the size of the request header

  • Does not know anything about HTTP headers, except content-length

  • Does not support chunked encoding, though this feature may be added if the author finds it useful

  • Does not know anything about URI:s. It is up to the application to interpret the request target

  • Currently only supports TCP. The application may use a different transport mechanism provided it uses a system-level file descriptor.

Example usage:

#include <west/io_inet_server_socket.hpp>
#include <west/service_registry.hpp>
#include <west/http_request_handler.hpp>
#include <west/http_session_factory.hpp>
#include <west/http_server.hpp>

namespace
{
	// Some stuff needed to convey processing results back to the framework

	enum class request_handler_error_code{no_error};

	struct read_result
	{
		size_t bytes_read;
		request_handler_error_code ec;
	};

	constexpr bool can_continue(request_handler_error_code ec)
	{
		switch(ec)
		{
			case request_handler_error_code::no_error:
				return true;
			default:
				__builtin_unreachable();
		}
	}

	constexpr bool is_error_indicator(request_handler_error_code ec)
	{
		switch(ec)
		{
			case request_handler_error_code::no_error:
				return false;
			default:
				__builtin_unreachable();
		}
	}

	constexpr char const* to_string(request_handler_error_code ec)
	{
		switch(ec)
		{
			case request_handler_error_code::no_error:
				return "No error";
			default:
				__builtin_unreachable();
		}
	}

	struct request_handler_write_result
	{
		size_t bytes_written;
		request_handler_error_code ec;
	};

	struct request_handler_read_result
	{
		size_t bytes_read;
		request_handler_error_code ec;
	};


	// The actual request handler. This class has all entry points for the application

	class your_request_handler
	{
	public:
		your_request_handler(/* request-handler-args */)
		{
			// Called per connection
		}

		auto finalize_state(west::http::request_header const& header)
		{
			// This function is called after the request header has been read. If you
			// had any state left over from a previous request like temporary buffers,
			// you may need to clear it in this function.
			//
			// `header` contains the request header sent by the client.
			// It contains the method, request target and all fields. If you
			// want to serve files, you should map header.request_target
			// to the corresponding file.

			// Return the result of this state
			west::http::finalize_state_result validation_result;
			validation_result.http_status = west::http::status::ok;
			validation_result.error_message = nullptr;

			return validation_result;
		}

		auto process_request_content(std::span<char const> buffer)
		{
			// This function may be called multiple times
			//
			// `buffer` always contains the lastly read portion of the request body.
			//
			return request_handler_write_result{
				.bytes_written = std::size(buffer),
				.ec = request_handler_error_code::no_error
			};
		}

		auto finalize_state(west::http::field_map& fields)
		{
			// This function is called after the request body has been processed. It is time
			// to prepare for the answer. You should append Content-Length and Content-Type to
			// `fields`

			fields.append("Content-Length", std::to_string(0)
				.append("Content-Type", "text/plain");

			west::http::finalize_state_result validation_result;
			validation_result.http_status = west::http::status::ok;
			validation_result.error_message = nullptr;
			return validation_result;
		}

		void finalize_state(west::http::field_map& fields, west::http::finalize_state_result&& res)
		{
			// This function is called in case of an error during the processing of the request. `res`
			// contains information about the error

			fields.append("Content-Length", std::to_string(0)
				.append("Content-Type", "text/plain");
		}

		auto read_response_content(std::span<char> buffer)
		{
			// This function will be called multiple times
			// Copy appropriate content to `buffer`

			return request_handler_read_result{
				std::size(buffer),
				request_handler_error_code::no_error
			};
		}

	private:
	// Session specific data
	};
}

int main()
{
	// Create a server socket listening for connections
	// on localhost. Choose first free port in range [49152, 65535].
	west::io::inet_server_socket http{
		west::io::inet_address{"127.0.0.1"},
		std::ranges::iota_view{49152, 65536},
		128
	};

	printf("Listening on port %u\n", http.port());
	fflush(stdout);

	// Register the service and start processing events
	west::service_registry services{};
	enroll_http_service<request_handler>(services, std::move(http), /* request-handler-args */)
		.process_events();
}