SNode.C
Loading...
Searching...
No Matches
SocketReader.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/SocketReader.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 <limits>
52#include <openssl/ssl.h>
53#include <string>
54
55#endif // DOXYGEN_SHOULD_SKIP_THIS
56
57namespace core::socket::stream::tls {
58
59 ssize_t SocketReader::read(char* chunk, std::size_t chunkLen) {
60 ssize_t ret = 0;
61
62 if ((SSL_get_shutdown(ssl) & SSL_RECEIVED_SHUTDOWN) != 0) {
63 ret = Super::read(chunk, chunkLen);
64 } else {
65 chunkLen = chunkLen > std::numeric_limits<int>::max() ? std::numeric_limits<int>::max() : chunkLen;
66 ret = SSL_read(ssl, chunk, static_cast<int>(chunkLen));
67
68 if (ret <= 0) {
69 const int ssl_err = SSL_get_error(ssl, static_cast<int>(ret));
70
71 switch (ssl_err) {
72 case SSL_ERROR_WANT_READ:
73 errno = EAGAIN;
74 ret = -1;
75 break;
76 case SSL_ERROR_WANT_WRITE:
77 LOG(TRACE) << getName() << " SSL/TLS: Start renegotiation on read";
79 [this]() {
80 LOG(DEBUG) << getName() << " SSL/TLS: Renegotiation on read success";
81 },
82 [this]() {
83 LOG(WARNING) << getName() << " SSL/TLS: Renegotiation on read timed out";
84 },
85 [this](int ssl_err) {
86 ssl_log(getName() + " SSL/TLS: Renegotiation on read", ssl_err);
87 });
88 errno = EAGAIN;
89 ret = -1;
90 break;
91 case SSL_ERROR_ZERO_RETURN: // received close_notify
93 errno = EAGAIN;
94 ret = -1;
95 break;
96 case SSL_ERROR_SYSCALL: // When SSL_get_error(ssl, ret) returns SSL_ERROR_SYSCALL
97 // and ret is 0, it indicates that the underlying TCP connection
98 // was closed unexpectedly by the peer. This situation typically
99 // happens when the peer closes (FIN) the connection without
100 // sending a close_notify alert, which violates the SSL/TLS
101 // protocol’s graceful shutdown procedure.
102 // In case ret is -1 a real syscall error (RST = ECONNRESET)
103 {
104 const utils::PreserveErrno pe;
105
106 if (ret == 0) {
107 PLOG(DEBUG) << getName() << " SSL/TLS: EOF detected: Connection closed by peer.";
108 } else {
109 PLOG(WARNING) << getName() + " SSL/TLS: Syscall error on read";
110 }
111 }
112 ret = -1;
113 break;
114 case SSL_ERROR_SSL:
115 ssl_log(getName() + " SSL/TLS: Read failed", ssl_err);
117 ret = -1;
118 break;
119 default:
120 ssl_log(getName() + " SSL/TLS: Unexpected error", ssl_err);
122 errno = EIO;
123 ret = -1;
124 break;
125 }
126 }
127 }
128
129 return ret;
130 }
131
132} // namespace core::socket::stream::tls
const std::string & getName() const
virtual ssize_t read(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 read(char *chunk, std::size_t chunkLen) override
core::socket::stream::SocketReader Super
void ssl_log(const std::string &message, int sslErr)