Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/logandk/restful_mapper
ORM for consuming RESTful APIs in C++
https://github.com/logandk/restful_mapper
cpp orm rest restful
Last synced: 3 months ago
JSON representation
ORM for consuming RESTful APIs in C++
- Host: GitHub
- URL: https://github.com/logandk/restful_mapper
- Owner: logandk
- License: other
- Created: 2013-03-30T16:18:14.000Z (almost 12 years ago)
- Default Branch: master
- Last Pushed: 2019-02-21T22:01:58.000Z (almost 6 years ago)
- Last Synced: 2024-05-03T03:16:58.003Z (9 months ago)
- Topics: cpp, orm, rest, restful
- Language: C++
- Homepage: http://logan.dk/restful_mapper
- Size: 546 KB
- Stars: 82
- Watchers: 13
- Forks: 19
- Open Issues: 4
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG
- License: LICENSE
Awesome Lists containing this project
README
# restful_mapper #
ORM for consuming RESTful APIs in C++
[![Build status](https://secure.travis-ci.org/logandk/restful_mapper.png)](http://travis-ci.org/logandk/restful_mapper)
# Introduction #
**restful\_mapper** connects business objects and Representational State
Transfer (REST) web services. It implements object-relational mapping for REST
web services to provide transparent proxying capabilities between a client
(using **restful_mapper**) and a RESTful web service (that follows the
conventions outlined below).The ideas and design philosophy are directly adopted from [Active Resource][3].
However, the API and functionality differ in some areas, where it makes sense.## Design goals ##
* _Simple_: Employ the principle of least surprise -- APIs should be clean and
intuitive
* _Portable_: Support old compilers -- all code is C++03 and C
* _Robust_: Keep the code base small -- rely on proven and stable libraries to
provide core functionality
* _Internationalized_: Handle timezones and character sets automatically## RESTful web service conventions ##
RESTful web services come in many different forms, which are not all suitable
for an ORM-style mapper such as **restful_mapper**. In order to be compatible
with as many web services as possible, **restful_mapper** complies with the
current best-practice for creating RESTful web services with JSON.Specifically, **restful_mapper** is modelled against the [Flask-Restless][4]
library, which provides full-featured RESTful web services that follow best-
practice design methods.The following basic conventions must be followed by the web service:
* Uses standard `DELETE`, `GET`, `POST` and `PUT` requests for CRUD-operations
* Collection of objects should use `objects` as the root JSON key
* Objects in a relationship should be represented as nested JSON structures
* If authentication is used, it must be HTTP basic authenticationFor exact details on expected request and response formats, see [Format of requests and responses][5].
# Building #
**restful\_mapper** is built using [CMake][6].
## Dependencies ##
The following libraries are required to build **restful_mapper**.
* [libcurl][7] - used for comminucating with the web service over HTTP
* [yajl][8] - used to parse and emit JSON
* [libiconv][9] - used to convert between character sets
* [googletest][10] - required for tests onlyInvoking the following command will download and build these libraries.
```shell
make vendor
```On UNIX platforms, [libcurl][7] and [libiconv][9] are typically present on the
system, and will not be built by the make command. If they are not present,
they should be installed using the system package manager.## Library ##
After building the dependencies, invoke the following command to build **restful_mapper**.
```shell
make
```This will install **restful_mapper** as a static library in the `lib` folder.
## Tests ##
The test suite can be built and run using the following command.
```shell
make test
```# Usage #
## API configuration ##
Before making any requests to the web service, it must be configured using the
following methods.The root URL of the web service is specified using the `set_url` method:
```c++
Api::set_url("http://localhost:5000/api");
```If the web service requires authentication, provide the username and password:
```c++
Api::set_username("admin");
Api::set_password("test");
```If you are using a proxy server, it can be specified through the `HTTP_PROXY`
environment variable or using the `set_proxy` method:```c++
Api::set_proxy("http://myproxy");
```## Mapper configuration ##
This example illustrates a complete object mapping:
```c++
using namespace std;
using namespace restful_mapper;class User;
class Alert;class Todo : public Model
{
public:
Primary id;
Field task;
Field priority;
Field time;
Field completed;
Field completed_on;
Foreign user_id;
BelongsTo user;
HasOne alert;virtual void map_set(Mapper &mapper) const
{
mapper.set("id", id);
mapper.set("task", task);
mapper.set("priority", priority);
mapper.set("time", time);
mapper.set("completed", completed);
mapper.set("completed_on", completed_on);
mapper.set("user_id", user_id);
mapper.set("user", user);
mapper.set("alert", alert);
}virtual void map_get(const Mapper &mapper)
{
mapper.get("id", id);
mapper.get("task", task);
mapper.get("priority", priority);
mapper.get("time", time);
mapper.get("completed", completed);
mapper.get("completed_on", completed_on);
mapper.get("user_id", user_id);
mapper.get("user", user);
mapper.get("alert", alert);
}virtual std::string endpoint() const
{
return "/todo";
}virtual const Primary &primary() const
{
return id;
}
};class User: public Model
{
public:
Primary id;
HasMany todos;
...
};
```An API entity is declared by creating a class that inherits from and follows the
interface defined in `restful_mapper::Model`.Each entity can hold a number of fields and relationships:
* `restful_mapper::Primary` -- maps a primary key (integer)
* `restful_mapper::Foreign` -- maps a foreign key (integer)
* `restful_mapper::Field` -- maps a string literal (represented in locale charset)
* `restful_mapper::Field` -- maps an integer value
* `restful_mapper::Field` -- maps a floating point value
* `restful_mapper::Field` -- maps a boolean value
* `restful_mapper::Field` -- maps a datetime value (represented in the local timezone)
* `restful_mapper::HasOne` -- maps a one-to-one or many-to-one relationship on the referenced side
* `restful_mapper::BelongsTo` -- maps a one-to-one or many-to-one relationship on the foreign key side
* `restful_mapper::HasMany` -- maps a one-to-many relationshipThe interface specified the following methods, which must be overriden:
* `virtual void restful_mapper::Model::map_set(Mapper &mapper) const`
Invoked upon saving an object to the web service.
* `virtual void restful_mapper::Model::map_get(const Mapper &mapper)`
Invoked upon requesting an object from the web service.
* `virtual std::string restful_mapper::Model::endpoint() const`
Specifies the API endpoint for this particular model. Relative to the web service
root URL.
* `virtual const Primary &restful_mapper::Model::primary()` const
Specifies the primary key of the model.## Working with objects ##
Using the models defined above, the following operations are made available by
**restful_mapper**.### Requesting data ###
```c++
// Find a single item by id
Todo t = Todo::find(2);// Outputting fields
cout << t.task.get();// Explicit conversions
cout << (string) t.task;// Reload data from server
t.reload();// Get all items in collection
Todo::Collection todos = Todo::find_all();// Find an item in the collection by id
todos.find(4);// Find all items in collection where task is "Do something"
todos.find("task", "Do something");// Find the first item in collection that has been completed
todos.find_first("completed", true);
```### Saving data ###
```c++
// Create a new item
Todo new_todo;
new_todo.task = "Use restful_mapper";
new_todo.save();// Update an existing item
Todo old_todo = Todo::find(2);
old_todo.completed = true;
old_todo.save();// Deleting an item
old_todo.destroy();// Create a clone with no id set (i.e. a new database object)
Todo todo_clone = old_todo.clone();
todo_clone.save();
```### Relationships ###
```c++
// Find an item including related items
User u = User::find(1);// Get a related todo
u.todos[2].task = "Do something else";// Delete last item
u.todos.pop_back();// Add a new related item
Todo new_todo;
new_todo.task = "Use restful_mapper";
u.todos.push_back(new_todo);// Save user including all changes to related todos - will delete one, update one and add one todo
u.save();// Objects in one-to-one and many-to-one relationships are managed pointers
Todo t = Todo::find(2);
cout << t.user->email.get();
```### Querying ###
Supports the query operations [specified][11] by [Flask-Restless][4].
```c++
// Query a single item
Query q;
q("task").like("Do someth%");
q("completed").eq(true);Todo todo = Todo::find(q);
// Query a collection of items
Query q;
q("time").gt("1.45").lt("3.0");
q.order_by_asc(q.field("priority"));Todo::Collection todos = Todo::find_all(q);
```### Exceptions ###
Some API errors are caught using custom exceptions.
* Internal [libcurl][7] errors are represented as a `restful_mapper::ApiError`,
which has adds a `code()` method to `std::runtime_error`.
* `restful_mapper::AuthenticationError` is thrown when authentication fails,
and has the same properties as `restful_mapper::ApiError`.
* `restful_mapper::BadRequestError` is thrown when an error occurs on the API
side, and has the same properties as `restful_mapper::ApiError`.
* `restful_mapper::ValidationError` is thrown when one or more validation
errors are thrown by the API. It has the same properties as
`restful_mapper::ApiError`, but adds an `errors()` method, which returns a
`restful_mapper::ValidationError::FieldMap` map with field names as keys and
error messages as values. This map may also be accessed directly through the
overloaded `operator[]`.# Contributing #
Pull requests on [GitHub][1] are very welcome! Please try to follow these simple rules:
* Please create a topic branch for every separate change you make.
* Make sure your patches are well tested. All tests must pass on [Travis CI][2].
* Update this [`README.md`](http://github.com/logandk/restful_mapper/blob/master/README.md) if applicable.# License #
This code is copyright 2013 Logan Raarup, and is released under the revised BSD License.
For more information, see [`LICENSE`](http://github.com/logandk/restful_mapper/blob/master/LICENSE).
[1]: http://github.com/logandk/restful_mapper
[2]: http://travis-ci.org/logandk/restful_mapper
[3]: https://github.com/rails/activeresource
[4]: https://github.com/jfinkels/flask-restless
[5]: https://flask-restless.readthedocs.org/en/latest/requestformat.html
[6]: http://www.cmake.org
[7]: http://curl.haxx.se/libcurl
[8]: http://lloyd.github.io/yajl
[9]: http://www.gnu.org/software/libiconv
[10]: https://code.google.com/p/googletest
[11]: https://flask-restless.readthedocs.org/en/latest/searchformat.html#queryformat