Skip to content

Latest commit

 

History

History
424 lines (357 loc) · 13.5 KB

1_joinmarket.md

File metadata and controls

424 lines (357 loc) · 13.5 KB

Qubes 4 & Whonix 15: JoinMarket

Create a VM without networking to host a JoinMarket wallet. The joinmarketd daemon will run on the bitcoind VM, communicate only over Tor onion services, and use stream isolation.

The offline joinmarket VM will communicate with your bitcoind VM using Qubes' qrexec.

What is JoinMarket?

Joinmarket is an open source, decentralized, and trustless market for coinjoins. Anyone holding Bitcoin can offer coinjoins for a fee, and anyone can pay a fee to have their transactions obfuscated.

See the Bitcoin wiki for a more detailed explanation of JoinMarket.

Why Do This?

This increases the privacy and security of your JoinMarket wallet while still maintaining full functionality. Enhanced privacy is achieved by using only Tor onion endpoints, and security is improved by removing the need for a network connection on the wallet VM.

Prerequisites

  • To complete this guide you must have first completed:

I. Set Up Dom0

A. In a dom0 terminal, create an AppVM.

  1. Create the AppVM for JoinMarket's wallet with no networking, using the whonix-ws-15-bitcoin TemplateVM.

Notes:

  • You must choose a label color, but it does not have to match this example.
  • It is safe to lower the maxmem and vcpus on this VM.
[user@dom0 ~]$ qvm-create --label black --prop maxmem='800' --prop netvm='' \
--prop vcpus='1' --template whonix-ws-15-bitcoin joinmarket

B. Allow comms from joinmarket to bitcoind VM.

[user@dom0 ~]$ echo 'joinmarket bitcoind allow' | sudo tee -a /etc/qubes-rpc/policy/qubes.ConnectTCP

II. Set Up TemplateVM

A. Create system user.

user@host:~$ sudo adduser --system joinmarket
Adding system user `joinmarket' (UID 117) ...
Adding new user `joinmarket' (UID 117) with group `nogroup' ...
Creating home directory `/home/joinmarket' ...

B. Shutdown TemplateVM.

user@host:~$ sudo poweroff

III. Install JoinMarket

A. In a bitcoind terminal, download and verify JoinMarket source code.

  1. Clone the JoinMarket repository.

Note: At the time of writing the current JoinMarket release is v0.6.3.1, modify the following steps accordingly if the version has changed.

user@host:~$ git clone --branch v0.6.3.1 \
https://github.com/JoinMarket-Org/joinmarket-clientserver ~/joinmarket-clientserver
Cloning into '/home/user/joinmarket-clientserver'...
remote: Enumerating objects: 249, done.
remote: Counting objects: 100% (249/249), done.
remote: Compressing objects: 100% (144/144), done.
remote: Total 6824 (delta 132), reused 195 (delta 105), pack-reused 6575
Receiving objects: 100% (6824/6824), 5.89 MiB | 227.00 KiB/s, done.
Resolving deltas: 100% (4526/4526), done.
Note: checking out '88b460a0ee459a1333f2a7baab4279a765471cb6'.
...
  1. Receive signing key.

Note: You can verify the key fingerprint in the release notes.

user@host:~$ gpg --recv-keys "2B6F C204 D9BF 332D 062B 461A 1410 01A1 AF77 F20B"
gpg: key 0x141001A1AF77F20B: public key "Adam Gibson (CODE SIGNING KEY) <[email protected]>" imported
gpg: Total number processed: 1
gpg:               imported: 1
  1. Change directory and verify.

Note: Your output may not match the example. Just check that it says Good signature.

user@host:~$ cd ~/joinmarket-clientserver/
user@host:~/joinmarket-clientserver$ git verify-tag v0.6.3.1
gpg: Signature made Mon 29 Jun 2020 03:56:54 PM UTC
gpg:                using RSA key 2B6FC204D9BF332D062B461A141001A1AF77F20B
gpg: Good signature from "Adam Gibson (CODE SIGNING KEY) <[email protected]>" [unknown]
gpg: WARNING: This key is not certified with a trusted signature!
gpg:          There is no indication that the signature belongs to the owner.
Primary key fingerprint: 2B6F C204 D9BF 332D 062B  461A 1410 01A1 AF77 F20B
  1. Copy ~/joinmarket-clientserver/ directory to user joinmarket.
user@host:~/joinmarket-clientserver$ sudo cp -r ~/joinmarket-clientserver/ /home/joinmarket/
  1. Fix ownership.
user@host:~/joinmarket-clientserver$ sudo chown -R joinmarket:nogroup /home/joinmarket/joinmarket-clientserver/

B. Install JoinMarket for main user.

  1. Create Python virtual environment.
user@host:~/joinmarket-clientserver$ virtualenv -p python3 ~/joinmarket-clientserver/jmvenv
Already using interpreter /usr/bin/python3
Using base prefix '/usr'
New python executable in /home/user/joinmarket-clientserver/jmvenv/bin/python3
Also creating executable in /home/user/joinmarket-clientserver/jmvenv/bin/python
Installing setuptools, pkg_resources, pip, wheel...done.
  1. Activate the virtual environment.
user@host:~/joinmarket-clientserver$ source ~/joinmarket-clientserver/jmvenv/bin/activate
  1. Install to virtual environment.

Note:

  • This step, and the next optional step, will produce a lot of output and take some time. This is normal, be patient.
(jmvenv) user@host:~/joinmarket-clientserver$ python ~/joinmarket-clientserver/setupall.py --all

3.1. Optional Step: Install QT dependencies for JoinMarket GUI. Note: You can safely skip this step if you do not intend to use the JoinMarket GUI.

(jmvenv) user@host:~/joinmarket-clientserver$ pip install PySide2 \
https://github.com/sunu/qt5reactor/archive/58410aaead2185e9917ae9cac9c50fe7b70e4a60.zip
  1. Copy the pip cache to user joinmarket.
(jmvenv) user@host:~/joinmarket-clientserver$ sudo cp -r ~/.cache/pip/ /home/joinmarket/.cache/
  1. Fix permissions.
(jmvenv) user@host:~/joinmarket-clientserver$ sudo chown -R joinmarket:nogroup /home/joinmarket/.cache/pip/

C. Copy joinmarket-clientserver/ directory to the joinmarket VM.

Note: Select joinmarket from the dom0 pop-up.

(jmvenv) user@host:~/joinmarket-clientserver$ qvm-copy ~/joinmarket-clientserver/

D. Install JoinMarket for daemon user.

  1. Change to user joinmarket.
(jmvenv) user@host:~/joinmarket-clientserver$ sudo -H -u joinmarket bash
  1. Change directory.
joinmarket@host:/home/user/joinmarket-clientserver$ cd ~/joinmarket-clientserver/
  1. Create virtual environment.
joinmarket@host:~/joinmarket-clientserver$ virtualenv -p python3 jmvenv
...
  1. Activate the virtual environment.
joinmarket@host:~/joinmarket-clientserver$ source jmvenv/bin/activate
  1. Install to virtual environment.

Note:

  • This step, and the next optional step, will produce a lot of output and take some time. This is normal, be patient.
(jmvenv) joinmarket@host:~/joinmarket-clientserver$ python setupall.py --daemon

IV. Configure bitcoind and joinmarketd

A. In a sys-bitcoin terminal, find out the gateway IP.

Note: Save your gateway IP (10.137.0.50 in this example) to replace <gateway-ip> in later examples.

user@host:~$ qubesdb-read /qubes-ip
10.137.0.50

B. In a bitcoind terminal, create RPC credentials for JoinMarket to communicate with bitcoind.

  1. Create a random RPC username. Do not use the one shown.

Note: Save your username (uJDzc07zxn5riJDx7N5m in this example) to replace <rpc-user> in later examples.

user@host:~$ head -c 15 /dev/urandom | base64
uJDzc07zxn5riJDx7N5m
  1. Use Bitcoin's tool to create a random RPC password and config entry. Do not use the one shown.

Notes:

  • Save the hased password (838c4dd74606918f1f27a5a2a52b168$9634018b87451bca05082f51b0b5b876fc72ef877bad98298e97e277abd5f90c in this example) to replace <hashed-pass> in later examples.
  • Save your password (IuziNnTsOUkonsDD3jn5WatPnFrFOMSnGUsRSUaq5Qg= in this example) to replace <rpc-pass> in later examples.
  • Replace <rpc-user> with the information noted earlier.
user@host:~$ ~/bitcoin/share/rpcauth/rpcauth.py <rpc-user>
String to be appended to bitcoin.conf:
rpcauth=uJDzc07zxn5riJDx7N5m:838c4dd74606918f1f27a5a2a52b168$9634018b87451bca05082f51b0b5b876fc72ef877bad98298e97e277abd5f90c
Your password:
IuziNnTsOUkonsDD3jn5WatPnFrFOMSnGUsRSUaq5Qg=

C. Configure bitcoind.

  1. Open bitcoin.conf.
user@host:~$ sudo -u bitcoin mousepad /home/bitcoin/.bitcoin/bitcoin.conf
  1. Paste the following at the bottom of the file.

Notes:

  • Be sure not to alter any of the existing information.
  • Replace <rpc-user> and <hashed-pass> with the information noted earlier.
# JoinMarket Auth
rpcauth=<rpc-user>:<hashed-pass>
# JoinMarket Wallet
wallet=joinmarket
  1. Save the file: Ctrl-S.
  2. Switch back to the terminal: Ctrl-Q.
  3. Restart the bitcoind service.
user@host:~$ sudo systemctl restart bitcoind.service

D. Use systemd to keep joinmarketd running.

  1. Create the service file.
user@host:~$ lxsu mousepad /rw/config/systemd/joinmarketd.service
  1. Paste the following.
[Unit]
Description=JoinMarket daemon

[Service]
WorkingDirectory=/home/joinmarket/joinmarket-clientserver
ExecStart=/bin/sh -c 'jmvenv/bin/python scripts/joinmarketd.py'

User=joinmarket
Type=idle
Restart=on-failure

PrivateTmp=true
ProtectSystem=full
NoNewPrivileges=true
PrivateDevices=true
MemoryDenyWriteExecute=true

[Install]
WantedBy=multi-user.target
  1. Save the file: Ctrl-S.
  2. Switch back to the terminal: Ctrl-Q.
  3. Fix permissions.
user@host:~$ chmod 0600 /rw/config/systemd/joinmarketd.service

E. Enable the service on boot.

  1. Edit the file /rw/config/rc.local.
user@host:~$ lxsu mousepad /rw/config/rc.local
  1. Paste the following at the bottom of the file.
cp /rw/config/systemd/joinmarketd.service /lib/systemd/system/
systemctl daemon-reload
systemctl start joinmarketd.service
  1. Save the file: Ctrl-S.
  2. Switch back to the terminal: Ctrl-Q.
  3. Execute the file.
user@host:~$ sudo /rw/config/rc.local

VI. Configure joinmarket VM

A. In a joinmarket terminal, open communication ports on boot.

  1. Edit the file /rw/config/rc.local.
user@host:~$ lxsu mousepad /rw/config/rc.local
  1. Paste the following at the bottom of the file.
qvm-connect-tcp 8332:bitcoind:8332
qvm-connect-tcp 27183:bitcoind:27183
qvm-connect-tcp 27184:bitcoind:27184
  1. Save the file: Ctrl-S.
  2. Switch back to the terminal: Ctrl-Q.
  3. Execute the file.
user@host:~$ sudo /rw/config/rc.local

B. Move copied JoinMarket directory to your home directory.

user@host:~$ mv ~/QubesIncoming/bitcoind/joinmarket-clientserver/ ~

C. Configure JoinMarket.

  1. Create data directory.
user@host:~$ mkdir -m 0700 ~/.joinmarket
  1. Source the virtual environment and change to the JoinMarket scripts/ directory.
user@host:~$ source ~/joinmarket-clientserver/jmvenv/bin/activate
(jmvenv) user@host:~$ cd ~/joinmarket-clientserver/scripts/
  1. Generate a configuration file.
(jmvenv) user@host:~/joinmarket-clientserver/scripts$ python wallet-tool.py
User data location: /home/user/.joinmarket/
Created a new `joinmarket.cfg`. Please review and adopt the settings and restart joinmarket.
  1. Make a backup and edit the file joinmarket.cfg.
(jmvenv) user@host:~/joinmarket-clientserver/scripts$ cp ~/.joinmarket/joinmarket.cfg ~/.joinmarket/joinmarket.cfg.orig
(jmvenv) user@host:~/joinmarket-clientserver/scripts$ echo > ~/.joinmarket/joinmarket.cfg
(jmvenv) user@host:~/joinmarket-clientserver/scripts$ mousepad ~/.joinmarket/joinmarket.cfg
  1. Paste the following.

Notes:

  • Be sure to replace <gateway-ip>, <rpc-user>, and <rpc-pass> with the information noted earlier.
  • For verbose desciptions of these setting, look to the original config file: ~/joinmarket-clientserver/scripts/joinmarket.cfg.orig.
[DAEMON]
no_daemon = 0
daemon_port = 27183
daemon_host = 127.0.0.1
use_ssl = false

[BLOCKCHAIN]
blockchain_source = bitcoin-rpc
network = mainnet
rpc_host = 127.0.0.1
rpc_port = 8332
rpc_user = <rpc-user>
rpc_password = <rpc-pass>
rpc_wallet_file = joinmarket

[MESSAGING:AgoraAnarplexIRC]
channel = joinmarket-pit
host = agora.anarplex.net
port = 14716
socks5 = true
socks5_host = <gateway-ip>
socks5_port = 9181
usessl = true

[MESSAGING:DarkScienceIRC]
channel = joinmarket-pit
host = darksci3bfoka7tw.onion
port = 6697
socks5 = true
socks5_host = <gateway-ip>
socks5_port = 9182
usessl = true

[LOGGING]
console_log_level = INFO
color = true

[TIMEOUT]
maker_timeout_sec = 45
unconfirm_timeout_sec = 180
confirm_timeout_hours = 6

[POLICY]
segwit = true
native = false
merge_algorithm = default
tx_fees = 3
absurd_fee_per_kb = 350000
tx_broadcast = self
minimum_makers = 2
taker_utxo_retries = 3
taker_utxo_age = 5
taker_utxo_amtpercent = 20
accept_commitment_broadcasts = 1
  1. Save the file: Ctrl-S.
  2. Close the editor: Ctrl-Q.

VI. Final Notes

  • Once bitcoind has finished syncing in the bitcoind VM you will be able to use JoinMarket's wallet from the joinmarket VM.

VII. Optional Steps

A. In a joinmarket terminal, source virtual envrionment and change to JoinMarket's scripts/ directory on boot.

  1. Edit the file ~/.bashrc.
user@host:~$ mousepad ~/.bashrc
  1. Paste the following at the bottom of the file.
source /home/user/joinmarket-clientserver/jmvenv/bin/activate
cd /home/user/joinmarket-clientserver/scripts/
  1. Save the file: Ctrl-S.
  2. Switch back to the terminal: Ctrl-Q.
  3. Source the file to take effect.
user@host:~$ source ~/.bashrc