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})>' | ||||||
|  |  | ||||||
							
								
								
									
										35
									
								
								main.py
									
										
									
									
									
								
							
							
						
						
									
										35
									
								
								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) | ||||||
|  |     if block: | ||||||
|         logger.debug('saving block number to state file') |         logger.debug('saving block number to state file') | ||||||
|         write_state(state_file, block_number=block.number) |         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) | ||||||
|  |         if miner: | ||||||
|             logger.debug('saving miner balance to state file') |             logger.debug('saving miner balance to state file') | ||||||
|             write_state(state_file, miner_balance=miner.raw_balance) |             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