2020-09-14 15:07:02 -07:00
# Manifest Mode
vcpkg has two modes of operation - classic mode and manifest mode.
In classic mode, vcpkg produces an "installed" tree,
whose contents are changed by explicit calls to `vcpkg install` or `vcpkg remove` .
The installed tree is intended for consumption by any number of projects:
for example, installing a bunch of libraries and then using those libraries from Visual Studio,
without additional configuration.
Because the installed tree is not associated with an individual project,
it's similar to tools like `brew` or `apt` ,
except that the installed tree is vcpkg-installation-local,
rather than global to a system or user.
In manifest mode, an installed tree is associated with a particular project rather than the vcpkg installation.
The set of installed ports is controlled by editing the project's "manifest file",
and the installed tree is placed in the project directory or build directory.
2021-02-10 16:22:33 -08:00
This mode acts more similarly to language package managers like Cargo, or npm.
2020-09-14 15:07:02 -07:00
We recommend using this manifest mode whenever possible,
because it allows one to encode a project's dependencies explicitly in a project file,
rather than in the documentation, making your project much easier to consume.
Manifest mode is in beta, but one can use it from the CMake integration,
which will be stable when used via things like `find_package` .
This is the recommended way to use manifest mode.
In this document, we have basic information on [Writing a Manifest ](#writing-a-manifest ),
the [vcpkg Command Line Interface ](#command-line-interface ),
and a little more information on [CMake ](#cmake-integration ) integration.
2020-10-12 10:22:47 -07:00
Check out the [manifest cmake example ](../examples/manifest-mode-cmake.md ) for an example project using CMake and
manifest mode.
2020-09-14 15:07:02 -07:00
2021-02-09 14:03:41 -08:00
## Simple Example Manifest
```json
{
"$schema": "https://raw.githubusercontent.com/microsoft/vcpkg/master/scripts/vcpkg.schema.json",
"name": "my-application",
"version": "0.15.2",
"dependencies": [
"boost-system",
{
"name": "cpprestsdk",
"default-features": false
},
"libxml2",
"yajl"
]
}
```
2020-09-14 15:07:02 -07:00
## Writing a Manifest
2020-10-12 10:22:47 -07:00
A manifest is a JSON-formatted file named `vcpkg.json` which lies at the root of your project.
2020-09-14 15:07:02 -07:00
It contains all the information a person needs to know to get dependencies for your project,
as well as all the metadata about your project that a person who depends on you might be interested in.
2020-10-12 10:22:47 -07:00
Manifests follow strict JSON: they can't contain C++-style comments (`//` ) nor trailing commas. However
you can use field names that start with `$` to write your comments in any object that has a well-defined set of keys.
These comment fields are not allowed in any objects which permit user-defined keys (such as `"features"` ).
2020-09-14 15:07:02 -07:00
2020-10-12 10:22:47 -07:00
Each manifest contains a top level object with the fields documented below; the most important ones are
[`"name"` ](#name ), the [version fields ](#version-fields ), and [`"dependencies"` ](#dependencies ):
2020-09-14 15:07:02 -07:00
### `"name"`
This is the name of your project! It must be formatted in a way that vcpkg understands - in other words,
it must be lowercase alphabetic characters, digits, and hyphens, and it must not start nor end with a hyphen.
For example, `Boost.Asio` might be given the name `boost-asio` .
This is a required field.
### Version fields
There is, at this point, only one version field - `"version-string"` . However, more will be added in the future.
You must have one (and only one) version field. There are different reasons to use each version field:
* `"version-string"` - used for packages that don't have orderable versions. This is pretty uncommon,
but since we don't have any versioning constraints yet, this is the only one that you can use.
Additionally, the `"port-version"` field is used by registries of packages,
as a way to version "the package gotten from `vcpkg install` " differently from the upstream package version.
You shouldn't need to worry about this at all.
2021-01-15 12:35:48 -08:00
#### Additional version fields
**Experimental behind the `versions` feature flag**
See [versioning.md ](versioning.md#version%20schemes ) for additional version types.
2020-09-14 15:07:02 -07:00
### `"description"`
This is where you describe your project. Give it a good description to help in searching for it!
This can be a single string, or it can be an array of strings;
in the latter case, the first string is treated as a summary,
while the remaining strings are treated as the full description.
2021-01-15 12:35:48 -08:00
### `"builtin-baseline"`
**Experimental behind the `versions` feature flag**
This field indicates the commit of vcpkg which provides global minimum version information for your manifest. It is required for top-level manifest files using versioning.
See also [versioning ](versioning.md#builtin-baseline ) for more semantic details.
2020-09-14 15:07:02 -07:00
### `"dependencies"`
This field lists all the dependencies you'll need to build your library (as well as any your dependents might need,
if they were to use you). It's an array of strings and objects:
* A string dependency (e.g., `"dependencies": [ "zlib" ]` ) is the simplest way one can depend on a library;
it means you don't depend on a single version, and don't need to write down any more information.
* On the other hand, an object dependency (e.g., `"dependencies": [ { "name": "zlib" } ]` )
allows you to add that extra information.
2021-02-09 14:03:41 -08:00
#### Example:
```json
"dependencies": [
{
"name": "arrow",
"default-features": false,
"features": [ "json" ]
},
"boost-asio",
"openssl",
{
"name": "picosha2",
"platform": "!windows"
}
]
```
2020-09-14 15:07:02 -07:00
2021-02-09 14:03:41 -08:00
#### `"name"` Field
2020-09-14 15:07:02 -07:00
The name of the dependency. This follows the same restrictions as the [`"name"` ](#name ) property for a project.
2021-02-09 14:03:41 -08:00
#### `"features"` and `"default-features"` Fields
2020-09-14 15:07:02 -07:00
`"features"` is an array of feature names which tell you the set of features that the
dependencies need to have at a minimum,
while `"default-features"` is a boolean that tells vcpkg whether or not to
install the features the package author thinks should be "most common for most people to use".
For example, `ffmpeg` is a library which supports many, many audio and video codecs;
however, for your specific project, you may only need mp3 encoding.
Then, you might just ask for:
```json
{
"name": "ffmpeg",
"default-features": false,
"features": [ "mp3lame" ]
}
```
2021-02-09 14:03:41 -08:00
#### `"platform"` Field
2020-09-14 15:07:02 -07:00
The `"platform"` field defines the platforms where the dependency should be installed - for example,
you might need to use sha256, and so you use platform primitives on Windows, but `picosha2` on non-Windows platforms.
```json
{
"name": "picosha2",
"platform": "!windows"
}
```
This is a string field which takes boolean expressions of the form `<identifier>` ,
`!expression` , `expression { & expression & expression...}` , and `expression { | expression | expression...}` ,
along with parentheses to denote precedence.
For example, a dependency that's only installed on the Windows OS, for the ARM64 architecture,
and on Linux on x64, would be written `(windows & arm64) | (linux & x64)` .
The common identifiers are:
- The operating system: `windows` , `uwp` , `linux` , `osx` (includes macOS), `android` , `emscripten`
- The architecture: `x86` , `x64` , `wasm32` , `arm64` , `arm` (includes both arm32 and arm64 due to backwards compatibility)
although one can define their own.
2021-02-09 14:03:41 -08:00
#### `"version>="` Field
2021-01-15 12:35:48 -08:00
**Experimental behind the `versions` feature flag**
A minimum version constraint on the dependency.
This field specifies the minimum version of the dependency using a '#' suffix to denote port-version if non-zero.
See also [versioning ](versioning.md#constraints ) for more semantic details.
### `"overrides"`
**Experimental behind the `versions` feature flag**
This field enables version resolution to be ignored for certain dependencies and to use specific versions instead.
See also [versioning ](versioning.md#overrides ) for more semantic details.
#### Example:
```json
"overrides": [
{
"name": "arrow", "version": "1.2.3", "port-version": 7
}
]
```
2020-09-14 15:07:02 -07:00
### `"supports"`
If your project doesn't support common platforms, you can tell your users this with the `"supports"` field.
It uses the same platform expressions as [`"platform"` ](#platform ), from dependencies, as well as the
`"supports"` field of features.
For example, if your library doesn't support linux, you might write `{ "supports": "!linux" }` .
### `"features"` and `"default-features"`
The `"features"` field defines _your_ project's optional features, that others may either depend on or not.
It's an object, where the keys are the names of the features, and the values are objects describing the feature.
`"description"` is required,
and acts exactly like the [`"description"` ](#description ) field on the global package,
and `"dependencies"` are optional,
and again act exactly like the [`"dependencies"` ](#dependencies ) field on the global package.
There's also the `"supports"` field,
which again acts exactly like the [`"supports"` ](#supports ) field on the global package.
You also have control over which features are default, if a person doesn't ask for anything specific,
and that's the `"default-features"` field, which is an array of feature names.
#### Example:
2021-02-09 14:03:41 -08:00
```json
2020-09-14 15:07:02 -07:00
{
"name": "libdb",
2021-02-09 14:03:41 -08:00
"version": "1.0.0",
2020-09-14 15:07:02 -07:00
"description": [
"An example database library.",
2021-01-15 12:35:48 -08:00
"Optionally can build with CBOR, JSON, or CSV as backends."
2020-09-14 15:07:02 -07:00
],
2021-01-15 12:35:48 -08:00
"$default-features-explanation": "Users using this library transitively will get all backends automatically",
2020-09-14 15:07:02 -07:00
"default-features": [ "cbor", "csv", "json" ],
"features": {
"cbor": {
"description": "The CBOR backend",
"dependencies": [
{
"$explanation": [
2021-01-15 12:35:48 -08:00
"This is how you tell vcpkg that the cbor feature depends on the json feature of this package"
2020-09-14 15:07:02 -07:00
],
"name": "libdb",
"default-features": false,
"features": [ "json" ]
}
]
},
"csv": {
"description": "The CSV backend",
"dependencies": [
"fast-cpp-csv-parser"
]
},
"json": {
"description": "The JSON backend",
"dependencies": [
"jsoncons"
]
2021-02-09 14:03:41 -08:00
}
2020-09-14 15:07:02 -07:00
}
}
```
## Command Line Interface
The command line interface around the new manifest mode is pretty simple.
There's only one command that one really needs to worry about, `vcpkg install` ,
although `vcpkg search` is still useful.
Since manifest mode is still in beta, you'll need to pass a feature flag: `manifests` .
There are a few ways to pass this feature flag:
2020-09-17 13:10:02 -07:00
* `--feature-flags` option: On any vcpkg command, you can pass `--feature-flags=manifests`
2020-09-14 15:07:02 -07:00
* `VCPKG_FEATURE_FLAGS` environment variable: one can set the environment variable `VCPKG_FEATURE_FLAGS` to
`manifests` .
### `vcpkg install`
Once one has written a manifest file,
they can run `vcpkg install` in any subdirectory of the directory containing `vcpkg.json` .
It will install all of the dependencies for the default triplet into
`<directory containing vcpkg.json>/vcpkg_installed` .
If you want to switch the triplet (for example, this is very common on windows, where the default triplet is x86-windows, not x64-windows),
you can pass it with the `--triplet` option: `vcpkg install --triplet x64-windows` (or whatever).
Then, vcpkg will install all the dependencies, and you're ready to go!
## CMake Integration
The CMake integration acts exactly like the existing CMake integration.
One passes the toolchain file, located at `[vcpkg root]/scripts/buildsystems/vcpkg.cmake` ,
to the CMake invocation via the `CMAKE_TOOLCHAIN_FILE` variable.
Then, CMake will install all dependencies into the build directory, and you're good!
It ends up that you only have to run CMake, and vcpkg is called only as part of the build process.
Unlike bare vcpkg, the feature flag is not required,
since the CMake integration won't break as long as you depending on the exact naming of vcpkg's installed directory.
### Example:
```
> cmake -B builddir -S . -DCMAKE_TOOLCHAIN_FILE=[vcpkg root]/scripts/buildsystems/vcpkg.cmake
> cmake --build builddir
```
with a `vcpkg.json` in the same directory as `CMakeLists.txt` should Just Work!
2021-02-09 14:03:41 -08:00
## MSBuild Integration
To use manifests with MSBuild, first you need to use an [existing integration method ](integration.md#with-msbuild ). Then, simply add a vcpkg.json above your project file (such as in the root of your source repository) and set the property `VcpkgEnableManifest` to `true` . You can set this property via the IDE in `Project Properties -> Vcpkg -> Use Vcpkg Manifest` .
As part of your project's build, vcpkg automatically be run and install any listed dependencies to `vcpkg_installed/` adjacent to the `vcpkg.json` file; these files will then automatically be included in and linked to your MSBuild projects.
2021-02-09 15:35:06 -08:00
Note: It is critical that all project files in a single build consuming the same `vcpkg.json` use the same triplet; if you need to use different triplets for different projects in your solution, they must consume from different `vcpkg.json` files.
### MSBuild Properties
These properties can be defined via the VS GUI under `Project Properties -> Vcpkg` or via a common `.props` file.
#### `VcpkgEnabled` (Use Vcpkg)
This can be set to "false" to explicitly disable vcpkg integration for the project
#### `VcpkgTriplet` (Triplet)
This can be set to a custom triplet to use for integration (such as x64-windows-static)
#### `VcpkgAdditionalInstallOptions` (Additional Options)
When using a manifest, this option specifies additional command line flags to pass to the underlying vcpkg tool invocation. This can be used to access features that have not yet been exposed through another option.
#### `VcpkgConfiguration` (Vcpkg Configuration)
If your configuration names are too complex for vcpkg to guess correctly, you can assign this property to `Release` or `Debug` to explicitly tell vcpkg what variant of libraries you want to consume.
#### `VcpkgEnableManifest` (Use Vcpkg Manifest)
This property must be set to true in order to consume from a local vcpkg.json file. If set to false, any local vcpkg.json files will be ignored. This will default to true in the future.
#### `VcpkgManifestInstall` (Install Vcpkg Dependencies)
*(Requires `Use Vcpkg Manifest` set to `true` )*
This property can be set to "false" to disable automatic dependency restoration on project build. Dependencies can be manually restored via the vcpkg command line.
2021-02-10 16:22:33 -08:00
#### `VcpkgInstalledDirectory` (Installed Directory)
This property defines the location where headers and binaries are consumed from. In manifest mode, this directory is created and populated based on your manifest.