begin work on parsing rpc server messages
This commit is contained in:
parent
353d38a5a4
commit
51d7ba7a33
@ -1,4 +1,6 @@
|
|||||||
#include "arch.hpp"
|
#include "arch.hpp"
|
||||||
|
#include "json.hpp"
|
||||||
|
#include "rpc_server.hpp"
|
||||||
#include "vm.hpp"
|
#include "vm.hpp"
|
||||||
#include <format>
|
#include <format>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
@ -88,12 +90,24 @@ auto compile_asm(const std::vector<AsmLine>& lines) -> std::vector<uint32_t>
|
|||||||
return output;
|
return output;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto execute_action(std::unique_ptr<sliger::json::Value> req,
|
||||||
|
std::unique_ptr<slige_rpc::BufferedWriter> writer) -> void
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
using R = Ref;
|
using R = Ref;
|
||||||
using L = Loc;
|
using L = Loc;
|
||||||
using enum sliger::Op;
|
using enum sliger::Op;
|
||||||
|
|
||||||
|
auto rpc = slige_rpc::RpcServer(
|
||||||
|
[&](std::unique_ptr<sliger::json::Value> req,
|
||||||
|
std::unique_ptr<slige_rpc::BufferedWriter> writer) {
|
||||||
|
execute_action(std::move(req), std::move(writer));
|
||||||
|
});
|
||||||
|
rpc.listen();
|
||||||
|
|
||||||
// fn add(a, b) {
|
// fn add(a, b) {
|
||||||
// + a b
|
// + a b
|
||||||
// }
|
// }
|
||||||
|
@ -1,72 +1,7 @@
|
|||||||
#include "rpc_server.hpp"
|
#include "rpc_server.hpp"
|
||||||
#include "json.hpp"
|
|
||||||
#include <memory>
|
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <utility>
|
|
||||||
|
|
||||||
auto create_address(uint16_t port) -> sockaddr_in
|
|
||||||
{
|
|
||||||
return {
|
|
||||||
.sin_family = AF_INET,
|
|
||||||
.sin_port = htons(port),
|
|
||||||
.sin_addr = { .s_addr = inet_addr("127.0.0.1") },
|
|
||||||
.sin_zero = { 0 },
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Functor>
|
|
||||||
auto slige_rpc::RpcServer<Functor>::listen() -> Res<Unit>
|
|
||||||
{
|
|
||||||
int socket_fd = ::socket(AF_INET, SOCK_STREAM, 0);
|
|
||||||
if (socket_fd < 0) {
|
|
||||||
return Err { "could not get socket" };
|
|
||||||
};
|
|
||||||
|
|
||||||
sockaddr_in address = create_address(13370);
|
|
||||||
unsigned int address_size = sizeof(this->address);
|
|
||||||
|
|
||||||
if (::bind(socket_fd, (struct sockaddr*)&address, sizeof(address)) < 0) {
|
|
||||||
return Err { "could not bind" };
|
|
||||||
};
|
|
||||||
|
|
||||||
if (::listen(socket_fd, 0) < 0) {
|
|
||||||
return Err { "could not listen" };
|
|
||||||
}
|
|
||||||
while (true) {
|
|
||||||
int client = ::accept(
|
|
||||||
this->fd, (struct sockaddr*)&this->address, &address_size);
|
|
||||||
if (client < 0) {
|
|
||||||
return Err { "could not accept" };
|
|
||||||
}
|
|
||||||
const size_t buf_len = 1024;
|
|
||||||
int8_t buffer[buf_len] = {};
|
|
||||||
auto bracket_finder = BracketFinder();
|
|
||||||
std::string message = {};
|
|
||||||
while (true) {
|
|
||||||
ssize_t bytes_read = read(client, buffer, buf_len);
|
|
||||||
if (bytes_read < 0) {
|
|
||||||
return Err { "could not read" };
|
|
||||||
} else if (bytes_read == 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
for (size_t i; i < (size_t)bytes_read; ++i) {
|
|
||||||
message += buffer[i];
|
|
||||||
bracket_finder.feed(buffer[i]);
|
|
||||||
if (!bracket_finder.bracket_closed()) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
auto req = sliger::json::parse_json(message);
|
|
||||||
auto writer = std::make_unique<BufferedWriter>(client);
|
|
||||||
this->functor(req, std::move(writer));
|
|
||||||
message.clear();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
std::unreachable();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto slige_rpc::BufferedWriter::write(std::string message) -> Res<Unit>
|
auto slige_rpc::BufferedWriter::write(std::string message) -> Res<Unit>
|
||||||
{
|
{
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "json.hpp"
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include <netdb.h>
|
#include <netdb.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
@ -80,6 +81,18 @@ private:
|
|||||||
int fd;
|
int fd;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
static inline auto create_address(uint16_t port) -> sockaddr_in
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
.sin_family = AF_INET,
|
||||||
|
.sin_port = htons(port),
|
||||||
|
.sin_addr = { .s_addr = inet_addr("127.0.0.1") },
|
||||||
|
.sin_zero = { 0 },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// - load code
|
/// - load code
|
||||||
/// - program input
|
/// - program input
|
||||||
/// - run
|
/// - run
|
||||||
@ -94,10 +107,73 @@ private:
|
|||||||
/// - json string
|
/// - json string
|
||||||
template <typename Functor> class RpcServer {
|
template <typename Functor> class RpcServer {
|
||||||
public:
|
public:
|
||||||
RpcServer(Functor functor)
|
RpcServer(Functor&& functor)
|
||||||
: functor(functor) {};
|
: functor(functor) {};
|
||||||
|
|
||||||
auto listen() -> Res<Unit>;
|
auto listen() -> Res<Unit>
|
||||||
|
{
|
||||||
|
|
||||||
|
int socket_fd = ::socket(AF_INET, SOCK_STREAM, 0);
|
||||||
|
if (socket_fd < 0) {
|
||||||
|
return Err { "could not get socket" };
|
||||||
|
};
|
||||||
|
|
||||||
|
{
|
||||||
|
auto address = create_address(13370);
|
||||||
|
if (::bind(socket_fd, (struct sockaddr*)&address, sizeof(address))
|
||||||
|
< 0) {
|
||||||
|
return Err { "could not bind" };
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (::listen(socket_fd, 0) < 0) {
|
||||||
|
return Err { "could not listen" };
|
||||||
|
}
|
||||||
|
while (true) {
|
||||||
|
|
||||||
|
auto client_address = create_address(13370);
|
||||||
|
socklen_t address_size = sizeof(client_address);
|
||||||
|
int client = ::accept(
|
||||||
|
socket_fd, (struct sockaddr*)&client_address, &address_size);
|
||||||
|
if (client < 0) {
|
||||||
|
return Err { "could not accept" };
|
||||||
|
}
|
||||||
|
const size_t buf_len = 1024;
|
||||||
|
int8_t buffer[buf_len] = {};
|
||||||
|
auto bracket_finder = BracketFinder();
|
||||||
|
std::string message = {};
|
||||||
|
while (true) {
|
||||||
|
ssize_t bytes_read = read(client, buffer, buf_len);
|
||||||
|
if (bytes_read < 0) {
|
||||||
|
return Err { "could not read" };
|
||||||
|
} else if (bytes_read == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
for (size_t i; i < (size_t)bytes_read; ++i) {
|
||||||
|
message += buffer[i];
|
||||||
|
bracket_finder.feed(buffer[i]);
|
||||||
|
if (!bracket_finder.bracket_closed()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
auto req = sliger::json::parse_json(message);
|
||||||
|
if (!req.ok()) {
|
||||||
|
auto err = req.err();
|
||||||
|
auto msg = std::format(
|
||||||
|
"error parsing rpc message: '{}' @ {}:{}\n",
|
||||||
|
err.msg, err.pos.line, err.pos.col);
|
||||||
|
return Err {
|
||||||
|
.msg = msg,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
auto writer = std::make_unique<BufferedWriter>(client);
|
||||||
|
this->functor(std::move(req.val()), std::move(writer));
|
||||||
|
message.clear();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
std::unreachable();
|
||||||
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Functor functor;
|
Functor functor;
|
||||||
|
Loading…
Reference in New Issue
Block a user