tw.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2019 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. tw: Tracking wallet methods for the MMGen suite
  20. """
  21. from mmgen.common import *
  22. from mmgen.obj import *
  23. from mmgen.tx import is_mmgen_id
  24. CUR_HOME,ERASE_ALL = '\033[H','\033[0J'
  25. def CUR_RIGHT(n): return '\033[{}C'.format(n)
  26. class TwUnspentOutputs(MMGenObject):
  27. def __new__(cls,*args,**kwargs):
  28. return MMGenObject.__new__(altcoin_subclass(cls,'tw','TwUnspentOutputs'))
  29. txid_w = 64
  30. disp_type = 'btc'
  31. can_group = True
  32. hdr_fmt = 'UNSPENT OUTPUTS (sort order: {}) Total {}: {}'
  33. desc = 'unspent outputs'
  34. item_desc = 'unspent output'
  35. dump_fn_pfx = 'listunspent'
  36. prompt_fs = 'Total to spend, excluding fees: {} {}\n\n'
  37. prompt = """
  38. Sort options: [t]xid, [a]mount, a[d]dress, [A]ge, [r]everse, [M]mgen addr
  39. Display options: show [D]ays, [g]roup, show [m]mgen addr, r[e]draw screen
  40. Actions: [q]uit view, [p]rint to file, pager [v]iew, [w]ide view, add [l]abel:
  41. """
  42. key_mappings = {
  43. 't':'s_txid','a':'s_amt','d':'s_addr','A':'s_age','r':'d_reverse','M':'s_twmmid',
  44. 'D':'d_days','g':'d_group','m':'d_mmid','e':'d_redraw',
  45. 'q':'a_quit','p':'a_print','v':'a_view','w':'a_view_wide','l':'a_lbl_add' }
  46. col_adj = 38
  47. class MMGenTwOutputList(list,MMGenObject): pass
  48. class MMGenTwUnspentOutput(MMGenListItem):
  49. # attrs = 'txid','vout','amt','label','twmmid','addr','confs','scriptPubKey','days','skip'
  50. txid = MMGenListItemAttr('txid','CoinTxID')
  51. vout = MMGenListItemAttr('vout',int,typeconv=False)
  52. amt = MMGenImmutableAttr('amt',g.proto.coin_amt.__name__)
  53. amt2 = MMGenListItemAttr('amt2',g.proto.coin_amt.__name__)
  54. label = MMGenListItemAttr('label','TwComment',reassign_ok=True)
  55. twmmid = MMGenImmutableAttr('twmmid','TwMMGenID')
  56. addr = MMGenImmutableAttr('addr','CoinAddr')
  57. confs = MMGenImmutableAttr('confs',int,typeconv=False)
  58. scriptPubKey = MMGenImmutableAttr('scriptPubKey','HexStr')
  59. days = MMGenListItemAttr('days',int,typeconv=False)
  60. skip = MMGenListItemAttr('skip',str,typeconv=False,reassign_ok=True)
  61. wmsg = {
  62. 'no_spendable_outputs': """
  63. No spendable outputs found! Import addresses with balances into your
  64. watch-only wallet using '{}-addrimport' and then re-run this program.
  65. """.strip().format(g.proj_name.lower())
  66. }
  67. def __init__(self,minconf=1):
  68. self.unspent = self.MMGenTwOutputList()
  69. self.fmt_display = ''
  70. self.fmt_print = ''
  71. self.cols = None
  72. self.reverse = False
  73. self.group = False
  74. self.show_mmid = True
  75. self.minconf = minconf
  76. self.get_unspent_data()
  77. self.age_fmt = 'days'
  78. self.sort_key = 'age'
  79. self.do_sort()
  80. self.total = self.get_total_coin()
  81. self.disp_prec = self.get_display_precision()
  82. @property
  83. def age_fmt(self):
  84. return self._age_fmt
  85. @age_fmt.setter
  86. def age_fmt(self,val):
  87. age_fmts = ('days','confs')
  88. if val not in age_fmts:
  89. raise BadAgeFormat("'{}': invalid age format (must be one of {!r})".format(val,age_fmts))
  90. self._age_fmt = val
  91. def get_display_precision(self):
  92. return g.proto.coin_amt.max_prec
  93. def get_total_coin(self):
  94. return sum(i.amt for i in self.unspent)
  95. def get_unspent_rpc(self):
  96. return g.rpch.listunspent(self.minconf)
  97. def get_unspent_data(self):
  98. if g.bogus_wallet_data: # for debugging purposes only
  99. us_rpc = eval(get_data_from_file(g.bogus_wallet_data)) # testing, so ok
  100. else:
  101. us_rpc = self.get_unspent_rpc()
  102. # write_data_to_file('bogus_unspent.json', repr(us), 'bogus unspent data')
  103. # sys.exit(0)
  104. if not us_rpc: die(0,self.wmsg['no_spendable_outputs'])
  105. confs_per_day = 60*60*24 // g.proto.secs_per_block
  106. tr_rpc = []
  107. lbl_id = ('account','label')['label_api' in g.rpch.caps]
  108. for o in us_rpc:
  109. if not lbl_id in o: continue # coinbase outputs have no account field
  110. l = TwLabel(o[lbl_id],on_fail='silent')
  111. if l:
  112. o.update({
  113. 'twmmid': l.mmid,
  114. 'label': l.comment,
  115. 'days': int(o['confirmations'] // confs_per_day),
  116. 'amt': g.proto.coin_amt(o['amount']),
  117. 'addr': CoinAddr(o['address']),
  118. 'confs': o['confirmations']
  119. })
  120. tr_rpc.append(o)
  121. self.unspent = self.MMGenTwOutputList(
  122. self.MMGenTwUnspentOutput(
  123. **{k:v for k,v in o.items() if k in dir(self.MMGenTwUnspentOutput)}
  124. ) for o in tr_rpc)
  125. for u in self.unspent:
  126. if u.label == None: u.label = ''
  127. if not self.unspent:
  128. die(1,'No tracked {}s in tracking wallet!'.format(self.item_desc))
  129. def do_sort(self,key=None,reverse=False):
  130. sort_funcs = {
  131. 'addr': lambda i: i.addr,
  132. 'age': lambda i: 0 - i.confs,
  133. 'amt': lambda i: i.amt,
  134. 'txid': lambda i: '{} {:04}'.format(i.txid,i.vout),
  135. 'twmmid': lambda i: i.twmmid.sort_key
  136. }
  137. key = key or self.sort_key
  138. if key not in sort_funcs:
  139. die(1,"'{}': invalid sort key. Valid options: {}".format(key,' '.join(sort_funcs.keys())))
  140. self.sort_key = key
  141. assert type(reverse) == bool
  142. self.unspent.sort(key=sort_funcs[key],reverse=reverse or self.reverse)
  143. def sort_info(self,include_group=True):
  144. ret = ([],['Reverse'])[self.reverse]
  145. ret.append(capfirst(self.sort_key).replace('Twmmid','MMGenID'))
  146. if include_group and self.group and (self.sort_key in ('addr','txid','twmmid')):
  147. ret.append('Grouped')
  148. return ret
  149. def set_term_columns(self):
  150. from mmgen.term import get_terminal_size
  151. while True:
  152. self.cols = g.terminal_width or get_terminal_size()[0]
  153. if self.cols >= g.min_screen_width: break
  154. m1 = 'Screen too narrow to display the tracking wallet\n'
  155. m2 = 'Please resize your screen to at least {} characters and hit ENTER '
  156. my_raw_input((m1+m2).format(g.min_screen_width))
  157. def format_for_display(self):
  158. unsp = self.unspent
  159. self.set_term_columns()
  160. # allow for 7-digit confirmation nums
  161. col1_w = max(3,len(str(len(unsp)))+1) # num + ')'
  162. mmid_w = max(len(('',i.twmmid)[i.twmmid.type=='mmgen']) for i in unsp) or 12 # DEADBEEF:S:1
  163. max_acct_w = max(i.label.screen_width for i in unsp) + mmid_w + 1
  164. max_btcaddr_w = max(len(i.addr) for i in unsp)
  165. min_addr_w = self.cols - self.col_adj
  166. addr_w = min(max_btcaddr_w + (0,1+max_acct_w)[self.show_mmid],min_addr_w)
  167. acct_w = min(max_acct_w, max(24,addr_w-10))
  168. btaddr_w = addr_w - acct_w - 1
  169. label_w = acct_w - mmid_w - 1
  170. tx_w = min(self.txid_w,self.cols-addr_w-28-col1_w) # min=7
  171. txdots = ('','..')[tx_w < self.txid_w]
  172. for i in unsp: i.skip = ''
  173. if self.group and (self.sort_key in ('addr','txid','twmmid')):
  174. for a,b in [(unsp[i],unsp[i+1]) for i in range(len(unsp)-1)]:
  175. for k in ('addr','txid','twmmid'):
  176. if self.sort_key == k and getattr(a,k) == getattr(b,k):
  177. b.skip = (k,'addr')[k=='twmmid']
  178. out = [self.hdr_fmt.format(' '.join(self.sort_info()),g.dcoin,self.total.hl())]
  179. if g.chain != 'mainnet': out += ['Chain: '+green(g.chain.upper())]
  180. fs = { 'btc': ' {n:%s} {t:%s} {v:2} {a} {A} {c:<}' % (col1_w,tx_w),
  181. 'eth': ' {n:%s} {a} {A}' % col1_w,
  182. 'token': ' {n:%s} {a} {A} {A2}' % col1_w }[self.disp_type]
  183. out += [fs.format( n='Num',
  184. t='TXid'.ljust(tx_w - 5) + ' Vout',
  185. v='',
  186. a='Address'.ljust(addr_w),
  187. A='Amt({})'.format(g.dcoin).ljust(self.disp_prec+3),
  188. A2=' Amt({})'.format(g.coin).ljust(self.disp_prec+4),
  189. c=('Confs','Age(d)')[self.age_fmt=='days']
  190. ).rstrip()]
  191. for n,i in enumerate(unsp):
  192. addr_dots = '|' + '.'*(addr_w-1)
  193. mmid_disp = MMGenID.fmtc('.'*mmid_w if i.skip=='addr'
  194. else i.twmmid if i.twmmid.type=='mmgen'
  195. else 'Non-{}'.format(g.proj_name),width=mmid_w,color=True)
  196. if self.show_mmid:
  197. addr_out = '{} {}'.format(
  198. type(i.addr).fmtc(addr_dots,width=btaddr_w,color=True) if i.skip == 'addr' \
  199. else i.addr.fmt(width=btaddr_w,color=True),
  200. '{} {}'.format(mmid_disp,i.label.fmt(width=label_w,color=True) \
  201. if label_w > 0 else ''))
  202. else:
  203. addr_out = type(i.addr).fmtc(addr_dots,width=addr_w,color=True) \
  204. if i.skip=='addr' else i.addr.fmt(width=addr_w,color=True)
  205. out.append(fs.format( n=str(n+1)+')',
  206. t='' if not i.txid else \
  207. ' ' * (tx_w-4) + '|...' if i.skip == 'txid' \
  208. else i.txid[:tx_w-len(txdots)] + txdots,
  209. v=i.vout,
  210. a=addr_out,
  211. A=i.amt.fmt(color=True,prec=self.disp_prec),
  212. A2=(i.amt2.fmt(color=True,prec=self.disp_prec) if i.amt2 is not None else ''),
  213. c=i.days if self.age_fmt == 'days' else i.confs
  214. ).rstrip())
  215. self.fmt_display = '\n'.join(out) + '\n'
  216. return self.fmt_display
  217. def format_for_printing(self,color=False):
  218. addr_w = max(len(i.addr) for i in self.unspent)
  219. mmid_w = max(len(('',i.twmmid)[i.twmmid.type=='mmgen']) for i in self.unspent) or 12 # DEADBEEF:S:1
  220. amt_w = g.proto.coin_amt.max_prec + 4
  221. fs = { 'btc': ' {n:4} {t:%s} {a} {m} {A:%s} {c:<8} {g:<6} {l}' % (self.txid_w+3,amt_w),
  222. 'eth': ' {n:4} {a} {m} {A:%s} {l}' % amt_w,
  223. 'token': ' {n:4} {a} {m} {A:%s} {A2:%s} {l}' % (amt_w,amt_w)
  224. }[self.disp_type]
  225. out = [fs.format( n='Num',
  226. t='Tx ID,Vout',
  227. a='Address'.ljust(addr_w),
  228. m='MMGen ID'.ljust(mmid_w+1),
  229. A='Amount({})'.format(g.dcoin).ljust(amt_w+1),
  230. A2='Amount({})'.format(g.coin),
  231. c='Confs', # skipped for eth
  232. g='Age(d)', # skipped for eth
  233. l='Label')]
  234. max_lbl_len = max([len(i.label) for i in self.unspent if i.label] or [2])
  235. for n,i in enumerate(self.unspent):
  236. addr = '|'+'.' * addr_w if i.skip == 'addr' and self.group else i.addr.fmt(color=color,width=addr_w)
  237. out.append(fs.format(
  238. n=str(n+1)+')',
  239. t='{},{}'.format('|'+'.'*63 if i.skip == 'txid' and self.group else i.txid,i.vout),
  240. a=addr,
  241. m=MMGenID.fmtc(i.twmmid if i.twmmid.type=='mmgen'
  242. else 'Non-{}'.format(g.proj_name),width=mmid_w,color=color),
  243. A=i.amt.fmt(color=color),
  244. A2=(i.amt2.fmt(color=color) if i.amt2 is not None else ''),
  245. c=i.confs,
  246. g=i.days,
  247. l=i.label.hl(color=color) if i.label else
  248. TwComment.fmtc('',color=color,nullrepl='-',width=max_lbl_len)).rstrip())
  249. fs = '{} ({} UTC)\nSort order: {}\n{}\n\nTotal {}: {}\n'
  250. self.fmt_print = fs.format(
  251. capfirst(self.desc),
  252. make_timestr(),
  253. ' '.join(self.sort_info(include_group=False)),
  254. '\n'.join(out),
  255. g.dcoin,
  256. self.total.hl(color=color))
  257. return self.fmt_print
  258. def display_total(self):
  259. fs = '\nTotal unspent: {} {} ({} outputs)'
  260. msg(fs.format(self.total.hl(),g.dcoin,len(self.unspent)))
  261. def get_idx_from_user(self,get_label=False):
  262. msg('')
  263. while True:
  264. ret = my_raw_input('Enter {} number (or RETURN to return to main menu): '.format(self.item_desc))
  265. if ret == '': return (None,None) if get_label else None
  266. n = AddrIdx(ret,on_fail='silent') # hacky way to test and convert to integer
  267. if not n or n < 1 or n > len(self.unspent):
  268. msg('Choice must be a single number between 1 and {}'.format(len(self.unspent)))
  269. # elif not self.unspent[n-1].mmid:
  270. # msg('Address #{} is not an {} address. No label can be added to it'.format(n,g.proj_name))
  271. else:
  272. if get_label:
  273. while True:
  274. s = my_raw_input("Enter label text (or 'q' to return to main menu): ")
  275. if s == 'q':
  276. return None,None
  277. elif s == '':
  278. fs = "Removing label for {} #{}. Is this what you want?"
  279. if keypress_confirm(fs.format(self.item_desc,n)):
  280. return n,s
  281. elif s:
  282. if TwComment(s,on_fail='return'):
  283. return n,s
  284. else:
  285. fs = "Removing {} #{} from tracking wallet. Is this what you want?"
  286. if keypress_confirm(fs.format(self.item_desc,n)):
  287. return n
  288. def view_and_sort(self,tx):
  289. from mmgen.term import get_char
  290. prompt = self.prompt.strip() + '\b'
  291. no_output,oneshot_msg = False,None
  292. while True:
  293. msg_r('' if no_output else '\n\n' if opt.no_blank else CUR_HOME+ERASE_ALL)
  294. reply = get_char('' if no_output else self.format_for_display()+'\n'+(oneshot_msg or '')+prompt,
  295. immed_chars=''.join(self.key_mappings.keys())).decode()
  296. no_output = False
  297. oneshot_msg = '' if oneshot_msg else None # tristate, saves previous state
  298. if reply not in self.key_mappings:
  299. msg_r('\ninvalid keypress ')
  300. time.sleep(0.5)
  301. continue
  302. action = self.key_mappings[reply]
  303. if action[:2] == 's_':
  304. self.do_sort(action[2:])
  305. if action == 's_twmmid': self.show_mmid = True
  306. elif action == 'd_days': self.age_fmt = ('days','confs')[self.age_fmt=='days']
  307. elif action == 'd_mmid': self.show_mmid = not self.show_mmid
  308. elif action == 'd_group':
  309. if self.can_group:
  310. self.group = not self.group
  311. elif action == 'd_redraw': pass
  312. elif action == 'd_reverse': self.unspent.reverse(); self.reverse = not self.reverse
  313. elif action == 'a_quit': msg(''); return self.unspent
  314. elif action == 'a_lbl_add':
  315. idx,lbl = self.get_idx_from_user(get_label=True)
  316. if idx:
  317. e = self.unspent[idx-1]
  318. if TrackingWallet(mode='w').add_label(e.twmmid,lbl,addr=e.addr):
  319. self.get_unspent_data()
  320. self.do_sort()
  321. a = 'added to' if lbl else 'removed from'
  322. oneshot_msg = yellow("Label {} {} #{}\n\n".format(a,self.item_desc,idx))
  323. else:
  324. oneshot_msg = red('Label could not be added\n\n')
  325. elif action == 'a_addr_remove':
  326. idx = self.get_idx_from_user()
  327. if idx:
  328. e = self.unspent[idx-1]
  329. if TrackingWallet(mode='w').remove_address(e.addr):
  330. self.get_unspent_data()
  331. self.do_sort()
  332. self.total = self.get_total_coin()
  333. oneshot_msg = yellow("{} #{} removed\n\n".format(capfirst(self.item_desc),idx))
  334. else:
  335. oneshot_msg = red('Address could not be removed\n\n')
  336. elif action == 'a_print':
  337. of = '{}-{}[{}].out'.format(self.dump_fn_pfx,g.dcoin,
  338. ','.join(self.sort_info(include_group=False)).lower())
  339. msg('')
  340. try:
  341. write_data_to_file(of,self.format_for_printing(),desc='{} listing'.format(self.desc))
  342. except UserNonConfirmation as e:
  343. oneshot_msg = red("File '{}' not overwritten by user request\n\n".format(of))
  344. else:
  345. oneshot_msg = yellow("Data written to '{}'\n\n".format(of))
  346. elif action in ('a_view','a_view_wide'):
  347. do_pager(self.fmt_display if action == 'a_view' else self.format_for_printing(color=True))
  348. if g.platform == 'linux' and oneshot_msg == None:
  349. msg_r(CUR_RIGHT(len(prompt.split('\n')[-1])-2))
  350. no_output = True
  351. class TwAddrList(MMGenDict):
  352. def __new__(cls,*args,**kwargs):
  353. return MMGenDict.__new__(altcoin_subclass(cls,'tw','TwAddrList'),*args,**kwargs)
  354. def __init__(self,usr_addr_list,minconf,showempty,showbtcaddrs,all_labels):
  355. def check_dup_mmid(acct_labels):
  356. mmid_prev,err = None,False
  357. for mmid in sorted(a.mmid for a in acct_labels if a):
  358. if mmid == mmid_prev:
  359. err = True
  360. msg('Duplicate MMGen ID ({}) discovered in tracking wallet!\n'.format(mmid))
  361. mmid_prev = mmid
  362. if err: rdie(3,'Tracking wallet is corrupted!')
  363. def check_addr_array_lens(acct_pairs):
  364. err = False
  365. for label,addrs in acct_pairs:
  366. if not label: continue
  367. if len(addrs) != 1:
  368. err = True
  369. if len(addrs) == 0:
  370. msg("Label '{}': has no associated address!".format(label))
  371. else:
  372. msg("'{}': more than one {} address in account!".format(addrs,g.coin))
  373. if err: rdie(3,'Tracking wallet is corrupted!')
  374. self.total = g.proto.coin_amt('0')
  375. rpc_init()
  376. lbl_id = ('account','label')['label_api' in g.rpch.caps]
  377. for d in g.rpch.listunspent(0):
  378. if not lbl_id in d: continue # skip coinbase outputs with missing account
  379. if d['confirmations'] < minconf: continue
  380. label = TwLabel(d[lbl_id],on_fail='silent')
  381. if label:
  382. if usr_addr_list and (label.mmid not in usr_addr_list): continue
  383. if label.mmid in self:
  384. if self[label.mmid]['addr'] != d['address']:
  385. die(2,'duplicate {} address ({}) for this MMGen address! ({})'.format(
  386. g.coin,d['address'],self[label.mmid]['addr']))
  387. else:
  388. self[label.mmid] = {'amt': g.proto.coin_amt('0'),
  389. 'lbl': label,
  390. 'addr': CoinAddr(d['address'])}
  391. self[label.mmid]['lbl'].mmid.confs = d['confirmations']
  392. self[label.mmid]['amt'] += d['amount']
  393. self.total += d['amount']
  394. # We use listaccounts only for empty addresses, as it shows false positive balances
  395. if showempty or all_labels:
  396. # for compatibility with old mmids, must use raw RPC rather than native data for matching
  397. # args: minconf,watchonly, MUST use keys() so we get list, not dict
  398. if 'label_api' in g.rpch.caps:
  399. acct_list = g.rpch.listlabels()
  400. acct_addrs = [list(a.keys()) for a in g.rpch.getaddressesbylabel([[k] for k in acct_list],batch=True)]
  401. else:
  402. acct_list = list(g.rpch.listaccounts(0,True).keys()) # raw list, no 'L'
  403. acct_addrs = g.rpch.getaddressesbyaccount([[a] for a in acct_list],batch=True) # use raw list here
  404. acct_labels = MMGenList([TwLabel(a,on_fail='silent') for a in acct_list])
  405. check_dup_mmid(acct_labels)
  406. assert len(acct_list) == len(acct_addrs),(
  407. 'listaccounts() and getaddressesbyaccount() not equal in length')
  408. addr_pairs = list(zip(acct_labels,acct_addrs))
  409. check_addr_array_lens(addr_pairs)
  410. for label,addr_arr in addr_pairs:
  411. if not label: continue
  412. if all_labels and not showempty and not label.comment: continue
  413. if usr_addr_list and (label.mmid not in usr_addr_list): continue
  414. if label.mmid not in self:
  415. self[label.mmid] = { 'amt':g.proto.coin_amt('0'), 'lbl':label, 'addr':'' }
  416. if showbtcaddrs:
  417. self[label.mmid]['addr'] = CoinAddr(addr_arr[0])
  418. def raw_list(self):
  419. return [((k if k.type == 'mmgen' else 'Non-MMGen'),self[k]['addr'],self[k]['amt']) for k in self]
  420. def coinaddr_list(self): return [self[k]['addr'] for k in self]
  421. def format(self,showbtcaddrs,sort,show_age,age_fmt):
  422. age_fmts = ('days','confs')
  423. if age_fmt not in age_fmts:
  424. raise BadAgeFormat("'{}': invalid age format (must be one of {!r})".format(age_fmt,age_fmts))
  425. out = ['Chain: '+green(g.chain.upper())] if g.chain != 'mainnet' else []
  426. fs = '{{mid}}{} {{cmt}} {{amt}}{}'.format(('',' {addr}')[showbtcaddrs],('',' {age}')[show_age])
  427. mmaddrs = [k for k in self.keys() if k.type == 'mmgen']
  428. max_mmid_len = max(len(k) for k in mmaddrs) + 2 if mmaddrs else 10
  429. max_cmt_width = max(max(v['lbl'].comment.screen_width for v in self.values()),7)
  430. addr_width = max(len(self[mmid]['addr']) for mmid in self)
  431. # fp: fractional part
  432. max_fp_len = max([len(a.split('.')[1]) for a in [str(v['amt']) for v in self.values()] if '.' in a] or [1])
  433. out += [fs.format(
  434. mid=MMGenID.fmtc('MMGenID',width=max_mmid_len),
  435. addr=(CoinAddr.fmtc('ADDRESS',width=addr_width) if showbtcaddrs else None),
  436. cmt=TwComment.fmtc('COMMENT',width=max_cmt_width+1),
  437. amt='BALANCE'.ljust(max_fp_len+4),
  438. age=('CONFS','DAYS')[age_fmt=='days'],
  439. )]
  440. def sort_algo(j):
  441. if sort and 'age' in sort:
  442. return '{}_{:>012}_{}'.format(
  443. j.obj.rsplit(':',1)[0],
  444. # Hack, but OK for the foreseeable future:
  445. (1000000000-j.confs if hasattr(j,'confs') and j.confs != None else 0),
  446. j.sort_key)
  447. else:
  448. return j.sort_key
  449. al_id_save = None
  450. confs_per_day = 60*60*24 // g.proto.secs_per_block
  451. for mmid in sorted(self,key=sort_algo,reverse=bool(sort and 'reverse' in sort)):
  452. if mmid.type == 'mmgen':
  453. if al_id_save and al_id_save != mmid.obj.al_id:
  454. out.append('')
  455. al_id_save = mmid.obj.al_id
  456. mmid_disp = mmid
  457. else:
  458. if al_id_save:
  459. out.append('')
  460. al_id_save = None
  461. mmid_disp = 'Non-MMGen'
  462. e = self[mmid]
  463. out.append(fs.format(
  464. mid=MMGenID.fmtc(mmid_disp,width=max_mmid_len,color=True),
  465. addr=(e['addr'].fmt(color=True,width=addr_width) if showbtcaddrs else None),
  466. cmt=e['lbl'].comment.fmt(width=max_cmt_width,color=True,nullrepl='-'),
  467. amt=e['amt'].fmt('4.{}'.format(max(max_fp_len,3)),color=True),
  468. age=mmid.confs // (1,confs_per_day)[age_fmt=='days'] if hasattr(mmid,'confs') and mmid.confs != None else '-'
  469. ))
  470. return '\n'.join(out + ['\nTOTAL: {} {}'.format(self.total.hl(color=True),g.dcoin)])
  471. class TrackingWallet(MMGenObject):
  472. def __new__(cls,*args,**kwargs):
  473. return MMGenObject.__new__(altcoin_subclass(cls,'tw','TrackingWallet'))
  474. mode = 'r'
  475. caps = ('rescan','batch')
  476. def __init__(self,mode='r'):
  477. m = "'{}': invalid 'mode' parameter for {} constructor"
  478. assert mode in ('r','w'),m.format(mode,type(self).__name__)
  479. self.mode = mode
  480. @write_mode
  481. def import_address(self,addr,label,rescan):
  482. return g.rpch.importaddress(addr,label,rescan,timeout=(False,3600)[rescan])
  483. @write_mode
  484. def batch_import_address(self,arg_list):
  485. return g.rpch.importaddress(arg_list,batch=True)
  486. @write_mode
  487. def write(self): pass
  488. def is_in_wallet(self,addr):
  489. return addr in TwAddrList([],0,True,True,True).coinaddr_list()
  490. @write_mode
  491. def set_label(self,coinaddr,lbl):
  492. if 'label_api' in g.rpch.caps:
  493. return g.rpch.setlabel(coinaddr,lbl,on_fail='return')
  494. else:
  495. # NOTE: this works because importaddress() removes the old account before
  496. # associating the new account with the address.
  497. # RPC args: addr,label,rescan[=true],p2sh[=none]
  498. return g.rpch.importaddress(coinaddr,lbl,False,on_fail='return')
  499. # returns on failure
  500. @write_mode
  501. def add_label(self,arg1,label='',addr=None,silent=False,on_fail='return'):
  502. from mmgen.tx import is_mmgen_id,is_coin_addr
  503. mmaddr,coinaddr = None,None
  504. if is_coin_addr(addr or arg1):
  505. coinaddr = CoinAddr(addr or arg1,on_fail='return')
  506. if is_mmgen_id(arg1):
  507. mmaddr = TwMMGenID(arg1)
  508. if mmaddr and not coinaddr:
  509. from mmgen.addr import AddrData
  510. coinaddr = AddrData(source='tw').mmaddr2coinaddr(mmaddr)
  511. try:
  512. if not is_mmgen_id(arg1):
  513. assert coinaddr,"Invalid coin address for this chain: {}".format(arg1)
  514. assert coinaddr,"{pn} address '{ma}' not found in tracking wallet"
  515. assert self.is_in_wallet(coinaddr),"Address '{ca}' not found in tracking wallet"
  516. except Exception as e:
  517. msg(e.args[0].format(pn=g.proj_name,ma=mmaddr,ca=coinaddr))
  518. return False
  519. # Allow for the possibility that BTC addr of MMGen addr was entered.
  520. # Do reverse lookup, so that MMGen addr will not be marked as non-MMGen.
  521. if not mmaddr:
  522. from mmgen.addr import AddrData
  523. mmaddr = AddrData(source='tw').coinaddr2mmaddr(coinaddr)
  524. if not mmaddr: mmaddr = '{}:{}'.format(g.proto.base_coin.lower(),coinaddr)
  525. mmaddr = TwMMGenID(mmaddr)
  526. cmt = TwComment(label,on_fail=on_fail)
  527. if cmt in (False,None): return False
  528. lbl = TwLabel(mmaddr + ('',' '+cmt)[bool(cmt)],on_fail=on_fail)
  529. ret = self.set_label(coinaddr,lbl)
  530. from mmgen.rpc import rpc_error,rpc_errmsg
  531. if rpc_error(ret):
  532. msg('From {}: {}'.format(g.proto.daemon_name,rpc_errmsg(ret)))
  533. if not silent:
  534. msg('Label could not be {}'.format(('removed','added')[bool(label)]))
  535. return False
  536. else:
  537. m = mmaddr.type.replace('mmg','MMG')
  538. a = mmaddr.replace(g.proto.base_coin.lower()+':','')
  539. s = '{} address {} in tracking wallet'.format(m,a)
  540. if label: msg("Added label '{}' to {}".format(label,s))
  541. else: msg('Removed label from {}'.format(s))
  542. return True
  543. @write_mode
  544. def remove_label(self,mmaddr):
  545. self.add_label(mmaddr,'')
  546. @write_mode
  547. def remove_address(self,addr):
  548. raise NotImplementedError('address removal not implemented for coin {}'.format(g.coin))
  549. class TwGetBalance(MMGenObject):
  550. fs = '{w:13} {u:<16} {p:<16} {c}\n'
  551. def __new__(cls,*args,**kwargs):
  552. return MMGenObject.__new__(altcoin_subclass(cls,'tw','TwGetBalance'))
  553. def __init__(self,minconf,quiet):
  554. rpc_init()
  555. self.minconf = minconf
  556. self.quiet = quiet
  557. self.data = {k:[g.proto.coin_amt('0')] * 4 for k in ('TOTAL','Non-MMGen','Non-wallet')}
  558. self.create_data()
  559. def create_data(self):
  560. # 0: unconfirmed, 1: below minconf, 2: confirmed, 3: spendable (privkey in wallet)
  561. lbl_id = ('account','label')['label_api' in g.rpch.caps]
  562. for d in g.rpch.listunspent(0):
  563. try:
  564. lbl = TwLabel(d[lbl_id],on_fail='silent')
  565. except:
  566. lbl,key = None,'Non-wallet'
  567. else:
  568. if lbl.mmid.type == 'mmgen':
  569. key = lbl.mmid.obj.sid
  570. if key not in self.data:
  571. self.data[key] = [g.proto.coin_amt('0')] * 4
  572. else:
  573. key = 'Non-MMGen'
  574. if not d['confirmations']:
  575. self.data['TOTAL'][0] += d['amount']
  576. self.data[key][0] += d['amount']
  577. conf_level = (1,2)[d['confirmations'] >= self.minconf]
  578. self.data['TOTAL'][conf_level] += d['amount']
  579. self.data[key][conf_level] += d['amount']
  580. if d['spendable']:
  581. self.data[key][3] += d['amount']
  582. def format(self):
  583. if self.quiet:
  584. o = str(self.data['TOTAL'][2] if self.data else 0) + '\n'
  585. else:
  586. o = self.fs.format( w='Wallet',
  587. u=' Unconfirmed',
  588. p=' <{} confirms'.format(self.minconf),
  589. c=' >={} confirms'.format(self.minconf))
  590. for key in sorted(self.data):
  591. if not any(self.data[key]): continue
  592. o += self.fs.format(**dict(zip(
  593. ('w','u','p','c'),
  594. [key+':'] + [a.fmt(color=True,suf=' '+g.dcoin) for a in self.data[key]]
  595. )))
  596. for key,vals in list(self.data.items()):
  597. if key == 'TOTAL': continue
  598. if vals[3]:
  599. o += red('Warning: this wallet contains PRIVATE KEYS for {} outputs!\n'.format(key))
  600. return o