-
Notifications
You must be signed in to change notification settings - Fork 0
/
hds.py
922 lines (768 loc) · 32.2 KB
/
hds.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
#!/usr/bin/python3
############################
#
# co8/HDS - Hotspot Discord Status
# https://github.com/co8/hds
#
# co8.com
# enrique r grullon
# discord: co8#1934
############################
########
# Set Crontab
# crontab -e
# - run script every minute. log to file
# */1 * * * * cd ~/hds; python3 hds.py >> cron.log 2>&1
#
## Optional
# clear cron.log weekly on Monday at 4:20am
# 20 4 * * 1 cd ~/hds; rm cron.log; echo 'Cron Log Cleared\n' >> cron.log 2>&1
########
########
# install Discord-Webhook module
# % pip3 install discord-webhook
########
#######
# Command Line Arguments
# REPORT
# python3 hds.py report - send miner report (if set Bobcat_local_endpoint)
# RESET
# python3 hds.py reset
#######
# modules/libraries
import sys
import time
import math
import requests
import json
import uuid
import logging
from datetime import datetime
from discord_webhook import DiscordWebhook
logging.basicConfig(
level=logging.INFO, format="%(asctime)s - %(levelname)s - %(message)s"
)
####
# Notes:
# To Add: Private methods, functions and data members
# _ : you shouldn’t access this method because it’s not part of the API
# __ : mangle the attribute names of a class to avoid conflicts of attribute names between classes
####
### vars
# override default values in config.json
wellness_check_hours = 12 # Default 8 hours. send status msg if X hours have lapsed since last message sent. slows miner, don't abuse
report_interval_hours = 72 # HOURS scheduled miner report. time after last report sent. slows miner, don't abuse
#
#
history_length_max = 200 # Trim activity history when reaches this length
history_length_min = 125 # when trim activity history, leave newest
sync_blocks_behind = 100 # Blocks Behind blockchain to be considered out of sync
api_sync_lag_multiple = 5 # Multiply sync_blocks_behind * api_sync_lag_multiple to balance with Helium API "Sync Status"
# pop_status_minutes = 7 # MINUTES remove status msg when sending activity if activity is recent to last activity sent. keep discord tidy
# interval_pop_status_seconds = int(60 * pop_status_minutes)
interval_pop_status_seconds = 390 # seconds - remove status msg when sending activity if activity is recent to last activity sent. keep discord tidy
helium_api_endpoint = "https://api.helium.io/v1/"
helium_explorer_tx = "https://explorer.helium.com/txns/"
config_file = "config.json"
activities = []
output_message = []
activity_history = []
hs = {}
wellness_check = history_repeats = wellness_check_seconds = 0
report_interval_seconds = output_message_length = 0
send = send_report = add_welcome = send_wellness_check = False
invalid_reason_short_names = {
"witness_too_close": "Too Close",
"witness_rssi_too_high": "RSSI Too High",
"witness_rssi_below_lower_bound": "RSSI BLB",
}
reward_short_names = {
"poc_witnesses": "Witness",
"poc_challengees": "Beacon",
"poc_challengers": "Challenger",
"data_credits": "Data",
}
# Generate a UUID from a host ID, sequence number, and the current time
headers = {"User-Agent": str(uuid.uuid1())}
#### functions
def local_bobcat_miner_report():
# only run if bobcat_local_endpoint is set
if "bobcat_local_endpoint" in config and bool(config["bobcat_local_endpoint"]):
global send_report, output_message, report_interval_hours, add_welcome
# send if next.report has been met
if "report" in config["next"] and hs["now"] > config["next"]["report"]:
send_report = True
hour_plural = "s" if report_interval_hours != 1 else ""
interval_msg = f"`⏰ Scheduled Miner Report, every {report_interval_hours}hr{hour_plural} `"
output_message.insert(0, interval_msg)
print(
f"\n{hs['time']} report interval met, every {report_interval_hours}hrs",
end="",
)
if bool(send_report) or bool(add_welcome):
# if 'bobcat_local_endpoint' in config and bool(config['bobcat_local_endpoint']) and bool(send_report):
# try to get json or return error
status = ""
try:
# LIVE local data
bobcat_miner_json = config["bobcat_local_endpoint"] + "miner.json"
bobcat_request = requests.get(bobcat_miner_json, headers=headers)
data = bobcat_request.json()
### Dev only
###LOCAL load miner.json
# with open("miner.json") as json_data_file:
# data = json.load(json_data_file)
except requests.RequestException:
status = "Connectivity"
except ValueError: # includes simplejson.decoder.JSONDecodeError
status = "Parsing JSON"
except (IndexError, KeyError):
status = "JSON format"
if bool(status):
print(f"\n{hs['time']} Bobcat API Error: {status}")
else:
# miner_temp = data["temp_alert"].title()
miner_temp = data["temp0"]
if "miner_temp" not in config["last"]["report"]:
config["last"]["report"]["miner_temp"] = ""
if miner_temp != config["last"]["report"]["miner_temp"]:
config["last"]["report"]["miner_temp"] = miner_temp
miner_temp = f"**{miner_temp}**"
miner_state = data["miner"]["State"].title()
if "miner_state" not in config["last"]["report"]:
config["last"]["report"]["miner_state"] = ""
if miner_state != config["last"]["report"]["miner_state"]:
config["last"]["report"]["miner_state"] = miner_state
miner_state = f"**{miner_state}**"
# miner_height
miner_height_int = int(data["miner_height"])
miner_height = "{:,}".format(miner_height_int)
if "miner_height" not in config["last"]["report"]:
config["last"]["report"]["miner_height"] = ""
if miner_height_int != config["last"]["report"]["miner_height"]:
config["last"]["report"]["miner_height"] = miner_height_int
miner_height = f"**{miner_height}**"
# miner_block
miner_block_int = int(data["blockchain_height"])
miner_block = "{:,}".format(miner_block_int)
if "miner_block" not in config["last"]["report"]:
config["last"]["report"]["miner_block"] = ""
if miner_block_int != config["last"]["report"]["miner_block"]:
config["last"]["report"]["miner_block"] = miner_block_int
miner_block = f"**{miner_block}**"
# miner_epoch
miner_epoch_int = int(data["epoch"])
miner_epoch = "{:,}".format(miner_epoch_int)
if "miner_epoch" not in config["last"]["report"]:
config["last"]["report"]["miner_epoch"] = ""
if miner_epoch_int != config["last"]["report"]["miner_epoch"]:
config["last"]["report"]["miner_epoch"] = miner_epoch_int
miner_epoch = f"**{miner_epoch}**"
# miner_gap
miner_gap_int = int(data["blockchain_height"]) - int(
data["miner_height"]
)
miner_gap_int = 0 if miner_gap_int < 0 else miner_gap_int
miner_gap = "{:,}".format(miner_gap_int)
if "miner_gap" not in config["last"]["report"]:
config["last"]["report"]["miner_gap"] = ""
if miner_gap != config["last"]["report"]["miner_gap"]:
config["last"]["report"]["miner_gap"] = miner_gap_int
miner_gap = f"**({miner_gap})**"
miner_gap = (
"**0**"
if miner_gap == "**(0)**" or miner_gap_int <= 0
else miner_gap
)
else:
miner_gap = f"({miner_gap})"
miner_gap = (
"Synced"
if miner_gap == "(0)" or miner_gap_int <= 0
else miner_gap
)
# miner_port_44158
# UPDATE - check peerbook for p2p(open) or ip4(relayed)
new_miner_port_44158 = False
miner_port_44158 = data["ports"]["44158"].title()
if "miner_port_44158" not in config["last"]["report"]:
config["last"]["report"]["miner_port_44158"] = ""
if miner_port_44158 != config["last"]["report"]["miner_port_44158"]:
config["last"]["report"]["miner_port_44158"] = miner_port_44158
new_miner_port_44158 = True
if miner_port_44158 == "Open":
miner_port_44158 = (
"✅ **Open**" if bool(new_miner_port_44158) else "✅ Open"
)
else:
miner_port_44158 = f"💩 **{miner_port_44158}, (RELAYED)**"
# miner_sync . Not Synced if more than 100 block behind miner_block
miner_sync = (
"Syncing"
if miner_gap_int > sync_blocks_behind # 0
else "Synced"
# "Synced" if miner_gap_int <= sync_blocks_behind else "Syncing"
)
if "miner_sync" not in config["last"]["report"]:
config["last"]["report"]["miner_sync"] = ""
if miner_sync != config["last"]["report"]["miner_sync"]:
config["last"]["report"]["miner_sync"] = miner_sync
miner_sync = f"**{miner_sync}**"
# helium OTA version
miner_ota_helium = data["miner"]["Image"]
miner_ota_helium = miner_ota_helium.split("_")
miner_ota_helium = str(miner_ota_helium[1])
if "miner_ota_helium" not in config["last"]["report"]:
config["last"]["report"]["miner_ota_helium"] = ""
if miner_ota_helium != config["last"]["report"]["miner_ota_helium"]:
config["last"]["report"]["miner_ota_helium"] = miner_ota_helium
miner_ota_helium = f"**{miner_ota_helium}**"
# bobcat OTA version
miner_ota_bobcat = data["ota_version"]
if "miner_ota_bobcat" not in config["last"]["report"]:
config["last"]["report"]["miner_ota_bobcat"] = ""
if miner_ota_bobcat != config["last"]["report"]["miner_ota_bobcat"]:
config["last"]["report"]["miner_ota_bobcat"] = miner_ota_bobcat
miner_ota_bobcat = f"**{miner_ota_bobcat}**"
report = (
f"🔩 **MINERity Report** `{nice_date(hs['now'])}`"
+ "\n"
+ f"Sync: {miner_sync} Height: 📦 {miner_height} Gap: {miner_gap}"
+ "\n"
+ f"Status: {miner_state} Temp: {miner_temp} Epoch: {miner_epoch}"
+ "\n"
+ f"Firmware: Helium {miner_ota_helium} | Bobcat {miner_ota_bobcat}"
# + "\n"
# + f"Inbound (44158): {miner_port_44158}"
# + "\n"
# + "Outbound:"
)
output_message.append(report)
# last report send timestamp
config["last"]["report"]["send"] = hs["now"]
config["last"]["report"]["send_nice"] = nice_date(hs["now"])
# config values. repeat every X hours
config["next"]["report"] = hs["now"] + report_interval_seconds
config["next"]["report_nice"] = nice_date(config["next"]["report"])
print(f"\n{hs['time']} bobcat local miner report", end="")
###load config.json vars
def load_config():
global config, send_report, activity_history, wellness_check_hours, report_interval_hours, wellness_check_seconds, report_interval_seconds, add_welcome
with open(config_file) as json_data_file:
config = json.load(json_data_file)
# wellness_check_hours - default sets config, or uses config value
if "wellness_check_hours" in config:
wellness_check_hours = config["wellness_check_hours"]
# else:
# config["wellness_check_hours"] = wellness_check_hours
wellness_check_seconds = int(60 * 60 * wellness_check_hours)
# report_interval_hours - default sets config, or uses config value
if "report_interval_hours" in config:
report_interval_hours = config["report_interval_hours"]
# else:
# config["report_interval_hours"] = report_interval_hours
report_interval_seconds = int(60 * 60 * report_interval_hours)
# add structure for elements
if "name" not in config:
config["name"] = ""
if "initials" not in config:
config["initials"] = ""
if "owner" not in config:
config["owner"] = ""
if "cursor" not in config:
config["cursor"] = ""
if "last" not in config:
config["last"] = {}
if "next" not in config:
config["next"] = {}
if "report" not in config["last"]:
config["last"]["report"] = {}
# send if no last.send in config
if "send" not in config["last"]:
add_welcome = True
# command line arguments
# send report if True
send_report = True if "report" in sys.argv else False
# reset hds. only clear config last/next and activity_history.
if "reset" in sys.argv:
config["last"] = config["next"] = {}
config["cursor"] = config["name"] = config["initials"] = config["owner"] = ""
update_config()
activity_history = []
update_activity_history()
def update_config():
global config
with open(config_file, "w") as outfile:
json.dump(config, outfile)
def load_activity_history():
global activity_history
with open("activity_history.json") as json_data_file:
activity_history = json.load(json_data_file)
def update_activity_history():
global activity_history, hs
if bool(activity_history):
# trim history. remove first 15 (oldest) elements if over 50 elements
if len(activity_history) > history_length_max:
print(f"\n{hs['time']} trimming activity_history", end="")
del activity_history[:history_length_min]
# save history details to config
if "activity_history" not in config["last"]:
config["last"]["activity_history"] = {}
config["last"]["activity_history"] = {
"count": len(activity_history),
"last": hs["now"],
"last_nice": nice_date(hs["now"]),
}
# write file
with open("activity_history.json", "w") as outfile:
json.dump(activity_history, outfile)
def get_time():
global hs
# time functions
now = datetime.now()
hs["now"] = round(datetime.timestamp(now))
hs["time"] = str(now.strftime("%H:%M %D"))
# functions
def nice_date(time):
timestamp = datetime.fromtimestamp(time)
return timestamp.strftime("%H:%M %d/%b").upper()
def nice_hotspot_name(name):
if not bool(config["name"]):
config["name"] = name.replace("-", " ").upper()
return config["name"]
def nice_hotspot_initials(name):
if not bool(config["initials"]):
name = nice_hotspot_name(name)
config["initials"] = "".join(item[0].upper() for item in name.split())
return config["initials"]
def nice_hnt_amount_or_seconds(amt):
niceNum = 0.00000001
niceNumSmall = 100000000
if isinstance(amt, float):
# float. for time i
amt_output = "{:.2f}".format(amt)
else:
# int. up to 3 decimal payments
amt_output = "{:.3f}".format(amt * niceNum)
# int. 8 decimal places for micropayments
# if amt > 0 and amt < 100000 :
if amt in range(0, 100000):
amt_output = "{:.8f}".format(amt / niceNumSmall).rstrip("0")
amt_output = f"`{amt_output}`"
return str(amt_output)
# invalid reason nice name, or raw reason if not in dict
def nice_invalid_reason(ir):
return (
invalid_reason_short_names[ir] if ir in invalid_reason_short_names else str(ir)
)
###activity type name to short name
def reward_short_name(reward_type):
return (
reward_short_names[reward_type]
if reward_type in reward_short_names
else reward_type.upper()
)
def load_activity_data():
global activities, config, hs, wellness_check, send, send_report, send_wellness_check
# try to get json or return error
status = ""
try:
# LIVE API data
activity_endpoint = (
helium_api_endpoint + "hotspots/" + config["hotspot"] + "/activity/"
)
activity_request = requests.get(activity_endpoint, headers=headers)
data = activity_request.json()
if data["data"]:
logging.debug("Got some data, will not continue with cursor.")
else:
if "cursor" in data:
logging.debug(f"Got cursor, fetching data again...")
activity_endpoint += "?cursor=" + data["cursor"]
activity_request = requests.get(activity_endpoint, headers=headers)
data = activity_request.json()
### DEV Only
###LOCAL load data.json
# with open("data.json") as json_data_file:
# data = json.load(json_data_file)
except requests.RequestException:
status = "Connectivity"
except ValueError: # includes simplejson.decoder.JSONDecodeError
status = "Parsing JSON"
except (IndexError, KeyError):
status = "JSON format"
if bool(status):
print(f"\n{hs['time']} Activity API Error: {status}")
quit()
# quit if no data
if "data" not in data:
print(f"\n{hs['time']} Activity API: Bad Data")
quit()
# set wellness_check if last.send exists
if "last" in config and "send" in config["last"]:
wellness_check = int(config["last"]["send"] + wellness_check_seconds)
# add/update cursor to config
if "cursor" not in config:
config["cursor"] = ""
if "cursor" in data and config["cursor"] != data["cursor"]:
config["cursor"] = data["cursor"]
# only send if send history. not for new users
if "send" in config["last"] and hs["now"] >= wellness_check:
print(
f"\n{hs['time']} Wellness Check, {wellness_check_hours}hrs, No New Activities",
end="",
)
send = send_wellness_check = send_report = True
# no data or send_report false
elif not data["data"] and not bool(send_report):
# print(f"{hs['time']} no activities")
print(".", end="")
quit()
# set activities, set last.send, update config
else:
send = True
activities = data["data"]
###activity type poc_receipts_v1
def poc_receipts(activity):
valid_text = "💩 Invalid"
time = nice_date(activity["time"])
txn_link = f'[🔎](<{helium_explorer_tx}{activity["hash"]}>)'
witnesses = {}
wit_count = 0
if "path" in activity and "witnesses" in activity["path"][0]:
witnesses = activity["path"][0]["witnesses"]
wit_count = len(witnesses)
# pluralize Witness
wit_plural = "es" if wit_count != 1 else ""
wit_text = f"{wit_count} Witness{wit_plural}"
# challenge accepted
if "challenger" in activity and activity["challenger"] == config["hotspot"]:
output_message.append(
f"🏁 ...Challenged Beaconer, {wit_text} `{time}` {txn_link}"
)
# beacon sent
elif (
"challengee" in activity["path"][0]
and activity["path"][0]["challengee"] == config["hotspot"]
):
# beacon sent plus witness count and valid count
valid_wit_count = 0
for wit in witnesses:
if bool(wit["is_valid"]):
valid_wit_count = valid_wit_count + 1
msg = f"🌋 Sent Beacon, {wit_text}"
if bool(wit_count):
if valid_wit_count == len(witnesses):
valid_wit_count = "All"
msg += f", {valid_wit_count} Valid"
msg += f" `{time}` {txn_link}"
output_message.append(msg)
# witnessed beacon plus valid or invalid and invalid reason
elif bool(witnesses):
vw = 0 # valid witnesses
valid_witness = False
for w in witnesses:
# valid witness count among witnesses
if "is_valid" in w and bool(w["is_valid"]):
vw = vw + 1
if w["gateway"] == config["hotspot"]:
witness_info = ""
if bool(w["is_valid"]):
valid_witness = True
valid_text = "🛸 Valid" # 🤙
witness_info = f", 1 of {wit_count}"
elif "invalid_reason" in w:
valid_text = "💩 Invalid"
witness_info = ", " + nice_invalid_reason(w["invalid_reason"])
# add valid witness count among witnesses
if bool(valid_witness) and vw >= 1:
vw = "All" if vw == len(witnesses) else vw
witness_info += f", {vw} Valid"
output_message.append(
f"{valid_text} Witness{witness_info} `{time}` {txn_link}"
)
# other
else:
ac_type = activity["type"]
output_message.append(
f"🏁 poc_receipts_v1 - {ac_type.upper()} `{time}` {txn_link}"
)
def loop_activities():
global send_report, history_repeats
if bool(activities): # and not bool(send_report):
# load history
load_activity_history()
# loop activities
for activity in activities:
# skip if activity is in history
if activity["hash"] in activity_history: # and not bool(send_report):
history_repeats = history_repeats + 1
continue # skip this element, continue for-loop
# save activity hash if not found
else:
activity_history.append(activity["hash"])
# activity time
time = nice_date(activity["time"])
txn_link = f'[🔎](<{helium_explorer_tx}{activity["hash"]}>)'
# reward
if activity["type"] == "rewards_v2":
for reward in activity["rewards"]:
rew = reward_short_name(reward["type"])
amt = nice_hnt_amount_or_seconds(reward["amount"])
output_message.append(
f"🍪 Reward 🥓{amt}, {rew} `{time}` {txn_link}"
)
# transferred data
elif activity["type"] == "state_channel_close_v1":
for summary in activity["state_channel"]["summaries"]:
packet_plural = "s" if summary["num_packets"] != 1 else ""
output_message.append(
f"🚛 Transferred {summary['num_packets']} Packet{packet_plural} ({summary['num_dcs']} DC) `{time}` {txn_link}"
)
## Disabled for Light Hotspots. no longer needed
# ...challenge accepted
# initial poc_request_v2 support
# elif activity["type"] == "poc_request_v1":
# output_message.append(f"🎲 Created Challenge... `{time}` {txn_link}")
# beacon sent, valid witness, invalid witness
# elif activity["type"] == "poc_receipts_v1":
elif activity["type"] == "poc_receipts_v1" or activity["type"] == "poc_receipts_v2":
poc_receipts(activity)
# other
else:
other_type = activity["type"]
output_message.append(f"🚀 {other_type.upper()} `{time}` {txn_link}")
def load_hotspot_data_and_status():
###hotspot data
global hs, config, add_welcome, send_wellness_check
new_balance = new_reward_scale = new_api_sync = new_status = False
# try to get json or return error
status = ""
try:
hs_endpoint = helium_api_endpoint + "hotspots/" + config["hotspot"]
hs_request = requests.get(hs_endpoint, headers=headers)
data = hs_request.json()
if not data["data"]:
print(f"no hotspot data {hs['time']}")
quit()
else:
hotspot_data = data["data"]
del hs_request
except requests.RequestException:
status = "Connectivity"
except ValueError: # includes simplejson.decoder.JSONDecodeError
status = "Parsing JSON"
except (IndexError, KeyError):
status = "JSON format"
if bool(status):
print(f"\n{hs['time']} Hotspot API Error: {status}")
quit()
# quit if no data
if "data" not in data:
print(f"\n{hs['time']} Helium Hotspot API. No 'data' key in Response")
quit()
### hotspot data
hs_add = {
"owner": hotspot_data["owner"],
"name": nice_hotspot_name(hotspot_data["name"]),
"initials": nice_hotspot_initials(hotspot_data["name"]),
"status": str(hotspot_data["status"]["online"]).upper(),
"height": hotspot_data["status"]["height"],
"block": hotspot_data["block"],
"reward_scale": "{:.2f}".format(round(hotspot_data["reward_scale"], 2)),
}
hs.update(hs_add)
del data, hotspot_data, hs_add
# add/update cursor to config. supports hotspot ownership transfers
if "owner" not in config or config["owner"] != hs["owner"]:
config["owner"] = hs["owner"]
########################################################
# dev
### API Sync
hs["api_sync"] = "*NSYNC"
block_gap_num = int(hs["block"] - hs["height"])
block_gap_num = 0 if block_gap_num <= 0 else block_gap_num
api_block_gap_exceeded = (
True if block_gap_num >= sync_blocks_behind * api_sync_lag_multiple else False
)
# config.last
if "api_sync" not in config["last"]:
new_api_sync = True
config["last"]["api_sync"] = ""
elif config["last"]["api_sync"] != hs["api_sync"]:
new_api_sync = True
# Show block gap number instead of *NSYNC if api_sync_lag_multiple is exceeded.
if bool(api_block_gap_exceeded):
last_sync = block_gap_num / 60
if last_sync > 24:
last_sync = math.floor(last_sync / 24)
last_sync = f"{last_sync}d"
else:
last_sync = math.trunc(math.floor(last_sync))
last_sync = f"{last_sync}h"
hs["api_sync"] = f"({last_sync})"
# hs["api_sync"] = f"({block_gap_num})"
## in_dev
# if exceed by add in a local bobcat Sync report using bobcat's status.json
config["last"]["api_sync"] = hs["api_sync"]
config["last"]["api_height"] = hs["height"]
config["last"]["api_block"] = hs["block"]
config["last"]["api_gap"] = block_gap_num
# Sync Display
hs["api_sync"] = (
"**" + hs["api_sync"] + "**" if bool(new_api_sync) else hs["api_sync"]
)
# Add 'API Gap: ' Text if gap
hs["api_sync"] = (
f"*NYSNC{hs['api_sync']}"
if bool(api_block_gap_exceeded)
else hs["api_sync"]
# f"API Gap:{hs['api_sync']}" if bool(api_block_gap_exceeded) else hs["api_sync"]
)
########################################################
###wallet data
wallet_request = requests.get(
helium_api_endpoint + "accounts/" + hs["owner"], headers=headers
)
w = wallet_request.json()
if "data" not in w:
print(f"\n{hs['time']} Wallet API Error", end="")
# quit()
hs["balance"] = nice_hnt_amount_or_seconds(w["data"]["balance"])
if "balance" not in config["last"]:
config["last"]["balance"] = "0"
###add to config if new
if hs["balance"] != config["last"]["balance"]:
new_balance = True
config["last"]["balance"] = hs["balance"]
del wallet_request, w
### reward_scale
if "reward_scale" not in config["last"]:
config["last"]["reward_scale"] = "0"
###add to config if new
if hs["reward_scale"] != config["last"]["reward_scale"]:
new_reward_scale = True
config["last"]["reward_scale"] = hs["reward_scale"]
### status
if "status" not in config["last"]:
config["last"]["status"] = ""
###add to config if new
if hs["status"] != config["last"]["status"]:
new_status = True
config["last"]["status"] = hs["status"]
#### STYLED status text
### bold balance if has changed
balance_styled = "**" + hs["balance"] + "**" if bool(new_balance) else hs["balance"]
### bold reward_scale if has changed
reward_scale_styled = (
"**" + hs["reward_scale"] + "**"
if bool(new_reward_scale)
else hs["reward_scale"]
)
### bold block_height if has changed
api_sync_styled = hs["api_sync"]
# api_sync_styled = f"**{hs['api_sync']}**" if bool(new_api_sync) else hs["api_sync"]
### bold status if not 'online'
status_styled = "**" + hs["status"] + "**" if bool(new_status) else hs["status"]
# default status msg
status_msg = (
"📡**"
+ hs["initials"]
+ "** 🔥"
+ status_styled
# + " ⛅️ LIGHT"
+ " 🥑LIGHT"
# + api_sync_styled
+ " 🍕"
+ reward_scale_styled
+ " 🥓"
+ balance_styled
)
# insert to top of output_message
output_message.insert(0, status_msg)
# add in wellness check message. not if new
# if not bool(add_welcome) and bool(send_wellness_check):
if (
not bool(add_welcome)
and bool(send_wellness_check)
and "send" in config["last"]
and bool(config["last"]["send"])
):
hour_plural = "s" if wellness_check_hours != 1 else ""
lapse_msg = (
f"`🚧 No API Activities in the Last {wellness_check_hours}hr{hour_plural} `"
)
output_message.insert(0, lapse_msg)
def discord_send():
global send, add_welcome, send_report, output_message_length
# send if no last.send in config
if "last" in config and "send" not in config["last"]:
send = add_welcome = True
# send if more than 1 (default) msg
elif len(output_message) > 1:
send = True
# unless send report, don't send, repeats only
elif not bool(send_report):
send = False
print(":", end="") # print(f"{hs['time']} repeat activities")
quit()
# add welcome msg to output if no config[last][send]
if bool(add_welcome):
output_message.insert(0, f"🤙 **{hs['name']} [ 📡 {hs['initials']} ]**")
print(f"\n{hs['time']} Welcome msg added", end="")
if bool(send):
# only send activity, remove status if recently sent. keep if report
if (
"last" in config
and "send" in config["last"]
and hs["now"] < (config["last"]["send"] + interval_pop_status_seconds)
):
output_message.pop(0)
# update last.send to be last status sent
config["last"]["send"] = hs["now"]
config["last"]["send_nice"] = nice_date(config["last"]["send"])
output_message_length = len(output_message)
discord_message = "\n".join(output_message)
### Dev only
# print(discord_message)
# exit()
webhook = DiscordWebhook(url=config["discord_webhook"], content=discord_message)
# send
webhook_response = webhook.execute()
return webhook_response.reason
#########################
### main
def main():
# time_execution
time_execution_t0 = time.time()
get_time()
load_config()
load_activity_data()
### Dev only
# print(activities)
# exit()
# if activity data...
load_hotspot_data_and_status()
loop_activities()
# if bobcat set in config
local_bobcat_miner_report()
# local_bobcat_sync_status()
# send
discord_response_reason = discord_send()
# write history
update_activity_history()
# write config
update_config()
# time_execution end
time_execution_t1 = time.time()
time_execution_seconds = nice_hnt_amount_or_seconds(
time_execution_t1 - time_execution_t0
)
# cron log
print(
f"\n{hs['time']} a:{str(len(activities))} r:{str(history_repeats)} m:{str(output_message_length)} discord:{discord_response_reason} sec:{time_execution_seconds}"
)
### execute main() if main is first module
if __name__ == "__main__":
main()