Dynamoose is a modeling tool for Amazon's DynamoDB (inspired by Mongoose)
In switching from MongoDB/Mongoose to DynamoDB, we missed the modeling provided by Mongoose. There are several great modules out there, but they didn't match our needs. We created Dynamoose based on our usage.
Dynamoose uses the official AWS SDK.
$ npm install dynamoose
Set AWS configurations in enviroment varable:
export AWS_ACCESS_KEY_ID="Your AWS Access Key ID"
export AWS_SECRET_ACCESS_KEY="Your AWS Secret Access Key"
export AWS_REGION="us-east-1"
Here's a simple example:
var dynamoose = require('dynamoose');
// Create cat model with default options
var Cat = dynamoose.model('Cat', { id: Number, name: String });
// Create a new cat object
var garfield = new Cat({id: 666, name: 'Garfield'});
// Save to DynamoDB
garfield.save();
// Lookup in DynamoDB
Cat.get(666)
.then(function (badCat) {
console.log('Never trust a smiling cat. - ' + badCat.name);
});
var dynamoose = require('dynamoose');
Compiles a new model or looks up an existing model. options
is optional.
Default options
:
{
create: true, // Create table in DB, if it does not exist,
update: false, // Update remote indexes if they do not match local index structure
waitForActive: true, // Wait for table to be created before trying to us it
waitForActiveTimeout: 180000 // wait 3 minutes for table to activate
}
Basic example:
var Cat = dynamoose.model('Cat', { id: Number, name: String });
Configure dynamoose to use a DynamoDB local for testing.
url
defaults to 'http://localhost:8000'
dynamoose.local();
Configures and returns the AWS.DynamoDB object
AWS object for dynamoose. Used to configure AWS for dynamoose.
dynamoose.AWS.config.update({
accessKeyId: 'AKID',
secretAccessKey: 'SECRET',
region: 'us-east-1'
});
Sets the default to be used when creating a model. Can be modified on a per model by passing options to .model()
.
Default options
:
{
create: true // Create table in DB if it does not exist
}
It is recommended that create
be disabled for production environments.
dynamoose.setDefaults( { create: false });
Schema class
Table class
var Schema = dynamoose.Schema;
Create a new Schema objects. The schema maps back to a DynamoDB table.
var dogSchema = new Schema({
ownerId: {
type: Number,
validate: function(v) { return v > 0; },
hashKey: true
},
name: {
type: String,
rangeKey: true,
index: true // name: nameLocalIndex, ProjectionType: ALL
},
breed: {
type: String,
trim: true,
required: true,
index: {
global: true,
rangeKey: 'ownerId',
name: 'BreedIndex',
project: true, // ProjectionType: ALL
throughput: 5 // read and write are both 5
}
},
color: {
lowercase: true,
type: [String],
default: ['Brown']
},
age: Number
},
{
throughput: {read: 15, write: 5}
});
Each key represents an attribute in the DynamoDB table. The value of the key can either be the attribute type or an Object describing the attribute.
Valid attribute types are:
- String
- Number
- Boolean
- Date
- Object
- Array
- Buffer
String, Boolean, Object, and Array all map to the DynamoDB type of 'S'. Number and Date map to 'N'. Buffer maps to 'B'.
Types can also be sets of the above types. This is done by making the type an array. For example, [String] would be a String Set (DynamoDB 'SS') and [Number] would be a Number Set (DynamoDB 'NS').
type: AttributeType required
Required for all attribute definitions. Defines the attribute type. See Attribute Types.
hashKey: boolean
Defines the hash key attribute for the table. If no attribute is defined as the hash key, the first attribute is assumed to be the hash key.
rangeKey: boolean
Defines the range key attribute.
required: boolean
Defines if the attribute is required. Prior to saving an entry, if the attribute is undefined or null, an error will be thrown.
index: boolean | {...}
Defines the attribute as a local or global secondary index. Index can either be true or an index definition object. The index definition object can contain the following keys:
- name: 'string' - Name of index (Default is `attribute.name + (global ? 'GlobalIndex' : 'LocalIndex')``).
- global: boolean - Set the index to be a global secondary index. Attribute will be the hash key for the Index.
- rangeKey: 'string' - The range key for a global secondary index.
- project: boolean | ['string', ...] - Sets the attributes to be projected for the index.
true
projects all attributes,false
projects only the key attributes, and ['string', ...] projects the attributes listed. Default istrue
. - throughput: number | {read: number, write: number} - Sets the throughput for the global secondary index.
default: function | value
Applies a default to the attribute's value when saving, if the values is null or undefined.
If default is a function, the function is called, and the response is assigned to the attribute's value.
If it is a value, the value is simply assigned.
validate: function | RegExp | value
Validation required before for saving.
If validate is a function, the function is used to validate the attribute's value. The function must have the signature:
function(value) {
if(valid)
return true;
else
return false;
}
If it is a RegExp, it is compared using RegExp.text(value)
.
If it is a value, it is compared with ===
.
set: function
Adds a setter function that will be used to transform the value before writing to the DB.
get: function
Adds a getter function that will be used to transform the value return from the DB.
trim: boolean
Trim whitespace from string when saving to DB.
lowercase: boolean
Convert to lowercase when saving to DB.
uppercase: boolean
Convert to uppercase when saving to DB.
throughput: boolean | {read: number, write: number}
Sets the throughput of the DynamoDB table. The value can either be a number or an Object with the keys read and write {read: 5, write: 2}
. If it is a number, both read and write are configured to the same number. If it is omitted, the default value is 1 for both read and write.
var schema = New Schema({...}, { throughput: 5});
var schema = New Schema({...}, { throughput: { read: 5, write: 2 } });
timestamps: boolean | {createdAt: string, updatedAt: string}
Defines that schema must contain fields to control creation and last update timestamps. If it is set to true, this fields will be createdAt for creation date and updatedAt for last update. for example:
var schema = New Schema({...}, { throughput: 5, timestamps: true});
Also it is possible to specify wich names that field will use, like in the following example:
var schema = New Schema({...}, { throughput: 5, timestamps: {createdAt: 'creationDate', updatedAt: 'lastUpdateDate'});
var Dog = dynamoose.model('Dog', dogSchema);
Creates a new instance of the model. Object keys are assigned to the new model.
var odie = new Dog({
ownerId: 4,
name: 'Odie',
breed: 'Beagle',
color: ['Tan'],
cartoon: true
});
Puts the item in the DynamoDB table. Will overwrite the item.
odie.save(function (err) {
if(err) { return console.log(err); }
console.log('Ta-da!');
});
odie.save({
condition: '#o = :ownerId',
conditionNames: { o: 'ownerId' },
conditionValues: { ownerId: 4 }
}, function (err) {
if(err) { return console.log(err); }
console.log('Ta-da!');
});
Puts multiple items in the table. Will overwrite existing items.
Dog.batchPut([
{
ownerId: 2,
name: 'Princes',
breed: 'Jack Russell Terrier',
color: ['White', 'Brown'],
cartoon: true
},
{
ownerId: 3,
name: 'Toto',
breed: 'Terrier',
color: ['Brown'],
cartoon: false
},
{
ownerId: 4,
name: 'Odie',
breed: 'Beagle',
color: ['Tan'],
cartoon: true
},
{
ownerId: 5,
name: 'Lassie',
breed: 'Beagle',
color: ['Tan'],
cartoon: false
}], function (err, dogs) {
if (err) { return console.log(err); }
console.log('Ta-da!');
});
overwrite: boolean
Overwrite existing item. Defaults to true.
condition: string
An expression for a conditional update. See the AWS documentation for more information about condition expressions.
conditionNames: object
A map of name substitutions for the condition expression.
conditionValues: object
A map of values for the condition expression. Note that in order for automatic object conversion to work, the keys in this object must match schema attribute names.
Creates a new instance of the model and save the item in the table.
Dog.create({
ownerId: 4,
name: 'Odie',
breed: 'Beagle',
color: ['Tan'],
cartoon: true
}, function(err, odie) {
if(err) { return console.log(err); }
console.log('Odie is a ' + odie.breed);
});
Gets an item from the table.
Dog.get({ownerId: 4, name: 'Odie'}, function(err, odie) {
if(err) { return console.log(err); }
console.log('Odie is a ' + odie.breed);
});
Gets multiple items from the table.
Dog.batchGet([{ownerId: 4, name: 'Odie'}, {ownerId: 5, name: 'Lassie'}], function (err, dogs) {
if (err) { return console.log(err); }
console.log('Retrieved two dogs: ' + dogs);
});
Deletes an item from the table.
Dog.delete({ownerId: 4, name: 'Odie'}, function(err) {
if(err) { return console.log(err); }
console.log('Bye bye Odie');
});
Deletes the item from the table.
odie.delete(function(err) {
if(err) { return console.log(err); }
console.log('Bye bye Odie');
});
Deletes multiple items from the table.
Dog.batchDelete([
{ ownerId: 2, name: 'Princes' },
{ ownerId: 3, name: 'Toto' },
{ ownerId: 4, name: 'Odie' },
{ ownerId: 5, name: 'Lassie'}
], function (err) {
if (err) { return console.log(err); }
console.log('Bye bye my friends');
});
Queries a table or index. If callback is not provided, then a Query object is returned. See Query.
Same functionality as query except only return the first result object (if any). See Query.
Scans a table. If callback is not provided, then a Scan object is returned. See Scan.
Updates and existing item in the table. Three types of updates: $PUT, $ADD, and $DELETE. Refer to DynamoDB's updateItem documentation for details on how PUT, ADD, and DELETE work.
allowEmptyArray: boolean
If true, the attribute can be updated to an empty array. If falsey, empty arrays will remove the attribute. Defaults to false.
$PUT
Put is the default behavior. The two example below are identical.
Dog.update({ownerId: 4, name: 'Odie'}, {age: 1}, function (err) {
if(err) { return console.log(err); }
console.log('Just a puppy');
})
Dog.update({ownerId: 4, name: 'Odie'}, {$PUT: {age: 1}}, function (err) {
if(err) { return console.log(err); }
console.log('Just a puppy');
})
$ADD
Dog.update({ownerId: 4, name: 'Odie'}, {$ADD: {age: 1}}, function (err) {
if(err) { return console.log(err); }
console.log('Birthday boy');
})
$DELETE
Dog.update({ownerId: 4, name: 'Odie'}, {$DELETE: {age: null}}, function (err) {
if(err) { return console.log(err); }
console.log('Too old to keep count');
})
Queries a table or index. The query parameter can either the the hash key of the table or global index or a complete query object. If the callback is provided, the exec command is called automatically, and the query parameter must be a query object.
Dog.query('breed').eq('Beagle').exec(function (err, dogs) {
// Look at all the beagles
});
Dog.query({breed: {eq: 'Beagle'} }, function (err, dogs) {
// Look at all the beagles
});
Executes the query against the table or index.
Set the range key of the table or index to query.
Set the atribulte on which to filter.
Use add logic for filters.
Use or logic for filters.
Inverts the filter logic that follows.
Filter attribute for null.
Hash, range key, or filter must equal the value provided. This is the only comparison option allowed for a hash key.
Range key or filter less than the value.
Range key or filter less than or equal value.
Range key or filter greater than or equal value.
Range key or filter greater than the value.
Range key or filter begins with value
Range key or filter is between values a and b.
Attribute is greater than the value.
Filter contains the value.
Filter begins with the value.
Filter is in values array.
Limit the number of responses.
Query with consistent read.
Sort in descending order.
Sort in ascending order (default).
Start query at key. Use LastEvaluatedKey returned in query.exec() callback.
Set the attributes to return.
Scans a table. The optional filter parameter can either be an attribute of the table or a complete filter object. If the callback is provided, the exec command is called automatically, and the scan parameter must be a Scan object.
Dog.scan('breed').contains('Terrier').exec(function (err, dogs) {
// Look at all the Terriers
});
Dog.scan({breed: {contains: 'Terrier'} }, function (err, dogs) {
// Look at all the Terriers
});
To get all the items in a table, do not provide a filter.
Dog.scan().exec(function (err, dogs) {
// Look at all the dogs
if(dogs.lastKey) { // More dogs to get
Dog.scan().startAt(dogs.lastKey).exec(function (err, dogs) {
// Look more dogs
});
}
});
Executes a scan against a table
For readability only. Scans us AND logic for multiple attributes. and()
does not provide any functionality and can be omitted.
Add additional attribute to the filter list.
Inverts the filter logic that follows.
Scan attribute for null.
Attribute is equal to the value.
Attribute is less than the value.
Attribute is less than or equal value.
Attribute is greater than or equal value.
Attribute is greater than the value.
Attribute contains the value.
Attribute begins with the value.
Attribute is in values array.
Attribute is between value a and b.
Limit scan response to limit.
Start scan at key. Use LastEvaluatedKey returned in scan.exec() callback.
Set the attributes to return.