/lerna-cola

Superpowers for your Lerna monorepos.

Primary LanguageJavaScriptMIT LicenseMIT

Work in Progress 👀

lerna-cola is being built in parallel to a large scale production grade project, thereby getting some serious dogfooding in order to work out the kinks and settle on a useful API. Whilst we have made a lot of progress this is still very much an alpha version of the project.


Lerna Cola 🥤

Superpowers for your Lerna monorepos.

Clean, build, develop, and deploy your packages utilising a rich plugin ecosystem.

TOC

Introduction

Lerna makes it crazy easy to manage cross package dependencies and provides sane methods to version them. It takes away the fear of creating and maintaining a wide set of packages, allowing us to fully embrace the module ethos by creating packages with isolated responsibilities.

Lerna Cola wants to build on top of these core principles by providing the following additional features:

  • Easily enrich your packages with a compilation/transpilation/bundling step (babel/flow/typescript/reasonml/webpack/parcel/etc/etc/etc).
  • Take away the fear of building a wide set of microservices/lambda packages by providing a rich development service that handles hot/auto reloading of your packages. This allows for a fluid development experience reminiscent of old school single package/repository based development.
  • Deploy your packages with a simple command to a cloud provider of your choice.

You access the features via one of the 4 CLI commands that Lerna Cola provides: clean, build, develop, and deploy.

The commands utilise a rich plugin eco-system, allowing for 3rd party contributions.

Lift your build, development and deployment to the root of your monorepo, keep your packages clean, and utilise the full benefits of a monorepo structure.

Requirements

  • Node >= 8

    Version 8 was LTS at the time of writing this so a decision was made to run with it.

(Not Really) Requirements

  • Lerna

    To tell you the truth, we don't strictly require that you use Lerna. You could very well use straight up Yarn workspaces or any other monorepo enabling tool, but in our opinion you would be missing out on some very cool features that Lerna provides.

Getting started

First, you need to install the cli

yarn add @lerna-cola/cli -DW

or, via NPM:

npm i @lerna-cola/cli -D

Secondly, you'll need a lerna-cola.js configuration file

You'll need to reference the Configuration on how to do this.

Lastly, fire up the lern-cola CLI

You can run the help to see the help commands

yarn lerna-cola help

or, via NPX

npx lerna-cola help

Sample Application

We definitely recommend that you read the all of the documentation first in order to gain a better understanding of Lerna Cola, however, we want to highlight early on that a sample application is maintained here:

This provides a great way for you to quickly clone and run a non-trivial project in order to assess the benefits that Lerna Cola could bring to your monorepos.

Video Walkthrough

Coming soon

Configuration

To use Lerna Cola you will need to create a configuration file named either lerna-cola.json or lerna-cola.js within the root of your monorepo.

Note: When creating a .js file ensure that you export the configuration object via module.exports.

Example Configuration

Before we describe the configuration schema, it may be helpful to review a "typical" configuration:

module.exports = {
  packages: {
    // The following two packages are microservices where we are using the babel
    // plugin to transpile them to support our current node version, and the
    // server develop plugin which will take care of executing and restarting
    // our microservices for any changes up their dependency trees.

    'microservice-one': {
      buildPlugin: '@lerna-cola/plugin-build-babel',
      developPlugin: 'plugin-develop-server',
    },

    'microservice-two': {
      buildPlugin: '@lerna-cola/plugin-build-babel',
      developPlugin: 'plugin-develop-server',
    },

    // The following package is a "Create React App" package which comes with
    // it's own build and develop (start) scripts. We will therefore use the
    // Lerna Cola script plugin to delegate to the CRA scripts.

    'my-ui': {
      buildPlugin: {
        name: 'plugin-script',
        options: {
          scriptName: 'build',
        },
      },
      developPlugin: {
        name: 'plugin-script',
        options: {
          scriptName: 'start',
        },
      },
    },
  },
}

Within this example we are providing a Lerna Cola configuration for 3 of our packages within our monorepo. We need not provide a configuration for every package within our monorepo; only the ones that we wish to execute Lerna Cola plugins against. Lerna Cola will still be aware of the other packages within our repo, for example providing hot reloading on our servers/apps when a change occurs on our shared utils package.

In our example we have two microservices that will be Babel transpiled by the build command, and will be treated as server process (with execution and hot reloading) by the develop command. The third package configuration utilities a special Create React App plugin that contains configuration for both the build and develop commands.

This is a very basic example, however, it is illustrative of how quickly you can provide a centralised and coordinated configuration for your monorepo packages. Plugins can of course be customised via options. We recommend you read the plugin docs for detailed information on them.

Configuration Schema

The configuration is an Object/JSON structure that supports the following schema:

  • commandHooks (Object, optional)

    A set of hooks to allow you to execute a function pre/post each command. You can specify a set of hooks for each of the commands:

    • clean
    • build
    • develop
    • deploy

    Each of them support the following configuration Object:

    • pre: (Function, optional)

      Runs prior to the specifed command. Can return a Promise, which will be waited on to resolve before proceeding.

    • post: (Function, optional)

      Runs after to the specifed command completes/fails. Can return a Promise, which will be waited on to resolve before proceeding.

    Example:

    // lerna-cola.js
    module.exports = {
      commandHooks: {
        build: {
          pre: () => Promise.resolve(),
          post: () => Promise.resolve(),
        },
      },
    }
  • packages (Object, optional)

    An object where each key is the name of a package within your repository (matching the name of the package within the package's package.json file). The value against each key is considered a package configuration object, supporting the following schema:

    • buildPlugin (string || Object, optional)

      The plugin to use when building the package. Please see the Plugins documentation for more information.

    • developPlugin (string || Object, optional)

      The plugin to use when developing the package. Please see the Plugins documentation for more information.

    • deployPlugin (string || Object, optional)

      The plugin to use when deploying the package. Please see the Plugins documentation for more information.

    • srcDir (string, optional, default: 'src')

      The directory within your package containing the source.

    • entryFile (string, optional, default: 'index.js')

      The file within your srcDir that should be considered the "entry" file for your package.

    • outputDir (string, optional, default: 'build')

      The directory which should be targetted for build output by the build plugins. Also used by the clean plugin to know which directory should be removed.

    • disableSrcWatching (boolean, optional, default: false)

      Use this to disable watching of the source when running the develop command. This can be useful if you don't care about tracking changes for a specific package within your repository, or you have a seperate process that will already track source changes for the respective package (e.g. the start script within a Create React App project).

  • packageSources (Array<string>, optional, default: see below)

    An array of globs, paths where your packages are contained. By default it uses the same configuration as Lerna, i.e.:

      "packageSources": [
        "packages/*"
      ]

CLI Commands

Below is an brief description of each command that the CLI provides. You can request for help via the CLI for all of the commands like so:

lerna-cola help

Or for an individual command like so:

lerna-cola build help

When executing commands all of the environment variables that are currently available (i.e. process.env) will be passed into the respective processes that are spawned for each package.

clean

Clean the output from the the build command.

By default the plugin-clean-build plugin is used, however, you can customise this within your configuration.

build

Take advantage of one of the wide array of plugins to babel/flow/typescript/reasonml/etc/etc/etc transpile/build your packages with the ability to easily share and manage configuration at the root of your monorepo.

When executed it will run all of the configured plugins for all of your packages that have a buildPlugin configured within the lerna-cola.json configuration file. The package build order is based upon on a topological sort against the dependency tree of your packages within the monorepo (i.e. they build in the correct order).

develop

Run a full auto-reloading development environment in which any changes to one package cascade through your package dependency tree. No more need to manually rebuild/restart servers during development allow for a blazingly hot development experience.

All of your packages will be watched (even the ones without any explicit configuration within your configuration), with any changes being propagated through the dependency tree subsequently executing any "develop" plugins that are configured against your projects.

Note: If you provide a "build" plugin against on your packages, but no explicit "develop" plugin, then the system will automatically execute the "build" plugin when handling changes.

All of the logs/output from your packages will be "merged" and printed within the console window where you ran the develop command. The console output contains a uniquely colored column to the left allowing you to easily identify the output for each respective package.

deploy

Deploy your apps with a simple one line command to a cloud provider supported by the plugin system.

When executing this command your packages will be built in topological sort order (based on their dependency tree between each other) and then will subsequently be deployed via their configured plugins.

Plugins

Lerna Cola is powered by a powerful plugin system that allows the build, develop, and deploy commands to support a multitude of targets. It is very easy to build your own plugin should you desire to so - please see the "Plugin Development" section for more information.

Plugins are split by "core" plugins, which are bundled with the main @lerna-cola/cli package, and "package" plugins which could either be official Lerna Cola packages, 3rd party packages, or private packages of your own making.

You define plugins against each of your package configurations within your Lerna Cola configuration.

Plugins allow two forms of assignment.

Form 1 - specify the plugin by name

// lerna-cola.js
module.exports = {
  packages: {
    'my-package': {
      buildPlugin: '@lerna-cola/plugin-build-babel',
    },
  },
}

Form 2 - provide options to the plugin

// lerna-cola.js
module.exports = {
  packages: {
    'my-package': {
      buildPlugin: {
        name: '@lerna-cola/plugin-build-babel',
        options: {
          config: {
            presets: ['babel-preset-env'],
          },
        },
      },
    },
  },
}

You can pass down any arbitrary set of options, which will be made available to the respective plugin. Please refer to the documentation for each plugin in terms of what options it supports.

Core Plugins

Below are the "core" plugins which will be immediately available when you add Lerna Cola to your repository.

plugin-clean-build

A clean command plugin.

This plugin will be automatically assigned to any package with a buildPlugin defined.

When executed it will remove the output directory targetted by your respective build plugin.

plugin-develop-server

A develop command plugin.

plugin-script

This is a special plugin that can be configured against any of your plugins for a package. i.e. cleanPlugin, buildPlugin, developPlugin, and deployPlugin.

It allows you to delegate to a script defined within the package.json of your targetted package.

This can be especially useful for packages that come with their own sets of scripts (e.g. A Create React App package).

Options

This plugin supports the following options:

  • scriptName (string, required)

    The name of the script to run within your target package.

  • runForEveryChange (boolean, default: false)

    Only used when assigned against the developPlugin. Setting this value to true will ensure that the your script will be executed any time a change is registered within your packages' source or against one of its dependencies. If your script spawns a long run child process (for example a Create React App server), then an existing running instance will be destroyed before the script is ran again.

Example

// lerna-cola.js
module.exports = {
  packages: {
    'my-create-react-app': {
      buildPlugin: {
        name: 'script',
        options: {
          scriptName: 'build',
        },
      },
    },
  },
}

Official Plugins

@lerna-cola/plugin-build-babel

A build command plugin.

This plugin will transpile your package's srcDir (see the configuration) using Babel, outputing the results into your packages outputDir (see the configuration).

By default it will use a .babelrc found within your packages root directory, and if one is not found then it will look for a .babelrc within the root of your monorepo. You can alternatively provide a babel configuration via the plugin's options.

Options

This plugin supports the following options:

  • config (string || Object, optional)

    The babel configuration to use for the transpilation.

    This can be one of two things:

    • The name of a package where the main export is a babel configuration object.
    • An object containing the babel configuration.

    Note: Lerna Cola ships a simple babel config package which you could use. It is called @lerna-cola/babel-config

We highly recommend that you enable sourcemaps output within your configuration to aid debugging.

Example

Specifying an inline config:

// lerna-cola.js
module.exports = {
  packages: {
    'my-lib': {
      buildPlugin: {
        name: '@lerna-cola/plugin-build-babel',
        options: {
          config: {
            presets: ['babel-preset-env'],
          },
        },
      },
    },
  },
}

Specifying a package containing the configuration:

// lerna-cola.js
module.exports = {
  packages: {
    'my-lib': {
      buildPlugin: {
        name: '@lerna-cola/plugin-build-babel',
        options: {
          config: 'my-babel-config-package',
        },
      },
    },
  },
}

@lerna-cola/plugin-build-flow

A build command plugin.

This plugin will transpile your package's srcDir (see the configuration), stripping your source of any flow annotations, and outputing the result (along with *.flow) files into your package's outputDir (see the configuration).

Options

This plugin supports the following options:

  • inputs (Array<string> , optional, default: see below)

    An array of glob strings which should be used to match the files that will be processed.

    Defaults to the following:

    ["**/*.js", "!__tests__", "!test.js"]

Example

// lerna-cola.js
module.exports = {
  packages: {
    'my-lib': {
      buildPlugin: '@lerna-cola/plugin-build-flow',
    },
  },
}

@lerna-cola/plugin-deploy-now

A deploy command plugin.

This plugin allows your package to be deployed to Zeit's now cloud service.

For this plugin to work you need to have installed the now CLI, i.e.:

npm i -g now

You also need to ensure that you have logged into your now account:

now login

Options

This plugin supports the following options:

  • settings (Object, optional)

    Any settings officially supported by now.

    We highly recommend you set the alias setting.

    Some defaults are automatically applied by the plugin, but you can override them:

    {
      forwardNpm: true,
      public: false
    }
  • disableRemovePrevious (boolean, optional, default: false)

    If you provide an alias within the settings option then this plugin will by default remove any previous deployments that were deployed against the target alias. This avoids any unnecessary build up of old deployments, which can become tedious to manage.

    Set this to true to prevent the removal of prior deployments of an alias target.

  • deployTimeoutMins (number, optional, default: 15)

    The number of minutes to wait before timing out the deployment, subsequently stopping the deployment process with a failure indicated.

  • passThroughEnvVars (Array<string> , optional)

    An array of strings, describing which environment variables (currently available on your process.env) should be passed into the now deployment.

    e.g.

    // lerna-cola.js
    module.exports = {
      packages: {
        'my-app': {
          deployPlugin: {
            name: '@lerna-cola/plugin-deploy-now',
            options: {
              alias: 'my-app.com',
              passThroughEnvVars: ['MY_DB_USERNAME', 'MY_DB_PASSWORD'],
            },
          },
        },
      },
    }
  • pathAlias (Object, optional)

    Any now path alias rules to be applied to the deployment.

    This is a really helpful setting, allowing you to proxy paths of your deployment to other deployments.

Example

// lerna-cola.js
module.exports = {
  packages: {
    'my-app': {
      buildPlugin: {
        name: '@lerna-cola/plugin-deploy-now',
        options: {
          settings: {
            alias: 'my-app.com',
            sfo1: {
              sfo1: {
                min: 1,
                max: 5,
              },
            },
          },
        },
      },
    },
  },
}

@lerna-cola/plugin-build-typescript

Coming soon

@lerna-cola/plugin-build-reasonml

Coming soon

@lerna-cola/plugin-build-parcel

Coming soon

@lerna-cola/plugin-build-rollup

Coming soon

3rd Party Plugins

Hopefully we will have some soon. 🤞

Please submit a PR to add yours here.

Plugin Development

Developing plugins is very easy. You can create a plugin to support one or more of the 4 commands that Lerna Cola provides.

To create a plugin simply create new package where the "main" on its package.json file points to the plugin module.

Your newly created package can adopt the behaviour of any of the plugin types as long as provides implementation for the interface specified for each plugin type. For example below is a template for a plugin package that supports all the command types (clean/build/develop/deploy):

// my-plugin.js
module.exports = {
  name: 'my-plugin',
  clean: (pkg, options, args) => Promise.resolve('todo'),
  build: (pkg, options, args) => Promise.resolve('todo'),
  develop: (pkg, options, args) => Promise.resolve('todo'),
  deploy: (pkg, options, args) => Promise.resolve('todo'),
}

Looking at the above, you can see that a plugin is really just a simple object, containing functions that map to the command names. Each of the functions will receive a set of useful arguments, which will be described below, and need to return a Promise in order to indicate when they have completed.

When running a command, the associated plugins will get executed for each package that they are configured against. The plugin functions will receive as an argument an object describing the package that it is running against, as well as the specific options that were defined when assigning the plugin to the package within the Lerna Cola configuration file.

Lets review the arguments that are provided to each of the plugin functions:

  • pkg (Package)

    The package for which the plugin command is getting executed against. This contains lots of really useful information about the package, such as it's dependency tree, src and build output paths etc. Please see the Package Schema for a full breakdown of what is available.

  • options (Object)

    Any options that were configured within the lerna-cola.js configuration file will be provided here.

    For example:

    // lerna-cola.js
    module.exports = {
      packages: {
        'my-lib': {
          buildPlugin: {
            name: '@lerna-cola/plugin-build-babel',
            // 👇 this stuff here
            options: {
              config: 'my-babel-config-package',
            },
          },
        },
      },
    }
  • args (Object)

    This will contain any other useful utils/data. Mostly plugin specific. Please read the docs for each plugin below.

Clean Plugin

Requires the following interface be satisfied:

module.exports = {
  name: 'my-plugin',
  clean: (pkg, options, args) => Promise.resolve('todo'),
}

Build Plugin

Requires the following interface be satisfied:

module.exports = {
  name: 'my-plugin',
  build: (pkg, options, args) => Promise.resolve('todo'),
}

Develop Plugin

Requires the following interface be satisfied:

module.exports = {
  name: 'my-plugin',
  develop: (pkg, options, args) => Promise.resolve('todo'),
}

A special note on the develop plugin; it gets executed via the development service and will be run multiple times. Specifically the plugin will be run under the following conditions:

  • When the development service first starts
  • Any time a change is detected within the src directory of the package it is configured against
  • Any time one of the packages within the monorepo that it depends on changes.

This allows you to know when to do restarting etc, and what type of behaviour you would like to implement for each scenario.

The args parameter is also well used by the develop plugin. It will contain the following items:

  • runType (string)

    It will contain one of the following values:

    • 'FIRST_RUN'

      Indicates that this is the first time the develop plugin is being executed during by the development service.

    • 'SELF_CHANGED'

      Indicates that a file within the src directory of the package it is configured against changed.

    • 'DEPENDENCY_CHANGED'

      Indicates that a package that it depends upon within the monorepo changed.

  • changedDependency

    This will contain the Package meta object describing the dependency that changed when the runType is "DEPENDENCY_CHANGED".

Deploy Plugin

Requires the following interface be satisfied:

module.exports = {
  name: 'my-plugin',
  develop: (pkg, options, args) => Promise.resolve('todo'),
}

Schemas

Below are schemas of the arguments that are provided to your plugins.

Package Schema

The holy grail of information for your plugins. The object contains the following:

  • name (string)

    The name of the package.

  • config (Object)

    The configuration that is being used (taken from the lerna-cola.js file with defaults applied) for it.

  • color (chalk package function)

    The chalk function representing the color we will use to uniquely identify console output for the package.

  • dependants (Array<string>)

    The names of the other packages within the monorepo that have a direct dependency on this package.

  • dependencies (Array<string>)

    The names of the other packages within the monorepo that this package has a direct dependency on.

  • allDependants (Array<string>)

    The names of the ALL other packages within the monorepo that directly, or indirectly (via dependency tree), depend on this package.

  • devDependencies (Array<string>)

    The names of the other packages within the monorepo that this package has a direct devDependency on.

  • packageJson (Object)

    The package.json for this package.

  • paths (Object)

    A set of paths for this package. Containing the following paths:

    • monoRepoRoot (string)

      The path to the root of the monorepo that this package belongs to.

    • monoRepoRootNodeModules (string)

      The path to the node_modules dir contained within the root of the monorepo that this package belongs to.

    • packageBuildOutput (string)

      The path to which build output should be emitted.

    • packageEntryFile (string)

      The source entry file for this package.

    • packageJson (string)

      The path to the package.json file for the package.

    • packageNodeModules (string)

      The path to the node_modules file for the package.

    • packageRoot (string)

      The path to the root dir of the package.

    • packageSrc (string)

      The path to the src dir of the package.

Config Schema

TODO