npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

mongo-migrations

v1.1.1

Published

Asynchronous MongoDB migration framework for Node.js based on node-migrate

Downloads

15

Readme

Mongo Migrations

NPM Version NPM Downloads MIT licensed npm downloads

Asynchronous MongoDB migration framework for Node.js based on node-migrate;

Installation

$ npm install mongo-migrations

Testing

To run tests use

$ npm run test

Tests require MongoDB running on default location/port;

Usage

Usage: mongo-migrations [options] [command]

Options:

  -V, --version  output the version number
  -h, --help     output usage information

Commands:

  init           Initalize the migrations tool in a project
  list           List migrations and their status
  create <name>  Create a new migration
  up [name]      Migrate up to a give migration
  down [name]    Migrate down to a given migration
  help [cmd]     display help for [cmd]

For help with the individual commands, see mongo-migrations help [cmd]. Each command has some helpful flags for customising the behavior of the tool.

Programmatic usage

const migrate = require('mongo-migrations')

(async () => {
  try {
    const set = await migrate.load(
      { stateStore: '.migrate' }
    );
    await set.up();
    console.log('migrations successfully ran');
  } catch (e) {
    console.error(e);
  }
})();

Configuration

You can overwrite basic functionalities of application by inserting migrate-config.json file inside of your project root directory. Default configuration is set inside of config.json file and looks like follow:

{
  "mongodb": {
    "connectionUrl": "mongodb://localhost:27017"
  },
  "stateFile": ".migrate",
  "dryRun": false,
  "migrationsDirectory": "migrations",
  "storeClassPath": "../stores/FileStore.js",
  "templateFactoryPath": "../factories/TemplateFactory.js",
  "templateFilePath": "../template.js",
  "useMongoStore": false,
  "mongoStore": {
    "connectionUrl": "mongodb://localhost:27017",
    "database": "test",
    "collection": "migrations",
    "idField": "test-migrations"
  }
}

For all available options please refer to Config.js file.

Please remember, configuration file will be overwritten by any run command options.

Since version 1.1.0 mongo-migrations allow usage of ENVs inside of migrate-config.json file. To indicate that specific field should be loaded from ENV use this structure:

{
  //...
  "mongodb": {
    "connectionUrl": "${MONGODB_CONNECTION_URL}"
  }
  //...
}

Note: when specified field ENV will be not present ConfigLoader will log warning information and use default one

Note: to load variables dynamically you can use --env switch for dotenv, aws-env or any other available library

Note: due to truthy/falsy nature, ENV value with empty string will be considered as false for boolean fields

Note: by default all config boolean fields are pointing to false

Creating Migrations

To create a migration, execute migrate create <title> with a title. By default, a file in ./migrations/ will be created with the following content:

module.exports.up = async (mongoClient) => {
  return undefined;
};

module.exports.down = async (mongoClient) => {
  return undefined;
};

For example:

$ migrate create add-pets
$ migrate create add-owners

The first call creates ./migrations/{timestamp in milliseconds}-add-pets.js, which we can populate:

module.exports.up = async (mongoClient) => {
  const db = mongoClient.db('test');
  await db.createCollection('pets');
  await db.collection('pets').insertOne({ _id: 'tobi' });
  await db.collection('pets').insertOne({ _id: 'loki' });
  await db.collection('pets').insertOne({ _id: 'jane' });
};

module.exports.down = async (mongoClient) => {
  const db = mongoClient.db('test');
  await db.collection('pets').deleteOne({ _id: 'tobi' });
  await db.collection('pets').deleteOne({ _id: 'loki' });
  await db.collection('pets').deleteOne({ _id: 'jane' });
  await db.collection('pets').drop();
};

The second creates ./migrations/{timestamp in milliseconds}-add-owners.js, which we can populate:

module.exports.up = (mongoClient) => {
  const db = mongoClient.db('test');
  await db.createCollection('owners');
  await db.collection('owners').insertOne({ _id: 'taylor' });
  await db.collection('owners').insertOne({ _id: 'tj' });
}

module.exports.down = (mongoClient) => {
  const db = mongoClient.db('test');
  await db.collection('owners').deleteOne({ _id: 'taylor' });
  await db.collection('owners').deleteOne({ _id: 'tj' });
  await db.collection('owners').drop();
}

Advanced migration creation

When creating migrations you have a bunch of other options to help you control how the migrations are created. You can fully configure the way the migration is made with a template-factory, which is just a class exported as a node module. A good example of a generator is the default one shipped with this package.

The create command accepts a flag for pointing the tool at a generator, for example:

$ mongo-migrations create --generator ./my-migrate-generator.js

A more simple and common thing you might want is to just change the default template file which is created. To do this, you can simply pass the template-file flag:

$ mongo-migrations create --template-file ./my-migration-template.js

Lastly, if you want to use newer ECMAscript features, or language addons like TypeScript, for your migrations, you can use the compiler flag. For example, to use babel with your migrations, you can do the following:

$ npm install --save babel-register
$ mongo-migrations create --compiler="js:babel-register" foo
$ mongo-migrations up --compiler="js:babel-register"

MongoDB Client

MongoDB Client wersion used inside of the project is locked to 3.1.6. For API reference please refer to mongodb github.io page.

Running Migrations

When first running the migrations, all will be executed in sequence.

$ mongo-migrations
  up : migrations/1316027432511-add-pets.js
  up : migrations/1316027432512-add-jane.js
  up : migrations/1316027432575-add-owners.js
  up : migrations/1316027433425-coolest-pet.js
  migration : complete

Subsequent attempts will simply output "complete", as they have already been executed. mongo-migrations knows this because it stores the current state in ./.migrate which is typically a file that SCMs like GIT should ignore.

$ mongo-migrations
  migration : complete

If we were to create another migration using mongo-migrations create, and then execute migrations again, we would execute only those not previously executed:

$ mongo-migrations
  up : migrates/1316027433455-coolest-owner.js

You can also run migrations incrementally by specifying a migration.

$ mongo-migrations up 1316027433425-coolest-pet.js
  up : migrations/1316027432511-add-pets.js
  up : migrations/1316027432512-add-jane.js
  up : migrations/1316027432575-add-owners.js
  up : migrations/1316027433425-coolest-pet.js
  migration : complete

This will run up-migrations up to (and including) 1316027433425-coolest-pet.js. Similarly you can run down-migrations up to (and including) a specific migration, instead of migrating all the way down.

$ mongo-migrations down 1316027432512-add-jane.js
  down : migrations/1316027432575-add-owners.js
  down : migrations/1316027432512-add-jane.js
  migration : complete

Any time you want to see the current state of the migrations, you can run mongo-migrations list to see an output like:

$ migrate list
  1316027432511-add-pets.js [2017-09-23] : <No Description>
  1316027432512-add-jane.js [2017-09-23] : <No Description>

The description can be added by exporting a description field from the migration file.

Dry-run

While running --dry-run mongodb module methods will be overwritten to avoid making changes inside of the database. All methods return log with information what operation will be made on DB unless stated diffrently.

List of currently stubbed methods:

  • Db
    • createCollection()
  • Collection
    • find() (returns original find() call with log)
    • findOne() (returns original findOne() call with log)
    • createIndex()
    • updateOne()
    • updateMany()
    • drop()
    • insertOne()
    • deleteOne()

MongoDB State Storage

Recently added MongoStore allows you to store migration state in MongoDB database instance. To use this store pass --mongo-store while running command (in which case mongo-migrations will load MongoStore default config) or set config option useMongoStore to true.

Note: MongoStore creates distinct connection to MongoDB to allow state to be stored inside of any DB user wants.

Note: inside of config file you can precise db, collection and _id field name which will be used while storing migration state. See Config for apropriate naming.

Custom State Storage

By default, mongo-migrations stores the state of the migrations which have been run in a file (.migrate). But you can provide a custom storage engine if you would like to do something different, like storing them in your database of choice. A storage engine has a simple interface of await load() and await save(set). As long as what goes in as set comes out the same on load, then you are good to go!

If you are using the provided cli, you can specify the store implementation with the --store flag, which is be a require-able node module. For example:

$ mongo-migrations up --store="my-migration-store"

API

await migrate.load(opts)

Calls the callback with a Set based on the options passed. Options:

  • set: A set instance if you created your own
  • config: An initialized Config object with mongodb reference
  • stateStore: A store instance to load and store migration state, or a string which is a path to the migration state file
  • migrationsDirectory: The path to the migrations directory
  • filterFunction: A filter function which will be called for each file found in the migrations directory
  • sortFunction: A sort function to ensure migration order

await set.up(migrationName)

Migrates up to the specified migrationName or, if none is specified, to the latest migration.

await set.down(migrationName)

Migrates down to the specified migrationName or, if none is specified, to the first migration.

TODO

  • Reconstruct unit test
    • Files should keep 1:1 structure with lib
    • Reconstruct/rewrite test cases for specific modules
    • Add missing unit test cases
  • Reconstruct integration test files
  • Add coverage checking tool
  • Add PR test runner
  • Introduce .eslint
  • Cleanup README to be more readable
  • Enhance CHANGELOG to include more information