#!/bin/python # # Cryptonote tipbot - utility functions # Copyright 2014,2015 moneromooo # # 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 from tipbot.ircutils import * 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 def GetNickFromPaymentID(p): nick = redis_hget("paymentid",p) log_log('PaymentID %s => %s' % (p, str(nick))) return nick 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 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: http = httplib.HTTPConnection(host,port) 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: http = httplib.HTTPConnection(host,port) 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) 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