SNode.C
Loading...
Searching...
No Matches
SocketWriter.cpp
Go to the documentation of this file.
1/*
2 * SNode.C - A Slim Toolkit for Network Communication
3 * Copyright (C) Volker Christian <me@vchrist.at>
4 * 2020, 2021, 2022, 2023, 2024, 2025
5 *
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Lesser General Public License as published
8 * by the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20/*
21 * MIT License
22 *
23 * Permission is hereby granted, free of charge, to any person obtaining a copy
24 * of this software and associated documentation files (the "Software"), to deal
25 * in the Software without restriction, including without limitation the rights
26 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
27 * copies of the Software, and to permit persons to whom the Software is
28 * furnished to do so, subject to the following conditions:
29 *
30 * The above copyright notice and this permission notice shall be included in
31 * all copies or substantial portions of the Software.
32 *
33 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
39 * THE SOFTWARE.
40 */
41
42#include "core/socket/stream/tls/SocketWriter.h"
43
44#ifndef DOXYGEN_SHOULD_SKIP_THIS
45
46#include "core/socket/stream/tls/ssl_utils.h"
47#include "log/Logger.h"
48#include "utils/PreserveErrno.h"
49
50#include <cerrno>
51#include <openssl/ssl.h>
52#include <string>
53
54#endif // DOXYGEN_SHOULD_SKIP_THIS
55
56namespace core::socket::stream::tls {
57
58 ssize_t SocketWriter::write(const char* chunk, std::size_t chunkLen) {
59 ssize_t ret = 0;
60
61 if ((SSL_get_shutdown(ssl) & SSL_SENT_SHUTDOWN) != 0) {
62 ret = Super::write(chunk, chunkLen);
63 } else {
64 ret = SSL_write(ssl, chunk, static_cast<int>(chunkLen));
65
66 if (ret <= 0) {
67 const int ssl_err = SSL_get_error(ssl, static_cast<int>(ret));
68
69 switch (ssl_err) {
70 case SSL_ERROR_WANT_READ:
71 LOG(TRACE) << getName() << " SSL/TLS: Start renegotiation on read";
73 [this]() {
74 LOG(DEBUG) << getName() << " SSL/TLS: Renegotiation on read success";
75 },
76 [this]() {
77 LOG(WARNING) << getName() << " SSL/TLS: Renegotiation on read timed out";
78 },
79 [this](int ssl_err) {
80 ssl_log(getName() + " SSL/TLS: Renegotiation", ssl_err);
81 });
82 errno = EAGAIN;
83 ret = -1;
84 break;
85 case SSL_ERROR_WANT_WRITE:
86 errno = EAGAIN;
87 ret = -1;
88 break;
89 case SSL_ERROR_ZERO_RETURN: // shutdown cleanly
90 LOG(DEBUG) << getName() << " SSL/TLS: Close_notify received. Is EOF? " << (closeNotifyIsEOF ? "true" : "false");
91 errno = closeNotifyIsEOF ? EPIPE : EAGAIN;
92 ret = -1; // on the write side this means a TCP broken pipe
93 break;
94 case SSL_ERROR_SYSCALL:
95 // In case ret is -1 a real syscall error (RST = ECONNRESET)
96 {
97 const utils::PreserveErrno pe;
98
99 if (errno == EPIPE) {
100 PLOG(WARNING) << getName() << " SSL/TLS: Syscal error (SIGPIPE detected) on write.";
101 } else if (errno == ECONNRESET) {
102 PLOG(WARNING) << getName() << " SSL/TLS: Connection reset by peer (ECONNRESET).";
103 } else {
104 PLOG(WARNING) << getName() << " SSL/TLS: Syscall error on write";
105 }
106 }
107 ret = -1;
108 break;
109 case SSL_ERROR_SSL:
110 ssl_log(getName() + " SSL/TLS: Failed", ssl_err);
111 ret = -1;
112 break;
113 default:
114 ssl_log(getName() + " SSL/TLS: Unexpected error", ssl_err);
115 errno = EIO;
116 ret = -1;
117 break;
118 }
119 }
120 }
121
122 return ret;
123 }
124
125} // namespace core::socket::stream::tls
const std::string & getName() const
virtual ssize_t write(const char *chunk, std::size_t chunkLen)
virtual bool doSSLHandshake(const std::function< void()> &onSuccess, const std::function< void()> &onTimeout, const std::function< void(int)> &onStatus)=0
ssize_t write(const char *chunk, std::size_t chunkLen) override
void ssl_log(const std::string &message, int sslErr)