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
|
|
|
|
import tipbot.config as config
|
|
|
|
import tipbot.coinspecs as coinspecs
|
|
|
|
from tipbot.log import log_error, log_warn, log_info, log_log
|
2015-01-01 11:42:06 +00:00
|
|
|
from tipbot.ircutils import *
|
2014-12-29 17:08:13 +00:00
|
|
|
from tipbot.redisdb import *
|
|
|
|
|
|
|
|
|
|
|
|
def GetPassword():
|
|
|
|
try:
|
|
|
|
f = open('tipbot-password.txt', 'r')
|
|
|
|
for p in f:
|
|
|
|
p = p.strip("\r\n")
|
|
|
|
f.close()
|
|
|
|
return p
|
|
|
|
except Exception,e:
|
|
|
|
log_error('could not fetch password: %s' % str(e))
|
|
|
|
raise
|
|
|
|
return "xxx"
|
|
|
|
|
|
|
|
def IsParamPresent(parms,idx):
|
|
|
|
return len(parms) > idx
|
|
|
|
|
|
|
|
def GetParam(parms,idx):
|
|
|
|
if IsParamPresent(parms,idx):
|
|
|
|
return parms[idx]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def GetPaymentID(nick):
|
|
|
|
salt="2u3g55bkwrui32fi3g4bGR$j5g4ugnujb-"+coinspecs.name+"-";
|
|
|
|
p = hashlib.sha256(salt+nick).hexdigest();
|
|
|
|
try:
|
|
|
|
redis_hset("paymentid",p,nick)
|
|
|
|
except Exception,e:
|
|
|
|
log_error('GetPaymentID: failed to set payment ID for %s to redis: %s' % (nick,str(e)))
|
|
|
|
return p
|
|
|
|
|
2015-01-10 20:35:10 +00:00
|
|
|
def GetNickFromPaymentID(p):
|
2014-12-29 17:08:13 +00:00
|
|
|
nick = redis_hget("paymentid",p)
|
2015-01-10 20:35:10 +00:00
|
|
|
log_log('PaymentID %s => %s' % (p, str(nick)))
|
2014-12-29 17:08:13 +00:00
|
|
|
return nick
|
|
|
|
|
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
|
|
|
|
|
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:
|
|
|
|
for den in coinspecs.denominations:
|
|
|
|
if lamount < den[0]:
|
|
|
|
samount = "%.16g %s" % (float(lamount) / den[1], den[2])
|
|
|
|
break
|
|
|
|
if not samount:
|
|
|
|
samount = "%.16g %s" % (float(lamount) / coinspecs.atomic_units, coinspecs.name)
|
|
|
|
return samount
|
|
|
|
|
|
|
|
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()
|
|
|
|
log_log('SendJSONRPCCommand: Received reply status: %s' % response.status)
|
|
|
|
if response.status != 200:
|
|
|
|
log_error('SendJSONRPCCommand: Error, not 200: %s' % str(response.status))
|
|
|
|
http.close()
|
|
|
|
raise RuntimeError("Error "+response.status)
|
|
|
|
s = response.read()
|
|
|
|
log_log('SendJSONRPCCommand: Received reply: %s' % str(s))
|
|
|
|
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()
|
|
|
|
log_log('SendHTMLCommand: Received reply status: %s' % response.status)
|
|
|
|
if response.status != 200:
|
|
|
|
log_error('SendHTMLCommand: Error, not 200: %s' % str(response.status))
|
|
|
|
http.close()
|
|
|
|
raise RuntimeError("Error "+response.status)
|
|
|
|
s = response.read()
|
|
|
|
log_log('SendHTMLCommand: Received reply: %s' % s)
|
|
|
|
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-01 17:33:07 +00:00
|
|
|
def RetrieveTipbotBalance():
|
|
|
|
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
|
|
|
|
return balance, unlocked_balance
|
|
|
|
|
|
|
|
def RetrieveHouseBalance():
|
|
|
|
balance, unlocked_balance = RetrieveTipbotBalance()
|
|
|
|
|
|
|
|
nicks = redis_hgetall("balances")
|
|
|
|
for nick in nicks:
|
|
|
|
nb = redis_hget("balances", nick)
|
|
|
|
unlocked_balance = unlocked_balance - long(nb)
|
|
|
|
log_log('RetrieveHouseBalance: subtracting %s from %s to give %s' % (AmountToString(nb), nick, AmountToString(unlocked_balance)))
|
|
|
|
|
|
|
|
if unlocked_balance < 0:
|
|
|
|
raise RuntimeError('Negative house balance')
|
|
|
|
return
|
|
|
|
return unlocked_balance
|
|
|
|
|