reactive-mongo
Meteor smart package which provides a fully reactive server-side MongoDB queries. This allows them to be used in server-side autorun, together with other fibers-enabled synchronous (blocking) code.
Adding this package to your Meteor application will make all MongoDB queries
reactive by default (you can still specify reactive: false
to
queries to disable reactivity for a specific query, or use
Tracker.nonreactive
). It will also automatically enable
server-side autorun. All this might break some existing
server-side code which might not expect to be reactive. Inspect locations where your code or packages you are using
already (before using this package) call Tracker.autorun
on the server. In most cases this occurs only in the code
which is shared between client and server.
Server side only.
Installation
meteor add peerlibrary:reactive-mongo
Ordered vs. unordered cursors
In Meteor, there's a concept of ordered cursors. If a cursor is ordered, then when the order of the documents in the result set changes, the reactive computation will be invalidated and it be will re-run.
By default, this package will use an ordered cursor if a sort
option is present in the query.
If no sort
option is specified, it will use an unordered cursor.
To override the default functionality, you can explicitly force or un-force the type of a cursor
by passing an ordered
option to your find
query:
1// Server-side code. 2 3// Will use an ordered cursor since sort option is present. 4Tracker.autorun(() => { 5 Posts.find({topic: 'news'}, {sort: {name: 1}}).fetch(); 6}); 7 8// Will use an unordered cursor since no sort option is present. 9Tracker.autorun(() => { 10 Posts.find({topic: 'news'}).fetch(); 11}); 12 13// Will not use an ordered cursor since it is explicitly disabled. 14Tracker.autorun(() => { 15 Posts.find({topic: 'news'}, {sort: {name: 1}, ordered: false}).fetch(); 16}); 17 18// Will use an ordered cursor since it is explicitly enabled. 19Tracker.autorun(() => { 20 Posts.find({topic: 'news'}, {ordered: true}).fetch(); 21});
You wan to choose the cursor so that invalidations happen when and only when you want them to happen.
Reactive queries and polling vs. oplog
A common use case for server-side reactive computations is using a findOne
to do a reactive join.
Meteor's server-side findOne
is a find(selector, {limit: 1}).fetch()[0]
under the hood.
Because Meteor oplog does not support
limit
without sort
, calling Collection.findOne(someId)
in a server-side reactive computation will
default to using polling.
If you would like queries inside a server-side reactive computation to use oplog, you will need to specify
a sort option for your findOne
query and pass ordered: false
to use an unordered cursor:
1// Server-side code. 2 3// Will use oplog since it has sort, limit, and is unordered. 4Tracker.autorun(() => { 5 Collection.findOne(someId, {sort: {_id: 1}, ordered: false}); 6}); 7 8// Will use polling because a limit (from findOne) but no sort is specified. 9Tracker.autorun(() => { 10 Collection.findOne(someId); 11});
Acknowledgments
This package is based on the great work by Diggory Blake who made the first implementation.