mif is a C++11 web-application framework designed for the backend micro-service development. The framework makes use of additional type meta-information.
1.5.1
NOTE: The master branch is actively developed, use latest release for production use.
Lastly tested on gcc 13.2
Linux (Lastly tested on Ubuntu 23.10)
git clone https://github.com/tdv/mif.git
cd mif
./download_third_party.sh
mkdir build
cd build
cmake ..
make
make install
NOTE: In Ubuntu you might have to install some additional packages: liblz4-dev, pkgconf, bison, flex
You can try using CMAKE_INSTALL_PREFIX to select the installation directory
After mif is built, you can build examples
cd mif/examples/{sample_project}
mkdir build
cd build
cmake ..
make
NOTE: To develop your applications, you can use the application template. After downloading the mif project, follow the steps
cd mif/template
mkdir build
cd build
cmake ..
make
After that you can change this template to create your own application. In addition, you can use the examples doned from the template. Examples are fully completed and have a script for build itself. All of them are in the _docs folder. You can find detailed documentation of the examples in the wiki.
Server:
cd mif/exammples/{sample_project}/bin
./{sample_project}_server
Client:
cd mif/exammples/{sample_project}/bin
./{sample_project}_client
Please use --help to get more information about runing an example
Source code
Description
The example demonstrates the work of the simple HTTP echo server.
HTTP-echo-server
// mif
#include <mif/application/http_server.h>
#include <mif/common/log.h>
#include <mif/net/http/constants.h>
class Application
: public mif::Application::HttpServer
{
public:
using HttpServer::HttpServer;
private:
// mif.Application.HttpServer
virtual void Init(mif::Net::Http::ServerHandlers &handlers) override final
{
handlers["/"] = [] (mif::Net::Http::IInputPack const &request,
mif::Net::Http::IOutputPack &response)
{
auto data = request.GetData();
mif_LOG(Info) << "Process request "" << request.GetPath() << request.GetQuery() << ""t Data: "
<< (data.empty() ? std::string{"null"} : std::string{std::begin(data), std::end(data)});
response.SetCode(mif::Net::Http::Code::Ok);
response.SetHeader(mif::Net::Http::Constants::Header::Connection::Value,
mif::Net::Http::Constants::Value::Connection::Close::Value);
response.SetData(std::move(data));
};
}
};
int main(int argc, char const **argv)
{
return mif::Application::Run<Application>(argc, argv);
}
Test
curl -iv -X POST "http://localhost:55555/" -d 'Test data'
Source code
Description
The example demonstrates the work of the HTTP server with dual interface for processing raw HTTP requests and mif RPC by HTTP. More you can see in the chapter "Web service. Additional"
Source code
Description
The "Hello World" example demonstrates a basic client-server application with the interface-based RPC marshaling and TCP communication with using boost.archives for data serialization
Basic steps for building a client-server application with RPC
Common interface
// STD
#include <string>
// mif
#include <mif/service/iservice.h>
namespace Service
{
struct IHelloWorld
: public mif::Service::Inherit<mif::Service::IService>
{
virtual void AddWord(std::string const &word) = 0;
virtual std::string GetText() const = 0;
virtual void Clean() = 0;
};
} // namespace Service
Common interface meta-information
// STD
#include <string>
// mif
#include <mif/remote/ps.h>
// THIS
#include "common/interface/ihello_world.h"
namespace Service
{
namespace Meta
{
using namespace ::Service;
mif_REMOTE_PS_BEGIN(IHelloWorld)
mif_REMOTE_METHOD(AddWord)
mif_REMOTE_METHOD(GetText)
mif_REMOTE_METHOD(Clean)
mif_REMOTE_PS_END()
} // namespace Meta
} // namespace Service
mif_REMOTE_REGISTER_PS(Service::Meta::IHelloWorld)
Server interface implementation
...
// mif
#include <mif/service/creator.h>
// THIS
#include "common/id/service.h"
#include "common/interface/ihello_world.h"
namespace Service
{
namespace Detail
{
namespace
{
class HelloWorld
: public IHelloWorld
{
public:
...
private:
...
// IHelloWorld
virtual void AddWord(std::string const &word) override final
{
...
}
virtual std::string GetText() const override final
{
...
}
virtual void Clean() override final
{
...
}
};
} // namespace
} // namespace Detail
} // namespace Service
mif_SERVICE_CREATOR
(
::Service::Id::HelloWorld,
::Service::Detail::HelloWorld
)
Server application
// mif
#include <mif/application/tcp_service.h>
// COMMON
#include "common/id/service.h"
#include "common/ps/ihello_world.h"
class Application
: public mif::Application::TcpService
{
public:
using TcpService::TcpService;
private:
// mif.Application.TcpService
virtual void Init(mif::Service::FactoryPtr factory) override final
{
factory->AddClass<Service::Id::HelloWorld>();
}
};
int main(int argc, char const **argv)
{
return mif::Application::Run<Application>(argc, argv);
}
Client application
// mif
#include <mif/application/tcp_service_client.h>
#include <mif/common/log.h>
// COMMON
#include "common/ps/ihello_world.h"
class Application
: public mif::Application::TcpServiceClient
{
public:
using TcpServiceClient::TcpServiceClient;
private:
// mif.Application.TcpServiceClient
virtual void Init(mif::Service::IFactoryPtr factory) override final
{
auto service = factory->Create<Service::IHelloWorld>("HelloWorld");
mif_LOG(Info) << "Add words.";
service->AddWord("Hello");
service->AddWord("World");
service->AddWord("!!!");
mif_LOG(Info) << "Result from server: "" << service->GetText() << """;
mif_LOG(Info) << "Clean.";
service->Clean();
mif_LOG(Info) << "Result from server: "" << service->GetText() << """;
}
};
int main(int argc, char const **argv)
{
return mif::Application::Run<Application>(argc, argv);
}
Source code
Description
This example is the same as "Hello World". The difference is in calling remote methods with user-defined data structures as parameters and returning a value. The project structure is the same as in the previous project example, we only add the definition of user-defined data structures and meta-information.
User data structs
// STD
#include <cstdint>
#include <map>
#include <string>
namespace Service
{
namespace Data
{
using ID = std::string;
struct Human
{
std::string name;
std::string lastName;
std::uint32_t age = 0;
};
enum class Position
{
Unknown,
Developer,
Manager
};
struct Employee
: public Human
{
Position position = Position::Unknown;
};
using Employees = std::map<ID, Employee>;
} // namespace Data
} // namespace Service
Meta-information
// mif
#include <mif/reflection/reflect_type.h>
// THIS
#include "common/data/data.h"
namespace Service
{
namespace Data
{
namespace Meta
{
mif_REFLECT_BEGIN(Human)
mif_REFLECT_FIELD(name)
mif_REFLECT_FIELD(lastName)
mif_REFLECT_FIELD(age)
mif_REFLECT_END()
mif_REFLECT_BEGIN(Position)
mif_REFLECT_FIELD(Unknown)
mif_REFLECT_FIELD(Developer)
mif_REFLECT_FIELD(Manager)
mif_REFLECT_END()
mif_REFLECT_BEGIN(Employee, Human)
mif_REFLECT_FIELD(position)
mif_REFLECT_END()
} // namespace Meta
} // namespace Data
} // namespace Service
mif_REGISTER_REFLECTED_TYPE(::Service::Data::Meta::Human)
mif_REGISTER_REFLECTED_TYPE(::Service::Data::Meta::Position)
mif_REGISTER_REFLECTED_TYPE(::Service::Data::Meta::Employee)
Source code
Description
Compared to the previous examples this one adds the inteface inheritance. In the implementation you can query an interface which is out of hierarchy.
Source code
Description
The "Visitor" example demonstrates the mechanism of remote callbacks for interface methods. This can be used as a starting point for publish / subscribe based applications.
Source code
Description
The example demonstrates the Json API (CRUD operations) on HTTP server.
Test
curl -i -X POST "http://localhost:55555/employee/create" -d '{"name":"Ivan", "lastName":"Ivanov","age":33,"email":"[email protected]","position":"Developer","rate":200000.00}'
curl -i "http://localhost:55555/employee/read?id=1"
curl -i "http://localhost:55555/employee/update?id=1" -d '{"name":"Ivan", "lastName":"Ivanov","age":33,"email":"[email protected]","position":"Developer","rate":220000.00}'
curl -i "http://localhost:55555/employee/list?limit=2&offset=0"
curl -i "http://localhost:55555/employee/delete?id=1"
Source code
Description
The example demonstrates communication between two microservices (the example is more powerful version of http crud).
Test
./storage —config=storage.xml
./service --config=service.xml
curl -i -X POST "http://localhost:55555/employee/create" -d '{"name":"Ivan", "lastName":"Ivanov","age":33,"email":"[email protected]","position":"Developer","rate":200000.00}'
curl -i "http://localhost:55555/employee/read?id=1"
curl -i "http://localhost:55555/employee/update?id=1" -d '{"name":"Ivan", "lastName":"Ivanov","age":33,"email":"[email protected]","position":"Developer","rate":220000.00}'
curl -i "http://localhost:55555/employee/list?limit=2&offset=0"
curl -i "http://localhost:55555/employee/delete?id=1"
Source code
Description
The example demonstrates the mechanism of C++ data struct reflection. This can be used as a starting point for building application with serialization, ORM and REST API.