/* * Author: Patrick-Christopher Mattulat * Company: Lynar Studios * E-Mail: webmaster@lynarstudios.com * Created: 2022-11-16 * Changed: 2022-12-28 * * */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include ls::std::network::Socket::Socket(ls::std::network::SocketParameter _parameter) : ls::std::core::Class("Socket"), parameter(::std::move(_parameter)) { this->_init(); } ls::std::network::Socket::~Socket() { delete[] this->readBuffer; #if LS_STD_UNIX_PLATFORM for (const auto& connection : this->unixDescriptors) { if (!this->_closeUnix(connection.second)) { ::std::cerr << "could not close socket with id \"" << connection.first << "\"" << ::std::endl; } } #endif } ls::std::core::type::byte_field ls::std::network::Socket::read() { if (!this->readBufferSet) { this->_initReadBuffer(); // TODO: is this really wise to initialize once? what if reading second time wouldn't work? - test it with unit tests and remove it this->readBufferSet = true; } return this->_read(); } bool ls::std::network::Socket::write(const ls::std::core::type::byte_field &_data) { return this->_write(_data); } ls::std::core::type::connection_id ls::std::network::Socket::accept() { if (this->parameter.socketAddress.protocolType != LS_STD_PROTOCOL_TYPE_TCP) { throw ls::std::core::WrongProtocolException{}; } #if LS_STD_UNIX_PLATFORM return ls::std::network::Socket::_acceptUnix(); #endif } bool ls::std::network::Socket::bind() { #if LS_STD_UNIX_PLATFORM return ls::std::network::Socket::_bindUnix(); #endif } bool ls::std::network::Socket::close() { return this->_close(); } bool ls::std::network::Socket::connect() { #if LS_STD_UNIX_PLATFORM return ls::std::network::Socket::_connectUnix(); #endif } bool ls::std::network::Socket::handle(const ls::std::core::type::connection_id &_connectionId) { return this->_handle(_connectionId); } bool ls::std::network::Socket::handle() { return this->_handle(1); } bool ls::std::network::Socket::isInitialized() const { return this->initialized; } bool ls::std::network::Socket::listen() { if (this->parameter.socketAddress.protocolType != LS_STD_PROTOCOL_TYPE_TCP) { throw ls::std::core::WrongProtocolException{}; } #if LS_STD_UNIX_PLATFORM return ls::std::network::Socket::_listenUnix(); #endif } #if LS_STD_UNIX_PLATFORM ls::std::core::type::connection_id ls::std::network::Socket::_acceptUnix() { ::sockaddr_in incoming{}; ::socklen_t length{}; ls::std::core::type::connection_id acceptedDescriptor = this->parameter.posixSocket->accept(this->unixDescriptors.at(1), reinterpret_cast(&incoming), &length); if (acceptedDescriptor >= 0) { this->_addUnixDescriptor(acceptedDescriptor); } else { throw ls::std::core::SocketOperationFailedException{}; } return this->unixUniqueDescriptorId; } void ls::std::network::Socket::_addUnixDescriptor(const int &_descriptor) { ++this->unixUniqueDescriptorId; this->unixDescriptors.insert({this->unixUniqueDescriptorId, _descriptor}); } bool ls::std::network::Socket::_bindUnix() { ls::std::network::ConvertedSocketAddress convertedSocketAddress = ls::std::network::SocketAddressMapper::from(ls::std::network::Socket::_createSocketAddressMapperParameter()); return this->parameter.posixSocket->bind(this->unixDescriptors.at(1), reinterpret_cast(&convertedSocketAddress.socketAddressUnix), convertedSocketAddress.addressLength) == 0; } #endif bool ls::std::network::Socket::_close() { #if LS_STD_UNIX_PLATFORM return ls::std::network::Socket::_closeUnix(this->unixDescriptors.at(1)); #endif } #if LS_STD_UNIX_PLATFORM bool ls::std::network::Socket::_closeUnix(const int& _descriptor) { return this->parameter.posixSocket->close(_descriptor) == 0; } bool ls::std::network::Socket::_connectUnix() { ls::std::network::ConvertedSocketAddress convertedSocketAddress = ls::std::network::SocketAddressMapper::from(ls::std::network::Socket::_createSocketAddressMapperParameter()); return this->parameter.posixSocket->connect(this->unixDescriptors.at(1), reinterpret_cast(&convertedSocketAddress.socketAddressUnix), convertedSocketAddress.addressLength) == 0; } #endif ls::std::network::SocketAddressMapperParameter ls::std::network::Socket::_createSocketAddressMapperParameter() const { ls::std::network::SocketAddressMapperParameter mapperParameter{}; mapperParameter.socketAddress = this->parameter.socketAddress; mapperParameter.protocolFamilyType = this->parameter.protocolFamilyType; return mapperParameter; } bool ls::std::network::Socket::_hasAcceptedConnection(const ls::std::core::type::connection_id &_connectionId) { #if LS_STD_UNIX_PLATFORM return this->_hasAcceptedConnectionUnix(_connectionId); #endif } #if LS_STD_UNIX_PLATFORM bool ls::std::network::Socket::_hasAcceptedConnectionUnix(const ls::std::core::type::connection_id &_connectionId) { return this->unixDescriptors.find(_connectionId) != this->unixDescriptors.end(); } #endif bool ls::std::network::Socket::_handle(const ls::std::core::type::connection_id& _connectionId) { bool focusSet{}; if (this->_hasAcceptedConnection(_connectionId)) { this->currentAcceptedConnection = _connectionId; focusSet = true; } return focusSet; } void ls::std::network::Socket::_init() { #if LS_STD_UNIX_PLATFORM this->initialized = ls::std::network::Socket::_initUnix(); #endif } void ls::std::network::Socket::_initReadBuffer() { if (this->parameter.readBufferSize <= 0) { throw ls::std::core::IllegalArgumentException{}; } this->readBuffer = new ls::std::core::type::byte[this->parameter.readBufferSize]; } #if LS_STD_UNIX_PLATFORM bool ls::std::network::Socket::_initUnix() { this->_setPosixReaderApi(); this->_setPosixSocketApi(); this->_setPosixWriterApi(); ls::std::network::ConvertedProtocolFamily convertedProtocolFamily = ls::std::network::ProtocolFamilyMapper::from(this->parameter.protocolFamilyType); ls::std::network::Protocol protocol = ls::std::network::ProtocolMapper::from(this->parameter.socketAddress.protocolType); int descriptor = this->parameter.posixSocket->create(convertedProtocolFamily.unixDomain, protocol.unixProtocol, 0); bool success = descriptor != -1; if (success) { this->_addUnixDescriptor(descriptor); this->currentAcceptedConnection = 1; } return success; } bool ls::std::network::Socket::_listenUnix() { return this->parameter.posixSocket->listen(this->unixDescriptors.at(1), this->parameter.queueSize) == 0; } #endif ls::std::core::type::byte_field ls::std::network::Socket::_read() { #if LS_STD_UNIX_PLATFORM return this->_readUnix(this->unixDescriptors.at(this->currentAcceptedConnection)); #endif } #if LS_STD_UNIX_PLATFORM ls::std::core::type::byte_field ls::std::network::Socket::_readUnix(const int& _descriptor) { size_t size = this->parameter.posixReader->read(_descriptor, this->readBuffer, this->parameter.readBufferSize); if (size == -1) { throw ls::std::core::FileOperationException{}; } return ls::std::core::type::byte_field{this->readBuffer, size}; } void ls::std::network::Socket::_setPosixReaderApi() { if (this->parameter.posixReader == nullptr) { this->parameter.posixReader = ::std::make_shared(); } } void ls::std::network::Socket::_setPosixSocketApi() { if (this->parameter.posixSocket == nullptr) { this->parameter.posixSocket = ::std::make_shared(); } } void ls::std::network::Socket::_setPosixWriterApi() { if (this->parameter.posixWriter == nullptr) { this->parameter.posixWriter = ::std::make_shared(); } } #endif bool ls::std::network::Socket::_write(const ls::std::core::type::byte_field &_data) { #if LS_STD_UNIX_PLATFORM return this->_writeUnix(this->unixDescriptors.at(this->currentAcceptedConnection), _data); #endif } #if LS_STD_UNIX_PLATFORM bool ls::std::network::Socket::_writeUnix(const int& _descriptor, const ls::std::core::type::byte_field &_data) { bool written{}; if (!_data.empty()) { size_t size = _data.size() + 1; char* buffer = new char[size]; ::std::strcpy(buffer, _data.c_str()); written = this->parameter.posixWriter->write(_descriptor, buffer, size) != -1; delete[] buffer; } return written; } #endif