Collection

The Collection object represents a group of related documents, and is backed by a RethinkDB table. Documents in a Collection are identified by a unique key stored in the id field.

// connect to the Horizon server, after Horizon has been loaded via
// <script> tag or require
const hz = new Horizon();

// get a handle to a Collection
const messages = hz("messages");

Collections that do not exist will automatically be created in development mode, or in production if the auto-create-collection option is set. Note that a collection name may not start with the prefix hz_.

Methods on a Collection object allow you to create, read, update and delete documents. Selections can be performed by matching on any field by passing an object to match against.

// Store a message
messages.store({
    id: 1,
    url: avatar_url,
    from: "bob",
    text: "Hello from RethinkDB"
});

// get the first message from Bob
messages.find({from: "bob"}).fetch().subscribe(msg => console.log(msg));

// the same, using a Horizon object directly
hz("messages").find({from: "bob"}).fetch().subscribe(msg => console.log(msg));

// get the message with ID 101; a "shortcut" that only works when fetching based on the
// unique document ID (the equivalent long form is `find({id: 101})`)
messages.find(101).fetch().subscribe(msg => console.log(msg));

// get all messages from Bob, ordered by ID
messages.order("id").findAll({from: "bob"}).fetch().subscribe(msg => console.log(msg));

Methods

Collection.fetch

Return a RxJS Observable containing the query result set as an array.

Collection.fetch()

Unlike watch, the fetch command does not update in real time, but rather returns a “snapshot” of the result set as it exists when fetch is executed. The fetch or watch command ends a Horizon query.

const hz = new Horizon();

hz("messages").fetch().subscribe(
    result => console.log('Result:', result),
    err => console.error(err),
    () => console.log('Results fetched')
);

Collection.subscribe

Provide handlers to a Collection result set.

Collection.fetch().subscribe(readFunction, errorFunction, completedFunction)
Collection.store().subscribe(writeFunction, errorFunction)
Collection.watch().subscribe(changefeedFunction, errorFunction)

This method is not actually part of the Collection class, but is instead a RxJS method.

When subscribe is chained off a read function (i.e., watch) it takes three callback functions:

const hz = new Horizon();

hz("messages").fetch().subscribe(
    result => console.log('Result:', result),
    err => console.error(err),
    () => console.log('Results fetched')
);

Note that when subscribe is chained after fetch, the next callback will receive a single array containing the entire result set at once.

When chained off a write function (e.g., store, upsert), it takes two callback functions:

hz("messages").store([
    {
        from: "agatha",
        text: "Meet at Smugglers' Cove on Saturday"
    },
    {
        id: 3,
        from: "bob",
        text: "Would Superman lose a fight against Wonder Woman?"
    }
]).subscribe(
    (id) => console.log("id value:", id),
    (err) => console.error(err)
);

This would produce output similar to:

id value: f8dd67dc-2301-487a-85ab-c4b573acad2d
id value: 3

In the first case, the id value is an automatically generated UUID; in the second, it was supplied with Bob’s message.

When subscribe is chained off watch, it takes two callback functions:

Read the documentation for watch for more details on returned changefeed dowcuments.

Collection.watch

Convert a query into a changefeed. This returns a RxJS Observable containing the query result set.

Collection.watch({options})

The watch command takes one option, rawChanges. If set true, your application will receive the actual change documents from the RethinkDB changefeed, rather than having Horizon update the result set for you based on the change documents. (Read the RethinkDB changefeed documentation for more details on change documents.)

const hz = new Horizon();

// get a handle to the channels table in a multi channel chat application
const channels = hz("channels");

// receive all active channels, listing them every time a channel is
// added, deleted or changed
channels.watch().subscribe(allChannels => {
    console.log('Channels: ', allChannels);
});

That will return output such as this:

Channels: []
Channels: [{id: 1, name: 'everyone'}]
Channels: [{id: 1, name: 'everyone'}, {id: 2, name: 'rethinkdb'}]
Channels: [{id: 1, name: 'general'}, {id: 2, name: 'rethinkdb'}]

To get raw change documents:

channels.watch({rawChanges: true}).subscribe(allChannels => {
    console.log('Change: ', allChannels)
});

The changes in the example above would produce these raw changes:

Change: {type: 'state', state: 'synced'}
Change: {type: 'add', new_val: {id: 1, name: 'everyone'}, old_val: null}
Change: {type: 'add', new_val: {id: 2, name: 'rethinkdb'}, old_val: null}
Change: {type: 'change', new_val: {id: 1, name: 'general'},
         old_val: {id: 1, name: 'everyone'}}

You can also use watch to turn more complex Horizon queries into changefeeds.

// only watch channels with 10 or more active users
channels.above({users: 10}, "closed").watch().subscribe(allChannels => {
    console.log('Popular channels: ', allChannels);
});

// maintain an updating "top 10" channel list
channels.order("users", "descending").limit(10).watch().subscribe(allChannels => {
    console.log('Popular channels: ', allChannels);
});

Collection.above

Restrict the range of results returned to values that sort above a given value.

Collection.above(id | object[, "closed" | "open"])

The above method may be called with either a key-value pair to match against (e.g., {name: "agatha"} or an id value to look up. The second optional parameter must be the string "closed" or "open", indicating that the specified value will be included (closed) or excluded (open) from the result set. The default is excluded (open): above(10) will return documents with id values higher than (but not equal to) 10.

Values in key-value pairs may be numbers, strings, or even arrays or objects; non-numeric values will be sorted lexicographically, and strings are sorted by UTF-8 codepoint. (Read about Sorting order and ReQL data types in general.)

The above method is often used in conjunction with order, but it may appear after any Horizon method with the exception of find and limit. (However, limit may appear after above.)

const hz = new Horizon();
const messages = hz("messages");

// get all messages with an ID over 100, sorted
messages.order("id").above({id: 100}).fetch();

// the same as above, but using the shorthand for fetching by document ID
messages.order("id").above(100).fetch();

// get all messages with an ID between 101 and 200, sorted
messages.order("id").below(200).above(100).fetch();

// get all users with a reputation score of 50 or over, unsorted
users.above({reputation: 50}, "closed").fetch();

Collection.below

Restrict the range of results returned to values that sort below a given value.

Collection.below(id | object[, "closed" | "open"])

The below method may be called with either a key-value pair to match against (e.g., {name: "agatha"} or an id value to look up. The second optional parameter must be the string "closed" or "open", indicating that the specified value will be included (closed) or excluded (open) from the result set. The default is excluded (open): below(10) will return documents with id values lower than (but not equal to) 10.

Values in key-value pairs may be numbers, strings, or even arrays or objects; non-numeric values will be sorted lexicographically, and strings are sorted by UTF-8 codepoint. (Read about Sorting order and ReQL data types in general.)

The below method may only be used after order, although other methods may be used after it.

const hz = new Horizon();
const messages = hz("messages");

// get all messages with an ID below 100, sorted
messages.order("id").below({id: 100}).fetch();

// the same as above, but using the shorthand for fetching by document ID
messages.order("id").below(100).fetch();

// get all messages with an ID between 101 and 200, sorted
messages.order("id").below(200).above(100).fetch();

// get all users with a reputation score of 50 or below, sorted
users.order("reputation").below({reputation: 50}, "closed").fetch();

Collection.find

Retrieve a single document from a Collection.

Collection.find(id | object)

The find method may be called with either a key-value pair to match against (e.g., {name: "agatha"} or an id value to look up.

const hz = new Horizon();
const messages = hz("messages");

// get the first message from Bob
messages.find({from: "bob"}).fetch();

// get the message with ID 101
messages.find({id: 101}).fetch();

// because we are fetching by the document ID, we can use a shorthand
messages.find(101).fetch();

If no matching document exists, no result will be returned and no error will thrown. To explicitly check for this condition, use RxJS’s defaultIfEmpty operator.

messages.find(id).fetch().defaultIfEmpty().subscribe(
    (msg) => {
        if (msg == null) {
            console.log('Message not found');
            return;
        }
    }
);

Collection.findAll

Retrieve multiple documents from a Collection.

Collection.findAll(object[, object, ...])

The findAll method can be called with one or more key-value pairs to match against (e.g., {email: "bob@example.com"}. Every document that matches the pairs will be returned in a list. (If no documents match, an empty list, [], will be returned.)

const hz = new Horizon();
const messages = hz("messages");

// get all messages from Bob, Agatha and Dave
var messageList = messages.findAll({from: "bob"}, {from: "agatha"}, {from: "dave"}).fetch();

// get all messages from Jane and all messages with a high priority
var messageList = messages.findAll({from: "jane"}, {priority: "high"}).fetch();

Collection.limit

Limit the results of the query to a maximum number of returned documents.

Collection.limit(integer)

The limit command takes a single argument, an integer representing the number of items to limit the results to.

const hz = new Horizon();
const users = hz("users");

// get the 10 most prolific posters
users.order("postCount", "descending").limit(10).fetch();

Collection.order

Sort the results of the query by the values of a given field.

Collection.order(field[, direction])

Fields passed to order may contain numbers, strings, or even arrays or objects; non-numeric values will be sorted lexicographically, and strings are sorted by UTF-8 codepoint. (Read about Sorting order and ReQL data types in general.)

The optional second argument must be a string indication sort direction, either "ascending" or "descending". The default is "ascending".

const hz = new Horizon();
const messages = hz("messages");

// get all messages, ordered ascending by ID value
messages.order("id", "ascending").fetch();

// "ascending" is the default, so it can be left out
messages.order("id");

// get all messages ordered by time, most recent first
messages.order("time", "descending").fetch();

Collection.remove

Delete a single document from a Collection.

Collection.remove(id | object)

The remove method may be called with either an object to be deleted or an id value. In the object case, the object must include an id key.

const hz = new Horizon();
const messages = hz("messages");

// get the message with an ID of 101
var messageObject;
messages.find(101).fetch().subscribe(
    (result) => { messageObject = result; }
);

// delete that object
messages.remove(messageObject);

// it may also be deleted by passing an object just with an ID key
messages.remove({id: 101});

// because we are deleting based on the document ID, we can use a shorthand
messages.remove(101);

Collection.removeAll

Delete multiple documents from a Collection.

Collection.removeAll([id, id, ...] | [object, object, ...])

The removeAll method must be called with an array of objects to be deleted, or id values to remove. The objects must have id keys. You can mix id values and objects within the array.

const hz = new Horizon();
const messages = hz("messages");

// delete messages with the IDs 101, 103 and 109
messages.removeAll([101, 103, 109]);

// find and delete all messages from Bob and Agatha
// this example uses RxJS's "map" command to execute the removeAll and return
// another observable to pass to subscribe for reporting purposes
messages.findAll({from: 'bob'}, {from: 'agatha'}).fetch()
    .mergeMap(messageList => messages.removeAll(messageList))
    .subscribe({
        next(id)   { console.log(`id ${id} was removed`) },
        error(err) { console.error(`Error: ${err}`) },
        complete() { console.log('All items removed successfully') }
    });

Collection.insert

Insert one or more new documents into a Collection.

Collection.insert(object | list of objects)

The insert method can be called either with an object representing a single document, or a list of objects. The objects must have unique id values, and must have id values that do not already exist in the collection or an error will be raised.

const hz = new Horizon();
const messages = hz("messages");

// Insert a single document. There must not be a document with an id of 1 in
// the messages collection.
messages.insert({
    id: 1,
    from: "bob",
    text: "Hello from RethinkDB"
});

// Insert multiple documents at once. All documents must be new.
messages.insert([
    {
        id: 2,
        from: "agatha",
        text: "Meet at Smugglers' Cove on Saturday"
    },
    {
        id: 3,
        from: "bob",
        text: "Would Superman lose a fight against Wonder Woman?"
    }
]);

Collection.replace

Replace one or more existing documents within a Collection.

Collection.replace(object | list of objects)

This is similar to insert, but instead of requiring documents to have new id values, replace requires documents to have existing id values in the Collection. If you try to insert new documents with replace, an error will be raised.

Collection.store

Insert one or more documents into a Collection, replacing existing ones or inserting new ones based on id value.

Collection.store(object | list of objects)

The store method is a combination of insert and replace:

Collection.update

Modify one or more existing documents within a Collection.

Collection.update(object | list of objects)

The update method can be called either with an object representing a single document, or a list of objects. The objects must have id values that already exist in the collection or an error will be raised. This functions similarly to replace, but the update command will only change the specified fields in documents it affects; unspecified fields will be left untouched. (This is similar to a ReQL merge operation.)

const hz = new Horizon();
const messages = hz("messages");

// Update a single document. This will raise an error if there is not an
// existing document in the messages collection with an id value of 1.
messages.update({
    id: 1,
    from: "bob",
    text: "Hello from RethinkDB"
});

// Update multiple documents at once. All documents must already exist.
// Fields that are not specified in the update command will be left
// unmodified.
messages.update([
    {
        id: 1,
        text: "Meet at Smugglers' Cove on Saturday"
    },
    {
        id: 2,
        text: "Would Superman lose in a fight against Wonder Woman?"
    }
]);

Collection.upsert

Update one or more documents in a Collection, modifying existing ones or inserting new ones based on id value.

Collection.upsert(object | list of objects)

The upsert method is a combination of insert and update:

Aggregates and models

Queries on multiple Horizon Collections can be combined using the Horizon.aggregate() method, and aggregates can be turned into parameterized templates using the Horizon.model() method. Read their documentation for details.

RxJS Observable methods and operators

The fetch and watch methods return RxJS Observables, and make all of the Observable methods available.