move marshaller to IPC component
This commit is contained in:
@@ -27,9 +27,8 @@ project/
|
||||
│ │ ├── IpcConfig.h # type alias: using IpcMessage = BaseIpcMessage<TextIpcSerializer>
|
||||
│ │ ├── IpcDispatcher.h
|
||||
│ │ ├── IpcPipeChannel.h
|
||||
│ │ ├── IpcMarshaller.h
|
||||
│ │ └── IpcSerializer.h # сериализаторы (TextIpcSerializer)
|
||||
│ ├── proxy/
|
||||
│ │ └── ProxyMarshaller.h
|
||||
│ └── rpc/
|
||||
│ ├── rpc_export.h
|
||||
│ ├── RpcInvoker.h
|
||||
@@ -59,15 +58,15 @@ project/
|
||||
* Сериализация вынесена в отдельные сериализаторы (`TextIpcSerializer` и т.д.)
|
||||
* Тип сырых данных параметризован через сериализатор (по умолчанию `std::string`, можно использовать `std::vector<std::byte>` для бинарных форматов)
|
||||
* Выбор сериализатора делается один раз в `IpcConfig.h` через type alias
|
||||
* **Уровень канала**: `RpcChannel` + `IpcPipeChannel`
|
||||
* **Уровень канала**: `IpcChannel` + `IpcPipeChannel`
|
||||
* `IpcChannel` — абстракция транспорта: `send(const IpcMessage&)`, `receive() -> IpcMessage`.
|
||||
* `IpcPipeChannel` — реализация поверх двух FIFO (`/tmp/fifo_to_server`, `/tmp/fifo_to_client`), которая внутри работает со строками, но наружу — только с `IpcMessage`.
|
||||
* **Уровень RPC-ядра**:
|
||||
* `ProxyMarshaller` — собирает `IpcMessage` из имени метода и аргументов, отправляет через `RpcChannel` и читает ответ.
|
||||
* `IpcMarshaller` — собирает `IpcMessage` из имени метода и аргументов, отправляет через `IpcChannel` и читает ответ.
|
||||
* `RpcInvoker` — по имени метода (первое поле сообщения) находит зарегистрированную функцию-член и вызывает её, читая аргументы через `get<T>()`.
|
||||
* **Сгенерированные обёртки**:
|
||||
* `*.proxy.*` — используют `ProxyMarshaller` и `RpcChannel`, не зависят от конкретного транспорта.
|
||||
* `*.skeleton.*` — используют `RpcInvoker` и принимают/возвращают `IpcMessage` для диспетчеризации вызовов.
|
||||
* `*.proxy.*` — шаблонные классы, зависящие только от абстрактного `impl` с методом `impl.callTyped<Ret>(method, args...)` и не знающие про конкретный транспорт.
|
||||
* `*.skeleton.*` — используют `RpcInvoker` и принимают/возвращают `RpcValue` для диспетчеризации вызовов.
|
||||
|
||||
---
|
||||
|
||||
|
||||
@@ -4,9 +4,13 @@
|
||||
#include <ipc/IpcCodec.h>
|
||||
#include <rpc/RpcValue.h>
|
||||
|
||||
class ProxyMarshaller {
|
||||
// Маршаллер, который знает, как превратить типизированный RPC-вызов
|
||||
// в IpcMessage и обратно. Живёт в IPC-слое и опирается на IpcChannel
|
||||
// и IpcCodec, но снаружи предъявляет только callTyped<T>(...).
|
||||
class IpcMarshaller {
|
||||
public:
|
||||
explicit ProxyMarshaller(IpcChannel& ch) : channel(ch) {}
|
||||
explicit IpcMarshaller(IpcChannel& ch)
|
||||
: channel(ch) {}
|
||||
|
||||
// Базовый type-erased вызов: принимает вектор RpcValue и возвращает RpcValue.
|
||||
RpcValue call(const std::string& method, const RpcArgs& args) {
|
||||
@@ -21,7 +25,7 @@ public:
|
||||
return IpcCodec::decodeResponse(resp);
|
||||
}
|
||||
|
||||
// Удобный шаблонный хелпер для сгенерированных прокси.
|
||||
// Типизированный хелпер: контрактом является только наличие этого метода.
|
||||
template<typename Ret, typename... Args>
|
||||
Ret callTyped(const std::string& method, const Args&... args) {
|
||||
RpcArgs packed;
|
||||
@@ -38,3 +42,4 @@ private:
|
||||
};
|
||||
|
||||
|
||||
|
||||
+4
-4
@@ -1,6 +1,6 @@
|
||||
#include "MyService.proxy.h"
|
||||
#include "ipc/IpcPipeChannel.h"
|
||||
#include "proxy/ProxyMarshaller.h"
|
||||
#include "ipc/IpcMarshaller.h"
|
||||
|
||||
#include <sys/stat.h>
|
||||
|
||||
@@ -16,9 +16,9 @@ int main() {
|
||||
// и читает из fifo, в который пишет сервер (fifo_to_client).
|
||||
IpcPipeChannel ch("/tmp/fifo_to_client", "/tmp/fifo_to_server");
|
||||
|
||||
// RPC‑уровень: создаём marshaller поверх канала и передаём его в прокси.
|
||||
ProxyMarshaller marshaller(ch);
|
||||
MyServiceProxy proxy(marshaller);
|
||||
// RPC‑уровень: создаём IpcMarshaller поверх канала и передаём его в прокси.
|
||||
IpcMarshaller marshaller(ch);
|
||||
MyServiceProxy<IpcMarshaller> proxy(marshaller);
|
||||
|
||||
proxy.add(7, 9);
|
||||
int counter = proxy.get();
|
||||
|
||||
@@ -1,11 +1,5 @@
|
||||
#include "{{ cls.name }}.proxy.h"
|
||||
|
||||
{{ cls.name }}Proxy::{{ cls.name }}Proxy(ProxyMarshaller& marshaller)
|
||||
: impl(marshaller) {}
|
||||
|
||||
{% for m in cls.methods %}
|
||||
{{ m.return_type }} {{ cls.name }}Proxy::{{ m.name }}({% for a in m.args %}{{ a.type }} {{ a.name }}{% if not loop.last %}, {% endif %}{% endfor %}) {
|
||||
return impl.callTyped<{{ m.return_type }}>("{{ cls.name }}.{{ m.name }}"{% for a in m.args %}, {{ a.name }}{% endfor %});
|
||||
}
|
||||
{% endfor %}
|
||||
|
||||
// Реализация шаблонного прокси целиком находится в заголовочном файле.
|
||||
// Этот cpp остаётся пустым, чтобы сгенерированные файлы по‑прежнему
|
||||
// могли участвовать в сборке как отдельная единица трансляции.
|
||||
|
||||
@@ -1,14 +1,24 @@
|
||||
#pragma once
|
||||
|
||||
#include "proxy/ProxyMarshaller.h"
|
||||
#include "{{ cls.name }}.h"
|
||||
|
||||
// Шаблонный прокси, который зависит только от контракта Impl:
|
||||
// Impl должен предоставлять метод
|
||||
// template<typename Ret, typename... Args>
|
||||
// Ret callTyped(const std::string& method, const Args&... args);
|
||||
template<typename Impl>
|
||||
class {{ cls.name }}Proxy {
|
||||
public:
|
||||
explicit {{ cls.name }}Proxy(ProxyMarshaller& marshaller);
|
||||
explicit {{ cls.name }}Proxy(Impl& impl)
|
||||
: impl(impl) {}
|
||||
|
||||
{% for m in cls.methods %}
|
||||
{{ m.return_type }} {{ m.name }}({% for a in m.args %}{{ a.type }} {{ a.name }}{% if not loop.last %}, {% endif %}{% endfor %});
|
||||
{{ m.return_type }} {{ m.name }}({% for a in m.args %}{{ a.type }} {{ a.name }}{% if not loop.last %}, {% endif %}{% endfor %}) {
|
||||
return impl.template callTyped<{{ m.return_type }}>("{{ cls.name }}.{{ m.name }}"{% for a in m.args %}, {{ a.name }}{% endfor %});
|
||||
}
|
||||
{% endfor %}
|
||||
|
||||
private:
|
||||
ProxyMarshaller& impl;
|
||||
Impl& impl;
|
||||
};
|
||||
|
||||
|
||||
Reference in New Issue
Block a user