Compare commits
2 Commits
f5430f3d0f
...
097dab01fd
| Author | SHA1 | Date | |
|---|---|---|---|
| 097dab01fd | |||
| bbafae2588 |
13
include/examples/DisconnectPacket.h
Normal file
13
include/examples/DisconnectPacket.h
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <sp/default/DefaultPacket.h>
|
||||||
|
#include <sp/protocol/Field.h>
|
||||||
|
#include <sp/protocol/MessageBase.h>
|
||||||
|
|
||||||
|
enum DisconnectPacketFields {
|
||||||
|
Reason = 0
|
||||||
|
};
|
||||||
|
|
||||||
|
using DisconnectFields = std::tuple<std::string /*Reason*/>;
|
||||||
|
|
||||||
|
DeclarePacket(Disconnect);
|
||||||
@@ -1,11 +1,15 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
enum PacketId {
|
enum PacketId {
|
||||||
KeepAlive
|
KeepAlive = 0,
|
||||||
|
Disconnect,
|
||||||
};
|
};
|
||||||
|
|
||||||
#include <examples/KeepAlivePacket.h>
|
#include <examples/KeepAlivePacket.h>
|
||||||
|
#include <examples/DisconnectPacket.h>
|
||||||
|
|
||||||
using AllPackets = std::tuple<KeepAlivePacket>;
|
// they must be in the same order !
|
||||||
|
using AllPackets = std::tuple<KeepAlivePacket, DisconnectPacket>;
|
||||||
|
|
||||||
#include <sp/default/DefaultPacketHandler.h>
|
#include <sp/default/DefaultPacketHandler.h>
|
||||||
|
#include <sp/default/DefaultPacketFactory.h>
|
||||||
12
include/sp/common/Templates.h
Normal file
12
include/sp/common/Templates.h
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
|
namespace sp {
|
||||||
|
|
||||||
|
/// @brief Concat multiple tuples in one big tuple
|
||||||
|
/// @tparam ...input_t Multiple std::tuple types to concat
|
||||||
|
template <typename... input_t>
|
||||||
|
using tuple_cat_t = decltype(std::tuple_cat(std::declval<input_t>()...));
|
||||||
|
|
||||||
|
} // namespace sp
|
||||||
7
include/sp/default/DefaultPacketFactory.h
Normal file
7
include/sp/default/DefaultPacketFactory.h
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <sp/protocol/MessageFactory.h>
|
||||||
|
|
||||||
|
namespace sp {
|
||||||
|
using PacketFactory = sp::MessageFactory<sp::PacketMessage, AllPackets>;
|
||||||
|
} // namespace sp
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <sp/common/DataBuffer.h>
|
#include <sp/common/DataBuffer.h>
|
||||||
#include <sp/protocol/Templates.h>
|
#include <sp/common/Templates.h>
|
||||||
|
|
||||||
namespace sp {
|
namespace sp {
|
||||||
|
|
||||||
@@ -63,6 +63,8 @@ class FieldWriter {
|
|||||||
DataBuffer& m_Buffer;
|
DataBuffer& m_Buffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
namespace details {
|
||||||
|
|
||||||
template <typename... TFields>
|
template <typename... TFields>
|
||||||
struct FieldsBuilder {};
|
struct FieldsBuilder {};
|
||||||
|
|
||||||
@@ -76,10 +78,10 @@ struct FieldsBuilder<std::tuple<TFields...>> {
|
|||||||
using Type = typename FieldsBuilder<TFields...>::Type;
|
using Type = typename FieldsBuilder<TFields...>::Type;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
template <typename TField, typename... TFields>
|
template <typename TField, typename... TFields>
|
||||||
struct FieldsBuilder<TField, TFields...> {
|
struct FieldsBuilder<TField, TFields...> {
|
||||||
using Type = sp::tuple_cat_t<std::tuple<Field<TField>>, typename FieldsBuilder<TFields...>::Type>;
|
using Type = sp::tuple_cat_t<std::tuple<Field<TField>>, typename FieldsBuilder<TFields...>::Type>;
|
||||||
};
|
};
|
||||||
|
} // namespace details
|
||||||
|
|
||||||
} // namespace sp
|
} // namespace sp
|
||||||
|
|||||||
@@ -5,9 +5,9 @@
|
|||||||
namespace sp {
|
namespace sp {
|
||||||
|
|
||||||
template <typename... TOptions>
|
template <typename... TOptions>
|
||||||
class Message : public option::MessageInterfaceBuilder<TOptions...>::Type {
|
class Message : public details::MessageInterfaceBuilder<TOptions...>::Type {
|
||||||
public:
|
public:
|
||||||
using ParsedOptions = typename option::MessageInterfaceBuilder<TOptions...>::ParsedOptions;
|
using ParsedOptions = typename details::MessageInterfaceBuilder<TOptions...>::ParsedOptions;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace sp
|
} // namespace sp
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <sp/protocol/Message.h>
|
#include <sp/protocol/Message.h>
|
||||||
#include <sp/protocol/Templates.h>
|
|
||||||
|
|
||||||
namespace sp {
|
namespace sp {
|
||||||
namespace option {
|
namespace option {
|
||||||
@@ -22,6 +21,9 @@ struct FieldsImpl {};
|
|||||||
} // namespace option
|
} // namespace option
|
||||||
|
|
||||||
|
|
||||||
|
namespace details {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -92,7 +94,7 @@ class MessageImplDispatchBase : public TBase {
|
|||||||
template <typename TBase, typename TFields>
|
template <typename TBase, typename TFields>
|
||||||
class MessageImplFieldsBase : public TBase {
|
class MessageImplFieldsBase : public TBase {
|
||||||
public:
|
public:
|
||||||
using AllFields = typename FieldsBuilder<TFields>::Type;
|
using AllFields = typename details::FieldsBuilder<TFields>::Type;
|
||||||
|
|
||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
void Construct(Args... args) {
|
void Construct(Args... args) {
|
||||||
@@ -274,11 +276,10 @@ struct MessageImplBuilder {
|
|||||||
using Type = Base6;
|
using Type = Base6;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
} // namespace details
|
||||||
|
|
||||||
template <typename TBase, typename... TOptions>
|
template <typename TBase, typename... TOptions>
|
||||||
class MessageBase : public MessageImplBuilder<TBase, TOptions...>::Type {};
|
class MessageBase : public details::MessageImplBuilder<TBase, TOptions...>::Type {};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
} // namespace sp
|
} // namespace sp
|
||||||
65
include/sp/protocol/MessageFactory.h
Normal file
65
include/sp/protocol/MessageFactory.h
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <functional>
|
||||||
|
#include <iostream>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
namespace sp {
|
||||||
|
|
||||||
|
namespace details {
|
||||||
|
|
||||||
|
template <typename TBase>
|
||||||
|
using ArrayType = std::vector<std::function<std::unique_ptr<TBase>(void)>>;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
template <typename TBase, typename... TMessages>
|
||||||
|
struct ArrayFiller {};
|
||||||
|
|
||||||
|
template <typename TBase, typename... TMessages>
|
||||||
|
struct ArrayFiller<TBase, std::tuple<TMessages...>> {
|
||||||
|
static void ArrayAppend(details::ArrayType<TBase>& array) {
|
||||||
|
ArrayFiller<TBase, TMessages...>::ArrayAppend(array);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename TBase, typename TMessage, typename... TMessages>
|
||||||
|
struct ArrayFiller<TBase, TMessage, TMessages...> {
|
||||||
|
static void ArrayAppend(details::ArrayType<TBase>& array) {
|
||||||
|
ArrayFiller<TBase, TMessage>::ArrayAppend(array);
|
||||||
|
ArrayFiller<TBase, TMessages...>::ArrayAppend(array);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename TBase, typename TMessage>
|
||||||
|
struct ArrayFiller<TBase, TMessage> {
|
||||||
|
static void ArrayAppend(details::ArrayType<TBase>& array) {
|
||||||
|
array.push_back([]() -> std::unique_ptr<TBase> { return std::make_unique<TMessage>(); });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace details
|
||||||
|
|
||||||
|
template <typename TBase, typename TTMessages>
|
||||||
|
class MessageFactory {
|
||||||
|
public:
|
||||||
|
using IdType = typename TBase::MsgIdType;
|
||||||
|
|
||||||
|
MessageFactory() {
|
||||||
|
details::ArrayFiller<TBase, TTMessages>::ArrayAppend(m_Factory);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<TBase> CreateMessage(IdType id) {
|
||||||
|
if (id >= m_Factory.size())
|
||||||
|
return nullptr;
|
||||||
|
return m_Factory.at(id)();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
details::ArrayType<TBase> m_Factory;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace sp
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
#include <sp/protocol/MessageInterfaces.h>
|
#include <sp/protocol/MessageInterfaces.h>
|
||||||
|
|
||||||
namespace sp {
|
namespace sp {
|
||||||
namespace option {
|
namespace details {
|
||||||
|
|
||||||
class EmptyBase {};
|
class EmptyBase {};
|
||||||
|
|
||||||
@@ -34,5 +34,5 @@ struct MessageInterfaceBuilder {
|
|||||||
using Type = Base6;
|
using Type = Base6;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace option
|
} // namespace details
|
||||||
} // namespace sp
|
} // namespace sp
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
#include <sp/protocol/Options.h>
|
#include <sp/protocol/Options.h>
|
||||||
|
|
||||||
namespace sp {
|
namespace sp {
|
||||||
namespace option {
|
namespace details {
|
||||||
|
|
||||||
template <typename... TOptions>
|
template <typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions {};
|
struct MessageInterfaceParsedOptions {};
|
||||||
@@ -23,34 +23,34 @@ struct MessageInterfaceParsedOptions<> {
|
|||||||
|
|
||||||
|
|
||||||
template <typename T, typename... TOptions>
|
template <typename T, typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions<MsgIdType<T>, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
struct MessageInterfaceParsedOptions<option::MsgIdType<T>, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
||||||
static const bool HasMsgIdType = true;
|
static const bool HasMsgIdType = true;
|
||||||
using MsgIdType = T;
|
using MsgIdType = T;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename... TOptions>
|
template <typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions<LittleEndian, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
struct MessageInterfaceParsedOptions<option::LittleEndian, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
||||||
static const bool HasLittleEndian = true;
|
static const bool HasLittleEndian = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename... TOptions>
|
template <typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions<ReadOperations, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
struct MessageInterfaceParsedOptions<option::ReadOperations, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
||||||
static const bool HasReadOperations = true;
|
static const bool HasReadOperations = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename... TOptions>
|
template <typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions<WriteOperations, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
struct MessageInterfaceParsedOptions<option::WriteOperations, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
||||||
static const bool HasWriteOperations = true;
|
static const bool HasWriteOperations = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename... TOptions>
|
template <typename T, typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions<Handler<T>, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
struct MessageInterfaceParsedOptions<option::Handler<T>, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
||||||
static const bool HasHandler = true;
|
static const bool HasHandler = true;
|
||||||
using HandlerType = Handler<T>;
|
using HandlerType = option::Handler<T>;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename... TOptions>
|
template <typename... TOptions>
|
||||||
struct MessageInterfaceParsedOptions<ValidCheckInterface, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
struct MessageInterfaceParsedOptions<option::ValidCheckInterface, TOptions...> : public MessageInterfaceParsedOptions<TOptions...> {
|
||||||
static const bool HasValid = true;
|
static const bool HasValid = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -239,5 +239,5 @@ template <typename TBase>
|
|||||||
struct MessageInterfaceProcessValid<TBase, false> {
|
struct MessageInterfaceProcessValid<TBase, false> {
|
||||||
using Type = TBase;
|
using Type = TBase;
|
||||||
};
|
};
|
||||||
} // namespace option
|
} // namespace details
|
||||||
} // namespace sp
|
} // namespace sp
|
||||||
|
|||||||
@@ -1,37 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
|
|
||||||
namespace sp {
|
|
||||||
|
|
||||||
/// @brief Default case, see field_index specialization for implementation details
|
|
||||||
template <size_t N, typename T, template <size_t, typename> typename U, typename = void>
|
|
||||||
static constexpr size_t field_index = 0;
|
|
||||||
|
|
||||||
/// @brief A templated size_t that counts the number of existing classes U with a "field_name" member
|
|
||||||
/// @tparam N Current counter for recursion (user should always call it with 0)
|
|
||||||
/// @tparam T Can be any type, must be different for each field we want to be counted later (used because we can't have a empty
|
|
||||||
/// template<> specialization nested in a class)
|
|
||||||
/// @tparam U The templated class that will be searched for match
|
|
||||||
template <size_t N, typename T, template <size_t, typename> typename U>
|
|
||||||
static constexpr size_t field_index<N, T, U, std::void_t<decltype(U<N, T>::field_name)>> = 1 + field_index<N + 1, T, U>;
|
|
||||||
|
|
||||||
/// @brief Concat multiple tuples in one big tuple
|
|
||||||
/// @tparam ...input_t Multiple std::tuple types to concat
|
|
||||||
template <typename... input_t>
|
|
||||||
using tuple_cat_t = decltype(std::tuple_cat(std::declval<input_t>()...));
|
|
||||||
|
|
||||||
template <typename T, typename Tuple>
|
|
||||||
constexpr bool tuple_contains_type = false;
|
|
||||||
template <typename T, typename... Ts>
|
|
||||||
constexpr bool tuple_contains_type<T, std::tuple<Ts...>> = std::disjunction_v<std::is_same<T, Ts>...>;
|
|
||||||
|
|
||||||
template <typename T, typename Tuple>
|
|
||||||
constexpr int get_tuple_index = 0;
|
|
||||||
template <typename T, typename... Rest>
|
|
||||||
constexpr int get_tuple_index<T, std::tuple<T, Rest...>> = 0;
|
|
||||||
template <typename T, typename First, typename... Rest>
|
|
||||||
constexpr int get_tuple_index<T, std::tuple<First, Rest...>> = 1 + get_tuple_index<T, std::tuple<Rest...>>;
|
|
||||||
|
|
||||||
|
|
||||||
} // namespace sp
|
|
||||||
14
src/main.cpp
14
src/main.cpp
@@ -7,6 +7,10 @@ class KeepAliveHandler : public sp::PacketHandler {
|
|||||||
void Handle(KeepAlivePacket& packet) {
|
void Handle(KeepAlivePacket& packet) {
|
||||||
std::cout << "KeepAlive handled !\n";
|
std::cout << "KeepAlive handled !\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Handle(DisconnectPacket& packet) {
|
||||||
|
std::cout << "Disconnect handled !\n";
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
@@ -24,7 +28,15 @@ int main() {
|
|||||||
|
|
||||||
std::cout << "KeepAlive2 : " << keepAlive2->GetField<KeepAliveId>() << "\n";
|
std::cout << "KeepAlive2 : " << keepAlive2->GetField<KeepAliveId>() << "\n";
|
||||||
|
|
||||||
//TODO: write ID and factory
|
//TODO: write ID
|
||||||
|
sp::PacketFactory factory;
|
||||||
|
auto packet = factory.CreateMessage(Disconnect);
|
||||||
|
if (packet == nullptr) {
|
||||||
|
std::cout << "Mauvais ID !\n";
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
std::cout << (unsigned) packet->GetId() << std::endl;
|
||||||
|
packet->Dispatch(handler);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user