Package Exports
- @momsfriendlydevco/mongoosy
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (@momsfriendlydevco/mongoosy) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
@MomsFriendlyDevCo/Mongoosy
The Mongoose module but with some quality-of-life additions:
Must haves:
- ObjectIds are automatically strings
- ObjectIds are always convered back to OIDs when saving to the database
- Schema types can be strings
-
meta()compatibility - Express ReST server
- Middleware compatibility
- Connect with sane defaults
- Pointer schema type
-
model.insert()/model.insertOne()(alias ofmodel.create()) -
mongoosy.scenario() - Iterators
- Search compatibility
- Hooks (+late binding hooks)
Nice to haves:
- Works with the MongoSh shell command
- Automatic field surpression (fields prefixed with '_')
-
DEBUGenv variable compatibility - Version incrementing
Differences from Mongoose
For the most part this module is a tiny wrapper around standard Mongoose but some additional quality-of-life fixes have been applied.
Sane connection defaults
Configuring the initial connection options in Mongoose can be a pain. Mongoosy ships with all the latest Mongoose switches tuned to their correct values, preventing any depreciation warnings.
NOTE: mongoosy.connect() and mongoosy.compileModels() need to be called seperately. This is so calls to schema construction can be buffered with additional hooks and virtuals declared before the entire schema structure is ready to compile.
Version incrementing
All documents have the meta __v property which now gets automatically incremented on each save.
ObjectIds are always strings
Mongoose makes comparing ObjectIds painful - always having to remember that while they look like strings they are actually objects and object comparison in JavaScript is unreliable. To make life easier, all ObjectIds fetched from the database are always simple strings which get converted back to the correct BSON type on save.
Promise.all([
mongoosy.models.users.findOne({name: 'Adam'}),
mongoosy.models.users.findOne({role: 'admin'}),
]).then(([adam, admin]) => {
console.log(
adam._id == admin._id // Simple string comparison
? 'Adam is admin'
: 'Adam is not admin'
);
})Easier debugging
Mongoosy uses wraps all data read/write functions in the debug NPM module for debugging.
To enable debugging set the environment variable to DEBUG=mongoosy for all debugging, DEBUG=mongoosy:METHOD for a specific method or combine globs as needed.
For example:
# Execute myFile.js showing all debugging (can be very loud)
DEBUG=mongoosy node myFile.js
# Execute myFile.js showing all updateOne calls
DEBUG=mongoosy:updateOne node myFile.js
# Execute myFile.js showing insert and delete calls
DEBUG=mongoosy:insert*,mongoosy:delete* node myFile.jsNote that enabling the debugging mode adds a small overhead to all model methods.
Models have extra alias functions
Models have the following additional aliased functions:
model.insert()/model.insertOne()(alias ofmodel.create()) - Bringing the syntax more into line withmodel.updateOne/model.updateMany()
Meta 'change' event
The Mongo middleware functions are a little annoying when trying to differenciate between new, saved and updated documents. Mongoosy implements a generic 'change' event which tracks all of these simultaniously with one binding:
mongoosy.models.widgets.pre('change', function() {
var doc = this; // Context is the full document (pulled automatically even on partial queries)
doc.$isNew() // Returns if the doc is being created for the first time (in the case of 'create' calls)
});As with all middleware this function is async compatible and stackable.
Pointer schema type
Pointers are really just one Mongo document pointing at another. The pointer schema type is actually just an ObjectId by default but it doesn't differenciate on storage methods (i.e. it can be an ObjectId but can be easily extended to storing UUIDs or some other item).
Schema virtuals are chainable
The Virtuals configuration in Mongoose is awkward when it comes to adding methods onto schemas.
Mongoosy supports a simple (id, {get, set}) or (id, getter, setter) syntax without exiting the model chain:
mongoosy.schema('users')
.virtual('password', ()=> 'RESTRICTED', function(pass) {
// Very crappy, yet predictable password hasher that removes all consonants
this._password = pass
.toLowerCase()
.replace(/[^aeiou]+/g, '');
})
.virtual('passwordStrength', function() {
// Returns the length of the (badly, see above) hashed password which is an approximate indicator of hash strength
return (this._password.length || 0);
})Scenario support built in
Importing large datasets (with linked OIDs) is now supported as built-in functionality.
See the mongoosy.scenario() function for details.
ReST server support built in
Connecting a Mongoose collection to Express compatible middleware is now supported as built-in functionality.
See the mongoosy.models.MODEL.serve() function for details.
API
In addition to the default Mongoose methods this module also provides a few conveinence functions.
mongoosy.dropCollection(name)
Drops a single collection by name. Returns a promise which will resolve with a boolean true if a collection was dropped or false if the collection didn't exist anyway.
mongoosy.dropDatabase(confirmation)
Drop an entire database allow with data, collections and indexes.
Requires a confirmation object exactly matching {$confirmDrop: true} for obvious reasons.
Returns a promise which will resolve when the database has dropped.
mongoosy.scenario(inputs, options)
Utility function to quickly load a JSON / JS file into a model. Inputs can be a JS object(s) or a file glob (or array of globs) to process.
This function acts similar to insertMany() with the following differences:
- Models are specified as the top level object key with an array of documents as the value - thus you can import to multiple models at the same time
- The special
$key is accepted as an identifier for a document, the string value is used as the identifier - i.e. give this inserted document a temporary alias - Any string field starting with
$should have the computed ID value of the named document inserted in place - Creation order is automatically calculated - documents with prerequisites are inserted in the correct order
mongoosy.scenario({
companies: [
{
$: '$company.acme',
name: 'Acme Inc',
},
],
users: [
{
$: 'users.joe',
name: 'Joe Random',
company: '$company.acme', // <- The ID of the first company is inserted here
},
],
});In the above scenario the company is inserted first, its ID remembered and used to populate the company field of the user.
mongoosy.serve(model, options)
Create a Express compatible middleware backend which functions as a ReST server.
This is an alternate method to call mongoosy.models.MODEL.serve(options) with an explicit model.
mongoosy.models.MODEL.serve(options)
Create a Express compatible middleware backend which functions as a ReST server.
var app = express();
app.use(bodyParser.json());
app.use('/api/movies/:id?', mongoosy.models.movies.serve({
create: true,
get: true,
query: true,
count: true,
save: true,
delete: true,
}));The following options are supported:
| Opition | Type | Default | Description |
|---|---|---|---|
get |
See notes | true |
Enable getting of records or specify middleware(s) to execute beforehand |
query |
See notes | true |
Enable querying of records or specify middleware(s) to execute beforehand |
queryForce |
Promiseable function | Called as (req) to override req.query with either a static object or an evaluated promise. Called as (req) |
|
queryValidate |
Promiseable function | Validate an incomming query, similar to queryForce. Throw an error to reject. Called as (req). |
|
count |
See notes | true |
Enable counting of records or specify middleware(s) to execute beforehand |
countParam |
string |
"count" |
Special case URL suffix to identify that we are performating a count operation and not looking up an ID |
create |
See notes | true |
Enable creating of records or specify middleware(s) to execute beforehand |
save |
See notes | true |
Enable updating of records or specify middleware(s) to execute beforehand |
delete |
See notes | true |
Enable deleting of records or specify middleware(s) to execute beforehand |
meta |
See notes | true |
Enable retrieving the structure of the collection |
metaCustomFields |
array<string> |
[] |
Array of additional custom fields to expose during a meta data query |
metaParam |
string |
"meta" |
Special case URL suffix to identify that we are performating a meta operation and not looking up an ID |
param |
string |
"id" |
Where to look in req.params for the document ID to get/update/delete |
searchId |
string |
"_id" |
What field to search by when fetching / updating / deleting documents |
selectHidden |
boolean |
false |
Automatically surpress all output fields prefixed with _ |
forbidHidden |
boolean |
true |
Forbid the selection of fields prefixed with _ |
neverHidden |
array<string> |
['_id', '__v'] |
Array of fields which are excluded from hiding |
errorHandler |
Function | See code | How to handle errors, default is to use Expresses res.status(code).send(text) method. Called as (res, code, text) |
Notes:
- The
get/query/count/create/save/delete/metamethods can be a simple boolean to enable / disable, an array of Express middleware functions or a single middleware function. Middleware are all called as(req, res, next)and can either callnext()to accept the request or handle output viares.send() - If
queryForceis a function it is expected to either mutatereq.queryor return the new absolute contents of that object which is spliced in place. Any falsy return is ignored
EVENT: model
Emitted as (modelInstance) when a model is declared.
Useful to extend the base model functionality when a new model appears.
Note that event hooks cannot be bound at this stage. Trap the schema event and add hooks to that instead.
EVENT: schema
Emitted as (schemaInstance) when a schema is declared.
Useful to extend the schema functionality with hooks and other custom functionality.
Migration
When migrating from Monoxide to Mongoose there are a few minor things to remember:
- An additional
mongoosy.compileModels()call is needed post-connection + post-schema loading to ready the models - Scenarios now use
$as the ID field (formally:_ref), they also require all ID lookup fields to have a dollar prefix and the ID to match (including the prefix) - Queries returning no documents no longer automatically fail if
$errNoDocsis set, usequery.orFail()instead model.use()->model.plugin()model.findOneByID()->model.findOne()- When setting
{ type: Object, default: {} }pass{ minimize: false }option toSchemain order to ensure the key is created. - Iterators now use the default Mongoose cursor system. Use the
Thing.find().cursor().map()pattern instead of filter / map / forEach - Virtuals do not pass the document as the first parameter. Use
thiswithin the getter / setter function to recieve the current document model.hook()is no longer supported. Use the Mongoose pre/post methods instead -model.pre('save', fn)/model.post('save', fn)instead.fnis called as(doc)and will be waited on if its a promise.