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
92 LOG(DEBUG) << getName() << " SSL/TLS: Close_notify is" << (closeNotifyIsEOF ? " " : " not ") << "EOF";
94 errno = closeNotifyIsEOF ? 0 : EAGAIN;
95 ret = closeNotifyIsEOF ? 0 : -1;
96 break;
97 case SSL_ERROR_SYSCALL: // When SSL_get_error(ssl, ret) returns SSL_ERROR_SYSCALL
98 // and ret is 0, it indicates that the underlying TCP connection
99 // was closed unexpectedly by the peer. This situation typically
100 // happens when the peer closes (FIN) the connection without
101 // sending a close_notify alert, which violates the SSL/TLS
102 // protocol’s graceful shutdown procedure.
103 // In case ret is -1 a real syscall error (RST = ECONNRESET)
104 {
105 const utils::PreserveErrno pe;
106
107 if (ret == 0) {
108 PLOG(DEBUG) << getName() << " SSL/TLS: EOF detected: Connection closed by peer.";
109 } else {
110 PLOG(WARNING) << getName() + " SSL/TLS: Syscall error on read";
111 }
112 }
113 ret = -1;
114 break;
115 case SSL_ERROR_SSL:
116 ssl_log(getName() + " SSL/TLS: Read failed", ssl_err);
118 ret = -1;
119 break;
120 default:
121 ssl_log(getName() + " SSL/TLS: Unexpected error", ssl_err);
123 errno = EIO;
124 ret = -1;
125 break;
126 }
127 }
128 }
129
130 return ret;
131 }
132
133} // 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
void ssl_log(const std::string &message, int sslErr)