test: put the generic parts from StaticContentsSock into a separate class

This allows reusing them in other mocked implementations.
This commit is contained in:
Vasil Dimov 2024-09-06 11:16:50 +02:00
parent 4b58d55878
commit f1864148c4
No known key found for this signature in database
GPG Key ID: 54DF06F64B55CBBF
2 changed files with 83 additions and 48 deletions

View File

@ -138,38 +138,31 @@ std::vector<NodeEvictionCandidate> GetRandomNodeEvictionCandidates(int n_candida
return candidates;
}
StaticContentsSock::StaticContentsSock(const std::string& contents)
: Sock{INVALID_SOCKET}, m_contents{contents}
// Have different ZeroSock (or others that inherit from it) objects have different
// m_socket because EqualSharedPtrSock compares m_socket and we want to avoid two
// different objects comparing as equal.
static std::atomic<SOCKET> g_mocked_sock_fd{0};
ZeroSock::ZeroSock() : Sock{g_mocked_sock_fd++} {}
// Sock::~Sock() would try to close(2) m_socket if it is not INVALID_SOCKET, avoid that.
ZeroSock::~ZeroSock() { m_socket = INVALID_SOCKET; }
ssize_t ZeroSock::Send(const void*, size_t len, int) const { return len; }
ssize_t ZeroSock::Recv(void* buf, size_t len, int flags) const
{
memset(buf, 0x0, len);
return len;
}
StaticContentsSock::~StaticContentsSock() { m_socket = INVALID_SOCKET; }
int ZeroSock::Connect(const sockaddr*, socklen_t) const { return 0; }
StaticContentsSock& StaticContentsSock::operator=(Sock&& other)
{
assert(false && "Move of Sock into MockSock not allowed.");
return *this;
}
int ZeroSock::Bind(const sockaddr*, socklen_t) const { return 0; }
ssize_t StaticContentsSock::Send(const void*, size_t len, int) const { return len; }
int ZeroSock::Listen(int) const { return 0; }
ssize_t StaticContentsSock::Recv(void* buf, size_t len, int flags) const
{
const size_t consume_bytes{std::min(len, m_contents.size() - m_consumed)};
std::memcpy(buf, m_contents.data() + m_consumed, consume_bytes);
if ((flags & MSG_PEEK) == 0) {
m_consumed += consume_bytes;
}
return consume_bytes;
}
int StaticContentsSock::Connect(const sockaddr*, socklen_t) const { return 0; }
int StaticContentsSock::Bind(const sockaddr*, socklen_t) const { return 0; }
int StaticContentsSock::Listen(int) const { return 0; }
std::unique_ptr<Sock> StaticContentsSock::Accept(sockaddr* addr, socklen_t* addr_len) const
std::unique_ptr<Sock> ZeroSock::Accept(sockaddr* addr, socklen_t* addr_len) const
{
if (addr != nullptr) {
// Pretend all connections come from 5.5.5.5:6789
@ -183,30 +176,28 @@ std::unique_ptr<Sock> StaticContentsSock::Accept(sockaddr* addr, socklen_t* addr
addr_in->sin_port = htons(6789);
}
}
return std::make_unique<StaticContentsSock>("");
};
return std::make_unique<ZeroSock>();
}
int StaticContentsSock::GetSockOpt(int level, int opt_name, void* opt_val, socklen_t* opt_len) const
int ZeroSock::GetSockOpt(int level, int opt_name, void* opt_val, socklen_t* opt_len) const
{
std::memset(opt_val, 0x0, *opt_len);
return 0;
}
int StaticContentsSock::SetSockOpt(int, int, const void*, socklen_t) const { return 0; }
int ZeroSock::SetSockOpt(int, int, const void*, socklen_t) const { return 0; }
int StaticContentsSock::GetSockName(sockaddr* name, socklen_t* name_len) const
int ZeroSock::GetSockName(sockaddr* name, socklen_t* name_len) const
{
std::memset(name, 0x0, *name_len);
return 0;
}
bool StaticContentsSock::SetNonBlocking() const { return true; }
bool ZeroSock::SetNonBlocking() const { return true; }
bool StaticContentsSock::IsSelectable() const { return true; }
bool ZeroSock::IsSelectable() const { return true; }
bool StaticContentsSock::Wait(std::chrono::milliseconds timeout,
Event requested,
Event* occurred) const
bool ZeroSock::Wait(std::chrono::milliseconds timeout, Event requested, Event* occurred) const
{
if (occurred != nullptr) {
*occurred = requested;
@ -214,7 +205,7 @@ bool StaticContentsSock::Wait(std::chrono::milliseconds timeout,
return true;
}
bool StaticContentsSock::WaitMany(std::chrono::milliseconds timeout, EventsPerSock& events_per_sock) const
bool ZeroSock::WaitMany(std::chrono::milliseconds timeout, EventsPerSock& events_per_sock) const
{
for (auto& [sock, events] : events_per_sock) {
(void)sock;
@ -223,7 +214,29 @@ bool StaticContentsSock::WaitMany(std::chrono::milliseconds timeout, EventsPerSo
return true;
}
bool StaticContentsSock::IsConnected(std::string&) const
ZeroSock& ZeroSock::operator=(Sock&& other)
{
return true;
assert(false && "Move of Sock into ZeroSock not allowed.");
return *this;
}
StaticContentsSock::StaticContentsSock(const std::string& contents)
: m_contents{contents}
{
}
ssize_t StaticContentsSock::Recv(void* buf, size_t len, int flags) const
{
const size_t consume_bytes{std::min(len, m_contents.size() - m_consumed)};
std::memcpy(buf, m_contents.data() + m_consumed, consume_bytes);
if ((flags & MSG_PEEK) == 0) {
m_consumed += consume_bytes;
}
return consume_bytes;
}
StaticContentsSock& StaticContentsSock::operator=(Sock&& other)
{
assert(false && "Move of Sock into StaticContentsSock not allowed.");
return *this;
}

View File

@ -134,18 +134,15 @@ constexpr auto ALL_NETWORKS = std::array{
};
/**
* A mocked Sock alternative that returns a statically contained data upon read and succeeds
* and ignores all writes. The data to be returned is given to the constructor and when it is
* exhausted an EOF is returned by further reads.
* A mocked Sock alternative that succeeds on all operations.
* Returns infinite amount of 0x0 bytes on reads.
*/
class StaticContentsSock : public Sock
class ZeroSock : public Sock
{
public:
explicit StaticContentsSock(const std::string& contents);
ZeroSock();
~StaticContentsSock() override;
StaticContentsSock& operator=(Sock&& other) override;
~ZeroSock() override;
ssize_t Send(const void*, size_t len, int) const override;
@ -175,9 +172,34 @@ public:
bool WaitMany(std::chrono::milliseconds timeout, EventsPerSock& events_per_sock) const override;
bool IsConnected(std::string&) const override;
private:
ZeroSock& operator=(Sock&& other) override;
};
/**
* A mocked Sock alternative that returns a statically contained data upon read and succeeds
* and ignores all writes. The data to be returned is given to the constructor and when it is
* exhausted an EOF is returned by further reads.
*/
class StaticContentsSock : public ZeroSock
{
public:
explicit StaticContentsSock(const std::string& contents);
/**
* Return parts of the contents that was provided at construction until it is exhausted
* and then return 0 (EOF).
*/
ssize_t Recv(void* buf, size_t len, int flags) const override;
bool IsConnected(std::string&) const override
{
return true;
}
private:
StaticContentsSock& operator=(Sock&& other) override;
const std::string m_contents;
mutable size_t m_consumed{0};
};