Offline Transaction Signing

This is NOT necessarily the recommended cold storage setup, due to high complexity, large room for errors and employing a general purpose computer for transaction signing (even if offline).

Published for educational purposes only to understand “what would it take to sign offline”.

It is generally better to use a hardware wallet like Trezor or Ledger.

Opinions may vary.

This is a guest tutorial contributed by crocket.

Offline transaction signing involves:

  • Creating an unsigned transaction on an online, view-only wallet
  • Moving the unsigned transaction to an offline machine
  • Signing the unsigned transaction on an offline machine
  • Moving the signed transaction back to online, view-only wallet
  • Broadcasting the transaction

Creating a new offline wallet

Constructing a new offline wallet is done by executing:

monero-wallet-cli --generate-new-wallet /path/to/wallet-file

on an offline machine. Record the seed on paper by executing seed on the offline wallet.

Creating a new offline wallet with seed offset passphrase

Seed and seed offset passphrase combine to create a new seed. You can store seed and seed offset passphrase in separate places so that a thief can’t steal your fund without stealing seed and seed offset passphrase. I recommend 6 to 8 (english) words as a seed offset passphrase as one english word has 11 bits of entropy on average and 8 words have 88 bits of entropy. With seed passphrase, you can also create decoy wallets that contain a little bit of money and can protect you from torturers or blackmailers who demand money from you.

If you want to create an offline wallet with seed and seed passphrase, create an offline wallet, record the seed on paper, delete the wallet file, generate seed offset passphrase, record seed offset passphrase on paper, and execute

monero-wallet-cli --generate-new-wallet /path/to/wallet-file \
---restore-deterministic-wallet

to restore from seed and seed offset passphrase. When you restore from seed, you can enter seed offset passphrase.

Generate seed offset passphrase on an offline machine or with diceware because humans are bad at creating random passphrases.

If you want to reconstruct an existing offline wallet that received or sent transactions, you need extra steps. Refer to Restoring offline wallet.

Creating a new view-only wallet

To create a view-only wallet, copy primary address and secret view key from an offline wallet to an online machine where a view-only wallet is going to be created. You can get primary address by executing address on an offline wallet and secret view key by executing viewkey on the offline wallet.

You can use a microSD card and two USB microSD card readers to exchange data between an offline wallet and a view-only wallet. You can also use a USB flash drive.

To create a view-only wallet on an online machine, execute

monero-wallet-cli --generate-from-view-key /path/to/wallet-file \
--daemon-address remote-node-address:port

If you want to reconstruct an existing view-only wallet that received or sent transactions, refer to Restoring view-only wallet.

Launching offline wallet

Execute

monero-wallet-cli --wallet-file /path/to/wallet-file

Launching a view-only wallet

Execute

monero-wallet-cli --wallet-file /path/to/wallet-file \
--daemon-address remote-node-address:port

It’s safe to sync your wallet over clearnet. If you want to broadcast a transaction without revealing your IP address, execute

monero-wallet-cli --wallet-file /path/to/wallet-file \
--daemon-address tor-or-i2p-remote-node-address:port \
--proxy 127.0.0.1:tor-or-i2p-port

Synchronizing wallet over clearnet is a lot faster than doing it on tor or i2p. Thus, consider synchronizing over clearnet even if you broadcast transactions over tor or i2p.

Offline transaction signing

Execute any wallet command that transfers monero to any address. For example,

transfer xmr-address amount-of-xmr-to-send

Any transfer command on a view-only wallet creates unsigned_monero_tx in the current working directory.

Move unsigned_monero_tx to an offline machine that has an offline wallet. Execute

sign_transfer

on the offline wallet in the directory with unsigned_monero_tx. signed_monero_tx file is created in the current working directory. Move signed_monero_tx to the online machine with a view-only wallet. In the directory with signed_monero_tx, launch the view-only wallet, and execute

submit_transfer

Because a view-only wallet doesn’t have key images, it can’t see outgoing transactions. To make a view-only wallet see outgoing transactions, it has to export new outputs created by submit_transfer to an offline wallet which creates key images out of new outputs.

Execute

export_outputs outputs

on a view-only wallet. Move outputs file to the offline machine with an offline wallet. Launch the offline wallet, and execute

import_outputs /path/to/outputs

Export key images derived from new outputs by executing

export_key_images key_images

on an offline wallet. Move key_images file to the machine with a view-only wallet. Launch the view-only wallet, and execute

import_key_images /path/to/key_images

Updating wallet software on an offline signing machine

When you update an offline machine with offline wallets, you can’t just connect the machine to the internet and update wallet software because doing so exposes offline wallets to the internet.

Instead, boot OS installation media, wipe filesystems, and then connect to the internet, and install everything from scratch again.

If your root filesystem is encrypted, OS installation media can connect to the internet from the beginning because encrypted data are safe until they are decrypted.

Restoring offline wallet

After updating wallet software on an offline signing machine by wiping it out and reinstalling everything, you have to restore offline wallet.

Restore an offline wallet from seed (and seed offset passphrase) by executing

monero-wallet-cli --generate-new-wallet wallet-file --restore-deterministic-wallet

The new offline wallet can’t sign new transactions because it doesn’t have all trasaction outputs that precede a new unsigned transaction. Thus, it first has to import all outputs from a view-only wallet.

On a view-only wallet that was derived from the offline wallet, execute

export_outputs all all_outputs

all is important because export_outputs exports only new outputs that weren’t exported before, but

export_outputs all

exports all outputs. Move all_outputs file to the offline machine with the offline wallet. Execute

import_outputs /path/to/all_outputs

on the new offline wallet.

Restoring view-only wallet

If you reconstruct view-only wallet, because it doesn’t have key images, it can’t see outgoing transactions. If it can’t see outgoing transactions, it reports wrong account balances. Thus, it has to import all key images from its corresponding offline wallet.

On the offline wallet, execute

export_key_images all all_key_images

export_key_images doesn’t work because it exports only new key images that weren’t exported before.

export_key_images all

exports all key images. Move all_key_images file to the machine with the view-only wallet. Execute

import_key_images /path/to/all_key_images