2016-03-14 04:51:04 +08:00
|
|
|
#pragma once
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
#include "io/network/stream_listener.hpp"
|
2016-03-14 04:51:04 +08:00
|
|
|
#include "memory/literals.hpp"
|
|
|
|
|
|
|
|
namespace io
|
|
|
|
{
|
|
|
|
using namespace memory::literals;
|
|
|
|
|
|
|
|
template <class Derived, class Stream>
|
2016-08-02 05:14:09 +08:00
|
|
|
class StreamReader : public StreamListener<Derived, Stream>
|
2016-03-14 04:51:04 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
struct Buffer
|
|
|
|
{
|
|
|
|
char* ptr;
|
|
|
|
size_t len;
|
|
|
|
};
|
|
|
|
|
2016-09-08 19:10:20 +08:00
|
|
|
StreamReader(uint32_t flags = 0) : StreamListener<Derived, Stream>(flags),
|
|
|
|
logger(logging::log->logger("io::StreamReader")) {}
|
2016-08-02 05:14:09 +08:00
|
|
|
|
|
|
|
bool accept(Socket& socket)
|
|
|
|
{
|
2016-09-08 19:10:20 +08:00
|
|
|
logger.trace("accept");
|
|
|
|
|
2016-08-02 05:14:09 +08:00
|
|
|
// accept a connection from a socket
|
|
|
|
auto s = socket.accept(nullptr, nullptr);
|
|
|
|
|
|
|
|
if(!s.is_open())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// make the recieved socket non blocking
|
|
|
|
s.set_non_blocking();
|
|
|
|
|
|
|
|
auto& stream = this->derived().on_connect(std::move(s));
|
|
|
|
|
|
|
|
// we want to listen to an incoming event which is edge triggered and
|
|
|
|
// we also want to listen on the hangup event
|
|
|
|
stream.event.events = EPOLLIN | EPOLLET | EPOLLRDHUP;
|
|
|
|
|
|
|
|
// add the connection to the event listener
|
|
|
|
this->add(stream);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2016-03-14 04:51:04 +08:00
|
|
|
|
|
|
|
void on_data(Stream& stream)
|
|
|
|
{
|
2016-09-08 19:10:20 +08:00
|
|
|
logger.trace("on data");
|
|
|
|
|
2016-03-14 04:51:04 +08:00
|
|
|
while(true)
|
|
|
|
{
|
2016-08-02 05:14:09 +08:00
|
|
|
if(UNLIKELY(!stream.alive()))
|
|
|
|
{
|
|
|
|
stream.close();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-03-14 04:51:04 +08:00
|
|
|
// allocate the buffer to fill the data
|
|
|
|
auto buf = this->derived().on_alloc(stream);
|
|
|
|
|
|
|
|
// read from the buffer at most buf.len bytes
|
|
|
|
buf.len = stream.socket.read(buf.ptr, buf.len);
|
|
|
|
|
|
|
|
// check for read errors
|
|
|
|
if(buf.len == -1)
|
|
|
|
{
|
|
|
|
// this means we have read all available data
|
|
|
|
if(LIKELY(errno == EAGAIN))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// some other error occurred, check errno
|
|
|
|
this->derived().on_error(stream);
|
2016-08-02 05:14:09 +08:00
|
|
|
break;
|
2016-03-14 04:51:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// end of file, the client has closed the connection
|
|
|
|
if(UNLIKELY(buf.len == 0))
|
|
|
|
{
|
|
|
|
stream.close();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->derived().on_read(stream, buf);
|
|
|
|
}
|
|
|
|
}
|
2016-09-08 19:10:20 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
Logger logger;
|
2016-03-14 04:51:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|