DatabaseSerialization

Serializations

When building APIs, you often need to convert models and their related relationships into an object or JSON. Kawkab Kawkab provides convenient methods to perform these conversions, along with control over the properties included in the serialized representation of the models.

Model and Collection Serialization

Converting to Data Object

To convert a model and its loaded relationships into an array, you can use the toData method. This method works recursively, converting all properties and relationships (including nested relationships) into an object:

const user = await User.query().with('roles').first();
 
return user.toData();

You can use the attributesToData method to convert only the model’s attributes into an object without including relationships:

const user = await User.query().first();
 
return user.attributesToData();

You can also convert entire collections of models into a data object using the toData method on the collection instance:

const users = await User.query().all();
 
return users.toData();

Converting to JSON

To convert a model to JSON, you can use the toJson method. Similar to toData, the toJson method works recursively, converting all properties and relationships into JSON. You can also specify supported encoding options in JavaScript:

const user = await User.query().find(1);
 
return user.toJson();
 
return user.toJson(null, 2);

Alternatively, you can convert a model or collection into a string, which will automatically call the toJson method:

const user = await User.query().find(1);
 
return String(user);
 
return JSON.stringify(user);

Since models and collections are converted to JSON when converted to a string, you can return Kawkab Kawkab objects directly from express or Koa applications. Kawkab Kawkab will automatically serialize models and collections into JSON when returned from routes or controllers:

const app = require('express')();
 
app.get('/', async (req, res) => {
  const user = await User.query().find(1);
 
  res.send(user);
});

Relationships

When a Kawkab Kawkab model is converted to JSON, its loaded relationships are automatically included as properties inside the JSON object.

Hiding Properties from JSON

Sometimes, you may want to restrict properties (such as passwords) from being included in the model’s data object or JSON representation. To achieve this, add a hidden property to the model. Properties listed in the hidden array will be excluded from the serialized representation of the model:

import { BaseModel } from "kawkab";
 
class User extends Model {
  hidden = ['password'];
}

Alternatively, you can use the visible property to define a “whitelist” of properties that should be included in the model’s data object and JSON representation. All properties not listed in the visible array will be hidden:

import { BaseModel } from "kawkab";
 
class User extends Model {
  visible = ['first_name', 'last_name'];
}

Temporarily Modify Property Visibility

If you want to make properties that are usually hidden visible for a particular model instance, you can use the makeVisible method. This method returns the model instance:

user.makeVisible('attribute').toData();
 
user.makeVisible(['attribute', 'another_attribute']).toData();

Similarly, if you want to hide properties that are usually visible, you can use the makeHidden method:

user.makeHidden('attribute').toData();
 
user.makeHidden(['attribute', 'another_attribute']).toData();

If you want to override all visible or hidden properties temporarily, you can use the setVisible and setHidden methods respectively:

user.setVisible(['id', 'name']).toData();
 
user.setHidden(['email', 'password', 'remember_token']).toData();

Appending Values to JSON

Sometimes, when converting models to a data object or JSON, you may want to add properties that do not have a corresponding column in the database. To achieve this, first define an appended value:

import { BaseModel, AttributeModel } from "kawkab";
 
class User extends Model {
  attributeIsAdmin() {
    return Attribute.make({
      get: (value, attributes) => (attributes.admin === 'yes')
    });
  }
}

After defining the appended value, add the property name to the model’s appends property. Note that property names are usually used in “snake_case” in the serialized representation, even if the appended value is defined in “camelCase”:

import { BaseModel } from "kawkab";
 
class User extends Model {
  appends = ['is_admin'];
}

Once the property is added to the appends list, it will be included in both the data object and JSON representations. Properties in the appends array will also respect the model’s visible and hidden settings.

Appending Values at Runtime

At runtime, you can tell a model instance to append additional properties using the append method. Or you can use the setAppends method to override the entire array of appended properties for the model instance:

user.append('is_admin').toData();
 
user.setAppends(['is_admin']).toData();