aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/clickhouse/src/Client/PacketReceiver.h
blob: deedf5cccdc7389548294de4a07e042b2d87179a (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#pragma once

#if defined(OS_LINUX)

#include <variant>

#include <Client/IConnections.h>
#include <Common/FiberStack.h>
#include <Common/Fiber.h>
#include <Common/Epoll.h>
#include <Common/TimerDescriptor.h>
#include <Common/AsyncTaskExecutor.h>

namespace DB
{

/// Class for nonblocking packet receiving. It runs connection->receivePacket
/// in fiber and sets special read callback which is called when
/// reading from socket blocks. When read callback is called,
/// socket and receive timeout are added in epoll and execution returns to the main program.
/// So, you can poll this epoll file descriptor to determine when to resume
/// packet receiving.
class PacketReceiver : public AsyncTaskExecutor
{
public:
    explicit PacketReceiver(Connection * connection_);

    bool isPacketReady() const { return !is_read_in_process && !is_timeout_expired && !exception; }
    Packet getPacket() { return std::move(packet); }

    bool hasException() const { return exception.operator bool(); }
    std::exception_ptr getException() const { return exception; }

    bool isTimeoutExpired() const { return is_timeout_expired; }
    Poco::Timespan getTimeout() const { return timeout; }

    void setTimeout(const Poco::Timespan & timeout_)
    {
        timeout_descriptor.setRelative(timeout_);
        timeout = timeout_;
    }

    int getFileDescriptor() const { return epoll.getFileDescriptor(); }

private:
    bool checkBeforeTaskResume() override;
    void afterTaskResume() override {}

    void processAsyncEvent(int fd, Poco::Timespan socket_timeout, AsyncEventTimeoutType, const std::string &, uint32_t) override;
    void clearAsyncEvent() override;

    void processException(std::exception_ptr e) override { exception = e; }

    struct Task : public AsyncTask
    {
        Task(PacketReceiver & receiver_) : receiver(receiver_) {}

        PacketReceiver & receiver;

        void run(AsyncCallback async_callback, SuspendCallback suspend_callback) override;
    };

    /// When epoll file descriptor is ready, check if it's an expired timeout.
    /// Return false if receive timeout expired and socket is not ready, return true otherwise.
    bool checkTimeout();

    Connection * connection;
    int socket_fd = -1;
    Packet packet;

    /// We use timer descriptor for checking socket timeouts.
    TimerDescriptor timeout_descriptor;
    Poco::Timespan timeout;
    bool is_timeout_expired = false;

    /// In read callback we add socket file descriptor and timer descriptor with receive timeout
    /// in epoll, so we can return epoll file descriptor outside for polling.
    Epoll epoll;

    /// If and exception occurred in fiber resume, we save it and rethrow.
    std::exception_ptr exception;

    bool is_read_in_process = false;
};

}
#endif