2014-12-29 17:08:13 +00:00
|
|
|
#!/bin/python
|
|
|
|
#
|
|
|
|
# Cryptonote tipbot - utility functions
|
2015-01-01 17:33:07 +00:00
|
|
|
# Copyright 2014,2015 moneromooo
|
2014-12-29 17:08:13 +00:00
|
|
|
#
|
|
|
|
# The Cryptonote tipbot is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License as published
|
|
|
|
# by the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
# any later version.
|
|
|
|
#
|
|
|
|
|
|
|
|
import redis
|
|
|
|
import hashlib
|
|
|
|
import json
|
|
|
|
import httplib
|
2015-01-25 21:09:56 +00:00
|
|
|
import time
|
2015-02-05 19:24:15 +00:00
|
|
|
import threading
|
2015-01-31 12:19:56 +00:00
|
|
|
import math
|
2015-02-02 20:44:03 +00:00
|
|
|
import string
|
2015-01-31 12:19:56 +00:00
|
|
|
from decimal import *
|
2014-12-29 17:08:13 +00:00
|
|
|
import tipbot.config as config
|
|
|
|
import tipbot.coinspecs as coinspecs
|
|
|
|
from tipbot.log import log_error, log_warn, log_info, log_log
|
2015-01-31 23:40:13 +00:00
|
|
|
from tipbot.link import Link
|
2014-12-29 17:08:13 +00:00
|
|
|
from tipbot.redisdb import *
|
|
|
|
|
2015-01-26 22:18:18 +00:00
|
|
|
registered_networks=dict()
|
2015-01-13 12:28:05 +00:00
|
|
|
networks=[]
|
2015-01-25 21:09:56 +00:00
|
|
|
cached_tipbot_balance=None
|
|
|
|
cached_tipbot_unlocked_balance=None
|
|
|
|
cached_tipbot_balance_timestamp=None
|
2014-12-29 17:08:13 +00:00
|
|
|
|
2015-02-05 21:40:42 +00:00
|
|
|
core_lock = threading.RLock()
|
2015-02-05 19:24:15 +00:00
|
|
|
|
2015-01-26 22:18:18 +00:00
|
|
|
def GetPassword(name):
|
2014-12-29 17:08:13 +00:00
|
|
|
try:
|
|
|
|
f = open('tipbot-password.txt', 'r')
|
|
|
|
for p in f:
|
|
|
|
p = p.strip("\r\n")
|
2015-01-26 22:18:18 +00:00
|
|
|
parts=p.split(':')
|
|
|
|
if parts[0]==name:
|
|
|
|
return parts[1]
|
2014-12-29 17:08:13 +00:00
|
|
|
except Exception,e:
|
|
|
|
log_error('could not fetch password: %s' % str(e))
|
|
|
|
raise
|
|
|
|
return "xxx"
|
2015-01-26 22:18:18 +00:00
|
|
|
finally:
|
|
|
|
f.close()
|
2014-12-29 17:08:13 +00:00
|
|
|
|
|
|
|
def IsParamPresent(parms,idx):
|
|
|
|
return len(parms) > idx
|
|
|
|
|
|
|
|
def GetParam(parms,idx):
|
|
|
|
if IsParamPresent(parms,idx):
|
|
|
|
return parms[idx]
|
|
|
|
return None
|
|
|
|
|
2015-01-13 12:28:05 +00:00
|
|
|
def GetPaymentID(link):
|
2014-12-29 17:08:13 +00:00
|
|
|
salt="2u3g55bkwrui32fi3g4bGR$j5g4ugnujb-"+coinspecs.name+"-";
|
2015-01-13 12:28:05 +00:00
|
|
|
p = hashlib.sha256(salt+link.identity()).hexdigest();
|
2014-12-29 17:08:13 +00:00
|
|
|
try:
|
2015-01-13 12:28:05 +00:00
|
|
|
redis_hset("paymentid",p,link.identity())
|
2014-12-29 17:08:13 +00:00
|
|
|
except Exception,e:
|
2015-01-13 12:28:05 +00:00
|
|
|
log_error('GetPaymentID: failed to set payment ID for %s to redis: %s' % (link.identity(),str(e)))
|
2014-12-29 17:08:13 +00:00
|
|
|
return p
|
|
|
|
|
2015-01-13 12:28:05 +00:00
|
|
|
def GetIdentityFromPaymentID(p):
|
|
|
|
if not redis_hexists("paymentid",p):
|
|
|
|
log_log('PaymentID %s not found' % p)
|
|
|
|
return None
|
|
|
|
identity = redis_hget("paymentid",p)
|
|
|
|
log_log('PaymentID %s => %s' % (p, str(identity)))
|
|
|
|
# HACK - grandfathering pre-network payment IDs
|
|
|
|
if identity.index(':') == -1:
|
|
|
|
log_warn('Pre-network payment ID found, assuming freenode')
|
|
|
|
identity = "freenode:"+identity
|
|
|
|
return identity
|
2014-12-29 17:08:13 +00:00
|
|
|
|
2015-01-11 14:08:12 +00:00
|
|
|
def IsValidAddress(address):
|
|
|
|
if len(address) < coinspecs.address_length[0] or len(address) > coinspecs.address_length[1]:
|
|
|
|
return False
|
|
|
|
for prefix in coinspecs.address_prefix:
|
|
|
|
if address.startswith(prefix):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2015-02-02 20:44:03 +00:00
|
|
|
def IsValidPaymentID(payment_id):
|
|
|
|
if len(payment_id)!=64:
|
|
|
|
return False
|
|
|
|
for char in payment_id:
|
|
|
|
if char not in string.hexdigits:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2015-01-31 12:19:56 +00:00
|
|
|
# Code taken from the Python documentation
|
|
|
|
def moneyfmt(value, places=2, curr='', sep=',', dp='.',
|
|
|
|
pos='', neg='-', trailneg=''):
|
|
|
|
"""Convert Decimal to a money formatted string.
|
|
|
|
|
|
|
|
places: required number of places after the decimal point
|
|
|
|
curr: optional currency symbol before the sign (may be blank)
|
|
|
|
sep: optional grouping separator (comma, period, space, or blank)
|
|
|
|
dp: decimal point indicator (comma or period)
|
|
|
|
only specify as blank when places is zero
|
|
|
|
pos: optional sign for positive numbers: '+', space or blank
|
|
|
|
neg: optional sign for negative numbers: '-', '(', space or blank
|
|
|
|
trailneg:optional trailing minus indicator: '-', ')', space or blank
|
|
|
|
|
|
|
|
>>> d = Decimal('-1234567.8901')
|
|
|
|
>>> moneyfmt(d, curr='$')
|
|
|
|
'-$1,234,567.89'
|
|
|
|
>>> moneyfmt(d, places=0, sep='.', dp='', neg='', trailneg='-')
|
|
|
|
'1.234.568-'
|
|
|
|
>>> moneyfmt(d, curr='$', neg='(', trailneg=')')
|
|
|
|
'($1,234,567.89)'
|
|
|
|
>>> moneyfmt(Decimal(123456789), sep=' ')
|
|
|
|
'123 456 789.00'
|
|
|
|
>>> moneyfmt(Decimal('-0.02'), neg='<', trailneg='>')
|
|
|
|
'<0.02>'
|
|
|
|
|
|
|
|
"""
|
|
|
|
q = Decimal(10) ** -places # 2 places --> '0.01'
|
|
|
|
sign, digits, exp = value.quantize(q).as_tuple()
|
|
|
|
result = []
|
|
|
|
digits = map(str, digits)
|
|
|
|
build, next = result.append, digits.pop
|
|
|
|
if sign:
|
|
|
|
build(trailneg)
|
|
|
|
for i in range(places):
|
|
|
|
build(next() if digits else '0')
|
|
|
|
build(dp)
|
|
|
|
if not digits:
|
|
|
|
build('0')
|
|
|
|
i = 0
|
|
|
|
while digits:
|
|
|
|
build(next())
|
|
|
|
i += 1
|
|
|
|
if i == 3 and digits:
|
|
|
|
i = 0
|
|
|
|
build(sep)
|
|
|
|
build(curr)
|
|
|
|
build(neg if sign else pos)
|
|
|
|
s = ''.join(reversed(result))
|
|
|
|
|
|
|
|
if dp in s:
|
|
|
|
s=s.strip('0').rstrip(dp)
|
|
|
|
if s=="" or s[0]==dp:
|
|
|
|
s="0"+s
|
|
|
|
return s
|
|
|
|
|
2014-12-29 17:08:13 +00:00
|
|
|
def AmountToString(amount):
|
|
|
|
if amount == None:
|
|
|
|
amount = 0
|
|
|
|
lamount=long(amount)
|
|
|
|
samount = None
|
|
|
|
if lamount == 0:
|
|
|
|
samount = "0 %s" % coinspecs.name
|
|
|
|
else:
|
2015-01-31 12:19:56 +00:00
|
|
|
places=long(0.5+math.log10(coinspecs.atomic_units))
|
2014-12-29 17:08:13 +00:00
|
|
|
for den in coinspecs.denominations:
|
|
|
|
if lamount < den[0]:
|
2015-01-31 12:19:56 +00:00
|
|
|
samount = moneyfmt(Decimal(lamount)/Decimal(den[1]),places=places) + " " + den[2]
|
2014-12-29 17:08:13 +00:00
|
|
|
break
|
|
|
|
if not samount:
|
2015-01-31 12:19:56 +00:00
|
|
|
samount = moneyfmt(Decimal(lamount)/Decimal(coinspecs.atomic_units),places=places) + " " + coinspecs.name
|
2014-12-29 17:08:13 +00:00
|
|
|
return samount
|
|
|
|
|
2015-01-18 22:18:46 +00:00
|
|
|
def TimeToString(seconds):
|
|
|
|
seconds=float(seconds)
|
|
|
|
if seconds < 1e-3:
|
|
|
|
return "%.2f microseconds" % (seconds*1e6)
|
|
|
|
if seconds < 1:
|
|
|
|
return "%.2f milliseconds" % (seconds*1e3)
|
|
|
|
if seconds < 60:
|
|
|
|
return "%.2f seconds" % (seconds)
|
|
|
|
if seconds < 3600:
|
|
|
|
return "%.2f minutes" % (seconds / 60)
|
|
|
|
if seconds < 3600 * 24:
|
|
|
|
return "%.2f hours" % (seconds / 3600)
|
|
|
|
if seconds < 3600 * 24 * 30.5:
|
|
|
|
return "%.2f days" % (seconds / (3600*24))
|
|
|
|
if seconds < 3600 * 24 * 365.25:
|
|
|
|
return "%.2f months" % (seconds / (3600*24*30.5))
|
|
|
|
if seconds < 3600 * 24 * 365.25 * 100:
|
|
|
|
return "%.2f years" % (seconds / (3600*24*365.25))
|
|
|
|
if seconds < 3600 * 24 * 365.25 * 1000:
|
|
|
|
return "%.2f centuries" % (seconds / (3600*24*365.25 * 100))
|
|
|
|
if seconds < 3600 * 24 * 365.25 * 1000000:
|
|
|
|
return "%.2f millenia" % (seconds / (3600*24*365.25 * 100))
|
2015-01-25 18:31:22 +00:00
|
|
|
return "like, forever, dude"
|
2015-01-18 22:18:46 +00:00
|
|
|
|
2015-02-02 12:22:20 +00:00
|
|
|
def StringToUnits(s):
|
|
|
|
try:
|
|
|
|
return long(Decimal(s)*long(coinspecs.atomic_units))
|
|
|
|
except Exception,e:
|
|
|
|
log_error('Failed to convert %s to units: %s' % (s,str(e)))
|
|
|
|
raise
|
|
|
|
|
2014-12-29 17:08:13 +00:00
|
|
|
def SendJSONRPCCommand(host,port,method,params):
|
|
|
|
try:
|
2015-01-15 11:35:21 +00:00
|
|
|
http = httplib.HTTPConnection(host,port,timeout=20)
|
2014-12-29 17:08:13 +00:00
|
|
|
except Exception,e:
|
|
|
|
log_error('SendJSONRPCCommand: Error connecting to %s:%u: %s' % (host, port, str(e)))
|
|
|
|
raise
|
|
|
|
d = dict(id="0",jsonrpc="2.0",method=method,params=params)
|
|
|
|
try:
|
|
|
|
j = json.dumps(d).encode()
|
|
|
|
except Exception,e:
|
|
|
|
log_error('SendJSONRPCCommand: Failed to encode JSON: %s' % str(e))
|
|
|
|
http.close()
|
|
|
|
raise
|
|
|
|
log_log('SendJSONRPCCommand: Sending json as body: %s' % j)
|
|
|
|
headers = None
|
|
|
|
try:
|
|
|
|
http.request("POST","/json_rpc",body=j)
|
|
|
|
except Exception,e:
|
|
|
|
log_error('SendJSONRPCCommand: Failed to post request: %s' % str(e))
|
|
|
|
http.close()
|
|
|
|
raise
|
|
|
|
response = http.getresponse()
|
|
|
|
if response.status != 200:
|
2015-01-25 17:13:03 +00:00
|
|
|
log_error('SendJSONRPCCommand: Error, received reply status %s' % str(response.status))
|
2014-12-29 17:08:13 +00:00
|
|
|
http.close()
|
|
|
|
raise RuntimeError("Error "+response.status)
|
|
|
|
s = response.read()
|
2015-01-25 17:13:03 +00:00
|
|
|
log_log('SendJSONRPCCommand: Received reply status %s: %s' % (response.status, str(s).replace('\r\n',' ').replace('\n',' ')))
|
2014-12-29 17:08:13 +00:00
|
|
|
try:
|
|
|
|
j = json.loads(s)
|
|
|
|
except Exception,e:
|
|
|
|
log_error('SendJSONRPCCommand: Failed to decode JSON: %s' % str(e))
|
|
|
|
http.close()
|
|
|
|
raise
|
|
|
|
http.close()
|
|
|
|
return j
|
|
|
|
|
|
|
|
def SendHTMLCommand(host,port,method):
|
|
|
|
try:
|
2015-01-15 11:35:21 +00:00
|
|
|
http = httplib.HTTPConnection(host,port,timeout=20)
|
2014-12-29 17:08:13 +00:00
|
|
|
except Exception,e:
|
|
|
|
log_error('SendHTMLCommand: Error connecting to %s:%u: %s' % (host, port, str(e)))
|
|
|
|
raise
|
|
|
|
headers = None
|
|
|
|
try:
|
|
|
|
http.request("POST","/"+method)
|
|
|
|
except Exception,e:
|
|
|
|
log_error('SendHTMLCommand: Failed to post request: %s' % str(e))
|
|
|
|
http.close()
|
|
|
|
raise
|
|
|
|
response = http.getresponse()
|
|
|
|
if response.status != 200:
|
2015-01-25 18:31:02 +00:00
|
|
|
log_error('SendHTMLCommand: Error, received reply status %s' % str(response.status))
|
2014-12-29 17:08:13 +00:00
|
|
|
http.close()
|
|
|
|
raise RuntimeError("Error "+response.status)
|
|
|
|
s = response.read()
|
2015-01-25 18:31:02 +00:00
|
|
|
log_log('SendHTMLCommand: Received reply status %s: %s' % (response.status,s.replace('\r\n',' ').replace('\n',' ')))
|
2014-12-29 17:08:13 +00:00
|
|
|
try:
|
|
|
|
j = json.loads(s)
|
|
|
|
except Exception,e:
|
|
|
|
log_error('SendHTMLCommand: Failed to decode JSON: %s' % str(e))
|
|
|
|
http.close()
|
|
|
|
raise
|
|
|
|
http.close()
|
|
|
|
return j
|
|
|
|
|
|
|
|
def SendWalletJSONRPCCommand(method,params):
|
|
|
|
return SendJSONRPCCommand(config.wallet_host,config.wallet_port,method,params)
|
|
|
|
|
|
|
|
def SendDaemonJSONRPCCommand(method,params):
|
|
|
|
return SendJSONRPCCommand(config.daemon_host,config.daemon_port,method,params)
|
|
|
|
|
|
|
|
def SendDaemonHTMLCommand(method):
|
|
|
|
return SendHTMLCommand(config.daemon_host,config.daemon_port,method)
|
|
|
|
|
2015-01-25 21:35:34 +00:00
|
|
|
def RetrieveTipbotBalance(force_refresh=False):
|
2015-01-25 21:09:56 +00:00
|
|
|
global cached_tipbot_balance, cached_tipbot_unlocked_balance, cached_tipbot_balance_timestamp
|
2015-01-25 21:35:34 +00:00
|
|
|
if not force_refresh and cached_tipbot_balance_timestamp and time.time()-cached_tipbot_balance_timestamp < config.tipbot_balance_cache_time:
|
2015-01-25 21:09:56 +00:00
|
|
|
return cached_tipbot_balance, cached_tipbot_unlocked_balance
|
|
|
|
|
2015-01-01 17:33:07 +00:00
|
|
|
j = SendWalletJSONRPCCommand("getbalance",None)
|
|
|
|
if not "result" in j:
|
|
|
|
log_error('RetrieveTipbotBalance: result not found in reply')
|
|
|
|
raise RuntimeError("")
|
|
|
|
return
|
|
|
|
result = j["result"]
|
|
|
|
if not "balance" in result:
|
|
|
|
log_error('RetrieveTipbotBalance: balance not found in result')
|
|
|
|
raise RuntimeError("")
|
|
|
|
return
|
|
|
|
if not "unlocked_balance" in result:
|
|
|
|
log_error('RetrieveTipbotBalance: unlocked_balance not found in result')
|
|
|
|
raise RuntimeError("")
|
|
|
|
return
|
|
|
|
balance = result["balance"]
|
|
|
|
unlocked_balance = result["unlocked_balance"]
|
|
|
|
log_log('RetrieveTipbotBalance: balance: %s' % str(balance))
|
|
|
|
log_log('RetrieveTipbotBalance: unlocked_balance: %s' % str(unlocked_balance))
|
|
|
|
pending = long(balance)-long(unlocked_balance)
|
|
|
|
if pending < 0:
|
|
|
|
log_error('RetrieveTipbotBalance: Negative pending balance! balance %s, unlocked %s' % (str(balance),str(unlocked_balance)))
|
|
|
|
raise RuntimeError("")
|
|
|
|
return
|
2015-01-25 21:09:56 +00:00
|
|
|
cached_tipbot_balance_timestamp=time.time()
|
|
|
|
cached_tipbot_balance=balance
|
|
|
|
cached_tipbot_unlocked_balance=unlocked_balance
|
2015-01-01 17:33:07 +00:00
|
|
|
return balance, unlocked_balance
|
|
|
|
|
2015-01-31 23:40:13 +00:00
|
|
|
def GetAccount(link_or_identity):
|
|
|
|
if isinstance(link_or_identity,Link):
|
|
|
|
identity=link_or_identity.identity()
|
|
|
|
else:
|
|
|
|
identity=link_or_identity
|
|
|
|
account = redis_hget('accounts',identity)
|
|
|
|
if account == None:
|
|
|
|
log_info('No account found for %s, creating new one' % identity)
|
|
|
|
next_account_id = long(redis_get('next_account_id') or 0)
|
|
|
|
account = next_account_id
|
|
|
|
if redis_hexists('accounts',account):
|
|
|
|
raise RuntimeError('Next account ID already exists (%d)', account)
|
|
|
|
redis_hset('accounts',identity,account)
|
|
|
|
next_account_id += 1
|
|
|
|
redis_set('next_account_id',next_account_id)
|
|
|
|
return account
|
|
|
|
|
2015-01-24 11:31:53 +00:00
|
|
|
def RetrieveBalance(link):
|
|
|
|
try:
|
2015-01-31 23:40:13 +00:00
|
|
|
account = GetAccount(link)
|
|
|
|
balance = redis_hget("balances",account) or 0
|
|
|
|
confirming = redis_hget("confirming_payments",account) or 0
|
2015-01-26 17:52:39 +00:00
|
|
|
return long(balance), long(confirming)
|
2015-01-24 11:31:53 +00:00
|
|
|
except Exception, e:
|
|
|
|
log_error('RetrieveBalance: exception: %s' % str(e))
|
|
|
|
raise
|
|
|
|
|
2015-02-06 19:36:58 +00:00
|
|
|
def LinkCore(link,other_identity):
|
|
|
|
try:
|
|
|
|
identity=link.identity()
|
|
|
|
if identity==other_identity:
|
|
|
|
return True, "same-identity"
|
|
|
|
links=redis_hget('links',identity)
|
|
|
|
if links:
|
|
|
|
if other_identity in links.split(chr(0)):
|
|
|
|
return True, "already"
|
|
|
|
links=links+chr(0)+other_identity
|
|
|
|
else:
|
|
|
|
links=other_identity
|
|
|
|
redis_hset('links',identity,links)
|
|
|
|
|
|
|
|
links=redis_hget('links',other_identity)
|
|
|
|
if links:
|
|
|
|
if identity in links.split(chr(0)):
|
|
|
|
# we have both
|
|
|
|
account=redis_hget('accounts',identity)
|
|
|
|
other_account=redis_hget('accounts',other_identity)
|
|
|
|
if account==other_account:
|
|
|
|
log_info('%s and %s already have the same account: %s' % (identity,other_identity,account))
|
|
|
|
return True, "same-account"
|
|
|
|
|
|
|
|
balance=long(redis_hget('balances',account))
|
|
|
|
log_info('Linking accounts %s (%s) and %s (%s)' % (account,identity,other_account,other_identity))
|
|
|
|
p=redis_pipeline()
|
|
|
|
p.hincrby('balances',other_account,balance)
|
|
|
|
p.hincrby('balances',account,-balance)
|
|
|
|
accounts=redis_hgetall('accounts')
|
|
|
|
for a in accounts:
|
|
|
|
if accounts[a]==account:
|
|
|
|
log_info('Changing %s\'s account from %s to %s' % (a,account,other_account))
|
|
|
|
p.hset('accounts',a,other_account)
|
|
|
|
p.execute()
|
|
|
|
return True, "linked"
|
|
|
|
except Exception,e:
|
|
|
|
log_error('Error linking %s and %s: %s' % (identity,other_identity,str(e)))
|
|
|
|
return False, "error"
|
|
|
|
|
|
|
|
return True, "ok"
|
|
|
|
|
2015-01-13 12:28:05 +00:00
|
|
|
def IdentityFromString(link,s):
|
|
|
|
if s.find(':') == -1:
|
2015-01-31 09:42:07 +00:00
|
|
|
network = link.network
|
2015-01-13 12:28:05 +00:00
|
|
|
nick=s
|
|
|
|
else:
|
|
|
|
parts=s.split(':')
|
2015-01-31 09:42:07 +00:00
|
|
|
network_name=parts[0]
|
|
|
|
network=GetNetworkByName(network_name)
|
2015-02-07 10:01:13 +00:00
|
|
|
if not network:
|
|
|
|
log_error('unknown network: %s' % network_name)
|
|
|
|
raise RuntimeError('Unknown network: %s' % network_name)
|
2015-01-13 12:28:05 +00:00
|
|
|
nick=parts[1]
|
2015-01-31 09:42:07 +00:00
|
|
|
return network.name+':'+network.canonicalize(nick)
|
2015-01-13 12:28:05 +00:00
|
|
|
|
|
|
|
def NickFromIdentity(identity):
|
|
|
|
return identity.split(':')[1]
|
|
|
|
|
2015-01-26 22:18:18 +00:00
|
|
|
def RegisterNetwork(name,type):
|
|
|
|
registered_networks[name]=type
|
|
|
|
|
2015-01-13 12:28:05 +00:00
|
|
|
def AddNetwork(network):
|
|
|
|
networks.append(network)
|
|
|
|
|
|
|
|
def GetNetworkByName(name):
|
|
|
|
for network in networks:
|
|
|
|
if network.name==name:
|
|
|
|
return network
|
|
|
|
return None
|
|
|
|
|
|
|
|
def GetNetworkByType(type):
|
|
|
|
for network in networks:
|
|
|
|
if isinstance(network,type):
|
|
|
|
return network
|
|
|
|
return None
|
|
|
|
|
2015-02-05 19:24:15 +00:00
|
|
|
def Lock():
|
|
|
|
return core_lock.acquire()
|
|
|
|
|
|
|
|
def Unlock():
|
|
|
|
core_lock.release()
|
|
|
|
return True
|
|
|
|
|