tippero/tipbot/modules/withdraw.py

175 lines
5.2 KiB
Python
Raw Normal View History

#!/bin/python
#
# Cryptonote tipbot - withdrawal commands
# Copyright 2014 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 json
import string
from tipbot.log import log_error, log_warn, log_info, log_log
import tipbot.coinspecs as coinspecs
2015-01-11 16:48:36 +00:00
import tipbot.config as config
from tipbot.utils import *
from tipbot.redisdb import *
from tipbot.command_manager import *
withdraw_disabled = False
2015-01-13 12:28:05 +00:00
def DisableWithdraw(link,cmd):
global withdraw_disabled
2015-01-13 12:28:05 +00:00
if link:
log_warn('DisableWithdraw: disabled by %s' % link.identity())
else:
log_warn('DisableWithdraw: disabled')
withdraw_disabled = True
2015-01-13 12:28:05 +00:00
def EnableWithdraw(link,cmd):
global withdraw_disabled
2015-01-13 12:28:05 +00:00
log_info('EnableWithdraw: enabled by %s' % link.identity())
withdraw_disabled = False
def CheckDisableWithdraw():
if config.disable_withdraw_on_error:
DisableWithdraw(None,None,None)
2015-01-13 12:28:05 +00:00
def Withdraw(link,cmd):
identity=link.identity()
local_withdraw_fee = config.withdrawal_fee or coinspecs.min_withdrawal_fee
local_min_withdraw_amount = config.min_withdraw_amount or local_withdraw_fee
if local_min_withdraw_amount <= 0 or local_withdraw_fee <= 0 or local_min_withdraw_amount < local_withdraw_fee:
log_error('Withdraw: Inconsistent withdrawal settings')
2015-01-13 12:28:05 +00:00
link.send("An error has occured")
return
try:
address=cmd[1]
except Exception,e:
2015-01-13 12:28:05 +00:00
link.send("Usage: withdraw address [amount]")
return
if not IsValidAddress(address):
2015-01-13 12:28:05 +00:00
link.send("Invalid address")
return
amount = GetParam(cmd,2)
if amount:
try:
famount=float(amount)
if (famount < 0):
raise RuntimeError("")
amount = long(famount * coinspecs.atomic_units)
amount += local_withdraw_fee
except Exception,e:
2015-01-13 12:28:05 +00:00
link.send("Invalid amount")
return
2015-01-13 12:28:05 +00:00
log_info("Withdraw: %s wants to withdraw %s to %s" % (identity, AmountToString(amount) if amount else "all", address))
if withdraw_disabled:
log_error('Withdraw: disabled')
2015-01-13 12:28:05 +00:00
link.send("Sorry, withdrawal is disabled due to a wallet error which requires admin assistance")
return
try:
2015-01-13 12:28:05 +00:00
balance = redis_hget("balances",identity)
if balance == None:
balance = 0
balance=long(balance)
except Exception, e:
log_error('Withdraw: exception: %s' % str(e))
2015-01-13 12:28:05 +00:00
link.send("An error has occured")
return
if amount:
if amount > balance:
2015-01-13 12:28:05 +00:00
log_info("Withdraw: %s trying to withdraw %s, but only has %s" % (identity,AmountToString(amount),AmountToString(balance)))
link.send("You only have %s" % AmountToString(balance))
return
else:
amount = balance
if amount <= 0 or amount < local_min_withdraw_amount:
log_info("Withdraw: Minimum withdrawal balance: %s, %s cannot withdraw %s" % (AmountToString(config.min_withdraw_amount),nick,AmountToString(amount)))
2015-01-13 12:28:05 +00:00
link.send("Minimum withdrawal balance: %s, cannot withdraw %s" % (AmountToString(config.min_withdraw_amount),AmountToString(amount)))
return
try:
fee = long(local_withdraw_fee)
topay = long(amount - fee)
log_info('Withdraw: Raw: fee: %s, to pay: %s' % (str(fee), str(topay)))
log_info('Withdraw: fee: %s, to pay: %s' % (AmountToString(fee), AmountToString(topay)))
params = {
'destinations': [{'address': address, 'amount': topay}],
2015-01-13 12:28:05 +00:00
'payment_id': GetPaymentID(link),
'fee': fee,
2015-01-11 16:28:20 +00:00
'mixin': config.withdrawal_mixin,
'unlock_time': 0,
}
j = SendWalletJSONRPCCommand("transfer",params)
except Exception,e:
log_error('Withdraw: Error in transfer: %s' % str(e))
CheckDisableWithdraw()
2015-01-13 12:28:05 +00:00
link.send("An error has occured")
return
if not "result" in j:
log_error('Withdraw: No result in transfer reply')
CheckDisableWithdraw()
2015-01-13 12:28:05 +00:00
link.send("An error has occured")
return
result = j["result"]
if not "tx_hash" in result:
log_error('Withdraw: No tx_hash in transfer reply')
CheckDisableWithdraw()
2015-01-13 12:28:05 +00:00
link.send("An error has occured")
return
tx_hash = result["tx_hash"]
2015-01-13 12:28:05 +00:00
log_info('%s has withdrawn %s, tx hash %s' % (identity, amount, str(tx_hash)))
link.send( "Tx sent: %s" % tx_hash)
try:
2015-01-13 12:28:05 +00:00
redis_hincrby("balances",identity,-amount)
except Exception, e:
log_error('Withdraw: FAILED TO SUBTRACT BALANCE: exception: %s' % str(e))
CheckDisableWithdraw()
2015-01-13 12:28:05 +00:00
def Help(link):
2015-01-01 14:23:34 +00:00
fee = config.withdrawal_fee or coinspecs.min_withdrawal_fee
min_amount = config.min_withdraw_amount or fee
2015-01-13 12:28:05 +00:00
link.send("Minimum withdrawal: %s" % AmountToString(min_amount))
link.send("Withdrawal fee: %s" % AmountToString(fee))
2015-01-01 14:23:34 +00:00
RegisterModule({
'name': __name__,
'help': Help,
})
RegisterCommand({
2015-01-01 10:12:03 +00:00
'module': __name__,
'name': 'withdraw',
'parms': '<address> [<amount>]',
'function': Withdraw,
'registered': True,
'help': "withdraw part or all of your balance"
})
RegisterCommand({
2015-01-01 10:12:03 +00:00
'module': __name__,
'name': 'enable_withdraw',
'function': EnableWithdraw,
'admin': True,
'help': "Enable withdrawals"
})
RegisterCommand({
2015-01-01 10:12:03 +00:00
'module': __name__,
'name': 'disable_withdraw',
'function': DisableWithdraw,
'admin': True,
'help': "Disable withdrawals"
})