pluto-src 0.1.1+0.10.4

Sources of Pluto (Lua 5.4 dialect) and logic to build it.
Documentation
#include "Server.hpp"

#if !SOUP_WASM

#include "CertStore.hpp"
#include "ServerService.hpp"
#include "ServerServiceUdp.hpp"
#include "SharedPtr.hpp"
#include "Socket.hpp"

NAMESPACE_SOUP
{
	struct CaptureServerPort
	{
		Server* server;
		ServerService* service;

		CaptureServerPort(Server* server, ServerService* service)
			: server(server), service(service)
		{
		}

		void processAccept(Socket&& sock) const
		{
			if (sock.hasConnection())
			{
				auto s = server->addSocket(std::move(sock));
				if (service->on_connection_established)
				{
					service->on_connection_established(*s, *service, *server);
				}
				service->on_tunnel_established(*s, *service, *server);
			}
		}
	};

	struct CaptureServerPortCrypto : public CaptureServerPort
	{
		SharedPtr<CertStore> certstore;
		tls_server_on_client_hello_t on_client_hello;

		CaptureServerPortCrypto(Server* server, ServerService* service, const SharedPtr<CertStore>& certstore, tls_server_on_client_hello_t on_client_hello)
			: CaptureServerPort(server, service), certstore(certstore), on_client_hello(on_client_hello)
		{
		}

		void processAccept(Socket&& sock) const
		{
			if (sock.hasConnection())
			{
				auto s = server->addSocket(std::move(sock));
				if (service->on_connection_established)
				{
					service->on_connection_established(*s, *service, *server);
				}
				s->enableCryptoServer(certstore, [](Socket& s, Capture&& _cap)
				{
					CaptureServerPortCrypto& cap = *_cap.get<CaptureServerPortCrypto*>();
					cap.service->on_tunnel_established(s, *cap.service, *cap.server);
				}, this, on_client_hello);
			}
		}
	};

	struct CaptureServerPortOptCrypto : public CaptureServerPortCrypto
	{
		CaptureServerPortOptCrypto(Server* server, ServerService* service, const SharedPtr<CertStore>& certstore, tls_server_on_client_hello_t on_client_hello)
			: CaptureServerPortCrypto(server, service, certstore, on_client_hello)
		{
		}

		void processAccept(Socket&& sock) const
		{
			if (sock.hasConnection())
			{
				auto s = server->addSocket(std::move(sock));
				if (service->on_connection_established)
				{
					service->on_connection_established(*s, *service, *server);
				}
				s->transport_recv([](Socket& s, std::string&& data, Capture&& _cap)
				{
					s.transport_unrecv(data);
					CaptureServerPortOptCrypto& cap = *_cap.get<CaptureServerPortOptCrypto*>();
					if (data.size() > 2 && data.at(0) == 22 && data.at(1) == 3) // TLS?
					{
						s.enableCryptoServer(cap.certstore, [](Socket& s, Capture&& _cap)
						{
							CaptureServerPortOptCrypto& cap = *_cap.get<CaptureServerPortOptCrypto*>();
							cap.service->on_tunnel_established(s, *cap.service, *cap.server);
						}, &cap, cap.on_client_hello);
					}
					else
					{
						cap.service->on_tunnel_established(s, *cap.service, *cap.server);
					}
				}, this);
				
			}
		}
	};

	bool Server::bind(uint16_t port, ServerService* service) SOUP_EXCAL
	{
		Socket sock6{};
		if (!sock6.bind6(port))
		{
			return false;
		}
		setDataAvailableHandler6(sock6);
		sock6.holdup_callback.cap = CaptureServerPort(this, service);
		addSocket(std::move(sock6));

#if SOUP_WINDOWS
		Socket sock4{};
		if (!sock4.bind4(port))
		{
			return false;
		}
		setDataAvailableHandler4(sock4);
		sock4.holdup_callback.cap = CaptureServerPort(this, service);
		addSocket(std::move(sock4));
#endif

		return true;
	}

	bool Server::bind(const IpAddr& ip, uint16_t port, ServerService* service) SOUP_EXCAL
	{
		Socket sock{};
#if SOUP_WINDOWS
		if (!ip.isV4())
#endif
		{
			SOUP_RETHROW_FALSE(sock.bind6(SOCK_STREAM, port, ip));
			setDataAvailableHandler6(sock);
		}
#if SOUP_WINDOWS
		else
		{
			SOUP_RETHROW_FALSE(sock.bind4(SOCK_STREAM, port, ip));
			setDataAvailableHandler4(sock);
		}
#endif
		sock.holdup_callback.cap = CaptureServerPort(this, service);
		addSocket(std::move(sock));
		return true;
	}

	bool Server::bindCrypto(uint16_t port, ServerService* service, SharedPtr<CertStore> certstore, tls_server_on_client_hello_t on_client_hello) SOUP_EXCAL
	{
		Socket sock6{};
		if (!sock6.bind6(port))
		{
			return false;
		}
		setDataAvailableHandlerCrypto6(sock6);
		sock6.holdup_callback.cap = CaptureServerPortCrypto(this, service, certstore, on_client_hello);
		addSocket(std::move(sock6));

#if SOUP_WINDOWS
		Socket sock4{};
		if (!sock4.bind4(port))
		{
			return false;
		}
		setDataAvailableHandlerCrypto4(sock4);
		sock4.holdup_callback.cap = CaptureServerPortCrypto(this, service, certstore, on_client_hello);
		addSocket(std::move(sock4));
#endif

		return true;
	}

	bool Server::bindOptCrypto(uint16_t port, ServerService* service, SharedPtr<CertStore> certstore, tls_server_on_client_hello_t on_client_hello) SOUP_EXCAL
	{
		Socket sock6{};
		if (!sock6.bind6(port))
		{
			return false;
		}
		setDataAvailableHandlerOptCrypto6(sock6);
		sock6.holdup_callback.cap = CaptureServerPortOptCrypto(this, service, certstore, on_client_hello);
		addSocket(std::move(sock6));

#if SOUP_WINDOWS
		Socket sock4{};
		if (!sock4.bind4(port))
		{
			return false;
		}
		setDataAvailableHandlerOptCrypto4(sock4);
		sock4.holdup_callback.cap = CaptureServerPortOptCrypto(this, service, certstore, on_client_hello);
		addSocket(std::move(sock4));
#endif

		return true;
	}

	bool Server::bindUdp(uint16_t port, udp_callback_t callback) SOUP_EXCAL
	{
		Socket sock6{};
		if (!sock6.udpBind6(port))
		{
			return false;
		}
		setDataAvailableHandlerUdp(sock6, callback);
		addSocket(std::move(sock6));

#if SOUP_WINDOWS
		Socket sock4{};
		if (!sock4.udpBind4(port))
		{
			return false;
		}
		setDataAvailableHandlerUdp(sock4, callback);
		addSocket(std::move(sock4));
#endif

		return true;
	}

	bool Server::bindUdp(const IpAddr& addr, uint16_t port, udp_callback_t callback) SOUP_EXCAL
	{
		Socket sock{};
		if (!sock.udpBind(addr, port))
		{
			return false;
		}
		setDataAvailableHandlerUdp(sock, callback);
		addSocket(std::move(sock));
		return true;
	}

	bool Server::bindUdp(uint16_t port, ServerServiceUdp* service) SOUP_EXCAL
	{
		Socket sock6{};
		if (!sock6.udpBind6(port))
		{
			return false;
		}
		setDataAvailableHandlerUdp(sock6, service);
		addSocket(std::move(sock6));

#if SOUP_WINDOWS
		Socket sock4{};
		if (!sock4.udpBind4(port))
		{
			return false;
		}
		setDataAvailableHandlerUdp(sock4, service);
		addSocket(std::move(sock4));
#endif

		return true;
	}

	bool Server::bindUdp(const IpAddr& addr, uint16_t port, ServerServiceUdp* service) SOUP_EXCAL
	{
		Socket sock{};
		if (!sock.udpBind(addr, port))
		{
			return false;
		}
		setDataAvailableHandlerUdp(sock, service);
		addSocket(std::move(sock));
		return true;
	}

	void Server::setDataAvailableHandler6(Socket& s) noexcept
	{
		s.holdup_type = Worker::SOCKET;
		s.holdup_callback.fp = [](Worker& w, Capture&& cap) SOUP_EXCAL
		{
			auto& s = static_cast<Socket&>(w);
			cap.get<CaptureServerPort>().processAccept(s.accept6());
		};
	}

	void Server::setDataAvailableHandlerCrypto6(Socket& s) noexcept
	{
		s.holdup_type = Worker::SOCKET;
		s.holdup_callback.fp = [](Worker& w, Capture&& cap) SOUP_EXCAL
		{
			auto& s = static_cast<Socket&>(w);
			cap.get<CaptureServerPortCrypto>().processAccept(s.accept6());
		};
	}

	void Server::setDataAvailableHandlerOptCrypto6(Socket& s) noexcept
	{
		s.holdup_type = Worker::SOCKET;
		s.holdup_callback.fp = [](Worker& w, Capture&& cap) SOUP_EXCAL
		{
			auto& s = static_cast<Socket&>(w);
			cap.get<CaptureServerPortOptCrypto>().processAccept(s.accept6());
		};
	}

#if SOUP_WINDOWS
	void Server::setDataAvailableHandler4(Socket& s) noexcept
	{
		s.holdup_type = Worker::SOCKET;
		s.holdup_callback.fp = [](Worker& w, Capture&& cap) SOUP_EXCAL
		{
			auto& s = static_cast<Socket&>(w);
			cap.get<CaptureServerPort>().processAccept(s.accept4());
		};
	}

	void Server::setDataAvailableHandlerCrypto4(Socket& s) noexcept
	{
		s.holdup_type = Worker::SOCKET;
		s.holdup_callback.fp = [](Worker& w, Capture&& cap) SOUP_EXCAL
		{
			auto& s = static_cast<Socket&>(w);
			cap.get<CaptureServerPortCrypto>().processAccept(s.accept4());
		};
	}

	void Server::setDataAvailableHandlerOptCrypto4(Socket& s) noexcept
	{
		s.holdup_type = Worker::SOCKET;
		s.holdup_callback.fp = [](Worker& w, Capture&& cap) SOUP_EXCAL
		{
			auto& s = static_cast<Socket&>(w);
			cap.get<CaptureServerPortOptCrypto>().processAccept(s.accept4());
		};
	}
#endif

	void Server::setDataAvailableHandlerUdp(Socket& s, udp_callback_t callback) noexcept
	{
		s.udpRecv([](Socket& s, SocketAddr&& sender, std::string&& data, Capture&& cap)
		{
			cap.get<udp_callback_t>()(s, std::move(sender), std::move(data));
			setDataAvailableHandlerUdp(s, cap.get<udp_callback_t>());
		}, callback);
	}

	void Server::setDataAvailableHandlerUdp(Socket& s, ServerServiceUdp* service) noexcept
	{
		s.udpRecv([](Socket& s, SocketAddr&& sender, std::string&& data, Capture&& cap)
		{
			cap.get<ServerServiceUdp*>()->callback(s, std::move(sender), std::move(data), *cap.get<ServerServiceUdp*>());
			setDataAvailableHandlerUdp(s, cap.get<ServerServiceUdp*>());
		}, service);
	}
}

#endif