Make sure you read all of the following Q&A before you start integrating your API:
- How can I change the property used for the primary key
- How do I handle a wrapped response object? (json root object)
- How can I extract response metadata from the root object?
- Does restmod supports linked elements side loaded in the response wrapper?
- My resource properties are being automatically renamed, is this normal?
- Are there any other conventions related to naming I should be aware of?
- How can I set a common url prefix for every resource?
- How can I add a trailing slash or extension to the url sent to server?
- How can I add a custom header or parameter to every request?
- I'm getting a 'No API style base was included' warning, what does it mean?
- Can I change the way url names are generated for relations?
The primary key is just a model configuration property and by default is set to 'id'.
To change it for a particular model use the $config
block:
restmod.model('/bikes').mix({
$config: { primaryKey: '_id' }
});
Or for every model, just set a base model definition using rebase
:
module.config(function(restmodProvider) {
restmodProvider.rebase({
$config: {
primaryKey: '_id'
}
});
});
In restmod response wrapping is handled by the model's unpack
and pack
static methods, by default this methods do nothing so records and collections are expected at response root level.
Restmod comes with a default packer mixin that should handle most situations, to enable the default packer just add the DefaultPacker
mixin:
module.config(function(restmodProvider) {
restmodProvider.rebase('DefaultPacker');
});
This will enable json root wrapping and by default it will use the resource's name for single records and the resource's plural name for collections.
So doing:
restmod.model('/api/bikes').$search();
Will expect:
{
"bikes": [{ "id": 1, "brand": "Trek" }]
}
And doing:
restmod.model('/api/bikes').$find(1);
Will expect:
{
"bike": { "id": 1, "brand": "Trek" }
}
The resoure's name is extracted from the resource url, if no url is given (nested resource) or if url does not match the resource's name, then you can change it's name and plural name by setting the name
and plural
configuration variables:
restmod.model().mix({
$config: {
name: 'mouse', // if you only set NAME, then plural is infered from it using the inflector.
plural: 'mice'
}
});
It's also posible to change the root property name without changing the resource name by setting the jsonRootSingle
, jsonRootMany
or the jsonRoot
(both single and many) configuration properties. This should only be used when the properties are fixed or not named after the resource.
Take a look at the previous question first and set up the default packer mixin.
By default, the default packer unpack
method extracts metadata from the meta
root property and stores it in the record's (or collection's) $metadata
property.
So, the following response to a collection's $fetch
:
{
"bikes": [{
"user_id": 1,
"part_ids": [1, 2, 3]
}],
"meta": {
"page": 1
}
}
Will put { page: 1 }
in the collection's $metadata
property.
To change the property from where the packer extracts the metadata set the jsonMeta
configuration property. Set it to '.' to extract metadata from root, so given:
{
"bikes": [{
"user_id": 1,
"part_ids": [1, 2, 3]
}],
"page": 1
}
var Bike = restmod.model('/api/bikes').mix('DefaultPacker',
$config: {
jsonMeta: '.'
}
});
Bike.$search().$then(function(_colection) {
_colection.$metadata; // = { page: 1 } it will ignore response's bikes property.
})
Yes it does, at least if its formated similar to what active_model_serializer generates.
Take a look at the wrapped response question first and set up the default packer.
By default, the default packer will expect inlined resources to come in a property named linked
(jsonapi.org standard).
The following example shows a simple use case scenario:
Given the following api response for the resource /api/bikes/:id
{
"bike": {
"user_id": 1,
"part_ids": [1, 2, 3]
},
"linked": {
"users": [
{ "id": 1, "name": "Joe", "role": "admin" }
],
"parts": [
{ "id": 1, "name": "handlebar" },
{ "id": 2, "name": "lever" },
{ "id": 3, "name": "wheel" }
]
}
}
The following model definition should correctly load every resource:
var Base = restmod.mixin('DefaultPacker'); // remember to enable the default packer.
var User = restmod.model('/api/users').mix(Base);
var Part = restmod.model('/api/parts').mix(Base);
var Bike = restmod.model('/api/bikes').mix(Base, {
user: { belongsTo: User },
parts: { belongsToMany: Part }
})
// Then calling
bike = Bike.$find(1).$then(function() {
bike.user.name; // = Joe
});
To change the property where the inlined resources are located set the jsonLinks
configuration property or set it to '.' to extract links from root.
module.config(function(restmodProvider) {
restmodProvider.rebase(function() {
this.setProperty('jsonLinks', 'links'); // or set jsonLinks: 'links' in a definition object.
});
});
Take a look at the default naming stardards, inlined resources are expected to use the pluralized names for their respective model names. See. By default the name is extracted from the url, you can change a model's name and plural name by setting the name
and plural
configuration variables:
restmod.model().mix({
$config: {
name: 'mouse', // if you only set name, then plural is infered from it.
plural: 'mice'
}
});
That is probably because the api style you selected has overriden the model's property renaming methods encodeName
and decodeName
.
You can disable or modify renaming by overriding the following methods:
module.config(function(restmodProvider) {
restmodProvider.rebase({
$extend: {
Model: {
encodeName: function(_name) { return _newName; }, // or null to disable renaming.
decodeName: function(_name) { return _newName; } // or null to disable renaming.
}
}
});
});
One last one!
By design, restmod will ignore response properties that start with '$' and won't send in requests local properties that start with '$'. This cannot be disabled.
To handle API's that require '$' prefixed properies you have two posibilities:
-
Change the name decoder so it renames properties that start with '$' and gives them another prefix and then change the name encoder to detect the special prefix and change it back to '$':
module.config(function(restmodProvider) { restmodProvider.rebase({ $extend: { Model: { decodeName: function(_name) { // change prefix to '_' return _name.charAt(0) == '$' ? '_' + inflector.camelize(_name.substr(1), true) : inflector.camelize(_name); }, encodeName: function(_name) { // change prefix back to '$' return _name.charAt(0) == '_' ? '$' + inflector.parameterize(_name.substr(1), '_') : inflector.parameterize(_name, '_'); } } } }); });
-
if there are only a couple of properties you need to let through, assign a special mapping to those properties:
restmod.model().mix({ myType: { map: '$type' }, myName: { map: '$name' } })
You can user the urlPrefix
configuration property, like this:
module.config(function(restmodProvider) {
restmodProvider.rebase({
$config: {
urlPrefix: '/api/v1' // or use setProperty('urlPrefix', '/api/v1') in a definition function
}
});
});
You can hook to the before-request
event and modify the url before is sent, restmod will remove trailing slashes from generated urls before this hook is called, so slashes appended here will be sent.
module.config(function(restmodProvider) {
restmodProvider.rebase({
$hooks: {
'before-request': function(_req) {
// _req is just a $http configuration object.
_req.url += '.json';
}
}
});
});
You can read more about hooks and object lifecycle in the Hooks Guide.
Same as before, you can hook to the before-request
event:
module.config(function(restmodProvider) {
restmodProvider.rebase({
$hooks: {
'before-request': function(_req) {
_req.headers = angular.extend(_req.headers, { 'X-My-Header': 'imateapot!' });
}
}
});
});
Restmod expect you to use a base API style for your models, you should include one of the included styles or create a new one:
For example to use the active_model_serializer style you should include the /dist/styles/ams.min.js
file in your project and then:
module.config(function(restmodProvider) {
restmodProvider.rebase('AMSApi');
});
or to define your own style do:
module.config(function(restmodProvider) {
restmodProvider.rebase({
$config: {
style: 'MyDjangoStyle', // By setting the style variable the warning is disabled.
primaryKey: '_id'
/* other style related configuration */
}
});
});
To see available styles or colaborate with a new one, take a look at the Style listing.
Yes you can, just override the Model.encodeUrlName
method to provide custom name transformation, like this:
module.config(function(restmodProvider) {
restmodProvider.rebase({
$extend: {
Model: {
encodeUrlName: function(_name) {
return _name.toLowerCase();
}
}
}
});
});
Transformations will be applied to relations where the extended model is referenced.