2014-12-29 17:08:13 +00:00
|
|
|
#!/bin/python
|
|
|
|
#
|
|
|
|
# Cryptonote tipbot - withdrawal commands
|
2015-02-09 23:52:32 +00:00
|
|
|
# Copyright 2014,2015 moneromooo
|
|
|
|
# DNS code largely copied from Electrum OpenAlias plugin, Copyright 2014-2015 The monero Project
|
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 json
|
|
|
|
import string
|
2015-02-09 23:52:32 +00:00
|
|
|
import re
|
|
|
|
import dns.name
|
|
|
|
import dns.dnssec
|
|
|
|
import dns.resolver
|
|
|
|
import dns.message
|
|
|
|
import dns.query
|
|
|
|
import dns.rdatatype
|
|
|
|
import dns.rdtypes.ANY.TXT
|
|
|
|
import dns.rdtypes.ANY.NS
|
2014-12-29 17:08:13 +00:00
|
|
|
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
|
2015-02-14 12:15:49 +00:00
|
|
|
from tipbot.user import User
|
|
|
|
from tipbot.link import Link
|
2014-12-29 17:08:13 +00:00
|
|
|
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):
|
2014-12-29 17:08:13 +00:00
|
|
|
global withdraw_disabled
|
2015-01-13 12:28:05 +00:00
|
|
|
if link:
|
|
|
|
log_warn('DisableWithdraw: disabled by %s' % link.identity())
|
2014-12-29 17:08:13 +00:00
|
|
|
else:
|
|
|
|
log_warn('DisableWithdraw: disabled')
|
|
|
|
withdraw_disabled = True
|
|
|
|
|
2015-01-13 12:28:05 +00:00
|
|
|
def EnableWithdraw(link,cmd):
|
2014-12-29 17:08:13 +00:00
|
|
|
global withdraw_disabled
|
2015-01-13 12:28:05 +00:00
|
|
|
log_info('EnableWithdraw: enabled by %s' % link.identity())
|
2014-12-29 17:08:13 +00:00
|
|
|
withdraw_disabled = False
|
|
|
|
|
|
|
|
def CheckDisableWithdraw():
|
|
|
|
if config.disable_withdraw_on_error:
|
2015-02-02 20:42:24 +00:00
|
|
|
DisableWithdraw(None,None)
|
2014-12-29 17:08:13 +00:00
|
|
|
|
2015-02-09 23:52:32 +00:00
|
|
|
def ValidateDNSSEC(address):
|
|
|
|
log_info('Validating DNSSEC for %s' % address)
|
|
|
|
try:
|
|
|
|
resolver = dns.resolver.get_default_resolver()
|
|
|
|
ns = resolver.nameservers[0]
|
|
|
|
parts = address.split('.')
|
|
|
|
for i in xrange(len(parts),0,-1):
|
|
|
|
subpart = '.'.join(parts[i-1:])
|
|
|
|
query = dns.message.make_query(subpart,dns.rdatatype.NS)
|
|
|
|
response = dns.query.udp(query,ns,1)
|
|
|
|
if response.rcode() != dns.rcode.NOERROR:
|
|
|
|
return False
|
|
|
|
if len(response.authority) > 0:
|
|
|
|
rrset = response.authority[0]
|
|
|
|
else:
|
|
|
|
rrset = response.answer[0]
|
|
|
|
rr = rrset[0]
|
|
|
|
if rr.rdtype == dns.rdatatype.SOA:
|
|
|
|
continue
|
|
|
|
query = dns.message.make_query(subpart,dns.rdatatype.DNSKEY,want_dnssec=True)
|
|
|
|
response = dns.query.udp(query,ns,1)
|
|
|
|
if response.rcode() != 0:
|
|
|
|
return False
|
|
|
|
answer = response.answer
|
|
|
|
if len(answer) != 2:
|
|
|
|
return False
|
|
|
|
name = dns.name.from_text(subpart)
|
|
|
|
dns.dnssec.validate(answer[0],answer[1],{name:answer[0]})
|
|
|
|
return True
|
|
|
|
except Exception,e:
|
|
|
|
log_error('Failed to validate DNSSEC for %s: %s' % (address, str(e)))
|
|
|
|
return False
|
|
|
|
|
|
|
|
def ResolveCore(address,ctype):
|
|
|
|
log_info('Resolving %s address for %s' % (ctype,address))
|
|
|
|
address=address.replace('@','.')
|
|
|
|
if not '.' in address:
|
|
|
|
return False,'invalid address'
|
|
|
|
|
|
|
|
try:
|
|
|
|
for attempt in range(3):
|
|
|
|
resolver = dns.resolver.Resolver()
|
|
|
|
resolver.timeout = 2
|
|
|
|
resolver.lifetime = 2
|
|
|
|
records = resolver.query(address,dns.rdatatype.TXT)
|
|
|
|
for record in records:
|
|
|
|
s = record.strings[0]
|
|
|
|
if s.lower().startswith('oa1:%s' % ctype.lower()):
|
|
|
|
a = re.sub('.*recipient_address[ \t]*=[ \t]*\"?([A-Za-z0-9]+)\"?.*','\\1',s)
|
|
|
|
if IsValidAddress(a):
|
|
|
|
log_info('Found %s address at %s: %s' % (ctype,address,a))
|
|
|
|
return True, [a,ValidateDNSSEC(address)]
|
|
|
|
except Exception,e:
|
|
|
|
log_error('Error resolving %s: %s' % (address,str(e)))
|
|
|
|
|
|
|
|
return False, 'not found'
|
|
|
|
|
2015-01-13 12:28:05 +00:00
|
|
|
def Withdraw(link,cmd):
|
|
|
|
identity=link.identity()
|
|
|
|
|
2014-12-29 17:08:13 +00:00
|
|
|
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")
|
2014-12-29 17:08:13 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
address=cmd[1]
|
|
|
|
except Exception,e:
|
2015-02-02 20:43:41 +00:00
|
|
|
link.send("Usage: withdraw address [amount] [paymentid]")
|
2014-12-29 17:08:13 +00:00
|
|
|
return
|
|
|
|
|
2015-02-09 23:52:32 +00:00
|
|
|
if '.' in address:
|
|
|
|
ok,extra=ResolveCore(address,coinspecs.symbol)
|
|
|
|
if not ok:
|
|
|
|
link.send('Error: %s' % extra)
|
|
|
|
return
|
|
|
|
a=extra[0]
|
|
|
|
dnssec=extra[1]
|
|
|
|
if not dnsssec:
|
|
|
|
link.send('%s address %s was found for %s' % (coinspecs.name,a,address))
|
|
|
|
link.send('Trust chain could not be verified, so withdrawal was not automatically performed')
|
|
|
|
link.send('Withdraw using this %s address if it is correct' % coinspecs.name)
|
|
|
|
return
|
|
|
|
address=a
|
|
|
|
|
2014-12-29 17:08:13 +00:00
|
|
|
if not IsValidAddress(address):
|
2015-01-13 12:28:05 +00:00
|
|
|
link.send("Invalid address")
|
2014-12-29 17:08:13 +00:00
|
|
|
return
|
2015-02-02 20:43:41 +00:00
|
|
|
|
|
|
|
if GetParam(cmd,3):
|
|
|
|
amount = GetParam(cmd,2)
|
|
|
|
paymentid = GetParam(cmd,3)
|
|
|
|
else:
|
2015-02-28 18:31:20 +00:00
|
|
|
if GetParam(cmd,2) and IsValidPaymentID(GetParam(cmd,2)):
|
2015-02-02 20:43:41 +00:00
|
|
|
amount = None
|
|
|
|
paymentid = GetParam(cmd,2)
|
|
|
|
else:
|
|
|
|
amount = GetParam(cmd,2)
|
|
|
|
paymentid = None
|
|
|
|
|
2014-12-29 17:08:13 +00:00
|
|
|
if amount:
|
|
|
|
try:
|
2015-02-02 12:22:20 +00:00
|
|
|
amount = StringToUnits(amount)
|
|
|
|
if (amount <= 0):
|
2014-12-29 17:08:13 +00:00
|
|
|
raise RuntimeError("")
|
|
|
|
amount += local_withdraw_fee
|
|
|
|
except Exception,e:
|
2015-01-13 12:28:05 +00:00
|
|
|
link.send("Invalid amount")
|
2014-12-29 17:08:13 +00:00
|
|
|
return
|
2015-02-02 20:43:41 +00:00
|
|
|
if paymentid != None:
|
|
|
|
if not IsValidPaymentID(paymentid):
|
|
|
|
link.send("Invalid payment ID")
|
|
|
|
return
|
2014-12-29 17:08:13 +00:00
|
|
|
|
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))
|
2014-12-29 17:08:13 +00:00
|
|
|
|
|
|
|
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")
|
2014-12-29 17:08:13 +00:00
|
|
|
return
|
|
|
|
|
2015-01-31 23:40:13 +00:00
|
|
|
account = GetAccount(identity)
|
2014-12-29 17:08:13 +00:00
|
|
|
try:
|
2015-02-02 20:43:17 +00:00
|
|
|
balance = redis_hget('balances',account)
|
2014-12-29 17:08:13 +00:00
|
|
|
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")
|
2014-12-29 17:08:13 +00:00
|
|
|
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))
|
2014-12-29 17:08:13 +00:00
|
|
|
return
|
|
|
|
else:
|
|
|
|
amount = balance
|
|
|
|
|
|
|
|
if amount <= 0 or amount < local_min_withdraw_amount:
|
2015-03-17 20:54:25 +00:00
|
|
|
log_info("Withdraw: Minimum withdrawal balance: %s, %s cannot withdraw %s" % (AmountToString(config.min_withdraw_amount),identity,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)))
|
2014-12-29 17:08:13 +00:00
|
|
|
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-02-02 20:43:41 +00:00
|
|
|
'payment_id': paymentid,
|
2015-02-01 12:46:49 +00:00
|
|
|
'fee': coinspecs.min_withdrawal_fee,
|
2015-01-11 16:28:20 +00:00
|
|
|
'mixin': config.withdrawal_mixin,
|
2014-12-29 17:08:13 +00:00
|
|
|
'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")
|
2014-12-29 17:08:13 +00:00
|
|
|
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")
|
2014-12-29 17:08:13 +00:00
|
|
|
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")
|
2014-12-29 17:08:13 +00:00
|
|
|
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)
|
2014-12-29 17:08:13 +00:00
|
|
|
|
|
|
|
try:
|
2015-01-31 23:40:13 +00:00
|
|
|
redis_hincrby("balances",account,-amount)
|
2014-12-29 17:08:13 +00:00
|
|
|
except Exception, e:
|
|
|
|
log_error('Withdraw: FAILED TO SUBTRACT BALANCE: exception: %s' % str(e))
|
|
|
|
CheckDisableWithdraw()
|
|
|
|
|
2015-02-09 23:52:32 +00:00
|
|
|
def Resolve(link,cmd):
|
|
|
|
try:
|
|
|
|
address=GetParam(cmd,1)
|
2015-02-10 23:48:54 +00:00
|
|
|
if not address:
|
|
|
|
raise RuntimeError("")
|
2015-02-09 23:52:32 +00:00
|
|
|
except Exception,e:
|
|
|
|
link.send('usage: !resolve <address>')
|
|
|
|
return
|
|
|
|
ok,extra=ResolveCore(address,coinspecs.symbol)
|
|
|
|
if not ok:
|
|
|
|
link.send('Error: %s' % extra)
|
|
|
|
return
|
|
|
|
a=extra[0]
|
|
|
|
dnssec=extra[1]
|
|
|
|
if dnssec:
|
|
|
|
link.send('Found %s address at %s: %s' % (coinspecs.symbol,address,a))
|
|
|
|
else:
|
|
|
|
link.send('Found %s address at %s via insecure DNS: %s' % (coinspecs.symbol,address,a))
|
|
|
|
|
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-02-09 23:52:32 +00:00
|
|
|
link.send_private("Partial or full withdrawals can be made to any %s address" % coinspecs.name)
|
|
|
|
link.send_private("OpenAlias is supported, to pay directly to a domain name which uses it")
|
2015-01-20 17:18:15 +00:00
|
|
|
link.send_private("Minimum withdrawal: %s" % AmountToString(min_amount))
|
|
|
|
link.send_private("Withdrawal fee: %s" % AmountToString(fee))
|
2015-01-01 14:23:34 +00:00
|
|
|
|
2014-12-29 17:08:13 +00:00
|
|
|
|
|
|
|
|
2015-01-03 18:32:09 +00:00
|
|
|
RegisterModule({
|
|
|
|
'name': __name__,
|
|
|
|
'help': Help,
|
|
|
|
})
|
2014-12-29 17:08:13 +00:00
|
|
|
RegisterCommand({
|
2015-01-01 10:12:03 +00:00
|
|
|
'module': __name__,
|
2014-12-29 17:08:13 +00:00
|
|
|
'name': 'withdraw',
|
2015-02-09 23:52:32 +00:00
|
|
|
'parms': '<address>|<domain-name> [<amount>] [paymentid]',
|
2014-12-29 17:08:13 +00:00
|
|
|
'function': Withdraw,
|
|
|
|
'registered': True,
|
|
|
|
'help': "withdraw part or all of your balance"
|
|
|
|
})
|
2015-02-09 23:52:32 +00:00
|
|
|
RegisterCommand({
|
|
|
|
'module': __name__,
|
|
|
|
'name': 'resolve',
|
|
|
|
'parms': '<address>',
|
|
|
|
'function': Resolve,
|
|
|
|
'registered': True,
|
|
|
|
'help': "Resolve a %s address from DNS with OpenAlias" % coinspecs.name
|
|
|
|
})
|
2014-12-29 17:08:13 +00:00
|
|
|
RegisterCommand({
|
2015-01-01 10:12:03 +00:00
|
|
|
'module': __name__,
|
2014-12-29 17:08:13 +00:00
|
|
|
'name': 'enable_withdraw',
|
|
|
|
'function': EnableWithdraw,
|
|
|
|
'admin': True,
|
|
|
|
'help': "Enable withdrawals"
|
|
|
|
})
|
|
|
|
RegisterCommand({
|
2015-01-01 10:12:03 +00:00
|
|
|
'module': __name__,
|
2014-12-29 17:08:13 +00:00
|
|
|
'name': 'disable_withdraw',
|
|
|
|
'function': DisableWithdraw,
|
|
|
|
'admin': True,
|
|
|
|
'help': "Disable withdrawals"
|
|
|
|
})
|