Meltano v2.0 is almost here! See what's on the roadmap.
Meltano takes a modular approach to data engineering in general and EL(T) in particular, where your project and pipelines are composed of plugins of different types, most notably extractors (Singer taps), loaders (Singer targets), transformers (dbt and dbt models), and orchestrators (currently Airflow, with Dagster in development).
Your project’s plugins are defined in your meltano.yml project file,
and are installed inside the .meltano directory.
They can be managed using various CLI commands as well as the UI.
You can add a new plugin to your project using meltano add, or
by directly modifying your meltano.yml project file
and installing the new plugin using meltano install.
If you’d like to add a discoverable plugin that’s supported by Meltano out of the box, like one of the extractors and loaders listed on the Extractors and Loaders pages, refer to the “Discoverable plugins” section below.
Alternatively, if you’d like to add a custom plugin that Meltano isn’t familiar with yet, like an arbitrary Singer tap or target, refer to the “Custom plugins” section.
Finally, if you’d like your new plugin to inherit from an existing plugin in your project, so that it can reuse the same package but override (parts of) its configuration, refer to the “Plugin inheritance” section.
Discoverable plugins can be added to your project by simply providing
meltano add with their type and name:
meltano add <type> <name>
# For example:
meltano add extractor tap-gitlab
meltano add loader target-postgres
meltano add transformer dbt
meltano add orchestrator airflow
This will add a shadowing plugin definition to your meltano.yml project file under the plugins property, inside an array named after the plugin type:
plugins:
  extractors:
  - name: tap-gitlab
    variant: meltano
    pip_url: git+https://gitlab.com/meltano/tap-gitlab.git
  loaders:
  - name: target-postgres
    variant: datamill-co
    pip_url: singer-target-postgres
  transformer:
  - name: dbt
    pip_url: dbt
  orchestrators:
  - name: airflow
    pip_url: apache-airflow
If multiple variants of the discoverable plugin are available,
the variant property is automatically set to the name of the default variant
(which is known to work well and recommended for new users),
so that your project is pinned to a specific package and its base plugin description.
If the variant property were omitted from the definition, Meltano would fall back on the original supported variant instead, which does not necessarily match the default.
The package’s pip_url (its pip install argument)
is repeated here for convenience, since you may want to update it to
point at a (custom) fork or to pin a package to a specific version.
If this property is omitted, it is inherited from the discoverable base plugin description identified by the name (and variant) instead.
As mentioned above, directly adding a plugin to your meltano.yml project file
and installing it using meltano install <type> <name>
has the same effect as adding it using meltano add.
If multiple variants of a discoverable plugin are available,
you can choose a specific (non-default) variant using the --variant option on meltano add:
meltano add <type> <name> --variant <variant>
# For example:
meltano add loader target-postgres --variant=transferwise
As you might expect, this will be reflected in the variant and pip_url properties in your meltano.yml project file:
plugins:
  loaders:
  - name: target-postgres
    variant: transferwise
    pip_url: pipelinewise-target-postgres
If you’d like to use multiple variants of the same discoverable plugin in your project at the same time, refer to “Multiple variants” under “Explicit inheritance” below.
If you’ve already added one variant to your project and would like to switch to another, refer to the “Switching from one variant to another” section below.
In the examples we’ve considered so far, plugins in your project implicitly inherit their base plugin descriptions from discoverable plugins by reusing their names, which is known as shadowing.
Alternatively, if you’d like to give the plugin a more descriptive name in your project,
you can use the --inherit-from (or --as) option on meltano add
to explicitly inherit from the discoverable plugin instead:
meltano add <type> <name> --inherit-from <discoverable-name>
# Or equivalently:
meltano add <type> <discoverable-name> --as <name>
# For example:
meltano add extractor tap-postgres--billing --inherit-from tap-postgres
meltano add extractor tap-postgres --as tap-postgres--billing
The corresponding inheriting plugin definition in your meltano.yml project file will use inherit_from:
plugins:
  extractors:
  - name: tap-postgres--billing
    inherit_from: tap-postgres
    variant: transferwise
    pip_url: pipelinewise-tap-postgres
Note that the variant and pip_url properties were populated automatically by meltano add as described above.
If you’d like to use multiple variants of the same discoverable plugin in your project at the same time, this feature will also come in handy.
Since plugins in your project need to have unique names, a discoverable plugin can only be shadowed once, but it can be inherited from multiple times, with each plugin free to choose its own variant:
meltano add loader target-snowflake --variant=transferwise --as target-snowflake--transferwise
meltano add loader target-snowflake --variant=meltano --as target-snowflake--meltano
Assuming a regular (shadowing) target-snowflake was added before using meltano add loader target-snowflake,
the resulting inheriting plugin definitions in meltano.yml project file will look as follows:
plugins:
  loaders:
  - name: target-snowflake
    variant: datamill-co
    pip_url: target-snowflake
  - name: target-snowflake--transferwise
    inherit_from: target-snowflake
    variant: transferwise
    pip_url: pipelinewise-target-snowflake
  - name: target-snowflake--meltano
    inherit_from: target-snowflake
    variant: meltano
    pip_url: git+https://gitlab.com/meltano/target-snowflake.git
Note that the --variant option and variant property are crucial here:
since inherit_from can also be used to inherit from another plugin in the project,
inherit_from: target-snowflake by itself would have resulted in the new plugin inheriting from the existing target-snowflake plugin (that uses the datamill-co variant) instead of the discoverable plugin we’re looking for.
Had there been no target-snowflake plugin in the project yet,
inherit_from: target-snowflake would necessarily refer to the discoverable plugin,
but without a variant the original variant would have been used rather than the default or a specific chosen one,
just like when shadowing with a name but no variant.
Custom plugins for packages that aren’t discoverable yet,
like arbitrary Singer taps and targets,
can be added to your project using the --custom option on meltano add:
meltano add --custom <type> <name>
# For example:
meltano add --custom extractor tap-covid-19
meltano add --custom loader target-bigquery--custom
# If you're using Docker, don't forget to mount the project directory,
# and ensure that interactive mode is enabled so that Meltano can ask you
# additional questions about the plugin and get your answers over STDIN:
docker run --interactive -v $(pwd):/project -w /project meltano/meltano add --custom extractor tap-covid-19
Since Meltano doesn’t have the base plugin description for the package in question yet,
meltano add --custom will ask you to find and provide this metadata yourself:
(Note that more context is provided in the actual command prompts.)
$ meltano add --custom extractor tap-covid-19
Adding new custom extractor with name 'tap-covid-19'...
Specify the plugin's namespace, which will serve as the:
- identifier to find related/compatible plugins
- default database schema (`load_schema` extra),
  for use by loaders that support a target schema
Hit Return to accept the default: plugin name with underscores instead of dashes
(namespace) [tap_covid_19]: tap_covid_19
Specify the plugin's `pip install` argument, for example:
- PyPI package name:
	tap-covid-19
- Git repository URL:
	git+https://gitlab.com/meltano/tap-covid-19.git
- local directory, in editable/development mode:
	-e extract/tap-covid-19
- 'n' if using a local executable (nothing to install)
Default: plugin name as PyPI package name
(pip_url) [tap-covid-19]: -e extract/tap-covid-19
Specify the plugin's executable name
Default: name derived from `pip_url`
(executable) [tap-covid-19]: tap-covid-19
Specify the tap's supported Singer features (executable flags), for example:
	`catalog`: supports the `--catalog` flag
	`discover`: supports the `--discover` flag
	`properties`: supports the `--properties` flag
	`state`: supports the `--state` flag
To find out what features a tap supports, reference its documentation or try one
of the tricks under [how to test a tap](/contribute/plugins#how-to-test-a-tap).
Multiple capabilities can be separated using commas.
Default: no capabilities
(capabilities) [[]]: catalog,discover,state
Specify the tap's supported settings (`config.json` keys)
Multiple setting names (keys) can be separated using commas.
A setting kind can be specified alongside the name (key) by using the `:` delimiter,
e.g. `port:integer` to set the kind `integer` for the name `port`
Supported setting kinds:
string | integer | boolean | date_iso8601 | email | password | oauth | options | file | array | object | hidden
- Credentials and other sensitive setting types should use the password kind.
- If not specified, setting kind defaults to string.
- Nested properties can be represented using the `.` separator, e.g. `auth.username` for `{ "auth": { "username": value } }`.
- To find out what settings a tap supports, reference its documentation.
Default: no settings
(settings) [[]]: api_token:password,user_agent:string,start_date:date_iso8601
Added extractor 'tap-covid-19' to your Meltano project
Installing extractor 'tap-covid-19'...
Installed extractor 'tap-covid-19'
If you’re adding a Singer tap or target that’s listed on Singer’s index of taps or targets,
simply providing the package name as pip_url and executable usually suffices.
The plugin’s name also typically matches the name of the package, but you are free to change it to be more descriptive.
If it’s a tap or target you have developed or are developing yourself,
you’ll want to set pip_url to either a Git repository URL or local directory path.
If you add the -e flag ahead of the local path, the package will be installed in editable mode.
To find out what settings a tap or target supports, reference the README in the repository and/or documentation.
If the capabilities a tap supports (executable flags like --discover and --state) are not described there,
you can try one of these tricks or refer directly to the source code.
This will add a custom plugin definition to your meltano.yml project file under the plugins property, inside an array named after the plugin type:
plugins:
  extractors:
  - name: tap-covid-19
    namespace: tap_covid_19
    pip_url: tap-covid-19
    executable: tap-covid-19
    capabilities:
    - catalog
    - discover
    - state
    settings:
    - name: api_token
    - name: user_agent
    - name: start_date
The pip_url, executable, capabilities, and settings properties
constitute the plugin’s base plugin description:
everything Meltano needs to know in order to be able to use the package as a plugin.
Once you've got the plugin working in your project, please consider contributing its description to the discovery.yml manifest to make it discoverable and supported out of the box for new users!
To add a new plugin to your project that inherits from an existing plugin,
so that it can reuse the same package but override (parts of) its configuration,
you can use the --inherit-from option on meltano add:
meltano add <type> <name> --inherit-from <existing-name>
# For example:
meltano add extractor tap-ga--client-foo --inherit-from tap-google-analytics
meltano add extractor tap-ga--client-bar --inherit-from tap-google-analytics
meltano add extractor tap-ga--client-foo--project-baz --inherit-from tap-ga--client-foo
The corresponding inheriting plugin definitions in your meltano.yml project file will use inherit_from:
plugins:
  extractors:
  - name: tap-google-analytics
    variant: meltano
    pip_url: git+https://gitlab.com/meltano/tap-google-analytics.git
  - name: tap-ga--client-foo
    inherit_from: tap-google-analytics
  - name: tap-ga--client-bar
    inherit_from: tap-google-analytics
  - name: tap-ga--client-foo--project-baz
    inherit_from: tap-ga--client-foo
Note that the --inherit-from option and inherit_from property can also be used to
explicitly inherit from a discoverable plugin.
Whenever you add a new plugin to a Meltano project using meltano add,
it will be installed into your project’s .meltano directory automatically.
However, since this directory is included in your project’s .gitignore file
by default, you’ll need to explicitly run meltano install
before any other meltano commands whenever you clone or pull an existing Meltano project from version control,
to install (or update) all plugins specified in your meltano.yml project file.
To install a specific plugin in your project, use meltano install <type> <name>, e.g. meltano install extractor tap-gitlab. Subsequent calls to meltano install will upgrade a plugin to its latest version, if any. To completely uninstall and reinstall a plugin, use --clean.
When you add a plugin to your project, the plugin definition’s pip_url property
(the package’s pip install argument)
in your meltano.yml project file
typically points at a PyPI package or Git repository without specifying a specific version,
to ensure that you always get the latest and (presumably) greatest.
This makes sense when a plugin is installed for the first time, but can lead to unwelcome surprises down the line, as your pipeline may break when a new version of the package is released that introduces a bug or changes its behavior in a backward-incompatible or otherwise undesireable way.
To ensure that meltano install always installs the same version that was used when you originally got the pipeline working,
you can modify the plugin definition in your meltano.yml project file to include a version identifier in the pip_url.
The exact steps to determine the version and modify the pip_url will depend on whether you are installing a package from PyPI or a Git repository:
If the plugin’s pip_url is set to a package name, e.g. tap-shopify, pip install will look for a package by that name on PyPI.
To pin the latest version:
Determine the latest version of the package by browsing to https://pypi.org/project/<package>, e.g. https://pypi.org/project/tap-shopify.
At the time of writing, the latest version of tap-shopify is 1.2.6.
Add an ==<version> or ~=<version> version specifier to the pip_url:
 # Before:
 pip_url: tap-shopify
 # After:
 pip_url: tap-shopify==1.2.6 # Always install version 1.2.6
 # Alternatively:
 pip_url: tap-shopify~=1.2.6 # Install 1.2.6 or a newer version in the 1.2.x range
If the plugin’s pip_url is set to a git+http(s) URL, e.g. git+https://gitlab.com/meltano/tap-gitlab.git or git+https://github.com/adswerve/target-bigquery.git, pip install will look for a package in that repository.
To pin the latest version:
Determine the latest version of the package by browsing to the https:// part of the repository URL, e.g. https://gitlab.com/meltano/tap-gitlab or https://github.com/adswerve/target-bigquery, and finding the latest Git tag.
At the time of writing, the latest tag of https://gitlab.com/meltano/tap-gitlab is v0.9.11,
 and the latest tag of https://github.com/adswerve/target-bigquery isv0.10.2.
If no tags are available, you can also use the SHA of the latest commit, e.g.
 2657b89e8896face4ce320a03b8413bbc196cec9 or
 3df97b951b7eebdfa331a1ff570f1fe3487d632f.
Add an @<tag> or @<sha> Git ref specifier to the pip_url:
 # Before:
 pip_url: git+https://gitlab.com/meltano/tap-gitlab.git
 pip_url: git+https://github.com/adswerve/target-bigquery.git
 # After:
 pip_url: git+https://gitlab.com/meltano/tap-gitlab.git@v0.9.11
 pip_url: git+https://github.com/adswerve/target-bigquery.git@v0.10.2
 # Alternatively:
 pip_url: git+https://gitlab.com/meltano/tap-gitlab.git@2657b89e8896face4ce320a03b8413bbc196cec9
 pip_url: git+https://github.com/adswerve/target-bigquery.git@3df97b951b7eebdfa331a1ff570f1fe3487d632f
If you need to fetch packages from a custom Python Package Index (PyPi), you can set the PIP_INDEX_URL environment variable to your custom URL before running meltano install.
In a Dockerfile, this would look like:
ARG PIP_INDEX_URL=<your_custom_pypi_url>
RUN meltano install
You can remove a plugin from your Meltano project by using meltano remove. You have to specify a type of plugin to remove, but you can remove multiple plugins of that type.
meltano remove <type> <name>
meltano remove <type> <name> <name_two>
# For example:
meltano remove extractor tap-gitlab
meltano remove loader target-postgres target-csv
Since the plugins section of your meltano.yml project file determines the plugins that make up your project, you can still manually remove a plugin from your project by deleting its entry from this file. Traces of the plugin may remain in the other locations mentioned above.
If you’ve forked a plugin’s repository and made changes to it, you can update your Meltano project to use your custom fork instead of the canonical source:
Modify the plugin definition’s pip_url in the plugins section of your meltano.yml project file to point at your fork using a git+http(s) URL, with an optional branch or tag name:
 plugins:
   extractors:
   - name: tap-gitlab
     variant: meltano
     pip_url: git+https://gitlab.com/meltano/tap-gitlab.git
     # pip_url: git+https://gitlab.com/meltano/tap-gitlab.git@ref-name
If your plugin source is stored in a private repository, you have two options:
Continue to authenticate over HTTP(S), and store your credentials in a .netrc file in your home directory:
machine <hostname> # e.g. gitlab.com or github.com
login <username>
password <personal-access-token-or-password>
Authenticate using SSH instead, and specify a git+ssh URL:
pip_url: git+ssh://git@gitlab.com/meltano/tap-gitlab.git
Reinstall the plugin from the new pip_url using meltano install:
 meltano install <type> <name>
 # For example:
 meltano install extractor tap-gitlab
If your fork supports additional settings, you can set them as custom settings.
The default variant of a discoverable plugin is recommended for new users, but may not always be a perfect fit for your use case.
If you’ve already added one variant to your project and would like to use another instead, you can add the new variant as a separate plugin or switch your existing plugin over to the new variant:
Modify the plugin definition’s variant and pip_url properties in the plugins section of your meltano.yml project file:
 # Before:
 plugins:
   loaders:
   - name: target-postgres
     variant: datamill-co
     pip_url: singer-target-postgres
 # After:
 plugins:
   loaders:
   - name: target-postgres
     variant: meltano
     pip_url: git+https://github.com/meltano/target-postgres.git # Optional
If you don’t know the new variant’s pip_url (its pip install argument),
 you can remove this property entirely so that Meltano will fall back on the default.
Reinstall the plugin from the new pip_url using meltano install:
 meltano install <type> <name>
 # For example:
 meltano install loader target-postgres
View the current configuration using meltano config <name> list to see if it is still valid:
 meltano config <name> list
 # For example:
 meltano config target-postgres list
Because different variants often use different setting names,
 you will likely see some of the settings used by the old variant show up as
 custom settings,
 indicating that they are not supported by the new variant,
 while settings that the new variant expects show up with a None or default value.
Assuming at least one setting did not carry over correctly from the old variant to the new variant,
 modify the plugin’s configuration in your meltano.yml project file to use the new setting names:
 # Before:
 config:
   postgres_host: postgres.example.com
   postgres_port: 5432
   postgres_username: my_user
   postgres_database: my_database
 # After:
 config:
   host: postgres.example.com
   port: 5432
   user: my_user
   dbname: my_database
If any of the old settings are stored in places other than meltano.yml,
 like a sensitive setting that may be stored in your project’s .env file,
 you can unset the old setting and set the new one using meltano config:
 meltano config <name> unset <old_setting>
 meltano config <name> set <setting> <value>
 # For example:
 meltano config target-postgres unset postgres_password
 meltano config target-postgres set password my_password
Keep doing this until meltano config <name> list shows a valid configuration for the new variant,
 without any of the old variant’s settings remaining as custom settings.
While Meltano is optimized for usage through the meltano CLI
and direct changes to the meltano.yml project file,
basic plugin management functionality is also available in the UI.