Automatic Node Upgrade: Nymvisor Setup & Usage
The Nymvisor binary was built in the building nym section. If you haven't yet built Nym and want to run the code, go there first. You can build just Nymvisor with
cargo build --release --bin nymvisor.
Our documentation often refer to syntax annotated in <> brackets. We use this expression for variables that are unique to each user (like path, local moniker, versions etcetra).
Any syntax in <> brackets needs to be substituted with your correct name or version, without the <> brackets. If you are unsure, please check our table of essential parameters and variables (opens in a new tab).
What is Nymvisor?
Nymvisor is a process manager for Nym binaries that monitors the Nym release information for any newly released binaries. If it detects any changes, Nymvisor can automatically download the binary, stop the current binary, switch from the old binary to the new one, and finally restart the underlying process with the new binary.
In essence, it tries to mirror the behaviour of Cosmovisor (opens in a new tab), a tool used by Cosmos blockchain operators for managing/automating chain upgrades. Nymvisor, however, introduces some Nym-specific changes since, for example, upgrade information is obtained from our GitHub releases page (opens in a new tab) instead of (in the case of Cosmos blockchains) governance proposals.
Supported Binaries
You can use Nymvisor to automate the upgrades of the following binaries:
nym-apinym-nodenym-clientnym-socks5-client
Nymvisor is an early and experimental software. Users should use it at their own risk.
Current version
Binary Name: nymvisor
Build Timestamp: 2024-12-18T17:33:13.184930493Z
Build Version: 0.1.10
Commit SHA: b628a5f8148f74c646915292c8b6dc0a46202a27
Commit Date: 2024-12-13T11:49:27.000000000+01:00
Commit Branch: master
rustc Version: 1.84.0-nightly
rustc Channel: nightly
cargo Profile: release
Preliminary steps
You need to have at least one Nym Node / client / Nym API instance already set up on the same VPS that you wish to run Nymvisor on.
Using Nymvisor presumes your VPS is running an operating system that is compatible with the pre-compiled binaries availiable on the Github releases page (opens in a new tab). If you're not, then until we're packaging for a greater variety of operating systems, you're stuck with manually upgrading your node.
Setup and Usage
Viewing command help
You can check that your binaries are properly compiled with:
./nymvisor --helpWhich should return a list of all available commands.
Usage: nymvisor [OPTIONS] <COMMAND>
Commands:
init Initialise a nymvisor instance with persistent Config.toml file
run Run the associated daemon with the preconfigured settings
build-info Show build information of this binary
daemon-build-info Show build information of the associated daemon
add-upgrade Queues up another upgrade for the associated daemon
config Show configuration options being used by this instance of nymvisor
help Print this message or the help of the given subcommand(s)
Options:
-c, --config-env-file <CONFIG_ENV_FILE>
Path pointing to an env file that configures the nymvisor and overrides any preconfigured values
-h, --help
Print help
-V, --version
Print versionYou can also check the various arguments required for individual commands with:
./nymvisor <COMMAND> --helpInitialising your Nymvisor Instance
This example will use the Nym Node binary as an example - however replacing
nym-nodewith any other supported binary will work the same.
Initialise your Nymvisor instance with the following command. You must initialise Nymvisor with the binary you wish to add upgrades for:
./nymvisor init --daemon-home ~/.nym/nym-node/<ID> <PATH>/nym-nodeWhere the value of --daemon-home might be ~/.nym/nym-nodes/default-nym-node and <PATH> might be /home/sakine/nym/target/release/nym-node, or wherever your node binary is located.
By default this will create config files at ~/.nym/nymvisors/instances/nym-node-default/config/config.toml as shown in the console output above. For config options look at the different --flags available, or the environment variables section below.
Running your Nymvisor Instance
Nymvisor acts as a wrapper around the specified node process - it has to do this in order to be able to pause and restart this process. As such, you need to run your node via Nymvisor!
The interface to the nymvisor run <ARGS> command is quite simple. Any argument passed after the run command will be passed directly to the underlying daemon, for example: nymvisor run run --id default-nym-node will run the $DAEMON_NAME run --id default-nym-node command (where DAEMON_NAME is the name of the binary itself (e.g. nym-api, nym-node, etc.)).
run Nymvisor and start your node via the following command. Make sure to stop any existing node before running this command.
./nymvisor run run --id <ID>Nymvisor will now manage your node process (for an in-depth overview of this command check the in-depth command information below). It will periodically poll this endpoint (opens in a new tab) (replace nym-node with whatever node you may actually be running via Nymvisor) and check for a new version of the binary it is watching. If this exists, it will then, using the information proceed with these steps:
- Pause your node process
- Grab the new binary (
version) - Verify it against the provided
checksum - Perform a data backup of the existing node
- Replace the old binary with the new one
- Restart the process
And that's it!
Creating an Adhoc Upgrade
This section is for advanced users. Generally users will not have to use this command.
nymvisor add-upgrade <PATH_TO_EXECUTABLE> --upgrade-name=<NAME> --arg1=value1 --arg2=value2 ... can be used to amend an existing upgrade-plan.json by creating new entries or to add an executable to an existing scheduled upgrade so that it would not have to be downloaded.
Situations in which this command might be used are:
- An adhoc upgrade if e.g. a patched version of a binary was required
- If a user doesn't trust the verification process of Nymvisor's pipeline and wishes to build/verify the binary themselves before using Nymvisor to perform the upgrade
- If a user isn't using a currently supported operating system and needs to manually specify a binary they have built themselves
Similarly to init, add-upgrade requires a positional argument specifying a valid path to the upgrade binary. Furthermore, the --upgrade-name keyword argument must be set in order to declare the upgrade name. The remaining arguments are optional. They include:
--force- if specified, will allow Nymvisor to overwrite existing upgrade binary /upgrade-info.jsonfiles if they already exist--add-binary- indicate that this command should only add binary to an existing scheduled upgrade--now- if specified will force the upgrade to be performed immediately (technically not 'immediately' within few seconds). It can't be specified alongside either--upgrade-timeor--upgrade-delayarguments--publish-date- if a newupgrade-info.jsonfile is going to be created, this argument will specify thepublish_datemetadata field. Otherwise, the current time will be used. The RFC3339 datetime (opens in a new tab) format is expected--upgrade-time- specifies the time at which the provided upgrade will be performed (RFC3339 formatted). If left unset, the upgrade will be performed in 15 minutes. It can't be specified alongside either--nowor--upgrade-delayarguments.--upgrade-delay- specifies delay until the provided upgrade is going to get performed. If let unset, the upgrade will be performed in 15 minutes. It can't be specified alongside either--upgrade_timeor--nowarguments.
Maintenance and Configuration
Check the Nymvisor Configuration page for information on Nymvisor process maintenance and automation. You can find more in-depth information about the various aspects of Nymvisor such as what happens under the hood for various commands.
Config
The output format of nymvisor config can be further configured with --output argument. By default a human-readable text representation is used:
id: nym-node-default
daemon name: nym-node
daemon home: /home/nym/.nym/nym-nodes/default-nym-node
upstream base upgrade url: https://nymtech.net/.wellknown/
disable nymvisor logs: false
CUSTOM upgrade data directory ""
upstream absolute upgrade url: ""
allow binaries download: true
enforce download checksum: true
restart after upgrade: true
restart on failure: false
on failure restart delay: 10s
max startup failures: 10
startup period duration: 2m
shutdown grace period: 10s
CUSTOM backup data directory ""
UNSAFE skip backups false- Adding
--output=jsonwould format the same data into JSON which can be more easily parsed programmatically to e.g. pipe the output intojqfor further processing.
nymvisor config --output=json- Outputs:
{"nymvisor":{"id":"nym-node-default","upstream_base_upgrade_url":"https://nymtech.net/.wellknown/","upstream_polling_rate":"1h","disable_logs":false,"upgrade_data_directory":null},"daemon":{"name":"nym-node","home":"/home/nym/.nym/nym-nodes/default-nym-nodee","absolute_upstream_upgrade_url":null,"allow_binaries_download":true,"enforce_download_checksum":true,"restart_after_upgrade":true,"restart_on_failure":false,"failure_restart_delay":"10s","max_startup_failures":10,"startup_period_duration":"2m","shutdown_grace_period":"10s","backup_data_directory":null,"unsafe_skip_backup":false}}CLI Overview
Command options are:
help,--help, or-h- Output Nymvisor help information and display the available commands.config- Display the current Nymvisor configuration, that means displaying the current configuration file that might have been overridden with environment variables value that Nymvisor is using.init- Generate aconfig.tomlfile for this instance of Nymvisor that will use the provided arguments alongside any environmental variables that are set.add-upgrade- Add an upgrade manually to Nymvisor. This command allows you to easily add the binary corresponding to an upgrade or amend the existingupgrade-plan.jsonwhilst creating newupgrade-info.jsonfile.build-info- Output the build information.daemon-build-info- Output the build information of the current binary used by the associated daemon.run- Run the configured binary using the rest of the provided arguments.-Vor--version- Output the Nymvisor version
Similarly to other Nym binaries, Nymvisor supports a global --config-env-file or -c flag that allows specifying path to a .env file defining any relevant environmental variables that are going to be applied to any of the Nymvisor commands as described in the Environment section.
For commands that depend on Nymvisor config file (i.e. all but init), the configuration file is loaded as follows:
- If available, reading
$NYMVISOR_CONFIG_PATH - Otherwise, if
$NYMVISOR_IDis set, a default path will be used, i.e.$HOME/.nym/nymvisors/instances/$NYMVISOR_ID/config/config.toml - Finally, if there's only a single
nymvisorinstance instantiated (as defined by directories in$HOME/.nym/nymvisors/instances), that one will be loaded - If all of the above fails, an error is returned
Nymvisor attempts to load the file from the derived path. If it fails, it attempts to use one of the older schemes and upgrade it as it goes, the loaded configuration is then overridden with any value that might have been set in the environment.
Environment Variables
Please note environmental variables take precedence over any arguments passed, i.e. if one were to specify --daemon_home="/foo" and set DAEMON_HOME="bar", the value of "bar" would end up being used.
For any of its commands as described in CLI Overview section, Nymvisor reads its configuration from the following environment variables:
NYMVISOR_IDis the human-readable identifier of the particular Nymvisor instance.NYMVISOR_CONFIG_PATHis used to manually override path to the configuration file of the Nymvisor instance.NYMVISOR_UPSTREAM_BASE_UPGRADE_URL(defaults to https://nymtech.net/.wellknown/ (opens in a new tab)) is the base url of the upstream source for obtaining upgrade information for the daemon. It will be used fo constructing the full url, i.e.$NYMVISOR_UPSTREAM_BASE_UPGRADE_URL/$DAEMON_NAME/upgrade-info.json.NYMVISOR_UPSTREAM_POLLING_RATE(defaults to 1h) is polling rate the upstream url for upgrade information.NYMVISOR_DISABLE_LOGS(defaults tofalse). If set totrue, this will disable Nymvisor logs (but not the underlying process) completely.NYMVISOR_UPGRADE_DATA_DIRECTORYis the custom directory for upgrade data - binaries and upgrade plans. If not set, the global Nymvisors' data directory will be used instead.DAEMON_NAMEis the name of the binary itself (e.g.nym-api,nym-node, etc.).DAEMON_HOMEis the location where thenymvisor/directory is kept that contains the auxiliary files associated with the underlying daemon instance, such as any backups or current version information, e.g.$HOME/.nym/nym-api/my-nym-api,$HOME/.nym/nym-nodes/default-nym-node, etc.DAEMON_ABSOLUTE_UPSTREAM_UPGRADE_URLis the absolute (i.e. the full url) upstream source for upgrade plans for this daemon. The url has to point to an endpoint containing a validUpgradeInfojson file. If set it takes precedence overNYMVISOR_UPSTREAM_BASE_UPGRADE_URL.DAEMON_ALLOW_BINARIES_DOWNLOAD(defaults totrue), if set totrue, it will enable auto-downloading of new binaries (as declared by urls in correspondingupgrade-info.jsonfiles). For security reasons one might wish to disable it and instead manually provide binaries by either placing them in the appropriate directory or by invokingadd-upgradecommand.DAEMON_ENFORCE_DOWNLOAD_CHECKSUM(defaults totrue), if set totrueNymvisor will require that a checksum is provided in the upgrade plan for the upgrade binary to be downloaded. If disabled, Nymvisor will not require a checksum to be provided, but still check the checksum if one is provided.DAEMON_RESTART_AFTER_UPGRADE(defaults totrue), if set totrueNymvisor will restart the subprocess with the same command-line arguments and flags (but with the new binary) after a successful upgrade. Otherwise (false), Nymvisor stops running after an upgrade and requires the system administrator to manually restart it. Note restart is only after the upgrade and does not auto-restart the subprocess after an error occurs. That is controlled viaDAEMON_RESTART_ON_FAILURE.DAEMON_RESTART_ON_FAILURE(defaults totrue), if set totrue, Nymvisor will restart the subprocess with the same command-line arguments and flags if it has terminated with a non-zero exit code.DAEMON_FAILURE_RESTART_DELAY(defaults to 10s), ifDAEMON_RESTART_ON_FAILUREis set totrue, this will specify a delay between the process shutdown (with a non-zero exit code) and it being restarted.DAEMON_MAX_STARTUP_FAILURES(defaults to 10) ifDAEMON_RESTART_ON_FAILUREis set totrue, this defines the maximum number of startup failures the subprocess can experience in a quick succession before no further restarts will be attempted and Nymvisor will terminate.DAEMON_STARTUP_PERIOD_DURATION(defaults to 120s) ifDAEMON_RESTART_ON_FAILUREis set totrue, this defines the length of time during which the subprocess is still considered to be in the startup phase when its failures are going to be counted towards the limit defined inDAEMON_MAX_STARTUP_FAILURES.DAEMON_SHUTDOWN_GRACE_PERIOD(defaults to 10s), specifies the amount of time Nymvisor is willing to wait for the subprocess to undergo graceful shutdown after receiving an interrupt before it sends a kill signal.DAEMON_BACKUP_DATA_DIRECTORYspecifies custom backup directory for daemon data. If not set,DAEMON_HOME/nymvisor/backupsis used instead.DAEMON_UNSAFE_SKIP_BACKUP(defaults tofalse), if set totrue, all upgrades will be performed directly without performing any backups. Otherwise (false), Nymvisor will back up the contents ofDAEMON_HOMEbefore trying the upgrade.
Dir structure
The folder structure of Nymvisor is heavily inspired by Cosmovisor, but with some notable changes to accommodate our binaries having possibly multiple instances due to their different --id flags. The data is spread through three main directories:
-
In a global
nymvisorsdata directory shared by all Nymvisor instances (default:$HOME/.nym/nymvisors/data) that contains daemon upgrade plans, binaries and upgrades histories. It includes a sub-directory for each version of the application (i.e.genesisorupgrades/<NAME>). Within each sub-directory is the application binary (i.e.bin/$DAEMON_NAME), the associatedupgrade-info.jsonand any additional auxiliary files associated with each binary.currentis a symbolic link to the currently active directory (i.e.genesisorupgrades/<NAME>) -
In a home directory of a particular
nymvisorinstance (e.g.$HOME/.nym/nymvisors/instances/<NYMVISOR_INSTANCE_ID>/). It includes sub-directories for its configuration file (i.e.config/config.toml), that preconfigures the instance, and for any additional persistent data that might be added in the future (i.e.data) -
In a
nymvisordata directory inside the home directory of the managed daemon instance (default:$HOME/.nym/$DAEMON_NAME/nymvisor) that contains sub-directories for data backups (i.e.backups/<NAME>) and current version information (current-version-info.json)
A sample full structure looks as follows:
~/.nym
├── nymvisors
│ ├── instances
│ │ ├── <id1>
│ │ │ ├── config
│ │ │ │ └── config.toml
│ │ │ └── data
│ │ │ └── ...
│ │ └── <id2>
│ │ └── ...
│ └── data
│ ├── nym-api
│ │ ├── current -> genesis or upgrades/<name>
│ │ ├── genesis
│ │ │ ├── bin
│ │ │ │ └── nym-api
│ │ │ └── upgrade-info.json
│ │ ├── upgrades
│ │ │ └── <upgrade-name>
│ │ │ ├── bin
│ │ │ │ └── nym-api
│ │ │ └── upgrade-info.json
│ │ ├── upgrade-history.json
│ │ └── upgrade-plan.json
│ ├── nym-node
│ │ └── ...
│ └── $DAEMON_NAME
│ └── ...
└── nym-api
├── <id1>
│ ├── config
│ │ └── <nym-api-config-data>
│ ├── data
│ │ └── <nym-api-data>
│ └── nymvisor
│ ├── backups
│ │ └── <upgrade-name>
│ │ └── ....
│ └── current-version-info.json
└── <id2>
└── ...Commands In-Depth
This section outlines what happens under the hood with the following commands:
init
init does the following:
- Executes the
$DAEMON_NAME build-infocommand on the daemon executable to check its validity and obtain its name - Creates the required directory structure:
$DAEMON_HOME/nymvisorfolder if it doesn't yet exist$DAEMON_BACKUP_DATA_DIRECTORYfolder if it doesn't yet exist$NYMVISOR_UPGRADE_DATA_DIRECTORYfolder if it doesn't yet exist$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/genesis/binfolder if it doesn't yet exist$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/upgradesfolder if it doesn't yet exist
- Copies the provided executable to
$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/genesis/bin/$DAEMON_NAME - Generates initial
$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/genesis/upgrade-info.jsonfile based on the provided binary - Generates initial
$DAEMON_HOME/nymvisor/current-version-info.jsonfile based on the provided binary - Creates a
$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/currentsymlink pointing to$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/genesis - Saves the Nymvisor instance's config file to
$NYMVISOR_CONFIG_PATHand creates the full directory structure for the file - Outputs (to
stdout) the full configuration used
nymvisor init is specifically for initializing Nymvisor, and should not be confused with a daemon's init command - such as nym-socks5-client init (e.g. nymvisor run init).
run
nymvisor run is a lightweight wrapper around the underlying daemon. It uses only a single thread and spawns three simple tasks:
- An upstream puller that checks the upstream source (as defined by
DAEMON_ABSOLUTE_UPSTREAM_UPGRADE_URLor derived fromNYMVISOR_UPSTREAM_BASE_UPGRADE_URL) for any recently released upgrades. It then creates appropriateupgrade-info.jsonfile and updates theupgrade-plan.json - A file watcher for the
upgrade-plan.jsonfile that can notify the main run loop of upgrades that were added by either the above upstream puller task or via theadd-upgradecommand, - The daemon run loop that:
- Runs the
DAEMON_NAMEwith the provided arguments until:- It completes the execution (with any exit code),
- Nymvisor receives a
SIGINT,SIGTERMorSIGQUIT, - A new upgrade is scheduled to be performed (by other task watching for changes in
upgrade-plan.jsonand waiting until theupgrade_time,
- If
DAEMON_UNSAFE_SKIP_BACKUPis not set totrue, it backups the content ofDAEMON_HOMEdirectory, - Performs the binary upgrade by:
- Creating a temporary, exclusive and non-blocking,
upgrade.lockfile for theDAEMON_NAME.flockwithLOCK_EX | LOCK_NBis used for that purpose. The file is created in case users didn't read any warnings and attempted to run multiple instances ofnymvisormanaging the sameDAEMON_NAME, - Downloading the upgrade binary for the runners architecture using one of the urls defined in
upgrade-info.json. Note, however, that this is only done if the binary associated with the<UPGRADE-NAME>does not already exist andDAEMON_ALLOW_DOWNLOAD_BINARIESis set totrue,- If the binary has been downloaded and
DAEMON_ENFORCE_DOWNLOAD_CHECKSUMis set to true, the file checksum is verified using the specified algorithm,
- If the binary has been downloaded and
- Verifying the upgrade binary - checking if it's a valid executable with expected
build-info. Note that this will also seta+xbits on the file if those permissions have not already been set,- removing the queued upgrade from
upgrade-plan.json, - Inserting new upgrade into the
upgrade-history.json, - Updating the
current-version-info.json, - Updating the
$NYMVISOR_UPGRADE_DATA_DIRECTORY/$DAEMON_NAME/currentsymlink to the upgrade directory, - Removing the
upgrade.lockfile.
- removing the queued upgrade from
- Creating a temporary, exclusive and non-blocking,
- The above loop is repeated if either:
- The daemon has crashed and
DAEMON_MAX_STARTUP_FAILUREShas not been reached yet, - The daemon has successfully been upgraded,
DAEMON_RESTART_AFTER_UPGRADEhas been set totrueand the manual flag on the performed upgrade has been set tofalse.
- The daemon has crashed and
- Runs the
add-upgrade
nymvisor add-upgrade does the following:
- Executes the
$DAEMON_NAME build-infocommand on the daemon executable to check its validity and obtain its name - Attempts to load existing
upgrade-info.jsonfor the provided<upgrade-name>. If it already exists and neither--forcenor--add-binarywas specified, Nymvisor will terminate - Checks if
upgrades/<UPGRADE_NAME>/$DAEMON_NAMEbinary already exists. If it does and--forceflag was not specified, Nymvisor will terminate the provided upgrade binary is copied to its appropriate location - if applicable, new
upgrade-info.jsonis created and written to its appropriate location upgrade-plan.jsonis updated with the new upgrade details. If there's an active Nymvisor instance running, this change will be detected to initialise upgrade process