Refactor client codes to use similar, standardized API
This commit is contained in:
29
clients/c_cpp/example.c
Normal file
29
clients/c_cpp/example.c
Normal file
@ -0,0 +1,29 @@
|
||||
#include "pipe_io.h"
|
||||
|
||||
|
||||
void run2() {
|
||||
pipe_io pipeio = pipeio_new_io("in", "out");
|
||||
const char* line;
|
||||
while ((line = pipeio_recv_msg(&pipeio.reader))) {
|
||||
pipeio_send_msg(&pipeio.writer, line);
|
||||
}
|
||||
pipeio_close_io(&pipeio);
|
||||
}
|
||||
|
||||
void msg_handler(const char* msg, void* ctx) {
|
||||
pipe_writer* writer = (pipe_writer*)(ctx);
|
||||
pipeio_send_msg(writer, msg);
|
||||
}
|
||||
|
||||
void run1() {
|
||||
pipe_io pipeio = pipeio_new_io("in", "out");
|
||||
pipeio_set_msg_handler(&pipeio.reader, &msg_handler, &pipeio.writer);
|
||||
pipeio_run(&pipeio.reader);
|
||||
pipeio_close_io(&pipeio);
|
||||
}
|
||||
|
||||
int main() {
|
||||
run1(); // or run2();
|
||||
// note that run1 and run2 do the same thing
|
||||
// and you can use whichever API you prefer
|
||||
}
|
13
clients/c_cpp/example.cpp
Normal file
13
clients/c_cpp/example.cpp
Normal file
@ -0,0 +1,13 @@
|
||||
#include <iostream>
|
||||
#include <string>
|
||||
#include <fstream>
|
||||
#include "pipe_io.hpp"
|
||||
|
||||
|
||||
int main() {
|
||||
PipeIO pipe_io("in", "out");
|
||||
pipe_io.reader.set_message_handler(
|
||||
[&pipe_io](std::string msg){pipe_io.writer.send_message(msg);}
|
||||
);
|
||||
pipe_io.reader.run();
|
||||
}
|
@ -9,25 +9,30 @@ typedef struct pipe_reader {
|
||||
FILE* stream;
|
||||
char* line_buf;
|
||||
size_t buf_size;
|
||||
void (*msg_handler)(const char*, void*);
|
||||
void* handler_ctx;
|
||||
} pipe_reader;
|
||||
|
||||
pipe_reader new_pipe_reader(const char* pipe_path) {
|
||||
static void null_handler(const char* msg, void* ctx) {}
|
||||
|
||||
pipe_reader pipeio_new_reader(const char* pipe_path) {
|
||||
pipe_reader preader;
|
||||
preader.stream = fopen(pipe_path, "r");
|
||||
preader.line_buf = NULL;
|
||||
preader.buf_size = 0;
|
||||
preader.msg_handler = &null_handler;
|
||||
return preader;
|
||||
}
|
||||
|
||||
bool check_pipe_reader(const pipe_reader* preader) {
|
||||
bool pipeio_check_reader(const pipe_reader* preader) {
|
||||
if (!preader || preader->stream == NULL) {
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
const char* recv_msg_pipe_reader(pipe_reader* preader) {
|
||||
if (!check_pipe_reader(preader)) {
|
||||
const char* pipeio_recv_msg(pipe_reader* preader) {
|
||||
if (!pipeio_check_reader(preader)) {
|
||||
return NULL;
|
||||
}
|
||||
ssize_t read = getline(&preader->line_buf, &preader->buf_size, preader->stream);
|
||||
@ -38,8 +43,22 @@ const char* recv_msg_pipe_reader(pipe_reader* preader) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
void close_pipe_reader(pipe_reader* preader) {
|
||||
if (!check_pipe_reader(preader)) {
|
||||
void pipeio_set_msg_handler(pipe_reader* preader, void (*msg_handler)(const char*, void*), void* handler_ctx) {
|
||||
if (msg_handler != NULL && handler_ctx != NULL) {
|
||||
preader->msg_handler = msg_handler;
|
||||
preader->handler_ctx = handler_ctx;
|
||||
}
|
||||
}
|
||||
|
||||
void pipeio_run(pipe_reader* preader) {
|
||||
const char* line;
|
||||
while ((line = pipeio_recv_msg(preader))) {
|
||||
(*preader->msg_handler)(line, preader->handler_ctx);
|
||||
}
|
||||
}
|
||||
|
||||
void pipeio_close_reader(pipe_reader* preader) {
|
||||
if (!pipeio_check_reader(preader)) {
|
||||
return;
|
||||
}
|
||||
fclose(preader->stream);
|
||||
@ -55,21 +74,21 @@ typedef struct pipe_writer {
|
||||
FILE* stream;
|
||||
} pipe_writer;
|
||||
|
||||
pipe_writer new_pipe_writer(const char* pipe_path) {
|
||||
pipe_writer pipeio_new_writer(const char* pipe_path) {
|
||||
pipe_writer pwriter;
|
||||
pwriter.stream = fopen(pipe_path, "w");
|
||||
return pwriter;
|
||||
}
|
||||
|
||||
bool check_pipe_writer(const pipe_writer* pwriter) {
|
||||
bool pipeio_check_writer(const pipe_writer* pwriter) {
|
||||
if (!pwriter || pwriter->stream == NULL) {
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
bool send_msg_pipe_writer(pipe_writer* pwriter, const char* msg) {
|
||||
if (!check_pipe_writer(pwriter)) {
|
||||
bool pipeio_send_msg(pipe_writer* pwriter, const char* msg) {
|
||||
if (!pipeio_check_writer(pwriter)) {
|
||||
return false;
|
||||
}
|
||||
fprintf(pwriter->stream, "%s\n", msg);
|
||||
@ -77,8 +96,8 @@ bool send_msg_pipe_writer(pipe_writer* pwriter, const char* msg) {
|
||||
return true;
|
||||
}
|
||||
|
||||
void close_pipe_writer(pipe_writer* pwriter) {
|
||||
if (!check_pipe_writer(pwriter)) {
|
||||
void pipeio_close_writer(pipe_writer* pwriter) {
|
||||
if (!pipeio_check_writer(pwriter)) {
|
||||
return;
|
||||
}
|
||||
fclose(pwriter->stream);
|
||||
@ -87,42 +106,42 @@ void close_pipe_writer(pipe_writer* pwriter) {
|
||||
|
||||
|
||||
typedef struct pipe_io {
|
||||
pipe_reader preader;
|
||||
pipe_writer pwriter;
|
||||
pipe_reader reader;
|
||||
pipe_writer writer;
|
||||
} pipe_io;
|
||||
|
||||
pipe_io new_pipe_io(const char* in_pipe_path, const char* out_pipe_path) {
|
||||
pipe_io pipeio_new_io(const char* in_pipe_path, const char* out_pipe_path) {
|
||||
pipe_io pipeio;
|
||||
pipeio.preader = new_pipe_reader(in_pipe_path);
|
||||
pipeio.pwriter = new_pipe_writer(out_pipe_path);
|
||||
pipeio.reader = pipeio_new_reader(in_pipe_path);
|
||||
pipeio.writer = pipeio_new_writer(out_pipe_path);
|
||||
return pipeio;
|
||||
}
|
||||
|
||||
bool check_pipe_io(const pipe_io* pipeio) {
|
||||
bool pipeio_check_io(const pipe_io* pipeio) {
|
||||
if (!pipeio) {
|
||||
return false;
|
||||
}
|
||||
return (check_pipe_reader(&pipeio->preader) && check_pipe_writer(&pipeio->pwriter));
|
||||
return (pipeio_check_reader(&pipeio->reader) && pipeio_check_writer(&pipeio->writer));
|
||||
}
|
||||
|
||||
const char* recv_msg_pipe_io(pipe_io* pipeio) {
|
||||
if (!check_pipe_io(pipeio)) {
|
||||
return NULL;
|
||||
}
|
||||
return recv_msg_pipe_reader(&pipeio->preader);
|
||||
}
|
||||
|
||||
bool send_msg_pipe_io(pipe_io* pipeio, const char* msg) {
|
||||
if (!check_pipe_io(pipeio)) {
|
||||
return false;
|
||||
}
|
||||
return send_msg_pipe_writer(&pipeio->pwriter, msg);
|
||||
}
|
||||
|
||||
void close_pipe_io(pipe_io* pipeio) {
|
||||
if (!check_pipe_io(pipeio)) {
|
||||
void pipeio_close_io(pipe_io* pipeio) {
|
||||
if (!pipeio_check_io(pipeio)) {
|
||||
return;
|
||||
}
|
||||
close_pipe_reader(&pipeio->preader);
|
||||
close_pipe_writer(&pipeio->pwriter);
|
||||
pipeio_close_reader(&pipeio->reader);
|
||||
pipeio_close_writer(&pipeio->writer);
|
||||
}
|
||||
|
||||
// const char* recv_msg_pipe_io(pipe_io* pipeio) {
|
||||
// if (!check_pipe_io(pipeio)) {
|
||||
// return NULL;
|
||||
// }
|
||||
// return recv_msg_pipe_reader(&pipeio->preader);
|
||||
// }
|
||||
|
||||
// bool send_msg_pipe_io(pipe_io* pipeio, const char* msg) {
|
||||
// if (!check_pipe_io(pipeio)) {
|
||||
// return false;
|
||||
// }
|
||||
// return send_msg_pipe_writer(&pipeio->pwriter, msg);
|
||||
// }
|
||||
|
@ -3,15 +3,16 @@ extern "C" {
|
||||
}
|
||||
#include <string>
|
||||
#include <iostream>
|
||||
#include <functional>
|
||||
|
||||
|
||||
class PipeReader {
|
||||
public:
|
||||
PipeReader(const char* pipe_path)
|
||||
:preader(new_pipe_reader(pipe_path)) {}
|
||||
PipeReader(const std::string& pipe_path)
|
||||
:preader(pipeio_new_reader(pipe_path.c_str())) {}
|
||||
|
||||
virtual ~PipeReader() {
|
||||
close_pipe_reader(&this->preader);
|
||||
pipeio_close_reader(&this->preader);
|
||||
}
|
||||
|
||||
PipeReader(PipeReader&) = delete;
|
||||
@ -19,31 +20,34 @@ public:
|
||||
PipeReader& operator=(const PipeReader&) = delete;
|
||||
PipeReader& operator=(const PipeReader&&) = delete;
|
||||
|
||||
void set_message_handler(std::function<void(std::string)> message_handler) {
|
||||
this->message_handler = message_handler;
|
||||
}
|
||||
|
||||
void run() {
|
||||
while (const char* linebuf = this->recv_message()) {
|
||||
std::string msg(linebuf);
|
||||
this->handle_message(msg);
|
||||
this->message_handler(msg);
|
||||
}
|
||||
}
|
||||
|
||||
const char* recv_message() {
|
||||
return recv_msg_pipe_reader(&this->preader);
|
||||
return pipeio_recv_msg(&this->preader);
|
||||
}
|
||||
|
||||
virtual void handle_message(std::string msg) {}
|
||||
|
||||
private:
|
||||
pipe_reader preader;
|
||||
std::function<void(std::string)> message_handler = [](std::string){};
|
||||
};
|
||||
|
||||
|
||||
class PipeWriter {
|
||||
public:
|
||||
PipeWriter(const char* pipe_path)
|
||||
:pwriter(new_pipe_writer(pipe_path)) {}
|
||||
PipeWriter(const std::string& pipe_path)
|
||||
:pwriter(pipeio_new_writer(pipe_path.c_str())) {}
|
||||
|
||||
virtual ~PipeWriter() {
|
||||
close_pipe_writer(&this->pwriter);
|
||||
pipeio_close_writer(&this->pwriter);
|
||||
}
|
||||
|
||||
PipeWriter(PipeReader&) = delete;
|
||||
@ -52,7 +56,7 @@ public:
|
||||
PipeWriter& operator=(const PipeReader&&) = delete;
|
||||
|
||||
void send_message(std::string msg) {
|
||||
send_msg_pipe_writer(&this->pwriter, msg.c_str());
|
||||
pipeio_send_msg(&this->pwriter, msg.c_str());
|
||||
}
|
||||
|
||||
private:
|
||||
@ -60,10 +64,13 @@ private:
|
||||
};
|
||||
|
||||
|
||||
class PipeIO : public PipeReader, public PipeWriter {
|
||||
class PipeIO {
|
||||
public:
|
||||
PipeIO(const char* in_pipe_path, const char* out_pipe_path)
|
||||
:PipeReader(in_pipe_path), PipeWriter(out_pipe_path) {}
|
||||
PipeReader reader;
|
||||
PipeWriter writer;
|
||||
|
||||
PipeIO(const std::string& in_pipe_path, const std::string& out_pipe_path)
|
||||
:reader(in_pipe_path), writer(out_pipe_path) {}
|
||||
|
||||
PipeIO(PipeReader&) = delete;
|
||||
PipeIO(PipeReader&&) = delete;
|
||||
|
@ -1,11 +0,0 @@
|
||||
#include "pipe_io.h"
|
||||
|
||||
|
||||
int main() {
|
||||
pipe_io pipeio = new_pipe_io("in", "out");
|
||||
const char* line;
|
||||
while ((line = read_line_pipe_io(&pipeio))) {
|
||||
send_msg_pipe_io(&pipeio, line);
|
||||
}
|
||||
close_pipe_io(&pipeio);
|
||||
}
|
@ -1,19 +0,0 @@
|
||||
#include <iostream>
|
||||
#include <string>
|
||||
#include <fstream>
|
||||
#include "pipe_io.hpp"
|
||||
|
||||
|
||||
class EchoPipeIO : public PipeIO {
|
||||
using PipeIO::PipeIO;
|
||||
|
||||
virtual void handle_message(std::string msg) override {
|
||||
this->send_message(msg);
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
int main() {
|
||||
EchoPipeIO echo_pipe_io("in", "out");
|
||||
echo_pipe_io.run();
|
||||
}
|
Reference in New Issue
Block a user