docker-develop v1.2.1
docker-develop
A simple CLI utility for developing with Docker locally and automatically rebuilding for changes.
Contents
Installation
The recommended usage is to install this package globally so that it can be used across any projects direct from the command line.
npm install -g docker-developTo facilitate use across larger teams, this can also be installed as a developer dependency for projects using a Node script layer.
npm install --save-dev docker-developNote that with the local installation, the CLI will need to be invoked via
npxor within a custom NPM script.
Prerequisites
The only prerequisites for docker-develop are Docker and Node. The CLI tool uses Docker CLI in the background, so please ensure that you have Docker appropriately installed and the Docker CLI available on your machine's PATH.
Usage
Calling the CLI
Invoking the CLI is as simple as follows. Within the root of your docker project:
docker-developNote that at this time, your Docker and watch configuration cannot be auto-discovered. A docker-develop configuration file is required.
CLI Arguments
The following additional arguments are supported in the docker-develop CLI:
| Flag | Description | Example | 
|---|---|---|
| --config,-c | The location of your docker-developconfiguration file. Defaults to./docker-develop.json. | docker-develop -c docker-develop-alternate.json | 
Configuration
Base Configuration
docker-develop is configured via a JSON schema that can be provided at launch. At this time, the schema must be saved to disk and loaded via file path and cannot be provided via CLI arguments. By default, this schema is expected to be named docker-develop.json and live at the root where you invoke the docker-develop CLI command. However, you may specify a path to the configuration using the --config or -c CLI argument as outlined in CLI Arguments.
The configuration schema is defined as below:
Note that the root of the schema is an array containing one or multiple
docker-developconfigurations.
[
    {
        "dockerConfig": {
            "imageName": {
                "type": "String",
                "description": "The tag (name) that you wish to use for your built Docker image. If desired, provide a tag with `:<tag>`. If not provided, Docker will automatically tag with 'latest'.",
                "required": true,
                "example": "sample-image"
            },
            "buildArgs": {
                "type": "Array<BuildArg>",
                "description": "List of arguments to provide to Docker during the `docker build` step.",
                "required": false,
                "default": [],
                "example": [ [ "--target", "dev" ] ],
                "customTypes": {
                    "BuildArg": {
                        "type": "Array<String>",
                        "description": "A single argument to provide to Docker during the `docker build` step. Each string in the array will be concatenated with a single space.",
                        "required": false,
                        "example": ["--target", "dev"]
                    }
                }
            },
            "envProviderConfig": {
                "aws": {
                    "profile": {
                        "type": "String",
                        "description": "The name of the AWS profile that the AWS Provider should automatically select for retrieving AWS credentials for the launched Docker container. If the `aws` object is provided but `profile` is not included, the AWS Provider will auto-discover profiles and prompt the user to select a profile. If the `aws` object is not provided, no AWS credentials will be provided to the launched Docker container.",
                        "required": false,
                        "example": "default"
                    }
                }
            },
            "runArgs": {
                "type": "Array<RunArg>",
                "description": "List of arguments to provide to Docker during the `docker run` step.",
                "required": false,
                "default": [],
                "example": [ [ "-p", "5000:80" ], [ "-e", "ENV_VAR=hello-world" ] ],
                "customTypes": {
                    "RunArg": {
                        "type": "Array<String>",
                        "description": "A single argument to provide to Docker during the `docker run` step. Each string in the array will be concatenated with a single space.",
                        "required": false,
                        "example": ["-p", "5000:80"]
                    }
                }
            }
        },
        "watchConfig": {
            "type": "Array<WatchConfiguration>",
            "description": "List of configurations describing directories to watch for changes as well as the filename patterns that should trigger the Docker application to restart. Required, but may be an empty array (`[]`) to specify no watched directories.",
            "required": true,
            "example": [
                {
                    "path": "./src",
                    "recursive": "true",
                    "regex": ".*"
                }
            ],
            "customTypes": {
                "WatchConfiguration": {
                    "path": {
                        "type": "String",
                        "description": "The path (absolute or relative when using `.`) to a directory to watch for changes.",
                        "required": true,
                        "example": "./src"
                    },
                    "recursive": {
                        "type": "Boolean",
                        "description": "Whether to watch the directory recursively into subdirectories (true) or to only watch the top level specified directory (false).",
                        "required": false,
                        "default": true,
                        "example": true
                    },
                    "regex": {
                        "type": "String",
                        "description": "The regular expression pattern to test against filenames when a change is detected. If the pattern is a match, relaunch the associated Docker application. If the pattern is not a match, ignore the change and do nothing.",
                        "required": false,
                        "default": ".*",
                        "example": ".*"
                    }
                }
            }
        }
    },
    // ...
]Example configuration:
[
    {
        "dockerConfig": {
            "imageName": "sample-application",
            "buildArgs": [
                ["--target", "dev"]
            ],
            "runArgs": [
                ["-p", "5000:80"],
                ["-p", "5001:5678"],
                ["-e", "EnvironmentVariable=hello-world"]
            ]
        },
        "envProviderConfig": {
            "aws": {}
        },
        "watchConfig": [
            {
                "path": "./src",
                "recursive": true,
                "regex": ".*"
            },
            {
                "path": ".",
                "recursive": false,
                "regex": "Dockerfile|.dockerignore|package.json"
            }
        ]
    }
]Local Configuration
In some scenarios, you may require secrets as part of your configuration that a provider does not currently support. In this case, local configurations (extensions) provide a mechanism for a local git ignored file to provide additional configuration to containers outlined in your base configuration.
The local extension configuration schema is defined as below:
Note that the root of the schema is an object containing one or multiple image names
{
    "<imageName>": {
        "dockerConfig": {
            "buildArgs": {
                "type": "Array<BuildArg>",
                "description": "List of arguments to provide to Docker during the `docker build` step.",
                "required": false,
                "default": [],
                "example": [ [ "--target", "dev" ] ],
                "customTypes": {
                    "BuildArg": {
                        "type": "Array<String>",
                        "description": "A single argument to provide to Docker during the `docker build` step. Each string in the array will be concatenated with a single space.",
                        "required": false,
                        "example": ["--target", "dev"]
                    }
                }
            },
            "runArgs": {
                "type": "Array<RunArg>",
                "description": "List of arguments to provide to Docker during the `docker run` step.",
                "required": false,
                "default": [],
                "example": [ [ "-p", "5000:80" ], [ "-e", "ENV_VAR=hello-world" ] ],
                "customTypes": {
                    "RunArg": {
                        "type": "Array<String>",
                        "description": "A single argument to provide to Docker during the `docker run` step. Each string in the array will be concatenated with a single space.",
                        "required": false,
                        "example": ["-p", "5000:80"]
                    }
                }
            }
        },
        "watchConfig": {
            "type": "Array<WatchConfiguration>",
            "description": "List of configurations describing directories to watch for changes as well as the filename patterns that should trigger the Docker application to restart. Required, but may be an empty array (`[]`) to specify no watched directories.",
            "required": true,
            "example": [
                {
                    "path": "./src",
                    "recursive": "true",
                    "regex": ".*"
                }
            ],
            "customTypes": {
                "WatchConfiguration": {
                    "path": {
                        "type": "String",
                        "description": "The path (absolute or relative when using `.`) to a directory to watch for changes.",
                        "required": true,
                        "example": "./src"
                    },
                    "recursive": {
                        "type": "Boolean",
                        "description": "Whether to watch the directory recursively into subdirectories (true) or to only watch the top level specified directory (false).",
                        "required": false,
                        "default": true,
                        "example": true
                    },
                    "regex": {
                        "type": "String",
                        "description": "The regular expression pattern to test against filenames when a change is detected. If the pattern is a match, relaunch the associated Docker application. If the pattern is not a match, ignore the change and do nothing.",
                        "required": false,
                        "default": ".*",
                        "example": ".*"
                    }
                }
            }
        }
    },
    //...
}Example configuration:
{
    "local-config-sample": {
        "dockerConfig": {
            "buildArgs": [
                ["--target", "dev"]
            ],
            "runArgs": [
                ["-p", "8080:80"]
            ]
        },
        "watchConfig": [
            {
                "path": "./src",
                "recursive": true,
                "regex": ".*"
            }
        ]
    }
}Examples
| Example | Description | Run Instructions | 
|---|---|---|
| Single Image Watch | Launching a single Hello World web application and watching the source HTML for changes. Available on localhost:8080. | From examples/singleImage, rundocker-develop | 
| Alternate Configuration | Launching a single Hello World web application with an alternate configuration and watching the source HTML for changes. Available on localhost:8081. | From examples/singleImage, rundocker-develop --config docker-develop-alternate.jsonordocker-develop -c docker-develop-alternate.json | 
| Multiple Image Watch | Launching two Hellow World web applications and watching the source HTML of each for changes. Each container rebuilds and relaunches only for changes to it's own source. Available on localhost:8080andlocalhost:8081. | From examples/multipleImages, rundocker-develop | 
| Local Configuration Extension | Launching a single Hello World web application and loading the run arguments from a local extension JSON file using default naming for configuration files. Available on localhost:8080. | From examples/localConfigExtension, rundocker-develop | 
| Local Configuration Extension | Launching a single Hello World web application and loading the run arguments from a local extension JSON file using custom naming for configuration files. Available on localhost:8081. | From examples/localConfigExtension, rundocker-develop --config docker-develop-alternate.jsonordocker-develop -c docker-develop-alternate.json | 
| Simple Provider | Launching a single Hello World web application but more importantly logging out simple environment variable during Docker build. | From examples/provider/simple, rundocker-develop | 
| AWS Select Profile Provider | Launching a single Hello World web application but more importantly prompting the user to select an AWS profile before launch and logging out the AWS Region (buildArg) during Docker build. | From examples/provider/aws, rundocker-develop | 
| AWS Default Provider | Launching a single Hello World web application but more importantly logging out the AWS Region (buildArg) of the default AWS profile during Docker build. | From examples/provider/aws, rundocker-develop --config docker-develop-alternate.json | 
Core Concepts
ConfigLoader
The DockerDevelopConfigLoader finds and then loads the docker-develop configuration file. This configuration is then used to determine which Docker applications should be launched as well as what configuration should be used to trigger relaunching each container. The config loader will search for docker-develop.json at the root of execution by default.
Additionally, the config loader will search for an optional local extension JSON configuration file using the naming schema <configFileName>.local.json. This configuration will map local extensions to any images outlined in your base configuration, but cannot provide additional images. By default, the config loader will search for docker-develop.local.json using the pattern from the default base configuration.
The primary purpose for the local extension configuration is to provide a mechanism for developer secrets that cannot be obtained via an existing provider. This local file is intended to be git ignored so that these secrets are not persisted in your source control.
Note: This extension process relies upon the
.jsonfile ending of your config file. If your config file does not end with.json, the config loader will still search for a file ending with.local.json. For example, a custom config file named.envwould still require a local config to be named.env.local.jsonfor the config loader to find it.
Please see the above section Configuration for more details on the structure of these configuration files.
Launcher
The DockerDevelopLauncher handles the building and launching of a given Docker image/container. A single launcher is created for each configuration found in the docker-develop configuration and maintains the lifecycle of that Docker application. The launcher utilizes the Docker CLI in order to facilitate these actions.
Build and run arguments are passed into the Docker CLI so that the Docker run configuration is fully configurable.
Additionally, restart requests are debounced to prevent flooding the application with stop and start commands. Please allow a couple of seconds for the restart process to initiate.
Watcher
The DockerDevelopWatcher handles watching the filesystem for changes according to the specified watch configuration. Upon a change that matches the provided regex pattern (or \.*\ if not provided), the watcher will restart the provided DockerLauncher.
Provider
To facilitate the collection of dynamic environment variables (e.g. AWS credentials, login information, etc.) - particularly those that you do not wish to include in source - Docker Develop introduces the concept of a Provider and the DockerDevelopBaseProvider class. A provider is referenced by name in the docker-develop.json configuration and uses it's custom configuration to asynchronously provide a dictionary of environment variables to be passed to Docker as both build-time buildArgs and runtime environment variables (-e).
Note that Providers will asynchronously obtain environment variable keys and values on first launch of the associated Docker container, but will cache and reuse these values for any container restarts. To force the Provider to refetch environment variables, you must kill the Docker Develop session and relaunch.
The DockerDevelopProviderManager reads in the envProviderConfig and creates/manages the lifecycle of all specified Providers. Each container has it's own Provider Manager.
Note that the Provider Manager will create a Provider so long as its configuration exists, even if that configuration is empty (i.e.
"envProviderConfig": { "aws": {} }will cause an AWS Provider to be created while"envProviderConfig": {}will cause no Provider to be created). In other words, the existence of an empty configuration object has meaning.
Supported Providers
Detailed provider configuration can be found above in the Configuration section. Below is a summary of supported providers and their root key.
| Name | Configuration Key | Description | 
|---|---|---|
| AWS Provider | aws | Loads AWS configuration (region, access key id, and secret access key) from host machine and makes these values available as AWS_REGION,AWS_ACCESS_KEY_ID, andAWS_SECRET_ACCESS_KEYrespectively. For applications using AWS services, these environment variables in your container will cause your application running locally to mimic an application running remotely with an IAM role that reflects the permissions of the provided credentials (likely your own user). |