Add payments notification
Signed-off-by: Julien Riou <julien@riou.xyz>
This commit is contained in:
parent
ddc2cd08d5
commit
e51e7c2698
6 changed files with 85 additions and 13 deletions
35
flexpool.py
35
flexpool.py
|
@ -57,6 +57,14 @@ class Miner:
|
||||||
self.balance_fiat = self.convert_balance()
|
self.balance_fiat = self.convert_balance()
|
||||||
self.balance_percentage = self.format_balance_percentage()
|
self.balance_percentage = self.format_balance_percentage()
|
||||||
|
|
||||||
|
last_transactions = miner.payments_paged(page=0)
|
||||||
|
if last_transactions:
|
||||||
|
trx = last_transactions[0]
|
||||||
|
self.last_transaction = Transaction(amount=trx.amount, time=trx.time, duration=trx.duration, txid=trx.txid,
|
||||||
|
exchange_rate=exchange_rate, currency=currency)
|
||||||
|
else:
|
||||||
|
self.last_transaction = None
|
||||||
|
|
||||||
def format_balance(self):
|
def format_balance(self):
|
||||||
return format_weis(self.raw_balance)
|
return format_weis(self.raw_balance)
|
||||||
|
|
||||||
|
@ -76,3 +84,30 @@ class Miner:
|
||||||
attributes['balance_fiat'] = self.balance_fiat
|
attributes['balance_fiat'] = self.balance_fiat
|
||||||
formatted_attributes = ' '.join([f'{k}="{v}"' for k, v in attributes.items()])
|
formatted_attributes = ' '.join([f'{k}="{v}"' for k, v in attributes.items()])
|
||||||
return f'<Miner #{self.address} ({formatted_attributes})>'
|
return f'<Miner #{self.address} ({formatted_attributes})>'
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class Transaction:
|
||||||
|
def __init__(self, amount, time, duration, txid, exchange_rate=None, currency=None):
|
||||||
|
self._exchange_rate = exchange_rate
|
||||||
|
self._currency = currency
|
||||||
|
self.raw_amount = amount
|
||||||
|
self.amount = format_weis(amount)
|
||||||
|
self.amount_fiat = self.convert_amount()
|
||||||
|
self.duration = format_timespan(duration)
|
||||||
|
self.txid = txid
|
||||||
|
self.time = time
|
||||||
|
|
||||||
|
def convert_amount(self):
|
||||||
|
if self._exchange_rate and self._currency:
|
||||||
|
converted = round(convert_weis(self.raw_amount)*self._exchange_rate, 2)
|
||||||
|
converted = f'{converted} {self._currency}'
|
||||||
|
return converted
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
attributes = {'time': self.time, 'amount': self.amount, 'raw_amount': self.raw_amount,
|
||||||
|
'duration': self.duration}
|
||||||
|
if self.amount_fiat:
|
||||||
|
attributes['amount_fiat'] = self.amount_fiat
|
||||||
|
formatted_attributes = ' '.join([f'{k}="{v}"' for k, v in attributes.items()])
|
||||||
|
return f'<Transaction #{self.txid} ({formatted_attributes})>'
|
||||||
|
|
43
main.py
43
main.py
|
@ -61,18 +61,19 @@ def watch_block(config, disable_notifications, last_block=None, exchange_rate=No
|
||||||
return block
|
return block
|
||||||
|
|
||||||
|
|
||||||
def watch_miner(address, config, disable_notifications, last_balance=None, exchange_rate=None, currency=None):
|
def watch_miner(address, config, disable_notifications, last_balance=None, last_transaction=None, exchange_rate=None,
|
||||||
|
currency=None):
|
||||||
logger.debug(f'watching miner {address}')
|
logger.debug(f'watching miner {address}')
|
||||||
try:
|
try:
|
||||||
miner = Miner(address=address, exchange_rate=exchange_rate, currency=currency)
|
miner = Miner(address=address, exchange_rate=exchange_rate, currency=currency)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
logger.error('failed to find miner')
|
logger.error('failed to find miner')
|
||||||
logger.debug(str(err))
|
logger.debug(err)
|
||||||
return
|
return
|
||||||
|
|
||||||
logger.debug(miner)
|
logger.debug(miner)
|
||||||
|
|
||||||
# watch balance
|
logger.debug('watching miner balance')
|
||||||
if miner.raw_balance != last_balance:
|
if miner.raw_balance != last_balance:
|
||||||
logger.info(f'miner {address} balance has changed')
|
logger.info(f'miner {address} balance has changed')
|
||||||
if not disable_notifications and config.get('telegram'):
|
if not disable_notifications and config.get('telegram'):
|
||||||
|
@ -88,6 +89,23 @@ def watch_miner(address, config, disable_notifications, last_balance=None, excha
|
||||||
logger.error('failed to send notification to telegram')
|
logger.error('failed to send notification to telegram')
|
||||||
logger.debug(str(err))
|
logger.debug(str(err))
|
||||||
|
|
||||||
|
logger.debug('watching miner payments')
|
||||||
|
if miner.last_transaction and miner.last_transaction.txid != last_transaction:
|
||||||
|
logger.info(f'new payment for miner {address}')
|
||||||
|
if not disable_notifications and config.get('telegram'):
|
||||||
|
logger.debug('sending payment notification to telegram')
|
||||||
|
variables = {'address': address, 'txid': miner.last_transaction.txid,
|
||||||
|
'amount': miner.last_transaction.amount, 'amount_fiat': miner.last_transaction.amount_fiat,
|
||||||
|
'time': miner.last_transaction.time, 'duration': miner.last_transaction.duration}
|
||||||
|
payload = telegram.create_payment_payload(chat_id=config['telegram']['chat_id'],
|
||||||
|
message_variables=variables)
|
||||||
|
try:
|
||||||
|
telegram.send_message(auth_key=config['telegram']['auth_key'], payload=payload)
|
||||||
|
logger.info('payment notification sent to telegram')
|
||||||
|
except HTTPError as err:
|
||||||
|
logger.error('failed to send notification to telegram')
|
||||||
|
logger.debug(str(err))
|
||||||
|
|
||||||
return miner
|
return miner
|
||||||
|
|
||||||
|
|
||||||
|
@ -115,15 +133,20 @@ def main():
|
||||||
|
|
||||||
block = watch_block(last_block=state.get('block'), config=config, disable_notifications=args.disable_notifications,
|
block = watch_block(last_block=state.get('block'), config=config, disable_notifications=args.disable_notifications,
|
||||||
exchange_rate=exchange_rate, currency=currency)
|
exchange_rate=exchange_rate, currency=currency)
|
||||||
logger.debug('saving block number to state file')
|
if block:
|
||||||
write_state(state_file, block_number=block.number)
|
logger.debug('saving block number to state file')
|
||||||
|
write_state(state_file, block_number=block.number)
|
||||||
|
|
||||||
if config.get('miner'):
|
if config.get('miner'):
|
||||||
miner = watch_miner(last_balance=state.get('balance'), address=config['miner'], config=config,
|
miner = watch_miner(last_balance=state.get('balance'), last_transaction=state.get('payment'),
|
||||||
disable_notifications=args.disable_notifications, exchange_rate=exchange_rate,
|
address=config['miner'], config=config, disable_notifications=args.disable_notifications,
|
||||||
currency=currency)
|
exchange_rate=exchange_rate, currency=currency)
|
||||||
logger.debug('saving miner balance to state file')
|
if miner:
|
||||||
write_state(state_file, miner_balance=miner.raw_balance)
|
logger.debug('saving miner balance to state file')
|
||||||
|
write_state(state_file, miner_balance=miner.raw_balance)
|
||||||
|
if miner.last_transaction and miner.last_transaction.txid:
|
||||||
|
logger.debug('saving miner payment to state file')
|
||||||
|
write_state(state_file, miner_payment=miner.last_transaction.txid)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|
|
@ -7,7 +7,7 @@ colorama==0.4.3
|
||||||
contextlib2==0.6.0
|
contextlib2==0.6.0
|
||||||
distlib==0.3.0
|
distlib==0.3.0
|
||||||
distro==1.4.0
|
distro==1.4.0
|
||||||
flexpoolapi==1.2.7.post1
|
flexpoolapi==1.2.7.post2
|
||||||
html5lib==1.0.1
|
html5lib==1.0.1
|
||||||
humanfriendly==9.1
|
humanfriendly==9.1
|
||||||
idna==2.8
|
idna==2.8
|
||||||
|
|
5
state.py
5
state.py
|
@ -7,7 +7,7 @@ def read_state(filename):
|
||||||
return json.load(fd)
|
return json.load(fd)
|
||||||
|
|
||||||
|
|
||||||
def write_state(filename, block_number=None, miner_balance=None):
|
def write_state(filename, block_number=None, miner_balance=None, miner_payment=None):
|
||||||
data = {}
|
data = {}
|
||||||
if os.path.isfile(filename):
|
if os.path.isfile(filename):
|
||||||
data = read_state(filename)
|
data = read_state(filename)
|
||||||
|
@ -18,6 +18,9 @@ def write_state(filename, block_number=None, miner_balance=None):
|
||||||
if miner_balance:
|
if miner_balance:
|
||||||
data['balance'] = miner_balance
|
data['balance'] = miner_balance
|
||||||
|
|
||||||
|
if miner_payment:
|
||||||
|
data['payment'] = miner_payment
|
||||||
|
|
||||||
with open(filename, 'w') as fd:
|
with open(filename, 'w') as fd:
|
||||||
json.dump(data, fd)
|
json.dump(data, fd)
|
||||||
|
|
||||||
|
|
|
@ -23,8 +23,12 @@ def create_balance_payload(chat_id, message_variables):
|
||||||
return generate_payload(chat_id, message_variables, 'balance.md.j2')
|
return generate_payload(chat_id, message_variables, 'balance.md.j2')
|
||||||
|
|
||||||
|
|
||||||
|
def create_payment_payload(chat_id, message_variables):
|
||||||
|
return generate_payload(chat_id, message_variables, 'payment.md.j2')
|
||||||
|
|
||||||
|
|
||||||
def generate_payload(chat_id, message_variables, template_name):
|
def generate_payload(chat_id, message_variables, template_name):
|
||||||
payload = {'chat_id': chat_id, 'parse_mode': 'MarkdownV2'}
|
payload = {'chat_id': chat_id, 'parse_mode': 'MarkdownV2', 'disable_web_page_preview': True}
|
||||||
template_path = os.path.join(absolute_path, 'templates')
|
template_path = os.path.join(absolute_path, 'templates')
|
||||||
loader = FileSystemLoader(template_path)
|
loader = FileSystemLoader(template_path)
|
||||||
env = Environment(loader=loader)
|
env = Environment(loader=loader)
|
||||||
|
|
7
templates/payment.md.j2
Normal file
7
templates/payment.md.j2
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
*💵 New payment*
|
||||||
|
|
||||||
|
*Amount*: {{amount}} {% if amount_fiat != 'None' %}\({{amount_fiat}}\){% endif %}
|
||||||
|
*ID*: [{{txid}}](https://etherscan.io/tx/{{txid}})
|
||||||
|
*Address*: {{address}}
|
||||||
|
*Date/Time*: {{time}}
|
||||||
|
*Duration*: {{duration}}
|
Reference in a new issue