-
Notifications
You must be signed in to change notification settings - Fork 8
/
publishes_claims.py
705 lines (578 loc) · 25.3 KB
/
publishes_claims.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
#!/usr/bin/env python3
# --------------------------------------------------------------------------- #
# The MIT License (MIT) #
# #
# Copyright (c) 2023 Eliud Cabrera Castillo <[email protected]> #
# #
# Permission is hereby granted, free of charge, to any person obtaining #
# a copy of this software and associated documentation files #
# (the "Software"), to deal in the Software without restriction, including #
# without limitation the rights to use, copy, modify, merge, publish, #
# distribute, sublicense, and/or sell copies of the Software, and to permit #
# persons to whom the Software is furnished to do so, subject to the #
# following conditions: #
# #
# The above copyright notice and this permission notice shall be included #
# in all copies or substantial portions of the Software. #
# #
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR #
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, #
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL #
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER #
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING #
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER #
# DEALINGS IN THE SOFTWARE. #
# --------------------------------------------------------------------------- #
"""Functions to get the list of published claims in the LBRY network."""
import time
import requests
import lbrytools.funcs as funcs
import lbrytools.search_utils as sutils
import lbrytools.publishes_ch as pubch
def get_channel_claims(wallet_id="default_wallet",
is_spent=False, reverse=False,
server="http://localhost:5279"):
"""Get all published claims by channel in the wallet."""
ch_claims = []
channels = pubch.get_channels(wallet_id=wallet_id,
is_spent=False, reverse=False,
server=server)
if not channels:
return ch_claims
for channel in channels:
msg = {"method": "claim_list",
"params": {"wallet_id": wallet_id,
"page_size": 99000,
"resolve": True,
"channel_id": channel["claim_id"]}}
if is_spent:
msg["params"]["is_spent"] = True
output = requests.post(server, json=msg).json()
if "error" in output:
name = output["error"]["data"]["name"]
mess = output["error"].get("message", "No error message")
print(f">>> {name}: {mess}")
continue
claims = output["result"]["items"]
# Order the claims by 'release_time' which exists for streams
# (video, audio, documents).
# For other claims (reposts, collections) it will use
# 'creation_timestamp'. For spent transactions, 'creation_timestamp'
# doesn't exist so we just use 'timestamp'.
for claim in claims:
if "relase_time" not in claim["value"]:
if "creation_timestamp" in claim["meta"]:
claim["value"]["release_time"] = \
claim["meta"]["creation_timestamp"]
else:
claim["value"]["release_time"] = claim["timestamp"]
claims = sorted(claims,
key=lambda c: int(c["value"]["release_time"]),
reverse=reverse)
ds = sutils.downloadable_size(claims, local=False, print_msg=False)
ch_name = channel["canonical_url"].split("lbry://")[1]
ch_claims.append({"name": ch_name,
"id": channel["claim_id"],
"address": channel["address"],
"claims": claims,
"size": ds["size"],
"duration": ds["duration"]})
return ch_claims
def get_anon_claims(wallet_id="default_wallet",
is_spent=False, reverse=False,
server="http://localhost:5279"):
"""Get all published claims that are not published by a channel."""
msg = {"method": "claim_list",
"params": {"wallet_id": wallet_id,
"page_size": 99000,
"resolve": True}}
if is_spent:
msg["params"]["is_spent"] = True
output = requests.post(server, json=msg).json()
if "error" in output:
name = output["error"]["data"]["name"]
mess = output["error"].get("message", "No error message")
print(f">>> {name}: {mess}")
return False
claims = output["result"]["items"]
# Only pick claims without a channel
anon_claims = []
for claim in claims:
if ("signing_channel" not in claim
and claim["value_type"] not in "channel"):
# Order the claims by 'release_time' which exists for streams
# (video, audio, documents).
# For other claims (reposts, collections) it will use
# 'creation_timestamp'. For spent transactions,
# 'creation_timestamp' doesn't exist so we just use 'timestamp'.
if "relase_time" not in claim["value"]:
if "creation_timestamp" in claim["meta"]:
claim["value"]["release_time"] = \
claim["meta"]["creation_timestamp"]
else:
claim["value"]["release_time"] = claim["timestamp"]
anon_claims.append(claim)
if not anon_claims:
return False
anon_claims = sorted(anon_claims,
key=lambda c: int(c["value"]["release_time"]),
reverse=reverse)
anon_ds = sutils.downloadable_size(anon_claims, local=False,
print_msg=False)
unknown = {"name": "_Unknown_",
"id": None,
"address": None,
"claims": anon_claims,
"size": anon_ds["size"],
"duration": anon_ds["duration"]}
return unknown
def get_claims(wallet_id="default_wallet",
is_spent=False, reverse=False,
server="http://localhost:5279"):
"""Get all claims published by channels or published anonymously.
Parameters
----------
wallet_id: str, optional
It defaults to 'default_wallet', in which case it will search
the accounts created in the default wallet created by `lbrynet`.
is_spent: bool, optional
It defaults to `False`, in which case it will show
active claims with a transaction that hasn't been spent.
If it is `True` it will get the claims with transactions
that have already been spent. This means it will show
some claims that are expired or no longer exist.
reverse: bool, optional
It defaults to `False`, in which case older items come first
in the output list.
If it is `True` newer claims are at the beginning of the list.
server: str, optional
It defaults to `'http://localhost:5279'`.
This is the address of the `lbrynet` daemon, which should be running
in your computer before using any `lbrynet` command.
Normally, there is no need to change this parameter from its default
value.
Returns
-------
list of dict
Each element in the list is a dictionary with the information
corresponding to a channel and its claims.
Each dictionary contains 6 keys:
- 'name': name of the channel starting with '@'.
- 'id': claim ID of the channel (40-character string).
- 'address': 34-character address that created the channel claim.
- 'claims': list of dictionaries corresponding to the output
of `claim_list`, indicating the claims published by this channel.
- 'size': total size of all claims in the channel, in bytes.
- 'duration': total duration of all claims in the channel, in seconds.
There may be one special dictionary with 'name' being '_Unknown_',
that is, not starting with '@'. This dictionary has the claims
that were published anonymously, that is, not under any channel.
For this dictionary 'id' and 'address' will be `None`.
False
If there is a problem, such as non-existing `wallet_id`,
it will return `False`.
"""
ch_claims = get_channel_claims(wallet_id=wallet_id,
is_spent=is_spent, reverse=reverse,
server=server)
unknown = get_anon_claims(wallet_id=wallet_id,
is_spent=is_spent, reverse=reverse,
server=server)
if unknown:
ch_claims.append(unknown)
return ch_claims
def print_s_claims(claims, output=None,
updates=False, claim_id=False, addresses=False,
typ=False, amounts=True, ch_name=False,
title=False, sanitize=False,
start=1, end=0,
reverse=False,
sep=";"):
"""Prepare output list in order to print the claims."""
if not output:
output = []
if reverse:
claims.reverse()
n_claims = len(claims)
if n_claims < 1:
output.append(" No claims")
for num, claim in enumerate(claims, start=1):
if num < start:
continue
if end != 0 and num > end:
break
meta = claim["meta"]
value = claim["value"]
rels_time = int(value.get("release_time", 0))
if not rels_time:
rels_time = meta.get("creation_timestamp", 0)
rels_time = time.strftime(funcs.TFMTp, time.gmtime(rels_time))
claim_op = claim["claim_op"]
timestamp = claim["timestamp"]
timestamp = time.strftime(funcs.TFMTp, time.gmtime(timestamp))
cid = claim["claim_id"]
ad = claim["address"]
vtype = claim["value_type"]
stream_type = value.get("stream_type", 8 * "_")
if "source" in value:
mtype = value["source"].get("media_type", 14 * "_")
else:
mtype = 14 * "_"
amount = float(claim["amount"])
t_amount = float(meta.get("effective_amount", 0))
if "signing_channel" in claim:
if "canonical_url" in claim["signing_channel"]:
channel = claim["signing_channel"]["canonical_url"]
channel = channel.split("lbry://")[1]
else:
channel = claim["signing_channel"]["permanent_url"]
_ch, _id = channel.split("#")
_ch = _ch.split("lbry://")[1]
channel = _ch + "#" + _id[0:3]
else:
channel = 14 * "_"
rep = meta.get("reposted", 0)
seconds = 0
if "video" in value and "duration" in value["video"]:
seconds = value["video"]["duration"]
if "audio" in value and "duration" in value["audio"]:
seconds = value["audio"]["duration"]
sec = seconds % 60
mi = seconds // 60
duration = f"{mi:3d}:{sec:02d}"
size = 0
if "source" in value and "size" in value["source"]:
size = float(value["source"]["size"])
size_mb = size / (1024**2) # to MB
name = claim["name"]
if title:
name = value.get("title") or name
name = '"' + name + '"'
if sanitize:
name = funcs.sanitize_text(name)
channel = funcs.sanitize_text(channel)
line = f"{num:4d}/{n_claims:4d}" + f"{sep} "
line += f"{rels_time}" + f"{sep} "
if updates:
line += f"{claim_op}" + f"{sep} "
line += f"{timestamp}" + f"{sep} "
if claim_id:
line += f"{cid}" + f"{sep} "
if addresses:
line += f"add. {ad}" + f"{sep} "
if typ:
line += f"{vtype:10s}" + f"{sep} "
line += f"{stream_type:9s}" + f"{sep} "
line += f"{mtype:17s}" + f"{sep} "
if amounts:
line += f"{amount:14.8f}" + f"{sep} "
line += f"{t_amount:14.8f}" + f"{sep} "
if ch_name:
line += f"{channel}" + f"{sep} "
line += f"r.{rep:3d}" + f"{sep} "
line += f"{duration}" + f"{sep} "
line += f"{size_mb:9.4f} MB" + f"{sep} "
line += f"{name}"
output.append(line)
return output
def print_claims(ch_claims,
updates=False, claim_id=False, addresses=False,
typ=False, amounts=True, ch_name=False,
title=False, sanitize=False,
file=None, fdate=False, sep=";"):
"""Print the list of channels and claims."""
n_chs = len(ch_claims)
out = []
t_n_claims = 0
t_size = 0
t_duration = 0
t_n_an_claims = 0
t_an_size = 0
t_an_duration = 0
anon_exists = False
is_anon = False
for ch_claim in ch_claims:
if ch_claim["name"] in "_Unknown_":
anon_exists = True
n_chs = n_chs - 1
for n_ch, ch_claim in enumerate(ch_claims, start=1):
chan_name = ch_claim["name"]
if chan_name in "_Unknown_":
is_anon = True
if sanitize:
chan_name = funcs.sanitize_text(chan_name)
chan_name = '"' + chan_name + '"'
chan_id = ch_claim["id"]
chan_add = ch_claim["address"]
chan_size = ch_claim["size"]
chan_duration = ch_claim["duration"]
claims = ch_claim["claims"]
if is_anon:
t_n_an_claims += len(claims)
t_an_size += chan_size
t_an_duration += chan_duration
else:
t_n_claims += len(claims)
t_size += chan_size
t_duration += chan_duration
GB = chan_size / (1024**3) # to GiB
hrs = chan_duration / 3600
days = hrs / 24
hr = chan_duration // 3600
mi = (chan_duration % 3600) // 60
sec = (chan_duration % 3600) % 60
if is_anon:
line = ""
else:
line = f"{n_ch:2d}/{n_chs:2d}" + f"{sep} "
line += f"{chan_name}" + f"{sep} "
line += f"{chan_id}" + f"{sep} "
line += f"{chan_add}" + f"{sep} "
line += f"{GB:.4f} GiB" + f"{sep} "
line += f"{hr} h {mi} min {sec} s, or {days:.4f} days"
out.append(line)
out = print_s_claims(claims, output=out,
updates=updates,
claim_id=claim_id, addresses=addresses,
typ=typ, amounts=amounts, ch_name=ch_name,
title=title, sanitize=sanitize,
sep=sep)
if not is_anon:
if n_ch < n_chs or anon_exists:
out.append("")
funcs.print_content(out, file=file, fdate=fdate)
def print_claims_summary(ch_claims,
file=None, fdate=False):
"""Print a summary paragraph of the channel claims."""
n_chs = len(ch_claims)
t_n_claims = 0
t_size = 0
t_duration = 0
t_n_anon_claims = 0
t_anon_size = 0
t_anon_duration = 0
is_anon = False
for ch_claim in ch_claims:
if ch_claim["name"] in "_Unknown_":
n_chs = n_chs - 1
for ch_claim in ch_claims:
if ch_claim["name"] in "_Unknown_":
is_anon = True
chan_size = ch_claim["size"]
chan_duration = ch_claim["duration"]
claims = ch_claim["claims"]
if is_anon:
t_n_anon_claims += len(claims)
t_anon_size += chan_size
t_anon_duration += chan_duration
else:
t_n_claims += len(claims)
t_size += chan_size
t_duration += chan_duration
t_GB = t_size / (1024**3) # to GiB
t_hrs = t_duration / 3600
t_days = t_hrs / 24
t_hr = t_duration // 3600
t_mi = (t_duration % 3600) // 60
t_sec = (t_duration % 3600) % 60
t_anon_GB = t_anon_size / (1024**3) # to GiB
t_anon_hrs = t_anon_duration / 3600
t_anon_days = t_anon_hrs / 24
t_anon_hr = t_anon_duration // 3600
t_anon_mi = (t_anon_duration % 3600) // 60
t_anon_sec = (t_anon_duration % 3600) % 60
out1 = [40 * "-",
f"Total unique channels: {n_chs}",
f"Total claims in channels: {t_n_claims}",
f"Total download size: {t_GB:.4f} GiB",
f"Total duration: {t_hr} h {t_mi} min {t_sec} s, "
f"or {t_days:.4f} days"]
out2 = [40 * "-",
f"Anonymous unique claims: {t_n_anon_claims}",
f"Total download size of anonymous claims: {t_anon_GB:.4f} GiB",
"Total duration of anonymous claims: "
f"{t_anon_hr} h {t_anon_mi} min {t_anon_sec} s, "
f"or {t_anon_days:.4f} days"]
out = []
if t_n_claims > 0:
out += out1
if t_n_anon_claims > 0:
out += out2
funcs.print_content(out, file=file, fdate=fdate)
return {"n_channels": n_chs,
"n_ch_claims": t_n_claims,
"chs_size": t_GB,
"chs_hr": t_hr,
"chs_min": t_mi,
"chs_sec": t_sec,
"chs_days": t_days,
"n_anon_claims": t_n_anon_claims,
"anon_size": t_anon_GB,
"anon_hr": t_anon_hr,
"anon_min": t_anon_mi,
"anon_sec": t_anon_sec,
"anon_days": t_anon_days}
def list_claims(wallet_id="default_wallet", is_spent=False,
channel=None, channel_id=None, anon=False,
updates=False, claim_id=False, addresses=False,
typ=False, amounts=True, ch_name=False,
title=False,
reverse=False, sanitize=False,
file=None, fdate=False, sep=";",
server="http://localhost:5279"):
"""List all claims published by channels or published anonymously.
Parameters
----------
wallet_id: str, optional
It defaults to 'default_wallet', in which case it will search
the accounts created in the default wallet created by `lbrynet`.
is_spent: bool, optional
It defaults to `False`, in which case it will show
active claims with a transaction that hasn't been spent.
If it is `True` it will get the claims with transactions
that have already been spent. This mean it will show
claims that are expired or no longer exist.
channel: str, optional
It defaults to `None` in which case it will print
the claims of all channels, as long as `channel_id` is also `None`.
If it is the name of one of our channels, it will display the claims
only for this channel.
channel_id: str, optional
It defaults to `None` in which case it will print
the claims of all channels, as long as `channel` is also `None`.
If it is a 40-character claim ID, it will display the claims
only for the channel with this claim ID.
anon: bool, optional
It defaults to `False`.
If it is `True` it will only print our claims published without
a channel, that is, that were published anonymously.
updates: bool, optional
It defaults to `False`.
If it is `True` it will print the last time the claims were
created or updated.
claim_id: bool, optional
It defaults to `False`.
If it is `True` it will print the claim ID of the claims.
addresses: bool, optional
It defaults to `False`.
If it is `True` it will print the 34-character address
that created the claim.
typ: bool, optional
It defaults to `False`.
If it is `True` it will print the type of claim, the type of stream,
and the media type, if any.
amounts: bool, optional
It defaults to `True`, in which case it prints two quantities,
the amount of LBC that is originally staked on the claim,
and the total amount, which may include tips and supports
by other people.
ch_name: bool, optional
It defaults to `False`.
If it is `True` it will print the channel that published the claims.
title: bool, optional
It defaults to `False`.
If it is `True` it will print the claim title instead of
the claim name.
reverse: bool, optional
It defaults to `False`, in which case older items come first
in the output list.
If it is `True` newer claims are at the beginning of the list.
sanitize: bool, optional
It defaults to `False`, in which case it will not remove the emojis
from the name and title of the claim and channel.
If it is `True` it will remove these unicode characters.
This option requires the `emoji` package to be installed.
file: str, optional
It defaults to `None`.
It must be a writable path to which the summary will be written.
Otherwise the summary will be printed to the terminal.
fdate: bool, optional
It defaults to `False`.
If it is `True` it will add the date to the name of the summary file.
sep: str, optional
It defaults to `;`. It is the separator character between
the data fields in the printed summary. Since the claim name
can have commas, a semicolon `;` is used by default.
server: str, optional
It defaults to `'http://localhost:5279'`.
This is the address of the `lbrynet` daemon, which should be running
in your computer before using any `lbrynet` command.
Normally, there is no need to change this parameter from its default
value.
Returns
-------
dict
There are two keys
- 'summary': a dictionary with 13 keys 'n_channels' (number
of channels), 'n_ch_claims' (number of claims in channels),
'chs_size' (GB), 'chs_hr' (hours), 'chs_min' (minutes),
'chs_sec' (seconds), 'chs_days' (days),
'n_anon_claims' (number of anonymous claims), and similarly
'anon_size', 'anon_hr', 'anon_min', 'anon_sec', and 'anon_days'.
- 'ch_claims': a list of dict.
Each element in the list is a dictionary with the information
corresponding to a channel and its claims.
Each dictionary contains 6 keys:
- 'name': name of the channel starting with '@'.
- 'id': claim ID of the channel (40-character string).
- 'address': 34-character address that created the channel claim.
- 'claims': list of dictionaries corresponding to the output
of `claim_list`, indicating the claims published by this channel.
If there are no claims, the list will be empty.
- 'size': total size of all claims in the channel, in bytes.
- 'duration': total duration of all claims in the channel,
in seconds.
If `channel` or `channel_id` were used to find a channel,
or `anon=True`, the list will only contain at most one dictionary.
If `anon=True` the single dictionary in the list will contain
the information of the claims that were published anonymously,
that is, not under any channel.
For this dictionary 'name' will be '_Unknown_' (it will not start
with '@'), and 'id' and 'address' will be `None`.
False
If there is a problem, such as non-existing `wallet_id`,
it will return `False`.
"""
print("Claims in the wallet")
print(80 * "-")
if not funcs.server_exists(server=server):
return False
ch_claims = get_claims(wallet_id=wallet_id,
is_spent=is_spent, reverse=reverse,
server=server)
if not ch_claims:
return False
found = False
# Filter the list of channels by the entered name or claim ID
if anon:
channel = "_Unknown_"
if channel or channel_id:
if channel:
if channel in "_Unknown_":
# Special name not starting with @ for anonymous claims
anon = True
elif not channel.startswith("@"):
channel = "@" + channel
for ch in ch_claims:
chan_name = ch["name"]
chan_id = ch["id"]
if (channel and channel in chan_name
or channel_id and channel_id in chan_id):
print(f"Found match")
ch_claims = [ch]
found = True
break
if not found:
if anon:
print(f"No anonymous claims")
else:
print(f'Not found: "{channel}", "{channel_id}"')
return False
print_claims(ch_claims,
updates=updates, claim_id=claim_id, addresses=addresses,
typ=typ, amounts=amounts, ch_name=ch_name,
title=title, sanitize=sanitize,
file=file, fdate=fdate, sep=sep)
summary = print_claims_summary(ch_claims, file=None, fdate=False)
return {"ch_claims": ch_claims,
"summary": summary}