main_autosign.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2023 The MMGen Project <mmgen@tuta.io>
  5. #
  6. # This program is free software: you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation, either version 3 of the License, or
  9. # (at your option) any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. """
  19. mmgen-autosign: Auto-sign MMGen transactions and message files
  20. """
  21. import sys,os,asyncio,signal,shutil
  22. from subprocess import run,PIPE,DEVNULL
  23. from collections import namedtuple
  24. from stat import *
  25. from .common import *
  26. mountpoint = '/mnt/tx'
  27. tx_dir = '/mnt/tx/tx'
  28. msg_dir = '/mnt/tx/msg'
  29. part_label = 'MMGEN_TX'
  30. wallet_dir = '/dev/shm/autosign'
  31. mn_fmts = {
  32. 'mmgen': 'words',
  33. 'bip39': 'bip39',
  34. }
  35. mn_fmt_dfl = 'mmgen'
  36. opts.UserOpts._set_ok += ('outdir','passwd_file')
  37. opts_data = {
  38. 'sets': [('stealth_led', True, 'led', True)],
  39. 'text': {
  40. 'desc': 'Auto-sign MMGen transactions and message files',
  41. 'usage':'[opts] [command]',
  42. 'options': f"""
  43. -h, --help Print this help message
  44. --, --longhelp Print help message for long options (common options)
  45. -c, --coins=c Coins to sign for (comma-separated list)
  46. -I, --no-insert-check Don’t check for device insertion
  47. -l, --led Use status LED to signal standby, busy and error
  48. -m, --mountpoint=M Specify an alternate mountpoint 'M' (default: {mountpoint!r})
  49. -M, --mnemonic-fmt=F During setup, prompt for mnemonic seed phrase of format
  50. 'F' (choices: {fmt_list(mn_fmts,fmt='no_spc')}; default: {mn_fmt_dfl!r})
  51. -n, --no-summary Don’t print a transaction summary
  52. -s, --stealth-led Stealth LED mode - signal busy and error only, and only
  53. after successful authorization.
  54. -S, --full-summary Print a full summary of each signed transaction after
  55. each autosign run. The default list of non-MMGen outputs
  56. will not be printed.
  57. -q, --quiet Produce quieter output
  58. -v, --verbose Produce more verbose output
  59. """,
  60. 'notes': f"""
  61. COMMANDS
  62. gen_key - generate the wallet encryption key and copy it to {mountpoint!r}
  63. setup - generate the wallet encryption key and wallet
  64. wait - start in loop mode: wait-mount-sign-unmount-wait
  65. USAGE NOTES
  66. If invoked with no command, the program mounts a removable device containing
  67. unsigned MMGen transactions and/or message files, signs them, unmounts the
  68. removable device and exits.
  69. If invoked with 'wait', the program waits in a loop, mounting the removable
  70. device, performing signing operations and unmounting the device every time it
  71. is inserted.
  72. On supported platforms (currently Orange Pi, Rock Pi and Raspberry Pi boards),
  73. the status LED indicates whether the program is busy or in standby mode, i.e.
  74. ready for device insertion or removal.
  75. The removable device must have a partition labeled MMGEN_TX with a user-
  76. writable root directory and a directory named '/tx', where unsigned MMGen
  77. transactions are placed. Optionally, the directory '/msg' may also be created
  78. and unsigned message files created by `mmgen-msg` placed in this directory.
  79. On the signing machine the mount point {mountpoint!r} must exist and /etc/fstab
  80. must contain the following entry:
  81. LABEL='MMGEN_TX' /mnt/tx auto noauto,user 0 0
  82. Transactions are signed with a wallet on the signing machine (in the directory
  83. {wallet_dir!r}) encrypted with a 64-character hexadecimal password saved
  84. in the file `autosign.key` in the root of the removable device partition.
  85. The password and wallet can be created in one operation by invoking the
  86. command with 'setup' with the removable device inserted. In this case, the
  87. user will be prompted for a seed mnemonic.
  88. Alternatively, the password and wallet can be created separately by first
  89. invoking the command with 'gen_key' and then creating and encrypting the
  90. wallet using the -P (--passwd-file) option:
  91. $ mmgen-walletconv -r0 -q -iwords -d{wallet_dir} -p1 -P/mnt/tx/autosign.key -Llabel
  92. Note that the hash preset must be '1'. Multiple wallets are permissible.
  93. For good security, it's advisable to re-generate a new wallet and key for
  94. each signing session.
  95. This command is currently available only on Linux-based platforms.
  96. """
  97. }
  98. }
  99. cmd_args = opts.init(
  100. opts_data,
  101. add_opts = ['outdir','passwd_file'], # in _set_ok, so must be set
  102. init_opts = {
  103. 'quiet': True,
  104. 'out_fmt': 'wallet',
  105. 'usr_randchars': 0,
  106. 'hash_preset': '1',
  107. 'label': 'Autosign Wallet',
  108. })
  109. exit_if_mswin('autosigning')
  110. if opt.mnemonic_fmt:
  111. if opt.mnemonic_fmt not in mn_fmts:
  112. die(1,'{!r}: invalid mnemonic format (must be one of: {})'.format(
  113. opt.mnemonic_fmt,
  114. fmt_list(mn_fmts,fmt='no_spc') ))
  115. from .wallet import Wallet
  116. from .tx import UnsignedTX
  117. from .tx.sign import txsign
  118. from .protocol import init_proto
  119. from .rpc import rpc_init
  120. if opt.mountpoint:
  121. mountpoint = opt.mountpoint
  122. keyfile = os.path.join(mountpoint,'autosign.key')
  123. msg_dir = os.path.join(mountpoint,'msg')
  124. tx_dir = os.path.join(mountpoint,'tx')
  125. opt.outdir = tx_dir
  126. opt.passwd_file = keyfile
  127. async def check_daemons_running():
  128. if opt.coin:
  129. die(1,'--coin option not supported with this command. Use --coins instead')
  130. if opt.coins:
  131. coins = opt.coins.upper().split(',')
  132. else:
  133. ymsg('Warning: no coins specified, defaulting to BTC')
  134. coins = ['BTC']
  135. for coin in coins:
  136. proto = init_proto( coin, testnet=g.network=='testnet', need_amt=True )
  137. if proto.sign_mode == 'daemon':
  138. vmsg(f'Checking {coin} daemon')
  139. from .exception import SocketError
  140. try:
  141. await rpc_init(proto)
  142. except SocketError as e:
  143. die(2,f'{coin} daemon not running or not listening on port {proto.rpc_port}')
  144. def get_wallet_files():
  145. try:
  146. dlist = os.listdir(wallet_dir)
  147. except:
  148. die(1,f"Cannot open wallet directory {wallet_dir!r}. Did you run 'mmgen-autosign setup'?")
  149. fns = [x for x in dlist if x.endswith('.mmdat')]
  150. if fns:
  151. return [os.path.join(wallet_dir,w) for w in fns]
  152. else:
  153. die(1,'No wallet files present!')
  154. def do_mount():
  155. if not os.path.ismount(mountpoint):
  156. if run(['mount',mountpoint],stderr=DEVNULL,stdout=DEVNULL).returncode == 0:
  157. msg(f'Mounting {mountpoint}')
  158. global have_msg_dir
  159. have_msg_dir = os.path.isdir(msg_dir)
  160. for cdir in [tx_dir] + ([msg_dir] if have_msg_dir else []):
  161. try:
  162. ds = os.stat(cdir)
  163. assert S_ISDIR(ds.st_mode), f'{cdir!r} is not a directory!'
  164. assert ds.st_mode & S_IWUSR|S_IRUSR == S_IWUSR|S_IRUSR, f'{cdir!r} is not read/write for this user!'
  165. except:
  166. die(1,f'{cdir!r} missing or not read/writable by user!')
  167. def do_umount():
  168. if os.path.ismount(mountpoint):
  169. run(['sync'],check=True)
  170. msg(f'Unmounting {mountpoint}')
  171. run(['umount',mountpoint],check=True)
  172. async def sign_object(d,fn):
  173. try:
  174. if d.desc == 'transaction':
  175. tx1 = UnsignedTX(filename=fn)
  176. if tx1.proto.sign_mode == 'daemon':
  177. tx1.rpc = await rpc_init(tx1.proto)
  178. tx2 = await txsign(tx1,wfs[:],None,None)
  179. if tx2:
  180. tx2.file.write(ask_write=False)
  181. return tx2
  182. else:
  183. return False
  184. elif d.desc == 'message file':
  185. from .msg import UnsignedMsg,SignedMsg
  186. m = UnsignedMsg(infile=fn)
  187. await m.sign(wallet_files=wfs[:])
  188. m = SignedMsg(data=m.__dict__)
  189. m.write_to_file(
  190. outdir = os.path.abspath(msg_dir),
  191. ask_overwrite = False )
  192. if m.data.get('failed_sids'):
  193. die('MsgFileFailedSID',f'Failed Seed IDs: {fmt_list(m.data["failed_sids"],fmt="bare")}')
  194. return m
  195. except Exception as e:
  196. ymsg(f'An error occurred with {d.desc} {fn!r}:\n {e!s}')
  197. return False
  198. except:
  199. ymsg(f'An error occurred with {d.desc} {fn!r}')
  200. return False
  201. async def sign(target):
  202. td = namedtuple('tdata',['desc','rawext','sigext','dir','fail_desc'])
  203. d = {
  204. 'msg': td('message file', 'rawmsg.json', 'sigmsg.json', msg_dir, 'sign or signed incompletely'),
  205. 'tx': td('transaction', 'rawtx', 'sigtx', tx_dir, 'sign'),
  206. }[target]
  207. raw = [fn[:-len(d.rawext)] for fn in os.listdir(d.dir) if fn.endswith('.'+d.rawext)]
  208. signed = [fn[:-len(d.sigext)] for fn in os.listdir(d.dir) if fn.endswith('.'+d.sigext)]
  209. unsigned = [os.path.join(d.dir,fn+d.rawext) for fn in raw if fn not in signed]
  210. if unsigned:
  211. ok,bad = ([],[])
  212. for fn in unsigned:
  213. ret = await sign_object(d,fn)
  214. if ret:
  215. ok.append(ret)
  216. else:
  217. bad.append(fn)
  218. qmsg('')
  219. await asyncio.sleep(0.3)
  220. msg(f'{len(ok)} {d.desc}{suf(ok)} signed')
  221. if bad:
  222. rmsg(f'{len(bad)} {d.desc}{suf(bad)} failed to {d.fail_desc}')
  223. if ok and not opt.no_summary:
  224. print_summary(d,ok)
  225. if bad:
  226. msg('')
  227. rmsg(f'Failed {d.desc}s:')
  228. def gen_bad_disp():
  229. if d.desc == 'transaction':
  230. for fn in sorted(bad):
  231. yield red(fn)
  232. elif d.desc == 'message file':
  233. for rawfn in sorted(bad):
  234. sigfn = rawfn[:-len(d.rawext)] + d.sigext
  235. yield orange(sigfn) if os.path.exists(sigfn) else red(rawfn)
  236. msg(' {}\n'.format( '\n '.join(gen_bad_disp()) ))
  237. return False if bad else True
  238. else:
  239. msg(f'No unsigned {d.desc}s')
  240. await asyncio.sleep(0.5)
  241. return True
  242. def decrypt_wallets():
  243. msg(f'Unlocking wallet{suf(wfs)} with key from {opt.passwd_file!r}')
  244. fails = 0
  245. for wf in wfs:
  246. try:
  247. Wallet(wf,ignore_in_fmt=True)
  248. except SystemExit as e:
  249. if e.code != 0:
  250. fails += 1
  251. return False if fails else True
  252. def print_summary(d,signed_objects):
  253. if d.desc == 'message file':
  254. gmsg('\nSigned message files:')
  255. for m in signed_objects:
  256. gmsg(' ' + os.path.join(msg_dir,m.signed_filename) )
  257. return
  258. if opt.full_summary:
  259. bmsg('\nAutosign summary:\n')
  260. def gen():
  261. for tx in signed_objects:
  262. yield tx.info.format(terse=True)
  263. msg_r(''.join(gen()))
  264. return
  265. def gen():
  266. for tx in signed_objects:
  267. non_mmgen = [o for o in tx.outputs if not o.mmid]
  268. if non_mmgen:
  269. yield (tx,non_mmgen)
  270. body = list(gen())
  271. if body:
  272. bmsg('\nAutosign summary:')
  273. fs = '{} {} {}'
  274. t_wid,a_wid = 6,44
  275. def gen():
  276. yield fs.format('TX ID ','Non-MMGen outputs'+' '*(a_wid-17),'Amount')
  277. yield fs.format('-'*t_wid, '-'*a_wid, '-'*7)
  278. for tx,non_mmgen in body:
  279. for nm in non_mmgen:
  280. yield fs.format(
  281. tx.txid.fmt(width=t_wid,color=True) if nm is non_mmgen[0] else ' '*t_wid,
  282. nm.addr.fmt(width=a_wid,color=True),
  283. nm.amt.hl() + ' ' + yellow(tx.coin))
  284. msg('\n'.join(gen()))
  285. else:
  286. msg('No non-MMGen outputs')
  287. async def do_sign():
  288. if not opt.stealth_led:
  289. led.set('busy')
  290. do_mount()
  291. key_ok = decrypt_wallets()
  292. if key_ok:
  293. if opt.stealth_led:
  294. led.set('busy')
  295. ret1 = await sign('tx')
  296. ret2 = await sign('msg') if have_msg_dir else True
  297. ret = ret1 and ret2
  298. do_umount()
  299. led.set(('standby','off','error')[(not ret)*2 or bool(opt.stealth_led)])
  300. return ret
  301. else:
  302. msg('Password is incorrect!')
  303. do_umount()
  304. if not opt.stealth_led:
  305. led.set('error')
  306. return False
  307. def wipe_existing_key():
  308. try: os.stat(keyfile)
  309. except: pass
  310. else:
  311. from .fileutil import shred_file
  312. msg(f'\nShredding existing key {keyfile!r}')
  313. shred_file( keyfile, verbose=opt.verbose )
  314. def create_key():
  315. kdata = os.urandom(32).hex()
  316. desc = f'key file {keyfile!r}'
  317. msg('Creating ' + desc)
  318. try:
  319. with open(keyfile,'w') as fp:
  320. fp.write(kdata+'\n')
  321. os.chmod(keyfile,0o400)
  322. msg('Wrote ' + desc)
  323. except:
  324. die(2,'Unable to write ' + desc)
  325. def gen_key(no_unmount=False):
  326. create_wallet_dir()
  327. if not get_insert_status():
  328. die(1,'Removable device not present!')
  329. do_mount()
  330. wipe_existing_key()
  331. create_key()
  332. if not no_unmount:
  333. do_umount()
  334. def remove_wallet_dir():
  335. msg(f'Deleting {wallet_dir!r}')
  336. try: shutil.rmtree(wallet_dir)
  337. except: pass
  338. def create_wallet_dir():
  339. try: os.mkdir(wallet_dir)
  340. except: pass
  341. try: os.stat(wallet_dir)
  342. except: die(2,f'Unable to create wallet directory {wallet_dir!r}')
  343. def setup():
  344. remove_wallet_dir()
  345. gen_key(no_unmount=True)
  346. ss_in = Wallet(in_fmt=mn_fmts[opt.mnemonic_fmt or mn_fmt_dfl])
  347. ss_out = Wallet(ss=ss_in)
  348. ss_out.write_to_file(desc='autosign wallet',outdir=wallet_dir)
  349. def get_insert_status():
  350. if opt.no_insert_check:
  351. return True
  352. try: os.stat(os.path.join('/dev/disk/by-label',part_label))
  353. except: return False
  354. else: return True
  355. async def do_loop():
  356. n,prev_status = 0,False
  357. if not opt.stealth_led:
  358. led.set('standby')
  359. while True:
  360. status = get_insert_status()
  361. if status and not prev_status:
  362. msg('Device insertion detected')
  363. await do_sign()
  364. prev_status = status
  365. if not n % 10:
  366. msg_r(f"\r{' '*17}\rWaiting")
  367. sys.stderr.flush()
  368. await asyncio.sleep(1)
  369. msg_r('.')
  370. n += 1
  371. if len(cmd_args) not in (0,1):
  372. opts.usage()
  373. if len(cmd_args) == 1:
  374. cmd = cmd_args[0]
  375. if cmd in ('gen_key','setup'):
  376. (gen_key if cmd == 'gen_key' else setup)()
  377. sys.exit(0)
  378. elif cmd != 'wait':
  379. die(1,f'{cmd!r}: unrecognized command')
  380. wfs = get_wallet_files()
  381. def at_exit(exit_val,message='\nCleaning up...'):
  382. if message:
  383. msg(message)
  384. led.stop()
  385. sys.exit(exit_val)
  386. def handler(a,b):
  387. at_exit(1)
  388. signal.signal(signal.SIGTERM,handler)
  389. signal.signal(signal.SIGINT,handler)
  390. from .led import LEDControl
  391. led = LEDControl(
  392. enabled = opt.led,
  393. simulate = os.getenv('MMGEN_TEST_SUITE_AUTOSIGN_LED_SIMULATE') )
  394. led.set('off')
  395. async def main():
  396. await check_daemons_running()
  397. if len(cmd_args) == 0:
  398. ret = await do_sign()
  399. at_exit(int(not ret),message='')
  400. elif cmd_args[0] == 'wait':
  401. await do_loop()
  402. async_run(main())