Pkg

Introduction

Pkg is the standard package manager for Julia 1.0 and newer. Unlike traditional package managers, which install and manage a single global set of packages, Pkg is designed around “environments”: independent sets of packages that can be local to an individual project or shared and selected by name. The exact set of packages and versions in an environment is captured in a manifest file which can be checked into a project repository and tracked in version control, significantly improving reproducibility of projects. If you’ve ever tried to run code you haven’t used in a while only to find that you can’t get anything to work because you’ve updated or uninstalled some of the packages your project was using, you’ll understand the motivation for this approach. In Pkg, since each project maintains its own independent set of package versions, you’ll never have this problem again. Moreover, if you check out a project on a new system, you can simply materialize the environment described by its manifest file and immediately be up and running with a known-good set of dependencies.

Since environments are managed and updated independently from each other, “dependency hell” is significantly alleviated in Pkg. If you want to use the latest and greatest version of some package in a new project but you’re stuck on an older version in a different project, that’s no problem – since they have separate environments they can just use different versions, which are both installed at the same time in different locations on your system. The location of each package version is canonical, so when environments use the same versions of packages, they can share installations, avoiding unnecessary duplication of the package. Old package versions that are no longer used by any environments are periodically “garbage collected” by the package manager.

Pkg’s approach to local environments may be familiar to people who have used Python’s virtualenv or Ruby’s bundler. In Julia, instead of hacking the language’s code loading mechanisms to support environments, we have the benefit that Julia natively understands them. In addition, Julia environments are “stackable”: you can overlay one environment with another and thereby have access to additional packages outside of the primary environment. This makes it easy to work on a project, which provides the primary environment, while still having access to all your usual dev tools like profilers, debuggers, and so on, just by having an environment including these dev tools later in the load path.

Last but not least, Pkg is designed to support federated package registries. This means that it allows multiple registries managed by different parties to interact seamlessly. In particular, this includes private registries which can live behind corporate firewalls. You can install and update your own packages from a private registry with exactly the same tools and workflows that you use to install and manage official Julia packages. If you urgently need to apply a hotfix for a public package that’s critical to your company’s product, you can tag a private version of it in your company’s internal registry and get a fix to your developers and ops teams quickly and easily without having to wait for an upstream patch to be accepted and published. Once an official fix is published, however, you can just upgrade your dependencies and you'll be back on an official release again.

Glossary

Project: a source tree with a standard layout, including a src directory for the main body of Julia code, a test directory for testing the project, docs for documentation files, and optionally a deps directory for a build script and its outputs. A project will typically also have a project file and may optionally have a manifest file:

  • Project file: a file in the root directory of a project, named Project.toml (or JuliaProject.toml) describing metadata about the project, including its name, UUID (for packages), authors, license, and the names and UUIDs of packages and libraries that it depends on.

  • Manifest file: a file in the root directory of a project, named Manifest.toml (or JuliaManifest.toml) describing a complete dependency graph and exact versions of each package and library used by a project.

Package: a project which provides reusable functionality that can be used by other Julia projects via import X or using X. A package should have a project file with a uuid entry giving its package UUID. This UUID is used to identify the package in projects that depend on it.

Note

For legacy reasons it is possible to load a package without a project file or UUID from the REPL or the top-level of a script. It is not possible, however, to load a package without a project file or UUID from a project with them. Once you've loaded from a project file, everything needs a project file and UUID.

Application: a project which provides standalone functionality not intended to be reused by other Julia projects. For example a web application or a commmand-line utility, or simulation/analytics code accompanying a scientific paper. An application may have a UUID but does not need one. An application may also provide global configuration options for packages it depends on. Packages, on the other hand, may not provide global configuration since that could conflict with the configuration of the main application.

Note

Projects vs. Packages vs. Applications:

  • Project is an umbrella term: packages and applications are kinds of projects.
  • Packages should have UUIDs, applications can have a UUIDs but don't need them.
  • Applications can provide global configuration, whereas packages cannot.

Library (future work): a compiled binary dependency (not written in Julia) packaged to be used by a Julia project. These are currently typically built in- place by a deps/build.jl script in a project’s source tree, but in the future we plan to make libraries first-class entities directly installed and upgraded by the package manager.

Environment: the combination of the top-level name map provided by a project file combined with the dependency graph and map from packages to their entry points provided by a manifest file. For more detail see the manual section on code loading.

  • Explicit environment: an environment in the form of an explicit project file and an optional corresponding manifest file together in a directory. If the manifest file is absent then the implied dependency graph and location maps are empty.

  • Implicit environment: an environment provided as a directory (without a project file or manifest file) containing packages with entry points of the form X.jl, X.jl/src/X.jl or X/src/X.jl. The top-level name map is implied by these entry points. The dependency graph is implied by the existence of project files inside of these package directories, e.g. X.jl/Project.toml or X/Project.toml. The dependencies of the X package are the dependencies in the corresponding project file if there is one. The location map is implied by the entry points themselves.

Registry: a source tree with a standard layout recording metadata about a registered set of packages, the tagged versions of them which are available, and which versions of packages are compatible or incompatible with each other. A registry is indexed by package name and UUID, and has a directory for each registered package providing the following metadata about it:

  • name – e.g. DataFrames
  • UUID – e.g. a93c6f00-e57d-5684-b7b6-d8193f3e46c0
  • authors – e.g. Jane Q. Developer jane@example.com
  • license – e.g. MIT, BSD3, or GPLv2
  • repository – e.g. https://github.com/JuliaData/DataFrames.jl.git
  • description – a block of text summarizing the functionality of a package
  • keywords – e.g. data, tabular, analysis, statistics
  • versions – a list of all registered version tags/jane@example.com
    For each registered version of a package, the following information is provided:

  • its semantic version number – e.g. v1.2.3

  • its git tree SHA-1 hash – e.g. 7ffb18ea3245ef98e368b02b81e8a86543a11103
  • a map from names to UUIDs of dependencies
  • which versions of other packages it is compatible/incompatible with
    Dependencies and compatibility are stored in a compressed but human-readable format using ranges of package versions.

Depot: a directory on a system where various package-related resources live, including:

  • environments: shared named environments (e.g. v0.7, devtools)
  • clones: bare clones of package repositories
  • compiled: cached compiled package images (.ji files)
  • config: global configuration files (e.g. startup.jl)
  • dev: default directory for package development
  • logs: log files (e.g. manifest_usage.toml, repl_history.jl)
  • packages: installed package versions
  • registries: clones of registries (e.g. General)
    Load path: a stack of environments where package identities, their dependencies, and entry-points are searched for. The load path is controlled in Julia by the LOAD_PATH global variable which is populated at startup based on the value of the JULIA_LOAD_PATH environment variable. The first entry is your primary environment, often the current project, while later entries provide additional packages one may want to use from the REPL or top-level scripts.

Depot path: a stack of depot locations where the package manager, as well as Julia's code loading mechanisms, look for registries, installed packages, named environments, repo clones, cached compiled package images, and configuration files. The depot path is controlled by the Julia DEPOT_PATH global variable which is populated at startup based on the value of the JULIA_DEPOT_PATH environment variable. The first entry is the “user depot” and should be writable by and owned by the current user. The user depot is where: registries are cloned, new package versions are installed, named environments are created and updated, package repos are cloned, newly compiled package image files are saved, log files are written, development packages are checked out by default, and global configuration data is saved. Later entries in the depot path are treated as read-only and are appropriate for registries, packages, etc. installed and managed by system administrators.

Getting Started

The Pkg REPL-mode is entered from the Julia REPL using the key ].

  1. (v0.7) pkg>

The part inside the parenthesis of the prompt shows the name of the current project. Since we haven't created our own project yet, we are in the default project, located at ~/.julia/environments/v0.7 (or whatever version of Julia you happen to run).

To return to the julia> prompt, either press backspace when the input line is empty or press Ctrl+C. Help is available by calling pkg> help. If you are in an environment that does not have access to a REPL you can still use the REPL mode commands using the string macro pkg available after using Pkg. The command pkg"cmd" would be equivalent to executing cmd in the REPL mode.

The documentation here describes using Pkg from the REPL mode. Documentation of using the Pkg API (by calling Pkg. functions) is in progress of being written.

Adding packages

There are two ways of adding packages, either using the add command or the dev command. The most frequently used one is add and its usage is described first.

Adding registered packages

In the Pkg REPL packages can be added with the add command followed by the name of the package, for example:

  1. (v0.7) pkg> add Example
  2. Cloning default registries into /Users/kristoffer/.julia/registries
  3. Cloning registry General from "https://github.com/JuliaRegistries/General.git"
  4. Updating registry at `~/.julia/registries/General`
  5. Updating git-repo `https://github.com/JuliaRegistries/General.git`
  6. Resolving package versions...
  7. Updating `~/.julia/environments/v0.7/Project.toml`
  8. [7876af07] + Example v0.5.1
  9. Updating `~/.julia/environments/v0.7/Manifest.toml`
  10. [7876af07] + Example v0.5.1
  11. [8dfed614] + Test

Here we added the package Example to the current project. In this example, we are using a fresh Julia installation, and this is our first time adding a package using Pkg. By default, Pkg clones Julia's General registry, and uses this registry to look up packages requested for inclusion in the current environment. The status update shows a short form of the package UUID to the left, then the package name, and the version. Since standard libraries (e.g. Test) are shipped with Julia, they do not have a version. The project status contains the packages you have added yourself, in this case, Example:

  1. (v0.7) pkg> st
  2. Status `Project.toml`
  3. [7876af07] Example v0.5.1

The manifest status, in addition, includes the dependencies of explicitly added packages.

  1. (v0.7) pkg> st --manifest
  2. Status `Manifest.toml`
  3. [7876af07] Example v0.5.1
  4. [8dfed614] Test

It is possible to add multiple packages in one command as pkg> add A B C.

After a package is added to the project, it can be loaded in Julia:

  1. julia> using Example
  2. julia> Example.hello("User")
  3. "Hello, User"

A specific version can be installed by appending a version after a @ symbol, e.g. @v0.4, to the package name:

  1. (v0.7) pkg> add Example@0.4
  2. Resolving package versions...
  3. Updating `~/.julia/environments/v0.7/Project.toml`
  4. [7876af07] + Example v0.4.1
  5. Updating `~/.julia/environments/v0.7/Manifest.toml`
  6. [7876af07] + Example v0.4.1

If the master branch (or a certain commit SHA) of Example has a hotfix that has not yet included in a registered version, we can explicitly track a branch (or commit) by appending #branch (or #commit) to the package name:

  1. (v0.7) pkg> add Example#master
  2. Updating git-repo `https://github.com/JuliaLang/Example.jl.git`
  3. Resolving package versions...
  4. Updating `~/.julia/environments/v0.7/Project.toml`
  5. [7876af07] ~ Example v0.5.1 v0.5.1+ #master (https://github.com/JuliaLang/Example.jl.git)
  6. Updating `~/.julia/environments/v0.7/Manifest.toml`
  7. [7876af07] ~ Example v0.5.1 v0.5.1+ #master (https://github.com/JuliaLang/Example.jl.git)

The status output now shows that we are tracking the master branch of Example. When updating packages, we will pull updates from that branch.

To go back to tracking the registry version of Example, the command free is used:

  1. (v0.7) pkg> free Example
  2. Resolving package versions...
  3. Updating `~/.julia/environments/v0.7/Project.toml`
  4. [7876af07] ~ Example v0.5.1+ #master (https://github.com/JuliaLang/Example.jl.git) ⇒ v0.5.1
  5. Updating `~/.julia/environments/v0.7/Manifest.toml`
  6. [7876af07] ~ Example v0.5.1+ #master )https://github.com/JuliaLang/Example.jl.git) ⇒ v0.5.1

Adding unregistered packages

If a package is not in a registry, it can still be added by instead of the package name giving the URL to the repository to add.

  1. (v0.7) pkg> add https://github.com/fredrikekre/ImportMacros.jl
  2. Updating git-repo `https://github.com/fredrikekre/ImportMacros.jl`
  3. Resolving package versions...
  4. Downloaded MacroTools v0.4.1
  5. Updating `~/.julia/environments/v0.7/Project.toml`
  6. [e6797606] + ImportMacros v0.0.0 # (https://github.com/fredrikekre/ImportMacros.jl)
  7. Updating `~/.julia/environments/v0.7/Manifest.toml`
  8. [e6797606] + ImportMacros v0.0.0 # (https://github.com/fredrikekre/ImportMacros.jl)
  9. [1914dd2f] + MacroTools v0.4.1

The dependencies of the unregistered package (here MacroTools) got installed. For unregistered packages we could have given a branch (or commit SHA) to track using #, just like for registered packages.

Adding a local package

Instead of giving a URL of a git repo to add we could instead have given a local path to a git repo. This works similarly to adding a URL. The local repository will be tracked (at some branch) and updates from that local repo are pulled when packages are updated. Note that changes to files in the local package repository will not immediately be reflected when loading that package. The changes would have to be committed and the packages updated in order to pull in the changes.

Developing packages

By only using add your Manifest will always have a "reproducible state", in other words, as long as the repositories and registries used are still accessible it is possible to retrieve the exact state of all the dependencies in the project. This has the advantage that you can send your project (Project.toml and Manifest.toml) to someone else and they can "instantiate" that project in the same state as you had it locally. However, when you are developing a package, it is more convenient to load packages at their current state at some path. For this reason, the dev command exists.

Let's try to dev a registered package:

  1. (v0.7) pkg> dev Example
  2. Updating git-repo `https://github.com/JuliaLang/Example.jl.git`
  3. Resolving package versions...
  4. Updating `~/.julia/environments/v0.7/Project.toml`
  5. [7876af07] + Example v0.5.1+ [`~/.julia/dev/Example`]
  6. Updating `~/.julia/environments/v0.7/Manifest.toml`
  7. [7876af07] + Example v0.5.1+ [`~/.julia/dev/Example`]

The dev command fetches a full clone of the package to ~/.julia/dev/ (the path can be changed by setting the environment variable JULIA_PKG_DEVDIR). When importing Example julia will now import it from ~/.julia/dev/Example and whatever local changes have been made to the files in that path are consequently reflected in the code loaded. When we used add we said that we tracked the package repository, we here say that we track the path itself. Note that the package manager will never touch any of the files at a tracked path. It is therefore up to you to pull updates, change branches etc. If we try to dev a package at some branch that already exists at ~/.julia/dev/ the package manager we will simply use the existing path. For example:

  1. (v0.7) pkg> dev Example
  2. Updating git-repo `https://github.com/JuliaLang/Example.jl.git`
  3. [ Info: Path `/Users/kristoffer/.julia/dev/Example` exists and looks like the correct package, using existing path instead of cloning

Note the info message saying that it is using the existing path. As a general rule, the package manager will never touch files that are tracking a path.

If dev is used on a local path, that path to that package is recorded and used when loading that package. The path will be recorded relative to the project file, unless it is given as an absolute path.

To stop tracking a path and use the registered version again, use free

  1. (v0.7) pkg> free Example
  2. Resolving package versions...
  3. Updating `~/.julia/environments/v0.7/Project.toml`
  4. [7876af07] Example v0.5.1+ [`~/.julia/dev/Example`] v0.5.1
  5. Updating `~/.julia/environments/v0.7/Manifest.toml`
  6. [7876af07] Example v0.5.1+ [`~/.julia/dev/Example`] v0.5.1

It should be pointed out that by using dev your project is now inherently stateful. Its state depends on the current content of the files at the path and the manifest cannot be "instantiated" by someone else without knowing the exact content of all the packages that are tracking a path.

Note that if you add a dependency to a package that tracks a local path, the Manifest (which contains the whole dependency graph) will become out of sync with the actual dependency graph. This means that the package will not be able to load that dependency since it is not recorded in the Manifest. To update sync the Manifest, use the REPL command resolve.

Removing packages

Packages can be removed from the current project by using pkg> rm Package. This will only remove packages that exist in the project, to remove a package that only exists as a dependency use pkg> rm —manifest DepPackage. Note that this will remove all packages that depends on DepPackage.

Updating packages

When new versions of packages the project is using are released, it is a good idea to update. Simply calling up will try to update all the dependencies of the project to the latest compatible version. Sometimes this is not what you want. You can specify a subset of the dependencies to upgrade by giving them as arguments to up, e.g:

  1. (v0.7) pkg> up Example

The version of all other packages direct dependencies will stay the same. If you only want to update the minor version of packages, to reduce the risk that your project breaks, you can give the —minor flag, e.g:

  1. (v0.7) pkg> up --minor Example

Packages that track a repository are not updated when a minor upgrade is done. Packages that track a path are never touched by the package manager.

Pinning a package

A pinned package will never be updated. A package can be pinned using pin as for example

  1. (v0.7) pkg> pin Example
  2. Resolving package versions...
  3. Updating `~/.julia/environments/v0.7/Project.toml`
  4. [7876af07] ~ Example v0.5.1 v0.5.1
  5. Updating `~/.julia/environments/v0.7/Manifest.toml`
  6. [7876af07] ~ Example v0.5.1 v0.5.1

Note the pin symbol showing that the package is pinned. Removing the pin is done using free

  1. (v0.7) pkg> free Example
  2. Updating `~/.julia/environments/v0.7/Project.toml`
  3. [7876af07] ~ Example v0.5.1 v0.5.1
  4. Updating `~/.julia/environments/v0.7/Manifest.toml`
  5. [7876af07] ~ Example v0.5.1 v0.5.1

Testing packages

The tests for a package can be run using testcommand:

  1. (v0.7) pkg> test Example
  2. Testing Example
  3. Testing Example tests passed

Building packages

The build step of a package is automatically run when a package is first installed. The output of the build process is directed to a file. To explicitly run the build step for a package the build command is used:

  1. (v0.7) pkg> build MbedTLS
  2. Building MbedTLS `~/.julia/packages/MbedTLS/h1Vu/deps/build.log`
  3. shell> cat ~/.julia/packages/MbedTLS/h1Vu/deps/build.log
  4. Warning: `wait(t::Task)` is deprecated, use `fetch(t)` instead.
  5. caller = macro expansion at OutputCollector.jl:63 [inlined]
  6. @ Core OutputCollector.jl:63
  7. ...
  8. [ Info: using prebuilt binaries

Creating your own projects

So far we have added packages to the default project at ~/.julia/environments/v0.7, it is, however, easy to create other, independent, projects. It should be pointed out if two projects uses the same package at the same version, the content of this package is not duplicated. In order to create a new project, create a directory for it and then activate that directory to make it the "active project" which package operations manipulate:

  1. shell> mkdir MyProject
  2. shell> cd MyProject
  3. /Users/kristoffer/MyProject
  4. (v0.7) pkg> activate .
  5. (MyProject) pkg> st
  6. Status `Project.toml`

Note that the REPL prompt changed when the new project is activated. Since this is a newly created project, the status command show it contains no packages, and in fact, it has no project or manifest file until we add a package to it:

  1. shell> ls -l
  2. total 0
  3. (MyProject) pkg> add Example
  4. Updating registry at `~/.julia/registries/General`
  5. Updating git-repo `https://github.com/JuliaRegistries/General.git`
  6. Resolving package versions...
  7. Updating `Project.toml`
  8. [7876af07] + Example v0.5.1
  9. Updating `Manifest.toml`
  10. [7876af07] + Example v0.5.1
  11. [8dfed614] + Test
  12. shell> ls -l
  13. total 8
  14. -rw-r--r-- 1 stefan staff 207 Jul 3 16:35 Manifest.toml
  15. -rw-r--r-- 1 stefan staff 56 Jul 3 16:35 Project.toml
  16. shell> cat Project.toml
  17. [deps]
  18. Example = "7876af07-990d-54b4-ab0e-23690620f79a"
  19. shell> cat Manifest.toml
  20. [[Example]]
  21. deps = ["Test"]
  22. git-tree-sha1 = "8eb7b4d4ca487caade9ba3e85932e28ce6d6e1f8"
  23. uuid = "7876af07-990d-54b4-ab0e-23690620f79a"
  24. version = "0.5.1"
  25. [[Test]]
  26. uuid = "8dfed614-e22c-5e08-85e1-65c5234f0b40"

This new environment is completely separate from the one we used earlier.

Garbage collecting old, unused packages

As packages are updated and projects are deleted, installed packages that were once used will inevitably become old and not used from any existing project. Pkg keeps a log of all projects used so it can go through the log and see exactly which projects still exist and what packages those projects used. The rest can be deleted. This is done with the gc command:

  1. (v0.7) pkg> gc
  2. Active manifests at:
  3. `/Users/kristoffer/BinaryProvider/Manifest.toml`
  4. ...
  5. `/Users/kristoffer/Compat.jl/Manifest.toml`
  6. Deleted /Users/kristoffer/.julia/packages/BenchmarkTools/1cAj: 146.302 KiB
  7. Deleted /Users/kristoffer/.julia/packages/Cassette/BXVB: 795.557 KiB
  8. ...
  9. Deleted /Users/kristoffer/.julia/packages/WeakRefStrings/YrK6: 27.328 KiB
  10. Deleted 36 package installations: 113.205 MiB

Note that only packages in ~/.julia/packages are deleted.

Creating your own packages

A package is a project with a name, uuid and version entry in the Project.toml file src/PackageName.jl file that defines the module PackageName. This file is executed when the package is loaded.

Generating files for a package

To generate files for a new package, use pkg> generate.

  1. (v0.7) pkg> generate HelloWorld

This creates a new project HelloWorld with the following files (visualized with the external tree command):

  1. shell> cd HelloWorld
  2. shell> tree .
  3. .
  4. ├── Project.toml
  5. └── src
  6. └── HelloWorld.jl
  7. 1 directory, 2 files

The Project.toml file contains the name of the package, its unique UUID, its version, the author and eventual dependencies:

  1. name = "HelloWorld"
  2. uuid = "b4cd1eb8-1e24-11e8-3319-93036a3eb9f3"
  3. version = "0.1.0"
  4. author = ["Some One <someone@email.com>"]
  5. [deps]

The content of src/HelloWorld.jl is:

  1. module HelloWorld
  2. greet() = print("Hello World!")
  3. end # module

We can now activate the project and load the package:

  1. pkg> activate .
  2. julia> import HelloWorld
  3. julia> HelloWorld.greet()
  4. Hello World!

Adding dependencies to the project

Let’s say we want to use the standard library package Random and the registered package JSON in our project. We simply add these packages (note how the prompt now shows the name of the newly generated project, since we are inside the HelloWorld project directory):

  1. (HelloWorld) pkg> add Random JSON
  2. Resolving package versions...
  3. Updating "~/Documents/HelloWorld/Project.toml"
  4. [682c06a0] + JSON v0.17.1
  5. [9a3f8284] + Random
  6. Updating "~/Documents/HelloWorld/Manifest.toml"
  7. [34da2185] + Compat v0.57.0
  8. [682c06a0] + JSON v0.17.1
  9. [4d1e1d77] + Nullables v0.0.4
  10. ...

Both Random and JSON got added to the project’s Project.toml file, and the resulting dependencies got added to the Manifest.toml file. The resolver has installed each package with the highest possible version, while still respecting the compatibility that each package enforce on its dependencies.

We can now use both Random and JSON in our project. Changing src/HelloWorld.jl to

  1. module HelloWorld
  2. import Random
  3. import JSON
  4. greet() = print("Hello World!")
  5. greet_alien() = print("Hello ", Random.randstring(8))
  6. end # module

and reloading the package, the new greet_alien function that uses Random can be used:

  1. julia> HelloWorld.greet_alien()
  2. Hello aT157rHV

Adding a build step to the package.

The build step is executed the first time a package is installed or when explicitly invoked with build. A package is built by executing the file deps/build.jl.

  1. shell> cat deps/build.log
  2. I am being built...
  3. (HelloWorld) pkg> build
  4. Building HelloWorld `deps/build.log`
  5. Resolving package versions...
  6. shell> cat deps/build.log
  7. I am being built...

If the build step fails, the output of the build step is printed to the console

  1. shell> cat deps/build.jl
  2. error("Ooops")
  3. (HelloWorld) pkg> build
  4. Building HelloWorld `deps/build.log`
  5. Resolving package versions...
  6. Error: Error building `HelloWorld`:
  7. ERROR: LoadError: Ooops
  8. Stacktrace:
  9. [1] error(::String) at ./error.jl:33
  10. [2] top-level scope at none:0
  11. [3] include at ./boot.jl:317 [inlined]
  12. [4] include_relative(::Module, ::String) at ./loading.jl:1071
  13. [5] include(::Module, ::String) at ./sysimg.jl:29
  14. [6] include(::String) at ./client.jl:393
  15. [7] top-level scope at none:0
  16. in expression starting at /Users/kristoffer/.julia/dev/Pkg/HelloWorld/deps/build.jl:1
  17. @ Pkg.Operations Operations.jl:938

Adding tests to the package

When a package is tested the file test/runtests.jl is executed.

  1. shell> cat test/runtests.jl
  2. println("Testing...")
  3. (HelloWorld) pkg> test
  4. Testing HelloWorld
  5. Resolving package versions...
  6. Testing...
  7. Testing HelloWorld tests passed

Test-specific dependencies

Sometimes one might want to use some packages only at testing time but not enforce a dependency on them when the package is used. This is possible by adding dependencies to [extras] and a test target in [targets] to the Project file. Here we add the Test standard library as a test-only dependency by adding the following to the Project file:

  1. [extras]
  2. Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40"
  3. [targets]
  4. test = ["Test"]

We can now use Test in the test script and we can see that it gets installed on testing:

  1. shell> cat test/runtests.jl
  2. using Test
  3. @test 1 == 1
  4. (HelloWorld) pkg> test
  5. Testing HelloWorld
  6. Resolving package versions...
  7. Updating `/var/folders/64/76tk_g152sg6c6t0b4nkn1vw0000gn/T/tmpPzUPPw/Project.toml`
  8. [d8327f2a] + HelloWorld v0.1.0 [`~/.julia/dev/Pkg/HelloWorld`]
  9. [8dfed614] + Test
  10. Updating `/var/folders/64/76tk_g152sg6c6t0b4nkn1vw0000gn/T/tmpPzUPPw/Manifest.toml`
  11. [d8327f2a] + HelloWorld v0.1.0 [`~/.julia/dev/Pkg/HelloWorld`]
  12. Testing HelloWorld tests passed```

Compatibility

Compatibility refers to the ability to restrict what version of the dependencies that your project is compatible with. If the compatibility for a dependency is not given, the project is assumed to be compatible with all versions of that dependency.

Compatibility for a dependency is entered in the Project.toml file as for example:

  1. [compat]
  2. Example = "0.4.3"

After a compatibility entry is put into the project file, up can be used to apply it.

The format of the version specifier is described in detail below.

Info

There is currently no way to give compatibility from the Pkg REPL mode so for now, one has to manually edit the project file.

Version specifier format

Similar to other package managers, the Julia package manager respects semantic versioning (semver). As an example, a version specifier is given as e.g. 1.2.3 is therefore assumed to be compatible with the versions [1.2.3 - 2.0.0) where ) is a non-inclusive upper bound. More specifically, a version specifier is either given as a caret specifier, e.g. ^1.2.3 or a tilde specifier ~1.2.3. Caret specifiers are the default and hence 1.2.3 == ^1.2.3. The difference between a caret and tilde is described in the next section. The intersection of multiple version specifiers can be formed by comma separating indiviual version specifiers.

Caret specifiers

A caret specifier allows upgrade that would be compatible according to semver. An updated dependency is considered compatible if the new version does not modify the left-most non zero digit in the version specifier.

Some examples are shown below.

  1. ^1.2.3 = [1.2.3, 2.0.0)
  2. ^1.2 = [1.2.0, 2.0.0)
  3. ^1 = [1.0.0, 2.0.0)
  4. ^0.2.3 = [0.2.3, 0.3.0)
  5. ^0.0.3 = [0.0.3, 0.0.4)
  6. ^0.0 = [0.0.0, 0.1.0)
  7. ^0 = [0.0.0, 1.0.0)

While the semver specification says that all versions with a major version of 0 are incompatible with each other, we have made that choice that a version given as 0.a.b is considered compatible with 0.a.c if a != 0 and c >= b.

Tilde specifiers

A tilde specifier provides more limited upgrade possibilities. With a tilde, only the last specified digit is allowed to increment by one. This gives the following example.

  1. ~1.2.3 = [1.2.3, 1.2.4)
  2. ~1.2 = [1.2.0, 1.3.0)
  3. ~1 = [1.0.0, 2.0.0)

Inequality specifiers

Inequalities can also be used to specify version ranges:

  1. >= 1.2.3 = [1.2.3, ∞)
  2. 1.2.3 = [1.2.3, ∞)
  3. = 1.2.3 = [1.2.3, 1.2.3]
  4. < 1.2.3 = [0.0.0, 1.2.2]

Precompiling a project

The REPL command precompile can be used to precompile all the dependencies in the project. You can for example do

  1. (HelloWorld) pkg> update; precompile

to update the dependencies and then precompile them.

Preview mode

If you just want to see the effects of running a command, but not change your state you can preview a command. For example:

  1. (HelloWorld) pkg> preview add Plots

or

  1. (HelloWorld) pkg> preview up

will show you the effects of adding Plots, or doing a full upgrade, respectively, would have on your project. However, nothing would be installed and your Project.toml and Manifest.toml are untouched.

Using someone else's project

Simply clone their project using e.g. git clone, cd to the project directory and call

  1. (v0.7) pkg> activate .
  2. (SomeProject) pkg> instantiate

If the project contains a manifest, this will install the packages in the same state that is given by that manifest. Otherwise, it will resolve the latest versions of the dependencies compatible with the project.

References

This section describes the "API mode" of interacting with Pkg.jl which is recommended for non-interactive usage, in i.e. scripts. In the REPL mode packages (with associated version, UUID, URL etc) are parsed from strings, for example, "Package#master","Package@v0.1", "www.mypkg.com/MyPkg#my/feature". It is possible to use strings as arguments for simple commands in the API mode (like Pkg.add(["PackageA", "PackageB"]), more complicated commands, that e.g. specify URLs or version range, uses a more structured format over strings. This is done by creating an instance of a PackageSpec which are passed in to functions.

Pkg.PackageSpec — Type.

  1. PackageSpec(name::String, [uuid::UUID, version::VersionNumber])
  2. PackageSpec(; name, url, path, rev, version, mode, level)

A PackageSpec is a representation of a package with various metadata. This includes:

  • The name of the package.
  • The package unique uuid.
  • A version (for example when adding a package. When upgrading, can also be an instance of
    the enum UpgradeLevel

  • A url and an optional git revision. rev could be a branch name or a git commit SHA.

  • A local path path. This is equivalent to using the url argument but can be more descriptive.
  • A mode, which is an instance of the enum PackageMode which can be either PKGMODE_PROJECT or
    PKGMODE_MANIFEST, defaults to PKGMODE_PROJECT. Used in e.g. Pkg.rm.

Most functions in Pkg take a Vector of PackageSpec and do the operation on all the packages in the vector.

Below is a comparison between the REPL version and the PackageSpec version:

REPL API
Package PackageSpec("Package")
Package@0.2 PackageSpec(name="Package", version="0.2")
Package=a67d… PackageSpec(name="Package", uuid="a67d…"
Package#master PackageSpec(name="Package", rev="master")
local/path#feature PackageSpec(path="local/path"; rev="feature)
www.mypkg.com PackageSpec(url="www.mypkg.com")
—manifest Package PackageSpec(name="Package", mode=PKGSPEC_MANIFEST)
—major Package PackageSpec(name="Package", version=PKGLEVEL_MAJOR)

Pkg.PackageMode — Type.

  1. PackageMode

An enum with the instances

  • PKGMODE_MANIFEST
  • PKGMODE_PROJECT
    Determines if operations should be made on a project or manifest level. Used as an argument to PackageSpec or as an argument to Pkg.rm.

Pkg.UpgradeLevel — Type.

  1. UpgradeLevel

An enum with the instances

  • UPLEVEL_FIXED
  • UPLEVEL_PATCH
  • UPLEVEL_MINOR
  • UPLEVEL_MAJOR
    Determines how much a package is allowed to be updated. Used as an argument to PackageSpec or as an argument to Pkg.update.

Pkg.add — Function.

  1. Pkg.add(pkg::Union{String, Vector{String})
  2. Pkg.add(pkg::Union{PackageSpec, Vector{PackageSpec}})

Add a package to the current project. This package will be available using the import and using keywords in the Julia REPL and if the current project is a package, also inside that package.

Examples

  1. Pkg.add("Example") # Add a package from registry
  2. Pkg.add(PackageSpec(name="Example", version="0.3")) # Specify version
  3. Pkg.add(PackageSpec(url="https://github.com/JuliaLang/Example.jl", rev="master")) # From url
  4. Pkg.add(PackageSpec(url="/remote/mycompany/juliapackages/OurPackage"))` # From path (has to be a gitrepo)

See also PackageSpec.

Pkg.develop — Function.

  1. Pkg.develop(pkg::Union{String, Vector{String})
  2. Pkg.develop(pkgs::Union{Packagespec, Vector{Packagespec}})

Make a package available for development by tracking it by path. If pkg is given with only a name or by a URL the packages will be downloaded to the location by the environment variable JULIA_PKG_DEVDIR with .julia/dev as the default.

If pkg is given as a local path, the package at that path will be tracked.

Examples

  1. # By name
  2. Pkg.develop("Example")
  3. # By url
  4. Pkg.develop(PackageSpec(url="https://github.com/JuliaLang/Compat.jl", rev="master"))
  5. # By path (also uses url keyword to PackageSpec)
  6. Pkg.develop(PackageSpec(url="MyJuliaPackages/Package.jl")

See also PackageSpec

Pkg.activate — Function.

  1. Pkg.activate([s::String]; shared::Bool=false)

Activate the environment at s. The active environment is the environment that is modified by executing package commands. The logic for what path is activated is as follows:

  • If shared is true, the first existing environment named s from the depots in the depot stack will be activated. If no such environment exists yet, activate it in the first depot.
  • If s is a path that exist, that environment will be activated.
  • If s is a package name in the current project activate that is tracking a path, activate the environment at that path.
  • If s is a non-existing path, activate that path.
    If no argument is given to activate, activate the home project, which is the one specified by either —project command line when starting julia, or JULIA_PROJECT environment variable.

Examples

  1. Pkg.activate()
  2. Pkg.activate("local/path")
  3. Pkg.activate("MyDependency")

Pkg.rm — Function.

  1. Pkg.rm(pkg::Union{String, Vector{String})
  2. Pkg.rm(pkg::Union{PackageSpec, Vector{PackageSpec}})

Remove a package from the current project. If the mode of pkg is PKGMODE_MANIFEST also remove it from the manifest including all recursive dependencies of pkg.

See also PackageSpec, PackageMode.

Pkg.update — Function.

  1. Pkg.update(; level::UpgradeLevel=UPLEVEL_MAJOR, mode::PackageMode = PKGMODE_PROJECT)
  2. Pkg.update(pkg::Union{String, Vector{String})
  3. Pkg.update(pkg::Union{PackageSpec, Vector{PackageSpec}})

Update a package pkg. If no posistional argument is given, update all packages in the manifest if mode is PKGMODE_MANIFEST and packages in both manifest and project if mode is PKGMODE_PROJECT. If no positional argument is given level can be used to control what how much packages are allowed to be upgraded (major, minor, patch, fixed).

See also PackageSpec, PackageMode, UpgradeLevel.

Pkg.test — Function.

  1. Pkg.test(; coverage::Bool=true)
  2. Pkg.test(pkg::Union{String, Vector{String}; coverage::Bool=true)
  3. Pkg.test(pkgs::Union{PackageSpec, Vector{PackageSpec}}; coverage::Bool=true)

Run the tests for package pkg or if no positional argument is given to test, the current project is tested (which thus needs to be a package). A package is tested by running its test/runtests.jl file.

The tests are run by generating a temporary environment with only pkg and its (recursive) dependencies (recursively) in it. If a manifest exist, the versions in that manifest is used, otherwise a feasible set of package are resolved and installed.

During the test, test-specific dependencies are active, which are given in the project file as e.g.

  1. [extras]
  2. Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40"
  3. [targets]
  4. test = [Test]

Coverage statistics for the packages may be generated by passing coverage=true. The default behavior is not to run coverage.

The tests are executed in a new process with check-bounds=yes and by default startup-file=no. If using the startup file (~/.julia/config/startup.jl) is desired, start julia with —startup-file=yes.

Pkg.build — Function.

  1. Pkg.build()
  2. Pkg.build(pkg::Union{String, Vector{String})
  3. Pkg.build(pkgs::Union{PackageSpec, Vector{PackageSpec}})

Run the build script in deps/build.jl for pkg and all of the dependencies in depth-first recursive order. If no argument is given to build, the current project is built, which thus needs to be a package. This function is called automatically one any package that gets installed for the first time.

Pkg.pin — Function.

  1. Pkg.pin(pkg::Union{String, Vector{String})
  2. Pkg.pin(pkgs::Union{Packagespec, Vector{Packagespec}})

Pin a package to the current version (or the one given in the packagespec or a certain git revision. A pinned package is never updated.

Pkg.free — Function.

  1. Pkg.free(pkg::Union{String, Vector{String})
  2. Pkg.free(pkgs::Union{Packagespec, Vector{Packagespec}})

Free a package which removes a pin if it exists, or if the package is tracking a path, e.g. after Pkg.develop, go back to tracking registered versions.

Examples

  1. Pkg.free("Package")
  2. Pkg.free(PackageSpec("Package"))

Pkg.instantiate — Function.

  1. Pkg.instantiate()

If a Manifest.toml file exist in the current project, download all the packages declared in that manifest. Else, resolve a set of feasible packages from the Project.toml files and install them.

Pkg.resolve — Function.

  1. Pkg.resolve()

Update the current manifest with eventual changes to the dependency graph from packages that are tracking a path.

Pkg.setprotocol! — Function.

  1. Pkg.setprotocol!(proto::Union{Nothing, AbstractString}=nothing)

Set the protocol used to access GitHub-hosted packages when adding a url or developing a package. Defaults to 'https', with proto == nothing delegating the choice to the package developer.

原文: https://juliacn.github.io/JuliaZH.jl/latest/stdlib/Pkg/