Refactor client codes to use similar, standardized API

This commit is contained in:
Kristóf Tóth 2019-04-11 18:30:25 +02:00
parent 74e08fd052
commit d28e90862d
16 changed files with 226 additions and 161 deletions

29
clients/c_cpp/example.c Normal file
View 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
View 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();
}

View File

@ -9,25 +9,30 @@ typedef struct pipe_reader {
FILE* stream; FILE* stream;
char* line_buf; char* line_buf;
size_t buf_size; size_t buf_size;
void (*msg_handler)(const char*, void*);
void* handler_ctx;
} pipe_reader; } 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; pipe_reader preader;
preader.stream = fopen(pipe_path, "r"); preader.stream = fopen(pipe_path, "r");
preader.line_buf = NULL; preader.line_buf = NULL;
preader.buf_size = 0; preader.buf_size = 0;
preader.msg_handler = &null_handler;
return preader; return preader;
} }
bool check_pipe_reader(const pipe_reader* preader) { bool pipeio_check_reader(const pipe_reader* preader) {
if (!preader || preader->stream == NULL) { if (!preader || preader->stream == NULL) {
return false; return false;
} }
return true; return true;
} }
const char* recv_msg_pipe_reader(pipe_reader* preader) { const char* pipeio_recv_msg(pipe_reader* preader) {
if (!check_pipe_reader(preader)) { if (!pipeio_check_reader(preader)) {
return NULL; return NULL;
} }
ssize_t read = getline(&preader->line_buf, &preader->buf_size, preader->stream); 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; return NULL;
} }
void close_pipe_reader(pipe_reader* preader) { void pipeio_set_msg_handler(pipe_reader* preader, void (*msg_handler)(const char*, void*), void* handler_ctx) {
if (!check_pipe_reader(preader)) { 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; return;
} }
fclose(preader->stream); fclose(preader->stream);
@ -55,21 +74,21 @@ typedef struct pipe_writer {
FILE* stream; FILE* stream;
} pipe_writer; } pipe_writer;
pipe_writer new_pipe_writer(const char* pipe_path) { pipe_writer pipeio_new_writer(const char* pipe_path) {
pipe_writer pwriter; pipe_writer pwriter;
pwriter.stream = fopen(pipe_path, "w"); pwriter.stream = fopen(pipe_path, "w");
return pwriter; return pwriter;
} }
bool check_pipe_writer(const pipe_writer* pwriter) { bool pipeio_check_writer(const pipe_writer* pwriter) {
if (!pwriter || pwriter->stream == NULL) { if (!pwriter || pwriter->stream == NULL) {
return false; return false;
} }
return true; return true;
} }
bool send_msg_pipe_writer(pipe_writer* pwriter, const char* msg) { bool pipeio_send_msg(pipe_writer* pwriter, const char* msg) {
if (!check_pipe_writer(pwriter)) { if (!pipeio_check_writer(pwriter)) {
return false; return false;
} }
fprintf(pwriter->stream, "%s\n", msg); fprintf(pwriter->stream, "%s\n", msg);
@ -77,8 +96,8 @@ bool send_msg_pipe_writer(pipe_writer* pwriter, const char* msg) {
return true; return true;
} }
void close_pipe_writer(pipe_writer* pwriter) { void pipeio_close_writer(pipe_writer* pwriter) {
if (!check_pipe_writer(pwriter)) { if (!pipeio_check_writer(pwriter)) {
return; return;
} }
fclose(pwriter->stream); fclose(pwriter->stream);
@ -87,42 +106,42 @@ void close_pipe_writer(pipe_writer* pwriter) {
typedef struct pipe_io { typedef struct pipe_io {
pipe_reader preader; pipe_reader reader;
pipe_writer pwriter; pipe_writer writer;
} pipe_io; } 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; pipe_io pipeio;
pipeio.preader = new_pipe_reader(in_pipe_path); pipeio.reader = pipeio_new_reader(in_pipe_path);
pipeio.pwriter = new_pipe_writer(out_pipe_path); pipeio.writer = pipeio_new_writer(out_pipe_path);
return pipeio; return pipeio;
} }
bool check_pipe_io(const pipe_io* pipeio) { bool pipeio_check_io(const pipe_io* pipeio) {
if (!pipeio) { if (!pipeio) {
return false; 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) { void pipeio_close_io(pipe_io* pipeio) {
if (!check_pipe_io(pipeio)) { if (!pipeio_check_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)) {
return; return;
} }
close_pipe_reader(&pipeio->preader); pipeio_close_reader(&pipeio->reader);
close_pipe_writer(&pipeio->pwriter); 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);
// }

View File

@ -3,15 +3,16 @@ extern "C" {
} }
#include <string> #include <string>
#include <iostream> #include <iostream>
#include <functional>
class PipeReader { class PipeReader {
public: public:
PipeReader(const char* pipe_path) PipeReader(const std::string& pipe_path)
:preader(new_pipe_reader(pipe_path)) {} :preader(pipeio_new_reader(pipe_path.c_str())) {}
virtual ~PipeReader() { virtual ~PipeReader() {
close_pipe_reader(&this->preader); pipeio_close_reader(&this->preader);
} }
PipeReader(PipeReader&) = delete; PipeReader(PipeReader&) = delete;
@ -19,31 +20,34 @@ public:
PipeReader& operator=(const PipeReader&) = delete; PipeReader& operator=(const PipeReader&) = delete;
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() { void run() {
while (const char* linebuf = this->recv_message()) { while (const char* linebuf = this->recv_message()) {
std::string msg(linebuf); std::string msg(linebuf);
this->handle_message(msg); this->message_handler(msg);
} }
} }
const char* recv_message() { 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: private:
pipe_reader preader; pipe_reader preader;
std::function<void(std::string)> message_handler = [](std::string){};
}; };
class PipeWriter { class PipeWriter {
public: public:
PipeWriter(const char* pipe_path) PipeWriter(const std::string& pipe_path)
:pwriter(new_pipe_writer(pipe_path)) {} :pwriter(pipeio_new_writer(pipe_path.c_str())) {}
virtual ~PipeWriter() { virtual ~PipeWriter() {
close_pipe_writer(&this->pwriter); pipeio_close_writer(&this->pwriter);
} }
PipeWriter(PipeReader&) = delete; PipeWriter(PipeReader&) = delete;
@ -52,7 +56,7 @@ public:
PipeWriter& operator=(const PipeReader&&) = delete; PipeWriter& operator=(const PipeReader&&) = delete;
void send_message(std::string msg) { void send_message(std::string msg) {
send_msg_pipe_writer(&this->pwriter, msg.c_str()); pipeio_send_msg(&this->pwriter, msg.c_str());
} }
private: private:
@ -60,10 +64,13 @@ private:
}; };
class PipeIO : public PipeReader, public PipeWriter { class PipeIO {
public: public:
PipeIO(const char* in_pipe_path, const char* out_pipe_path) PipeReader reader;
:PipeReader(in_pipe_path), PipeWriter(out_pipe_path) {} 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;
PipeIO(PipeReader&&) = delete; PipeIO(PipeReader&&) = delete;

View File

@ -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);
}

View File

@ -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();
}

View File

@ -7,8 +7,8 @@ class Program
{ {
using (var t = new PipeIO("in", "out")) using (var t = new PipeIO("in", "out"))
{ {
t.reader.OnMessage += (String msg) => t.writer.SendMessage(msg); t.Reader.OnMessage += (String msg) => t.Writer.SendMessage(msg);
t.reader.Run(); t.Reader.Run();
} }
} }
} }

View File

@ -65,19 +65,19 @@ namespace Pipe.IO
public class PipeIO : IDisposable public class PipeIO : IDisposable
{ {
public PipeReader reader; public PipeReader Reader;
public PipeWriter writer; public PipeWriter Writer;
public PipeIO(String in_pipe_path, String out_pipe_path) public PipeIO(String in_pipe_path, String out_pipe_path)
{ {
this.reader = new PipeReader(in_pipe_path); this.Reader = new PipeReader(in_pipe_path);
this.writer = new PipeWriter(out_pipe_path); this.Writer = new PipeWriter(out_pipe_path);
} }
public void Dispose() public void Dispose()
{ {
this.reader.Dispose(); this.Reader.Dispose();
this.writer.Dispose(); this.Writer.Dispose();
} }
} }
} }

View File

@ -13,8 +13,8 @@ func main() {
} }
defer pipeio.Close() defer pipeio.Close()
pipeio.HandleMessage(func(msg []byte) { pipeio.Reader.SetMessageHandler(func(msg []byte) {
pipeio.Writer.SendMessage(msg) pipeio.Writer.SendMessage(msg)
}) })
pipeio.Run() pipeio.Reader.Run()
} }

View File

@ -9,8 +9,11 @@ import (
type pipeReader struct { type pipeReader struct {
pipe *os.File pipe *os.File
scanner *bufio.Scanner scanner *bufio.Scanner
messageHandler messageHandlerFunc
} }
type messageHandlerFunc func([]byte)
func NewPipeReader(pipePath string) (*pipeReader, error) { func NewPipeReader(pipePath string) (*pipeReader, error) {
pipe, err := os.Open(pipePath) pipe, err := os.Open(pipePath)
scanner := bufio.NewScanner(pipe) scanner := bufio.NewScanner(pipe)
@ -21,9 +24,28 @@ func NewPipeReader(pipePath string) (*pipeReader, error) {
return &pipeReader{ return &pipeReader{
pipe: pipe, pipe: pipe,
scanner: scanner, scanner: scanner,
messageHandler: func([]byte){},
}, nil }, nil
} }
func (preader *pipeReader) Close() {
preader.pipe.Close()
}
func (preader *pipeReader) SetMessageHandler(fun messageHandlerFunc) {
preader.messageHandler = fun
}
func (preader *pipeReader) Run() {
for {
msg := preader.RecvMessage()
if msg == nil {
break
}
preader.messageHandler(msg)
}
}
func (preader *pipeReader) RecvMessage() []byte { func (preader *pipeReader) RecvMessage() []byte {
if preader.scanner.Scan() { if preader.scanner.Scan() {
return preader.scanner.Bytes() return preader.scanner.Bytes()
@ -31,10 +53,6 @@ func (preader *pipeReader) RecvMessage() []byte {
return nil return nil
} }
func (preader *pipeReader) Close() {
preader.pipe.Close()
}
type pipeWriter struct { type pipeWriter struct {
pipe *os.File pipe *os.File
@ -60,11 +78,8 @@ func (pwriter *pipeWriter) SendMessage(msg []byte) {
type pipeIO struct { type pipeIO struct {
Reader pipeReader Reader pipeReader
Writer pipeWriter Writer pipeWriter
handleMessage handleMessageFunc
} }
type handleMessageFunc func([]byte)
func NewPipeIO(inPipePath, outPipePath string) (*pipeIO, error) { func NewPipeIO(inPipePath, outPipePath string) (*pipeIO, error) {
reader, err := NewPipeReader(inPipePath) reader, err := NewPipeReader(inPipePath)
if err != nil { if err != nil {
@ -77,7 +92,6 @@ func NewPipeIO(inPipePath, outPipePath string) (*pipeIO, error) {
return &pipeIO{ return &pipeIO{
Reader: *reader, Reader: *reader,
Writer: *writer, Writer: *writer,
handleMessage: func([]byte){},
}, nil }, nil
} }
@ -85,17 +99,3 @@ func (pipeio *pipeIO) Close() {
pipeio.Reader.Close() pipeio.Reader.Close()
pipeio.Writer.Close() pipeio.Writer.Close()
} }
func (pipeio *pipeIO) Run() {
for {
msg := pipeio.Reader.RecvMessage()
if msg == nil {
break
}
pipeio.handleMessage(msg)
}
}
func (pipeio *pipeIO) HandleMessage(fun handleMessageFunc) {
pipeio.handleMessage = fun
}

13
clients/java/Example.java Normal file
View File

@ -0,0 +1,13 @@
public class Example {
public static void main(String[] args) {
try (PipeIO pipeio = new PipeIO("in", "out")) {
pipeio.reader.setMessageHandler(new MessageHandler(){
@Override
public void call(String msg) {
pipeio.writer.sendMessage(msg);
}
});
pipeio.reader.run();
}
}
}

View File

@ -0,0 +1,3 @@
public interface MessageHandler {
public void call(String msg);
}

View File

@ -2,25 +2,16 @@ import java.io.IOException;
public class PipeIO implements AutoCloseable { public class PipeIO implements AutoCloseable {
protected PipeReader reader; public PipeReader reader;
protected PipeWriter writer; public PipeWriter writer;
public PipeIO(String in_pipe_path, String out_pipe_path) throws IOException { public PipeIO(String in_pipe_path, String out_pipe_path) {
this.reader = new PipeReader(in_pipe_path); this.reader = new PipeReader(in_pipe_path);
this.writer = new PipeWriter(out_pipe_path); this.writer = new PipeWriter(out_pipe_path);
} }
public void close() throws IOException { public void close() {
this.reader.close(); this.reader.close();
this.writer.close(); this.writer.close();
} }
public void run() throws IOException {
String msg;
while ((msg = this.reader.recvMessage()) != null) {
this.handleMessage(msg);
}
}
public void handleMessage(String msg) throws IOException {}
} }

View File

@ -1,21 +1,50 @@
import java.io.BufferedReader; import java.io.BufferedReader;
import java.io.FileReader; import java.io.FileReader;
import java.io.IOException; import java.io.IOException;
import java.io.UncheckedIOException;
import java.lang.AutoCloseable; import java.lang.AutoCloseable;
public class PipeReader implements AutoCloseable { public class PipeReader implements AutoCloseable {
private BufferedReader pipe; private BufferedReader pipe;
private MessageHandler messageHandler;
public PipeReader(String pipe_path) throws IOException { public PipeReader(String pipe_path) {
this.pipe = new BufferedReader(new FileReader(pipe_path)); try {
this.pipe = new BufferedReader(new FileReader(pipe_path));
} catch (IOException e) {
throw new UncheckedIOException(e);
}
this.messageHandler = new MessageHandler(){
@Override
public void call(String msg) {}
};
} }
public void close() throws IOException { public void close() {
this.pipe.close(); try {
this.pipe.close();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
} }
public String recvMessage() throws IOException { public void setMessageHandler(MessageHandler handler) {
return this.pipe.readLine(); this.messageHandler = handler;
}
public void run() {
String msg;
while ((msg = this.recvMessage()) != null) {
this.messageHandler.call(msg);
}
}
public String recvMessage() {
try {
return this.pipe.readLine();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
} }
} }

View File

@ -1,22 +1,35 @@
import java.io.BufferedWriter; import java.io.BufferedWriter;
import java.io.FileWriter; import java.io.FileWriter;
import java.io.IOException; import java.io.IOException;
import java.io.UncheckedIOException;
import java.lang.AutoCloseable; import java.lang.AutoCloseable;
public class PipeWriter implements AutoCloseable { public class PipeWriter implements AutoCloseable {
private BufferedWriter pipe; private BufferedWriter pipe;
public PipeWriter(String pipe_path) throws IOException { public PipeWriter(String pipe_path) {
this.pipe = new BufferedWriter(new FileWriter(pipe_path)); try {
this.pipe = new BufferedWriter(new FileWriter(pipe_path));
} catch(IOException e) {
throw new UncheckedIOException(e);
}
} }
public void close() throws IOException { public void close() {
this.pipe.close(); try {
this.pipe.close();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
} }
public void sendMessage(String msg) throws IOException { public void sendMessage(String msg) {
this.pipe.write(msg + "\n"); try {
this.pipe.flush(); this.pipe.write(msg + "\n");
this.pipe.flush();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
} }
} }

View File

@ -1,22 +0,0 @@
import java.io.IOException;
public class Test extends PipeIO {
public Test(String in_pipe_path, String out_pipe_path) throws IOException {
super(in_pipe_path, out_pipe_path);
}
@Override
public void handleMessage(String msg) throws IOException {
this.writer.sendMessage(msg);
}
public static void main(String[] args) {
try(Test test = new Test("in", "out")) {
test.run();
}
catch (IOException e) {
e.printStackTrace();
}
}
}