Compare commits
3499 commits
managed-st
...
master
Author | SHA1 | Date | |
---|---|---|---|
|
eb5da9511e | ||
|
8722ef840e | ||
|
6e75a1a89b | ||
|
ef3189de1d | ||
|
c2d2080034 | ||
|
d0b5a0a8fd | ||
|
1d0e17be21 | ||
|
4ef03bb1f4 | ||
|
4bd4bcdc27 | ||
|
e5ca967fa2 | ||
|
eed7d02e8b | ||
|
02aecad52b | ||
|
585962d930 | ||
|
ea4fba39a6 | ||
|
7a86406746 | ||
|
c8a3eb97a4 | ||
|
20213628d7 | ||
|
2d1649f972 | ||
|
5cb04b86a0 | ||
|
93ab6b3be3 | ||
|
b9762c3e64 | ||
|
82592d00ef | ||
|
c118174c1a | ||
|
d284acd8b8 | ||
|
235c98372d | ||
|
d2f5073ef4 | ||
|
84e5e43117 | ||
|
7bd025ae54 | ||
|
8f28ce65b0 | ||
|
d36e305129 | ||
|
2609dee8fb | ||
|
a2da86d4b5 | ||
|
aa16c7fee5 | ||
|
3266f72b82 | ||
|
77cd2a3f8a | ||
|
308e586e9a | ||
84beddfd77 | |||
|
6258651650 | ||
|
cc5f0b6630 | ||
|
f64d507d39 | ||
|
001819d5c2 | ||
|
8b4c046d28 | ||
|
2c20ad6c43 | ||
|
9e610cc54c | ||
|
b9d25c6d01 | ||
|
419b5b45f2 | ||
|
516c2dd5d0 | ||
|
b99102f9c9 | ||
|
8c6c7b655c | ||
|
48c6873fc4 | ||
|
15dc52bd9a | ||
|
52d555078f | ||
|
cc976bd010 | ||
|
9cc6992011 | ||
|
a1b87460c5 | ||
|
007e1115c4 | ||
|
20ae51b949 | ||
|
24e9c7b435 | ||
|
b452e76e1d | ||
|
341834c30d | ||
|
12bac730bd | ||
|
1027337833 | ||
|
97fef21f75 | ||
|
9dafd5f69b | ||
|
fd4f0b2049 | ||
|
734f0651a4 | ||
|
94deaf55df | ||
|
d957d46f96 | ||
|
0217aede3d | ||
|
e4e1600f51 | ||
|
d0aad8ccaf | ||
|
ab50cfa5c1 | ||
|
5a26aea398 | ||
|
bd1cebdb4c | ||
|
ec433f069f | ||
|
cd6d3fec9c | ||
|
8c474a69de | ||
|
8903056648 | ||
|
749a92d0e5 | ||
|
a7d7efecc7 | ||
|
c88f0797a3 | ||
|
137ebd503d | ||
|
c3f5dd780e | ||
|
20b1865879 | ||
|
231b982422 | ||
|
fd69401791 | ||
|
718d046833 | ||
|
e10f57d1ed | ||
|
8a033d58df | ||
|
c07c369a28 | ||
|
5be990fc55 | ||
|
8f26010c04 | ||
|
3021962e3d | ||
|
84d89ce5af | ||
|
0961cad716 | ||
|
5c543cb374 | ||
|
f78d7896a5 | ||
|
78a28de2aa | ||
|
45a255e7a2 | ||
|
d2738c2e72 | ||
|
a7c7ab7f7b | ||
|
988f288715 | ||
|
38e9b5b432 | ||
|
f7455600cc | ||
|
c7c2d6fe5a | ||
|
c6c0228970 | ||
|
8d9d2c76ae | ||
|
0b059a5445 | ||
|
ab67f417ee | ||
|
0e7a1aee0a | ||
|
d0497cf6b5 | ||
|
c38573d5de | ||
|
f077e56cec | ||
|
5e58c2f224 | ||
|
cc64789e96 | ||
|
b5c390ca04 | ||
|
da2ffb000e | ||
|
df77392fe0 | ||
|
9aa9ecdc0a | ||
|
43b45a939b | ||
|
e2922a434f | ||
|
0d6125de0b | ||
|
13af7800c2 | ||
|
47a5d37d7c | ||
|
4a3a7e318d | ||
|
85ff487af5 | ||
|
62eb9d5c75 | ||
|
cfe5c8de8a | ||
|
0497698c5b | ||
|
508bdb8e94 | ||
|
cd42f0d726 | ||
|
2706b66a92 | ||
|
29c2d5715d | ||
|
965389b759 | ||
|
174439f517 | ||
|
baf422fc03 | ||
|
61f7fbe230 | ||
|
c6c27925b7 | ||
|
be4c62cf32 | ||
|
443a1c32fa | ||
|
90c2a58470 | ||
|
adc79ec404 | ||
|
137d8ca4ac | ||
|
abf4d888af | ||
|
6c350e57dd | ||
|
fb7a93096e | ||
|
7ea88e7b31 | ||
|
2361e34541 | ||
|
be06378437 | ||
|
a334a93757 | ||
|
e3ee3892b2 | ||
|
d61accea1a | ||
|
e887453aa5 | ||
|
c3e4f0b988 | ||
|
318728aebd | ||
|
d8c1aaebc2 | ||
|
d7b65c15d2 | ||
|
972db80246 | ||
|
0d343ecb2f | ||
|
01cd95fe46 | ||
|
6dc57fc02c | ||
|
10df0c1fba | ||
|
ec751e5add | ||
|
3e3974f813 | ||
|
ec82486e15 | ||
|
e16f6b07b8 | ||
|
9a842c273b | ||
|
40f7d3ee4b | ||
|
1dc2f0458b | ||
|
3924b28cc3 | ||
|
020487b6a0 | ||
|
14037c9b2f | ||
|
0cb37a5c4b | ||
|
fa5f3e7e55 | ||
|
30aa0724ec | ||
|
059890e4e5 | ||
|
9654d4f003 | ||
|
956b52a2c1 | ||
|
2e975c8b61 | ||
|
656e299100 | ||
|
352e45b6b7 | ||
|
a9a1076362 | ||
|
6d370b0a12 | ||
|
c9fac27b66 | ||
|
59bc0b9682 | ||
|
ba60aeeebc | ||
|
dc427ecf6c | ||
|
87b4404767 | ||
|
ba9ac489c3 | ||
|
7049629ad7 | ||
|
3ae4aeea47 | ||
|
8becf1f69f | ||
|
582f79ba1c | ||
|
3c28d869f4 | ||
|
fe61b90610 | ||
|
c04fbb2908 | ||
|
571e71b28e | ||
|
39fcfcccfb | ||
|
2313d30996 | ||
|
ac7e94c6ed | ||
|
a391fe9fc7 | ||
|
ea8adc5367 | ||
|
0ea8ba72dd | ||
|
7a8d5da0e8 | ||
|
da30f003e8 | ||
|
6257948ad7 | ||
|
a7f606d62c | ||
|
1d95eb1549 | ||
|
e5e9873f79 | ||
|
530f9c72ea | ||
|
fad84c771c | ||
|
fe07aac79c | ||
|
91a6eae831 | ||
|
5852fcd287 | ||
|
4767bb9dee | ||
|
82d7f81f41 | ||
|
b036961954 | ||
|
5c708e1c6f | ||
|
9436600267 | ||
|
4ab29c4d5f | ||
|
6944c4a7c4 | ||
|
2735484fae | ||
|
03b0d5e250 | ||
|
629812337b | ||
|
e54cc8850c | ||
|
7cba51ca7d | ||
|
3dc145fe68 | ||
|
7d560df9fd | ||
|
b3f894e480 | ||
|
235cc5dc05 | ||
|
c276053301 | ||
|
2e85e29ef1 | ||
|
1169a02c8b | ||
|
a7cea4082e | ||
|
7e6ea97499 | ||
|
3c46cc4fdd | ||
|
6e5c7a1927 | ||
|
4e09b35012 | ||
|
16a2023bbd | ||
|
99fc7178c1 | ||
|
d4aca89a48 | ||
|
2918d8c7b4 | ||
|
407c570f8b | ||
|
e299a9c159 | ||
|
cc4a578578 | ||
|
0e4f1eae5b | ||
|
eccf0e6234 | ||
|
a3da041412 | ||
|
2f1617eee4 | ||
|
05124d41ae | ||
|
42fd1c962e | ||
|
47e432b4bb | ||
|
61c99abcf1 | ||
|
28fdd62945 | ||
|
3855db6c66 | ||
|
30acde0afc | ||
|
2d9c5742c7 | ||
|
43e50f7f04 | ||
|
888e9918a6 | ||
|
9e9a64d989 | ||
|
7acaecaed2 | ||
|
24eb189b7f | ||
|
2344aca146 | ||
|
758f9deafe | ||
|
7b425eb2ac | ||
|
30e8728f7f | ||
|
3989eef84b | ||
|
dc6f8c4fc4 | ||
|
2df8a1d99d | ||
|
4ea858fdd3 | ||
|
006391dd26 | ||
|
4a0bf8a702 | ||
|
d0e715feb9 | ||
|
fd73412f12 | ||
|
3819552861 | ||
|
ca6fd5b7b9 | ||
|
b8867cd18c | ||
|
8209eafc6b | ||
|
858e72a555 | ||
|
d3880fffa0 | ||
|
0a51898722 | ||
|
63cef81015 | ||
|
9279865078 | ||
|
fba7fc7aba | ||
|
a3d9d5bce7 | ||
|
23ecbc8ebe | ||
|
42b2dbd92e | ||
|
37eb55375a | ||
|
94bf357817 | ||
|
eca69391ef | ||
|
d0c5b32a90 | ||
|
84ef52cf4d | ||
|
8fb14bf713 | ||
|
16eb50a291 | ||
|
dd503fbb82 | ||
|
ae79314869 | ||
|
0cbc514a8e | ||
|
5777f3e15c | ||
|
8cdcd770c0 | ||
|
2d20458bc2 | ||
|
2bd2088248 | ||
|
5818270803 | ||
|
79a5f0e375 | ||
|
c830784f65 | ||
|
3fc538104d | ||
|
96490fdb15 | ||
|
5a0c225c6f | ||
|
c3e524cb8b | ||
|
9faf6e46ca | ||
|
e89acac235 | ||
|
200761ff13 | ||
|
cb78e95e3d | ||
|
f01cf98d62 | ||
|
c9c2495611 | ||
|
aac72fa512 | ||
|
c5e2f19dde | ||
|
34bd9e5cb4 | ||
|
ad489ed606 | ||
|
bb541901d9 | ||
|
ca4ba19a5e | ||
|
f05943ff79 | ||
|
7ded8a1333 | ||
|
c2478d4add | ||
|
f69747bc89 | ||
|
441cc950aa | ||
|
a76a0ac8c4 | ||
|
8b1009161a | ||
|
868a620e91 | ||
|
a0e34b0bc8 | ||
|
612dbcb2f3 | ||
|
b3614d965d | ||
|
5d7137255e | ||
|
6ff867ef55 | ||
|
c14915df29 | ||
|
7d4966e2ae | ||
|
3876e0317d | ||
|
0b2b10f759 | ||
|
9a79b33664 | ||
|
af1a6edd15 | ||
|
b78929f4d5 | ||
|
fb6e342043 | ||
|
0faa2d35da | ||
|
511e57c231 | ||
|
d762d675c4 | ||
|
3fdadee87c | ||
|
1aa4d9d585 | ||
|
8019f4bdb3 | ||
|
ca65c1ebc5 | ||
|
f0e47aae86 | ||
|
dc7cd545ba | ||
|
76bd59d82e | ||
|
461687ffb4 | ||
|
dd5b9ca81b | ||
|
89ed04f8a7 | ||
|
ec0d9f06c5 | ||
|
03b59ac6fc | ||
|
43ac3336d7 | ||
|
d12c78db74 | ||
|
bfaf1b0957 | ||
|
bb60c385d5 | ||
|
c96d1d9c32 | ||
|
7c7a0d4bdf | ||
|
cc829a7bf4 | ||
|
e0ea6383e2 | ||
|
bcec5dc2ae | ||
|
cba9c16a06 | ||
|
dd68fb077b | ||
|
c2294e97db | ||
|
c0f512ace7 | ||
|
3305eb67c6 | ||
|
c9d637b4da | ||
|
ae3e8fadf5 | ||
|
a1abd94387 | ||
|
9b463a8cab | ||
|
babc54a240 | ||
|
5836a93b21 | ||
|
557348e345 | ||
|
9adfec6b00 | ||
|
3a496902f8 | ||
|
b5ead91746 | ||
|
302461b446 | ||
|
ac201c718e | ||
|
f78e3825ca | ||
|
0618053bd4 | ||
|
8e6fa3490c | ||
|
8a1a1a4000 | ||
|
fd9dcbf9a8 | ||
|
beb8583436 | ||
|
b44e2c0b38 | ||
|
06e94640b5 | ||
|
ff36bdc802 | ||
|
46f576de46 | ||
|
7b09c34fce | ||
|
a22f50aa84 | ||
|
2d9130b4e0 | ||
|
470ee72462 | ||
|
add147b409 | ||
|
371df6e6c2 | ||
|
7ed5fe8f66 | ||
|
a6ca7a6f38 | ||
|
1c857b8dd8 | ||
|
87ff3f95ff | ||
|
5cb4c06d0c | ||
|
e7d9079389 | ||
|
9cdcff0e1e | ||
|
a4dce8cf9f | ||
|
aaa11c02bf | ||
|
d2ebbf5db6 | ||
|
e6efc1ad4a | ||
|
a8523996a9 | ||
|
f586de2bbe | ||
|
7df02303b2 | ||
|
f89c75e642 | ||
|
d2c1961101 | ||
|
2a4c5a48bf | ||
|
5f5f39a4aa | ||
|
df54cc04af | ||
|
0439616480 | ||
|
19fa274227 | ||
|
8076000c27 | ||
|
c80b30f070 | ||
|
486d5c48b0 | ||
|
4822792ee2 | ||
|
569f1d42b1 | ||
|
23c10faff5 | ||
|
1eaa195363 | ||
|
fb57cfa5d8 | ||
|
d33086c8f7 | ||
|
d815a6f02c | ||
|
8216f4a873 | ||
|
e4cc4521d9 | ||
|
6bd9b3744d | ||
|
f741b00768 | ||
|
5eb95d7dd4 | ||
|
e5268f43e7 | ||
|
54d6fb9da4 | ||
|
3d5c9cc1c2 | ||
|
442326f1d8 | ||
|
d66f46e07b | ||
|
757b53443d | ||
|
3436965b33 | ||
|
df71132957 | ||
|
1b322dc404 | ||
|
58341f4ff1 | ||
|
0d3ca80008 | ||
|
63437712cd | ||
|
26d0e87f46 | ||
|
2cad4fa1ce | ||
|
7bb293e5d6 | ||
|
e4777f9314 | ||
|
3508f562a7 | ||
|
1aa66c6038 | ||
|
e7458edb72 | ||
|
7f97013703 | ||
|
9e43060d41 | ||
|
d69486fb6e | ||
|
d4ebfdbc3c | ||
|
e00c3db71a | ||
|
11c3ea0b87 | ||
|
7531401623 | ||
|
e6c1dc251e | ||
|
dca7977051 | ||
|
d19e07d661 | ||
|
751ff6e21f | ||
|
3f6fe995b8 | ||
|
1e00fb369d | ||
|
54b522383a | ||
|
90a7de3b5c | ||
|
3fe1582432 | ||
|
85eddd2100 | ||
|
f5f8775c59 | ||
|
0ca98678f7 | ||
|
a19060c08d | ||
|
fa2ad88cc4 | ||
|
63cbcd0956 | ||
|
d6d0ebf8f4 | ||
|
0d810d92ca | ||
|
1ff914a6f4 | ||
|
5959b1be72 | ||
|
d12a214c05 | ||
|
3a83052f2e | ||
|
510b44ca92 | ||
|
15edb6756d | ||
|
fbfd02b08b | ||
|
b39c26fc86 | ||
|
95b2c8d175 | ||
|
d52748b09f | ||
|
34d18a3a9a | ||
|
3b27d6a9b5 | ||
|
703c391f99 | ||
|
4f1dc29df1 | ||
|
13667df374 | ||
|
8800d6985f | ||
|
364b8f2605 | ||
|
67b9ea9deb | ||
|
b78f2336a7 | ||
|
c7ba637c7d | ||
|
23a5ce3df7 | ||
|
8f88e28e50 | ||
|
9cf6139557 | ||
|
d556065a8b | ||
|
951716f7dc | ||
|
1ddc7ddda3 | ||
|
903ed9f3dc | ||
|
c42b76dcb8 | ||
|
a73582d9ae | ||
|
42c4fc7557 | ||
|
ddbbb6f1dd | ||
|
ff21a92330 | ||
|
07f76f7ad1 | ||
|
c90ccffd7b | ||
|
a00d5f18af | ||
|
1e391d211b | ||
|
d87f9672fa | ||
|
2b5838aa01 | ||
|
e10486d6ec | ||
|
1a74d6604d | ||
|
6d118536b6 | ||
|
ca4d758db9 | ||
|
dc18c26aa4 | ||
|
48505c2968 | ||
|
a98ea1e66a | ||
|
3dec697816 | ||
|
88fd41e597 | ||
|
b05d071a1c | ||
|
a27d3b9689 | ||
|
1facc0cd01 | ||
|
837f91d830 | ||
|
9c5f5aefb0 | ||
|
6b8d4a444b | ||
|
6bef09a3b1 | ||
|
e35319e5a2 | ||
|
0e548b3812 | ||
|
bfac02ccab | ||
|
7ea1a2b361 | ||
|
99df418f1d | ||
|
6416d8ce9c | ||
|
22b43a2b01 | ||
|
05e5d24c5e | ||
|
eabcc30367 | ||
|
f5e0ef5223 | ||
|
f46d9330b0 | ||
|
b62a0b4607 | ||
|
1f044321fb | ||
|
a841d49483 | ||
|
9509acc490 | ||
|
02d356ef12 | ||
|
d3516f299e | ||
|
79630767c2 | ||
|
084a76d075 | ||
|
bc6822e397 | ||
|
43432a9e48 | ||
|
d64a5bc12f | ||
|
b2922d18e2 | ||
|
ccf03fc07b | ||
|
a7c45da10c | ||
|
e03f01e24a | ||
|
0939589557 | ||
|
8167af9b4a | ||
|
4cf76123e5 | ||
|
01ee4b23e6 | ||
|
b198f79214 | ||
|
09db868a28 | ||
|
33e8ef75ff | ||
|
11dcb16b14 | ||
|
86f21da28b | ||
|
89cd6a9aa4 | ||
|
18e1256037 | ||
|
02cf478d91 | ||
|
6ec70192fe | ||
|
8c75098a9a | ||
|
72500f6948 | ||
|
37ec9ab464 | ||
|
82fe2a4c8d | ||
|
aa50e6ee66 | ||
|
91a07cfaee | ||
|
709f5e9a65 | ||
|
b2f9ef21cc | ||
|
be6b72edcd | ||
|
ece2d1e78a | ||
|
1ee1a5f2a1 | ||
|
a567326853 | ||
|
6231861dd6 | ||
|
1ff7b77ee0 | ||
|
9365708bb2 | ||
|
d23a0a8589 | ||
|
701b39b043 | ||
|
58ad1f3876 | ||
|
2138e7ea33 | ||
|
32f8c9e59f | ||
|
57028eab39 | ||
|
3a16edd8a6 | ||
|
165f3bb270 | ||
|
0ba75153f3 | ||
|
db2789990f | ||
|
acaf299bcb | ||
|
1940301824 | ||
|
34576e880d | ||
|
65c0668d40 | ||
|
53bd2bcbfe | ||
|
388724fccb | ||
|
231eabb013 | ||
|
54903fc2ea | ||
|
3a1baf0700 | ||
|
0c0e36b6f8 | ||
|
234c03db09 | ||
|
59db5e7889 | ||
|
28aa7da349 | ||
|
c51e344b87 | ||
|
54461dfa75 | ||
|
2d48e93f74 | ||
|
af22646322 | ||
|
722b42a93e | ||
|
8f9e7f77a7 | ||
|
09bb1ba494 | ||
|
d4137428ff | ||
|
b4d6c4f5b7 | ||
|
ffbe59ece5 | ||
|
fab9c90ccb | ||
|
fb1a774bc4 | ||
|
98bc7d1e0e | ||
|
f7622f24b2 | ||
|
f0a195a6d4 | ||
|
180ba27d84 | ||
|
f944671f86 | ||
|
def2903f7d | ||
|
0273a4e839 | ||
|
f8d2f02c5d | ||
|
25147d8897 | ||
|
0fb6f05fba | ||
|
4e4e899356 | ||
|
5a01dbf269 | ||
|
30b923b283 | ||
|
73ba381d20 | ||
|
1a5912877e | ||
|
813e506b68 | ||
|
077ca987f7 | ||
|
c632a7a6a5 | ||
|
e33e767510 | ||
|
ac82617aa9 | ||
|
a35dfd1fd1 | ||
|
c28aae9913 | ||
|
c26a99e65c | ||
|
ca57dcfc2f | ||
|
df5662dd69 | ||
|
8927a4889e | ||
|
1ac7831f3c | ||
|
292d272a94 | ||
|
a6ee8dc66e | ||
|
496f89f184 | ||
|
7a56eff1ac | ||
|
07e182aa16 | ||
|
7de06aa1e0 | ||
|
3955b64405 | ||
|
2bb55d681d | ||
|
f94e6ac527 | ||
|
b344f17b86 | ||
|
677b8cb633 | ||
|
6f3342e09e | ||
|
a1ddd762e0 | ||
|
68474e4057 | ||
|
a84b9ee396 | ||
|
b9c2ee745a | ||
|
c91a47fcaa | ||
|
615e489d8d | ||
|
c68f9f6f16 | ||
|
229cb85a6a | ||
|
e5c22fa665 | ||
|
8bcfff05d7 | ||
|
6416ee8151 | ||
|
f8eceb48e6 | ||
|
310c483bfa | ||
|
a8f20361aa | ||
|
290be69d99 | ||
|
3b96bd7ea0 | ||
|
dc2f22f5fa | ||
|
821be29f41 | ||
|
52ff1a12ff | ||
|
814699ef11 | ||
|
0c30838b25 | ||
|
cf66c2a1ee | ||
|
2ee419ffca | ||
|
bfb9d696d7 | ||
|
bb2a34dd6b | ||
|
ed652c0c56 | ||
|
1dc961d6eb | ||
|
d119fcfc98 | ||
|
4d3573724a | ||
|
8b37a66075 | ||
|
ba4f32075a | ||
|
218be22576 | ||
|
7688293716 | ||
|
458f8533c4 | ||
|
34502752fc | ||
|
d6758fd823 | ||
|
65700e790e | ||
|
7c34e4bb96 | ||
|
d0d6e3563b | ||
|
a2619f8c78 | ||
|
42d07fd2f0 | ||
|
8bea10960f | ||
|
9cbb19c304 | ||
|
1b94dfd712 | ||
|
9f3604d739 | ||
|
4a1b2be269 | ||
|
962dc1b55b | ||
|
07c86502f6 | ||
|
adb188e5d0 | ||
|
ce031dc6b8 | ||
|
18b5f03247 | ||
|
8a555ecf1c | ||
|
1b325b9acd | ||
|
1bdaddb319 | ||
|
7896e177ef | ||
|
ce8e659008 | ||
|
27be5deeb2 | ||
|
515f270c3a | ||
|
ffff3bd334 | ||
|
f493f13b25 | ||
|
e605c14b13 | ||
|
338488f16d | ||
|
2abc67c3e8 | ||
|
eb1ba143ec | ||
|
6f5bca0f67 | ||
|
407cd8dd4b | ||
|
62a4f0fc04 | ||
|
77cde411f1 | ||
|
3eb9d23108 | ||
|
410d4aeb21 | ||
|
0a28d216fd | ||
|
b69faf6920 | ||
|
efb92ea37a | ||
|
e77f9981df | ||
|
d27c2cc1e9 | ||
|
586b19675e | ||
|
f2907536b4 | ||
|
4aa4e35d1c | ||
|
9a11ac06bf | ||
|
aa3b18f848 | ||
|
103bdc151f | ||
|
6d4c1cd879 | ||
|
cacbe30871 | ||
|
bfeeacb230 | ||
|
04bb7b4919 | ||
|
b7df277a5c | ||
|
c681041b48 | ||
|
923834c784 | ||
|
588edf98be | ||
|
28c603ad5f | ||
|
6988a47e02 | ||
|
2c8ceb1217 | ||
|
ccac4ffa24 | ||
|
4258cef9bd | ||
|
62cc6dfe76 | ||
|
9f224a971b | ||
|
cf5dba9157 | ||
|
23035b9aa0 | ||
|
84908ec8ec | ||
|
dade49743b | ||
|
f29bf35c2a | ||
|
dfa6701c43 | ||
|
763ca69a73 | ||
|
6bf3b152bf | ||
|
aa19f85996 | ||
|
156d89567e | ||
|
ecc71baf61 | ||
|
90c743d963 | ||
|
b926293fa7 | ||
|
71a19191f8 | ||
|
38a0f20a33 | ||
|
c35192108c | ||
|
245b564f13 | ||
|
0d8d1ea4f3 | ||
|
27a427a363 | ||
|
2ff028a694 | ||
|
c211338218 | ||
|
8ac89af8bd | ||
|
bbbaf59591 | ||
|
169419896f | ||
|
0543dca502 | ||
|
cc6011d57a | ||
|
fc4407ef7e | ||
|
03735a125f | ||
|
5baeda9ff1 | ||
|
9b9794b5e0 | ||
|
0697d60a48 | ||
|
cfe6c82a31 | ||
|
3e30228d95 | ||
|
7264b53e5f | ||
|
60836d8523 | ||
|
ef89c2e47a | ||
|
2d9e3e1847 | ||
|
30136a9697 | ||
|
db7ccd66d3 | ||
|
cfe6483102 | ||
|
561566e723 | ||
|
c2dcc4c898 | ||
|
d09bfdc4ff | ||
|
358ef4536f | ||
|
5061a35e66 | ||
|
cd9a1e8c9e | ||
|
646902e75e | ||
|
40d26cb868 | ||
|
b64aa51c0c | ||
|
8206441834 | ||
|
d713783736 | ||
|
57dffaa2ce | ||
|
9e81dd2360 | ||
|
e2798969d7 | ||
|
1c31ec66f2 | ||
|
241f9fc7b0 | ||
|
270192486a | ||
|
a799503c97 | ||
|
9685928087 | ||
|
0e4b2fad99 | ||
|
3c4571a4e0 | ||
|
046147eb1d | ||
|
7834520e54 | ||
|
8e5b4d4b6f | ||
|
4544a074d9 | ||
|
9b78501392 | ||
|
f59ddcc88d | ||
|
a4955a2b79 | ||
|
92ae1a565b | ||
|
15a56ca25e | ||
|
9dcaa829ea | ||
|
9f65799a3d | ||
|
886587848b | ||
|
a97fc6dba8 | ||
|
c124e88d12 | ||
|
17f3870296 | ||
|
4626d42d08 | ||
|
e1e760055c | ||
|
45bf6c3bf3 | ||
|
fef0cc764d | ||
|
72049afcf6 | ||
|
d26c06dbf3 | ||
|
268decd655 | ||
|
7ae246c839 | ||
|
c7c454e4fb | ||
|
8e27297a81 | ||
|
2cdec72985 | ||
|
0085ac534d | ||
|
7828a79a96 | ||
|
5576c21e67 | ||
|
e49cfb1d2b | ||
|
1e541d0225 | ||
|
0974afd26d | ||
|
8d93594771 | ||
|
1136ac70e8 | ||
|
dc8d5a39ea | ||
|
8329e649b0 | ||
|
66da8b164f | ||
|
ea48577864 | ||
|
597146b136 | ||
|
30dd0c1e11 | ||
|
88772c4266 | ||
|
dc1d9e1c84 | ||
|
69ea65835d | ||
|
d5bae3a8c6 | ||
|
f14010bd5b | ||
|
87094fc83f | ||
|
7c179cfeab | ||
|
7582c221d1 | ||
|
c109895848 | ||
|
eccedada40 | ||
|
25d54accf8 | ||
|
d07685f0e9 | ||
|
2445c00c7e | ||
|
4c1d3ef514 | ||
|
4614c7d4c2 | ||
|
bbf1ef0dc3 | ||
|
3433c9e708 | ||
|
2cd5d75a2e | ||
|
2535b8adef | ||
|
4edab7bb7f | ||
|
fd8658e317 | ||
|
51d21d8c86 | ||
|
b4c3307cdf | ||
|
4e8d10cb44 | ||
|
e96875a425 | ||
|
5ab0035348 | ||
|
4ddff96b1e | ||
|
a08d84c1df | ||
|
21c71bfac1 | ||
|
6baaed3581 | ||
|
152dbfd5d1 | ||
|
a56d14086b | ||
|
aee87693f8 | ||
|
976b4affd9 | ||
|
e222b6ad9c | ||
|
19b17374e8 | ||
|
43989122bb | ||
|
72712d6047 | ||
|
0b52d2cc15 | ||
|
8304102136 | ||
|
3381aefcfa | ||
|
279a365cb1 | ||
|
2c9e00da56 | ||
|
f7cae69704 | ||
|
b7d58bcdbc | ||
|
13a856b843 | ||
|
8da38985c3 | ||
|
60cf6c6b97 | ||
|
35c2b34564 | ||
|
ef2e048efc | ||
|
6b3261aa33 | ||
|
1849c02cb6 | ||
|
1ec74a89e2 | ||
|
c591792de9 | ||
|
3108543ae5 | ||
|
1eb221c743 | ||
|
bebf6bc2e7 | ||
|
9e91cc2138 | ||
|
c5b939cfb7 | ||
|
5bd411ca27 | ||
|
a533cda6f0 | ||
|
fe4b07b8ae | ||
|
f9f2ccd904 | ||
|
d9e87d7c32 | ||
|
a0092c0770 | ||
|
3100131125 | ||
|
988880cf83 | ||
|
c3fb9672c4 | ||
|
0a2d94e425 | ||
|
8d9073cd31 | ||
|
d075961ffa | ||
|
7a72409b61 | ||
|
34fc530fba | ||
|
f257ff2f97 | ||
|
7ad5822c5b | ||
|
9a8f9f0a94 | ||
|
6421cecafb | ||
|
be544d6d89 | ||
|
3c89ecafdd | ||
|
35ec4eec52 | ||
|
e47f737a2f | ||
|
ac671a065b | ||
|
74116cc550 | ||
|
406070a5c3 | ||
|
0ccafd5b53 | ||
|
940f517aa3 | ||
|
216e5f65ad | ||
|
a74685d66d | ||
|
b7791d2845 | ||
|
d151a82d78 | ||
|
8ce61fbd52 | ||
|
90c24aade3 | ||
|
6b3f787fee | ||
|
4ebe4ce1b7 | ||
|
8c79740ee8 | ||
|
59d027ca02 | ||
|
37a7345a90 | ||
|
c519d4651b | ||
|
9b3b609e40 | ||
|
6254f53716 | ||
|
f05dc46432 | ||
|
3de0982a4a | ||
|
c2184fb3bf | ||
|
919c09fcb0 | ||
|
1d9dbd40ec | ||
|
0cd953a6f3 | ||
|
4db2b72351 | ||
|
dd54fcbdbd | ||
|
3123cf7ac6 | ||
|
6b579dd4ce | ||
|
16dfaa3e27 | ||
|
d7842b9f84 | ||
|
115034fccb | ||
|
309e957a85 | ||
|
d7007e402e | ||
|
91323a21cf | ||
|
fea893d76c | ||
|
761bc6ba4c | ||
|
75172feb4e | ||
|
3285fb1608 | ||
|
03a4c6910d | ||
|
485b958599 | ||
|
da47ba2f67 | ||
|
c39195488a | ||
|
227fb0ae9b | ||
|
b12ff5b503 | ||
|
0946c72b88 | ||
|
7d49b046d4 | ||
|
5f0426c840 | ||
|
73e239cc5f | ||
|
ad670f721a | ||
|
028a4a70cf | ||
|
77d7960347 | ||
|
39821146bd | ||
|
7d505a41ac | ||
|
e457b2f0d6 | ||
|
c9cf7fd4d4 | ||
|
b0371dd33d | ||
|
25e16c3565 | ||
|
7b39527863 | ||
|
d861b08866 | ||
|
fb438dc108 | ||
|
4e6b4f179b | ||
|
00d038c8f3 | ||
|
a9f6a68952 | ||
|
b9142bbc5a | ||
|
6c812f663e | ||
|
a93ec9783a | ||
|
2d184d77b6 | ||
|
bce299ccc7 | ||
|
235cebd14a | ||
|
a638aa9d53 | ||
|
67cce0ef7e | ||
|
82f4267bf6 | ||
|
45a9ca29c4 | ||
|
7f4e813277 | ||
|
3805ff4a0c | ||
|
464cfd475e | ||
|
fe469ae57f | ||
|
550ef9a1c4 | ||
|
935adfb51a | ||
|
3974df4a62 | ||
|
4870974161 | ||
|
8c4b0037f5 | ||
|
2c6f763ef2 | ||
|
ca28de02d8 | ||
|
bfc15ea029 | ||
|
6e8b8a5920 | ||
|
099f3b6a62 | ||
|
142d182bc1 | ||
|
1437871d88 | ||
|
352bf69409 | ||
|
9bdf3d23e1 | ||
|
be8ecfa707 | ||
|
51da0d0259 | ||
|
f55b78a994 | ||
|
e1a44c93f8 | ||
|
07e7087a09 | ||
|
2c79c7e2f6 | ||
|
09f6637fe0 | ||
|
3784db3308 | ||
|
2b950ff5dd | ||
|
09339c9cfb | ||
|
ccadd88af5 | ||
|
cc02a0efc2 | ||
|
43a1385b79 | ||
|
5101464e3b | ||
|
3d71478d38 | ||
|
4989ed445e | ||
|
d9413039ec | ||
|
eba0c9be34 | ||
|
48c9e9f3cc | ||
|
81ebde88db | ||
|
79ced9d0f8 | ||
|
a4058b84ce | ||
|
7bf211a52b | ||
|
d5f722792f | ||
|
0f02906c9b | ||
|
9582e228b1 | ||
|
45f20431f9 | ||
|
7554e6d7f9 | ||
|
cb8f26f177 | ||
|
b5dfce7861 | ||
|
2ca5a65544 | ||
|
17deb136db | ||
|
8c9710c76c | ||
|
32f7ecb261 | ||
|
fb77fde710 | ||
|
3c67bb90d7 | ||
|
dabb168853 | ||
|
45e5b3b219 | ||
|
a6b7469923 | ||
|
cb5dab3033 | ||
|
21d0038ff2 | ||
|
c094d8f2e8 | ||
|
c465d6a6c2 | ||
|
73d35bc985 | ||
|
2a8ccb065b | ||
|
8f04a50ce1 | ||
|
888aa5586b | ||
|
99f56f5d22 | ||
|
ad6281090d | ||
|
f0d334d3e2 | ||
|
5f829b048f | ||
|
1a961e66ff | ||
|
fdb0e22656 | ||
|
132ee1915f | ||
|
44bf4f3c8f | ||
|
6237767d5a | ||
|
dec9d96417 | ||
|
b167c87267 | ||
|
2280fe8e8e | ||
|
575d6dcd2d | ||
|
f729490c6b | ||
|
b32124cdd6 | ||
|
3d4321ee38 | ||
|
85034b382e | ||
|
77a51d1ad4 | ||
|
33e0cdc2d7 | ||
|
6519faa2fe | ||
|
5e3a234cbe | ||
|
e54c31d2d5 | ||
|
66c0537251 | ||
|
ac58516593 | ||
|
c3da6322b5 | ||
|
3d241500cf | ||
|
ded8224f66 | ||
|
f8814881a1 | ||
|
cc2852cd48 | ||
|
467637a9eb | ||
|
3cfc292d84 | ||
|
6acf94a810 | ||
|
31367fb4c4 | ||
|
12d6074e3b | ||
|
ff30386051 | ||
|
601f99ac16 | ||
|
87fe5c6101 | ||
|
68399ca31c | ||
|
2a6d7fd80f | ||
|
4725f510d8 | ||
|
be0ba22222 | ||
|
c8781392be | ||
|
b97164fcfb | ||
|
0dfb92281b | ||
|
4fe80c40da | ||
|
f0fac5115a | ||
|
46dd389d0d | ||
|
1e28e21ab5 | ||
|
7832c62c5d | ||
|
d025ee9dbe | ||
|
a9a9cb4319 | ||
|
aa727cb9b1 | ||
|
b8c9a99f20 | ||
|
aff995b0d0 | ||
|
2cc7e5dfdc | ||
|
5235a150b1 | ||
|
c6372ea9de | ||
|
7df4cc44c4 | ||
|
d47cf40544 | ||
|
7f5d88e95c | ||
|
d09663c066 | ||
|
ef97c9b69f | ||
|
d855e6c8b1 | ||
|
cd66f7eb43 | ||
|
6a35a7ba4c | ||
|
a3e146dc68 | ||
|
b81305a4a9 | ||
|
73884b34bc | ||
|
6166a34db2 | ||
|
6fa7da4b1c | ||
|
c3e426c491 | ||
|
21e023f0db | ||
|
063be001b3 | ||
|
5dff02e8bc | ||
|
60a59407d8 | ||
|
20a5aecfca | ||
|
c2e7b5a67d | ||
|
8f32303d07 | ||
|
891b1e7782 | ||
|
f26394fd3b | ||
|
4d83d42b4c | ||
|
57f1108df2 | ||
|
2641a9abe5 | ||
|
6b193ab350 | ||
|
b1bb37511c | ||
|
319187d6d6 | ||
|
02eb789f84 | ||
|
5a9338a27f | ||
|
eb6924277f | ||
|
325419404d | ||
|
bd8f371fd5 | ||
|
1783ff2845 | ||
|
d388527ffa | ||
|
19494088bd | ||
|
920dad524a | ||
|
ec89bcac8e | ||
|
a916c1f4ad | ||
|
a9a0ac92d7 | ||
|
da8a8bd1ef | ||
|
d9c746891d | ||
|
67817005b5 | ||
|
24d11de5a7 | ||
|
9251c87323 | ||
|
e12fab90d1 | ||
|
0a194b5b01 | ||
|
8d028adc53 | ||
|
dfca15395e | ||
|
e21f2362fe | ||
|
1ce328e8a9 | ||
|
038a5f999f | ||
|
5d3704c7ea | ||
|
87037c06c9 | ||
|
dd412c0f50 | ||
|
bf44befff6 | ||
|
e61874bb6f | ||
|
1e5331768f | ||
|
ec9a3a4f7c | ||
|
e439a3a8dc | ||
|
19f70d7a11 | ||
|
afe7ed5b05 | ||
|
d4bf004d74 | ||
|
e4d06a088b | ||
|
0929088b12 | ||
|
7b4838fc9b | ||
|
0cf9533248 | ||
|
84ff0b8a9f | ||
|
d467dcfeaf | ||
|
8e68ba4751 | ||
|
0f2a85ba9f | ||
|
7674a0a91e | ||
|
5bc1a66572 | ||
|
9b56067213 | ||
|
9a9df2fc3c | ||
|
9989d8d1d4 | ||
|
f9471f297e | ||
|
146b693e4a | ||
|
7295b7e329 | ||
|
e2441ea3e7 | ||
|
119e51912e | ||
|
dd950f5b0d | ||
|
78a9bad1e1 | ||
|
0c6eaf5484 | ||
|
1010068ddb | ||
|
82eec3d8d7 | ||
|
ee7b37d3f3 | ||
|
143d82d242 | ||
|
8b91b38855 | ||
|
1098f0d2a3 | ||
|
ab53cec022 | ||
|
6f5f8e5648 | ||
|
edfd707c22 | ||
|
1870f30af8 | ||
|
90106f5f08 | ||
|
9924b7b438 | ||
|
aa37faab0a | ||
|
dc10f8ce72 | ||
|
996686c1da | ||
|
488785d013 | ||
|
3abdc01230 | ||
|
8da04a584f | ||
|
27cc61d45e | ||
|
7371c30064 | ||
|
140d163895 | ||
|
dc33bdc1dc | ||
|
74df4fab83 | ||
|
1e5cd3d7a1 | ||
|
a54e9b64aa | ||
|
74660704e3 | ||
|
7439893a2a | ||
|
e27e49e9dc | ||
|
34ed729c59 | ||
|
adaeeca3fd | ||
|
dac75563d3 | ||
|
cbc76adcaa | ||
|
69a9cb383d | ||
|
4343073c00 | ||
|
fe60d4be88 | ||
|
ae337807f5 | ||
|
9ae30ac08e | ||
|
62fa85c0a4 | ||
|
7bb873dad9 | ||
|
5f6c1c14cb | ||
|
d43189ad33 | ||
|
fcad76fc51 | ||
|
97e6e1684e | ||
|
67a0d3e926 | ||
|
183fb9f9ff | ||
|
9815ddef1f | ||
|
f6d0847453 | ||
|
b0b9f0d65f | ||
|
0cec80f676 | ||
|
48c64143e3 | ||
|
a8712422bc | ||
|
97f65bd283 | ||
|
fd3c1c50f1 | ||
|
b153e4bb9f | ||
|
db9856a8db | ||
|
75ecea265d | ||
|
be8751cb73 | ||
|
fb25ecb4a1 | ||
|
f1cb7d27ac | ||
|
dee494e12f | ||
|
b13a121915 | ||
|
7486ee9537 | ||
|
4a20ccc28e | ||
|
f80dd2b307 | ||
|
b208cf6d32 | ||
|
39e78ff17e | ||
|
a8177ea7fe | ||
|
bedcfc154b | ||
|
4c38b4aa3c | ||
|
f6cfe266e0 | ||
|
4905e65f14 | ||
|
ccb250b410 | ||
|
aca57ffc62 | ||
|
7f375f42d8 | ||
|
eedcc2034d | ||
|
24c9a167d7 | ||
|
909df8ef1f | ||
|
3b27cb3671 | ||
|
3fe0db4a7d | ||
|
8b55814ab2 | ||
|
575e471553 | ||
|
0f5f1aebed | ||
|
50e17eb1ab | ||
|
158cc2f660 | ||
|
1066a31acd | ||
|
1f9d0f4582 | ||
|
a6d65233f1 | ||
|
eff2fe7a1b | ||
|
20efdc70b3 | ||
|
f0d8fb8f1a | ||
|
f7a380e9b7 | ||
|
e9c7cf6f63 | ||
|
68f1661452 | ||
|
36fd1b91ae | ||
|
a4ec430ac0 | ||
|
519614b2fd | ||
|
bf0118c8ef | ||
|
a4db0820bc | ||
|
ee7528413e | ||
|
7952fc8324 | ||
|
652773d2cf | ||
|
2a17787242 | ||
|
0a53ad5721 | ||
|
6da6bdc863 | ||
|
42ad2bb83f | ||
|
f309a65cb4 | ||
|
77e19ab1a4 | ||
|
1a996b6ef3 | ||
|
b882f1a010 | ||
|
82a030e6ff | ||
|
0758b85179 | ||
|
ab3d9bd080 | ||
|
8ff813f689 | ||
|
88ae72c0d3 | ||
|
312aa4be26 | ||
|
cbb06fce9d | ||
|
f259e497c4 | ||
|
dd4172ac66 | ||
|
66029e60d3 | ||
|
364f484f04 | ||
|
9dd5159414 | ||
|
13e38d6fd8 | ||
|
10dcb64715 | ||
|
7551b51e7d | ||
|
adb418aafc | ||
|
270da80d64 | ||
|
b2027cfd66 | ||
|
7f1f4eeac6 | ||
|
7a7446c8bd | ||
|
ddbae294e6 | ||
|
8c71b744f3 | ||
|
479b5d31a9 | ||
|
4cbf4230e8 | ||
|
6a610187e0 | ||
|
eb2a4aebba | ||
|
21a2e67755 | ||
|
3b9e312615 | ||
|
26dab04c9e | ||
|
00713c0d11 | ||
|
751b5f3027 | ||
|
e8261b000e | ||
|
41ecb70297 | ||
|
09ee104b8c | ||
|
e3a4964787 | ||
|
9bf72910a4 | ||
|
ee39e20e6d | ||
|
399d6db6f6 | ||
|
0821ce44b5 | ||
|
ea279111c6 | ||
|
674ce02e58 | ||
|
8dfa2767ec | ||
|
20dad7f07f | ||
|
751cc4c44d | ||
|
2318e6d8e9 | ||
|
61b4a492c3 | ||
|
9db3d01e09 | ||
|
8da73ad3dd | ||
|
b8c16d8ac5 | ||
|
429c0951f3 | ||
|
74e103c791 | ||
|
f941950ee2 | ||
|
846df2eef1 | ||
|
34ed058c97 | ||
|
eae0290978 | ||
|
561368570e | ||
|
3467d1fed0 | ||
|
d02ff232e5 | ||
|
2d1c6a5402 | ||
|
eab3b65629 | ||
|
20b435732a | ||
|
929617273d | ||
|
2717bf7d49 | ||
|
5cd2ebc960 | ||
|
9b4afe9816 | ||
|
23bb5598d5 | ||
|
af1d7813e9 | ||
|
16c2e5a585 | ||
|
c02750edbd | ||
|
7204ddafec | ||
|
faeba9a7e4 | ||
|
190d238a1f | ||
|
715451b5fb | ||
|
87f1895405 | ||
|
923d817751 | ||
|
0728209b66 | ||
|
b8b9dcc2ee | ||
|
f35e879852 | ||
|
34f4f12eb9 | ||
|
fa63bf758d | ||
|
f6b396ae64 | ||
|
2c7fd58e34 | ||
|
982f2c9634 | ||
|
f2fd42b47a | ||
|
1b4ccad938 | ||
|
a9de1ce8e0 | ||
|
ac752d5ec2 | ||
|
632d8d02d2 | ||
|
48aeb26e02 | ||
|
1694af8b5e | ||
|
83bcab9cd2 | ||
|
bdc7f4b3f5 | ||
|
39202a3d79 | ||
|
912065a121 | ||
|
c8466afac2 | ||
|
2619e162c1 | ||
|
e1112e17f8 | ||
|
92b2ead74c | ||
|
bbed9b94c1 | ||
|
73d07311db | ||
|
1cdff47477 | ||
|
511a5c3f82 | ||
|
853885e2ff | ||
|
d83936a66a | ||
|
5517d2bf56 | ||
|
f21ab49ac5 | ||
|
925a458abe | ||
|
76946c447f | ||
|
2faa29b1c4 | ||
|
6826cc311d | ||
|
5e17ce0a0b | ||
|
e8d299d3b6 | ||
|
7637aa2ab6 | ||
|
ab067d1d3a | ||
|
e6f84666c7 | ||
|
4c5429af15 | ||
|
0a0ac3b7c9 | ||
|
24833ce9fb | ||
|
ec2c18dc87 | ||
|
7384609e74 | ||
|
3047649650 | ||
|
d298dac3f3 | ||
|
1574bca8a8 | ||
|
ec2f6c6b80 | ||
|
838cc60161 | ||
|
310c61a5cc | ||
|
318cc15323 | ||
|
3a64ceb4d6 | ||
|
d0f21c0095 | ||
|
46dc15dd29 | ||
|
8dc654b513 | ||
|
7000ac3f3f | ||
|
43c2e8d8e9 | ||
|
0231139b01 | ||
|
d6ee6446dd | ||
|
7b666efcf8 | ||
|
eb5d2198fc | ||
|
34e44ebd1c | ||
|
bf2f4bc040 | ||
|
9dc4559aba | ||
|
eba8856261 | ||
|
a8c5aa471a | ||
|
52f6dcf092 | ||
|
dec79f3742 | ||
|
8bdcac0f3e | ||
|
8426b674a3 | ||
|
da391bcc8d | ||
|
2d7443d454 | ||
|
991987ed76 | ||
|
ec24ebf2cf | ||
|
6ba0976085 | ||
|
2b88d01a01 | ||
|
0c09f24cbf | ||
|
61d22afeba | ||
|
9f1ed6e8c3 | ||
|
bbc4113cac | ||
|
91194bf422 | ||
|
9c5f940b00 | ||
|
455b4043b8 | ||
|
bd83ee7931 | ||
|
f6bdf7c09a | ||
|
2db8afb8c2 | ||
|
e033129dd3 | ||
|
f9dc590100 | ||
|
8996aafe0d | ||
|
9dc25ef7af | ||
|
2b84f4d407 | ||
|
097c8b674c | ||
|
ba649d4b94 | ||
|
6ed1614db0 | ||
|
df5b6a8380 | ||
|
1f82a8b99e | ||
|
0c95d96f32 | ||
|
c2f5f84118 | ||
|
b3b5e3d8f0 | ||
|
506d3f3cd9 | ||
|
516a8c5ee5 | ||
|
2f81e9d374 | ||
|
2d8703bb8d | ||
|
76e60d9bc3 | ||
|
9d5370be5f | ||
|
fc1a06bc45 | ||
|
fce80374f4 | ||
|
420c9f10c2 | ||
|
5a39681a2e | ||
|
7a1b7db7c8 | ||
|
03a643da52 | ||
|
383f0ce450 | ||
|
c6c668676c | ||
|
7b01dde063 | ||
|
8c25f65024 | ||
|
1e478e3545 | ||
|
6c75a8978b | ||
|
640267fc2b | ||
|
33c7c3ee12 | ||
|
f9b41d34ae | ||
|
d4bec79451 | ||
|
ac1a8b4daf | ||
|
28838c1759 | ||
|
50ecb0dac9 | ||
|
e22bc01cbd | ||
|
6c28713a4c | ||
|
fc9023386c | ||
|
e6cae9bcc3 | ||
|
a9eeca1302 | ||
|
8c695e42ca | ||
|
0aa7fd47d5 | ||
|
70596042d6 | ||
|
caf616234b | ||
|
375187aa70 | ||
|
71eccdc0e3 | ||
|
639b1e48f5 | ||
|
0bb4cb4472 | ||
|
cc51543851 | ||
|
22540390e1 | ||
|
98565eb67c | ||
|
cf6a47ecb7 | ||
|
fa60b9f9d3 | ||
|
644120ca31 | ||
|
a50a625b3b | ||
|
7297c13331 | ||
|
f73399bfac | ||
|
a056cd78f7 | ||
|
a30f3c86c2 | ||
|
e70bdd86a7 | ||
|
bc9f33c2e0 | ||
|
872b89ee93 | ||
|
ae53062518 | ||
|
17f76c9cb3 | ||
|
5de944146a | ||
|
9dc6092cb0 | ||
|
a32a2ef04e | ||
|
ecfa0ae3da | ||
|
03595052ce | ||
|
1f94c53dd2 | ||
|
9c426373f2 | ||
|
c03e30a01f | ||
|
07f7a77ac0 | ||
|
3b9ea2c9a4 | ||
|
1beb13dd80 | ||
|
ddae84abb3 | ||
|
863b9a2c98 | ||
|
9d44bbdb48 | ||
|
8d93dd5adc | ||
|
48502961cf | ||
|
3c8bec61d3 | ||
|
7296c7df1a | ||
|
f3ee6603de | ||
|
ee0aabda1d | ||
|
08d37a4b0f | ||
|
f975ea99cb | ||
|
f030d41dc7 | ||
|
8d079bfcd1 | ||
|
d39b8654a6 | ||
|
ce7816a968 | ||
|
2ee572e68f | ||
|
4bbd850898 | ||
|
34eae6e608 | ||
|
6a0302fec6 | ||
|
c94cc293c2 | ||
|
cae7792a1e | ||
|
7f6b2fe4f1 | ||
|
eba430bbc0 | ||
|
01280c8d04 | ||
|
ec141ac9c9 | ||
|
590c892a6a | ||
|
ff8a50c366 | ||
|
46ef6c8ab7 | ||
|
b09eabc478 | ||
|
e49fcea6e3 | ||
|
68ed9f4ffc | ||
|
af94687d45 | ||
|
26964ecf0f | ||
|
77d19af359 | ||
|
120b82f243 | ||
|
a0fea30a11 | ||
|
8bb3e0a64d | ||
|
bbded12923 | ||
|
e8ba5d7606 | ||
|
af66b31a44 | ||
|
b000a40f28 | ||
|
3c85322523 | ||
|
a469b8bc04 | ||
|
78b8261a3a | ||
|
f20ca70c01 | ||
|
4e4148fc1c | ||
|
c22482f907 | ||
|
870e139fce | ||
|
4d58648c02 | ||
|
ebbb182537 | ||
|
1cd5377b45 | ||
|
1d1f0527ee | ||
|
37a5f77415 | ||
|
ced3c7efe4 | ||
|
c3b8f366ed | ||
|
de78876b1a | ||
|
64c25b049c | ||
|
8a4fe4f3ad | ||
|
8811b8c1fd | ||
|
190b01fdf9 | ||
|
f145d08c10 | ||
|
53382b7e15 | ||
|
6ad0242617 | ||
|
a7c2408c0a | ||
|
ce1eabaed6 | ||
|
f602541ede | ||
|
3f718e6efc | ||
|
ce7a985df6 | ||
|
6d83f7e7bd | ||
|
b73c00943c | ||
|
abaac8ef48 | ||
|
a2f8e7068e | ||
|
4d47873219 | ||
|
cf985486e5 | ||
|
b930c3fc93 | ||
|
dd26a96828 | ||
|
6865ddfc12 | ||
|
e888e69d4d | ||
|
2089059792 | ||
|
27739e0364 | ||
|
698ee271d6 | ||
|
543c75b293 | ||
|
b09c46f6f7 | ||
|
f2cc19e6aa | ||
|
814a0a123f | ||
|
179383540f | ||
|
8be1c8310d | ||
|
ef02d776ca | ||
|
750ff448ad | ||
|
e3abab6d4d | ||
|
d3ffae72fb | ||
|
87f751188e | ||
|
3469abaefd | ||
|
797364ee5c | ||
|
36c05fc4b9 | ||
|
79624febc0 | ||
|
0a9d4de126 | ||
|
b6f6994db4 | ||
|
ff7bed720a | ||
|
00c0f48b02 | ||
|
94c45cf2a0 | ||
|
58f77b2a1c | ||
|
7170e69b22 | ||
|
239ee2437c | ||
|
ced368db31 | ||
|
6a991e5c15 | ||
|
523d22262b | ||
|
08197a327e | ||
|
b7cb2a7aa5 | ||
|
decc5c74ef | ||
|
fbe0f886b6 | ||
|
7e23d6e2ef | ||
|
49458d1085 | ||
|
289c12bd9a | ||
|
9a6326b027 | ||
|
51f573f1ea | ||
|
c8b7cd8862 | ||
|
21c112d059 | ||
|
9432e1b5b2 | ||
|
8269d2f83c | ||
|
5a4b6be974 | ||
|
35e8ce60a9 | ||
|
8b6dd9f603 | ||
|
084f0ebdab | ||
|
58640c1521 | ||
|
7ffdfd12f8 | ||
|
cb9a30f285 | ||
|
e48bef809f | ||
|
f5d7570102 | ||
|
a600c60cf8 | ||
|
695eabd026 | ||
|
e81b51a647 | ||
|
b7e95ff090 | ||
|
3ca41be686 | ||
|
3152046173 | ||
|
3a98dc8a95 | ||
|
d737b28916 | ||
|
97c0dac876 | ||
|
006494b1fa | ||
|
149d343201 | ||
|
496cc79ba8 | ||
|
3591768745 | ||
|
d615f6761a | ||
|
5b29894048 | ||
|
d17f0cfa40 | ||
|
b2f70c7120 | ||
|
238707bd93 | ||
|
87cdf1e3a0 | ||
|
57fd47022e | ||
|
5eafd3bf6b | ||
|
640b5b0ea9 | ||
|
e4fb2f4680 | ||
|
a4909f54e4 | ||
|
f7065c6f0c | ||
|
64f7f837e7 | ||
|
130d36acd9 | ||
|
e94c28cfa2 | ||
|
a697dec30c | ||
|
bac09e9b9f | ||
|
b179f8f86a | ||
|
962d04ae17 | ||
|
f28e3bfe37 | ||
|
6474c86d32 | ||
|
769ea8cdfe | ||
|
052e77dd5a | ||
|
1a4b0cb12a | ||
|
f9aa95c987 | ||
|
a5d06fb4a4 | ||
|
d5e5d90bdc | ||
|
e62678e4e6 | ||
|
558ac24f7e | ||
|
6f22f6a59f | ||
|
5ec74f8abe | ||
|
16d7547e03 | ||
|
767112dcda | ||
|
81e23d1d8c | ||
|
ace90a4354 | ||
|
267e7096cc | ||
|
e7cded7511 | ||
|
33fbd715c0 | ||
|
25ba5b867c | ||
|
c8f431447c | ||
|
1b83a1d09a | ||
|
d2fb7a7151 | ||
|
7ad3447598 | ||
|
5e5bc8e705 | ||
|
886d1e8a19 | ||
|
6de7a035fa | ||
|
48d2497eb2 | ||
|
6494754ab9 | ||
|
a8153627c6 | ||
|
ed38966edb | ||
|
85551d1e54 | ||
|
ca31363180 | ||
|
a8f11eb3c3 | ||
|
151805121c | ||
|
0422d2a021 | ||
|
8586762dde | ||
|
ffdf70257b | ||
|
6079d60aa1 | ||
|
d46d012f8c | ||
|
baf926b360 | ||
|
82339869dc | ||
|
f70343bb63 | ||
|
4ab56ae3d1 | ||
|
39b4031684 | ||
|
71f8965393 | ||
|
66857e72a4 | ||
|
15eb5d47eb | ||
|
86242139da | ||
|
460bdc4148 | ||
|
1052126522 | ||
|
fdb42ac876 | ||
|
6486f986e8 | ||
|
bc4075e2ed | ||
|
11525c7d0d | ||
|
bb12ae9ce6 | ||
|
66b4ad0c44 | ||
|
4bff2d718e | ||
|
af7a7b4dc6 | ||
|
ec20d9a2a8 | ||
|
b0aee3d335 | ||
|
5e2ddbfd86 | ||
|
f086ebbb8e | ||
|
e5bf6a5bfc | ||
|
76fa86d54b | ||
|
60d89506a5 | ||
|
2893f1eb9e | ||
|
a08cbf412d | ||
|
0bd65356f9 | ||
|
2e2b39455f | ||
|
1f9fbe34e4 | ||
|
506d7ae50e | ||
|
342cb00625 | ||
|
952fc01efd | ||
|
3eebe301fe | ||
|
19c0a81c42 | ||
|
9fc7f9904b | ||
|
b04a516063 | ||
|
241e946d91 | ||
|
e45375dc26 | ||
|
ec8e243323 | ||
|
af0e9368d4 | ||
|
285483b81a | ||
|
73a91d5569 | ||
|
e24c78be32 | ||
|
b11184de68 | ||
|
61d02fc5d7 | ||
|
f0217f6821 | ||
|
09644914a6 | ||
|
b5c24d6a48 | ||
|
bdd2ac2c25 | ||
|
76376f0d33 | ||
|
9749da46ae | ||
|
1c05295e89 | ||
|
4d0f28215a | ||
|
0bba72bc5a | ||
|
87089b8e83 | ||
|
6a58148a89 | ||
|
15091052be | ||
|
5cd7e9a9b8 | ||
|
7cb530c334 | ||
|
5e0324cc91 | ||
|
6293e227ea | ||
|
d6d83a5c76 | ||
|
93fc883b90 | ||
|
dd21803598 | ||
|
af2f2282c2 | ||
|
869a76c9bb | ||
|
44b2964a6a | ||
|
8fddb57e0a | ||
|
c7a5a0cab0 | ||
|
c13aab3ffc | ||
|
95a9a76598 | ||
|
923cfa3d50 | ||
|
2c489168c2 | ||
|
f3292b4d34 | ||
|
9769829b72 | ||
|
36243d15cc | ||
|
f4645f570c | ||
|
61603ccfce | ||
|
a26cfc639c | ||
|
90602931d8 | ||
|
d1b330028c | ||
|
7a6b1930bf | ||
|
c271361552 | ||
|
17789bc814 | ||
|
d7c16e161a | ||
|
e71c17d7e7 | ||
|
17d87eb157 | ||
|
bf11bcc084 | ||
|
bd291109df | ||
|
ac89ba9b8d | ||
|
5ab634e375 | ||
|
bb1978d976 | ||
|
bea94ce8ac | ||
|
9561f93594 | ||
|
d44d5c3304 | ||
|
ec541e2057 | ||
|
8169bf6b97 | ||
|
56c8ad1221 | ||
|
6814f2e38c | ||
|
5f043b9a78 | ||
|
c9092cd1c7 | ||
|
86cc65d894 | ||
|
8ef2647fa9 | ||
|
3ff9e99416 | ||
|
1731046011 | ||
|
c3d96184b6 | ||
|
c255c606a7 | ||
|
cc69faa1fd | ||
|
0f53cd86c8 | ||
|
212c8f188d | ||
|
dec248adec | ||
|
98e6a066f4 | ||
|
c90db54a3d | ||
|
927d156933 | ||
|
c71f91073f | ||
|
ab28387692 | ||
|
c4905d02b9 | ||
|
ee39880fb5 | ||
|
926b3e56b9 | ||
|
e060df5367 | ||
|
a3294d4a0d | ||
|
19ce0ab246 | ||
|
75a1cc0d33 | ||
|
05228529b0 | ||
|
3cbeadfbc3 | ||
|
3d4938c0e2 | ||
|
1b850b8a2b | ||
|
ebd33f1869 | ||
|
ac846b4df3 | ||
|
0d427c9b90 | ||
|
dc6194f862 | ||
|
de50214e1f | ||
|
5ec4a88c35 | ||
|
31c141e757 | ||
|
9d5760d899 | ||
|
7945e1ea3c | ||
|
d94c40e371 | ||
|
31f22122e8 | ||
|
506582aa2b | ||
|
a02b251c9b | ||
|
34cbb6fa79 | ||
|
33679b56fd | ||
|
8897c4d560 | ||
|
f0c18ccbe7 | ||
|
e560d83c51 | ||
|
eca7addc67 | ||
|
263f3ba5c9 | ||
|
f514123ef0 | ||
|
9f461db0d0 | ||
|
7de8670616 | ||
|
8c7908b200 | ||
|
e7722e039f | ||
|
039bc0208a | ||
|
d57900a069 | ||
|
c1153302aa | ||
|
1086a3297f | ||
|
5c613934ca | ||
|
6d7e9092f8 | ||
|
8bb6f328dc | ||
|
7f03b13579 | ||
|
09008ea991 | ||
|
ee234212e6 | ||
|
277b243f52 | ||
|
0973ac753f | ||
|
f41cfbfb97 | ||
|
a70980c81a | ||
|
12a962f656 | ||
|
5936444f3e | ||
|
c7d42f00c6 | ||
|
b1318a9958 | ||
|
d2560d260c | ||
|
f40a61cf9a | ||
|
e0623578bf | ||
|
ead8daaa14 | ||
|
23b4b9e230 | ||
|
947017e334 | ||
|
7fd0d6507f | ||
|
ad1e9ef086 | ||
|
2f1d08e417 | ||
|
55f4eb80ba | ||
|
7c160ff65e | ||
|
fc5d5faaed | ||
|
5394f1763c | ||
|
050b67c9d6 | ||
|
4b87cb45ee | ||
|
f0e1db319c | ||
|
3950715237 | ||
|
ae9ba14b59 | ||
|
ad6c6fbe35 | ||
|
ffa5c20c88 | ||
|
5a6218eeca | ||
|
200531dd96 | ||
|
15a2f048ac | ||
|
d317a4042c | ||
|
c2d717aba5 | ||
|
ef74777df1 | ||
|
f6dbf99fb5 | ||
|
0b2d9e15b4 | ||
|
d14e5e75e8 | ||
|
df11ef34d2 | ||
|
4fc619262c | ||
|
2750eee86d | ||
|
71cb03345b | ||
|
22c8ca6c40 | ||
|
237a8965eb | ||
|
7d872c7863 | ||
|
0d08858dfb | ||
|
5c6d63c93d | ||
|
c8c10d2bb0 | ||
|
f4d3e9ea6a | ||
|
3713d3488d | ||
|
9b46d03c91 | ||
|
fe4e738580 | ||
|
205b0c4263 | ||
|
78eef25f1a | ||
|
c9305fd070 | ||
|
20774280b9 | ||
|
74bd2557af | ||
|
605fc8ecd8 | ||
|
6b745c53dc | ||
|
98c60b38fc | ||
|
be4515ec9a | ||
|
dcb1b64696 | ||
|
1190a60c7a | ||
|
82ecf207c6 | ||
|
84c7f1bf58 | ||
|
5ac453fe07 | ||
|
c2f39d653d | ||
|
748178f536 | ||
|
ffbef69dc7 | ||
|
3757124323 | ||
|
e65f214b3c | ||
|
f14004e56b | ||
|
dadc004dd6 | ||
|
3697d9e1f0 | ||
|
94524f7330 | ||
|
17d8a3e5d6 | ||
|
39996d7612 | ||
|
601ee4320f | ||
|
6e4f6fa92d | ||
|
b7103c29dd | ||
|
e53209e5ae | ||
|
501fbd3114 | ||
|
5626f43e2b | ||
|
0185cb48fb | ||
|
9a49eb06da | ||
|
3fbc012231 | ||
|
3938cf33d8 | ||
|
38b3d0c87c | ||
|
15a2fa6199 | ||
|
e834209d40 | ||
|
9607d21828 | ||
|
6fbbf36143 | ||
|
db25f3282e | ||
|
ceb92838e7 | ||
|
afb98f2cfc | ||
|
ed785fb087 | ||
|
3921180d77 | ||
|
ef36e0311e | ||
|
bf4cbe1204 | ||
|
28eee4da25 | ||
|
933ccf6deb | ||
|
9ac41322e5 | ||
|
c3233e03ef | ||
|
111871bb28 | ||
|
bf5b5f43e3 | ||
|
2ed8ebff09 | ||
|
34eb856d09 | ||
|
2d644bdfb0 | ||
|
741e5cf103 | ||
|
d745c04fe6 | ||
|
3a3c63956a | ||
|
153bdf576a | ||
|
6525ee6510 | ||
|
1780ddd329 | ||
|
85ad972ca8 | ||
|
a90b60799a | ||
|
47e8f74da9 | ||
|
fac28072ab | ||
|
ab77541f36 | ||
|
9eb769f340 | ||
|
1e81d6f848 | ||
|
dc393f4b77 | ||
|
dd1de530c3 | ||
|
74cd887249 | ||
|
b40bb35652 | ||
|
dfe257af86 | ||
|
ebb2a09107 | ||
|
c157909b4e | ||
|
a744239b22 | ||
|
9b4417be87 | ||
|
84f807e278 | ||
|
448635a945 | ||
|
638e3e6b3d | ||
|
bfe711bd42 | ||
|
d5eed91e7f | ||
|
15abf49211 | ||
|
b7eec0586c | ||
|
e639124a69 | ||
|
22bca7a16e | ||
|
6f66b354e5 | ||
|
f98e3320ae | ||
|
73c958222b | ||
|
e43230e46a | ||
|
dd07bc29eb | ||
|
790bbfcc99 | ||
|
994e70d43a | ||
|
7d5e19716a | ||
|
11530c675d | ||
|
1c474352fe | ||
|
3b2d635390 | ||
|
055c532d31 | ||
|
55e08f96a6 | ||
|
3e0a9180bc | ||
|
01ed71b243 | ||
|
595c058517 | ||
|
e04b0a6995 | ||
|
8961848ab1 | ||
|
6fb1c72b7d | ||
|
400d79d6ab | ||
|
ce6018f387 | ||
|
65d9dca917 | ||
|
2232b08351 | ||
|
0732ab6ad5 | ||
|
fb568768c5 | ||
|
7fd56e0add | ||
|
38a3f8cf4c | ||
|
8d8a5b36b6 | ||
|
a70cc7beb1 | ||
|
c5421699b1 | ||
|
d9f809864f | ||
|
bb23f509d7 | ||
|
254e184677 | ||
|
6c07141abd | ||
|
5822fc1d5b | ||
|
2cd7ea257c | ||
|
0a21b72f9c | ||
|
80280f6f4a | ||
|
810a8e76d1 | ||
|
bd3b4906d1 | ||
|
d9c740f014 | ||
|
ec63a18960 | ||
|
fab0618b6b | ||
|
13e57e7aa8 | ||
|
1299c9162c | ||
|
a64f33dbcc | ||
|
836b66e110 | ||
|
802119d789 | ||
|
f94135cadd | ||
|
474c0f980e | ||
|
2cf0f791f2 | ||
|
aec4d1dc60 | ||
|
8d28bb5a66 | ||
|
1d5e553f9c | ||
|
9d79c52d20 | ||
|
1ac03e2f1d | ||
|
801f05f45e | ||
|
345196aa3b | ||
|
a01b52421e | ||
|
86cedfe8b2 | ||
|
709128225b | ||
|
59a5bacb2e | ||
|
3a21df31ee | ||
|
0e3407ec2f | ||
|
fd6609e961 | ||
|
b0b5e045ff | ||
|
f293d7cccc | ||
|
974ed29c36 | ||
|
53eb033034 | ||
|
f2545b98ab | ||
|
1bf51e855d | ||
|
e4da2a695b | ||
|
1a905d9a42 | ||
|
7d9d0c15d3 | ||
|
d34f5c2712 | ||
|
a56dd66c98 | ||
|
3946bc6662 | ||
|
6647dd8f08 | ||
|
9371122bed | ||
|
734cd8ee08 | ||
|
d85d9d05bb | ||
|
1d7b87b3a9 | ||
|
f0cfde36f2 | ||
|
7e220d2741 | ||
|
c4b1351a43 | ||
|
90d2597bc3 | ||
|
c1508c94c2 | ||
|
1a0d805bad | ||
|
1a802469f5 | ||
|
646ae7e2d1 | ||
|
563d76ff9f | ||
|
54d0473e85 | ||
|
ec8a61c3af | ||
|
0315eaff8a | ||
|
b34f44c867 | ||
|
ea7056835f | ||
|
0bb4cdadd9 | ||
|
08f6520557 | ||
|
38b108752e | ||
|
0ee7870bdf | ||
|
37d46ecdb2 | ||
|
2a7911c7d6 | ||
|
fc2d9b4fd2 | ||
|
832020fa81 | ||
|
78606ed4b8 | ||
|
a04ba606e6 | ||
|
ebf2e7ee70 | ||
|
b31881f424 | ||
|
c0a6f6fd08 | ||
|
7e1b1ca730 | ||
|
163d176f0a | ||
|
0f7eab5100 | ||
|
cbf78d474a | ||
|
3615e0de34 | ||
|
e73c081d36 | ||
|
663bf6e3af | ||
|
60194cbafd | ||
|
7d2eb5faf7 | ||
|
5969f3d213 | ||
|
6ddde88389 | ||
|
e9992ed6b4 | ||
|
4db823a3be | ||
|
78732513c8 | ||
|
79557bb878 | ||
|
4094b02ae9 | ||
|
9cc7c118a8 | ||
|
d3dad51c76 | ||
|
5bf35de955 | ||
|
f8928c654b | ||
|
b3903f4ffd | ||
|
46c3f76edc | ||
|
3488408b7a | ||
|
b7b164b84c | ||
|
7c353b7d76 | ||
|
3d9841b61a | ||
|
87e2d2c7b0 | ||
|
ff59619af4 | ||
|
b79f2b86b5 | ||
|
386fd7a459 | ||
|
d4f41901ef | ||
|
f170da3e78 | ||
|
5dc15be98a | ||
|
88c7cfc745 | ||
|
867478697d | ||
|
d27e8cf73a | ||
|
4490caa4f7 | ||
|
1b25f2c531 | ||
|
9d911d1fa0 | ||
|
494feb9f6d | ||
|
2a04943a67 | ||
|
731b29ce91 | ||
|
28fbb70858 | ||
|
44f402c64e | ||
|
c7f391ca44 | ||
|
5ed9c5e168 | ||
|
51b1a86d19 | ||
|
20c46677d0 | ||
|
efb5f232f7 | ||
|
9edf9561b2 | ||
|
5d4eb018ee | ||
|
10fbce056b | ||
|
cbc6d6a572 | ||
|
fb1af9e3d2 | ||
|
c9e410a6f4 | ||
|
c8d72b59c0 | ||
|
cd71d4a2f7 | ||
|
ed0cc59e66 | ||
|
d4f1b48f52 | ||
|
08bec02170 | ||
|
c9f27b83e1 | ||
|
8327585b3b | ||
|
727815d6dd | ||
|
2968f74c6c | ||
|
431499f43f | ||
|
d809283d23 | ||
|
b49275fe6b | ||
|
87c7ce588e | ||
|
26d0a7c742 | ||
|
b01887c7ec | ||
|
745031c020 | ||
|
3ccfc09e01 | ||
|
e8750275c9 | ||
|
f0d7ea4cc6 | ||
|
ea5322af82 | ||
|
0b23f68fb2 | ||
|
1c00129f76 | ||
|
454700af05 | ||
|
d9b482d90a | ||
|
09c2c97069 | ||
|
88faf0ce78 | ||
|
aef83eec31 | ||
|
5bd222c266 | ||
|
d3e7f789b2 | ||
|
87c6e292f1 | ||
|
b597ad0b3f | ||
|
40dd99dea3 | ||
|
6ea2686219 | ||
|
6bca90c4f7 | ||
|
20fa7bd852 | ||
|
6b80119eff | ||
|
1fc5469eb1 | ||
|
bf2769a0c2 | ||
|
325b601bea | ||
|
126642912e | ||
|
57ee16d565 | ||
|
9f0fba063d | ||
|
757fb51415 | ||
|
b1c5655138 | ||
|
4d97dcaa93 | ||
|
96db1b4685 | ||
|
9e21c52d04 | ||
|
5fa801f9f2 | ||
|
0301768b79 | ||
|
4c6dedfa4f | ||
|
6652d55455 | ||
|
3e03dd3e80 | ||
|
84b8a22423 | ||
|
ada03e12fc | ||
|
d66c801350 | ||
|
91846939f6 | ||
|
c85127d76b | ||
|
166bf65e88 | ||
|
f68bdc406f | ||
|
8390482d6c | ||
|
783f454f8e | ||
|
86a67bd245 | ||
|
099b0b65c0 | ||
|
505652efa0 | ||
|
a20088330f | ||
|
a20e2504bb | ||
|
b07bc150e1 | ||
|
46c6588aa1 | ||
|
e36c672c9a | ||
|
c044e1ea8c | ||
|
25b6c1b6ca | ||
|
c25d72d911 | ||
|
73613d1583 | ||
|
1e6542d12d | ||
|
3cffaa43f7 | ||
|
08792e794f | ||
|
7fa26d59da | ||
|
74469829e5 | ||
|
ba414742a6 | ||
|
f6fcfb6dde | ||
|
c5b90c0144 | ||
|
f669881849 | ||
|
d0b70ea6df | ||
|
51af43d492 | ||
|
8a49ad4586 | ||
|
2ae068c2d2 | ||
|
5e07e8dbbd | ||
|
8d2c68505e | ||
|
44f1e7c401 | ||
|
fd2945469e | ||
|
b0f0827cff | ||
|
c1e2c415c4 | ||
|
a258b96602 | ||
|
82ef76b1cb | ||
|
aaa9982932 | ||
|
1cdb4bc9e7 | ||
|
fddb0c9e66 | ||
|
5c78d2b6d2 | ||
|
a4113cbb52 | ||
|
afa3bda3c8 | ||
|
61d7edd15f | ||
|
c2106de7e1 | ||
|
0bf1be6198 | ||
|
af7c20e440 | ||
|
17771a2b89 | ||
|
8840097fe2 | ||
|
7d333efd45 | ||
|
ec65815c57 | ||
|
640b0c71e6 | ||
|
4208876ef3 | ||
|
bff9117360 | ||
|
6b3c4c70d2 | ||
|
0292fd8d91 | ||
|
94a41270d0 | ||
|
75d78bfa53 | ||
|
88263db831 | ||
|
3ef96404ee | ||
|
13f9370f8c | ||
|
1c349270bf | ||
|
c28f0f6286 | ||
|
b20939d211 | ||
|
b76c45fc2b | ||
|
1bbfccd082 | ||
|
cee7e06832 | ||
|
03dff4b382 | ||
|
f546b20a01 | ||
|
0e8d9a264f | ||
|
66a4c98bee | ||
|
36101db500 | ||
|
6bff298d1e | ||
|
c832f8ffbb | ||
|
d7fe46dbde | ||
|
5951186463 | ||
|
7191042bb8 | ||
|
a80fbcc252 | ||
|
c321758afd | ||
|
ca8f59a643 | ||
|
8110882617 | ||
|
a283b66541 | ||
|
569de37e16 | ||
|
669f3394c7 | ||
|
87ec2c7378 | ||
|
ea7a31cad1 | ||
|
c894ade25e | ||
|
a3fe127a92 | ||
|
7f526be879 | ||
|
71ae174e29 | ||
|
2277d134cc | ||
|
880aa265f1 | ||
|
ea639e64a4 | ||
|
b60fdee5e2 | ||
|
44f07fb81d | ||
|
60a85fadd3 | ||
|
68367125e1 | ||
|
6112f60681 | ||
|
ecc74e2ae5 | ||
|
c2dae43965 | ||
|
6e3a25d255 | ||
|
4e13cf1d85 | ||
|
0fee1897eb | ||
|
9dd51467fe | ||
|
cec6dca72f | ||
|
58df2b9bcf | ||
|
cc6bdd6295 | ||
|
7a044f9f78 | ||
|
e77acde005 | ||
|
ece5082096 | ||
|
0723026a56 | ||
|
0b7346f4a1 | ||
|
a357db1ff6 | ||
|
a2287f8d0a | ||
|
076decd6e9 | ||
|
d40461de94 | ||
|
4b1492be0d | ||
|
1676246743 | ||
|
f7c9f6d142 | ||
|
6022339091 | ||
|
25d97afa60 | ||
|
df635b963c | ||
|
6d0a054170 | ||
|
177d3af77f | ||
|
35f71c619b | ||
|
31a80ccce4 | ||
|
3194cec8df | ||
|
57fd3c5801 | ||
|
86617f1bda | ||
|
35f36c598c | ||
|
36bd7816fe | ||
|
fd632392d4 | ||
|
423b48866f | ||
|
745bc24343 | ||
|
04a823c7d0 | ||
|
56af4c2fcb | ||
|
77daf6812a | ||
|
c7e964ec42 | ||
|
694e2c2a4f | ||
|
df6537fae8 | ||
|
71f6542571 | ||
|
5a6294c08e | ||
|
f0390786d6 | ||
|
719d18c670 | ||
|
a116dcd3bb | ||
|
e2dd3dcf88 | ||
|
822f0c3cb2 | ||
|
63fb39016b | ||
|
e79f8d58ca | ||
|
a469dfb583 | ||
|
6cbc545f84 | ||
|
ea1e24d8f9 | ||
|
6fcb4d93c9 | ||
|
36031c9d0f | ||
|
a70f5e2440 | ||
|
6c42aab567 | ||
|
27631a1746 | ||
|
320f5cebc7 | ||
|
ea2a583803 | ||
|
9965801258 | ||
|
24ed0521c7 | ||
|
d8fed79810 | ||
|
c30db15efa | ||
|
817869b915 | ||
|
3038dd9440 | ||
|
276b2c737a | ||
|
62543b94c9 | ||
|
4d607ed276 | ||
|
26e2865e35 | ||
|
f7c94c9eb5 | ||
|
25a91f89fd | ||
|
246f055145 | ||
|
8c3bf0d4cb | ||
|
e49f24e95f | ||
|
48a18ff771 | ||
|
37161057c6 | ||
|
68d1157495 | ||
|
19c3b2f4b6 | ||
|
7051c332a1 | ||
|
0f2a0b7c97 | ||
|
bb6ef42d0c | ||
|
3c591d4968 | ||
|
cf356a8e35 | ||
|
3a0ce58cda | ||
|
2d7038dc18 | ||
|
25a0e67841 | ||
|
be64209292 | ||
|
0006cb1c27 | ||
|
836d38ed2a | ||
|
ec71017785 | ||
|
faf9449107 | ||
|
f93bbd6bc3 | ||
|
ca12d655ea | ||
|
72e486791b | ||
|
874c28bd88 | ||
|
aa7c0a3544 | ||
|
3624a3b450 | ||
|
c4c50699cc | ||
|
6a3d760cfe | ||
|
8ceaf66a81 | ||
|
fdd2562f32 | ||
|
28457021f8 | ||
|
459716bd6e | ||
|
41e0e6762c | ||
|
a5ee60c1c9 | ||
|
1391c1d607 | ||
|
8e5d47061f | ||
|
28ea43c89e | ||
|
59adcfe1a1 | ||
|
cb38ba12a7 | ||
|
dc8e146aa1 | ||
|
9200f5b516 | ||
|
5c1026c745 | ||
|
95e8c85aa9 | ||
|
0d179d67a3 | ||
|
99ce40f736 | ||
|
d2315f4fd7 | ||
|
6370404985 | ||
|
1ed38fff6f | ||
|
ba48713219 | ||
|
dfc1e6a2b9 | ||
|
4237331222 | ||
|
223bcdc261 | ||
|
087124ce42 | ||
|
c5cfd8f40e | ||
|
b70a23a10a | ||
|
ccf55bcd01 | ||
|
d227660d6c | ||
|
ca6202fc19 | ||
|
f6b5a40813 | ||
|
69c0cefe5e | ||
|
ad8772e191 | ||
|
e095767bc6 | ||
|
610f19bea1 | ||
|
4637559c6e | ||
|
766dcff6c6 | ||
|
b30dc39e37 | ||
|
a23178633a | ||
|
7c074e1cd4 | ||
|
4a8b96a6d2 | ||
|
290a206012 | ||
|
7b86b3843f | ||
|
553adb2ad0 | ||
|
cc55a78fb4 | ||
|
bd1df09098 | ||
|
c61fae05d6 | ||
|
e3cb5bfa07 | ||
|
09f6034f2f | ||
|
63b8a76bf6 | ||
|
5fe536a058 | ||
|
342184e532 | ||
|
e5c169ddf8 | ||
|
622ba2b0e3 | ||
|
fbfe8f8d14 | ||
|
3c96fda59b | ||
|
7cb03dde44 | ||
|
db34f89251 | ||
|
98b9972fa5 | ||
|
dd5507036f | ||
|
75da209de1 | ||
|
bb8a2bfff7 | ||
|
a7f8febed7 | ||
|
21d291531e | ||
|
d133fa0d74 | ||
|
2653a9447b | ||
|
bc04903741 | ||
|
5501418727 | ||
|
2737b3635f | ||
|
e29688530d | ||
|
2eec26b648 | ||
|
d4d1d87680 | ||
|
869be3e361 | ||
|
15376c19be | ||
|
f92b610a03 | ||
|
268f6447cb | ||
|
ed93c647d2 | ||
|
f60db358ee | ||
|
ad6bd840a9 | ||
|
0fa9e8d9f5 | ||
|
481e265ec7 | ||
|
583833017b | ||
|
d50ebcf9eb | ||
|
c448531551 | ||
|
81ddf7e1d2 | ||
|
180035296e | ||
|
afe24a3e65 | ||
|
9058201de0 | ||
|
49a54f8a13 | ||
|
5cebc7f5cd | ||
|
4e7e475a5e | ||
|
f8a5bc94e5 | ||
|
6b0b441329 | ||
|
1c8aad7a07 | ||
|
7743a6a71d | ||
|
65a18e78dd | ||
|
12a0cbf4e1 | ||
|
55c2073813 | ||
|
39bc8c45c3 | ||
|
46465f5409 | ||
|
7b5f23e17d | ||
|
380efc5ae0 | ||
|
d1a123997d | ||
|
722419dffc | ||
|
bcd69f812e | ||
|
1075cd9995 | ||
|
e5be1ff157 | ||
|
398fe55083 | ||
|
b4ee537f7c | ||
|
b0ad30a5d8 | ||
|
26d17e2e69 | ||
|
43b619b2c6 | ||
|
0e4426e084 | ||
|
a27943ea14 | ||
|
b8b24da292 | ||
|
45eef03576 | ||
|
270f77df24 | ||
|
dc580b9eff | ||
|
fed7e94d8d | ||
|
9bb0ca2f0d | ||
|
3d82becadc | ||
|
e2b92c99ef | ||
|
fea7275148 | ||
|
64581fc81b | ||
|
37ae302fc6 | ||
|
c51cc02a87 | ||
|
6c92fbd8de | ||
|
065215cc27 | ||
|
eb8a88dbd6 | ||
|
62e3258dda | ||
|
9959d2c70d | ||
|
19e8b6e11f | ||
|
63447d1ec3 | ||
|
75fac93544 | ||
|
23f7faa74c | ||
|
284fdb2fcb | ||
|
44b27bbfc8 | ||
|
35d2d55a27 | ||
|
292f1953e1 | ||
|
5cd0828dd3 | ||
|
077281a77a | ||
|
7055d250d0 | ||
|
8c0196301d | ||
|
f99aacd9dc | ||
|
43b83bec83 | ||
|
fc1b4658fc | ||
|
05a28ea346 | ||
|
624f986978 | ||
|
91a9957918 | ||
|
8cd18e47e7 | ||
|
517868f73c | ||
|
18d80f4ed5 | ||
|
2e2c751d75 | ||
|
7aa5e2cbe1 | ||
|
7d6db20081 | ||
|
911ec9444b | ||
|
2883c6bab3 | ||
|
f5fed54195 | ||
|
91b1446375 | ||
|
56df145052 | ||
|
9e873f77f9 | ||
|
27bbd91635 | ||
|
2c73c80c9f | ||
|
3a3a9b5f4e | ||
|
3f89ed4579 | ||
|
02b01b89f2 | ||
|
4a6af0b938 | ||
|
9c5b4e3cc9 | ||
|
b3f087dbc7 | ||
|
aa90257ec7 | ||
|
348bbc81ca | ||
|
4231e7f3ff | ||
|
4cbc82619d | ||
|
1b0d802941 | ||
|
dc72a1e58d | ||
|
1d5333c737 | ||
|
d289041ff3 | ||
|
f287483397 | ||
|
1ac3b6c173 | ||
|
1d090765ba | ||
|
6c6050528a | ||
|
835d376ca6 | ||
|
68e9fd8541 | ||
|
e49f8f8c61 | ||
|
b396322f1e | ||
|
ea434e3469 | ||
|
84587ac232 | ||
|
9fbc83fe9d | ||
|
965972a719 | ||
|
34d19c4a72 | ||
|
9b0910767d | ||
|
32cc077e1f | ||
|
1e7bec48b6 | ||
|
c035837a14 | ||
|
cb5eb8983b | ||
|
9c86cb7954 | ||
|
ef94c1eccc | ||
|
64eaa5e3aa | ||
|
826f584aa8 | ||
|
c5530aee5f | ||
|
817f01593e | ||
|
79727f0e97 | ||
|
abac82c1c6 | ||
|
02c51c6059 | ||
|
46b45d0387 | ||
|
e6c549c457 | ||
|
1aa2983216 | ||
|
a095f89c81 | ||
|
287902f01b | ||
|
7dc3e4e7a1 | ||
|
0b31aad901 | ||
|
1791d8e4ee | ||
|
b7c2054a66 | ||
|
81f16d7061 | ||
|
5c151a3a09 | ||
|
3c46c6cbb9 | ||
|
d31f09c174 | ||
|
063eeb34ec | ||
|
f06192601c | ||
|
091221ba2c | ||
|
7c22111ae0 | ||
|
daac3eed09 | ||
|
32ee7feea8 | ||
|
2026848a77 | ||
|
a7b205603c | ||
|
291d14f2d4 | ||
|
4a68137cfa | ||
|
b2f0374426 | ||
|
5fcca6ed0c | ||
|
a05c44ad1e | ||
|
f9880c8931 | ||
|
b876896628 | ||
|
375c2a3363 | ||
|
5dbc749000 | ||
|
36fbd91280 | ||
|
d2ad3e7ff1 | ||
|
072f1f112e | ||
|
6c79f98d3c | ||
|
8edebbb684 | ||
|
4839ca631a | ||
|
5018fe90de | ||
|
379144bcfe | ||
|
d2cd0ece5f | ||
|
10b7ccaa92 | ||
|
925eb618de | ||
|
96800de052 | ||
|
9df659b647 | ||
|
aec0bd5d11 | ||
|
c1c9c2c578 | ||
|
c739fa3ca1 | ||
|
355dfcb6c1 | ||
|
8f7c291e33 | ||
|
71fbb3df24 | ||
|
9cfa9b2c57 | ||
|
b9b8178e30 | ||
|
089cefb77b | ||
|
ee9ea1faab | ||
|
22b2cf8b0c | ||
|
10bd3c9dda | ||
|
6c3147fcf4 | ||
|
5e8d169c50 | ||
|
8657a35fd7 | ||
|
97c0b60a22 | ||
|
8a284d6c49 | ||
|
ff4b2f2dec | ||
|
aedce256af | ||
|
48b4b97132 | ||
|
6b1f88df66 | ||
|
d246c46227 | ||
|
7fdd9dc293 | ||
|
202861a900 | ||
|
4c41e7cb29 | ||
|
92823dcea9 | ||
|
795a5227bc | ||
|
bbd4bb99b7 | ||
|
f7443061ef | ||
|
416c3c16ab | ||
|
a11956ece0 | ||
|
c4e0c659a7 | ||
|
f6ba09b171 | ||
|
b827881024 | ||
|
89e395b5f4 | ||
|
c590954eb9 | ||
|
a5af3866fd | ||
|
1bef56f030 | ||
|
9045737504 | ||
|
0e48772c84 | ||
|
72690b5cff | ||
|
07bae26fd3 | ||
|
7b0e4617d3 | ||
|
af797296ed | ||
|
c826c7da0d | ||
|
20af1396b9 | ||
|
f567aca532 | ||
|
c60b658443 | ||
|
fde8c34088 | ||
|
deb6a07160 | ||
|
0a66e1b24c | ||
|
819a551b77 | ||
|
5119b71035 | ||
|
33945e14d3 | ||
|
1d9705fb17 | ||
|
1949937b5e | ||
|
956e21269e | ||
|
ff73418fc1 | ||
|
76d40f44d8 | ||
|
ab6944b7b0 | ||
|
4d741bd200 | ||
|
f3a163b382 | ||
|
011b7f090f | ||
|
4ead92cfbe | ||
|
9ee2f30df4 | ||
|
4a749f6c38 | ||
|
cf924373f6 | ||
|
d1f5b25418 | ||
|
822f53c888 | ||
|
f04c5ee902 | ||
|
022d60ec2c | ||
|
8fce374cae | ||
|
e78e2738ef | ||
|
5728493abb | ||
|
af1c8ec35c | ||
|
cb84abfb88 | ||
|
df1e349470 | ||
|
357ce881a6 | ||
|
aa417f5410 | ||
|
0ca29b9518 | ||
|
22db04068e | ||
|
65e74281e0 | ||
|
46d89929ba | ||
|
1c941a355b | ||
|
0d6248fbeb | ||
|
98d4d00f96 | ||
|
e2d618f472 | ||
|
79d643db6c | ||
|
13891330d9 | ||
|
6fd0c2bf77 | ||
|
b8b42a1fd2 | ||
|
fba1b28615 | ||
|
98828387e3 | ||
|
1c12247b71 | ||
|
ee10ebd5d7 | ||
|
9937ad6e85 | ||
|
60ce393531 | ||
|
bc33cad999 | ||
|
4261b50d9d | ||
|
e2b3141536 | ||
|
266aa1b15d | ||
|
b57a7475c5 | ||
|
8cf48eb5f7 | ||
|
35541bbbb9 | ||
|
59424a3e73 | ||
|
2056888667 | ||
|
1e54785742 | ||
|
a58ed6f331 | ||
|
9a79b71d9f | ||
|
e9098887fb | ||
|
9e8a3ef7fc | ||
|
221585908a | ||
|
f4c111f04c | ||
|
de12e25450 | ||
|
2545ebbece | ||
|
14ec6ce43b | ||
|
0cac1e65aa | ||
|
1e4d57ab83 | ||
|
49b42b7e03 | ||
|
89c0b5dbdf | ||
|
8138675808 | ||
|
726129836a | ||
|
356b3c85e5 | ||
|
1a0680ead9 | ||
|
c876d891fa | ||
|
a08232f164 | ||
|
0f8cf46626 | ||
|
95017b5fc8 | ||
|
1374d337d8 | ||
|
35e7cc685f | ||
|
2ac4883bda | ||
|
5a7212aebb | ||
|
3f39cc2955 | ||
|
8a5bfe884a | ||
|
046dd53913 | ||
|
d2f174635c | ||
|
01a760d8f0 | ||
|
9215d2c9f9 | ||
|
0bb09ae91e | ||
|
afeff61268 | ||
|
6f5a595975 | ||
|
8a69ec062e | ||
|
45fb673e49 | ||
|
93b5f1809e | ||
|
ff05eaae48 | ||
|
b48b67ed10 | ||
|
3ead6f63b6 | ||
|
77235f0ad8 | ||
|
f6a80ba929 | ||
|
295ceb90d5 | ||
|
df41a82b23 | ||
|
34500e07c6 | ||
|
1e6097e3b2 | ||
|
521facedc4 | ||
|
de80a2ba97 | ||
|
3afd615447 | ||
|
87e52bec0a | ||
|
b89f04b1f1 | ||
|
e4f802e5f4 | ||
|
16fc2c456f | ||
|
ef7656ae98 | ||
|
ad2df4871e | ||
|
8d5f50b695 | ||
|
bbc056eef0 | ||
|
82fd3c632e | ||
|
dbae939808 | ||
|
c2dcacb0de | ||
|
8f45618cd0 | ||
|
29a87c850c | ||
|
395518ff76 | ||
|
08da763327 | ||
|
d7ec3c182d | ||
|
ffe6418392 | ||
|
88adf3ee27 | ||
|
f406db85dc | ||
|
d5f8af471a | ||
|
046459a72d | ||
|
49b0f59790 | ||
|
71551058ce | ||
|
2a9e6473eb | ||
|
576ef9c4ed | ||
|
4d64fca9de | ||
|
8b1307e1ca | ||
|
eae4ed7b27 | ||
|
6997ea608d | ||
|
d97d738880 | ||
|
f72b07d0f4 | ||
|
b89916a3d7 | ||
|
e4aeb66257 | ||
|
578f8f4062 | ||
|
8fbf82f4ea | ||
|
478ae7e657 | ||
|
d319761483 | ||
|
16fb716872 | ||
|
b8d9d097d8 | ||
|
ec79a1d0e8 | ||
|
31284e0c5e | ||
|
8e64b1840d | ||
|
dd9bf04e35 | ||
|
76a27fdef1 | ||
|
f96443515a | ||
|
0f1bde2274 | ||
|
6fb94e445f | ||
|
e02a4e44c0 | ||
|
71dc882d5d | ||
|
ae4a29d41c | ||
|
d27a3b6421 | ||
|
03a58cf50a | ||
|
f43e370e89 | ||
|
45bdb9023a | ||
|
6d7aa93157 | ||
|
d083910255 | ||
|
db3f9af174 | ||
|
7d2ad5e4ed | ||
|
84d3771984 | ||
|
1eebe006ae | ||
|
0327ce53a6 | ||
|
85442c1d07 | ||
|
a6ed51b7a7 | ||
|
dc8b4b5252 | ||
|
8ca74d3de2 | ||
|
1923e6f152 | ||
|
0d621e085b | ||
|
8749c78622 | ||
|
510d88fff1 | ||
|
d9460dcd9e | ||
|
d0607b6fec | ||
|
2481305a1a | ||
|
9990889cce | ||
|
14f743c493 | ||
|
df8fe2e3ef | ||
|
906fea972e | ||
|
04a82d5c23 | ||
|
f2944fd33b | ||
|
dcb1e53aba | ||
|
a4d8508b40 | ||
|
f109aa1f73 | ||
|
f09c18539b | ||
|
c5aeac6898 | ||
|
b81c801f2d | ||
|
f3cedf2f44 | ||
|
910999e7e4 | ||
|
5bceb06fc5 | ||
|
cc098f3090 | ||
|
a73c5aec27 | ||
|
68b29d9529 | ||
|
57f069d750 | ||
|
67e12bc590 | ||
|
ff2a08747b | ||
|
3791d22713 | ||
|
638ecafb69 | ||
|
5176feed81 | ||
|
8c63033461 | ||
|
8c878f8e25 | ||
|
c1c7e30ec9 | ||
|
28ae7f81e9 | ||
|
71de4d769d | ||
|
f502944c5b | ||
|
99d21e93f3 | ||
|
350e610f15 | ||
|
4f208b6b31 | ||
|
dcb781a57e | ||
|
09db71ae7f | ||
|
60dc43340d | ||
|
256b10bd7d | ||
|
77adcaeb50 | ||
|
abf2ca40a2 | ||
|
d5183123fa | ||
|
8b0d96acba | ||
|
66122a45ed | ||
|
a2953ac9ac | ||
|
c51db5294f | ||
|
8cba43bfed | ||
|
4b1c7aa6c5 | ||
|
14c1e95841 | ||
|
4f5a4d9ce0 | ||
|
82f2ad21e1 | ||
|
8a5b803788 | ||
|
86b46ebf19 | ||
|
ca999053ae | ||
|
8b152f3c96 | ||
|
c68bfbdf2e | ||
|
7e8180fefc | ||
|
305226a3bd | ||
|
c8752b03c1 | ||
|
8195deebe3 | ||
|
88f2156aeb | ||
|
b1982a81dc | ||
|
18e19213f4 | ||
|
6fb67b86f5 | ||
|
a4c15769d0 | ||
|
894e86d3ee | ||
|
9712ef8abf | ||
|
908107d24f | ||
|
d667efa9de | ||
|
e649d7219b | ||
|
1f5698ec86 | ||
|
15341aa80e | ||
|
3f4895bd40 | ||
|
103c93a06c | ||
|
21f02d3e6d | ||
|
9ce93a1da6 | ||
|
d0e1468acd | ||
|
8a5bb239c5 | ||
|
816cde32a9 | ||
|
2b514a9df7 | ||
|
ef39ba26a0 | ||
|
59ca3d3d16 | ||
|
80ff60f0aa | ||
|
488c2e2f17 | ||
|
9e11fc1e99 | ||
|
a8b590ecbb | ||
|
8ca8591ed6 | ||
|
e393314423 | ||
|
c54373bd9c | ||
|
f459b6833a | ||
|
0a6eb76542 | ||
|
e6d42b5319 | ||
|
0d2804ce55 | ||
|
8efdd9d29a | ||
|
27c36a4dce | ||
|
2f74611844 | ||
|
d8265add2d | ||
|
01cd02e4c5 | ||
|
2b98f85961 | ||
|
338d1bd8e7 | ||
|
7dedaeeef7 | ||
|
54830b8968 | ||
|
41b866c1ee | ||
|
fe7224e537 | ||
|
f5e38f6e23 | ||
|
063335377e | ||
|
ee6f511126 | ||
|
75574486d3 | ||
|
03a09f0f42 | ||
|
075b3ec3d2 | ||
|
3d5aa6a44b | ||
|
8350b42c1b | ||
|
7061a62e88 | ||
|
133fceeee5 | ||
|
38331b8918 | ||
|
5bbba0b65a | ||
|
07a63b2a0a | ||
|
9ddc70fc81 | ||
|
2b6f5fde13 | ||
|
d4b006656d | ||
|
b1e70723c7 | ||
|
1ec1ad19f8 | ||
|
b96b27ed91 | ||
|
8b145cb1d8 | ||
|
85d7adc0b5 | ||
|
d530599b5d | ||
|
16215d5b96 | ||
|
db685e5807 | ||
|
190304344c | ||
|
24e8a3c62e | ||
|
571b02f381 | ||
|
36e3f1b3ec | ||
|
564d6bdc5d | ||
|
1e2b040f75 | ||
|
5876d5c295 | ||
|
54e53932bd | ||
|
c9b3c2e6aa | ||
|
912c102d9a | ||
|
2b7cbf9759 | ||
|
1cc6b2e31e | ||
|
7fa797712a | ||
|
636f36ebad | ||
|
8e0888b846 | ||
|
ce5dd5022b | ||
|
4a24cc0095 | ||
|
f2bd0edc51 | ||
|
756d17bada | ||
|
81e061af30 | ||
|
a60cd1d475 | ||
|
f78323e0fb | ||
|
fd6cbe0f6c | ||
|
c30e905e19 | ||
|
f5650c1b13 | ||
|
ecebbc86c6 | ||
|
bfeed40bfe | ||
|
2ad22d7d19 | ||
|
262735c219 | ||
|
3abc98acc8 | ||
|
eaa8ba814e | ||
|
794eac0519 | ||
|
b75139662b | ||
|
9e24f4ca54 | ||
|
1b5281c23e | ||
|
be15601a67 | ||
|
35cd42bfea | ||
|
03e48c0968 | ||
|
3ea92c0d6e | ||
|
8e43e40810 | ||
|
a14b91cdda | ||
|
08609e9760 | ||
|
ca0a3dbf9e | ||
|
9d0b9f0ded | ||
|
8911af58cf | ||
|
d23920cc61 | ||
|
ee353ca776 | ||
|
bba4c6e4f4 | ||
|
1d126913e9 | ||
|
b3afee2f86 | ||
|
31983fbb8e | ||
|
ba8870823c | ||
|
5f1bc01636 | ||
|
ffa9e508e9 | ||
|
cfb051396a | ||
|
397ebe8428 | ||
|
2626a28e6d | ||
|
fb2d7cbe28 | ||
|
874b3faf8c | ||
|
a54a11dca0 | ||
|
e7beb6b1a7 | ||
|
30edd53f3e | ||
|
fe446862c2 | ||
|
6788e09ae9 | ||
|
f0c2d16749 | ||
|
9b21bd1a6f | ||
|
169ef00b9e | ||
|
aa93b94b61 | ||
|
d7dadc2f0e | ||
|
33ac9eea13 | ||
|
9ff41f748e | ||
|
9589ac6d02 | ||
|
cfa74985e0 | ||
|
38ae660a68 | ||
|
ca4a07f766 | ||
|
9281bdb61b | ||
|
80b65ae7e7 | ||
|
61a254f580 | ||
|
4ba90a727b | ||
|
27558ffd89 | ||
|
c44de48481 | ||
|
a021ddeffd | ||
|
cdd25e3b0f | ||
|
1286417aec | ||
|
fcac4f1488 | ||
|
f655fefe52 | ||
|
7979f0a683 | ||
|
d9df2ead84 | ||
|
b07ca20e0e | ||
|
3a1cd8985c | ||
|
a79be0513b | ||
|
569ac502ed | ||
|
b86d2e6e73 | ||
|
f8d24d3402 | ||
|
589768864a | ||
|
bd2512c27e | ||
|
9e4c1120fa | ||
|
4eff0d2660 | ||
|
09726876ad | ||
|
45c6ad3f6d | ||
|
97e4d51c29 | ||
|
f134bc264b | ||
|
57eb56b92d | ||
|
6c68ba1d22 | ||
|
cabc58feb4 | ||
|
364f95612a | ||
|
45669dad33 | ||
|
19e326c140 | ||
|
e1c0caa357 | ||
|
9d3e1af69b | ||
|
662f2bff4c | ||
|
9a9d37cc50 | ||
|
3e4c9f25a8 | ||
|
2cb9d5c636 | ||
|
a53b60fca4 | ||
|
09b77972d9 | ||
|
7a8c2fc98b | ||
|
355989d6d1 | ||
|
08814ad811 | ||
|
6a00cf9942 | ||
|
e79202de3c | ||
|
527e288568 | ||
|
ed32d17936 | ||
|
3a8e1cd4a5 | ||
|
186de045b3 | ||
|
feee534829 | ||
|
f4bf260cdb | ||
|
442138ef36 | ||
|
b9b411ec30 | ||
|
d72a152195 | ||
|
390226d2ef | ||
|
a16495d06f | ||
|
7092f40701 | ||
|
458189366f | ||
|
9d772a45d5 | ||
|
1c860e8578 | ||
|
50fc027289 | ||
|
e310601266 | ||
|
78fd434686 | ||
|
a6de3a9642 | ||
|
5ed478ed11 | ||
|
b93f9d4c94 | ||
|
f4ec20a2e2 | ||
|
8bc07fad32 | ||
|
612066165a | ||
|
b273d58a80 | ||
|
b2544139e6 | ||
|
367ffdea1f | ||
|
4c854dd5a7 | ||
|
e96b2f702d | ||
|
cff22a8951 | ||
|
1b7c5a1373 | ||
|
2bd60021aa | ||
|
2b8c06bda7 | ||
|
c36b4626a9 | ||
|
b5f245d9b6 | ||
|
899a6f0d4a | ||
|
ba5fc2a627 | ||
|
9525bd6ca0 | ||
|
6c4f6dd59a | ||
|
997b573aa4 | ||
|
ca67c56222 | ||
|
56fc003a96 | ||
|
8cb811ca1d | ||
|
9e5246c619 | ||
|
b2bcf39872 | ||
|
0fc864c2d5 | ||
|
4b39aad07e | ||
|
10c1f8e1cf | ||
|
ed880be3e5 | ||
|
e5714dc1fc | ||
|
1732167af1 | ||
|
ee28bbc19b | ||
|
345d4f8ab1 | ||
|
e745b6c16e | ||
|
ee57af8e99 | ||
|
700f73e8e9 | ||
|
fb6a7442a6 | ||
|
b8a3b62f0f | ||
|
78b84ba698 | ||
|
fdf479241a | ||
|
8fcbf48bdf | ||
|
268e599423 | ||
|
ae8fece9bb | ||
|
c91c69540b | ||
|
c497385bc7 | ||
|
37468cad59 | ||
|
5f70ae9923 | ||
|
c1b96a36a7 | ||
|
5cdda0f3a8 | ||
|
f0e1fc1a49 | ||
|
77d8c68b13 | ||
|
48d7bfd225 | ||
|
f60435c878 | ||
|
41332f22c3 | ||
|
a6ff18c872 | ||
|
b90378d739 | ||
|
a5a0089daa | ||
|
82c71e4255 | ||
|
dce5541c3a | ||
|
8bbda75865 | ||
|
e650562758 | ||
|
d3d60071ae | ||
|
85503cf92b | ||
|
f67b872bd1 | ||
|
c497e9884a | ||
|
3b021af1bd | ||
|
6fe648b9a9 | ||
|
03594dab6a | ||
|
b830f4c3ca | ||
|
28b638f0aa | ||
|
4d0ecd1fb4 | ||
|
b62c5c57af | ||
|
c7b647b1eb | ||
|
63563c412c | ||
|
9fed3fce3e | ||
|
a8f8ff30a2 | ||
|
d270e84810 | ||
|
0b8c67f2f0 | ||
|
2b73d3d109 | ||
|
a7838b80c3 | ||
|
f2f2a12ea1 | ||
|
382ebb55b9 | ||
|
c6b32fa946 | ||
|
ae0e1bfaef | ||
|
aa73cb9bbf | ||
|
d60b98236b | ||
|
2cd1947ef5 | ||
|
e10f1df321 | ||
|
2c5fd4aade | ||
|
0ce4b9a7de | ||
|
44b1178cfe | ||
|
7bdcaa6737 | ||
|
cab8981716 | ||
|
36ad834d43 | ||
|
b999abb9ea | ||
|
55f5c7491c | ||
|
52b7bb0fce | ||
|
b19fd8ed85 | ||
|
6bdd9c9d78 | ||
|
87c210952b | ||
|
4b9c3e8409 | ||
|
c6f2d8b6fc | ||
|
b23481bd21 | ||
|
64a626cbb5 | ||
|
d4f964ef8e | ||
|
7f09405335 | ||
|
e37da81dc6 | ||
|
febadc5dd6 | ||
|
4aea68ecd1 | ||
|
6a97f96745 | ||
|
3ed03c267a | ||
|
c60e17b341 | ||
|
689c578081 | ||
|
2723a55766 | ||
|
d02da2f3a5 | ||
|
462fd8bad1 | ||
|
75dd93436c | ||
|
eebfe2be08 | ||
|
3a264c1eec | ||
|
9c4dfa7f84 | ||
|
d9c21ce37b | ||
|
1d84440b74 | ||
|
9aa22e1071 | ||
|
9cea8d2725 | ||
|
944238d542 | ||
|
95cbdcb131 | ||
|
38b1fb5fba | ||
|
fe59ed0105 | ||
|
356ab9666f | ||
|
0960762694 | ||
|
1f4a9cff26 | ||
|
a6f97dfbde | ||
|
04aa559037 | ||
|
a55feee960 | ||
|
29c0e3f15f | ||
|
b0bc06ae00 | ||
|
cdbb34e976 | ||
|
5c5b0ace53 | ||
|
a879c822c2 | ||
|
841d49554c | ||
|
0d22b8c23c | ||
|
c29b4c476d | ||
|
5977f42a7e | ||
|
58d2c04b9f | ||
|
c45c792657 | ||
|
c0e0b4b745 | ||
|
aa47f10602 | ||
|
9268b6ca13 | ||
|
4832526477 | ||
|
53e0fe9a69 | ||
|
8ed0791b26 | ||
|
8a87195f55 | ||
|
0b75bb4052 | ||
|
543f533a68 | ||
|
89f4ec12e9 | ||
|
d76758bd1d | ||
|
18629c93a2 | ||
|
b3f3fdeb1d | ||
|
71d99046d1 | ||
|
eedc9bf9ed | ||
|
94b920ab2a | ||
|
873437e6a9 | ||
|
57abad2181 | ||
|
91d95bd42e | ||
|
b102afd68a | ||
|
9e55694512 | ||
|
00cf8131ec | ||
|
026bcd4d4d | ||
|
a27f89157e | ||
|
9a2e9d68e7 | ||
|
c961dd609a | ||
|
dabedc17d0 | ||
|
e7dcd9216b | ||
|
95a22831aa | ||
|
ac2fe3476a | ||
|
df251def26 | ||
|
133a86cd89 | ||
|
1a5654d50b | ||
|
5b28e77ab5 | ||
|
f8c1790558 | ||
|
e2537e43b5 | ||
|
d4107600a5 | ||
|
e996717b60 | ||
|
75ddf209e2 | ||
|
f5515e5e77 | ||
|
1eb375ba80 | ||
|
d3a171e638 | ||
|
e154c83ffe | ||
|
69ad8e384a | ||
|
cfeb7b249b | ||
|
3a35951b46 | ||
|
420ba43d5a | ||
|
2a42bda0a0 | ||
|
6be95dceab | ||
|
a20c060931 | ||
|
b0bd0b1fc0 | ||
|
746aef474a | ||
|
5a2a3727b2 | ||
|
40651322d0 | ||
|
c2a9027c19 | ||
|
25046124c0 | ||
|
2f83ebb3b5 | ||
|
a03069bf89 | ||
|
93555cad3c | ||
|
0a2d916931 | ||
|
a4519d3add | ||
|
67dc297c97 | ||
|
7c39e55089 | ||
|
3f17dd336b | ||
|
10fd654edc | ||
|
686eb7b1f0 | ||
|
97532fb300 | ||
|
e36e4c6354 | ||
|
fcc7177a9a | ||
|
0f786a8c43 | ||
|
184d3a5910 | ||
|
9370b1d2fa | ||
|
f46550262f | ||
|
adec715010 | ||
|
1ce5da80d6 | ||
|
0ffec82e35 | ||
|
32ba594e55 | ||
|
feac754925 | ||
|
565b095a8d | ||
|
5db792e99c | ||
|
c9c3585e1e | ||
|
4a0d461446 | ||
|
630a805c49 | ||
|
5eb276655d | ||
|
2006ad68ee | ||
|
ae56b81a08 | ||
|
e2296d4491 | ||
|
dd40d2e6ce | ||
|
c3cb297205 | ||
|
6bb81e4d6f | ||
|
c58a1c9ead | ||
|
da265198f1 | ||
|
1dfc18683d | ||
|
9a467f8840 | ||
|
aba3ed7ca0 | ||
|
22b897db2e | ||
|
833ef98ff5 | ||
|
43cd9c4100 | ||
|
4dee59a49d | ||
|
52846a3b24 | ||
|
ac0c88768d | ||
|
9977a398af | ||
|
643998332e | ||
|
13444ae6c3 | ||
|
05ec13394c | ||
|
b7cbe16159 | ||
|
f0ecf58752 | ||
|
e9fbc875e0 | ||
|
99ae39012f | ||
|
fbd1b2ec30 | ||
|
eb6781481a | ||
|
78e4135159 | ||
|
54350bcbc4 | ||
|
09a32174c8 | ||
|
740af0465e | ||
|
08f3ecbf25 | ||
|
02768a8181 | ||
|
c86113b288 | ||
|
246cc234c4 | ||
|
4e4c4b5f96 | ||
|
ca6d5937ba | ||
|
ece2db08da |
391 changed files with 57398 additions and 24647 deletions
1
.gitattributes
vendored
1
.gitattributes
vendored
|
@ -1 +0,0 @@
|
||||||
/CHANGELOG.md merge=union
|
|
74
.github/ISSUE_TEMPLATE.md
vendored
74
.github/ISSUE_TEMPLATE.md
vendored
|
@ -1,74 +0,0 @@
|
||||||
<!--
|
|
||||||
Thanks for reporting an issue to LBRY and helping us improve!
|
|
||||||
|
|
||||||
To make it possible for us to help you, please fill out below information carefully.
|
|
||||||
|
|
||||||
Before reporting any issues, please make sure that you're using the latest version.
|
|
||||||
- App: https://github.com/lbryio/lbry-desktop/releases
|
|
||||||
- Daemon: https://github.com/lbryio/lbry/releases
|
|
||||||
|
|
||||||
We are also available on Discord at https://chat.lbry.com
|
|
||||||
-->
|
|
||||||
|
|
||||||
|
|
||||||
## The Issue
|
|
||||||
|
|
||||||
In order to <achieve some value>,
|
|
||||||
as a <type of user>,
|
|
||||||
I want <some functionality>.
|
|
||||||
|
|
||||||
|
|
||||||
### Steps to reproduce
|
|
||||||
1.
|
|
||||||
2.
|
|
||||||
3.
|
|
||||||
|
|
||||||
### Expected behaviour
|
|
||||||
Tell us what should happen
|
|
||||||
|
|
||||||
### Actual behaviour
|
|
||||||
Tell us what happens instead
|
|
||||||
|
|
||||||
|
|
||||||
## System Configuration
|
|
||||||
|
|
||||||
<!-- For the app, this info is in the About section at the bottom of the Help page.
|
|
||||||
You can include a screenshot instead of typing it out -->
|
|
||||||
|
|
||||||
<!-- For the daemon, run:
|
|
||||||
curl 'http://localhost:5279' --data '{"method":"version"}'
|
|
||||||
and include the full output -->
|
|
||||||
|
|
||||||
- LBRY Daemon version:
|
|
||||||
- LBRY App version:
|
|
||||||
- LBRY Installation ID:
|
|
||||||
- Operating system:
|
|
||||||
|
|
||||||
|
|
||||||
## Anything Else
|
|
||||||
<!-- Include anything else that does not fit into the above sections -->
|
|
||||||
|
|
||||||
|
|
||||||
## Screenshots
|
|
||||||
<!-- If a screenshot would help explain the bug, please include one or two here -->
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Internal Use
|
|
||||||
|
|
||||||
### Acceptance Criteria
|
|
||||||
1.
|
|
||||||
2.
|
|
||||||
3.
|
|
||||||
|
|
||||||
### Definition of Done
|
|
||||||
- [ ] Tested against acceptance criteria
|
|
||||||
- [ ] Tested against the assumptions of user story
|
|
||||||
- [ ] The project builds without errors
|
|
||||||
- [ ] Unit tests are written and passing
|
|
||||||
- [ ] Tests on devices/browsers listed in the issue have passed
|
|
||||||
- [ ] QA performed & issues resolved
|
|
||||||
- [ ] Refactoring completed
|
|
||||||
- [ ] Any configuration or build changes documented
|
|
||||||
- [ ] Documentation updated
|
|
||||||
- [ ] Peer Code Review performed
|
|
38
.github/PULL_REQUEST_TEMPLATE.md
vendored
38
.github/PULL_REQUEST_TEMPLATE.md
vendored
|
@ -1,38 +0,0 @@
|
||||||
## PR Checklist
|
|
||||||
Please check all that apply to this PR using "x":
|
|
||||||
|
|
||||||
- [ ] I have checked that this PR is not a duplicate of an existing PR (open, closed or merged)
|
|
||||||
- [ ] I have checked that this PR does not introduce a breaking change
|
|
||||||
- [ ] This PR introduces breaking changes and I have provided a detailed explanation below
|
|
||||||
|
|
||||||
|
|
||||||
## PR Type
|
|
||||||
What kind of change does this PR introduce?
|
|
||||||
|
|
||||||
Why is this change necessary?
|
|
||||||
|
|
||||||
<!-- Please check all that apply to this PR using "x". -->
|
|
||||||
|
|
||||||
- [ ] Bugfix
|
|
||||||
- [ ] Feature
|
|
||||||
- [ ] Breaking changes (bugfix or feature that introduces breaking changes)
|
|
||||||
- [ ] Code style update (formatting)
|
|
||||||
- [ ] Refactoring (no functional changes)
|
|
||||||
- [ ] Documentation changes
|
|
||||||
- [ ] Other - Please describe:
|
|
||||||
|
|
||||||
## Fixes
|
|
||||||
|
|
||||||
Issue Number: N/A
|
|
||||||
|
|
||||||
|
|
||||||
## What is the current behavior?
|
|
||||||
|
|
||||||
|
|
||||||
## What is the new behavior?
|
|
||||||
|
|
||||||
|
|
||||||
## Other information
|
|
||||||
|
|
||||||
|
|
||||||
<!-- If this PR contains a breaking change, please describe the impact and solution strategy for existing applications below. -->
|
|
206
.github/workflows/main.yml
vendored
Normal file
206
.github/workflows/main.yml
vendored
Normal file
|
@ -0,0 +1,206 @@
|
||||||
|
name: ci
|
||||||
|
on: ["push", "pull_request", "workflow_dispatch"]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
|
||||||
|
lint:
|
||||||
|
name: lint
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.9'
|
||||||
|
- name: extract pip cache
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/pip
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
||||||
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
|
- run: pip install --user --upgrade pip wheel
|
||||||
|
- run: pip install -e .[lint]
|
||||||
|
- run: make lint
|
||||||
|
|
||||||
|
tests-unit:
|
||||||
|
name: "tests / unit"
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
os:
|
||||||
|
- ubuntu-20.04
|
||||||
|
- macos-latest
|
||||||
|
- windows-latest
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.9'
|
||||||
|
- name: set pip cache dir
|
||||||
|
shell: bash
|
||||||
|
run: echo "PIP_CACHE_DIR=$(pip cache dir)" >> $GITHUB_ENV
|
||||||
|
- name: extract pip cache
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ${{ env.PIP_CACHE_DIR }}
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
||||||
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
|
- id: os-name
|
||||||
|
uses: ASzc/change-string-case-action@v5
|
||||||
|
with:
|
||||||
|
string: ${{ runner.os }}
|
||||||
|
- run: python -m pip install --user --upgrade pip wheel
|
||||||
|
- if: startsWith(runner.os, 'linux')
|
||||||
|
run: pip install -e .[test]
|
||||||
|
- if: startsWith(runner.os, 'linux')
|
||||||
|
env:
|
||||||
|
HOME: /tmp
|
||||||
|
run: make test-unit-coverage
|
||||||
|
- if: startsWith(runner.os, 'linux') != true
|
||||||
|
run: pip install -e .[test]
|
||||||
|
- if: startsWith(runner.os, 'linux') != true
|
||||||
|
env:
|
||||||
|
HOME: /tmp
|
||||||
|
run: coverage run --source=lbry -m unittest tests/unit/test_conf.py
|
||||||
|
- name: submit coverage report
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
COVERALLS_FLAG_NAME: tests-unit-${{ steps.os-name.outputs.lowercase }}
|
||||||
|
COVERALLS_PARALLEL: true
|
||||||
|
run: |
|
||||||
|
pip install coveralls
|
||||||
|
coveralls --service=github
|
||||||
|
|
||||||
|
tests-integration:
|
||||||
|
name: "tests / integration"
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
test:
|
||||||
|
- datanetwork
|
||||||
|
- blockchain
|
||||||
|
- claims
|
||||||
|
- takeovers
|
||||||
|
- transactions
|
||||||
|
- other
|
||||||
|
steps:
|
||||||
|
- name: Configure sysctl limits
|
||||||
|
run: |
|
||||||
|
sudo swapoff -a
|
||||||
|
sudo sysctl -w vm.swappiness=1
|
||||||
|
sudo sysctl -w fs.file-max=262144
|
||||||
|
sudo sysctl -w vm.max_map_count=262144
|
||||||
|
- name: Runs Elasticsearch
|
||||||
|
uses: elastic/elastic-github-actions/elasticsearch@master
|
||||||
|
with:
|
||||||
|
stack-version: 7.12.1
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.9'
|
||||||
|
- if: matrix.test == 'other'
|
||||||
|
run: |
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y --no-install-recommends ffmpeg
|
||||||
|
- name: extract pip cache
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ./.tox
|
||||||
|
key: tox-integration-${{ matrix.test }}-${{ hashFiles('setup.py') }}
|
||||||
|
restore-keys: txo-integration-${{ matrix.test }}-
|
||||||
|
- run: pip install tox coverage coveralls
|
||||||
|
- if: matrix.test == 'claims'
|
||||||
|
run: rm -rf .tox
|
||||||
|
- run: tox -e ${{ matrix.test }}
|
||||||
|
- name: submit coverage report
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
COVERALLS_FLAG_NAME: tests-integration-${{ matrix.test }}
|
||||||
|
COVERALLS_PARALLEL: true
|
||||||
|
run: |
|
||||||
|
coverage combine tests
|
||||||
|
coveralls --service=github
|
||||||
|
|
||||||
|
|
||||||
|
coverage:
|
||||||
|
needs: ["tests-unit", "tests-integration"]
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
steps:
|
||||||
|
- name: finalize coverage report submission
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
run: |
|
||||||
|
pip install coveralls
|
||||||
|
coveralls --service=github --finish
|
||||||
|
|
||||||
|
build:
|
||||||
|
needs: ["lint", "tests-unit", "tests-integration"]
|
||||||
|
name: "build / binary"
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
os:
|
||||||
|
- ubuntu-20.04
|
||||||
|
- macos-latest
|
||||||
|
- windows-latest
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.9'
|
||||||
|
- id: os-name
|
||||||
|
uses: ASzc/change-string-case-action@v5
|
||||||
|
with:
|
||||||
|
string: ${{ runner.os }}
|
||||||
|
- name: set pip cache dir
|
||||||
|
shell: bash
|
||||||
|
run: echo "PIP_CACHE_DIR=$(pip cache dir)" >> $GITHUB_ENV
|
||||||
|
- name: extract pip cache
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ${{ env.PIP_CACHE_DIR }}
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
||||||
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
|
- run: pip install pyinstaller==4.6
|
||||||
|
- run: pip install -e .
|
||||||
|
- if: startsWith(github.ref, 'refs/tags/v')
|
||||||
|
run: python docker/set_build.py
|
||||||
|
- if: startsWith(runner.os, 'linux') || startsWith(runner.os, 'mac')
|
||||||
|
name: Build & Run (Unix)
|
||||||
|
run: |
|
||||||
|
pyinstaller --onefile --name lbrynet lbry/extras/cli.py
|
||||||
|
dist/lbrynet --version
|
||||||
|
- if: startsWith(runner.os, 'windows')
|
||||||
|
name: Build & Run (Windows)
|
||||||
|
run: |
|
||||||
|
pip install pywin32==301
|
||||||
|
pyinstaller --additional-hooks-dir=scripts/. --icon=icons/lbry256.ico --onefile --name lbrynet lbry/extras/cli.py
|
||||||
|
dist/lbrynet.exe --version
|
||||||
|
- uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: lbrynet-${{ steps.os-name.outputs.lowercase }}
|
||||||
|
path: dist/
|
||||||
|
|
||||||
|
release:
|
||||||
|
name: "release"
|
||||||
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
|
needs: ["build"]
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v1
|
||||||
|
- uses: actions/download-artifact@v2
|
||||||
|
- name: upload binaries
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.RELEASE_API_TOKEN }}
|
||||||
|
run: |
|
||||||
|
pip install githubrelease
|
||||||
|
chmod +x lbrynet-macos/lbrynet
|
||||||
|
chmod +x lbrynet-linux/lbrynet
|
||||||
|
zip --junk-paths lbrynet-mac.zip lbrynet-macos/lbrynet
|
||||||
|
zip --junk-paths lbrynet-linux.zip lbrynet-linux/lbrynet
|
||||||
|
zip --junk-paths lbrynet-windows.zip lbrynet-windows/lbrynet.exe
|
||||||
|
ls -lh
|
||||||
|
githubrelease release lbryio/lbry-sdk info ${GITHUB_REF#refs/tags/}
|
||||||
|
githubrelease asset lbryio/lbry-sdk upload ${GITHUB_REF#refs/tags/} \
|
||||||
|
lbrynet-mac.zip lbrynet-linux.zip lbrynet-windows.zip
|
||||||
|
githubrelease release lbryio/lbry-sdk publish ${GITHUB_REF#refs/tags/}
|
||||||
|
|
22
.github/workflows/release.yml
vendored
Normal file
22
.github/workflows/release.yml
vendored
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
name: slack
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
release:
|
||||||
|
name: "slack notification"
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
steps:
|
||||||
|
- uses: LoveToKnow/slackify-markdown-action@v1.0.0
|
||||||
|
id: markdown
|
||||||
|
with:
|
||||||
|
text: "There is a new SDK release: ${{github.event.release.html_url}}\n${{ github.event.release.body }}"
|
||||||
|
- uses: slackapi/slack-github-action@v1.14.0
|
||||||
|
env:
|
||||||
|
CHANGELOG: '<!channel> ${{ steps.markdown.outputs.text }}'
|
||||||
|
SLACK_WEBHOOK_URL: ${{ secrets.SLACK_RELEASE_BOT_WEBHOOK }}
|
||||||
|
with:
|
||||||
|
payload: '{"type": "mrkdwn", "text": ${{ toJSON(env.CHANGELOG) }} }'
|
||||||
|
|
19
.gitignore
vendored
19
.gitignore
vendored
|
@ -1,13 +1,22 @@
|
||||||
.DS_Store
|
/.idea
|
||||||
|
/.DS_Store
|
||||||
/build
|
/build
|
||||||
/dist
|
/dist
|
||||||
/.tox
|
/.tox
|
||||||
/.idea
|
/.coverage*
|
||||||
/.coverage
|
|
||||||
/lbry-venv
|
/lbry-venv
|
||||||
|
/venv
|
||||||
|
/lbry/blockchain
|
||||||
|
|
||||||
lbrynet.egg-info
|
lbry.egg-info
|
||||||
__pycache__
|
__pycache__
|
||||||
_trial_temp/
|
_trial_temp/
|
||||||
|
trending*.log
|
||||||
|
|
||||||
/tests/integration/files
|
/tests/integration/claims/files
|
||||||
|
/tests/.coverage.*
|
||||||
|
|
||||||
|
/lbry/wallet/bin
|
||||||
|
|
||||||
|
/.vscode
|
||||||
|
/.gitignore
|
||||||
|
|
445
.pylintrc
445
.pylintrc
|
@ -1,445 +0,0 @@
|
||||||
[MASTER]
|
|
||||||
|
|
||||||
# Specify a configuration file.
|
|
||||||
#rcfile=
|
|
||||||
|
|
||||||
# Python code to execute, usually for sys.path manipulation such as
|
|
||||||
# pygtk.require().
|
|
||||||
#init-hook=
|
|
||||||
|
|
||||||
# Add files or directories to the blacklist. They should be base names, not
|
|
||||||
# paths.
|
|
||||||
ignore=CVS,schema
|
|
||||||
|
|
||||||
# Add files or directories matching the regex patterns to the
|
|
||||||
# blacklist. The regex matches against base names, not paths.
|
|
||||||
# `\.#.*` - add emacs tmp files to the blacklist
|
|
||||||
ignore-patterns=\.#.*
|
|
||||||
|
|
||||||
# Pickle collected data for later comparisons.
|
|
||||||
persistent=yes
|
|
||||||
|
|
||||||
# List of plugins (as comma separated values of python modules names) to load,
|
|
||||||
# usually to register additional checkers.
|
|
||||||
load-plugins=
|
|
||||||
|
|
||||||
# Use multiple processes to speed up Pylint.
|
|
||||||
jobs=1
|
|
||||||
|
|
||||||
# Allow loading of arbitrary C extensions. Extensions are imported into the
|
|
||||||
# active Python interpreter and may run arbitrary code.
|
|
||||||
unsafe-load-any-extension=no
|
|
||||||
|
|
||||||
# A comma-separated list of package or module names from where C extensions may
|
|
||||||
# be loaded. Extensions are loading into the active Python interpreter and may
|
|
||||||
# run arbitrary code
|
|
||||||
# extension-pkg-whitelist=
|
|
||||||
|
|
||||||
# Allow optimization of some AST trees. This will activate a peephole AST
|
|
||||||
# optimizer, which will apply various small optimizations. For instance, it can
|
|
||||||
# be used to obtain the result of joining multiple strings with the addition
|
|
||||||
# operator. Joining a lot of strings can lead to a maximum recursion error in
|
|
||||||
# Pylint and this flag can prevent that. It has one side effect, the resulting
|
|
||||||
# AST will be different than the one from reality.
|
|
||||||
optimize-ast=no
|
|
||||||
|
|
||||||
|
|
||||||
[MESSAGES CONTROL]
|
|
||||||
|
|
||||||
# Only show warnings with the listed confidence levels. Leave empty to show
|
|
||||||
# all. Valid levels: HIGH, INFERENCE, INFERENCE_FAILURE, UNDEFINED
|
|
||||||
confidence=
|
|
||||||
|
|
||||||
# Enable the message, report, category or checker with the given id(s). You can
|
|
||||||
# either give multiple identifier separated by comma (,) or put this option
|
|
||||||
# multiple time (only on the command line, not in the configuration file where
|
|
||||||
# it should appear only once). See also the "--disable" option for examples.
|
|
||||||
#enable=
|
|
||||||
|
|
||||||
# Disable the message, report, category or checker with the given id(s). You
|
|
||||||
# can either give multiple identifiers separated by comma (,) or put this
|
|
||||||
# option multiple times (only on the command line, not in the configuration
|
|
||||||
# file where it should appear only once).You can also use "--disable=all" to
|
|
||||||
# disable everything first and then re-enable specific checks. For example, if
|
|
||||||
# you want to run only the similarities checker, you can use "--disable=all
|
|
||||||
# --enable=similarities". If you want to run only the classes checker, but have
|
|
||||||
# no Warning level messages displayed, use"--disable=all --enable=classes
|
|
||||||
# --disable=W"
|
|
||||||
disable=
|
|
||||||
anomalous-backslash-in-string,
|
|
||||||
arguments-differ,
|
|
||||||
attribute-defined-outside-init,
|
|
||||||
bad-continuation,
|
|
||||||
bare-except,
|
|
||||||
broad-except,
|
|
||||||
cell-var-from-loop,
|
|
||||||
consider-iterating-dictionary,
|
|
||||||
dangerous-default-value,
|
|
||||||
duplicate-code,
|
|
||||||
fixme,
|
|
||||||
global-statement,
|
|
||||||
inherit-non-class,
|
|
||||||
invalid-name,
|
|
||||||
len-as-condition,
|
|
||||||
locally-disabled,
|
|
||||||
logging-not-lazy,
|
|
||||||
missing-docstring,
|
|
||||||
no-else-return,
|
|
||||||
no-init,
|
|
||||||
no-member,
|
|
||||||
no-self-use,
|
|
||||||
protected-access,
|
|
||||||
redefined-builtin,
|
|
||||||
redefined-outer-name,
|
|
||||||
redefined-variable-type,
|
|
||||||
relative-import,
|
|
||||||
signature-differs,
|
|
||||||
super-init-not-called,
|
|
||||||
too-few-public-methods,
|
|
||||||
too-many-arguments,
|
|
||||||
too-many-branches,
|
|
||||||
too-many-instance-attributes,
|
|
||||||
too-many-lines,
|
|
||||||
too-many-locals,
|
|
||||||
too-many-nested-blocks,
|
|
||||||
too-many-public-methods,
|
|
||||||
too-many-return-statements,
|
|
||||||
too-many-statements,
|
|
||||||
trailing-newlines,
|
|
||||||
undefined-loop-variable,
|
|
||||||
ungrouped-imports,
|
|
||||||
unnecessary-lambda,
|
|
||||||
unused-argument,
|
|
||||||
unused-variable,
|
|
||||||
wildcard-import,
|
|
||||||
wrong-import-order,
|
|
||||||
wrong-import-position,
|
|
||||||
deprecated-lambda,
|
|
||||||
simplifiable-if-statement,
|
|
||||||
unidiomatic-typecheck,
|
|
||||||
global-at-module-level,
|
|
||||||
inconsistent-return-statements,
|
|
||||||
keyword-arg-before-vararg,
|
|
||||||
assignment-from-no-return,
|
|
||||||
useless-return,
|
|
||||||
assignment-from-none,
|
|
||||||
stop-iteration-return
|
|
||||||
|
|
||||||
|
|
||||||
[REPORTS]
|
|
||||||
|
|
||||||
# Set the output format. Available formats are text, parseable, colorized, msvs
|
|
||||||
# (visual studio) and html. You can also give a reporter class, eg
|
|
||||||
# mypackage.mymodule.MyReporterClass.
|
|
||||||
output-format=text
|
|
||||||
|
|
||||||
# Put messages in a separate file for each module / package specified on the
|
|
||||||
# command line instead of printing them on stdout. Reports (if any) will be
|
|
||||||
# written in a file name "pylint_global.[txt|html]".
|
|
||||||
files-output=no
|
|
||||||
|
|
||||||
# Tells whether to display a full report or only the messages
|
|
||||||
reports=no
|
|
||||||
|
|
||||||
# Python expression which should return a note less than 10 (10 is the highest
|
|
||||||
# note). You have access to the variables errors warning, statement which
|
|
||||||
# respectively contain the number of errors / warnings messages and the total
|
|
||||||
# number of statements analyzed. This is used by the global evaluation report
|
|
||||||
# (RP0004).
|
|
||||||
evaluation=10.0 - ((float(5 * error + warning + refactor + convention) / statement) * 10)
|
|
||||||
|
|
||||||
# Template used to display messages. This is a python new-style format string
|
|
||||||
# used to format the message information. See doc for all details
|
|
||||||
#msg-template=
|
|
||||||
|
|
||||||
|
|
||||||
[VARIABLES]
|
|
||||||
|
|
||||||
# Tells whether we should check for unused import in __init__ files.
|
|
||||||
init-import=no
|
|
||||||
|
|
||||||
# A regular expression matching the name of dummy variables (i.e. expectedly
|
|
||||||
# not used).
|
|
||||||
dummy-variables-rgx=_$|dummy
|
|
||||||
|
|
||||||
# List of additional names supposed to be defined in builtins. Remember that
|
|
||||||
# you should avoid to define new builtins when possible.
|
|
||||||
additional-builtins=
|
|
||||||
|
|
||||||
# List of strings which can identify a callback function by name. A callback
|
|
||||||
# name must start or end with one of those strings.
|
|
||||||
callbacks=cb_,_cb
|
|
||||||
|
|
||||||
|
|
||||||
[LOGGING]
|
|
||||||
|
|
||||||
# Logging modules to check that the string format arguments are in logging
|
|
||||||
# function parameter format
|
|
||||||
logging-modules=logging
|
|
||||||
|
|
||||||
|
|
||||||
[BASIC]
|
|
||||||
|
|
||||||
# List of builtins function names that should not be used, separated by a comma
|
|
||||||
bad-functions=map,filter,input
|
|
||||||
|
|
||||||
# Good variable names which should always be accepted, separated by a comma
|
|
||||||
# allow `d` as its used frequently for deferred callback chains
|
|
||||||
good-names=i,j,k,ex,Run,_,d
|
|
||||||
|
|
||||||
# Bad variable names which should always be refused, separated by a comma
|
|
||||||
bad-names=foo,bar,baz,toto,tutu,tata
|
|
||||||
|
|
||||||
# Colon-delimited sets of names that determine each other's naming style when
|
|
||||||
# the name regexes allow several styles.
|
|
||||||
name-group=
|
|
||||||
|
|
||||||
# Include a hint for the correct naming format with invalid-name
|
|
||||||
include-naming-hint=no
|
|
||||||
|
|
||||||
# Regular expression matching correct function names
|
|
||||||
function-rgx=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Naming hint for function names
|
|
||||||
function-name-hint=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Regular expression matching correct variable names
|
|
||||||
variable-rgx=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Naming hint for variable names
|
|
||||||
variable-name-hint=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Regular expression matching correct constant names
|
|
||||||
const-rgx=(([A-Z_][A-Z0-9_]*)|(__.*__))$
|
|
||||||
|
|
||||||
# Naming hint for constant names
|
|
||||||
const-name-hint=(([A-Z_][A-Z0-9_]*)|(__.*__))$
|
|
||||||
|
|
||||||
# Regular expression matching correct attribute names
|
|
||||||
attr-rgx=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Naming hint for attribute names
|
|
||||||
attr-name-hint=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Regular expression matching correct argument names
|
|
||||||
argument-rgx=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Naming hint for argument names
|
|
||||||
argument-name-hint=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Regular expression matching correct class attribute names
|
|
||||||
class-attribute-rgx=([A-Za-z_][A-Za-z0-9_]{2,30}|(__.*__))$
|
|
||||||
|
|
||||||
# Naming hint for class attribute names
|
|
||||||
class-attribute-name-hint=([A-Za-z_][A-Za-z0-9_]{2,30}|(__.*__))$
|
|
||||||
|
|
||||||
# Regular expression matching correct inline iteration names
|
|
||||||
inlinevar-rgx=[A-Za-z_][A-Za-z0-9_]*$
|
|
||||||
|
|
||||||
# Naming hint for inline iteration names
|
|
||||||
inlinevar-name-hint=[A-Za-z_][A-Za-z0-9_]*$
|
|
||||||
|
|
||||||
# Regular expression matching correct class names
|
|
||||||
class-rgx=[A-Z_][a-zA-Z0-9]+$
|
|
||||||
|
|
||||||
# Naming hint for class names
|
|
||||||
class-name-hint=[A-Z_][a-zA-Z0-9]+$
|
|
||||||
|
|
||||||
# Regular expression matching correct module names
|
|
||||||
module-rgx=(([a-z_][a-z0-9_]*)|([A-Z][a-zA-Z0-9]+))$
|
|
||||||
|
|
||||||
# Naming hint for module names
|
|
||||||
module-name-hint=(([a-z_][a-z0-9_]*)|([A-Z][a-zA-Z0-9]+))$
|
|
||||||
|
|
||||||
# Regular expression matching correct method names
|
|
||||||
method-rgx=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Naming hint for method names
|
|
||||||
method-name-hint=[a-z_][a-z0-9_]{2,30}$
|
|
||||||
|
|
||||||
# Regular expression which should only match function or class names that do
|
|
||||||
# not require a docstring.
|
|
||||||
no-docstring-rgx=^_
|
|
||||||
|
|
||||||
# Minimum line length for functions/classes that require docstrings, shorter
|
|
||||||
# ones are exempt.
|
|
||||||
docstring-min-length=-1
|
|
||||||
|
|
||||||
|
|
||||||
[ELIF]
|
|
||||||
|
|
||||||
# Maximum number of nested blocks for function / method body
|
|
||||||
max-nested-blocks=5
|
|
||||||
|
|
||||||
|
|
||||||
[SPELLING]
|
|
||||||
|
|
||||||
# Spelling dictionary name. Available dictionaries: none. To make it working
|
|
||||||
# install python-enchant package.
|
|
||||||
spelling-dict=
|
|
||||||
|
|
||||||
# List of comma separated words that should not be checked.
|
|
||||||
spelling-ignore-words=
|
|
||||||
|
|
||||||
# A path to a file that contains private dictionary; one word per line.
|
|
||||||
spelling-private-dict-file=
|
|
||||||
|
|
||||||
# Tells whether to store unknown words to indicated private dictionary in
|
|
||||||
# --spelling-private-dict-file option instead of raising a message.
|
|
||||||
spelling-store-unknown-words=no
|
|
||||||
|
|
||||||
|
|
||||||
[FORMAT]
|
|
||||||
|
|
||||||
# Maximum number of characters on a single line.
|
|
||||||
max-line-length=120
|
|
||||||
|
|
||||||
# Regexp for a line that is allowed to be longer than the limit.
|
|
||||||
ignore-long-lines=^\s*(# )?<?https?://\S+>?$
|
|
||||||
|
|
||||||
# Allow the body of an if to be on the same line as the test if there is no
|
|
||||||
# else.
|
|
||||||
single-line-if-stmt=no
|
|
||||||
|
|
||||||
# List of optional constructs for which whitespace checking is disabled. `dict-
|
|
||||||
# separator` is used to allow tabulation in dicts, etc.: {1 : 1,\n222: 2}.
|
|
||||||
# `trailing-comma` allows a space between comma and closing bracket: (a, ).
|
|
||||||
# `empty-line` allows space-only lines.
|
|
||||||
no-space-check=trailing-comma,dict-separator
|
|
||||||
|
|
||||||
# Maximum number of lines in a module
|
|
||||||
max-module-lines=1000
|
|
||||||
|
|
||||||
# String used as indentation unit. This is usually " " (4 spaces) or "\t" (1
|
|
||||||
# tab).
|
|
||||||
indent-string=' '
|
|
||||||
|
|
||||||
# Number of spaces of indent required inside a hanging or continued line.
|
|
||||||
indent-after-paren=4
|
|
||||||
|
|
||||||
# Expected format of line ending, e.g. empty (any line ending), LF or CRLF.
|
|
||||||
expected-line-ending-format=
|
|
||||||
|
|
||||||
|
|
||||||
[MISCELLANEOUS]
|
|
||||||
|
|
||||||
# List of note tags to take in consideration, separated by a comma.
|
|
||||||
notes=FIXME,XXX,TODO
|
|
||||||
|
|
||||||
|
|
||||||
[SIMILARITIES]
|
|
||||||
|
|
||||||
# Minimum lines number of a similarity.
|
|
||||||
min-similarity-lines=4
|
|
||||||
|
|
||||||
# Ignore comments when computing similarities.
|
|
||||||
ignore-comments=yes
|
|
||||||
|
|
||||||
# Ignore docstrings when computing similarities.
|
|
||||||
ignore-docstrings=yes
|
|
||||||
|
|
||||||
# Ignore imports when computing similarities.
|
|
||||||
ignore-imports=no
|
|
||||||
|
|
||||||
|
|
||||||
[TYPECHECK]
|
|
||||||
|
|
||||||
# Tells whether missing members accessed in mixin class should be ignored. A
|
|
||||||
# mixin class is detected if its name ends with "mixin" (case insensitive).
|
|
||||||
ignore-mixin-members=yes
|
|
||||||
|
|
||||||
# List of module names for which member attributes should not be checked
|
|
||||||
# (useful for modules/projects where namespaces are manipulated during runtime
|
|
||||||
# and thus existing member attributes cannot be deduced by static analysis. It
|
|
||||||
# supports qualified module names, as well as Unix pattern matching.
|
|
||||||
ignored-modules=leveldb,distutils
|
|
||||||
# Ignoring distutils because: https://github.com/PyCQA/pylint/issues/73
|
|
||||||
|
|
||||||
# List of classes names for which member attributes should not be checked
|
|
||||||
# (useful for classes with attributes dynamically set). This supports can work
|
|
||||||
# with qualified names.
|
|
||||||
# ignored-classes=
|
|
||||||
|
|
||||||
|
|
||||||
# List of members which are set dynamically and missed by pylint inference
|
|
||||||
# system, and so shouldn't trigger E1101 when accessed. Python regular
|
|
||||||
# expressions are accepted.
|
|
||||||
generated-members=lbrynet.lbrynet_daemon.LBRYDaemon.Parameters
|
|
||||||
|
|
||||||
|
|
||||||
[IMPORTS]
|
|
||||||
|
|
||||||
# Deprecated modules which should not be used, separated by a comma
|
|
||||||
deprecated-modules=regsub,TERMIOS,Bastion,rexec
|
|
||||||
|
|
||||||
# Create a graph of every (i.e. internal and external) dependencies in the
|
|
||||||
# given file (report RP0402 must not be disabled)
|
|
||||||
import-graph=
|
|
||||||
|
|
||||||
# Create a graph of external dependencies in the given file (report RP0402 must
|
|
||||||
# not be disabled)
|
|
||||||
ext-import-graph=
|
|
||||||
|
|
||||||
# Create a graph of internal dependencies in the given file (report RP0402 must
|
|
||||||
# not be disabled)
|
|
||||||
int-import-graph=
|
|
||||||
|
|
||||||
|
|
||||||
[DESIGN]
|
|
||||||
|
|
||||||
# Maximum number of arguments for function / method
|
|
||||||
max-args=10
|
|
||||||
|
|
||||||
# Argument names that match this expression will be ignored. Default to name
|
|
||||||
# with leading underscore
|
|
||||||
ignored-argument-names=_.*
|
|
||||||
|
|
||||||
# Maximum number of locals for function / method body
|
|
||||||
max-locals=15
|
|
||||||
|
|
||||||
# Maximum number of return / yield for function / method body
|
|
||||||
max-returns=6
|
|
||||||
|
|
||||||
# Maximum number of branch for function / method body
|
|
||||||
max-branches=12
|
|
||||||
|
|
||||||
# Maximum number of statements in function / method body
|
|
||||||
max-statements=50
|
|
||||||
|
|
||||||
# Maximum number of parents for a class (see R0901).
|
|
||||||
max-parents=8
|
|
||||||
|
|
||||||
# Maximum number of attributes for a class (see R0902).
|
|
||||||
max-attributes=7
|
|
||||||
|
|
||||||
# Minimum number of public methods for a class (see R0903).
|
|
||||||
min-public-methods=2
|
|
||||||
|
|
||||||
# Maximum number of public methods for a class (see R0904).
|
|
||||||
max-public-methods=20
|
|
||||||
|
|
||||||
# Maximum number of boolean expressions in a if statement
|
|
||||||
max-bool-expr=5
|
|
||||||
|
|
||||||
|
|
||||||
[CLASSES]
|
|
||||||
|
|
||||||
# List of method names used to declare (i.e. assign) instance attributes.
|
|
||||||
defining-attr-methods=__init__,__new__,setUp
|
|
||||||
|
|
||||||
# List of valid names for the first argument in a class method.
|
|
||||||
valid-classmethod-first-arg=cls
|
|
||||||
|
|
||||||
# List of valid names for the first argument in a metaclass class method.
|
|
||||||
valid-metaclass-classmethod-first-arg=mcs
|
|
||||||
|
|
||||||
# List of member names, which should be excluded from the protected access
|
|
||||||
# warning.
|
|
||||||
exclude-protected=_asdict,_fields,_replace,_source,_make
|
|
||||||
|
|
||||||
|
|
||||||
[EXCEPTIONS]
|
|
||||||
|
|
||||||
# Exceptions that will emit a warning when being caught. Defaults to
|
|
||||||
# "Exception"
|
|
||||||
overgeneral-exceptions=Exception
|
|
145
.travis.yml
145
.travis.yml
|
@ -1,145 +0,0 @@
|
||||||
sudo: required
|
|
||||||
dist: xenial
|
|
||||||
language: python
|
|
||||||
python: "3.7"
|
|
||||||
|
|
||||||
env:
|
|
||||||
global:
|
|
||||||
# must also be updated in wine_build.sh
|
|
||||||
- TORBA=master
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
include:
|
|
||||||
|
|
||||||
- stage: code quality
|
|
||||||
name: "pylint lbrynet"
|
|
||||||
install:
|
|
||||||
- pip install astroid pylint
|
|
||||||
- pip install git+https://github.com/lbryio/torba.git@${TORBA}#egg=torba
|
|
||||||
- pip install -e .
|
|
||||||
script: pylint lbrynet
|
|
||||||
|
|
||||||
- stage: test
|
|
||||||
name: "Unit Tests"
|
|
||||||
install:
|
|
||||||
- pip install coverage
|
|
||||||
- pip install git+https://github.com/lbryio/torba.git@${TORBA}#egg=torba
|
|
||||||
- pip install -e .
|
|
||||||
script:
|
|
||||||
- HOME=/tmp coverage run -p --source=lbrynet -m unittest discover -vv tests.unit
|
|
||||||
after_success:
|
|
||||||
- coverage combine
|
|
||||||
- bash <(curl -s https://codecov.io/bash)
|
|
||||||
|
|
||||||
- name: "Integration Tests"
|
|
||||||
install:
|
|
||||||
- pip install tox-travis coverage
|
|
||||||
- pushd .. && git clone --single-branch --branch ${TORBA} https://github.com/lbryio/torba.git && popd
|
|
||||||
|
|
||||||
script: tox
|
|
||||||
after_success:
|
|
||||||
- coverage combine tests/
|
|
||||||
- bash <(curl -s https://codecov.io/bash)
|
|
||||||
|
|
||||||
- name: "Run Examples"
|
|
||||||
install:
|
|
||||||
- pip install coverage
|
|
||||||
- pip install git+https://github.com/lbryio/torba.git@${TORBA}#egg=torba
|
|
||||||
- pip install -e .
|
|
||||||
script:
|
|
||||||
- HOME=/tmp coverage run -p --source=lbrynet scripts/generate_json_api.py
|
|
||||||
after_success:
|
|
||||||
- coverage combine
|
|
||||||
- bash <(curl -s https://codecov.io/bash)
|
|
||||||
|
|
||||||
- stage: build
|
|
||||||
name: "Windows"
|
|
||||||
language: generic
|
|
||||||
services:
|
|
||||||
- docker
|
|
||||||
install:
|
|
||||||
- docker pull lbry/pyinstaller34_32bits:py371
|
|
||||||
script:
|
|
||||||
- python scripts/set_build.py
|
|
||||||
- docker run -v "$(pwd):/src/lbry" lbry/pyinstaller34_32bits:py371 lbry/scripts/wine_build.sh
|
|
||||||
- sudo zip -j dist/lbrynet-windows.zip dist/lbrynet.exe
|
|
||||||
deploy:
|
|
||||||
provider: releases
|
|
||||||
api_key: $GITHUB_OAUTH_TOKEN
|
|
||||||
file: dist/lbrynet-windows.zip
|
|
||||||
skip_cleanup: true
|
|
||||||
overwrite: true
|
|
||||||
draft: true
|
|
||||||
on:
|
|
||||||
tags: true
|
|
||||||
addons:
|
|
||||||
artifacts:
|
|
||||||
working_dir: dist
|
|
||||||
paths:
|
|
||||||
- lbrynet-windows.zip
|
|
||||||
target_paths:
|
|
||||||
- /daemon/build-${TRAVIS_BUILD_NUMBER}_commit-${TRAVIS_COMMIT:0:7}_branch-${TRAVIS_BRANCH}$([ ! -z ${TRAVIS_TAG} ] && echo _tag-${TRAVIS_TAG})
|
|
||||||
|
|
||||||
- &build
|
|
||||||
name: "Linux"
|
|
||||||
env: OS=linux
|
|
||||||
install:
|
|
||||||
- pip3 install pyinstaller
|
|
||||||
- pip3 install git+https://github.com/lbryio/torba.git@${TORBA}#egg=torba
|
|
||||||
- python3 scripts/set_build.py
|
|
||||||
- pip3 install -e .
|
|
||||||
script:
|
|
||||||
- pyinstaller -F -n lbrynet lbrynet/extras/cli.py
|
|
||||||
- chmod +x dist/lbrynet
|
|
||||||
- zip -j dist/lbrynet-${OS}.zip dist/lbrynet
|
|
||||||
- ./dist/lbrynet --version
|
|
||||||
deploy:
|
|
||||||
provider: releases
|
|
||||||
api_key: $GITHUB_OAUTH_TOKEN
|
|
||||||
file: dist/lbrynet-${OS}.zip
|
|
||||||
skip_cleanup: true
|
|
||||||
overwrite: true
|
|
||||||
draft: true
|
|
||||||
on:
|
|
||||||
tags: true
|
|
||||||
addons:
|
|
||||||
artifacts:
|
|
||||||
working_dir: dist
|
|
||||||
paths:
|
|
||||||
- lbrynet-${OS}.zip
|
|
||||||
# artifact uploader thinks lbrynet is a directory, https://github.com/travis-ci/artifacts/issues/78
|
|
||||||
target_paths:
|
|
||||||
- /daemon/build-${TRAVIS_BUILD_NUMBER}_commit-${TRAVIS_COMMIT:0:7}_branch-${TRAVIS_BRANCH}$([ ! -z ${TRAVIS_TAG} ] && echo _tag-${TRAVIS_TAG})
|
|
||||||
|
|
||||||
- <<: *build
|
|
||||||
name: "Mac"
|
|
||||||
os: osx
|
|
||||||
osx_image: xcode8.3
|
|
||||||
language: generic
|
|
||||||
env: OS=mac
|
|
||||||
cache: false
|
|
||||||
before_install:
|
|
||||||
- brew upgrade python || true
|
|
||||||
- brew upgrade python || true
|
|
||||||
- if: tag IS present
|
|
||||||
stage: build
|
|
||||||
name: "Wallet Server Docker Image - Tagged Release"
|
|
||||||
script:
|
|
||||||
- set -e
|
|
||||||
- echo "$DOCKER_PASSWORD" | docker login --username "$DOCKER_USERNAME" --password-stdin
|
|
||||||
- travis_retry docker build -t lbry/wallet-server:$TRAVIS_TAG -f scripts/Dockerfile.wallet_server --build-arg TORBA_VERSION=${TORBA} .
|
|
||||||
- docker push lbry/wallet-server:$TRAVIS_TAG
|
|
||||||
- if: tag IS blank AND branch = master AND NOT type IN (pull_request)
|
|
||||||
stage: build
|
|
||||||
name: "Wallet Server Docker Image - Master"
|
|
||||||
script:
|
|
||||||
- set -e
|
|
||||||
- echo "$DOCKER_PASSWORD" | docker login --username "$DOCKER_USERNAME" --password-stdin
|
|
||||||
- travis_retry docker build -t lbry/wallet-server:master -f scripts/Dockerfile.wallet_server --build-arg TORBA_VERSION=${TORBA} .
|
|
||||||
- docker push lbry/wallet-server:master
|
|
||||||
|
|
||||||
cache:
|
|
||||||
directories:
|
|
||||||
- $HOME/.cache/pip
|
|
||||||
- $HOME/Library/Caches/pip
|
|
||||||
- $TRAVIS_BUILD_DIR/.tox
|
|
1019
CHANGELOG.md
1019
CHANGELOG.md
File diff suppressed because it is too large
Load diff
|
@ -1,6 +0,0 @@
|
||||||
# Add a CHANGELOG entry for app changes
|
|
||||||
has_app_changes = !git.modified_files.grep(/lbrynet/).empty?
|
|
||||||
if !git.modified_files.include?("CHANGELOG.md") && has_app_changes
|
|
||||||
fail("Please include a CHANGELOG entry.")
|
|
||||||
message "See http://keepachangelog.com/en/0.3.0/ for details on good changelog guidelines"
|
|
||||||
end
|
|
163
INSTALL.md
163
INSTALL.md
|
@ -1,6 +1,6 @@
|
||||||
# Installing LBRY
|
# Installing LBRY
|
||||||
|
|
||||||
If only the JSON-RPC API server is needed, the recommended way to install LBRY is to use a pre-built binary. We provide binaries for all major operating systems. See the [README](README.md).
|
If only the JSON-RPC API server is needed, the recommended way to install LBRY is to use a pre-built binary. We provide binaries for all major operating systems. See the [README](README.md)!
|
||||||
|
|
||||||
These instructions are for installing LBRY from source, which is recommended if you are interested in doing development work or LBRY is not available on your operating system (godspeed, TempleOS users).
|
These instructions are for installing LBRY from source, which is recommended if you are interested in doing development work or LBRY is not available on your operating system (godspeed, TempleOS users).
|
||||||
|
|
||||||
|
@ -9,33 +9,47 @@ Here's a video walkthrough of this setup, which is itself hosted by the LBRY net
|
||||||
|
|
||||||
## Prerequisites
|
## Prerequisites
|
||||||
|
|
||||||
Running `lbrynet` from source requires Python 3.7 or higher. Get the installer for your OS [here](https://www.python.org/downloads/release/python-370/)
|
Running `lbrynet` from source requires Python 3.7. Get the installer for your OS [here](https://www.python.org/downloads/release/python-370/).
|
||||||
|
|
||||||
After installing python 3, you'll need to install some additional libraries depending on your operating system.
|
After installing Python 3.7, you'll need to install some additional libraries depending on your operating system.
|
||||||
|
|
||||||
|
Because of [issue #2769](https://github.com/lbryio/lbry-sdk/issues/2769)
|
||||||
|
at the moment the `lbrynet` daemon will only work correctly with Python 3.7.
|
||||||
|
If Python 3.8+ is used, the daemon will start but the RPC server
|
||||||
|
may not accept messages, returning the following:
|
||||||
|
```
|
||||||
|
Could not connect to daemon. Are you sure it's running?
|
||||||
|
```
|
||||||
|
|
||||||
### macOS
|
### macOS
|
||||||
|
|
||||||
macOS users will need to install [xcode command line tools](https://developer.xamarin.com/guides/testcloud/calabash/configuring/osx/install-xcode-command-line-tools/) and [homebrew](http://brew.sh/).
|
macOS users will need to install [xcode command line tools](https://developer.xamarin.com/guides/testcloud/calabash/configuring/osx/install-xcode-command-line-tools/) and [homebrew](http://brew.sh/).
|
||||||
|
|
||||||
These environment variables also need to be set
|
These environment variables also need to be set:
|
||||||
1. PYTHONUNBUFFERED=1
|
```
|
||||||
2. EVENT_NOKQUEUE=1
|
PYTHONUNBUFFERED=1
|
||||||
|
EVENT_NOKQUEUE=1
|
||||||
|
```
|
||||||
|
|
||||||
Remaining dependencies can then be installed by running:
|
Remaining dependencies can then be installed by running:
|
||||||
|
|
||||||
```
|
```
|
||||||
brew install python protobuf
|
brew install python protobuf
|
||||||
```
|
```
|
||||||
|
|
||||||
Assistance installing Python3: https://docs.python-guide.org/starting/install3/osx/
|
Assistance installing Python3: https://docs.python-guide.org/starting/install3/osx/.
|
||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
|
|
||||||
On Ubuntu (we recommend 18.04), install the following:
|
On Ubuntu (we recommend 18.04 or 20.04), install the following:
|
||||||
|
```
|
||||||
|
sudo add-apt-repository ppa:deadsnakes/ppa
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install build-essential python3.7 python3.7-dev git python3.7-venv libssl-dev python-protobuf
|
||||||
|
```
|
||||||
|
|
||||||
```
|
The [deadsnakes PPA](https://launchpad.net/~deadsnakes/+archive/ubuntu/ppa) provides Python 3.7
|
||||||
sudo apt-get install build-essential python3.7 python3.7-dev git python3-venv libssl-dev python-protobuf
|
for those Ubuntu distributions that no longer have it in their
|
||||||
```
|
official repositories.
|
||||||
|
|
||||||
On Raspbian, you will also need to install `python-pyparsing`.
|
On Raspbian, you will also need to install `python-pyparsing`.
|
||||||
|
|
||||||
|
@ -43,38 +57,121 @@ If you're running another Linux distro, install the equivalent of the above pack
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
To install:
|
### Linux/Mac
|
||||||
|
|
||||||
```
|
Clone the repository:
|
||||||
git clone https://github.com/lbryio/lbry.git
|
```bash
|
||||||
cd lbry
|
git clone https://github.com/lbryio/lbry-sdk.git
|
||||||
|
cd lbry-sdk
|
||||||
|
```
|
||||||
|
|
||||||
Creating venv:
|
Create a Python virtual environment for lbry-sdk:
|
||||||
python -m venv lbry-venv
|
```bash
|
||||||
|
python3.7 -m venv lbry-venv
|
||||||
Activating lbry-venv on Linux/Mac:
|
```
|
||||||
source lbry-venv/bin/activate
|
|
||||||
|
|
||||||
Activating lbry-venv on Windows:
|
|
||||||
lbry-venv\Scripts\activate
|
|
||||||
|
|
||||||
python --version # Python 2 is not supported. Make sure you're on Python 3.7
|
Activate virtual environment:
|
||||||
|
```bash
|
||||||
|
source lbry-venv/bin/activate
|
||||||
|
```
|
||||||
|
|
||||||
pip install -e .
|
Make sure you're on Python 3.7+ as default in the virtual environment:
|
||||||
```
|
```bash
|
||||||
|
python --version
|
||||||
|
```
|
||||||
|
|
||||||
To verify your installation, `which lbrynet` should return a path inside of the `lbry-venv` folder created by the `virtualenv` command.
|
Install packages:
|
||||||
|
```bash
|
||||||
|
make install
|
||||||
|
```
|
||||||
|
|
||||||
|
If you are on Linux and using PyCharm, generates initial configs:
|
||||||
|
```bash
|
||||||
|
make idea
|
||||||
|
```
|
||||||
|
|
||||||
|
To verify your installation, `which lbrynet` should return a path inside
|
||||||
|
of the `lbry-venv` folder.
|
||||||
|
```bash
|
||||||
|
(lbry-venv) $ which lbrynet
|
||||||
|
/opt/lbry-sdk/lbry-venv/bin/lbrynet
|
||||||
|
```
|
||||||
|
|
||||||
|
To exit the virtual environment simply use the command `deactivate`.
|
||||||
|
|
||||||
|
### Windows
|
||||||
|
|
||||||
|
Clone the repository:
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/lbryio/lbry-sdk.git
|
||||||
|
cd lbry-sdk
|
||||||
|
```
|
||||||
|
|
||||||
|
Create a Python virtual environment for lbry-sdk:
|
||||||
|
```bash
|
||||||
|
python -m venv lbry-venv
|
||||||
|
```
|
||||||
|
|
||||||
|
Activate virtual environment:
|
||||||
|
```bash
|
||||||
|
lbry-venv\Scripts\activate
|
||||||
|
```
|
||||||
|
|
||||||
|
Install packages:
|
||||||
|
```bash
|
||||||
|
pip install -e .
|
||||||
|
```
|
||||||
|
|
||||||
## Run the tests
|
## Run the tests
|
||||||
To run the unit tests from the repo directory:
|
### Elasticsearch
|
||||||
```
|
|
||||||
trial --reactor=asyncio tests.unit
|
For running integration tests, Elasticsearch is required to be available at localhost:9200/
|
||||||
```
|
|
||||||
|
The easiest way to start it is using docker with:
|
||||||
|
```bash
|
||||||
|
make elastic-docker
|
||||||
|
```
|
||||||
|
|
||||||
|
Alternative installation methods are available [at Elasticsearch website](https://www.elastic.co/guide/en/elasticsearch/reference/current/install-elasticsearch.html).
|
||||||
|
|
||||||
|
To run the unit and integration tests from the repo directory:
|
||||||
|
```
|
||||||
|
python -m unittest discover tests.unit
|
||||||
|
python -m unittest discover tests.integration
|
||||||
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
To start the API server:
|
To start the API server:
|
||||||
`lbrynet start`
|
```
|
||||||
|
lbrynet start
|
||||||
|
```
|
||||||
|
|
||||||
|
Whenever the code inside [lbry-sdk/lbry](./lbry)
|
||||||
|
is modified we should run `make install` to recompile the `lbrynet`
|
||||||
|
executable with the newest code.
|
||||||
|
|
||||||
|
## Development
|
||||||
|
|
||||||
|
When developing, remember to enter the environment,
|
||||||
|
and if you wish start the server interactively.
|
||||||
|
```bash
|
||||||
|
$ source lbry-venv/bin/activate
|
||||||
|
|
||||||
|
(lbry-venv) $ python lbry/extras/cli.py start
|
||||||
|
```
|
||||||
|
|
||||||
|
Parameters can be passed in the same way.
|
||||||
|
```bash
|
||||||
|
(lbry-venv) $ python lbry/extras/cli.py wallet balance
|
||||||
|
```
|
||||||
|
|
||||||
|
If a Python debugger (`pdb` or `ipdb`) is installed we can also start it
|
||||||
|
in this way, set up break points, and step through the code.
|
||||||
|
```bash
|
||||||
|
(lbry-venv) $ pip install ipdb
|
||||||
|
|
||||||
|
(lbry-venv) $ ipdb lbry/extras/cli.py
|
||||||
|
```
|
||||||
|
|
||||||
Happy hacking!
|
Happy hacking!
|
||||||
|
|
2
LICENSE
2
LICENSE
|
@ -1,6 +1,6 @@
|
||||||
The MIT License (MIT)
|
The MIT License (MIT)
|
||||||
|
|
||||||
Copyright (c) 2015-2019 LBRY Inc
|
Copyright (c) 2015-2022 LBRY Inc
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the
|
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,
|
"Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
|
4
MANIFEST.in
Normal file
4
MANIFEST.in
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
include README.md
|
||||||
|
include CHANGELOG.md
|
||||||
|
include LICENSE
|
||||||
|
recursive-include lbry *.txt *.py
|
26
Makefile
Normal file
26
Makefile
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
.PHONY: install tools lint test test-unit test-unit-coverage test-integration idea
|
||||||
|
|
||||||
|
install:
|
||||||
|
pip install -e .
|
||||||
|
|
||||||
|
lint:
|
||||||
|
pylint --rcfile=setup.cfg lbry
|
||||||
|
#mypy --ignore-missing-imports lbry
|
||||||
|
|
||||||
|
test: test-unit test-integration
|
||||||
|
|
||||||
|
test-unit:
|
||||||
|
python -m unittest discover tests.unit
|
||||||
|
|
||||||
|
test-unit-coverage:
|
||||||
|
coverage run --source=lbry -m unittest discover -vv tests.unit
|
||||||
|
|
||||||
|
test-integration:
|
||||||
|
tox
|
||||||
|
|
||||||
|
idea:
|
||||||
|
mkdir -p .idea
|
||||||
|
cp -r scripts/idea/* .idea
|
||||||
|
|
||||||
|
elastic-docker:
|
||||||
|
docker run -d -v lbryhub:/usr/share/elasticsearch/data -p 9200:9200 -p 9300:9300 -e"ES_JAVA_OPTS=-Xms512m -Xmx512m" -e "discovery.type=single-node" docker.elastic.co/elasticsearch/elasticsearch:7.12.1
|
22
README.md
22
README.md
|
@ -1,19 +1,19 @@
|
||||||
# <img src="https://raw.githubusercontent.com/lbryio/lbry/master/lbry.png" alt="LBRY" width="48" height="36" /> LBRY SDK [![Build Status](https://travis-ci.org/lbryio/lbry.svg?branch=master)](https://travis-ci.org/lbryio/lbry) [![Test Coverage](https://codecov.io/gh/lbryio/lbry/branch/master/graph/badge.svg)](https://codecov.io/gh/lbryio/lbry)
|
# <img src="https://raw.githubusercontent.com/lbryio/lbry-sdk/master/lbry.png" alt="LBRY" width="48" height="36" /> LBRY SDK [![build](https://github.com/lbryio/lbry-sdk/actions/workflows/main.yml/badge.svg)](https://github.com/lbryio/lbry-sdk/actions/workflows/main.yml) [![coverage](https://coveralls.io/repos/github/lbryio/lbry-sdk/badge.svg)](https://coveralls.io/github/lbryio/lbry-sdk)
|
||||||
|
|
||||||
LBRY is a decentralized peer-to-peer protocol for publishing and accessing digital content. It utilizes the [LBRY blockchain](https://github.com/lbryio/lbrycrd) as a global namespace and database of digital content. Blockchain entries contain searchable content metadata, identities, rights and access rules. LBRY also provides a data network that consists of peers (seeders) uploading and downloading data from other peers, possibly in exchange for payments, as well as a distributed hash table used by peers to discover other peers.
|
LBRY is a decentralized peer-to-peer protocol for publishing and accessing digital content. It utilizes the [LBRY blockchain](https://github.com/lbryio/lbrycrd) as a global namespace and database of digital content. Blockchain entries contain searchable content metadata, identities, rights and access rules. LBRY also provides a data network that consists of peers (seeders) uploading and downloading data from other peers, possibly in exchange for payments, as well as a distributed hash table used by peers to discover other peers.
|
||||||
|
|
||||||
LBRY SDK for Python is currently the most full featured implementation of the LBRY Network protocols and includes many useful components and tools for building decentralized applications. Primary features and components:
|
LBRY SDK for Python is currently the most fully featured implementation of the LBRY Network protocols and includes many useful components and tools for building decentralized applications. Primary features and components include:
|
||||||
|
|
||||||
* Built on Python 3.7+ and `asyncio`.
|
* Built on Python 3.7 and `asyncio`.
|
||||||
* Kademlia DHT (Distributed Hash Table) implementation for finding peers to download from and announcing to peers what we have to host ([lbrynet.dht](https://github.com/lbryio/lbry/tree/master/lbrynet/dht)).
|
* Kademlia DHT (Distributed Hash Table) implementation for finding peers to download from and announcing to peers what we have to host ([lbry.dht](https://github.com/lbryio/lbry-sdk/tree/master/lbry/dht)).
|
||||||
* Blob exchange protocol for transferring encrypted blobs of content and negotiating payments ([lbrynet.blob_exchange](https://github.com/lbryio/lbry/tree/master/lbrynet/blob_exchange)).
|
* Blob exchange protocol for transferring encrypted blobs of content and negotiating payments ([lbry.blob_exchange](https://github.com/lbryio/lbry-sdk/tree/master/lbry/blob_exchange)).
|
||||||
* Protobuf schema for encoding and decoding metadata stored on the blockchain ([lbrynet.schema](https://github.com/lbryio/lbry/tree/master/lbrynet/schema)).
|
* Protobuf schema for encoding and decoding metadata stored on the blockchain ([lbry.schema](https://github.com/lbryio/lbry-sdk/tree/master/lbry/schema)).
|
||||||
* Wallet implementation for the LBRY blockchain ([lbrynet.wallet](https://github.com/lbryio/lbry/tree/master/lbrynet/wallet)).
|
* Wallet implementation for the LBRY blockchain ([lbry.wallet](https://github.com/lbryio/lbry-sdk/tree/master/lbry/wallet)).
|
||||||
* Daemon with a JSON-RPC API to ease building end user applications in any language and for automating various tasks ([lbrynet.extras.daemon](https://github.com/lbryio/lbry/tree/master/lbrynet/extras/daemon)).
|
* Daemon with a JSON-RPC API to ease building end user applications in any language and for automating various tasks ([lbry.extras.daemon](https://github.com/lbryio/lbry-sdk/tree/master/lbry/extras/daemon)).
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
Our [releases page](https://github.com/lbryio/lbry/releases) contains pre-built binaries of the latest release, pre-releases, and past releases for macOS, Debian-based Linux, and Windows. [Automated travis builds](http://build.lbry.io/daemon/) are also available for testing.
|
Our [releases page](https://github.com/lbryio/lbry-sdk/releases) contains pre-built binaries of the latest release, pre-releases, and past releases for macOS, Debian-based Linux, and Windows. [Automated travis builds](http://build.lbry.io/daemon/) are also available for testing.
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
|
@ -41,7 +41,7 @@ This project is MIT licensed. For the full license, see [LICENSE](LICENSE).
|
||||||
|
|
||||||
## Security
|
## Security
|
||||||
|
|
||||||
We take security seriously. Please contact security@lbry.com regarding any security issues. [Our GPG key is here](https://lbry.com/faq/gpg-key) if you need it.
|
We take security seriously. Please contact security@lbry.com regarding any security issues. [Our PGP key is here](https://lbry.com/faq/pgp-key) if you need it.
|
||||||
|
|
||||||
## Contact
|
## Contact
|
||||||
|
|
||||||
|
@ -53,4 +53,4 @@ The documentation for the API can be found [here](https://lbry.tech/api/sdk).
|
||||||
|
|
||||||
Daemon defaults, ports, and other settings are documented [here](https://lbry.tech/resources/daemon-settings).
|
Daemon defaults, ports, and other settings are documented [here](https://lbry.tech/resources/daemon-settings).
|
||||||
|
|
||||||
Settings can be configured using a daemon-settings.yml file. An example can be found [here](https://github.com/lbryio/lbry/blob/master/example_daemon_settings.yml).
|
Settings can be configured using a daemon-settings.yml file. An example can be found [here](https://github.com/lbryio/lbry-sdk/blob/master/example_daemon_settings.yml).
|
||||||
|
|
9
SECURITY.md
Normal file
9
SECURITY.md
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
# Security Policy
|
||||||
|
|
||||||
|
## Supported Versions
|
||||||
|
|
||||||
|
While we are not at v1.0 yet, only the latest release will be supported.
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
See https://lbry.com/faq/security
|
43
docker/Dockerfile.dht_node
Normal file
43
docker/Dockerfile.dht_node
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
FROM debian:10-slim
|
||||||
|
|
||||||
|
ARG user=lbry
|
||||||
|
ARG projects_dir=/home/$user
|
||||||
|
ARG db_dir=/database
|
||||||
|
|
||||||
|
ARG DOCKER_TAG
|
||||||
|
ARG DOCKER_COMMIT=docker
|
||||||
|
ENV DOCKER_TAG=$DOCKER_TAG DOCKER_COMMIT=$DOCKER_COMMIT
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
apt-get -y --no-install-recommends install \
|
||||||
|
wget \
|
||||||
|
automake libtool \
|
||||||
|
tar unzip \
|
||||||
|
build-essential \
|
||||||
|
pkg-config \
|
||||||
|
libleveldb-dev \
|
||||||
|
python3.7 \
|
||||||
|
python3-dev \
|
||||||
|
python3-pip \
|
||||||
|
python3-wheel \
|
||||||
|
python3-setuptools && \
|
||||||
|
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
||||||
|
rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
||||||
|
|
||||||
|
COPY . $projects_dir
|
||||||
|
RUN chown -R $user:$user $projects_dir
|
||||||
|
RUN mkdir -p $db_dir
|
||||||
|
RUN chown -R $user:$user $db_dir
|
||||||
|
|
||||||
|
USER $user
|
||||||
|
WORKDIR $projects_dir
|
||||||
|
|
||||||
|
RUN python3 -m pip install -U setuptools pip
|
||||||
|
RUN make install
|
||||||
|
RUN python3 docker/set_build.py
|
||||||
|
RUN rm ~/.cache -rf
|
||||||
|
VOLUME $db_dir
|
||||||
|
ENTRYPOINT ["python3", "scripts/dht_node.py"]
|
||||||
|
|
56
docker/Dockerfile.wallet_server
Normal file
56
docker/Dockerfile.wallet_server
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
FROM debian:10-slim
|
||||||
|
|
||||||
|
ARG user=lbry
|
||||||
|
ARG db_dir=/database
|
||||||
|
ARG projects_dir=/home/$user
|
||||||
|
|
||||||
|
ARG DOCKER_TAG
|
||||||
|
ARG DOCKER_COMMIT=docker
|
||||||
|
ENV DOCKER_TAG=$DOCKER_TAG DOCKER_COMMIT=$DOCKER_COMMIT
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
apt-get -y --no-install-recommends install \
|
||||||
|
wget \
|
||||||
|
tar unzip \
|
||||||
|
build-essential \
|
||||||
|
automake libtool \
|
||||||
|
pkg-config \
|
||||||
|
libleveldb-dev \
|
||||||
|
python3.7 \
|
||||||
|
python3-dev \
|
||||||
|
python3-pip \
|
||||||
|
python3-wheel \
|
||||||
|
python3-cffi \
|
||||||
|
python3-setuptools && \
|
||||||
|
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
||||||
|
rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
||||||
|
RUN mkdir -p $db_dir
|
||||||
|
RUN chown -R $user:$user $db_dir
|
||||||
|
|
||||||
|
COPY . $projects_dir
|
||||||
|
RUN chown -R $user:$user $projects_dir
|
||||||
|
|
||||||
|
USER $user
|
||||||
|
WORKDIR $projects_dir
|
||||||
|
|
||||||
|
RUN pip install uvloop
|
||||||
|
RUN make install
|
||||||
|
RUN python3 docker/set_build.py
|
||||||
|
RUN rm ~/.cache -rf
|
||||||
|
|
||||||
|
# entry point
|
||||||
|
ARG host=0.0.0.0
|
||||||
|
ARG tcp_port=50001
|
||||||
|
ARG daemon_url=http://lbry:lbry@localhost:9245/
|
||||||
|
VOLUME $db_dir
|
||||||
|
ENV TCP_PORT=$tcp_port
|
||||||
|
ENV HOST=$host
|
||||||
|
ENV DAEMON_URL=$daemon_url
|
||||||
|
ENV DB_DIRECTORY=$db_dir
|
||||||
|
ENV MAX_SESSIONS=1000000000
|
||||||
|
ENV MAX_SEND=1000000000000000000
|
||||||
|
ENV EVENT_LOOP_POLICY=uvloop
|
||||||
|
COPY ./docker/wallet_server_entrypoint.sh /entrypoint.sh
|
||||||
|
ENTRYPOINT ["/entrypoint.sh"]
|
45
docker/Dockerfile.web
Normal file
45
docker/Dockerfile.web
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
FROM debian:10-slim
|
||||||
|
|
||||||
|
ARG user=lbry
|
||||||
|
ARG downloads_dir=/database
|
||||||
|
ARG projects_dir=/home/$user
|
||||||
|
|
||||||
|
ARG DOCKER_TAG
|
||||||
|
ARG DOCKER_COMMIT=docker
|
||||||
|
ENV DOCKER_TAG=$DOCKER_TAG DOCKER_COMMIT=$DOCKER_COMMIT
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
apt-get -y --no-install-recommends install \
|
||||||
|
wget \
|
||||||
|
automake libtool \
|
||||||
|
tar unzip \
|
||||||
|
build-essential \
|
||||||
|
pkg-config \
|
||||||
|
libleveldb-dev \
|
||||||
|
python3.7 \
|
||||||
|
python3-dev \
|
||||||
|
python3-pip \
|
||||||
|
python3-wheel \
|
||||||
|
python3-setuptools && \
|
||||||
|
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
||||||
|
rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
||||||
|
RUN mkdir -p $downloads_dir
|
||||||
|
RUN chown -R $user:$user $downloads_dir
|
||||||
|
|
||||||
|
COPY . $projects_dir
|
||||||
|
RUN chown -R $user:$user $projects_dir
|
||||||
|
|
||||||
|
USER $user
|
||||||
|
WORKDIR $projects_dir
|
||||||
|
|
||||||
|
RUN pip install uvloop
|
||||||
|
RUN make install
|
||||||
|
RUN python3 docker/set_build.py
|
||||||
|
RUN rm ~/.cache -rf
|
||||||
|
|
||||||
|
# entry point
|
||||||
|
VOLUME $downloads_dir
|
||||||
|
COPY ./docker/webconf.yaml /webconf.yaml
|
||||||
|
ENTRYPOINT ["/home/lbry/.local/bin/lbrynet", "start", "--config=/webconf.yaml"]
|
9
docker/README.md
Normal file
9
docker/README.md
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
### How to run with docker-compose
|
||||||
|
1. Edit config file and after that fix permissions with
|
||||||
|
```
|
||||||
|
sudo chown -R 999:999 webconf.yaml
|
||||||
|
```
|
||||||
|
2. Start SDK with
|
||||||
|
```
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
49
docker/docker-compose-wallet-server.yml
Normal file
49
docker/docker-compose-wallet-server.yml
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
version: "3"
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
wallet_server:
|
||||||
|
es01:
|
||||||
|
|
||||||
|
services:
|
||||||
|
wallet_server:
|
||||||
|
depends_on:
|
||||||
|
- es01
|
||||||
|
image: lbry/wallet-server:${WALLET_SERVER_TAG:-latest-release}
|
||||||
|
restart: always
|
||||||
|
network_mode: host
|
||||||
|
ports:
|
||||||
|
- "50001:50001" # rpc port
|
||||||
|
- "2112:2112" # uncomment to enable prometheus
|
||||||
|
volumes:
|
||||||
|
- "wallet_server:/database"
|
||||||
|
environment:
|
||||||
|
- DAEMON_URL=http://lbry:lbry@127.0.0.1:9245
|
||||||
|
- MAX_QUERY_WORKERS=4
|
||||||
|
- CACHE_MB=1024
|
||||||
|
- CACHE_ALL_TX_HASHES=
|
||||||
|
- CACHE_ALL_CLAIM_TXOS=
|
||||||
|
- MAX_SEND=1000000000000000000
|
||||||
|
- MAX_RECEIVE=1000000000000000000
|
||||||
|
- MAX_SESSIONS=100000
|
||||||
|
- HOST=0.0.0.0
|
||||||
|
- TCP_PORT=50001
|
||||||
|
- PROMETHEUS_PORT=2112
|
||||||
|
- FILTERING_CHANNEL_IDS=770bd7ecba84fd2f7607fb15aedd2b172c2e153f 95e5db68a3101df19763f3a5182e4b12ba393ee8
|
||||||
|
- BLOCKING_CHANNEL_IDS=dd687b357950f6f271999971f43c785e8067c3a9 06871aa438032244202840ec59a469b303257cad b4a2528f436eca1bf3bf3e10ff3f98c57bd6c4c6
|
||||||
|
es01:
|
||||||
|
image: docker.elastic.co/elasticsearch/elasticsearch:7.11.0
|
||||||
|
container_name: es01
|
||||||
|
environment:
|
||||||
|
- node.name=es01
|
||||||
|
- discovery.type=single-node
|
||||||
|
- indices.query.bool.max_clause_count=8192
|
||||||
|
- bootstrap.memory_lock=true
|
||||||
|
- "ES_JAVA_OPTS=-Xms4g -Xmx4g" # no more than 32, remember to disable swap
|
||||||
|
ulimits:
|
||||||
|
memlock:
|
||||||
|
soft: -1
|
||||||
|
hard: -1
|
||||||
|
volumes:
|
||||||
|
- es01:/usr/share/elasticsearch/data
|
||||||
|
ports:
|
||||||
|
- 127.0.0.1:9200:9200
|
9
docker/docker-compose.yml
Normal file
9
docker/docker-compose.yml
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
websdk:
|
||||||
|
image: vshyba/websdk
|
||||||
|
ports:
|
||||||
|
- '5279:5279'
|
||||||
|
- '5280:5280'
|
||||||
|
volumes:
|
||||||
|
- ./webconf.yaml:/webconf.yaml
|
7
docker/hooks/build
Normal file
7
docker/hooks/build
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )"
|
||||||
|
cd "$DIR/../.." ## make sure we're in the right place. Docker Hub screws this up sometimes
|
||||||
|
echo "docker build dir: $(pwd)"
|
||||||
|
|
||||||
|
docker build --build-arg DOCKER_TAG=$DOCKER_TAG --build-arg DOCKER_COMMIT=$SOURCE_COMMIT -f $DOCKERFILE_PATH -t $IMAGE_NAME .
|
11
docker/install_choco.ps1
Normal file
11
docker/install_choco.ps1
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
# requires powershell and .NET 4+. see https://chocolatey.org/install for more info.
|
||||||
|
|
||||||
|
$chocoVersion = powershell choco -v
|
||||||
|
if(-not($chocoVersion)){
|
||||||
|
Write-Output "Chocolatey is not installed, installing now"
|
||||||
|
Write-Output "IF YOU KEEP GETTING THIS MESSAGE ON EVERY BUILD, TRY RESTARTING THE GITLAB RUNNER SO IT GETS CHOCO INTO IT'S ENV"
|
||||||
|
Set-ExecutionPolicy Bypass -Scope Process -Force; iex ((New-Object System.Net.WebClient).DownloadString('https://chocolatey.org/install.ps1'))
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
Write-Output "Chocolatey version $chocoVersion is already installed"
|
||||||
|
}
|
44
docker/set_build.py
Normal file
44
docker/set_build.py
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import logging
|
||||||
|
import lbry.build_info as build_info_mod
|
||||||
|
|
||||||
|
log = logging.getLogger()
|
||||||
|
log.addHandler(logging.StreamHandler())
|
||||||
|
log.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
|
|
||||||
|
def _check_and_set(d: dict, key: str, value: str):
|
||||||
|
try:
|
||||||
|
d[key]
|
||||||
|
except KeyError:
|
||||||
|
raise Exception(f"{key} var does not exist in {build_info_mod.__file__}")
|
||||||
|
d[key] = value
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
build_info = {item: build_info_mod.__dict__[item] for item in dir(build_info_mod) if not item.startswith("__")}
|
||||||
|
|
||||||
|
commit_hash = os.getenv('DOCKER_COMMIT', os.getenv('GITHUB_SHA'))
|
||||||
|
if commit_hash is None:
|
||||||
|
raise ValueError("Commit hash not found in env vars")
|
||||||
|
_check_and_set(build_info, "COMMIT_HASH", commit_hash[:6])
|
||||||
|
|
||||||
|
docker_tag = os.getenv('DOCKER_TAG')
|
||||||
|
if docker_tag:
|
||||||
|
_check_and_set(build_info, "DOCKER_TAG", docker_tag)
|
||||||
|
_check_and_set(build_info, "BUILD", "docker")
|
||||||
|
else:
|
||||||
|
if re.match(r'refs/tags/v\d+\.\d+\.\d+$', str(os.getenv('GITHUB_REF'))):
|
||||||
|
_check_and_set(build_info, "BUILD", "release")
|
||||||
|
else:
|
||||||
|
_check_and_set(build_info, "BUILD", "qa")
|
||||||
|
|
||||||
|
log.debug("build info: %s", ", ".join([f"{k}={v}" for k, v in build_info.items()]))
|
||||||
|
with open(build_info_mod.__file__, 'w') as f:
|
||||||
|
f.write("\n".join([f"{k} = \"{v}\"" for k, v in build_info.items()]) + "\n")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
sys.exit(main())
|
25
docker/wallet_server_entrypoint.sh
Executable file
25
docker/wallet_server_entrypoint.sh
Executable file
|
@ -0,0 +1,25 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# entrypoint for wallet server Docker image
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
SNAPSHOT_URL="${SNAPSHOT_URL:-}" #off by default. latest snapshot at https://lbry.com/snapshot/wallet
|
||||||
|
|
||||||
|
if [[ -n "$SNAPSHOT_URL" ]] && [[ ! -f /database/lbry-leveldb ]]; then
|
||||||
|
files="$(ls)"
|
||||||
|
echo "Downloading wallet snapshot from $SNAPSHOT_URL"
|
||||||
|
wget --no-verbose --trust-server-names --content-disposition "$SNAPSHOT_URL"
|
||||||
|
echo "Extracting snapshot..."
|
||||||
|
filename="$(grep -vf <(echo "$files") <(ls))" # finds the file that was not there before
|
||||||
|
case "$filename" in
|
||||||
|
*.tgz|*.tar.gz|*.tar.bz2 ) tar xvf "$filename" --directory /database ;;
|
||||||
|
*.zip ) unzip "$filename" -d /database ;;
|
||||||
|
* ) echo "Don't know how to extract ${filename}. SNAPSHOT COULD NOT BE LOADED" && exit 1 ;;
|
||||||
|
esac
|
||||||
|
rm "$filename"
|
||||||
|
fi
|
||||||
|
|
||||||
|
/home/lbry/.local/bin/lbry-hub-elastic-sync
|
||||||
|
echo 'starting server'
|
||||||
|
/home/lbry/.local/bin/lbry-hub "$@"
|
9
docker/webconf.yaml
Normal file
9
docker/webconf.yaml
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
allowed_origin: "*"
|
||||||
|
max_key_fee: "0.0 USD"
|
||||||
|
save_files: false
|
||||||
|
save_blobs: false
|
||||||
|
streaming_server: "0.0.0.0:5280"
|
||||||
|
api: "0.0.0.0:5279"
|
||||||
|
data_dir: /tmp
|
||||||
|
download_dir: /tmp
|
||||||
|
wallet_dir: /tmp
|
307
docs/404.html
307
docs/404.html
|
@ -1,307 +0,0 @@
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en" class="no-js">
|
|
||||||
<head>
|
|
||||||
|
|
||||||
<meta charset="utf-8">
|
|
||||||
<meta name="viewport" content="width=device-width,initial-scale=1">
|
|
||||||
<meta http-equiv="x-ua-compatible" content="ie=edge">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<meta name="lang:clipboard.copy" content="Copy to clipboard">
|
|
||||||
|
|
||||||
<meta name="lang:clipboard.copied" content="Copied to clipboard">
|
|
||||||
|
|
||||||
<meta name="lang:search.language" content="en">
|
|
||||||
|
|
||||||
<meta name="lang:search.pipeline.stopwords" content="True">
|
|
||||||
|
|
||||||
<meta name="lang:search.pipeline.trimmer" content="True">
|
|
||||||
|
|
||||||
<meta name="lang:search.result.none" content="No matching documents">
|
|
||||||
|
|
||||||
<meta name="lang:search.result.one" content="1 matching document">
|
|
||||||
|
|
||||||
<meta name="lang:search.result.other" content="# matching documents">
|
|
||||||
|
|
||||||
<meta name="lang:search.tokenizer" content="[\s\-]+">
|
|
||||||
|
|
||||||
<link rel="shortcut icon" href="/assets/images/favicon.png">
|
|
||||||
<meta name="generator" content="mkdocs-0.17.3, mkdocs-material-2.7.0">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<title>LBRY</title>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<link rel="stylesheet" href="/assets/stylesheets/application.78aab2dc.css">
|
|
||||||
|
|
||||||
<link rel="stylesheet" href="/assets/stylesheets/application-palette.6079476c.css">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<script src="/assets/javascripts/modernizr.1aa3b519.js"></script>
|
|
||||||
|
|
||||||
|
|
||||||
<link href="https://fonts.gstatic.com" rel="preconnect" crossorigin>
|
|
||||||
|
|
||||||
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto:300,400,400i,700|Roboto+Mono">
|
|
||||||
<style>body,input{font-family:"Roboto","Helvetica Neue",Helvetica,Arial,sans-serif}code,kbd,pre{font-family:"Roboto Mono","Courier New",Courier,monospace}</style>
|
|
||||||
|
|
||||||
<link rel="stylesheet" href="https://fonts.googleapis.com/icon?family=Material+Icons">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
</head>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<body dir="ltr" data-md-color-primary="teal" data-md-color-accent="green">
|
|
||||||
|
|
||||||
<svg class="md-svg">
|
|
||||||
<defs>
|
|
||||||
|
|
||||||
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="416" height="448"
|
|
||||||
viewBox="0 0 416 448" id="github">
|
|
||||||
<path fill="currentColor" d="M160 304q0 10-3.125 20.5t-10.75 19-18.125
|
|
||||||
8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19 18.125-8.5
|
|
||||||
18.125 8.5 10.75 19 3.125 20.5zM320 304q0 10-3.125 20.5t-10.75
|
|
||||||
19-18.125 8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19
|
|
||||||
18.125-8.5 18.125 8.5 10.75 19 3.125 20.5zM360
|
|
||||||
304q0-30-17.25-51t-46.75-21q-10.25 0-48.75 5.25-17.75 2.75-39.25
|
|
||||||
2.75t-39.25-2.75q-38-5.25-48.75-5.25-29.5 0-46.75 21t-17.25 51q0 22 8
|
|
||||||
38.375t20.25 25.75 30.5 15 35 7.375 37.25 1.75h42q20.5 0
|
|
||||||
37.25-1.75t35-7.375 30.5-15 20.25-25.75 8-38.375zM416 260q0 51.75-15.25
|
|
||||||
82.75-9.5 19.25-26.375 33.25t-35.25 21.5-42.5 11.875-42.875 5.5-41.75
|
|
||||||
1.125q-19.5 0-35.5-0.75t-36.875-3.125-38.125-7.5-34.25-12.875-30.25-20.25-21.5-28.75q-15.5-30.75-15.5-82.75
|
|
||||||
0-59.25 34-99-6.75-20.5-6.75-42.5 0-29 12.75-54.5 27 0 47.5 9.875t47.25
|
|
||||||
30.875q36.75-8.75 77.25-8.75 37 0 70 8 26.25-20.5
|
|
||||||
46.75-30.25t47.25-9.75q12.75 25.5 12.75 54.5 0 21.75-6.75 42 34 40 34
|
|
||||||
99.5z" />
|
|
||||||
</svg>
|
|
||||||
|
|
||||||
</defs>
|
|
||||||
</svg>
|
|
||||||
<input class="md-toggle" data-md-toggle="drawer" type="checkbox" id="drawer">
|
|
||||||
<input class="md-toggle" data-md-toggle="search" type="checkbox" id="search">
|
|
||||||
<label class="md-overlay" data-md-component="overlay" for="drawer"></label>
|
|
||||||
|
|
||||||
|
|
||||||
<header class="md-header" data-md-component="header">
|
|
||||||
<nav class="md-header-nav md-grid">
|
|
||||||
<div class="md-flex">
|
|
||||||
<div class="md-flex__cell md-flex__cell--shrink">
|
|
||||||
<a href="/" title="LBRY" class="md-header-nav__button md-logo">
|
|
||||||
|
|
||||||
<img src="https://s3.amazonaws.com/files.lbry.io/logo-square-white-bookonly.png" alt="LBRY logo" width="24" height="24">
|
|
||||||
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div class="md-flex__cell md-flex__cell--shrink">
|
|
||||||
<label class="md-icon md-icon--menu md-header-nav__button" for="drawer"></label>
|
|
||||||
</div>
|
|
||||||
<div class="md-flex__cell md-flex__cell--stretch">
|
|
||||||
<div class="md-flex__ellipsis md-header-nav__title" data-md-component="title">
|
|
||||||
|
|
||||||
|
|
||||||
<span class="md-header-nav__topic">
|
|
||||||
LBRY
|
|
||||||
</span>
|
|
||||||
<span class="md-header-nav__topic">
|
|
||||||
|
|
||||||
</span>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="md-flex__cell md-flex__cell--shrink">
|
|
||||||
|
|
||||||
|
|
||||||
<label class="md-icon md-icon--search md-header-nav__button" for="search"></label>
|
|
||||||
|
|
||||||
<div class="md-search" data-md-component="search" role="dialog">
|
|
||||||
<label class="md-search__overlay" for="search"></label>
|
|
||||||
<div class="md-search__inner" role="search">
|
|
||||||
<form class="md-search__form" name="search">
|
|
||||||
<input type="text" class="md-search__input" name="query" placeholder="Search" autocapitalize="off" autocorrect="off" autocomplete="off" spellcheck="false" data-md-component="query" data-md-state="active">
|
|
||||||
<label class="md-icon md-search__icon" for="search"></label>
|
|
||||||
<button type="reset" class="md-icon md-search__icon" data-md-component="reset" tabindex="-1">
|
|
||||||

|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
<div class="md-search__output">
|
|
||||||
<div class="md-search__scrollwrap" data-md-scrollfix>
|
|
||||||
<div class="md-search-result" data-md-component="result">
|
|
||||||
<div class="md-search-result__meta">
|
|
||||||
Type to start searching
|
|
||||||
</div>
|
|
||||||
<ol class="md-search-result__list"></ol>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="md-flex__cell md-flex__cell--shrink">
|
|
||||||
<div class="md-header-nav__source">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<a href="https://github.com/lbryio/lbry/" title="Go to repository" class="md-source" data-md-source="github">
|
|
||||||
|
|
||||||
<div class="md-source__icon">
|
|
||||||
<svg viewBox="0 0 24 24" width="24" height="24">
|
|
||||||
<use xlink:href="#github" width="24" height="24"></use>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="md-source__repository">
|
|
||||||
GitHub
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
<div class="md-container">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<main class="md-main">
|
|
||||||
<div class="md-main__inner md-grid" data-md-component="container">
|
|
||||||
|
|
||||||
|
|
||||||
<div class="md-sidebar md-sidebar--primary" data-md-component="navigation">
|
|
||||||
<div class="md-sidebar__scrollwrap">
|
|
||||||
<div class="md-sidebar__inner">
|
|
||||||
<nav class="md-nav md-nav--primary" data-md-level="0">
|
|
||||||
<label class="md-nav__title md-nav__title--site" for="drawer">
|
|
||||||
<span class="md-nav__button md-logo">
|
|
||||||
|
|
||||||
<img src="https://s3.amazonaws.com/files.lbry.io/logo-square-white-bookonly.png" alt="LBRY logo" width="48" height="48">
|
|
||||||
|
|
||||||
</span>
|
|
||||||
LBRY
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<div class="md-nav__source">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<a href="https://github.com/lbryio/lbry/" title="Go to repository" class="md-source" data-md-source="github">
|
|
||||||
|
|
||||||
<div class="md-source__icon">
|
|
||||||
<svg viewBox="0 0 24 24" width="24" height="24">
|
|
||||||
<use xlink:href="#github" width="24" height="24"></use>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="md-source__repository">
|
|
||||||
GitHub
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<ul class="md-nav__list" data-md-scrollfix>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<li class="md-nav__item">
|
|
||||||
<a href="/" title="API" class="md-nav__link">
|
|
||||||
API
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<li class="md-nav__item">
|
|
||||||
<a href="/cli/" title="CLI" class="md-nav__link">
|
|
||||||
CLI
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
|
|
||||||
</ul>
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<div class="md-content">
|
|
||||||
<article class="md-content__inner md-typeset">
|
|
||||||
|
|
||||||
<h1>404 - Not found</h1>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
</article>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
|
|
||||||
|
|
||||||
<footer class="md-footer">
|
|
||||||
|
|
||||||
<div class="md-footer-meta md-typeset">
|
|
||||||
<div class="md-footer-meta__inner md-grid">
|
|
||||||
<div class="md-footer-copyright">
|
|
||||||
|
|
||||||
powered by
|
|
||||||
<a href="http://www.mkdocs.org">MkDocs</a>
|
|
||||||
and
|
|
||||||
<a href="https://squidfunk.github.io/mkdocs-material/">
|
|
||||||
Material for MkDocs</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</footer>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<script src="/assets/javascripts/application.8eb9be28.js"></script>
|
|
||||||
|
|
||||||
<script>app.initialize({version:"0.17.3",url:{base:""}})</script>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<script>!function(e,a,t,n,o,c,i){e.GoogleAnalyticsObject=o,e.ga=e.ga||function(){(e.ga.q=e.ga.q||[]).push(arguments)},e.ga.l=1*new Date,c=a.createElement(t),i=a.getElementsByTagName(t)[0],c.async=1,c.src="https://www.google-analytics.com/analytics.js",i.parentNode.insertBefore(c,i)}(window,document,"script",0,"ga"),ga("create","UA-60403362-1","auto"),ga("set","anonymizeIp",!0),ga("send","pageview");var links=document.getElementsByTagName("a");if(Array.prototype.map.call(links,function(e){e.host!=document.location.host&&e.addEventListener("click",function(){var a=e.getAttribute("data-md-action")||"follow";ga("send","event","outbound",a,e.href)})}),document.forms.search){var query=document.forms.search.query;query.addEventListener("blur",function(){if(this.value){var e=document.location.pathname;ga("send","pageview",e+"?q="+this.value)}})}</script>
|
|
||||||
|
|
||||||
|
|
||||||
</body>
|
|
||||||
</html>
|
|
2880
docs/api.json
2880
docs/api.json
File diff suppressed because one or more lines are too long
Binary file not shown.
Before Width: | Height: | Size: 521 B |
|
@ -1,20 +0,0 @@
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="352" height="448"
|
|
||||||
viewBox="0 0 352 448" id="bitbucket">
|
|
||||||
<path fill="currentColor" d="M203.75 214.75q2 15.75-12.625 25.25t-27.875
|
|
||||||
1.5q-9.75-4.25-13.375-14.5t-0.125-20.5 13-14.5q9-4.5 18.125-3t16 8.875
|
|
||||||
6.875 16.875zM231.5 209.5q-3.5-26.75-28.25-41t-49.25-3.25q-15.75
|
|
||||||
7-25.125 22.125t-8.625 32.375q1 22.75 19.375 38.75t41.375 14q22.75-2
|
|
||||||
38-21t12.5-42zM291.25
|
|
||||||
74q-5-6.75-14-11.125t-14.5-5.5-17.75-3.125q-72.75-11.75-141.5 0.5-10.75
|
|
||||||
1.75-16.5 3t-13.75 5.5-12.5 10.75q7.5 7 19 11.375t18.375 5.5 21.875
|
|
||||||
2.875q57 7.25 112 0.25 15.75-2 22.375-3t18.125-5.375 18.75-11.625zM305.5
|
|
||||||
332.75q-2 6.5-3.875 19.125t-3.5 21-7.125 17.5-14.5 14.125q-21.5
|
|
||||||
12-47.375 17.875t-50.5 5.5-50.375-4.625q-11.5-2-20.375-4.5t-19.125-6.75-18.25-10.875-13-15.375q-6.25-24-14.25-73l1.5-4
|
|
||||||
4.5-2.25q55.75 37 126.625 37t126.875-37q5.25 1.5 6 5.75t-1.25 11.25-2
|
|
||||||
9.25zM350.75 92.5q-6.5 41.75-27.75 163.75-1.25 7.5-6.75 14t-10.875
|
|
||||||
10-13.625 7.75q-63 31.5-152.5
|
|
||||||
22-62-6.75-98.5-34.75-3.75-3-6.375-6.625t-4.25-8.75-2.25-8.5-1.5-9.875-1.375-8.75q-2.25-12.5-6.625-37.5t-7-40.375-5.875-36.875-5.5-39.5q0.75-6.5
|
|
||||||
4.375-12.125t7.875-9.375 11.25-7.5 11.5-5.625 12-4.625q31.25-11.5
|
|
||||||
78.25-16 94.75-9.25 169 12.5 38.75 11.5 53.75 30.5 4 5 4.125
|
|
||||||
12.75t-1.375 13.5z" />
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.4 KiB |
|
@ -1,18 +0,0 @@
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="416" height="448"
|
|
||||||
viewBox="0 0 416 448" id="github">
|
|
||||||
<path fill="currentColor" d="M160 304q0 10-3.125 20.5t-10.75 19-18.125
|
|
||||||
8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19 18.125-8.5
|
|
||||||
18.125 8.5 10.75 19 3.125 20.5zM320 304q0 10-3.125 20.5t-10.75
|
|
||||||
19-18.125 8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19
|
|
||||||
18.125-8.5 18.125 8.5 10.75 19 3.125 20.5zM360
|
|
||||||
304q0-30-17.25-51t-46.75-21q-10.25 0-48.75 5.25-17.75 2.75-39.25
|
|
||||||
2.75t-39.25-2.75q-38-5.25-48.75-5.25-29.5 0-46.75 21t-17.25 51q0 22 8
|
|
||||||
38.375t20.25 25.75 30.5 15 35 7.375 37.25 1.75h42q20.5 0
|
|
||||||
37.25-1.75t35-7.375 30.5-15 20.25-25.75 8-38.375zM416 260q0 51.75-15.25
|
|
||||||
82.75-9.5 19.25-26.375 33.25t-35.25 21.5-42.5 11.875-42.875 5.5-41.75
|
|
||||||
1.125q-19.5 0-35.5-0.75t-36.875-3.125-38.125-7.5-34.25-12.875-30.25-20.25-21.5-28.75q-15.5-30.75-15.5-82.75
|
|
||||||
0-59.25 34-99-6.75-20.5-6.75-42.5 0-29 12.75-54.5 27 0 47.5 9.875t47.25
|
|
||||||
30.875q36.75-8.75 77.25-8.75 37 0 70 8 26.25-20.5
|
|
||||||
46.75-30.25t47.25-9.75q12.75 25.5 12.75 54.5 0 21.75-6.75 42 34 40 34
|
|
||||||
99.5z" />
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.2 KiB |
|
@ -1,38 +0,0 @@
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="500" height="500"
|
|
||||||
viewBox="0 0 500 500" id="gitlab">
|
|
||||||
<g transform="translate(156.197863, 1.160267)">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M93.667,473.347L93.667,473.347l90.684-279.097H2.983L93.667,
|
|
||||||
473.347L93.667,473.347z" />
|
|
||||||
</g>
|
|
||||||
<g transform="translate(28.531199, 1.160800)" opacity="0.7">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M221.333,473.345L130.649,194.25H3.557L221.333,473.345L221.333,
|
|
||||||
473.345z" />
|
|
||||||
</g>
|
|
||||||
<g transform="translate(0.088533, 0.255867)" opacity="0.5">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M32,195.155L32,195.155L4.441,279.97c-2.513,7.735,0.24,16.21,6.821,
|
|
||||||
20.99l238.514,173.29 L32,195.155L32,195.155z" />
|
|
||||||
</g>
|
|
||||||
<g transform="translate(29.421866, 280.255593)">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M2.667-84.844h127.092L75.14-252.942c-2.811-8.649-15.047-8.649-17.856,
|
|
||||||
0L2.667-84.844 L2.667-84.844z" />
|
|
||||||
</g>
|
|
||||||
<g transform="translate(247.197860, 1.160800)" opacity="0.7">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M2.667,473.345L93.351,194.25h127.092L2.667,473.345L2.667,
|
|
||||||
473.345z" />
|
|
||||||
</g>
|
|
||||||
<g transform="translate(246.307061, 0.255867)" opacity="0.5">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M221.334,195.155L221.334,195.155l27.559,84.815c2.514,7.735-0.24,
|
|
||||||
16.21-6.821,20.99 L3.557,474.25L221.334,195.155L221.334,195.155z" />
|
|
||||||
</g>
|
|
||||||
<g transform="translate(336.973725, 280.255593)">
|
|
||||||
<path fill="currentColor"
|
|
||||||
d="M130.667-84.844H3.575l54.618-168.098c2.811-8.649,15.047-8.649,
|
|
||||||
17.856,0L130.667-84.844 L130.667-84.844z" />
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.6 KiB |
File diff suppressed because one or more lines are too long
|
@ -1 +0,0 @@
|
||||||
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r,i,n;e.da=function(){this.pipeline.reset(),this.pipeline.add(e.da.trimmer,e.da.stopWordFilter,e.da.stemmer),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add(e.da.stemmer))},e.da.wordCharacters="A-Za-zªºÀ-ÖØ-öø-ʸˠ-ˤᴀ-ᴥᴬ-ᵜᵢ-ᵥᵫ-ᵷᵹ-ᶾḀ-ỿⁱⁿₐ-ₜKÅℲⅎⅠ-ↈⱠ-ⱿꜢ-ꞇꞋ-ꞭꞰ-ꞷꟷ-ꟿꬰ-ꭚꭜ-ꭤff-stA-Za-z",e.da.trimmer=e.trimmerSupport.generateTrimmer(e.da.wordCharacters),e.Pipeline.registerFunction(e.da.trimmer,"trimmer-da"),e.da.stemmer=(r=e.stemmerSupport.Among,i=e.stemmerSupport.SnowballProgram,n=new function(){var e,n,t,s=[new r("hed",-1,1),new r("ethed",0,1),new r("ered",-1,1),new r("e",-1,1),new r("erede",3,1),new r("ende",3,1),new r("erende",5,1),new r("ene",3,1),new r("erne",3,1),new r("ere",3,1),new r("en",-1,1),new r("heden",10,1),new r("eren",10,1),new r("er",-1,1),new r("heder",13,1),new r("erer",13,1),new r("s",-1,2),new r("heds",16,1),new r("es",16,1),new r("endes",18,1),new r("erendes",19,1),new r("enes",18,1),new r("ernes",18,1),new r("eres",18,1),new r("ens",16,1),new r("hedens",24,1),new r("erens",24,1),new r("ers",16,1),new r("ets",16,1),new r("erets",28,1),new r("et",-1,1),new r("eret",30,1)],o=[new r("gd",-1,-1),new r("dt",-1,-1),new r("gt",-1,-1),new r("kt",-1,-1)],a=[new r("ig",-1,1),new r("lig",0,1),new r("elig",1,1),new r("els",-1,1),new r("løst",-1,2)],d=[17,65,16,1,0,0,0,0,0,0,0,0,0,0,0,0,48,0,128],u=[239,254,42,3,0,0,0,0,0,0,0,0,0,0,0,0,16],c=new i;function l(){var e,r=c.limit-c.cursor;c.cursor>=n&&(e=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,c.find_among_b(o,4)?(c.bra=c.cursor,c.limit_backward=e,c.cursor=c.limit-r,c.cursor>c.limit_backward&&(c.cursor--,c.bra=c.cursor,c.slice_del())):c.limit_backward=e)}this.setCurrent=function(e){c.setCurrent(e)},this.getCurrent=function(){return c.getCurrent()},this.stem=function(){var r,i=c.cursor;return function(){var r,i=c.cursor+3;if(n=c.limit,0<=i&&i<=c.limit){for(e=i;;){if(r=c.cursor,c.in_grouping(d,97,248)){c.cursor=r;break}if(c.cursor=r,r>=c.limit)return;c.cursor++}for(;!c.out_grouping(d,97,248);){if(c.cursor>=c.limit)return;c.cursor++}(n=c.cursor)<e&&(n=e)}}(),c.limit_backward=i,c.cursor=c.limit,function(){var e,r;if(c.cursor>=n&&(r=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,e=c.find_among_b(s,32),c.limit_backward=r,e))switch(c.bra=c.cursor,e){case 1:c.slice_del();break;case 2:c.in_grouping_b(u,97,229)&&c.slice_del()}}(),c.cursor=c.limit,l(),c.cursor=c.limit,function(){var e,r,i,t=c.limit-c.cursor;if(c.ket=c.cursor,c.eq_s_b(2,"st")&&(c.bra=c.cursor,c.eq_s_b(2,"ig")&&c.slice_del()),c.cursor=c.limit-t,c.cursor>=n&&(r=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,e=c.find_among_b(a,5),c.limit_backward=r,e))switch(c.bra=c.cursor,e){case 1:c.slice_del(),i=c.limit-c.cursor,l(),c.cursor=c.limit-i;break;case 2:c.slice_from("løs")}}(),c.cursor=c.limit,c.cursor>=n&&(r=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,c.out_grouping_b(d,97,248)?(c.bra=c.cursor,t=c.slice_to(t),c.limit_backward=r,c.eq_v_b(t)&&c.slice_del()):c.limit_backward=r),!0}},function(e){return"function"==typeof e.update?e.update(function(e){return n.setCurrent(e),n.stem(),n.getCurrent()}):(n.setCurrent(e),n.stem(),n.getCurrent())}),e.Pipeline.registerFunction(e.da.stemmer,"stemmer-da"),e.da.stopWordFilter=e.generateStopWordFilter("ad af alle alt anden at blev blive bliver da de dem den denne der deres det dette dig din disse dog du efter eller en end er et for fra ham han hans har havde have hende hendes her hos hun hvad hvis hvor i ikke ind jeg jer jo kunne man mange med meget men mig min mine mit mod ned noget nogle nu når og også om op os over på selv sig sin sine sit skal skulle som sådan thi til ud under var vi vil ville vor være været".split(" ")),e.Pipeline.registerFunction(e.da.stopWordFilter,"stopWordFilter-da")}});
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1 +0,0 @@
|
||||||
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r="2"==e.version[0];e.jp=function(){this.pipeline.reset(),this.pipeline.add(e.jp.stopWordFilter,e.jp.stemmer),r?this.tokenizer=e.jp.tokenizer:(e.tokenizer&&(e.tokenizer=e.jp.tokenizer),this.tokenizerFn&&(this.tokenizerFn=e.jp.tokenizer))};var t=new e.TinySegmenter;e.jp.tokenizer=function(n){if(!arguments.length||null==n||null==n)return[];if(Array.isArray(n))return n.map(function(t){return r?new e.Token(t.toLowerCase()):t.toLowerCase()});for(var i=n.toString().toLowerCase().replace(/^\s+/,""),o=i.length-1;o>=0;o--)if(/\S/.test(i.charAt(o))){i=i.substring(0,o+1);break}return t.segment(i).filter(function(e){return!!e}).map(function(t){return r?new e.Token(t):t})},e.jp.stemmer=function(e){return e},e.Pipeline.registerFunction(e.jp.stemmer,"stemmer-jp"),e.jp.wordCharacters="一二三四五六七八九十百千万億兆一-龠々〆ヵヶぁ-んァ-ヴーア-ン゙a-zA-Za-zA-Z0-90-9",e.jp.stopWordFilter=function(t){if(-1===e.jp.stopWordFilter.stopWords.indexOf(r?t.toString():t))return t},e.jp.stopWordFilter=e.generateStopWordFilter("これ それ あれ この その あの ここ そこ あそこ こちら どこ だれ なに なん 何 私 貴方 貴方方 我々 私達 あの人 あのかた 彼女 彼 です あります おります います は が の に を で え から まで より も どの と し それで しかし".split(" ")),e.Pipeline.registerFunction(e.jp.stopWordFilter,"stopWordFilter-jp")}});
|
|
|
@ -1 +0,0 @@
|
||||||
!function(e,i){"function"==typeof define&&define.amd?define(i):"object"==typeof exports?module.exports=i():i()(e.lunr)}(this,function(){return function(e){e.multiLanguage=function(){for(var i=Array.prototype.slice.call(arguments),t=i.join("-"),r="",n=[],s=[],p=0;p<i.length;++p)"en"==i[p]?(r+="\\w",n.unshift(e.stopWordFilter),n.push(e.stemmer),s.push(e.stemmer)):(r+=e[i[p]].wordCharacters,n.unshift(e[i[p]].stopWordFilter),n.push(e[i[p]].stemmer),s.push(e[i[p]].stemmer));var o=e.trimmerSupport.generateTrimmer(r);return e.Pipeline.registerFunction(o,"lunr-multi-trimmer-"+t),n.unshift(o),function(){this.pipeline.reset(),this.pipeline.add.apply(this.pipeline,n),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add.apply(this.searchPipeline,s))}}}});
|
|
|
@ -1 +0,0 @@
|
||||||
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r,n,i;e.no=function(){this.pipeline.reset(),this.pipeline.add(e.no.trimmer,e.no.stopWordFilter,e.no.stemmer),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add(e.no.stemmer))},e.no.wordCharacters="A-Za-zªºÀ-ÖØ-öø-ʸˠ-ˤᴀ-ᴥᴬ-ᵜᵢ-ᵥᵫ-ᵷᵹ-ᶾḀ-ỿⁱⁿₐ-ₜKÅℲⅎⅠ-ↈⱠ-ⱿꜢ-ꞇꞋ-ꞭꞰ-ꞷꟷ-ꟿꬰ-ꭚꭜ-ꭤff-stA-Za-z",e.no.trimmer=e.trimmerSupport.generateTrimmer(e.no.wordCharacters),e.Pipeline.registerFunction(e.no.trimmer,"trimmer-no"),e.no.stemmer=(r=e.stemmerSupport.Among,n=e.stemmerSupport.SnowballProgram,i=new function(){var e,i,t=[new r("a",-1,1),new r("e",-1,1),new r("ede",1,1),new r("ande",1,1),new r("ende",1,1),new r("ane",1,1),new r("ene",1,1),new r("hetene",6,1),new r("erte",1,3),new r("en",-1,1),new r("heten",9,1),new r("ar",-1,1),new r("er",-1,1),new r("heter",12,1),new r("s",-1,2),new r("as",14,1),new r("es",14,1),new r("edes",16,1),new r("endes",16,1),new r("enes",16,1),new r("hetenes",19,1),new r("ens",14,1),new r("hetens",21,1),new r("ers",14,1),new r("ets",14,1),new r("et",-1,1),new r("het",25,1),new r("ert",-1,3),new r("ast",-1,1)],o=[new r("dt",-1,-1),new r("vt",-1,-1)],s=[new r("leg",-1,1),new r("eleg",0,1),new r("ig",-1,1),new r("eig",2,1),new r("lig",2,1),new r("elig",4,1),new r("els",-1,1),new r("lov",-1,1),new r("elov",7,1),new r("slov",7,1),new r("hetslov",9,1)],a=[17,65,16,1,0,0,0,0,0,0,0,0,0,0,0,0,48,0,128],m=[119,125,149,1],l=new n;this.setCurrent=function(e){l.setCurrent(e)},this.getCurrent=function(){return l.getCurrent()},this.stem=function(){var r,n,u,d,c=l.cursor;return function(){var r,n=l.cursor+3;if(i=l.limit,0<=n||n<=l.limit){for(e=n;;){if(r=l.cursor,l.in_grouping(a,97,248)){l.cursor=r;break}if(r>=l.limit)return;l.cursor=r+1}for(;!l.out_grouping(a,97,248);){if(l.cursor>=l.limit)return;l.cursor++}(i=l.cursor)<e&&(i=e)}}(),l.limit_backward=c,l.cursor=l.limit,function(){var e,r,n;if(l.cursor>=i&&(r=l.limit_backward,l.limit_backward=i,l.ket=l.cursor,e=l.find_among_b(t,29),l.limit_backward=r,e))switch(l.bra=l.cursor,e){case 1:l.slice_del();break;case 2:n=l.limit-l.cursor,l.in_grouping_b(m,98,122)?l.slice_del():(l.cursor=l.limit-n,l.eq_s_b(1,"k")&&l.out_grouping_b(a,97,248)&&l.slice_del());break;case 3:l.slice_from("er")}}(),l.cursor=l.limit,n=l.limit-l.cursor,l.cursor>=i&&(r=l.limit_backward,l.limit_backward=i,l.ket=l.cursor,l.find_among_b(o,2)?(l.bra=l.cursor,l.limit_backward=r,l.cursor=l.limit-n,l.cursor>l.limit_backward&&(l.cursor--,l.bra=l.cursor,l.slice_del())):l.limit_backward=r),l.cursor=l.limit,l.cursor>=i&&(d=l.limit_backward,l.limit_backward=i,l.ket=l.cursor,(u=l.find_among_b(s,11))?(l.bra=l.cursor,l.limit_backward=d,1==u&&l.slice_del()):l.limit_backward=d),!0}},function(e){return"function"==typeof e.update?e.update(function(e){return i.setCurrent(e),i.stem(),i.getCurrent()}):(i.setCurrent(e),i.stem(),i.getCurrent())}),e.Pipeline.registerFunction(e.no.stemmer,"stemmer-no"),e.no.stopWordFilter=e.generateStopWordFilter("alle at av bare begge ble blei bli blir blitt både båe da de deg dei deim deira deires dem den denne der dere deres det dette di din disse ditt du dykk dykkar då eg ein eit eitt eller elles en enn er et ett etter for fordi fra før ha hadde han hans har hennar henne hennes her hjå ho hoe honom hoss hossen hun hva hvem hver hvilke hvilken hvis hvor hvordan hvorfor i ikke ikkje ikkje ingen ingi inkje inn inni ja jeg kan kom korleis korso kun kunne kva kvar kvarhelst kven kvi kvifor man mange me med medan meg meget mellom men mi min mine mitt mot mykje ned no noe noen noka noko nokon nokor nokre nå når og også om opp oss over på samme seg selv si si sia sidan siden sin sine sitt sjøl skal skulle slik so som som somme somt så sånn til um upp ut uten var vart varte ved vere verte vi vil ville vore vors vort vår være være vært å".split(" ")),e.Pipeline.registerFunction(e.no.stopWordFilter,"stopWordFilter-no")}});
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1 +0,0 @@
|
||||||
!function(r,t){"function"==typeof define&&define.amd?define(t):"object"==typeof exports?module.exports=t():t()(r.lunr)}(this,function(){return function(r){r.stemmerSupport={Among:function(r,t,i,s){if(this.toCharArray=function(r){for(var t=r.length,i=new Array(t),s=0;s<t;s++)i[s]=r.charCodeAt(s);return i},!r&&""!=r||!t&&0!=t||!i)throw"Bad Among initialisation: s:"+r+", substring_i: "+t+", result: "+i;this.s_size=r.length,this.s=this.toCharArray(r),this.substring_i=t,this.result=i,this.method=s},SnowballProgram:function(){var r;return{bra:0,ket:0,limit:0,cursor:0,limit_backward:0,setCurrent:function(t){r=t,this.cursor=0,this.limit=t.length,this.limit_backward=0,this.bra=this.cursor,this.ket=this.limit},getCurrent:function(){var t=r;return r=null,t},in_grouping:function(t,i,s){if(this.cursor<this.limit){var e=r.charCodeAt(this.cursor);if(e<=s&&e>=i&&t[(e-=i)>>3]&1<<(7&e))return this.cursor++,!0}return!1},in_grouping_b:function(t,i,s){if(this.cursor>this.limit_backward){var e=r.charCodeAt(this.cursor-1);if(e<=s&&e>=i&&t[(e-=i)>>3]&1<<(7&e))return this.cursor--,!0}return!1},out_grouping:function(t,i,s){if(this.cursor<this.limit){var e=r.charCodeAt(this.cursor);if(e>s||e<i)return this.cursor++,!0;if(!(t[(e-=i)>>3]&1<<(7&e)))return this.cursor++,!0}return!1},out_grouping_b:function(t,i,s){if(this.cursor>this.limit_backward){var e=r.charCodeAt(this.cursor-1);if(e>s||e<i)return this.cursor--,!0;if(!(t[(e-=i)>>3]&1<<(7&e)))return this.cursor--,!0}return!1},eq_s:function(t,i){if(this.limit-this.cursor<t)return!1;for(var s=0;s<t;s++)if(r.charCodeAt(this.cursor+s)!=i.charCodeAt(s))return!1;return this.cursor+=t,!0},eq_s_b:function(t,i){if(this.cursor-this.limit_backward<t)return!1;for(var s=0;s<t;s++)if(r.charCodeAt(this.cursor-t+s)!=i.charCodeAt(s))return!1;return this.cursor-=t,!0},find_among:function(t,i){for(var s=0,e=i,n=this.cursor,u=this.limit,o=0,h=0,c=!1;;){for(var a=s+(e-s>>1),f=0,l=o<h?o:h,_=t[a],m=l;m<_.s_size;m++){if(n+l==u){f=-1;break}if(f=r.charCodeAt(n+l)-_.s[m])break;l++}if(f<0?(e=a,h=l):(s=a,o=l),e-s<=1){if(s>0||e==s||c)break;c=!0}}for(;;){if(o>=(_=t[s]).s_size){if(this.cursor=n+_.s_size,!_.method)return _.result;var b=_.method();if(this.cursor=n+_.s_size,b)return _.result}if((s=_.substring_i)<0)return 0}},find_among_b:function(t,i){for(var s=0,e=i,n=this.cursor,u=this.limit_backward,o=0,h=0,c=!1;;){for(var a=s+(e-s>>1),f=0,l=o<h?o:h,_=(m=t[a]).s_size-1-l;_>=0;_--){if(n-l==u){f=-1;break}if(f=r.charCodeAt(n-1-l)-m.s[_])break;l++}if(f<0?(e=a,h=l):(s=a,o=l),e-s<=1){if(s>0||e==s||c)break;c=!0}}for(;;){var m;if(o>=(m=t[s]).s_size){if(this.cursor=n-m.s_size,!m.method)return m.result;var b=m.method();if(this.cursor=n-m.s_size,b)return m.result}if((s=m.substring_i)<0)return 0}},replace_s:function(t,i,s){var e=s.length-(i-t),n=r.substring(0,t),u=r.substring(i);return r=n+s+u,this.limit+=e,this.cursor>=i?this.cursor+=e:this.cursor>t&&(this.cursor=t),e},slice_check:function(){if(this.bra<0||this.bra>this.ket||this.ket>this.limit||this.limit>r.length)throw"faulty slice operation"},slice_from:function(r){this.slice_check(),this.replace_s(this.bra,this.ket,r)},slice_del:function(){this.slice_from("")},insert:function(r,t,i){var s=this.replace_s(r,t,i);r<=this.bra&&(this.bra+=s),r<=this.ket&&(this.ket+=s)},slice_to:function(){return this.slice_check(),r.substring(this.bra,this.ket)},eq_v_b:function(r){return this.eq_s_b(r.length,r)}}}},r.trimmerSupport={generateTrimmer:function(r){var t=new RegExp("^[^"+r+"]+"),i=new RegExp("[^"+r+"]+$");return function(r){return"function"==typeof r.update?r.update(function(r){return r.replace(t,"").replace(i,"")}):r.replace(t,"").replace(i,"")}}}}});
|
|
|
@ -1 +0,0 @@
|
||||||
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r,n,t;e.sv=function(){this.pipeline.reset(),this.pipeline.add(e.sv.trimmer,e.sv.stopWordFilter,e.sv.stemmer),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add(e.sv.stemmer))},e.sv.wordCharacters="A-Za-zªºÀ-ÖØ-öø-ʸˠ-ˤᴀ-ᴥᴬ-ᵜᵢ-ᵥᵫ-ᵷᵹ-ᶾḀ-ỿⁱⁿₐ-ₜKÅℲⅎⅠ-ↈⱠ-ⱿꜢ-ꞇꞋ-ꞭꞰ-ꞷꟷ-ꟿꬰ-ꭚꭜ-ꭤff-stA-Za-z",e.sv.trimmer=e.trimmerSupport.generateTrimmer(e.sv.wordCharacters),e.Pipeline.registerFunction(e.sv.trimmer,"trimmer-sv"),e.sv.stemmer=(r=e.stemmerSupport.Among,n=e.stemmerSupport.SnowballProgram,t=new function(){var e,t,i=[new r("a",-1,1),new r("arna",0,1),new r("erna",0,1),new r("heterna",2,1),new r("orna",0,1),new r("ad",-1,1),new r("e",-1,1),new r("ade",6,1),new r("ande",6,1),new r("arne",6,1),new r("are",6,1),new r("aste",6,1),new r("en",-1,1),new r("anden",12,1),new r("aren",12,1),new r("heten",12,1),new r("ern",-1,1),new r("ar",-1,1),new r("er",-1,1),new r("heter",18,1),new r("or",-1,1),new r("s",-1,2),new r("as",21,1),new r("arnas",22,1),new r("ernas",22,1),new r("ornas",22,1),new r("es",21,1),new r("ades",26,1),new r("andes",26,1),new r("ens",21,1),new r("arens",29,1),new r("hetens",29,1),new r("erns",21,1),new r("at",-1,1),new r("andet",-1,1),new r("het",-1,1),new r("ast",-1,1)],s=[new r("dd",-1,-1),new r("gd",-1,-1),new r("nn",-1,-1),new r("dt",-1,-1),new r("gt",-1,-1),new r("kt",-1,-1),new r("tt",-1,-1)],a=[new r("ig",-1,1),new r("lig",0,1),new r("els",-1,1),new r("fullt",-1,3),new r("löst",-1,2)],o=[17,65,16,1,0,0,0,0,0,0,0,0,0,0,0,0,24,0,32],u=[119,127,149],m=new n;this.setCurrent=function(e){m.setCurrent(e)},this.getCurrent=function(){return m.getCurrent()},this.stem=function(){var r,n=m.cursor;return function(){var r,n=m.cursor+3;if(t=m.limit,0<=n||n<=m.limit){for(e=n;;){if(r=m.cursor,m.in_grouping(o,97,246)){m.cursor=r;break}if(m.cursor=r,m.cursor>=m.limit)return;m.cursor++}for(;!m.out_grouping(o,97,246);){if(m.cursor>=m.limit)return;m.cursor++}(t=m.cursor)<e&&(t=e)}}(),m.limit_backward=n,m.cursor=m.limit,function(){var e,r=m.limit_backward;if(m.cursor>=t&&(m.limit_backward=t,m.cursor=m.limit,m.ket=m.cursor,e=m.find_among_b(i,37),m.limit_backward=r,e))switch(m.bra=m.cursor,e){case 1:m.slice_del();break;case 2:m.in_grouping_b(u,98,121)&&m.slice_del()}}(),m.cursor=m.limit,r=m.limit_backward,m.cursor>=t&&(m.limit_backward=t,m.cursor=m.limit,m.find_among_b(s,7)&&(m.cursor=m.limit,m.ket=m.cursor,m.cursor>m.limit_backward&&(m.bra=--m.cursor,m.slice_del())),m.limit_backward=r),m.cursor=m.limit,function(){var e,r;if(m.cursor>=t){if(r=m.limit_backward,m.limit_backward=t,m.cursor=m.limit,m.ket=m.cursor,e=m.find_among_b(a,5))switch(m.bra=m.cursor,e){case 1:m.slice_del();break;case 2:m.slice_from("lös");break;case 3:m.slice_from("full")}m.limit_backward=r}}(),!0}},function(e){return"function"==typeof e.update?e.update(function(e){return t.setCurrent(e),t.stem(),t.getCurrent()}):(t.setCurrent(e),t.stem(),t.getCurrent())}),e.Pipeline.registerFunction(e.sv.stemmer,"stemmer-sv"),e.sv.stopWordFilter=e.generateStopWordFilter("alla allt att av blev bli blir blivit de dem den denna deras dess dessa det detta dig din dina ditt du där då efter ej eller en er era ert ett från för ha hade han hans har henne hennes hon honom hur här i icke ingen inom inte jag ju kan kunde man med mellan men mig min mina mitt mot mycket ni nu när någon något några och om oss på samma sedan sig sin sina sitta själv skulle som så sådan sådana sådant till under upp ut utan vad var vara varför varit varje vars vart vem vi vid vilka vilkas vilken vilket vår våra vårt än är åt över".split(" ")),e.Pipeline.registerFunction(e.sv.stopWordFilter,"stopWordFilter-sv")}});
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
2587
docs/cli/index.html
2587
docs/cli/index.html
File diff suppressed because it is too large
Load diff
2313
docs/index.html
2313
docs/index.html
File diff suppressed because it is too large
Load diff
File diff suppressed because one or more lines are too long
|
@ -2,6 +2,6 @@
|
||||||
.tox
|
.tox
|
||||||
__pycache__
|
__pycache__
|
||||||
dist
|
dist
|
||||||
lbrynet.egg-info
|
lbry.egg-info
|
||||||
docs
|
docs
|
||||||
tests
|
tests
|
2
lbry/__init__.py
Normal file
2
lbry/__init__.py
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
__version__ = "0.113.0"
|
||||||
|
version = tuple(map(int, __version__.split('.'))) # pylint: disable=invalid-name
|
6
lbry/blob/__init__.py
Normal file
6
lbry/blob/__init__.py
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
from lbry.utils import get_lbry_hash_obj
|
||||||
|
|
||||||
|
MAX_BLOB_SIZE = 2 * 2 ** 20
|
||||||
|
|
||||||
|
# digest_size is in bytes, and blob hashes are hex encoded
|
||||||
|
BLOBHASH_LENGTH = get_lbry_hash_obj().digest_size * 2
|
|
@ -1,5 +1,6 @@
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
import time
|
||||||
import asyncio
|
import asyncio
|
||||||
import binascii
|
import binascii
|
||||||
import logging
|
import logging
|
||||||
|
@ -9,18 +10,20 @@ from io import BytesIO
|
||||||
from cryptography.hazmat.primitives.ciphers import Cipher, modes
|
from cryptography.hazmat.primitives.ciphers import Cipher, modes
|
||||||
from cryptography.hazmat.primitives.ciphers.algorithms import AES
|
from cryptography.hazmat.primitives.ciphers.algorithms import AES
|
||||||
from cryptography.hazmat.primitives.padding import PKCS7
|
from cryptography.hazmat.primitives.padding import PKCS7
|
||||||
|
from cryptography.hazmat.backends import default_backend
|
||||||
|
|
||||||
from lbrynet.cryptoutils import backend, get_lbry_hash_obj
|
from lbry.utils import get_lbry_hash_obj
|
||||||
from lbrynet.error import DownloadCancelledError, InvalidBlobHashError, InvalidDataError
|
from lbry.error import DownloadCancelledError, InvalidBlobHashError, InvalidDataError
|
||||||
|
|
||||||
from lbrynet.blob import MAX_BLOB_SIZE, blobhash_length
|
from lbry.blob import MAX_BLOB_SIZE, BLOBHASH_LENGTH
|
||||||
from lbrynet.blob.blob_info import BlobInfo
|
from lbry.blob.blob_info import BlobInfo
|
||||||
from lbrynet.blob.writer import HashBlobWriter
|
from lbry.blob.writer import HashBlobWriter
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
_hexmatch = re.compile("^[a-f,0-9]+$")
|
HEXMATCH = re.compile("^[a-f,0-9]+$")
|
||||||
|
BACKEND = default_backend()
|
||||||
|
|
||||||
|
|
||||||
def is_valid_blobhash(blobhash: str) -> bool:
|
def is_valid_blobhash(blobhash: str) -> bool:
|
||||||
|
@ -31,11 +34,11 @@ def is_valid_blobhash(blobhash: str) -> bool:
|
||||||
|
|
||||||
@return: True/False
|
@return: True/False
|
||||||
"""
|
"""
|
||||||
return len(blobhash) == blobhash_length and _hexmatch.match(blobhash)
|
return len(blobhash) == BLOBHASH_LENGTH and HEXMATCH.match(blobhash)
|
||||||
|
|
||||||
|
|
||||||
def encrypt_blob_bytes(key: bytes, iv: bytes, unencrypted: bytes) -> typing.Tuple[bytes, str]:
|
def encrypt_blob_bytes(key: bytes, iv: bytes, unencrypted: bytes) -> typing.Tuple[bytes, str]:
|
||||||
cipher = Cipher(AES(key), modes.CBC(iv), backend=backend)
|
cipher = Cipher(AES(key), modes.CBC(iv), backend=BACKEND)
|
||||||
padder = PKCS7(AES.block_size).padder()
|
padder = PKCS7(AES.block_size).padder()
|
||||||
encryptor = cipher.encryptor()
|
encryptor = cipher.encryptor()
|
||||||
encrypted = encryptor.update(padder.update(unencrypted) + padder.finalize()) + encryptor.finalize()
|
encrypted = encryptor.update(padder.update(unencrypted) + padder.finalize()) + encryptor.finalize()
|
||||||
|
@ -47,7 +50,7 @@ def encrypt_blob_bytes(key: bytes, iv: bytes, unencrypted: bytes) -> typing.Tupl
|
||||||
def decrypt_blob_bytes(data: bytes, length: int, key: bytes, iv: bytes) -> bytes:
|
def decrypt_blob_bytes(data: bytes, length: int, key: bytes, iv: bytes) -> bytes:
|
||||||
if len(data) != length:
|
if len(data) != length:
|
||||||
raise ValueError("unexpected length")
|
raise ValueError("unexpected length")
|
||||||
cipher = Cipher(AES(key), modes.CBC(iv), backend=backend)
|
cipher = Cipher(AES(key), modes.CBC(iv), backend=BACKEND)
|
||||||
unpadder = PKCS7(AES.block_size).unpadder()
|
unpadder = PKCS7(AES.block_size).unpadder()
|
||||||
decryptor = cipher.decryptor()
|
decryptor = cipher.decryptor()
|
||||||
return unpadder.update(decryptor.update(data) + decryptor.finalize()) + unpadder.finalize()
|
return unpadder.update(decryptor.update(data) + decryptor.finalize()) + unpadder.finalize()
|
||||||
|
@ -68,21 +71,27 @@ class AbstractBlob:
|
||||||
'writers',
|
'writers',
|
||||||
'verified',
|
'verified',
|
||||||
'writing',
|
'writing',
|
||||||
'readers'
|
'readers',
|
||||||
|
'added_on',
|
||||||
|
'is_mine',
|
||||||
]
|
]
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
def __init__(
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
self, loop: asyncio.AbstractEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
||||||
blob_directory: typing.Optional[str] = None):
|
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
||||||
|
blob_directory: typing.Optional[str] = None, added_on: typing.Optional[int] = None, is_mine: bool = False,
|
||||||
|
):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.blob_hash = blob_hash
|
self.blob_hash = blob_hash
|
||||||
self.length = length
|
self.length = length
|
||||||
self.blob_completed_callback = blob_completed_callback
|
self.blob_completed_callback = blob_completed_callback
|
||||||
self.blob_directory = blob_directory
|
self.blob_directory = blob_directory
|
||||||
self.writers: typing.Dict[typing.Tuple[typing.Optional[str], typing.Optional[int]], HashBlobWriter] = {}
|
self.writers: typing.Dict[typing.Tuple[typing.Optional[str], typing.Optional[int]], HashBlobWriter] = {}
|
||||||
self.verified: asyncio.Event = asyncio.Event(loop=self.loop)
|
self.verified: asyncio.Event = asyncio.Event()
|
||||||
self.writing: asyncio.Event = asyncio.Event(loop=self.loop)
|
self.writing: asyncio.Event = asyncio.Event()
|
||||||
self.readers: typing.List[typing.BinaryIO] = []
|
self.readers: typing.List[typing.BinaryIO] = []
|
||||||
|
self.added_on = added_on or time.time()
|
||||||
|
self.is_mine = is_mine
|
||||||
|
|
||||||
if not is_valid_blobhash(blob_hash):
|
if not is_valid_blobhash(blob_hash):
|
||||||
raise InvalidBlobHashError(blob_hash)
|
raise InvalidBlobHashError(blob_hash)
|
||||||
|
@ -108,7 +117,7 @@ class AbstractBlob:
|
||||||
if reader in self.readers:
|
if reader in self.readers:
|
||||||
self.readers.remove(reader)
|
self.readers.remove(reader)
|
||||||
|
|
||||||
def _write_blob(self, blob_bytes: bytes):
|
def _write_blob(self, blob_bytes: bytes) -> asyncio.Task:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def set_length(self, length) -> None:
|
def set_length(self, length) -> None:
|
||||||
|
@ -142,7 +151,7 @@ class AbstractBlob:
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
while self.writers:
|
while self.writers:
|
||||||
peer, writer = self.writers.popitem()
|
_, writer = self.writers.popitem()
|
||||||
if writer and writer.finished and not writer.finished.done() and not self.loop.is_closed():
|
if writer and writer.finished and not writer.finished.done() and not self.loop.is_closed():
|
||||||
writer.finished.cancel()
|
writer.finished.cancel()
|
||||||
while self.readers:
|
while self.readers:
|
||||||
|
@ -163,7 +172,10 @@ class AbstractBlob:
|
||||||
if not self.is_readable():
|
if not self.is_readable():
|
||||||
raise OSError('blob files cannot be read')
|
raise OSError('blob files cannot be read')
|
||||||
with self.reader_context() as handle:
|
with self.reader_context() as handle:
|
||||||
return await self.loop.sendfile(writer.transport, handle, count=self.get_length())
|
try:
|
||||||
|
return await self.loop.sendfile(writer.transport, handle, count=self.get_length())
|
||||||
|
except (ConnectionError, BrokenPipeError, RuntimeError, OSError, AttributeError):
|
||||||
|
return -1
|
||||||
|
|
||||||
def decrypt(self, key: bytes, iv: bytes) -> bytes:
|
def decrypt(self, key: bytes, iv: bytes) -> bytes:
|
||||||
"""
|
"""
|
||||||
|
@ -175,35 +187,42 @@ class AbstractBlob:
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
async def create_from_unencrypted(
|
async def create_from_unencrypted(
|
||||||
cls, loop: asyncio.BaseEventLoop, blob_dir: typing.Optional[str], key: bytes, iv: bytes,
|
cls, loop: asyncio.AbstractEventLoop, blob_dir: typing.Optional[str], key: bytes, iv: bytes,
|
||||||
unencrypted: bytes, blob_num: int,
|
unencrypted: bytes, blob_num: int, added_on: int, is_mine: bool,
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], None]] = None) -> BlobInfo:
|
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], None]] = None,
|
||||||
|
) -> BlobInfo:
|
||||||
"""
|
"""
|
||||||
Create an encrypted BlobFile from plaintext bytes
|
Create an encrypted BlobFile from plaintext bytes
|
||||||
"""
|
"""
|
||||||
|
|
||||||
blob_bytes, blob_hash = encrypt_blob_bytes(key, iv, unencrypted)
|
blob_bytes, blob_hash = encrypt_blob_bytes(key, iv, unencrypted)
|
||||||
length = len(blob_bytes)
|
length = len(blob_bytes)
|
||||||
blob = cls(loop, blob_hash, length, blob_completed_callback, blob_dir)
|
blob = cls(loop, blob_hash, length, blob_completed_callback, blob_dir, added_on, is_mine)
|
||||||
writer = blob.get_blob_writer()
|
writer = blob.get_blob_writer()
|
||||||
writer.write(blob_bytes)
|
writer.write(blob_bytes)
|
||||||
await blob.verified.wait()
|
await blob.verified.wait()
|
||||||
return BlobInfo(blob_num, length, binascii.hexlify(iv).decode(), blob_hash)
|
return BlobInfo(blob_num, length, binascii.hexlify(iv).decode(), added_on, blob_hash, is_mine)
|
||||||
|
|
||||||
def save_verified_blob(self, verified_bytes: bytes):
|
def save_verified_blob(self, verified_bytes: bytes):
|
||||||
if self.verified.is_set():
|
if self.verified.is_set():
|
||||||
return
|
return
|
||||||
if self.is_writeable():
|
|
||||||
self._write_blob(verified_bytes)
|
def update_events(_):
|
||||||
self.verified.set()
|
self.verified.set()
|
||||||
|
self.writing.clear()
|
||||||
|
|
||||||
|
if self.is_writeable():
|
||||||
|
self.writing.set()
|
||||||
|
task = self._write_blob(verified_bytes)
|
||||||
|
task.add_done_callback(update_events)
|
||||||
if self.blob_completed_callback:
|
if self.blob_completed_callback:
|
||||||
self.blob_completed_callback(self)
|
task.add_done_callback(lambda _: self.blob_completed_callback(self))
|
||||||
|
|
||||||
def get_blob_writer(self, peer_address: typing.Optional[str] = None,
|
def get_blob_writer(self, peer_address: typing.Optional[str] = None,
|
||||||
peer_port: typing.Optional[int] = None) -> HashBlobWriter:
|
peer_port: typing.Optional[int] = None) -> HashBlobWriter:
|
||||||
if (peer_address, peer_port) in self.writers and not self.writers[(peer_address, peer_port)].closed():
|
if (peer_address, peer_port) in self.writers and not self.writers[(peer_address, peer_port)].closed():
|
||||||
raise OSError(f"attempted to download blob twice from {peer_address}:{peer_port}")
|
raise OSError(f"attempted to download blob twice from {peer_address}:{peer_port}")
|
||||||
fut = asyncio.Future(loop=self.loop)
|
fut = asyncio.Future()
|
||||||
writer = HashBlobWriter(self.blob_hash, self.get_length, fut)
|
writer = HashBlobWriter(self.blob_hash, self.get_length, fut)
|
||||||
self.writers[(peer_address, peer_port)] = writer
|
self.writers[(peer_address, peer_port)] = writer
|
||||||
|
|
||||||
|
@ -237,11 +256,13 @@ class BlobBuffer(AbstractBlob):
|
||||||
"""
|
"""
|
||||||
An in-memory only blob
|
An in-memory only blob
|
||||||
"""
|
"""
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
def __init__(
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
self, loop: asyncio.AbstractEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
||||||
blob_directory: typing.Optional[str] = None):
|
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
||||||
|
blob_directory: typing.Optional[str] = None, added_on: typing.Optional[int] = None, is_mine: bool = False
|
||||||
|
):
|
||||||
self._verified_bytes: typing.Optional[BytesIO] = None
|
self._verified_bytes: typing.Optional[BytesIO] = None
|
||||||
super().__init__(loop, blob_hash, length, blob_completed_callback, blob_directory)
|
super().__init__(loop, blob_hash, length, blob_completed_callback, blob_directory, added_on, is_mine)
|
||||||
|
|
||||||
@contextlib.contextmanager
|
@contextlib.contextmanager
|
||||||
def _reader_context(self) -> typing.ContextManager[typing.BinaryIO]:
|
def _reader_context(self) -> typing.ContextManager[typing.BinaryIO]:
|
||||||
|
@ -256,9 +277,11 @@ class BlobBuffer(AbstractBlob):
|
||||||
self.verified.clear()
|
self.verified.clear()
|
||||||
|
|
||||||
def _write_blob(self, blob_bytes: bytes):
|
def _write_blob(self, blob_bytes: bytes):
|
||||||
if self._verified_bytes:
|
async def write():
|
||||||
raise OSError("already have bytes for blob")
|
if self._verified_bytes:
|
||||||
self._verified_bytes = BytesIO(blob_bytes)
|
raise OSError("already have bytes for blob")
|
||||||
|
self._verified_bytes = BytesIO(blob_bytes)
|
||||||
|
return self.loop.create_task(write())
|
||||||
|
|
||||||
def delete(self):
|
def delete(self):
|
||||||
if self._verified_bytes:
|
if self._verified_bytes:
|
||||||
|
@ -276,10 +299,12 @@ class BlobFile(AbstractBlob):
|
||||||
"""
|
"""
|
||||||
A blob existing on the local file system
|
A blob existing on the local file system
|
||||||
"""
|
"""
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
def __init__(
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
self, loop: asyncio.AbstractEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
||||||
blob_directory: typing.Optional[str] = None):
|
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
||||||
super().__init__(loop, blob_hash, length, blob_completed_callback, blob_directory)
|
blob_directory: typing.Optional[str] = None, added_on: typing.Optional[int] = None, is_mine: bool = False
|
||||||
|
):
|
||||||
|
super().__init__(loop, blob_hash, length, blob_completed_callback, blob_directory, added_on, is_mine)
|
||||||
if not blob_directory or not os.path.isdir(blob_directory):
|
if not blob_directory or not os.path.isdir(blob_directory):
|
||||||
raise OSError(f"invalid blob directory '{blob_directory}'")
|
raise OSError(f"invalid blob directory '{blob_directory}'")
|
||||||
self.file_path = os.path.join(self.blob_directory, self.blob_hash)
|
self.file_path = os.path.join(self.blob_directory, self.blob_hash)
|
||||||
|
@ -314,22 +339,28 @@ class BlobFile(AbstractBlob):
|
||||||
handle.close()
|
handle.close()
|
||||||
|
|
||||||
def _write_blob(self, blob_bytes: bytes):
|
def _write_blob(self, blob_bytes: bytes):
|
||||||
with open(self.file_path, 'wb') as f:
|
def _write_blob():
|
||||||
f.write(blob_bytes)
|
with open(self.file_path, 'wb') as f:
|
||||||
|
f.write(blob_bytes)
|
||||||
|
|
||||||
|
async def write_blob():
|
||||||
|
await self.loop.run_in_executor(None, _write_blob)
|
||||||
|
|
||||||
|
return self.loop.create_task(write_blob())
|
||||||
|
|
||||||
def delete(self):
|
def delete(self):
|
||||||
|
super().delete()
|
||||||
if os.path.isfile(self.file_path):
|
if os.path.isfile(self.file_path):
|
||||||
os.remove(self.file_path)
|
os.remove(self.file_path)
|
||||||
return super().delete()
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
async def create_from_unencrypted(
|
async def create_from_unencrypted(
|
||||||
cls, loop: asyncio.BaseEventLoop, blob_dir: typing.Optional[str], key: bytes, iv: bytes,
|
cls, loop: asyncio.AbstractEventLoop, blob_dir: typing.Optional[str], key: bytes, iv: bytes,
|
||||||
unencrypted: bytes, blob_num: int,
|
unencrypted: bytes, blob_num: int, added_on: float, is_mine: bool,
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'],
|
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None
|
||||||
asyncio.Task]] = None) -> BlobInfo:
|
) -> BlobInfo:
|
||||||
if not blob_dir or not os.path.isdir(blob_dir):
|
if not blob_dir or not os.path.isdir(blob_dir):
|
||||||
raise OSError(f"cannot create blob in directory: '{blob_dir}'")
|
raise OSError(f"cannot create blob in directory: '{blob_dir}'")
|
||||||
return await super().create_from_unencrypted(
|
return await super().create_from_unencrypted(
|
||||||
loop, blob_dir, key, iv, unencrypted, blob_num, blob_completed_callback
|
loop, blob_dir, key, iv, unencrypted, blob_num, added_on, is_mine, blob_completed_callback
|
||||||
)
|
)
|
|
@ -7,13 +7,19 @@ class BlobInfo:
|
||||||
'blob_num',
|
'blob_num',
|
||||||
'length',
|
'length',
|
||||||
'iv',
|
'iv',
|
||||||
|
'added_on',
|
||||||
|
'is_mine'
|
||||||
]
|
]
|
||||||
|
|
||||||
def __init__(self, blob_num: int, length: int, iv: str, blob_hash: typing.Optional[str] = None):
|
def __init__(
|
||||||
|
self, blob_num: int, length: int, iv: str, added_on,
|
||||||
|
blob_hash: typing.Optional[str] = None, is_mine=False):
|
||||||
self.blob_hash = blob_hash
|
self.blob_hash = blob_hash
|
||||||
self.blob_num = blob_num
|
self.blob_num = blob_num
|
||||||
self.length = length
|
self.length = length
|
||||||
self.iv = iv
|
self.iv = iv
|
||||||
|
self.added_on = added_on
|
||||||
|
self.is_mine = is_mine
|
||||||
|
|
||||||
def as_dict(self) -> typing.Dict:
|
def as_dict(self) -> typing.Dict:
|
||||||
d = {
|
d = {
|
|
@ -2,21 +2,21 @@ import os
|
||||||
import typing
|
import typing
|
||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
from lbrynet.utils import LRUCache
|
from lbry.utils import LRUCacheWithMetrics
|
||||||
from lbrynet.blob.blob_file import is_valid_blobhash, BlobFile, BlobBuffer, AbstractBlob
|
from lbry.blob.blob_file import is_valid_blobhash, BlobFile, BlobBuffer, AbstractBlob
|
||||||
from lbrynet.stream.descriptor import StreamDescriptor
|
from lbry.stream.descriptor import StreamDescriptor
|
||||||
from lbrynet.connection_manager import ConnectionManager
|
from lbry.connection_manager import ConnectionManager
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbrynet.conf import Config
|
from lbry.conf import Config
|
||||||
from lbrynet.dht.protocol.data_store import DictDataStore
|
from lbry.dht.protocol.data_store import DictDataStore
|
||||||
from lbrynet.extras.daemon.storage import SQLiteStorage
|
from lbry.extras.daemon.storage import SQLiteStorage
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class BlobManager:
|
class BlobManager:
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, blob_dir: str, storage: 'SQLiteStorage', config: 'Config',
|
def __init__(self, loop: asyncio.AbstractEventLoop, blob_dir: str, storage: 'SQLiteStorage', config: 'Config',
|
||||||
node_data_store: typing.Optional['DictDataStore'] = None):
|
node_data_store: typing.Optional['DictDataStore'] = None):
|
||||||
"""
|
"""
|
||||||
This class stores blobs on the hard disk
|
This class stores blobs on the hard disk
|
||||||
|
@ -32,47 +32,43 @@ class BlobManager:
|
||||||
else self._node_data_store.completed_blobs
|
else self._node_data_store.completed_blobs
|
||||||
self.blobs: typing.Dict[str, AbstractBlob] = {}
|
self.blobs: typing.Dict[str, AbstractBlob] = {}
|
||||||
self.config = config
|
self.config = config
|
||||||
self.decrypted_blob_lru_cache = None if not self.config.blob_lru_cache_size else LRUCache(
|
self.decrypted_blob_lru_cache = None if not self.config.blob_lru_cache_size else LRUCacheWithMetrics(
|
||||||
self.config.blob_lru_cache_size)
|
self.config.blob_lru_cache_size)
|
||||||
self.connection_manager = ConnectionManager(loop)
|
self.connection_manager = ConnectionManager(loop)
|
||||||
|
|
||||||
def _get_blob(self, blob_hash: str, length: typing.Optional[int] = None):
|
def _get_blob(self, blob_hash: str, length: typing.Optional[int] = None, is_mine: bool = False):
|
||||||
if self.config.save_blobs:
|
if self.config.save_blobs or (
|
||||||
|
is_valid_blobhash(blob_hash) and os.path.isfile(os.path.join(self.blob_dir, blob_hash))):
|
||||||
return BlobFile(
|
return BlobFile(
|
||||||
self.loop, blob_hash, length, self.blob_completed, self.blob_dir
|
self.loop, blob_hash, length, self.blob_completed, self.blob_dir, is_mine=is_mine
|
||||||
)
|
|
||||||
else:
|
|
||||||
if is_valid_blobhash(blob_hash) and os.path.isfile(os.path.join(self.blob_dir, blob_hash)):
|
|
||||||
return BlobFile(
|
|
||||||
self.loop, blob_hash, length, self.blob_completed, self.blob_dir
|
|
||||||
)
|
|
||||||
return BlobBuffer(
|
|
||||||
self.loop, blob_hash, length, self.blob_completed, self.blob_dir
|
|
||||||
)
|
)
|
||||||
|
return BlobBuffer(
|
||||||
|
self.loop, blob_hash, length, self.blob_completed, self.blob_dir, is_mine=is_mine
|
||||||
|
)
|
||||||
|
|
||||||
def get_blob(self, blob_hash, length: typing.Optional[int] = None):
|
def get_blob(self, blob_hash, length: typing.Optional[int] = None, is_mine: bool = False):
|
||||||
if blob_hash in self.blobs:
|
if blob_hash in self.blobs:
|
||||||
if self.config.save_blobs and isinstance(self.blobs[blob_hash], BlobBuffer):
|
if self.config.save_blobs and isinstance(self.blobs[blob_hash], BlobBuffer):
|
||||||
buffer = self.blobs.pop(blob_hash)
|
buffer = self.blobs.pop(blob_hash)
|
||||||
if blob_hash in self.completed_blob_hashes:
|
if blob_hash in self.completed_blob_hashes:
|
||||||
self.completed_blob_hashes.remove(blob_hash)
|
self.completed_blob_hashes.remove(blob_hash)
|
||||||
self.blobs[blob_hash] = self._get_blob(blob_hash, length)
|
self.blobs[blob_hash] = self._get_blob(blob_hash, length, is_mine)
|
||||||
if buffer.is_readable():
|
if buffer.is_readable():
|
||||||
with buffer.reader_context() as reader:
|
with buffer.reader_context() as reader:
|
||||||
self.blobs[blob_hash].write_blob(reader.read())
|
self.blobs[blob_hash].write_blob(reader.read())
|
||||||
if length and self.blobs[blob_hash].length is None:
|
if length and self.blobs[blob_hash].length is None:
|
||||||
self.blobs[blob_hash].set_length(length)
|
self.blobs[blob_hash].set_length(length)
|
||||||
else:
|
else:
|
||||||
self.blobs[blob_hash] = self._get_blob(blob_hash, length)
|
self.blobs[blob_hash] = self._get_blob(blob_hash, length, is_mine)
|
||||||
return self.blobs[blob_hash]
|
return self.blobs[blob_hash]
|
||||||
|
|
||||||
def is_blob_verified(self, blob_hash: str, length: typing.Optional[int] = None) -> bool:
|
def is_blob_verified(self, blob_hash: str, length: typing.Optional[int] = None) -> bool:
|
||||||
if not is_valid_blobhash(blob_hash):
|
if not is_valid_blobhash(blob_hash):
|
||||||
raise ValueError(blob_hash)
|
raise ValueError(blob_hash)
|
||||||
if blob_hash in self.blobs:
|
|
||||||
return self.blobs[blob_hash].get_is_verified()
|
|
||||||
if not os.path.isfile(os.path.join(self.blob_dir, blob_hash)):
|
if not os.path.isfile(os.path.join(self.blob_dir, blob_hash)):
|
||||||
return False
|
return False
|
||||||
|
if blob_hash in self.blobs:
|
||||||
|
return self.blobs[blob_hash].get_is_verified()
|
||||||
return self._get_blob(blob_hash, length).get_is_verified()
|
return self._get_blob(blob_hash, length).get_is_verified()
|
||||||
|
|
||||||
async def setup(self) -> bool:
|
async def setup(self) -> bool:
|
||||||
|
@ -82,11 +78,15 @@ class BlobManager:
|
||||||
return {
|
return {
|
||||||
item.name for item in os.scandir(self.blob_dir) if is_valid_blobhash(item.name)
|
item.name for item in os.scandir(self.blob_dir) if is_valid_blobhash(item.name)
|
||||||
}
|
}
|
||||||
|
|
||||||
in_blobfiles_dir = await self.loop.run_in_executor(None, get_files_in_blob_dir)
|
in_blobfiles_dir = await self.loop.run_in_executor(None, get_files_in_blob_dir)
|
||||||
to_add = await self.storage.sync_missing_blobs(in_blobfiles_dir)
|
to_add = await self.storage.sync_missing_blobs(in_blobfiles_dir)
|
||||||
if to_add:
|
if to_add:
|
||||||
self.completed_blob_hashes.update(to_add)
|
self.completed_blob_hashes.update(to_add)
|
||||||
self.connection_manager.start()
|
# check blobs that aren't set as finished but were seen on disk
|
||||||
|
await self.ensure_completed_blobs_status(in_blobfiles_dir - to_add)
|
||||||
|
if self.config.track_bandwidth:
|
||||||
|
self.connection_manager.start()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def stop(self):
|
def stop(self):
|
||||||
|
@ -107,13 +107,26 @@ class BlobManager:
|
||||||
if isinstance(blob, BlobFile):
|
if isinstance(blob, BlobFile):
|
||||||
if blob.blob_hash not in self.completed_blob_hashes:
|
if blob.blob_hash not in self.completed_blob_hashes:
|
||||||
self.completed_blob_hashes.add(blob.blob_hash)
|
self.completed_blob_hashes.add(blob.blob_hash)
|
||||||
return self.loop.create_task(self.storage.add_blobs((blob.blob_hash, blob.length), finished=True))
|
return self.loop.create_task(self.storage.add_blobs(
|
||||||
|
(blob.blob_hash, blob.length, blob.added_on, blob.is_mine), finished=True)
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
return self.loop.create_task(self.storage.add_blobs((blob.blob_hash, blob.length), finished=False))
|
return self.loop.create_task(self.storage.add_blobs(
|
||||||
|
(blob.blob_hash, blob.length, blob.added_on, blob.is_mine), finished=False)
|
||||||
|
)
|
||||||
|
|
||||||
def check_completed_blobs(self, blob_hashes: typing.List[str]) -> typing.List[str]:
|
async def ensure_completed_blobs_status(self, blob_hashes: typing.Iterable[str]):
|
||||||
"""Returns of the blobhashes_to_check, which are valid"""
|
"""Ensures that completed blobs from a given list of blob hashes are set as 'finished' in the database."""
|
||||||
return [blob_hash for blob_hash in blob_hashes if self.is_blob_verified(blob_hash)]
|
to_add = []
|
||||||
|
for blob_hash in blob_hashes:
|
||||||
|
if not self.is_blob_verified(blob_hash):
|
||||||
|
continue
|
||||||
|
blob = self.get_blob(blob_hash)
|
||||||
|
to_add.append((blob.blob_hash, blob.length, blob.added_on, blob.is_mine))
|
||||||
|
if len(to_add) > 500:
|
||||||
|
await self.storage.add_blobs(*to_add, finished=True)
|
||||||
|
to_add.clear()
|
||||||
|
return await self.storage.add_blobs(*to_add, finished=True)
|
||||||
|
|
||||||
def delete_blob(self, blob_hash: str):
|
def delete_blob(self, blob_hash: str):
|
||||||
if not is_valid_blobhash(blob_hash):
|
if not is_valid_blobhash(blob_hash):
|
77
lbry/blob/disk_space_manager.py
Normal file
77
lbry/blob/disk_space_manager.py
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
import asyncio
|
||||||
|
import logging
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class DiskSpaceManager:
|
||||||
|
|
||||||
|
def __init__(self, config, db, blob_manager, cleaning_interval=30 * 60, analytics=None):
|
||||||
|
self.config = config
|
||||||
|
self.db = db
|
||||||
|
self.blob_manager = blob_manager
|
||||||
|
self.cleaning_interval = cleaning_interval
|
||||||
|
self.running = False
|
||||||
|
self.task = None
|
||||||
|
self.analytics = analytics
|
||||||
|
self._used_space_bytes = None
|
||||||
|
|
||||||
|
async def get_free_space_mb(self, is_network_blob=False):
|
||||||
|
limit_mb = self.config.network_storage_limit if is_network_blob else self.config.blob_storage_limit
|
||||||
|
space_used_mb = await self.get_space_used_mb()
|
||||||
|
space_used_mb = space_used_mb['network_storage'] if is_network_blob else space_used_mb['content_storage']
|
||||||
|
return max(0, limit_mb - space_used_mb)
|
||||||
|
|
||||||
|
async def get_space_used_bytes(self):
|
||||||
|
self._used_space_bytes = await self.db.get_stored_blob_disk_usage()
|
||||||
|
return self._used_space_bytes
|
||||||
|
|
||||||
|
async def get_space_used_mb(self, cached=True):
|
||||||
|
cached = cached and self._used_space_bytes is not None
|
||||||
|
space_used_bytes = self._used_space_bytes if cached else await self.get_space_used_bytes()
|
||||||
|
return {key: int(value/1024.0/1024.0) for key, value in space_used_bytes.items()}
|
||||||
|
|
||||||
|
async def clean(self):
|
||||||
|
await self._clean(False)
|
||||||
|
await self._clean(True)
|
||||||
|
|
||||||
|
async def _clean(self, is_network_blob=False):
|
||||||
|
space_used_mb = await self.get_space_used_mb(cached=False)
|
||||||
|
if is_network_blob:
|
||||||
|
space_used_mb = space_used_mb['network_storage']
|
||||||
|
else:
|
||||||
|
space_used_mb = space_used_mb['content_storage'] + space_used_mb['private_storage']
|
||||||
|
storage_limit_mb = self.config.network_storage_limit if is_network_blob else self.config.blob_storage_limit
|
||||||
|
if self.analytics:
|
||||||
|
asyncio.create_task(
|
||||||
|
self.analytics.send_disk_space_used(space_used_mb, storage_limit_mb, is_network_blob)
|
||||||
|
)
|
||||||
|
delete = []
|
||||||
|
available = storage_limit_mb - space_used_mb
|
||||||
|
if storage_limit_mb == 0 if not is_network_blob else available >= 0:
|
||||||
|
return 0
|
||||||
|
for blob_hash, file_size, _ in await self.db.get_stored_blobs(is_mine=False, is_network_blob=is_network_blob):
|
||||||
|
delete.append(blob_hash)
|
||||||
|
available += int(file_size/1024.0/1024.0)
|
||||||
|
if available >= 0:
|
||||||
|
break
|
||||||
|
if delete:
|
||||||
|
await self.db.stop_all_files()
|
||||||
|
await self.blob_manager.delete_blobs(delete, delete_from_db=True)
|
||||||
|
self._used_space_bytes = None
|
||||||
|
return len(delete)
|
||||||
|
|
||||||
|
async def cleaning_loop(self):
|
||||||
|
while self.running:
|
||||||
|
await asyncio.sleep(self.cleaning_interval)
|
||||||
|
await self.clean()
|
||||||
|
|
||||||
|
async def start(self):
|
||||||
|
self.running = True
|
||||||
|
self.task = asyncio.create_task(self.cleaning_loop())
|
||||||
|
self.task.add_done_callback(lambda _: log.info("Stopping blob cleanup service."))
|
||||||
|
|
||||||
|
async def stop(self):
|
||||||
|
if self.running:
|
||||||
|
self.running = False
|
||||||
|
self.task.cancel()
|
|
@ -2,8 +2,8 @@ import typing
|
||||||
import logging
|
import logging
|
||||||
import asyncio
|
import asyncio
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
from lbrynet.error import InvalidBlobHashError, InvalidDataError
|
from lbry.error import InvalidBlobHashError, InvalidDataError
|
||||||
from lbrynet.cryptoutils import get_lbry_hash_obj
|
from lbry.utils import get_lbry_hash_obj
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -33,13 +33,13 @@ class HashBlobWriter:
|
||||||
def write(self, data: bytes):
|
def write(self, data: bytes):
|
||||||
expected_length = self.get_length()
|
expected_length = self.get_length()
|
||||||
if not expected_length:
|
if not expected_length:
|
||||||
raise IOError("unknown blob length")
|
raise OSError("unknown blob length")
|
||||||
if self.buffer is None:
|
if self.buffer is None:
|
||||||
log.warning("writer has already been closed")
|
log.warning("writer has already been closed")
|
||||||
if not self.finished.done():
|
if not self.finished.done():
|
||||||
self.finished.cancel()
|
self.finished.cancel()
|
||||||
return
|
return
|
||||||
raise IOError('I/O operation on closed file')
|
raise OSError('I/O operation on closed file')
|
||||||
|
|
||||||
self._hashsum.update(data)
|
self._hashsum.update(data)
|
||||||
self.len_so_far += len(data)
|
self.len_so_far += len(data)
|
|
@ -1,20 +1,22 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import time
|
||||||
import logging
|
import logging
|
||||||
import typing
|
import typing
|
||||||
import binascii
|
import binascii
|
||||||
from lbrynet.error import InvalidBlobHashError, InvalidDataError
|
from typing import Optional
|
||||||
from lbrynet.blob_exchange.serialization import BlobResponse, BlobRequest
|
from lbry.error import InvalidBlobHashError, InvalidDataError
|
||||||
from lbrynet.utils import cache_concurrent
|
from lbry.blob_exchange.serialization import BlobResponse, BlobRequest
|
||||||
|
from lbry.utils import cache_concurrent
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbrynet.blob.blob_file import AbstractBlob
|
from lbry.blob.blob_file import AbstractBlob
|
||||||
from lbrynet.blob.writer import HashBlobWriter
|
from lbry.blob.writer import HashBlobWriter
|
||||||
from lbrynet.connection_manager import ConnectionManager
|
from lbry.connection_manager import ConnectionManager
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class BlobExchangeClientProtocol(asyncio.Protocol):
|
class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, peer_timeout: typing.Optional[float] = 10,
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_timeout: typing.Optional[float] = 10,
|
||||||
connection_manager: typing.Optional['ConnectionManager'] = None):
|
connection_manager: typing.Optional['ConnectionManager'] = None):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.peer_port: typing.Optional[int] = None
|
self.peer_port: typing.Optional[int] = None
|
||||||
|
@ -30,7 +32,7 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.buf = b''
|
self.buf = b''
|
||||||
|
|
||||||
# this is here to handle the race when the downloader is closed right as response_fut gets a result
|
# this is here to handle the race when the downloader is closed right as response_fut gets a result
|
||||||
self.closed = asyncio.Event(loop=self.loop)
|
self.closed = asyncio.Event()
|
||||||
|
|
||||||
def data_received(self, data: bytes):
|
def data_received(self, data: bytes):
|
||||||
if self.connection_manager:
|
if self.connection_manager:
|
||||||
|
@ -39,8 +41,6 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.peer_address, self.peer_port = addr_info
|
self.peer_address, self.peer_port = addr_info
|
||||||
# assert self.peer_address is not None
|
# assert self.peer_address is not None
|
||||||
self.connection_manager.received_data(f"{self.peer_address}:{self.peer_port}", len(data))
|
self.connection_manager.received_data(f"{self.peer_address}:{self.peer_port}", len(data))
|
||||||
#log.debug("%s:%d -- got %s bytes -- %s bytes on buffer -- %s blob bytes received",
|
|
||||||
# self.peer_address, self.peer_port, len(data), len(self.buf), self._blob_bytes_received)
|
|
||||||
if not self.transport or self.transport.is_closing():
|
if not self.transport or self.transport.is_closing():
|
||||||
log.warning("transport closing, but got more bytes from %s:%i\n%s", self.peer_address, self.peer_port,
|
log.warning("transport closing, but got more bytes from %s:%i\n%s", self.peer_address, self.peer_port,
|
||||||
binascii.hexlify(data))
|
binascii.hexlify(data))
|
||||||
|
@ -67,7 +67,8 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.blob.set_length(blob_response.length)
|
self.blob.set_length(blob_response.length)
|
||||||
elif blob_response and not blob_response.error and self.blob.blob_hash != blob_response.blob_hash:
|
elif blob_response and not blob_response.error and self.blob.blob_hash != blob_response.blob_hash:
|
||||||
# the server started sending a blob we didn't request
|
# the server started sending a blob we didn't request
|
||||||
log.warning("mismatch with self.blob %s", self.blob.blob_hash)
|
log.warning("%s started sending blob we didn't request %s instead of %s", self.peer_address,
|
||||||
|
blob_response.blob_hash, self.blob.blob_hash)
|
||||||
return
|
return
|
||||||
if response.responses:
|
if response.responses:
|
||||||
log.debug("got response from %s:%i <- %s", self.peer_address, self.peer_port, response.to_dict())
|
log.debug("got response from %s:%i <- %s", self.peer_address, self.peer_port, response.to_dict())
|
||||||
|
@ -82,24 +83,23 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
if len(data) > (self.blob.get_length() - self._blob_bytes_received):
|
if len(data) > (self.blob.get_length() - self._blob_bytes_received):
|
||||||
data = data[:(self.blob.get_length() - self._blob_bytes_received)]
|
data = data[:(self.blob.get_length() - self._blob_bytes_received)]
|
||||||
log.warning("got more than asked from %s:%d, probable sendfile bug", self.peer_address, self.peer_port)
|
log.warning("got more than asked from %s:%d, probable sendfile bug", self.peer_address, self.peer_port)
|
||||||
else:
|
|
||||||
data = data
|
|
||||||
self._blob_bytes_received += len(data)
|
self._blob_bytes_received += len(data)
|
||||||
try:
|
try:
|
||||||
self.writer.write(data)
|
self.writer.write(data)
|
||||||
except IOError as err:
|
except OSError as err:
|
||||||
log.error("error downloading blob from %s:%i: %s", self.peer_address, self.peer_port, err)
|
log.error("error downloading blob from %s:%i: %s", self.peer_address, self.peer_port, err)
|
||||||
if self._response_fut and not self._response_fut.done():
|
if self._response_fut and not self._response_fut.done():
|
||||||
self._response_fut.set_exception(err)
|
self._response_fut.set_exception(err)
|
||||||
except (asyncio.TimeoutError) as err: # TODO: is this needed?
|
except asyncio.TimeoutError as err:
|
||||||
log.error("%s downloading blob from %s:%i", str(err), self.peer_address, self.peer_port)
|
log.error("%s downloading blob from %s:%i", str(err), self.peer_address, self.peer_port)
|
||||||
if self._response_fut and not self._response_fut.done():
|
if self._response_fut and not self._response_fut.done():
|
||||||
self._response_fut.set_exception(err)
|
self._response_fut.set_exception(err)
|
||||||
|
|
||||||
async def _download_blob(self) -> typing.Tuple[int, typing.Optional[asyncio.Transport]]:
|
async def _download_blob(self) -> typing.Tuple[int, Optional['BlobExchangeClientProtocol']]: # pylint: disable=too-many-return-statements
|
||||||
"""
|
"""
|
||||||
:return: download success (bool), keep connection (bool)
|
:return: download success (bool), connected protocol (BlobExchangeClientProtocol)
|
||||||
"""
|
"""
|
||||||
|
start_time = time.perf_counter()
|
||||||
request = BlobRequest.make_request_for_blob_hash(self.blob.blob_hash)
|
request = BlobRequest.make_request_for_blob_hash(self.blob.blob_hash)
|
||||||
blob_hash = self.blob.blob_hash
|
blob_hash = self.blob.blob_hash
|
||||||
if not self.peer_address:
|
if not self.peer_address:
|
||||||
|
@ -111,7 +111,7 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.transport.write(msg)
|
self.transport.write(msg)
|
||||||
if self.connection_manager:
|
if self.connection_manager:
|
||||||
self.connection_manager.sent_data(f"{self.peer_address}:{self.peer_port}", len(msg))
|
self.connection_manager.sent_data(f"{self.peer_address}:{self.peer_port}", len(msg))
|
||||||
response: BlobResponse = await asyncio.wait_for(self._response_fut, self.peer_timeout, loop=self.loop)
|
response: BlobResponse = await asyncio.wait_for(self._response_fut, self.peer_timeout)
|
||||||
availability_response = response.get_availability_response()
|
availability_response = response.get_availability_response()
|
||||||
price_response = response.get_price_response()
|
price_response = response.get_price_response()
|
||||||
blob_response = response.get_blob_response()
|
blob_response = response.get_blob_response()
|
||||||
|
@ -125,16 +125,21 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.peer_port)
|
self.peer_port)
|
||||||
log.warning(response.to_dict())
|
log.warning(response.to_dict())
|
||||||
return self._blob_bytes_received, self.close()
|
return self._blob_bytes_received, self.close()
|
||||||
elif availability_response.available_blobs and \
|
elif availability_response and availability_response.available_blobs and \
|
||||||
availability_response.available_blobs != [self.blob.blob_hash]:
|
availability_response.available_blobs != [self.blob.blob_hash]:
|
||||||
log.warning("blob availability response doesn't match our request from %s:%i",
|
log.warning("blob availability response doesn't match our request from %s:%i",
|
||||||
self.peer_address, self.peer_port)
|
self.peer_address, self.peer_port)
|
||||||
return self._blob_bytes_received, self.close()
|
return self._blob_bytes_received, self.close()
|
||||||
|
elif not availability_response:
|
||||||
|
log.warning("response from %s:%i did not include an availability response (we requested %s)",
|
||||||
|
self.peer_address, self.peer_port, blob_hash)
|
||||||
|
return self._blob_bytes_received, self.close()
|
||||||
|
|
||||||
if not price_response or price_response.blob_data_payment_rate != 'RATE_ACCEPTED':
|
if not price_response or price_response.blob_data_payment_rate != 'RATE_ACCEPTED':
|
||||||
log.warning("data rate rejected by %s:%i", self.peer_address, self.peer_port)
|
log.warning("data rate rejected by %s:%i", self.peer_address, self.peer_port)
|
||||||
return self._blob_bytes_received, self.close()
|
return self._blob_bytes_received, self.close()
|
||||||
if not blob_response or blob_response.error:
|
if not blob_response or blob_response.error:
|
||||||
log.warning("blob cant be downloaded from %s:%i", self.peer_address, self.peer_port)
|
log.warning("blob can't be downloaded from %s:%i", self.peer_address, self.peer_port)
|
||||||
return self._blob_bytes_received, self.close()
|
return self._blob_bytes_received, self.close()
|
||||||
if not blob_response.error and blob_response.blob_hash != self.blob.blob_hash:
|
if not blob_response.error and blob_response.blob_hash != self.blob.blob_hash:
|
||||||
log.warning("incoming blob hash mismatch from %s:%i", self.peer_address, self.peer_port)
|
log.warning("incoming blob hash mismatch from %s:%i", self.peer_address, self.peer_port)
|
||||||
|
@ -146,10 +151,14 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
f" timeout in {self.peer_timeout}"
|
f" timeout in {self.peer_timeout}"
|
||||||
log.debug(msg)
|
log.debug(msg)
|
||||||
msg = f"downloaded {self.blob.blob_hash[:8]} from {self.peer_address}:{self.peer_port}"
|
msg = f"downloaded {self.blob.blob_hash[:8]} from {self.peer_address}:{self.peer_port}"
|
||||||
await asyncio.wait_for(self.writer.finished, self.peer_timeout, loop=self.loop)
|
await asyncio.wait_for(self.writer.finished, self.peer_timeout)
|
||||||
log.info(msg)
|
# wait for the io to finish
|
||||||
|
await self.blob.verified.wait()
|
||||||
|
log.info("%s at %fMB/s", msg,
|
||||||
|
round((float(self._blob_bytes_received) /
|
||||||
|
float(time.perf_counter() - start_time)) / 1000000.0, 2))
|
||||||
# await self.blob.finished_writing.wait() not necessary, but a dangerous change. TODO: is it needed?
|
# await self.blob.finished_writing.wait() not necessary, but a dangerous change. TODO: is it needed?
|
||||||
return self._blob_bytes_received, self.transport
|
return self._blob_bytes_received, self
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
return self._blob_bytes_received, self.close()
|
return self._blob_bytes_received, self.close()
|
||||||
except (InvalidBlobHashError, InvalidDataError):
|
except (InvalidBlobHashError, InvalidDataError):
|
||||||
|
@ -170,19 +179,19 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.transport = None
|
self.transport = None
|
||||||
self.buf = b''
|
self.buf = b''
|
||||||
|
|
||||||
async def download_blob(self, blob: 'AbstractBlob') -> typing.Tuple[int, typing.Optional[asyncio.Transport]]:
|
async def download_blob(self, blob: 'AbstractBlob') -> typing.Tuple[int, Optional['BlobExchangeClientProtocol']]:
|
||||||
self.closed.clear()
|
self.closed.clear()
|
||||||
blob_hash = blob.blob_hash
|
blob_hash = blob.blob_hash
|
||||||
if blob.get_is_verified() or not blob.is_writeable():
|
if blob.get_is_verified() or not blob.is_writeable():
|
||||||
return 0, self.transport
|
return 0, self
|
||||||
try:
|
try:
|
||||||
self._blob_bytes_received = 0
|
self._blob_bytes_received = 0
|
||||||
self.blob, self.writer = blob, blob.get_blob_writer(self.peer_address, self.peer_port)
|
self.blob, self.writer = blob, blob.get_blob_writer(self.peer_address, self.peer_port)
|
||||||
self._response_fut = asyncio.Future(loop=self.loop)
|
self._response_fut = asyncio.Future()
|
||||||
return await self._download_blob()
|
return await self._download_blob()
|
||||||
except OSError as e:
|
except OSError:
|
||||||
# i'm not sure how to fix this race condition - jack
|
# i'm not sure how to fix this race condition - jack
|
||||||
log.warning("race happened downloading %s from %s:%i", blob_hash, self.peer_address, self.peer_port)
|
log.warning("race happened downloading %s from %s:%s", blob_hash, self.peer_address, self.peer_port)
|
||||||
# return self._blob_bytes_received, self.transport
|
# return self._blob_bytes_received, self.transport
|
||||||
raise
|
raise
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
|
@ -206,40 +215,41 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.connection_manager.connection_made(f"{self.peer_address}:{self.peer_port}")
|
self.connection_manager.connection_made(f"{self.peer_address}:{self.peer_port}")
|
||||||
log.debug("connection made to %s:%i", self.peer_address, self.peer_port)
|
log.debug("connection made to %s:%i", self.peer_address, self.peer_port)
|
||||||
|
|
||||||
def connection_lost(self, reason):
|
def connection_lost(self, exc):
|
||||||
if self.connection_manager:
|
if self.connection_manager:
|
||||||
self.connection_manager.outgoing_connection_lost(f"{self.peer_address}:{self.peer_port}")
|
self.connection_manager.outgoing_connection_lost(f"{self.peer_address}:{self.peer_port}")
|
||||||
log.debug("connection lost to %s:%i (reason: %s, %s)", self.peer_address, self.peer_port, str(reason),
|
log.debug("connection lost to %s:%i (reason: %s, %s)", self.peer_address, self.peer_port, str(exc),
|
||||||
str(type(reason)))
|
str(type(exc)))
|
||||||
self.close()
|
self.close()
|
||||||
|
|
||||||
|
|
||||||
@cache_concurrent
|
@cache_concurrent
|
||||||
async def request_blob(loop: asyncio.BaseEventLoop, blob: 'AbstractBlob', address: str, tcp_port: int,
|
async def request_blob(loop: asyncio.AbstractEventLoop, blob: Optional['AbstractBlob'], address: str,
|
||||||
peer_connect_timeout: float, blob_download_timeout: float,
|
tcp_port: int, peer_connect_timeout: float, blob_download_timeout: float,
|
||||||
connected_transport: asyncio.Transport = None, connection_id: int = 0,
|
connected_protocol: Optional['BlobExchangeClientProtocol'] = None,
|
||||||
connection_manager: typing.Optional['ConnectionManager'] = None)\
|
connection_id: int = 0, connection_manager: Optional['ConnectionManager'] = None)\
|
||||||
-> typing.Tuple[int, typing.Optional[asyncio.Transport]]:
|
-> typing.Tuple[int, Optional['BlobExchangeClientProtocol']]:
|
||||||
"""
|
"""
|
||||||
Returns [<downloaded blob>, <keep connection>]
|
Returns [<amount of bytes received>, <client protocol if connected>]
|
||||||
"""
|
"""
|
||||||
|
|
||||||
protocol = BlobExchangeClientProtocol(
|
protocol = connected_protocol
|
||||||
loop, blob_download_timeout, connection_manager
|
if not connected_protocol or not connected_protocol.transport or connected_protocol.transport.is_closing():
|
||||||
)
|
connected_protocol = None
|
||||||
if connected_transport and not connected_transport.is_closing():
|
protocol = BlobExchangeClientProtocol(
|
||||||
connected_transport.set_protocol(protocol)
|
loop, blob_download_timeout, connection_manager
|
||||||
protocol.transport = connected_transport
|
)
|
||||||
log.debug("reusing connection for %s:%d", address, tcp_port)
|
|
||||||
else:
|
else:
|
||||||
connected_transport = None
|
log.debug("reusing connection for %s:%d", address, tcp_port)
|
||||||
try:
|
try:
|
||||||
if not connected_transport:
|
if not connected_protocol:
|
||||||
await asyncio.wait_for(loop.create_connection(lambda: protocol, address, tcp_port),
|
await asyncio.wait_for(loop.create_connection(lambda: protocol, address, tcp_port),
|
||||||
peer_connect_timeout, loop=loop)
|
peer_connect_timeout)
|
||||||
if blob.get_is_verified() or not blob.is_writeable():
|
connected_protocol = protocol
|
||||||
|
if blob is None or blob.get_is_verified() or not blob.is_writeable():
|
||||||
|
# blob is None happens when we are just opening a connection
|
||||||
# file exists but not verified means someone is writing right now, give it time, come back later
|
# file exists but not verified means someone is writing right now, give it time, come back later
|
||||||
return 0, connected_transport
|
return 0, connected_protocol
|
||||||
return await protocol.download_blob(blob)
|
return await connected_protocol.download_blob(blob)
|
||||||
except (asyncio.TimeoutError, ConnectionRefusedError, ConnectionAbortedError, OSError):
|
except (asyncio.TimeoutError, ConnectionRefusedError, ConnectionAbortedError, OSError):
|
||||||
return 0, None
|
return 0, None
|
|
@ -1,14 +1,16 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import typing
|
import typing
|
||||||
import logging
|
import logging
|
||||||
from lbrynet.utils import cache_concurrent
|
from lbry.utils import cache_concurrent
|
||||||
from lbrynet.blob_exchange.client import request_blob
|
from lbry.blob_exchange.client import request_blob
|
||||||
|
from lbry.dht.node import get_kademlia_peers_from_hosts
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbrynet.conf import Config
|
from lbry.conf import Config
|
||||||
from lbrynet.dht.node import Node
|
from lbry.dht.node import Node
|
||||||
from lbrynet.dht.peer import KademliaPeer
|
from lbry.dht.peer import KademliaPeer
|
||||||
from lbrynet.blob.blob_manager import BlobManager
|
from lbry.blob.blob_manager import BlobManager
|
||||||
from lbrynet.blob.blob_file import AbstractBlob
|
from lbry.blob.blob_file import AbstractBlob
|
||||||
|
from lbry.blob_exchange.client import BlobExchangeClientProtocol
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -16,7 +18,7 @@ log = logging.getLogger(__name__)
|
||||||
class BlobDownloader:
|
class BlobDownloader:
|
||||||
BAN_FACTOR = 2.0 # fixme: when connection manager gets implemented, move it out from here
|
BAN_FACTOR = 2.0 # fixme: when connection manager gets implemented, move it out from here
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, config: 'Config', blob_manager: 'BlobManager',
|
def __init__(self, loop: asyncio.AbstractEventLoop, config: 'Config', blob_manager: 'BlobManager',
|
||||||
peer_queue: asyncio.Queue):
|
peer_queue: asyncio.Queue):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.config = config
|
self.config = config
|
||||||
|
@ -26,41 +28,44 @@ class BlobDownloader:
|
||||||
self.ignored: typing.Dict['KademliaPeer', int] = {}
|
self.ignored: typing.Dict['KademliaPeer', int] = {}
|
||||||
self.scores: typing.Dict['KademliaPeer', int] = {}
|
self.scores: typing.Dict['KademliaPeer', int] = {}
|
||||||
self.failures: typing.Dict['KademliaPeer', int] = {}
|
self.failures: typing.Dict['KademliaPeer', int] = {}
|
||||||
self.connections: typing.Dict['KademliaPeer', asyncio.Transport] = {}
|
self.connection_failures: typing.Set['KademliaPeer'] = set()
|
||||||
self.is_running = asyncio.Event(loop=self.loop)
|
self.connections: typing.Dict['KademliaPeer', 'BlobExchangeClientProtocol'] = {}
|
||||||
|
self.is_running = asyncio.Event()
|
||||||
|
|
||||||
def should_race_continue(self, blob: 'AbstractBlob'):
|
def should_race_continue(self, blob: 'AbstractBlob'):
|
||||||
if len(self.active_connections) >= self.config.max_connections_per_download:
|
max_probes = self.config.max_connections_per_download * (1 if self.connections else 10)
|
||||||
|
if len(self.active_connections) >= max_probes:
|
||||||
return False
|
return False
|
||||||
return not (blob.get_is_verified() or not blob.is_writeable())
|
return not (blob.get_is_verified() or not blob.is_writeable())
|
||||||
|
|
||||||
async def request_blob_from_peer(self, blob: 'AbstractBlob', peer: 'KademliaPeer', connection_id: int = 0):
|
async def request_blob_from_peer(self, blob: 'AbstractBlob', peer: 'KademliaPeer', connection_id: int = 0,
|
||||||
|
just_probe: bool = False):
|
||||||
if blob.get_is_verified():
|
if blob.get_is_verified():
|
||||||
return
|
return
|
||||||
transport = self.connections.get(peer)
|
|
||||||
start = self.loop.time()
|
start = self.loop.time()
|
||||||
bytes_received, transport = await request_blob(
|
bytes_received, protocol = await request_blob(
|
||||||
self.loop, blob, peer.address, peer.tcp_port, self.config.peer_connect_timeout,
|
self.loop, blob if not just_probe else None, peer.address, peer.tcp_port, self.config.peer_connect_timeout,
|
||||||
self.config.blob_download_timeout, connected_transport=transport, connection_id=connection_id,
|
self.config.blob_download_timeout, connected_protocol=self.connections.get(peer),
|
||||||
connection_manager=self.blob_manager.connection_manager
|
connection_id=connection_id, connection_manager=self.blob_manager.connection_manager
|
||||||
|
|
||||||
)
|
)
|
||||||
if not transport and peer not in self.ignored:
|
if not bytes_received and not protocol and peer not in self.connection_failures:
|
||||||
|
self.connection_failures.add(peer)
|
||||||
|
if not protocol and peer not in self.ignored:
|
||||||
self.ignored[peer] = self.loop.time()
|
self.ignored[peer] = self.loop.time()
|
||||||
log.debug("drop peer %s:%i", peer.address, peer.tcp_port)
|
log.debug("drop peer %s:%i", peer.address, peer.tcp_port)
|
||||||
self.failures[peer] = self.failures.get(peer, 0) + 1
|
self.failures[peer] = self.failures.get(peer, 0) + 1
|
||||||
if peer in self.connections:
|
if peer in self.connections:
|
||||||
del self.connections[peer]
|
del self.connections[peer]
|
||||||
elif transport:
|
elif protocol:
|
||||||
log.debug("keep peer %s:%i", peer.address, peer.tcp_port)
|
log.debug("keep peer %s:%i", peer.address, peer.tcp_port)
|
||||||
self.failures[peer] = 0
|
self.failures[peer] = 0
|
||||||
self.connections[peer] = transport
|
self.connections[peer] = protocol
|
||||||
elapsed = self.loop.time() - start
|
elapsed = self.loop.time() - start
|
||||||
self.scores[peer] = bytes_received / elapsed if bytes_received and elapsed else 1
|
self.scores[peer] = bytes_received / elapsed if bytes_received and elapsed else 1
|
||||||
|
|
||||||
async def new_peer_or_finished(self):
|
async def new_peer_or_finished(self):
|
||||||
active_tasks = list(self.active_connections.values()) + [asyncio.sleep(1)]
|
active_tasks = list(self.active_connections.values()) + [asyncio.create_task(asyncio.sleep(1))]
|
||||||
await asyncio.wait(active_tasks, loop=self.loop, return_when='FIRST_COMPLETED')
|
await asyncio.wait(active_tasks, return_when='FIRST_COMPLETED')
|
||||||
|
|
||||||
def cleanup_active(self):
|
def cleanup_active(self):
|
||||||
if not self.active_connections and not self.connections:
|
if not self.active_connections and not self.connections:
|
||||||
|
@ -71,10 +76,10 @@ class BlobDownloader:
|
||||||
|
|
||||||
def clearbanned(self):
|
def clearbanned(self):
|
||||||
now = self.loop.time()
|
now = self.loop.time()
|
||||||
self.ignored = dict((
|
self.ignored = {
|
||||||
(peer, when) for (peer, when) in self.ignored.items()
|
peer: when for (peer, when) in self.ignored.items()
|
||||||
if (now - when) < min(30.0, (self.failures.get(peer, 0) ** self.BAN_FACTOR))
|
if (now - when) < min(30.0, (self.failures.get(peer, 0) ** self.BAN_FACTOR))
|
||||||
))
|
}
|
||||||
|
|
||||||
@cache_concurrent
|
@cache_concurrent
|
||||||
async def download_blob(self, blob_hash: str, length: typing.Optional[int] = None,
|
async def download_blob(self, blob_hash: str, length: typing.Optional[int] = None,
|
||||||
|
@ -85,42 +90,48 @@ class BlobDownloader:
|
||||||
self.is_running.set()
|
self.is_running.set()
|
||||||
try:
|
try:
|
||||||
while not blob.get_is_verified() and self.is_running.is_set():
|
while not blob.get_is_verified() and self.is_running.is_set():
|
||||||
batch: typing.Set['KademliaPeer'] = set()
|
batch: typing.Set['KademliaPeer'] = set(self.connections.keys())
|
||||||
while not self.peer_queue.empty():
|
while not self.peer_queue.empty():
|
||||||
batch.update(self.peer_queue.get_nowait())
|
batch.update(self.peer_queue.get_nowait())
|
||||||
if batch:
|
|
||||||
self.peer_queue.put_nowait(list(batch))
|
|
||||||
log.debug(
|
log.debug(
|
||||||
"running, %d peers, %d ignored, %d active",
|
"%s running, %d peers, %d ignored, %d active, %s connections", blob_hash[:6],
|
||||||
len(batch), len(self.ignored), len(self.active_connections)
|
len(batch), len(self.ignored), len(self.active_connections), len(self.connections)
|
||||||
)
|
)
|
||||||
for peer in sorted(batch, key=lambda peer: self.scores.get(peer, 0), reverse=True):
|
for peer in sorted(batch, key=lambda peer: self.scores.get(peer, 0), reverse=True):
|
||||||
if not self.should_race_continue(blob):
|
if peer in self.ignored:
|
||||||
break
|
continue
|
||||||
if peer not in self.active_connections and peer not in self.ignored:
|
if peer in self.active_connections or not self.should_race_continue(blob):
|
||||||
log.debug("request %s from %s:%i", blob_hash[:8], peer.address, peer.tcp_port)
|
continue
|
||||||
t = self.loop.create_task(self.request_blob_from_peer(blob, peer, connection_id))
|
log.debug("request %s from %s:%i", blob_hash[:8], peer.address, peer.tcp_port)
|
||||||
self.active_connections[peer] = t
|
t = self.loop.create_task(self.request_blob_from_peer(blob, peer, connection_id))
|
||||||
|
self.active_connections[peer] = t
|
||||||
|
self.peer_queue.put_nowait(list(batch))
|
||||||
await self.new_peer_or_finished()
|
await self.new_peer_or_finished()
|
||||||
self.cleanup_active()
|
self.cleanup_active()
|
||||||
log.debug("downloaded %s", blob_hash[:8])
|
log.debug("downloaded %s", blob_hash[:8])
|
||||||
return blob
|
return blob
|
||||||
finally:
|
finally:
|
||||||
blob.close()
|
blob.close()
|
||||||
|
if self.loop.is_running():
|
||||||
|
self.loop.call_soon(self.cleanup_active)
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
|
self.connection_failures.clear()
|
||||||
self.scores.clear()
|
self.scores.clear()
|
||||||
self.ignored.clear()
|
self.ignored.clear()
|
||||||
self.is_running.clear()
|
self.is_running.clear()
|
||||||
for transport in self.connections.values():
|
for protocol in self.connections.values():
|
||||||
transport.close()
|
protocol.close()
|
||||||
|
|
||||||
|
|
||||||
async def download_blob(loop, config: 'Config', blob_manager: 'BlobManager', node: 'Node',
|
async def download_blob(loop, config: 'Config', blob_manager: 'BlobManager', dht_node: 'Node',
|
||||||
blob_hash: str) -> 'AbstractBlob':
|
blob_hash: str) -> 'AbstractBlob':
|
||||||
search_queue = asyncio.Queue(loop=loop, maxsize=config.max_connections_per_download)
|
search_queue = asyncio.Queue(maxsize=config.max_connections_per_download)
|
||||||
search_queue.put_nowait(blob_hash)
|
search_queue.put_nowait(blob_hash)
|
||||||
peer_queue, accumulate_task = node.accumulate_peers(search_queue)
|
peer_queue, accumulate_task = dht_node.accumulate_peers(search_queue)
|
||||||
|
fixed_peers = None if not config.fixed_peers else await get_kademlia_peers_from_hosts(config.fixed_peers)
|
||||||
|
if fixed_peers:
|
||||||
|
loop.call_later(config.fixed_peer_delay, peer_queue.put_nowait, fixed_peers)
|
||||||
downloader = BlobDownloader(loop, config, blob_manager, peer_queue)
|
downloader = BlobDownloader(loop, config, blob_manager, peer_queue)
|
||||||
try:
|
try:
|
||||||
return await downloader.download_blob(blob_hash)
|
return await downloader.download_blob(blob_hash)
|
|
@ -46,7 +46,7 @@ class BlobAvailabilityRequest(BlobMessage):
|
||||||
|
|
||||||
def __init__(self, requested_blobs: typing.List[str], lbrycrd_address: typing.Optional[bool] = True,
|
def __init__(self, requested_blobs: typing.List[str], lbrycrd_address: typing.Optional[bool] = True,
|
||||||
**kwargs) -> None:
|
**kwargs) -> None:
|
||||||
assert len(requested_blobs)
|
assert len(requested_blobs) > 0
|
||||||
self.requested_blobs = requested_blobs
|
self.requested_blobs = requested_blobs
|
||||||
self.lbrycrd_address = lbrycrd_address
|
self.lbrycrd_address = lbrycrd_address
|
||||||
|
|
||||||
|
@ -134,9 +134,9 @@ class BlobErrorResponse(BlobMessage):
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
blob_request_types = typing.Union[BlobPriceRequest, BlobAvailabilityRequest, BlobDownloadRequest,
|
blob_request_types = typing.Union[BlobPriceRequest, BlobAvailabilityRequest, BlobDownloadRequest, # pylint: disable=invalid-name
|
||||||
BlobPaymentAddressRequest]
|
BlobPaymentAddressRequest]
|
||||||
blob_response_types = typing.Union[BlobPriceResponse, BlobAvailabilityResponse, BlobDownloadResponse,
|
blob_response_types = typing.Union[BlobPriceResponse, BlobAvailabilityResponse, BlobDownloadResponse, # pylint: disable=invalid-name
|
||||||
BlobErrorResponse, BlobPaymentAddressResponse]
|
BlobErrorResponse, BlobPaymentAddressResponse]
|
||||||
|
|
||||||
|
|
||||||
|
@ -157,10 +157,10 @@ def _parse_blob_response(response_msg: bytes) -> typing.Tuple[typing.Optional[ty
|
||||||
except ValueError:
|
except ValueError:
|
||||||
continue
|
continue
|
||||||
possible_response_keys = {
|
possible_response_keys = {
|
||||||
BlobPaymentAddressResponse.key,
|
BlobPaymentAddressResponse.key,
|
||||||
BlobAvailabilityResponse.key,
|
BlobAvailabilityResponse.key,
|
||||||
BlobPriceResponse.key,
|
BlobPriceResponse.key,
|
||||||
BlobDownloadResponse.key
|
BlobDownloadResponse.key
|
||||||
}
|
}
|
||||||
if isinstance(response, dict) and response.keys():
|
if isinstance(response, dict) and response.keys():
|
||||||
if set(response.keys()).issubset(possible_response_keys):
|
if set(response.keys()).issubset(possible_response_keys):
|
||||||
|
@ -179,7 +179,7 @@ class BlobRequest:
|
||||||
return d
|
return d
|
||||||
|
|
||||||
def _get_request(self, request_type: blob_request_types):
|
def _get_request(self, request_type: blob_request_types):
|
||||||
request = tuple(filter(lambda r: type(r) == request_type, self.requests))
|
request = tuple(filter(lambda r: type(r) == request_type, self.requests)) # pylint: disable=unidiomatic-typecheck
|
||||||
if request:
|
if request:
|
||||||
return request[0]
|
return request[0]
|
||||||
|
|
||||||
|
@ -235,7 +235,7 @@ class BlobResponse:
|
||||||
return d
|
return d
|
||||||
|
|
||||||
def _get_response(self, response_type: blob_response_types):
|
def _get_response(self, response_type: blob_response_types):
|
||||||
response = tuple(filter(lambda r: type(r) == response_type, self.responses))
|
response = tuple(filter(lambda r: type(r) == response_type, self.responses)) # pylint: disable=unidiomatic-typecheck
|
||||||
if response:
|
if response:
|
||||||
return response[0]
|
return response[0]
|
||||||
|
|
||||||
|
@ -280,4 +280,3 @@ class BlobResponse:
|
||||||
if response_type.key in response
|
if response_type.key in response
|
||||||
])
|
])
|
||||||
return cls(requests, extra)
|
return cls(requests, extra)
|
||||||
|
|
194
lbry/blob_exchange/server.py
Normal file
194
lbry/blob_exchange/server.py
Normal file
|
@ -0,0 +1,194 @@
|
||||||
|
import asyncio
|
||||||
|
import binascii
|
||||||
|
import logging
|
||||||
|
import socket
|
||||||
|
import typing
|
||||||
|
from json.decoder import JSONDecodeError
|
||||||
|
from lbry.blob_exchange.serialization import BlobResponse, BlobRequest, blob_response_types
|
||||||
|
from lbry.blob_exchange.serialization import BlobAvailabilityResponse, BlobPriceResponse, BlobDownloadResponse, \
|
||||||
|
BlobPaymentAddressResponse
|
||||||
|
|
||||||
|
if typing.TYPE_CHECKING:
|
||||||
|
from lbry.blob.blob_manager import BlobManager
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
# a standard request will be 295 bytes
|
||||||
|
MAX_REQUEST_SIZE = 1200
|
||||||
|
|
||||||
|
|
||||||
|
class BlobServerProtocol(asyncio.Protocol):
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, blob_manager: 'BlobManager', lbrycrd_address: str,
|
||||||
|
idle_timeout: float = 30.0, transfer_timeout: float = 60.0):
|
||||||
|
self.loop = loop
|
||||||
|
self.blob_manager = blob_manager
|
||||||
|
self.idle_timeout = idle_timeout
|
||||||
|
self.transfer_timeout = transfer_timeout
|
||||||
|
self.server_task: typing.Optional[asyncio.Task] = None
|
||||||
|
self.started_listening = asyncio.Event()
|
||||||
|
self.buf = b''
|
||||||
|
self.transport: typing.Optional[asyncio.Transport] = None
|
||||||
|
self.lbrycrd_address = lbrycrd_address
|
||||||
|
self.peer_address_and_port: typing.Optional[str] = None
|
||||||
|
self.started_transfer = asyncio.Event()
|
||||||
|
self.transfer_finished = asyncio.Event()
|
||||||
|
self.close_on_idle_task: typing.Optional[asyncio.Task] = None
|
||||||
|
|
||||||
|
async def close_on_idle(self):
|
||||||
|
while self.transport:
|
||||||
|
try:
|
||||||
|
await asyncio.wait_for(self.started_transfer.wait(), self.idle_timeout)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
log.debug("closing idle connection from %s", self.peer_address_and_port)
|
||||||
|
return self.close()
|
||||||
|
self.started_transfer.clear()
|
||||||
|
await self.transfer_finished.wait()
|
||||||
|
self.transfer_finished.clear()
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
if self.transport:
|
||||||
|
self.transport.close()
|
||||||
|
|
||||||
|
def connection_made(self, transport):
|
||||||
|
self.transport = transport
|
||||||
|
self.close_on_idle_task = self.loop.create_task(self.close_on_idle())
|
||||||
|
self.peer_address_and_port = "%s:%i" % self.transport.get_extra_info('peername')
|
||||||
|
self.blob_manager.connection_manager.connection_received(self.peer_address_and_port)
|
||||||
|
log.debug("received connection from %s", self.peer_address_and_port)
|
||||||
|
|
||||||
|
def connection_lost(self, exc: typing.Optional[Exception]) -> None:
|
||||||
|
log.debug("lost connection from %s", self.peer_address_and_port)
|
||||||
|
self.blob_manager.connection_manager.incoming_connection_lost(self.peer_address_and_port)
|
||||||
|
self.transport = None
|
||||||
|
if self.close_on_idle_task and not self.close_on_idle_task.done():
|
||||||
|
self.close_on_idle_task.cancel()
|
||||||
|
self.close_on_idle_task = None
|
||||||
|
|
||||||
|
def send_response(self, responses: typing.List[blob_response_types]):
|
||||||
|
to_send = []
|
||||||
|
while responses:
|
||||||
|
to_send.append(responses.pop())
|
||||||
|
serialized = BlobResponse(to_send).serialize()
|
||||||
|
self.transport.write(serialized)
|
||||||
|
self.blob_manager.connection_manager.sent_data(self.peer_address_and_port, len(serialized))
|
||||||
|
|
||||||
|
async def handle_request(self, request: BlobRequest):
|
||||||
|
addr = self.transport.get_extra_info('peername')
|
||||||
|
peer_address, peer_port = addr
|
||||||
|
|
||||||
|
responses = []
|
||||||
|
address_request = request.get_address_request()
|
||||||
|
if address_request:
|
||||||
|
responses.append(BlobPaymentAddressResponse(lbrycrd_address=self.lbrycrd_address))
|
||||||
|
availability_request = request.get_availability_request()
|
||||||
|
if availability_request:
|
||||||
|
responses.append(BlobAvailabilityResponse(available_blobs=list(set(
|
||||||
|
filter(lambda blob_hash: blob_hash in self.blob_manager.completed_blob_hashes,
|
||||||
|
availability_request.requested_blobs)
|
||||||
|
))))
|
||||||
|
price_request = request.get_price_request()
|
||||||
|
if price_request:
|
||||||
|
responses.append(BlobPriceResponse(blob_data_payment_rate='RATE_ACCEPTED'))
|
||||||
|
download_request = request.get_blob_request()
|
||||||
|
|
||||||
|
if download_request:
|
||||||
|
blob = self.blob_manager.get_blob(download_request.requested_blob)
|
||||||
|
if blob.get_is_verified():
|
||||||
|
incoming_blob = {'blob_hash': blob.blob_hash, 'length': blob.length}
|
||||||
|
responses.append(BlobDownloadResponse(incoming_blob=incoming_blob))
|
||||||
|
self.send_response(responses)
|
||||||
|
blob_hash = blob.blob_hash[:8]
|
||||||
|
log.debug("send %s to %s:%i", blob_hash, peer_address, peer_port)
|
||||||
|
self.started_transfer.set()
|
||||||
|
try:
|
||||||
|
sent = await asyncio.wait_for(blob.sendfile(self), self.transfer_timeout)
|
||||||
|
if sent and sent > 0:
|
||||||
|
self.blob_manager.connection_manager.sent_data(self.peer_address_and_port, sent)
|
||||||
|
log.info("sent %s (%i bytes) to %s:%i", blob_hash, sent, peer_address, peer_port)
|
||||||
|
else:
|
||||||
|
self.close()
|
||||||
|
log.debug("stopped sending %s to %s:%i", blob_hash, peer_address, peer_port)
|
||||||
|
return
|
||||||
|
except (OSError, ValueError, asyncio.TimeoutError) as err:
|
||||||
|
if isinstance(err, asyncio.TimeoutError):
|
||||||
|
log.debug("timed out sending blob %s to %s", blob_hash, peer_address)
|
||||||
|
else:
|
||||||
|
log.warning("could not read blob %s to send %s:%i", blob_hash, peer_address, peer_port)
|
||||||
|
self.close()
|
||||||
|
return
|
||||||
|
finally:
|
||||||
|
self.transfer_finished.set()
|
||||||
|
else:
|
||||||
|
log.info("don't have %s to send %s:%i", blob.blob_hash[:8], peer_address, peer_port)
|
||||||
|
if responses and not self.transport.is_closing():
|
||||||
|
self.send_response(responses)
|
||||||
|
|
||||||
|
def data_received(self, data):
|
||||||
|
request = None
|
||||||
|
if len(self.buf) + len(data or b'') >= MAX_REQUEST_SIZE:
|
||||||
|
log.warning("request from %s is too large", self.peer_address_and_port)
|
||||||
|
self.close()
|
||||||
|
return
|
||||||
|
if data:
|
||||||
|
self.blob_manager.connection_manager.received_data(self.peer_address_and_port, len(data))
|
||||||
|
_, separator, remainder = data.rpartition(b'}')
|
||||||
|
if not separator:
|
||||||
|
self.buf += data
|
||||||
|
return
|
||||||
|
try:
|
||||||
|
request = BlobRequest.deserialize(self.buf + data)
|
||||||
|
self.buf = remainder
|
||||||
|
except (UnicodeDecodeError, JSONDecodeError):
|
||||||
|
log.error("request from %s is not valid json (%i bytes): %s", self.peer_address_and_port,
|
||||||
|
len(self.buf + data), '' if not data else binascii.hexlify(self.buf + data).decode())
|
||||||
|
self.close()
|
||||||
|
return
|
||||||
|
if not request.requests:
|
||||||
|
log.error("failed to decode request from %s (%i bytes): %s", self.peer_address_and_port,
|
||||||
|
len(self.buf + data), '' if not data else binascii.hexlify(self.buf + data).decode())
|
||||||
|
self.close()
|
||||||
|
return
|
||||||
|
self.loop.create_task(self.handle_request(request))
|
||||||
|
|
||||||
|
|
||||||
|
class BlobServer:
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, blob_manager: 'BlobManager', lbrycrd_address: str,
|
||||||
|
idle_timeout: float = 30.0, transfer_timeout: float = 60.0):
|
||||||
|
self.loop = loop
|
||||||
|
self.blob_manager = blob_manager
|
||||||
|
self.server_task: typing.Optional[asyncio.Task] = None
|
||||||
|
self.started_listening = asyncio.Event()
|
||||||
|
self.lbrycrd_address = lbrycrd_address
|
||||||
|
self.idle_timeout = idle_timeout
|
||||||
|
self.transfer_timeout = transfer_timeout
|
||||||
|
self.server_protocol_class = BlobServerProtocol
|
||||||
|
|
||||||
|
def start_server(self, port: int, interface: typing.Optional[str] = '0.0.0.0'):
|
||||||
|
if self.server_task is not None:
|
||||||
|
raise Exception("already running")
|
||||||
|
|
||||||
|
async def _start_server():
|
||||||
|
# checking if the port is in use
|
||||||
|
# thx https://stackoverflow.com/a/52872579
|
||||||
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
||||||
|
if s.connect_ex(('localhost', port)) == 0:
|
||||||
|
# the port is already in use!
|
||||||
|
log.error("Failed to bind TCP %s:%d", interface, port)
|
||||||
|
|
||||||
|
server = await self.loop.create_server(
|
||||||
|
lambda: self.server_protocol_class(self.loop, self.blob_manager, self.lbrycrd_address,
|
||||||
|
self.idle_timeout, self.transfer_timeout),
|
||||||
|
interface, port
|
||||||
|
)
|
||||||
|
self.started_listening.set()
|
||||||
|
log.info("Blob server listening on TCP %s:%i", interface, port)
|
||||||
|
async with server:
|
||||||
|
await server.serve_forever()
|
||||||
|
|
||||||
|
self.server_task = self.loop.create_task(_start_server())
|
||||||
|
|
||||||
|
def stop_server(self):
|
||||||
|
if self.server_task:
|
||||||
|
self.server_task.cancel()
|
||||||
|
self.server_task = None
|
||||||
|
log.info("Stopped blob server")
|
|
@ -1,3 +1,4 @@
|
||||||
# don't touch this. CI server changes this during build/deployment
|
# don't touch this. CI server changes this during build/deployment
|
||||||
BUILD = "dev"
|
BUILD = "dev"
|
||||||
BUILD_COMMIT = "source installation"
|
COMMIT_HASH = "none"
|
||||||
|
DOCKER_TAG = "none"
|
|
@ -1,43 +1,41 @@
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
import typing
|
|
||||||
import logging
|
import logging
|
||||||
import yaml
|
from typing import List, Dict, Tuple, Union, TypeVar, Generic, Optional
|
||||||
from argparse import ArgumentParser
|
from argparse import ArgumentParser
|
||||||
from contextlib import contextmanager
|
from contextlib import contextmanager
|
||||||
from appdirs import user_data_dir, user_config_dir
|
from appdirs import user_data_dir, user_config_dir
|
||||||
from lbrynet.error import InvalidCurrencyError
|
import yaml
|
||||||
from lbrynet.dht import constants
|
from lbry.error import InvalidCurrencyError
|
||||||
|
from lbry.dht import constants
|
||||||
|
from lbry.wallet.coinselection import STRATEGIES
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
NOT_SET = type(str('NOT_SET'), (object,), {})
|
NOT_SET = type('NOT_SET', (object,), {}) # pylint: disable=invalid-name
|
||||||
T = typing.TypeVar('T')
|
T = TypeVar('T')
|
||||||
|
|
||||||
CURRENCIES = {
|
CURRENCIES = {
|
||||||
'BTC': {'type': 'crypto'},
|
'BTC': {'type': 'crypto'},
|
||||||
'LBC': {'type': 'crypto'},
|
'LBC': {'type': 'crypto'},
|
||||||
'USD': {'type': 'fiat'},
|
'USD': {'type': 'fiat'},
|
||||||
}
|
}
|
||||||
HEADERS_FILE_SHA256_CHECKSUM = (
|
|
||||||
366295, 'b0c8197153a33ccbc52fb81a279588b6015b68b7726f73f6a2b81f7e25bfe4b9'
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class Setting(typing.Generic[T]):
|
class Setting(Generic[T]):
|
||||||
|
|
||||||
def __init__(self, doc: str, default: typing.Optional[T] = None,
|
def __init__(self, doc: str, default: Optional[T] = None,
|
||||||
previous_names: typing.Optional[typing.List[str]] = None,
|
previous_names: Optional[List[str]] = None,
|
||||||
metavar: typing.Optional[str] = None):
|
metavar: Optional[str] = None):
|
||||||
self.doc = doc
|
self.doc = doc
|
||||||
self.default = default
|
self.default = default
|
||||||
self.previous_names = previous_names or []
|
self.previous_names = previous_names or []
|
||||||
self.metavar = metavar
|
self.metavar = metavar
|
||||||
|
|
||||||
def __set_name__(self, owner, name):
|
def __set_name__(self, owner, name):
|
||||||
self.name = name
|
self.name = name # pylint: disable=attribute-defined-outside-init
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def cli_name(self):
|
def cli_name(self):
|
||||||
|
@ -47,7 +45,7 @@ class Setting(typing.Generic[T]):
|
||||||
def no_cli_name(self):
|
def no_cli_name(self):
|
||||||
return f"--no-{self.name.replace('_', '-')}"
|
return f"--no-{self.name.replace('_', '-')}"
|
||||||
|
|
||||||
def __get__(self, obj: typing.Optional['BaseConfig'], owner) -> T:
|
def __get__(self, obj: Optional['BaseConfig'], owner) -> T:
|
||||||
if obj is None:
|
if obj is None:
|
||||||
return self
|
return self
|
||||||
for location in obj.search_order:
|
for location in obj.search_order:
|
||||||
|
@ -55,7 +53,7 @@ class Setting(typing.Generic[T]):
|
||||||
return location[self.name]
|
return location[self.name]
|
||||||
return self.default
|
return self.default
|
||||||
|
|
||||||
def __set__(self, obj: 'BaseConfig', val: typing.Union[T, NOT_SET]):
|
def __set__(self, obj: 'BaseConfig', val: Union[T, NOT_SET]):
|
||||||
if val == NOT_SET:
|
if val == NOT_SET:
|
||||||
for location in obj.modify_order:
|
for location in obj.modify_order:
|
||||||
if self.name in location:
|
if self.name in location:
|
||||||
|
@ -65,13 +63,25 @@ class Setting(typing.Generic[T]):
|
||||||
for location in obj.modify_order:
|
for location in obj.modify_order:
|
||||||
location[self.name] = val
|
location[self.name] = val
|
||||||
|
|
||||||
def validate(self, val):
|
def is_set(self, obj: 'BaseConfig') -> bool:
|
||||||
|
for location in obj.search_order:
|
||||||
|
if self.name in location:
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def is_set_to_default(self, obj: 'BaseConfig') -> bool:
|
||||||
|
for location in obj.search_order:
|
||||||
|
if self.name in location:
|
||||||
|
return location[self.name] == self.default
|
||||||
|
return False
|
||||||
|
|
||||||
|
def validate(self, value):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def deserialize(self, value):
|
def deserialize(self, value): # pylint: disable=no-self-use
|
||||||
return value
|
return value
|
||||||
|
|
||||||
def serialize(self, value):
|
def serialize(self, value): # pylint: disable=no-self-use
|
||||||
return value
|
return value
|
||||||
|
|
||||||
def contribute_to_argparse(self, parser: ArgumentParser):
|
def contribute_to_argparse(self, parser: ArgumentParser):
|
||||||
|
@ -84,14 +94,18 @@ class Setting(typing.Generic[T]):
|
||||||
|
|
||||||
|
|
||||||
class String(Setting[str]):
|
class String(Setting[str]):
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, str), \
|
assert isinstance(value, str), \
|
||||||
f"Setting '{self.name}' must be a string."
|
f"Setting '{self.name}' must be a string."
|
||||||
|
|
||||||
|
# TODO: removes this after pylint starts to understand generics
|
||||||
|
def __get__(self, obj: Optional['BaseConfig'], owner) -> str: # pylint: disable=useless-super-delegation
|
||||||
|
return super().__get__(obj, owner)
|
||||||
|
|
||||||
|
|
||||||
class Integer(Setting[int]):
|
class Integer(Setting[int]):
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, int), \
|
assert isinstance(value, int), \
|
||||||
f"Setting '{self.name}' must be an integer."
|
f"Setting '{self.name}' must be an integer."
|
||||||
|
|
||||||
def deserialize(self, value):
|
def deserialize(self, value):
|
||||||
|
@ -99,8 +113,8 @@ class Integer(Setting[int]):
|
||||||
|
|
||||||
|
|
||||||
class Float(Setting[float]):
|
class Float(Setting[float]):
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, float), \
|
assert isinstance(value, float), \
|
||||||
f"Setting '{self.name}' must be a decimal."
|
f"Setting '{self.name}' must be a decimal."
|
||||||
|
|
||||||
def deserialize(self, value):
|
def deserialize(self, value):
|
||||||
|
@ -108,8 +122,8 @@ class Float(Setting[float]):
|
||||||
|
|
||||||
|
|
||||||
class Toggle(Setting[bool]):
|
class Toggle(Setting[bool]):
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, bool), \
|
assert isinstance(value, bool), \
|
||||||
f"Setting '{self.name}' must be a true/false value."
|
f"Setting '{self.name}' must be a true/false value."
|
||||||
|
|
||||||
def contribute_to_argparse(self, parser: ArgumentParser):
|
def contribute_to_argparse(self, parser: ArgumentParser):
|
||||||
|
@ -129,10 +143,10 @@ class Toggle(Setting[bool]):
|
||||||
|
|
||||||
|
|
||||||
class Path(String):
|
class Path(String):
|
||||||
def __init__(self, doc: str, default: str = '', *args, **kwargs):
|
def __init__(self, doc: str, *args, default: str = '', **kwargs):
|
||||||
super().__init__(doc, default, *args, **kwargs)
|
super().__init__(doc, default, *args, **kwargs)
|
||||||
|
|
||||||
def __get__(self, obj, owner):
|
def __get__(self, obj, owner) -> str:
|
||||||
value = super().__get__(obj, owner)
|
value = super().__get__(obj, owner)
|
||||||
if isinstance(value, str):
|
if isinstance(value, str):
|
||||||
return os.path.expanduser(os.path.expandvars(value))
|
return os.path.expanduser(os.path.expandvars(value))
|
||||||
|
@ -150,7 +164,12 @@ class MaxKeyFee(Setting[dict]):
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _parse_list(l):
|
def _parse_list(l):
|
||||||
assert len(l) == 2, 'Max key fee is made up of two values: "AMOUNT CURRENCY".'
|
if l == ['null']:
|
||||||
|
return None
|
||||||
|
assert len(l) == 2, (
|
||||||
|
'Max key fee is made up of either two values: '
|
||||||
|
'"AMOUNT CURRENCY", or "null" (to set no limit)'
|
||||||
|
)
|
||||||
try:
|
try:
|
||||||
amount = float(l[0])
|
amount = float(l[0])
|
||||||
except ValueError:
|
except ValueError:
|
||||||
|
@ -178,13 +197,13 @@ class MaxKeyFee(Setting[dict]):
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
self.cli_name,
|
self.cli_name,
|
||||||
help=self.doc,
|
help=self.doc,
|
||||||
nargs=2,
|
nargs='+',
|
||||||
metavar=('AMOUNT', 'CURRENCY'),
|
metavar=('AMOUNT', 'CURRENCY'),
|
||||||
default=NOT_SET
|
default=NOT_SET
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
self.no_cli_name,
|
self.no_cli_name,
|
||||||
help=f"Disable maximum key fee check.",
|
help="Disable maximum key fee check.",
|
||||||
dest=self.name,
|
dest=self.name,
|
||||||
const=None,
|
const=None,
|
||||||
action="store_const",
|
action="store_const",
|
||||||
|
@ -192,10 +211,25 @@ class MaxKeyFee(Setting[dict]):
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class StringChoice(String):
|
||||||
|
def __init__(self, doc: str, valid_values: List[str], default: str, *args, **kwargs):
|
||||||
|
super().__init__(doc, default, *args, **kwargs)
|
||||||
|
if not valid_values:
|
||||||
|
raise ValueError("No valid values provided")
|
||||||
|
if default not in valid_values:
|
||||||
|
raise ValueError(f"Default value must be one of: {', '.join(valid_values)}")
|
||||||
|
self.valid_values = valid_values
|
||||||
|
|
||||||
|
def validate(self, value):
|
||||||
|
super().validate(value)
|
||||||
|
if value not in self.valid_values:
|
||||||
|
raise ValueError(f"Setting '{self.name}' value must be one of: {', '.join(self.valid_values)}")
|
||||||
|
|
||||||
|
|
||||||
class ListSetting(Setting[list]):
|
class ListSetting(Setting[list]):
|
||||||
|
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, (tuple, list)), \
|
assert isinstance(value, (tuple, list)), \
|
||||||
f"Setting '{self.name}' must be a tuple or list."
|
f"Setting '{self.name}' must be a tuple or list."
|
||||||
|
|
||||||
def contribute_to_argparse(self, parser: ArgumentParser):
|
def contribute_to_argparse(self, parser: ArgumentParser):
|
||||||
|
@ -208,10 +242,10 @@ class ListSetting(Setting[list]):
|
||||||
|
|
||||||
class Servers(ListSetting):
|
class Servers(ListSetting):
|
||||||
|
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, (tuple, list)), \
|
assert isinstance(value, (tuple, list)), \
|
||||||
f"Setting '{self.name}' must be a tuple or list of servers."
|
f"Setting '{self.name}' must be a tuple or list of servers."
|
||||||
for idx, server in enumerate(val):
|
for idx, server in enumerate(value):
|
||||||
assert isinstance(server, (tuple, list)) and len(server) == 2, \
|
assert isinstance(server, (tuple, list)) and len(server) == 2, \
|
||||||
f"Server defined '{server}' at index {idx} in setting " \
|
f"Server defined '{server}' at index {idx} in setting " \
|
||||||
f"'{self.name}' must be a tuple or list of two items."
|
f"'{self.name}' must be a tuple or list of two items."
|
||||||
|
@ -242,26 +276,104 @@ class Servers(ListSetting):
|
||||||
|
|
||||||
class Strings(ListSetting):
|
class Strings(ListSetting):
|
||||||
|
|
||||||
def validate(self, val):
|
def validate(self, value):
|
||||||
assert isinstance(val, (tuple, list)), \
|
assert isinstance(value, (tuple, list)), \
|
||||||
f"Setting '{self.name}' must be a tuple or list of strings."
|
f"Setting '{self.name}' must be a tuple or list of strings."
|
||||||
for idx, string in enumerate(val):
|
for idx, string in enumerate(value):
|
||||||
assert isinstance(string, str), \
|
assert isinstance(string, str), \
|
||||||
f"Value of '{string}' at index {idx} in setting " \
|
f"Value of '{string}' at index {idx} in setting " \
|
||||||
f"'{self.name}' must be a string."
|
f"'{self.name}' must be a string."
|
||||||
|
|
||||||
|
|
||||||
|
class KnownHubsList:
|
||||||
|
|
||||||
|
def __init__(self, config: 'Config' = None, file_name: str = 'known_hubs.yml'):
|
||||||
|
self.file_name = file_name
|
||||||
|
self.path = os.path.join(config.wallet_dir, self.file_name) if config else None
|
||||||
|
self.hubs: Dict[Tuple[str, int], Dict] = {}
|
||||||
|
if self.exists:
|
||||||
|
self.load()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def exists(self):
|
||||||
|
return self.path and os.path.exists(self.path)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def serialized(self) -> Dict[str, Dict]:
|
||||||
|
return {f"{host}:{port}": details for (host, port), details in self.hubs.items()}
|
||||||
|
|
||||||
|
def filter(self, match_none=False, **kwargs):
|
||||||
|
if not kwargs:
|
||||||
|
return self.hubs
|
||||||
|
result = {}
|
||||||
|
for hub, details in self.hubs.items():
|
||||||
|
for key, constraint in kwargs.items():
|
||||||
|
value = details.get(key)
|
||||||
|
if value == constraint or (match_none and value is None):
|
||||||
|
result[hub] = details
|
||||||
|
break
|
||||||
|
return result
|
||||||
|
|
||||||
|
def load(self):
|
||||||
|
if self.path:
|
||||||
|
with open(self.path, 'r') as known_hubs_file:
|
||||||
|
raw = known_hubs_file.read()
|
||||||
|
for hub, details in yaml.safe_load(raw).items():
|
||||||
|
self.set(hub, details)
|
||||||
|
|
||||||
|
def save(self):
|
||||||
|
if self.path:
|
||||||
|
with open(self.path, 'w') as known_hubs_file:
|
||||||
|
known_hubs_file.write(yaml.safe_dump(self.serialized, default_flow_style=False))
|
||||||
|
|
||||||
|
def set(self, hub: str, details: Dict):
|
||||||
|
if hub and hub.count(':') == 1:
|
||||||
|
host, port = hub.split(':')
|
||||||
|
hub_parts = (host, int(port))
|
||||||
|
if hub_parts not in self.hubs:
|
||||||
|
self.hubs[hub_parts] = details
|
||||||
|
return hub
|
||||||
|
|
||||||
|
def add_hubs(self, hubs: List[str]):
|
||||||
|
added = False
|
||||||
|
for hub in hubs:
|
||||||
|
if self.set(hub, {}) is not None:
|
||||||
|
added = True
|
||||||
|
return added
|
||||||
|
|
||||||
|
def items(self):
|
||||||
|
return self.hubs.items()
|
||||||
|
|
||||||
|
def __bool__(self):
|
||||||
|
return len(self) > 0
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return self.hubs.__len__()
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
return iter(self.hubs)
|
||||||
|
|
||||||
|
|
||||||
class EnvironmentAccess:
|
class EnvironmentAccess:
|
||||||
PREFIX = 'LBRY_'
|
PREFIX = 'LBRY_'
|
||||||
|
|
||||||
def __init__(self, environ: dict):
|
def __init__(self, config: 'BaseConfig', environ: dict):
|
||||||
self.environ = environ
|
self.configuration = config
|
||||||
|
self.data = {}
|
||||||
|
if environ:
|
||||||
|
self.load(environ)
|
||||||
|
|
||||||
|
def load(self, environ):
|
||||||
|
for setting in self.configuration.get_settings():
|
||||||
|
value = environ.get(f'{self.PREFIX}{setting.name.upper()}', NOT_SET)
|
||||||
|
if value != NOT_SET and not (isinstance(setting, ListSetting) and value is None):
|
||||||
|
self.data[setting.name] = setting.deserialize(value)
|
||||||
|
|
||||||
def __contains__(self, item: str):
|
def __contains__(self, item: str):
|
||||||
return f'{self.PREFIX}{item.upper()}' in self.environ
|
return item in self.data
|
||||||
|
|
||||||
def __getitem__(self, item: str):
|
def __getitem__(self, item: str):
|
||||||
return self.environ[f'{self.PREFIX}{item.upper()}']
|
return self.data[item]
|
||||||
|
|
||||||
|
|
||||||
class ArgumentAccess:
|
class ArgumentAccess:
|
||||||
|
@ -302,7 +414,7 @@ class ConfigFileAccess:
|
||||||
cls = type(self.configuration)
|
cls = type(self.configuration)
|
||||||
with open(self.path, 'r') as config_file:
|
with open(self.path, 'r') as config_file:
|
||||||
raw = config_file.read()
|
raw = config_file.read()
|
||||||
serialized = yaml.load(raw) or {}
|
serialized = yaml.safe_load(raw) or {}
|
||||||
for key, value in serialized.items():
|
for key, value in serialized.items():
|
||||||
attr = getattr(cls, key, None)
|
attr = getattr(cls, key, None)
|
||||||
if attr is None:
|
if attr is None:
|
||||||
|
@ -346,6 +458,9 @@ class ConfigFileAccess:
|
||||||
del self.data[key]
|
del self.data[key]
|
||||||
|
|
||||||
|
|
||||||
|
TBC = TypeVar('TBC', bound='BaseConfig')
|
||||||
|
|
||||||
|
|
||||||
class BaseConfig:
|
class BaseConfig:
|
||||||
|
|
||||||
config = Path("Path to configuration file.", metavar='FILE')
|
config = Path("Path to configuration file.", metavar='FILE')
|
||||||
|
@ -361,12 +476,11 @@ class BaseConfig:
|
||||||
|
|
||||||
@contextmanager
|
@contextmanager
|
||||||
def update_config(self):
|
def update_config(self):
|
||||||
if not isinstance(self.persisted, ConfigFileAccess):
|
|
||||||
raise TypeError("Config file cannot be updated.")
|
|
||||||
self._updating_config = True
|
self._updating_config = True
|
||||||
yield self
|
yield self
|
||||||
self._updating_config = False
|
self._updating_config = False
|
||||||
self.persisted.save()
|
if isinstance(self.persisted, ConfigFileAccess):
|
||||||
|
self.persisted.save()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def modify_order(self):
|
def modify_order(self):
|
||||||
|
@ -402,7 +516,7 @@ class BaseConfig:
|
||||||
}
|
}
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def create_from_arguments(cls, args):
|
def create_from_arguments(cls, args) -> TBC:
|
||||||
conf = cls()
|
conf = cls()
|
||||||
conf.set_arguments(args)
|
conf.set_arguments(args)
|
||||||
conf.set_environment()
|
conf.set_environment()
|
||||||
|
@ -418,7 +532,7 @@ class BaseConfig:
|
||||||
self.arguments = ArgumentAccess(self, args)
|
self.arguments = ArgumentAccess(self, args)
|
||||||
|
|
||||||
def set_environment(self, environ=None):
|
def set_environment(self, environ=None):
|
||||||
self.environment = EnvironmentAccess(environ or os.environ)
|
self.environment = EnvironmentAccess(self, environ or os.environ)
|
||||||
|
|
||||||
def set_persisted(self, config_file_path=None):
|
def set_persisted(self, config_file_path=None):
|
||||||
if config_file_path is None:
|
if config_file_path is None:
|
||||||
|
@ -437,7 +551,27 @@ class BaseConfig:
|
||||||
self.persisted.save()
|
self.persisted.save()
|
||||||
|
|
||||||
|
|
||||||
class CLIConfig(BaseConfig):
|
class TranscodeConfig(BaseConfig):
|
||||||
|
|
||||||
|
ffmpeg_path = String('A list of places to check for ffmpeg and ffprobe. '
|
||||||
|
f'$data_dir/ffmpeg/bin and $PATH are checked afterward. Separator: {os.pathsep}',
|
||||||
|
'', previous_names=['ffmpeg_folder'])
|
||||||
|
video_encoder = String('FFmpeg codec and parameters for the video encoding. '
|
||||||
|
'Example: libaom-av1 -crf 25 -b:v 0 -strict experimental',
|
||||||
|
'libx264 -crf 24 -preset faster -pix_fmt yuv420p')
|
||||||
|
video_bitrate_maximum = Integer('Maximum bits per second allowed for video streams (0 to disable).', 5_000_000)
|
||||||
|
video_scaler = String('FFmpeg scaling parameters for reducing bitrate. '
|
||||||
|
'Example: -vf "scale=-2:720,fps=24" -maxrate 5M -bufsize 3M',
|
||||||
|
r'-vf "scale=if(gte(iw\,ih)\,min(1920\,iw)\,-2):if(lt(iw\,ih)\,min(1920\,ih)\,-2)" '
|
||||||
|
r'-maxrate 5500K -bufsize 5000K')
|
||||||
|
audio_encoder = String('FFmpeg codec and parameters for the audio encoding. '
|
||||||
|
'Example: libopus -b:a 128k',
|
||||||
|
'aac -b:a 160k')
|
||||||
|
volume_filter = String('FFmpeg filter for audio normalization. Exmple: -af loudnorm', '')
|
||||||
|
volume_analysis_time = Integer('Maximum seconds into the file that we examine audio volume (0 to disable).', 240)
|
||||||
|
|
||||||
|
|
||||||
|
class CLIConfig(TranscodeConfig):
|
||||||
|
|
||||||
api = String('Host name and port for lbrynet daemon API.', 'localhost:5279', metavar='HOST:PORT')
|
api = String('Host name and port for lbrynet daemon API.', 'localhost:5279', metavar='HOST:PORT')
|
||||||
|
|
||||||
|
@ -455,6 +589,9 @@ class CLIConfig(BaseConfig):
|
||||||
|
|
||||||
|
|
||||||
class Config(CLIConfig):
|
class Config(CLIConfig):
|
||||||
|
|
||||||
|
jurisdiction = String("Limit interactions to wallet server in this jurisdiction.")
|
||||||
|
|
||||||
# directories
|
# directories
|
||||||
data_dir = Path("Directory path to store blobs.", metavar='DIR')
|
data_dir = Path("Directory path to store blobs.", metavar='DIR')
|
||||||
download_dir = Path(
|
download_dir = Path(
|
||||||
|
@ -465,6 +602,10 @@ class Config(CLIConfig):
|
||||||
"Directory containing a 'wallets' subdirectory with 'default_wallet' file.",
|
"Directory containing a 'wallets' subdirectory with 'default_wallet' file.",
|
||||||
previous_names=['lbryum_wallet_dir'], metavar='DIR'
|
previous_names=['lbryum_wallet_dir'], metavar='DIR'
|
||||||
)
|
)
|
||||||
|
wallets = Strings(
|
||||||
|
"Wallet files in 'wallet_dir' to load at startup.",
|
||||||
|
['default_wallet']
|
||||||
|
)
|
||||||
|
|
||||||
# network
|
# network
|
||||||
use_upnp = Toggle(
|
use_upnp = Toggle(
|
||||||
|
@ -472,7 +613,8 @@ class Config(CLIConfig):
|
||||||
"ports or have firewall rules you likely want to disable this.", True
|
"ports or have firewall rules you likely want to disable this.", True
|
||||||
)
|
)
|
||||||
udp_port = Integer("UDP port for communicating on the LBRY DHT", 4444, previous_names=['dht_node_port'])
|
udp_port = Integer("UDP port for communicating on the LBRY DHT", 4444, previous_names=['dht_node_port'])
|
||||||
tcp_port = Integer("TCP port to listen for incoming blob requests", 3333, previous_names=['peer_port'])
|
tcp_port = Integer("TCP port to listen for incoming blob requests", 4444, previous_names=['peer_port'])
|
||||||
|
prometheus_port = Integer("Port to expose prometheus metrics (off by default)", 0)
|
||||||
network_interface = String("Interface to use for the DHT and blob exchange", '0.0.0.0')
|
network_interface = String("Interface to use for the DHT and blob exchange", '0.0.0.0')
|
||||||
|
|
||||||
# routing table
|
# routing table
|
||||||
|
@ -480,17 +622,24 @@ class Config(CLIConfig):
|
||||||
"Routing table bucket index below which we always split the bucket if given a new key to add to it and "
|
"Routing table bucket index below which we always split the bucket if given a new key to add to it and "
|
||||||
"the bucket is full. As this value is raised the depth of the routing table (and number of peers in it) "
|
"the bucket is full. As this value is raised the depth of the routing table (and number of peers in it) "
|
||||||
"will increase. This setting is used by seed nodes, you probably don't want to change it during normal "
|
"will increase. This setting is used by seed nodes, you probably don't want to change it during normal "
|
||||||
"use.", 1
|
"use.", 2
|
||||||
|
)
|
||||||
|
is_bootstrap_node = Toggle(
|
||||||
|
"When running as a bootstrap node, disable all logic related to balancing the routing table, so we can "
|
||||||
|
"add as many peers as possible and better help first-runs.", False
|
||||||
)
|
)
|
||||||
|
|
||||||
# protocol timeouts
|
# protocol timeouts
|
||||||
download_timeout = Float("Cumulative timeout for a stream to begin downloading before giving up", 30.0)
|
download_timeout = Float("Cumulative timeout for a stream to begin downloading before giving up", 30.0)
|
||||||
blob_download_timeout = Float("Timeout to download a blob from a peer", 30.0)
|
blob_download_timeout = Float("Timeout to download a blob from a peer", 30.0)
|
||||||
|
hub_timeout = Float("Timeout when making a hub request", 30.0)
|
||||||
peer_connect_timeout = Float("Timeout to establish a TCP connection to a peer", 3.0)
|
peer_connect_timeout = Float("Timeout to establish a TCP connection to a peer", 3.0)
|
||||||
node_rpc_timeout = Float("Timeout when making a DHT request", constants.rpc_timeout)
|
node_rpc_timeout = Float("Timeout when making a DHT request", constants.RPC_TIMEOUT)
|
||||||
|
|
||||||
# blob announcement and download
|
# blob announcement and download
|
||||||
save_blobs = Toggle("Save encrypted blob files for hosting, otherwise download blobs to memory only.", True)
|
save_blobs = Toggle("Save encrypted blob files for hosting, otherwise download blobs to memory only.", True)
|
||||||
|
network_storage_limit = Integer("Disk space in MB to be allocated for helping the P2P network. 0 = disable", 0)
|
||||||
|
blob_storage_limit = Integer("Disk space in MB to be allocated for blob storage. 0 = no limit", 0)
|
||||||
blob_lru_cache_size = Integer(
|
blob_lru_cache_size = Integer(
|
||||||
"LRU cache size for decrypted downloaded blobs used to minimize re-downloading the same blobs when "
|
"LRU cache size for decrypted downloaded blobs used to minimize re-downloading the same blobs when "
|
||||||
"replying to a range request. Set to 0 to disable.", 32
|
"replying to a range request. Set to 0 to disable.", 32
|
||||||
|
@ -507,13 +656,16 @@ class Config(CLIConfig):
|
||||||
"Maximum number of peers to connect to while downloading a blob", 4,
|
"Maximum number of peers to connect to while downloading a blob", 4,
|
||||||
previous_names=['max_connections_per_stream']
|
previous_names=['max_connections_per_stream']
|
||||||
)
|
)
|
||||||
|
concurrent_hub_requests = Integer("Maximum number of concurrent hub requests", 32)
|
||||||
fixed_peer_delay = Float(
|
fixed_peer_delay = Float(
|
||||||
"Amount of seconds before adding the reflector servers as potential peers to download from in case dht"
|
"Amount of seconds before adding the reflector servers as potential peers to download from in case dht"
|
||||||
"peers are not found or are slow", 2.0
|
"peers are not found or are slow", 2.0
|
||||||
)
|
)
|
||||||
max_key_fee = MaxKeyFee(
|
max_key_fee = MaxKeyFee(
|
||||||
"Don't download streams with fees exceeding this amount", {'currency': 'USD', 'amount': 50.0}
|
"Don't download streams with fees exceeding this amount. When set to "
|
||||||
|
"null, the amount is unbounded.", {'currency': 'USD', 'amount': 50.0}
|
||||||
)
|
)
|
||||||
|
max_wallet_server_fee = String("Maximum daily LBC amount allowed as payment for wallet servers.", "0.0")
|
||||||
|
|
||||||
# reflector settings
|
# reflector settings
|
||||||
reflect_streams = Toggle(
|
reflect_streams = Toggle(
|
||||||
|
@ -525,43 +677,80 @@ class Config(CLIConfig):
|
||||||
)
|
)
|
||||||
|
|
||||||
# servers
|
# servers
|
||||||
reflector_servers = Servers("Reflector re-hosting servers", [
|
reflector_servers = Servers("Reflector re-hosting servers for mirroring publishes", [
|
||||||
('reflector.lbry.com', 5566)
|
('reflector.lbry.com', 5566)
|
||||||
])
|
])
|
||||||
|
|
||||||
|
fixed_peers = Servers("Fixed peers to fall back to if none are found on P2P for a blob", [
|
||||||
|
('cdn.reflector.lbry.com', 5567)
|
||||||
|
])
|
||||||
|
|
||||||
|
tracker_servers = Servers("BitTorrent-compatible (BEP15) UDP trackers for helping P2P discovery", [
|
||||||
|
('tracker.lbry.com', 9252),
|
||||||
|
('tracker.lbry.grin.io', 9252),
|
||||||
|
('tracker.lbry.pigg.es', 9252),
|
||||||
|
('tracker.lizard.technology', 9252),
|
||||||
|
('s1.lbry.network', 9252),
|
||||||
|
])
|
||||||
|
|
||||||
lbryum_servers = Servers("SPV wallet servers", [
|
lbryum_servers = Servers("SPV wallet servers", [
|
||||||
('lbryumx1.lbry.com', 50001), # US EAST
|
('spv11.lbry.com', 50001),
|
||||||
('lbryumx2.lbry.com', 50001), # US EAST
|
('spv12.lbry.com', 50001),
|
||||||
('lbryumx4.lbry.com', 50001) #EU
|
('spv13.lbry.com', 50001),
|
||||||
|
('spv14.lbry.com', 50001),
|
||||||
|
('spv15.lbry.com', 50001),
|
||||||
|
('spv16.lbry.com', 50001),
|
||||||
|
('spv17.lbry.com', 50001),
|
||||||
|
('spv18.lbry.com', 50001),
|
||||||
|
('spv19.lbry.com', 50001),
|
||||||
|
('hub.lbry.grin.io', 50001),
|
||||||
|
('hub.lizard.technology', 50001),
|
||||||
|
('s1.lbry.network', 50001),
|
||||||
])
|
])
|
||||||
known_dht_nodes = Servers("Known nodes for bootstrapping connection to the DHT", [
|
known_dht_nodes = Servers("Known nodes for bootstrapping connection to the DHT", [
|
||||||
|
('dht.lbry.grin.io', 4444), # Grin
|
||||||
|
('dht.lbry.madiator.com', 4444), # Madiator
|
||||||
|
('dht.lbry.pigg.es', 4444), # Pigges
|
||||||
('lbrynet1.lbry.com', 4444), # US EAST
|
('lbrynet1.lbry.com', 4444), # US EAST
|
||||||
('lbrynet2.lbry.com', 4444), # US WEST
|
('lbrynet2.lbry.com', 4444), # US WEST
|
||||||
('lbrynet3.lbry.com', 4444), # EU
|
('lbrynet3.lbry.com', 4444), # EU
|
||||||
('lbrynet4.lbry.com', 4444) # ASIA
|
('lbrynet4.lbry.com', 4444), # ASIA
|
||||||
|
('dht.lizard.technology', 4444), # Jack
|
||||||
|
('s2.lbry.network', 4444),
|
||||||
])
|
])
|
||||||
|
|
||||||
comment_server = String("Comment server API URL", "https://comments.lbry.com/api")
|
|
||||||
|
|
||||||
# blockchain
|
# blockchain
|
||||||
blockchain_name = String("Blockchain name - lbrycrd_main, lbrycrd_regtest, or lbrycrd_testnet", 'lbrycrd_main')
|
blockchain_name = String("Blockchain name - lbrycrd_main, lbrycrd_regtest, or lbrycrd_testnet", 'lbrycrd_main')
|
||||||
s3_headers_depth = Integer("download headers from s3 when the local height is more than 10 chunks behind", 96 * 10)
|
|
||||||
cache_time = Integer("Time to cache resolved claims", 150) # TODO: use this
|
|
||||||
|
|
||||||
# daemon
|
# daemon
|
||||||
save_files = Toggle("Save downloaded files when calling `get` by default", True)
|
save_files = Toggle("Save downloaded files when calling `get` by default", False)
|
||||||
components_to_skip = Strings("components which will be skipped during start-up of daemon", [])
|
components_to_skip = Strings("components which will be skipped during start-up of daemon", [])
|
||||||
share_usage_data = Toggle(
|
share_usage_data = Toggle(
|
||||||
"Whether to share usage stats and diagnostic info with LBRY.", True,
|
"Whether to share usage stats and diagnostic info with LBRY.", False,
|
||||||
previous_names=['upload_log', 'upload_log', 'share_debug_info']
|
previous_names=['upload_log', 'upload_log', 'share_debug_info']
|
||||||
)
|
)
|
||||||
|
track_bandwidth = Toggle("Track bandwidth usage", True)
|
||||||
|
allowed_origin = String(
|
||||||
|
"Allowed `Origin` header value for API request (sent by browser), use * to allow "
|
||||||
|
"all hosts; default is to only allow API requests with no `Origin` value.", "")
|
||||||
|
|
||||||
# media server
|
# media server
|
||||||
|
|
||||||
streaming_server = String('Host name and port to serve streaming media over range requests',
|
streaming_server = String('Host name and port to serve streaming media over range requests',
|
||||||
'localhost:5280', metavar='HOST:PORT')
|
'localhost:5280', metavar='HOST:PORT')
|
||||||
streaming_get = Toggle("Enable the /get endpoint for the streaming media server. "
|
streaming_get = Toggle("Enable the /get endpoint for the streaming media server. "
|
||||||
"Disable to prevent new streams from being added.", True)
|
"Disable to prevent new streams from being added.", True)
|
||||||
|
|
||||||
|
coin_selection_strategy = StringChoice(
|
||||||
|
"Strategy to use when selecting UTXOs for a transaction",
|
||||||
|
STRATEGIES, "prefer_confirmed"
|
||||||
|
)
|
||||||
|
|
||||||
|
transaction_cache_size = Integer("Transaction cache size", 2 ** 17)
|
||||||
|
save_resolved_claims = Toggle(
|
||||||
|
"Save content claims to the database when they are resolved to keep file_list up to date, "
|
||||||
|
"only disable this if file_x commands are not needed", True
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def streaming_host(self):
|
def streaming_host(self):
|
||||||
return self.streaming_server.split(':')[0]
|
return self.streaming_server.split(':')[0]
|
||||||
|
@ -573,6 +762,7 @@ class Config(CLIConfig):
|
||||||
def __init__(self, **kwargs):
|
def __init__(self, **kwargs):
|
||||||
super().__init__(**kwargs)
|
super().__init__(**kwargs)
|
||||||
self.set_default_paths()
|
self.set_default_paths()
|
||||||
|
self.known_hubs = KnownHubsList(self)
|
||||||
|
|
||||||
def set_default_paths(self):
|
def set_default_paths(self):
|
||||||
if 'darwin' in sys.platform.lower():
|
if 'darwin' in sys.platform.lower():
|
||||||
|
@ -594,10 +784,14 @@ class Config(CLIConfig):
|
||||||
return os.path.join(self.data_dir, 'lbrynet.log')
|
return os.path.join(self.data_dir, 'lbrynet.log')
|
||||||
|
|
||||||
|
|
||||||
def get_windows_directories() -> typing.Tuple[str, str, str]:
|
def get_windows_directories() -> Tuple[str, str, str]:
|
||||||
from lbrynet.winpaths import get_path, FOLDERID, UserHandle
|
from lbry.winpaths import get_path, FOLDERID, UserHandle, \
|
||||||
|
PathNotFoundException # pylint: disable=import-outside-toplevel
|
||||||
|
|
||||||
download_dir = get_path(FOLDERID.Downloads, UserHandle.current)
|
try:
|
||||||
|
download_dir = get_path(FOLDERID.Downloads, UserHandle.current)
|
||||||
|
except PathNotFoundException:
|
||||||
|
download_dir = os.getcwd()
|
||||||
|
|
||||||
# old
|
# old
|
||||||
appdata = get_path(FOLDERID.RoamingAppData, UserHandle.current)
|
appdata = get_path(FOLDERID.RoamingAppData, UserHandle.current)
|
||||||
|
@ -609,24 +803,24 @@ def get_windows_directories() -> typing.Tuple[str, str, str]:
|
||||||
# new
|
# new
|
||||||
data_dir = user_data_dir('lbrynet', 'lbry')
|
data_dir = user_data_dir('lbrynet', 'lbry')
|
||||||
lbryum_dir = user_data_dir('lbryum', 'lbry')
|
lbryum_dir = user_data_dir('lbryum', 'lbry')
|
||||||
download_dir = get_path(FOLDERID.Downloads, UserHandle.current)
|
|
||||||
return data_dir, lbryum_dir, download_dir
|
return data_dir, lbryum_dir, download_dir
|
||||||
|
|
||||||
|
|
||||||
def get_darwin_directories() -> typing.Tuple[str, str, str]:
|
def get_darwin_directories() -> Tuple[str, str, str]:
|
||||||
data_dir = user_data_dir('LBRY')
|
data_dir = user_data_dir('LBRY')
|
||||||
lbryum_dir = os.path.expanduser('~/.lbryum')
|
lbryum_dir = os.path.expanduser('~/.lbryum')
|
||||||
download_dir = os.path.expanduser('~/Downloads')
|
download_dir = os.path.expanduser('~/Downloads')
|
||||||
return data_dir, lbryum_dir, download_dir
|
return data_dir, lbryum_dir, download_dir
|
||||||
|
|
||||||
|
|
||||||
def get_linux_directories() -> typing.Tuple[str, str, str]:
|
def get_linux_directories() -> Tuple[str, str, str]:
|
||||||
try:
|
try:
|
||||||
with open(os.path.join(user_config_dir(), 'user-dirs.dirs'), 'r') as xdg:
|
with open(os.path.join(user_config_dir(), 'user-dirs.dirs'), 'r') as xdg:
|
||||||
down_dir = re.search(r'XDG_DOWNLOAD_DIR=(.+)', xdg.read()).group(1)
|
down_dir = re.search(r'XDG_DOWNLOAD_DIR=(.+)', xdg.read())
|
||||||
down_dir = re.sub('\$HOME', os.getenv('HOME') or os.path.expanduser("~/"), down_dir)
|
if down_dir:
|
||||||
download_dir = re.sub('\"', '', down_dir)
|
down_dir = re.sub(r'\$HOME', os.getenv('HOME') or os.path.expanduser("~/"), down_dir.group(1))
|
||||||
except EnvironmentError:
|
download_dir = re.sub('\"', '', down_dir)
|
||||||
|
except OSError:
|
||||||
download_dir = os.getenv('XDG_DOWNLOAD_DIR')
|
download_dir = os.getenv('XDG_DOWNLOAD_DIR')
|
||||||
if not download_dir:
|
if not download_dir:
|
||||||
download_dir = os.path.expanduser('~/Downloads')
|
download_dir = os.path.expanduser('~/Downloads')
|
|
@ -1,3 +1,4 @@
|
||||||
|
import time
|
||||||
import asyncio
|
import asyncio
|
||||||
import typing
|
import typing
|
||||||
import collections
|
import collections
|
||||||
|
@ -18,8 +19,10 @@ class ConnectionManager:
|
||||||
self.incoming: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
self.incoming: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
||||||
self.outgoing_connected: typing.Set[str] = set()
|
self.outgoing_connected: typing.Set[str] = set()
|
||||||
self.outgoing: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
self.outgoing: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
||||||
|
self._max_incoming_mbs = 0.0
|
||||||
|
self._max_outgoing_mbs = 0.0
|
||||||
self._status = {}
|
self._status = {}
|
||||||
|
self._running = False
|
||||||
self._task: typing.Optional[asyncio.Task] = None
|
self._task: typing.Optional[asyncio.Task] = None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -27,53 +30,63 @@ class ConnectionManager:
|
||||||
return self._status
|
return self._status
|
||||||
|
|
||||||
def sent_data(self, host_and_port: str, size: int):
|
def sent_data(self, host_and_port: str, size: int):
|
||||||
self.outgoing[host_and_port] += size
|
if self._running:
|
||||||
|
self.outgoing[host_and_port] += size
|
||||||
|
|
||||||
def received_data(self, host_and_port: str, size: int):
|
def received_data(self, host_and_port: str, size: int):
|
||||||
self.incoming[host_and_port] += size
|
if self._running:
|
||||||
|
self.incoming[host_and_port] += size
|
||||||
|
|
||||||
def connection_made(self, host_and_port: str):
|
def connection_made(self, host_and_port: str):
|
||||||
self.outgoing_connected.add(host_and_port)
|
if self._running:
|
||||||
|
self.outgoing_connected.add(host_and_port)
|
||||||
|
|
||||||
def connection_received(self, host_and_port: str):
|
def connection_received(self, host_and_port: str):
|
||||||
# self.incoming_connected.add(host_and_port)
|
# self.incoming_connected.add(host_and_port)
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def outgoing_connection_lost(self, host_and_port: str):
|
def outgoing_connection_lost(self, host_and_port: str):
|
||||||
if host_and_port in self.outgoing_connected:
|
if self._running and host_and_port in self.outgoing_connected:
|
||||||
self.outgoing_connected.remove(host_and_port)
|
self.outgoing_connected.remove(host_and_port)
|
||||||
|
|
||||||
def incoming_connection_lost(self, host_and_port: str):
|
def incoming_connection_lost(self, host_and_port: str):
|
||||||
if host_and_port in self.incoming_connected:
|
if self._running and host_and_port in self.incoming_connected:
|
||||||
self.incoming_connected.remove(host_and_port)
|
self.incoming_connected.remove(host_and_port)
|
||||||
|
|
||||||
async def _update(self):
|
async def _update(self):
|
||||||
|
|
||||||
self._status = {
|
self._status = {
|
||||||
'incoming_bps': {},
|
'incoming_bps': {},
|
||||||
'outgoing_bps': {},
|
'outgoing_bps': {},
|
||||||
'total_incoming_mbs': 0.0,
|
'total_incoming_mbs': 0.0,
|
||||||
'total_outgoing_mbs': 0.0,
|
'total_outgoing_mbs': 0.0,
|
||||||
'time': self.loop.time()
|
'total_sent': 0,
|
||||||
|
'total_received': 0,
|
||||||
|
'max_incoming_mbs': 0.0,
|
||||||
|
'max_outgoing_mbs': 0.0
|
||||||
}
|
}
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
last = self.loop.time()
|
last = time.perf_counter()
|
||||||
await asyncio.sleep(1, loop=self.loop)
|
await asyncio.sleep(0.1)
|
||||||
self._status['incoming_bps'].clear()
|
self._status['incoming_bps'].clear()
|
||||||
self._status['outgoing_bps'].clear()
|
self._status['outgoing_bps'].clear()
|
||||||
|
now = time.perf_counter()
|
||||||
while self.outgoing:
|
while self.outgoing:
|
||||||
k, v = self.outgoing.popitem()
|
k, sent = self.outgoing.popitem()
|
||||||
self._status['outgoing_bps'][k] = v
|
self._status['total_sent'] += sent
|
||||||
|
self._status['outgoing_bps'][k] = sent / (now - last)
|
||||||
while self.incoming:
|
while self.incoming:
|
||||||
k, v = self.incoming.popitem()
|
k, received = self.incoming.popitem()
|
||||||
self._status['incoming_bps'][k] = v
|
self._status['total_received'] += received
|
||||||
now = self.loop.time()
|
self._status['incoming_bps'][k] = received / (now - last)
|
||||||
self._status['total_outgoing_mbs'] = int(sum(list(self._status['outgoing_bps'].values())
|
self._status['total_outgoing_mbs'] = int(sum(list(self._status['outgoing_bps'].values())
|
||||||
) / (now - last)) / 1000000.0
|
)) / 1000000.0
|
||||||
self._status['total_incoming_mbs'] = int(sum(list(self._status['incoming_bps'].values())
|
self._status['total_incoming_mbs'] = int(sum(list(self._status['incoming_bps'].values())
|
||||||
) / (now - last)) / 1000000.0
|
)) / 1000000.0
|
||||||
self._status['time'] = now
|
self._max_incoming_mbs = max(self._max_incoming_mbs, self._status['total_incoming_mbs'])
|
||||||
|
self._max_outgoing_mbs = max(self._max_outgoing_mbs, self._status['total_outgoing_mbs'])
|
||||||
|
self._status['max_incoming_mbs'] = self._max_incoming_mbs
|
||||||
|
self._status['max_outgoing_mbs'] = self._max_outgoing_mbs
|
||||||
|
|
||||||
def stop(self):
|
def stop(self):
|
||||||
if self._task:
|
if self._task:
|
||||||
|
@ -84,7 +97,9 @@ class ConnectionManager:
|
||||||
self.incoming.clear()
|
self.incoming.clear()
|
||||||
self.incoming_connected.clear()
|
self.incoming_connected.clear()
|
||||||
self._status.clear()
|
self._status.clear()
|
||||||
|
self._running = False
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
self.stop()
|
self.stop()
|
||||||
|
self._running = True
|
||||||
self._task = self.loop.create_task(self._update())
|
self._task = self.loop.create_task(self._update())
|
2
lbry/constants.py
Normal file
2
lbry/constants.py
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
CENT = 1000000
|
||||||
|
COIN = 100*CENT
|
86
lbry/crypto/base58.py
Normal file
86
lbry/crypto/base58.py
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
from lbry.crypto.hash import double_sha256
|
||||||
|
from lbry.crypto.util import bytes_to_int, int_to_bytes
|
||||||
|
|
||||||
|
|
||||||
|
class Base58Error(Exception):
|
||||||
|
""" Exception used for Base58 errors. """
|
||||||
|
|
||||||
|
|
||||||
|
class Base58:
|
||||||
|
""" Class providing base 58 functionality. """
|
||||||
|
|
||||||
|
chars = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'
|
||||||
|
assert len(chars) == 58
|
||||||
|
char_map = {c: n for n, c in enumerate(chars)}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def char_value(cls, c):
|
||||||
|
val = cls.char_map.get(c)
|
||||||
|
if val is None:
|
||||||
|
raise Base58Error(f'invalid base 58 character "{c}"')
|
||||||
|
return val
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def decode(cls, txt):
|
||||||
|
""" Decodes txt into a big-endian bytearray. """
|
||||||
|
if isinstance(txt, memoryview):
|
||||||
|
txt = str(txt)
|
||||||
|
|
||||||
|
if isinstance(txt, bytes):
|
||||||
|
txt = txt.decode()
|
||||||
|
|
||||||
|
if not isinstance(txt, str):
|
||||||
|
raise TypeError('a string is required')
|
||||||
|
|
||||||
|
if not txt:
|
||||||
|
raise Base58Error('string cannot be empty')
|
||||||
|
|
||||||
|
value = 0
|
||||||
|
for c in txt:
|
||||||
|
value = value * 58 + cls.char_value(c)
|
||||||
|
|
||||||
|
result = int_to_bytes(value)
|
||||||
|
|
||||||
|
# Prepend leading zero bytes if necessary
|
||||||
|
count = 0
|
||||||
|
for c in txt:
|
||||||
|
if c != '1':
|
||||||
|
break
|
||||||
|
count += 1
|
||||||
|
if count:
|
||||||
|
result = bytes((0,)) * count + result
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def encode(cls, be_bytes):
|
||||||
|
"""Converts a big-endian bytearray into a base58 string."""
|
||||||
|
value = bytes_to_int(be_bytes)
|
||||||
|
|
||||||
|
txt = ''
|
||||||
|
while value:
|
||||||
|
value, mod = divmod(value, 58)
|
||||||
|
txt += cls.chars[mod]
|
||||||
|
|
||||||
|
for byte in be_bytes:
|
||||||
|
if byte != 0:
|
||||||
|
break
|
||||||
|
txt += '1'
|
||||||
|
|
||||||
|
return txt[::-1]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def decode_check(cls, txt, hash_fn=double_sha256):
|
||||||
|
""" Decodes a Base58Check-encoded string to a payload. The version prefixes it. """
|
||||||
|
be_bytes = cls.decode(txt)
|
||||||
|
result, check = be_bytes[:-4], be_bytes[-4:]
|
||||||
|
if check != hash_fn(result)[:4]:
|
||||||
|
raise Base58Error(f'invalid base 58 checksum for {txt}')
|
||||||
|
return result
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def encode_check(cls, payload, hash_fn=double_sha256):
|
||||||
|
""" Encodes a payload bytearray (which includes the version byte(s))
|
||||||
|
into a Base58Check string."""
|
||||||
|
be_bytes = payload + hash_fn(payload)[:4]
|
||||||
|
return cls.encode(be_bytes)
|
71
lbry/crypto/crypt.py
Normal file
71
lbry/crypto/crypt.py
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
import os
|
||||||
|
import base64
|
||||||
|
import typing
|
||||||
|
from cryptography.hazmat.primitives.kdf.scrypt import Scrypt
|
||||||
|
from cryptography.hazmat.primitives.ciphers import Cipher, modes
|
||||||
|
from cryptography.hazmat.primitives.ciphers.algorithms import AES
|
||||||
|
from cryptography.hazmat.primitives.padding import PKCS7
|
||||||
|
from cryptography.hazmat.backends import default_backend
|
||||||
|
|
||||||
|
from lbry.error import InvalidPasswordError
|
||||||
|
from lbry.crypto.hash import double_sha256
|
||||||
|
|
||||||
|
|
||||||
|
def aes_encrypt(secret: str, value: str, init_vector: bytes = None) -> str:
|
||||||
|
if init_vector is not None:
|
||||||
|
assert len(init_vector) == 16
|
||||||
|
else:
|
||||||
|
init_vector = os.urandom(16)
|
||||||
|
key = double_sha256(secret.encode())
|
||||||
|
encryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).encryptor()
|
||||||
|
padder = PKCS7(AES.block_size).padder()
|
||||||
|
padded_data = padder.update(value.encode()) + padder.finalize()
|
||||||
|
encrypted_data = encryptor.update(padded_data) + encryptor.finalize()
|
||||||
|
return base64.b64encode(init_vector + encrypted_data).decode()
|
||||||
|
|
||||||
|
|
||||||
|
def aes_decrypt(secret: str, value: str) -> typing.Tuple[str, bytes]:
|
||||||
|
try:
|
||||||
|
data = base64.b64decode(value.encode())
|
||||||
|
key = double_sha256(secret.encode())
|
||||||
|
init_vector, data = data[:16], data[16:]
|
||||||
|
decryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).decryptor()
|
||||||
|
unpadder = PKCS7(AES.block_size).unpadder()
|
||||||
|
result = unpadder.update(decryptor.update(data)) + unpadder.finalize()
|
||||||
|
return result.decode(), init_vector
|
||||||
|
except UnicodeDecodeError:
|
||||||
|
raise InvalidPasswordError()
|
||||||
|
except ValueError as e:
|
||||||
|
if e.args[0] == 'Invalid padding bytes.':
|
||||||
|
raise InvalidPasswordError()
|
||||||
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
def better_aes_encrypt(secret: str, value: bytes) -> bytes:
|
||||||
|
init_vector = os.urandom(16)
|
||||||
|
key = scrypt(secret.encode(), salt=init_vector)
|
||||||
|
encryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).encryptor()
|
||||||
|
padder = PKCS7(AES.block_size).padder()
|
||||||
|
padded_data = padder.update(value) + padder.finalize()
|
||||||
|
encrypted_data = encryptor.update(padded_data) + encryptor.finalize()
|
||||||
|
return base64.b64encode(b's:8192:16:1:' + init_vector + encrypted_data)
|
||||||
|
|
||||||
|
|
||||||
|
def better_aes_decrypt(secret: str, value: bytes) -> bytes:
|
||||||
|
try:
|
||||||
|
data = base64.b64decode(value)
|
||||||
|
_, scryp_n, scrypt_r, scrypt_p, data = data.split(b':', maxsplit=4)
|
||||||
|
init_vector, data = data[:16], data[16:]
|
||||||
|
key = scrypt(secret.encode(), init_vector, int(scryp_n), int(scrypt_r), int(scrypt_p))
|
||||||
|
decryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).decryptor()
|
||||||
|
unpadder = PKCS7(AES.block_size).unpadder()
|
||||||
|
return unpadder.update(decryptor.update(data)) + unpadder.finalize()
|
||||||
|
except ValueError as e:
|
||||||
|
if e.args[0] == 'Invalid padding bytes.':
|
||||||
|
raise InvalidPasswordError()
|
||||||
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
def scrypt(passphrase, salt, scrypt_n=1<<13, scrypt_r=16, scrypt_p=1):
|
||||||
|
kdf = Scrypt(salt, length=32, n=scrypt_n, r=scrypt_r, p=scrypt_p, backend=default_backend())
|
||||||
|
return kdf.derive(passphrase)
|
47
lbry/crypto/hash.py
Normal file
47
lbry/crypto/hash.py
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
import hashlib
|
||||||
|
import hmac
|
||||||
|
from binascii import hexlify, unhexlify
|
||||||
|
|
||||||
|
|
||||||
|
def sha256(x):
|
||||||
|
""" Simple wrapper of hashlib sha256. """
|
||||||
|
return hashlib.sha256(x).digest()
|
||||||
|
|
||||||
|
|
||||||
|
def sha512(x):
|
||||||
|
""" Simple wrapper of hashlib sha512. """
|
||||||
|
return hashlib.sha512(x).digest()
|
||||||
|
|
||||||
|
|
||||||
|
def ripemd160(x):
|
||||||
|
""" Simple wrapper of hashlib ripemd160. """
|
||||||
|
h = hashlib.new('ripemd160')
|
||||||
|
h.update(x)
|
||||||
|
return h.digest()
|
||||||
|
|
||||||
|
|
||||||
|
def double_sha256(x):
|
||||||
|
""" SHA-256 of SHA-256, as used extensively in bitcoin. """
|
||||||
|
return sha256(sha256(x))
|
||||||
|
|
||||||
|
|
||||||
|
def hmac_sha512(key, msg):
|
||||||
|
""" Use SHA-512 to provide an HMAC. """
|
||||||
|
return hmac.new(key, msg, hashlib.sha512).digest()
|
||||||
|
|
||||||
|
|
||||||
|
def hash160(x):
|
||||||
|
""" RIPEMD-160 of SHA-256.
|
||||||
|
Used to make bitcoin addresses from pubkeys. """
|
||||||
|
return ripemd160(sha256(x))
|
||||||
|
|
||||||
|
|
||||||
|
def hash_to_hex_str(x):
|
||||||
|
""" Convert a big-endian binary hash to displayed hex string.
|
||||||
|
Display form of a binary hash is reversed and converted to hex. """
|
||||||
|
return hexlify(reversed(x))
|
||||||
|
|
||||||
|
|
||||||
|
def hex_str_to_hash(x):
|
||||||
|
""" Convert a displayed hex string to a binary hash. """
|
||||||
|
return reversed(unhexlify(x))
|
13
lbry/crypto/util.py
Normal file
13
lbry/crypto/util.py
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
from binascii import unhexlify, hexlify
|
||||||
|
|
||||||
|
|
||||||
|
def bytes_to_int(be_bytes):
|
||||||
|
""" Interprets a big-endian sequence of bytes as an integer. """
|
||||||
|
return int(hexlify(be_bytes), 16)
|
||||||
|
|
||||||
|
|
||||||
|
def int_to_bytes(value):
|
||||||
|
""" Converts an integer to a big-endian sequence of bytes. """
|
||||||
|
length = (value.bit_length() + 7) // 8
|
||||||
|
s = '%x' % value
|
||||||
|
return unhexlify(('0' * (len(s) % 2) + s).zfill(length * 2))
|
78
lbry/dht/blob_announcer.py
Normal file
78
lbry/dht/blob_announcer.py
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
import asyncio
|
||||||
|
import typing
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from prometheus_client import Counter, Gauge
|
||||||
|
|
||||||
|
if typing.TYPE_CHECKING:
|
||||||
|
from lbry.dht.node import Node
|
||||||
|
from lbry.extras.daemon.storage import SQLiteStorage
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class BlobAnnouncer:
|
||||||
|
announcements_sent_metric = Counter(
|
||||||
|
"announcements_sent", "Number of announcements sent and their respective status.", namespace="dht_node",
|
||||||
|
labelnames=("peers", "error"),
|
||||||
|
)
|
||||||
|
announcement_queue_size_metric = Gauge(
|
||||||
|
"announcement_queue_size", "Number of hashes waiting to be announced.", namespace="dht_node",
|
||||||
|
labelnames=("scope",)
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, node: 'Node', storage: 'SQLiteStorage'):
|
||||||
|
self.loop = loop
|
||||||
|
self.node = node
|
||||||
|
self.storage = storage
|
||||||
|
self.announce_task: asyncio.Task = None
|
||||||
|
self.announce_queue: typing.List[str] = []
|
||||||
|
self._done = asyncio.Event()
|
||||||
|
self.announced = set()
|
||||||
|
|
||||||
|
async def _run_consumer(self):
|
||||||
|
while self.announce_queue:
|
||||||
|
try:
|
||||||
|
blob_hash = self.announce_queue.pop()
|
||||||
|
peers = len(await self.node.announce_blob(blob_hash))
|
||||||
|
self.announcements_sent_metric.labels(peers=peers, error=False).inc()
|
||||||
|
if peers > 4:
|
||||||
|
self.announced.add(blob_hash)
|
||||||
|
else:
|
||||||
|
log.debug("failed to announce %s, could only find %d peers, retrying soon.", blob_hash[:8], peers)
|
||||||
|
except Exception as err:
|
||||||
|
self.announcements_sent_metric.labels(peers=0, error=True).inc()
|
||||||
|
log.warning("error announcing %s: %s", blob_hash[:8], str(err))
|
||||||
|
|
||||||
|
async def _announce(self, batch_size: typing.Optional[int] = 10):
|
||||||
|
while batch_size:
|
||||||
|
if not self.node.joined.is_set():
|
||||||
|
await self.node.joined.wait()
|
||||||
|
await asyncio.sleep(60)
|
||||||
|
if not self.node.protocol.routing_table.get_peers():
|
||||||
|
log.warning("No peers in DHT, announce round skipped")
|
||||||
|
continue
|
||||||
|
self.announce_queue.extend(await self.storage.get_blobs_to_announce())
|
||||||
|
self.announcement_queue_size_metric.labels(scope="global").set(len(self.announce_queue))
|
||||||
|
log.debug("announcer task wake up, %d blobs to announce", len(self.announce_queue))
|
||||||
|
while len(self.announce_queue) > 0:
|
||||||
|
log.info("%i blobs to announce", len(self.announce_queue))
|
||||||
|
await asyncio.gather(*[self._run_consumer() for _ in range(batch_size)])
|
||||||
|
announced = list(filter(None, self.announced))
|
||||||
|
if announced:
|
||||||
|
await self.storage.update_last_announced_blobs(announced)
|
||||||
|
log.info("announced %i blobs", len(announced))
|
||||||
|
self.announced.clear()
|
||||||
|
self._done.set()
|
||||||
|
self._done.clear()
|
||||||
|
|
||||||
|
def start(self, batch_size: typing.Optional[int] = 10):
|
||||||
|
assert not self.announce_task or self.announce_task.done(), "already running"
|
||||||
|
self.announce_task = self.loop.create_task(self._announce(batch_size))
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
if self.announce_task and not self.announce_task.done():
|
||||||
|
self.announce_task.cancel()
|
||||||
|
|
||||||
|
def wait(self):
|
||||||
|
return self._done.wait()
|
40
lbry/dht/constants.py
Normal file
40
lbry/dht/constants.py
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
import hashlib
|
||||||
|
import os
|
||||||
|
|
||||||
|
HASH_CLASS = hashlib.sha384 # pylint: disable=invalid-name
|
||||||
|
HASH_LENGTH = HASH_CLASS().digest_size
|
||||||
|
HASH_BITS = HASH_LENGTH * 8
|
||||||
|
ALPHA = 5
|
||||||
|
K = 8
|
||||||
|
SPLIT_BUCKETS_UNDER_INDEX = 1
|
||||||
|
REPLACEMENT_CACHE_SIZE = 8
|
||||||
|
RPC_TIMEOUT = 5.0
|
||||||
|
RPC_ATTEMPTS = 5
|
||||||
|
RPC_ATTEMPTS_PRUNING_WINDOW = 600
|
||||||
|
ITERATIVE_LOOKUP_DELAY = RPC_TIMEOUT / 2.0 # TODO: use config val / 2 if rpc timeout is provided
|
||||||
|
REFRESH_INTERVAL = 3600 # 1 hour
|
||||||
|
REPLICATE_INTERVAL = REFRESH_INTERVAL
|
||||||
|
DATA_EXPIRATION = 86400 # 24 hours
|
||||||
|
TOKEN_SECRET_REFRESH_INTERVAL = 300 # 5 minutes
|
||||||
|
MAYBE_PING_DELAY = 300 # 5 minutes
|
||||||
|
CHECK_REFRESH_INTERVAL = REFRESH_INTERVAL / 5
|
||||||
|
RPC_ID_LENGTH = 20
|
||||||
|
PROTOCOL_VERSION = 1
|
||||||
|
MSG_SIZE_LIMIT = 1400
|
||||||
|
|
||||||
|
|
||||||
|
def digest(data: bytes) -> bytes:
|
||||||
|
h = HASH_CLASS()
|
||||||
|
h.update(data)
|
||||||
|
return h.digest()
|
||||||
|
|
||||||
|
|
||||||
|
def generate_id(num=None) -> bytes:
|
||||||
|
if num is not None:
|
||||||
|
return digest(str(num).encode())
|
||||||
|
else:
|
||||||
|
return digest(os.urandom(32))
|
||||||
|
|
||||||
|
|
||||||
|
def generate_rpc_id(num=None) -> bytes:
|
||||||
|
return generate_id(num)[:RPC_ID_LENGTH]
|
282
lbry/dht/node.py
Normal file
282
lbry/dht/node.py
Normal file
|
@ -0,0 +1,282 @@
|
||||||
|
import logging
|
||||||
|
import asyncio
|
||||||
|
import typing
|
||||||
|
import socket
|
||||||
|
|
||||||
|
from prometheus_client import Gauge
|
||||||
|
|
||||||
|
from lbry.utils import aclosing, resolve_host
|
||||||
|
from lbry.dht import constants
|
||||||
|
from lbry.dht.peer import make_kademlia_peer
|
||||||
|
from lbry.dht.protocol.distance import Distance
|
||||||
|
from lbry.dht.protocol.iterative_find import IterativeNodeFinder, IterativeValueFinder
|
||||||
|
from lbry.dht.protocol.protocol import KademliaProtocol
|
||||||
|
|
||||||
|
if typing.TYPE_CHECKING:
|
||||||
|
from lbry.dht.peer import PeerManager
|
||||||
|
from lbry.dht.peer import KademliaPeer
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class Node:
|
||||||
|
storing_peers_metric = Gauge(
|
||||||
|
"storing_peers", "Number of peers storing blobs announced to this node", namespace="dht_node",
|
||||||
|
labelnames=("scope",),
|
||||||
|
)
|
||||||
|
stored_blob_with_x_bytes_colliding = Gauge(
|
||||||
|
"stored_blobs_x_bytes_colliding", "Number of blobs with at least X bytes colliding with this node id prefix",
|
||||||
|
namespace="dht_node", labelnames=("amount",)
|
||||||
|
)
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', node_id: bytes, udp_port: int,
|
||||||
|
internal_udp_port: int, peer_port: int, external_ip: str, rpc_timeout: float = constants.RPC_TIMEOUT,
|
||||||
|
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX, is_bootstrap_node: bool = False,
|
||||||
|
storage: typing.Optional['SQLiteStorage'] = None):
|
||||||
|
self.loop = loop
|
||||||
|
self.internal_udp_port = internal_udp_port
|
||||||
|
self.protocol = KademliaProtocol(loop, peer_manager, node_id, external_ip, udp_port, peer_port, rpc_timeout,
|
||||||
|
split_buckets_under_index, is_bootstrap_node)
|
||||||
|
self.listening_port: asyncio.DatagramTransport = None
|
||||||
|
self.joined = asyncio.Event()
|
||||||
|
self._join_task: asyncio.Task = None
|
||||||
|
self._refresh_task: asyncio.Task = None
|
||||||
|
self._storage = storage
|
||||||
|
|
||||||
|
@property
|
||||||
|
def stored_blob_hashes(self):
|
||||||
|
return self.protocol.data_store.keys()
|
||||||
|
|
||||||
|
async def refresh_node(self, force_once=False):
|
||||||
|
while True:
|
||||||
|
# remove peers with expired blob announcements from the datastore
|
||||||
|
self.protocol.data_store.removed_expired_peers()
|
||||||
|
|
||||||
|
total_peers: typing.List['KademliaPeer'] = []
|
||||||
|
# add all peers in the routing table
|
||||||
|
total_peers.extend(self.protocol.routing_table.get_peers())
|
||||||
|
# add all the peers who have announced blobs to us
|
||||||
|
storing_peers = self.protocol.data_store.get_storing_contacts()
|
||||||
|
self.storing_peers_metric.labels("global").set(len(storing_peers))
|
||||||
|
total_peers.extend(storing_peers)
|
||||||
|
|
||||||
|
counts = {0: 0, 1: 0, 2: 0}
|
||||||
|
node_id = self.protocol.node_id
|
||||||
|
for blob_hash in self.protocol.data_store.keys():
|
||||||
|
bytes_colliding = 0 if blob_hash[0] != node_id[0] else 2 if blob_hash[1] == node_id[1] else 1
|
||||||
|
counts[bytes_colliding] += 1
|
||||||
|
self.stored_blob_with_x_bytes_colliding.labels(amount=0).set(counts[0])
|
||||||
|
self.stored_blob_with_x_bytes_colliding.labels(amount=1).set(counts[1])
|
||||||
|
self.stored_blob_with_x_bytes_colliding.labels(amount=2).set(counts[2])
|
||||||
|
|
||||||
|
# get ids falling in the midpoint of each bucket that hasn't been recently updated
|
||||||
|
node_ids = self.protocol.routing_table.get_refresh_list(0, True)
|
||||||
|
|
||||||
|
if self.protocol.routing_table.get_peers():
|
||||||
|
# if we have node ids to look up, perform the iterative search until we have k results
|
||||||
|
while node_ids:
|
||||||
|
peers = await self.peer_search(node_ids.pop())
|
||||||
|
total_peers.extend(peers)
|
||||||
|
else:
|
||||||
|
if force_once:
|
||||||
|
break
|
||||||
|
fut = asyncio.Future()
|
||||||
|
self.loop.call_later(constants.REFRESH_INTERVAL // 4, fut.set_result, None)
|
||||||
|
await fut
|
||||||
|
continue
|
||||||
|
|
||||||
|
# ping the set of peers; upon success/failure the routing able and last replied/failed time will be updated
|
||||||
|
to_ping = [peer for peer in set(total_peers) if self.protocol.peer_manager.peer_is_good(peer) is not True]
|
||||||
|
if to_ping:
|
||||||
|
self.protocol.ping_queue.enqueue_maybe_ping(*to_ping, delay=0)
|
||||||
|
if self._storage:
|
||||||
|
await self._storage.save_kademlia_peers(self.protocol.routing_table.get_peers())
|
||||||
|
if force_once:
|
||||||
|
break
|
||||||
|
|
||||||
|
fut = asyncio.Future()
|
||||||
|
self.loop.call_later(constants.REFRESH_INTERVAL, fut.set_result, None)
|
||||||
|
await fut
|
||||||
|
|
||||||
|
async def announce_blob(self, blob_hash: str) -> typing.List[bytes]:
|
||||||
|
hash_value = bytes.fromhex(blob_hash)
|
||||||
|
assert len(hash_value) == constants.HASH_LENGTH
|
||||||
|
peers = await self.peer_search(hash_value)
|
||||||
|
|
||||||
|
if not self.protocol.external_ip:
|
||||||
|
raise Exception("Cannot determine external IP")
|
||||||
|
log.debug("Store to %i peers", len(peers))
|
||||||
|
for peer in peers:
|
||||||
|
log.debug("store to %s %s %s", peer.address, peer.udp_port, peer.tcp_port)
|
||||||
|
stored_to_tup = await asyncio.gather(
|
||||||
|
*(self.protocol.store_to_peer(hash_value, peer) for peer in peers)
|
||||||
|
)
|
||||||
|
stored_to = [node_id for node_id, contacted in stored_to_tup if contacted]
|
||||||
|
if stored_to:
|
||||||
|
log.debug(
|
||||||
|
"Stored %s to %i of %i attempted peers", hash_value.hex()[:8],
|
||||||
|
len(stored_to), len(peers)
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
log.debug("Failed announcing %s, stored to 0 peers", blob_hash[:8])
|
||||||
|
return stored_to
|
||||||
|
|
||||||
|
def stop(self) -> None:
|
||||||
|
if self.joined.is_set():
|
||||||
|
self.joined.clear()
|
||||||
|
if self._join_task:
|
||||||
|
self._join_task.cancel()
|
||||||
|
if self._refresh_task and not (self._refresh_task.done() or self._refresh_task.cancelled()):
|
||||||
|
self._refresh_task.cancel()
|
||||||
|
if self.protocol and self.protocol.ping_queue.running:
|
||||||
|
self.protocol.ping_queue.stop()
|
||||||
|
self.protocol.stop()
|
||||||
|
if self.listening_port is not None:
|
||||||
|
self.listening_port.close()
|
||||||
|
self._join_task = None
|
||||||
|
self.listening_port = None
|
||||||
|
log.info("Stopped DHT node")
|
||||||
|
|
||||||
|
async def start_listening(self, interface: str = '0.0.0.0') -> None:
|
||||||
|
if not self.listening_port:
|
||||||
|
self.listening_port, _ = await self.loop.create_datagram_endpoint(
|
||||||
|
lambda: self.protocol, (interface, self.internal_udp_port)
|
||||||
|
)
|
||||||
|
log.info("DHT node listening on UDP %s:%i", interface, self.internal_udp_port)
|
||||||
|
self.protocol.start()
|
||||||
|
else:
|
||||||
|
log.warning("Already bound to port %s", self.listening_port)
|
||||||
|
|
||||||
|
async def join_network(self, interface: str = '0.0.0.0',
|
||||||
|
known_node_urls: typing.Optional[typing.List[typing.Tuple[str, int]]] = None):
|
||||||
|
def peers_from_urls(urls: typing.Optional[typing.List[typing.Tuple[bytes, str, int, int]]]):
|
||||||
|
peer_addresses = []
|
||||||
|
for node_id, address, udp_port, tcp_port in urls:
|
||||||
|
if (node_id, address, udp_port, tcp_port) not in peer_addresses and \
|
||||||
|
(address, udp_port) != (self.protocol.external_ip, self.protocol.udp_port):
|
||||||
|
peer_addresses.append((node_id, address, udp_port, tcp_port))
|
||||||
|
return [make_kademlia_peer(*peer_address) for peer_address in peer_addresses]
|
||||||
|
|
||||||
|
if not self.listening_port:
|
||||||
|
await self.start_listening(interface)
|
||||||
|
self.protocol.ping_queue.start()
|
||||||
|
self._refresh_task = self.loop.create_task(self.refresh_node())
|
||||||
|
|
||||||
|
while True:
|
||||||
|
if self.protocol.routing_table.get_peers():
|
||||||
|
if not self.joined.is_set():
|
||||||
|
self.joined.set()
|
||||||
|
log.info(
|
||||||
|
"joined dht, %i peers known in %i buckets", len(self.protocol.routing_table.get_peers()),
|
||||||
|
self.protocol.routing_table.buckets_with_contacts()
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if self.joined.is_set():
|
||||||
|
self.joined.clear()
|
||||||
|
seed_peers = peers_from_urls(
|
||||||
|
await self._storage.get_persisted_kademlia_peers()
|
||||||
|
) if self._storage else []
|
||||||
|
if not seed_peers:
|
||||||
|
try:
|
||||||
|
seed_peers.extend(peers_from_urls([
|
||||||
|
(None, await resolve_host(address, udp_port, 'udp'), udp_port, None)
|
||||||
|
for address, udp_port in known_node_urls or []
|
||||||
|
]))
|
||||||
|
except socket.gaierror:
|
||||||
|
await asyncio.sleep(30)
|
||||||
|
continue
|
||||||
|
|
||||||
|
self.protocol.peer_manager.reset()
|
||||||
|
self.protocol.ping_queue.enqueue_maybe_ping(*seed_peers, delay=0.0)
|
||||||
|
await self.peer_search(self.protocol.node_id, shortlist=seed_peers, count=32)
|
||||||
|
|
||||||
|
await asyncio.sleep(1)
|
||||||
|
|
||||||
|
def start(self, interface: str, known_node_urls: typing.Optional[typing.List[typing.Tuple[str, int]]] = None):
|
||||||
|
self._join_task = self.loop.create_task(self.join_network(interface, known_node_urls))
|
||||||
|
|
||||||
|
def get_iterative_node_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
||||||
|
max_results: int = constants.K) -> IterativeNodeFinder:
|
||||||
|
shortlist = shortlist or self.protocol.routing_table.find_close_peers(key)
|
||||||
|
return IterativeNodeFinder(self.loop, self.protocol, key, max_results, shortlist)
|
||||||
|
|
||||||
|
def get_iterative_value_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
||||||
|
max_results: int = -1) -> IterativeValueFinder:
|
||||||
|
shortlist = shortlist or self.protocol.routing_table.find_close_peers(key)
|
||||||
|
return IterativeValueFinder(self.loop, self.protocol, key, max_results, shortlist)
|
||||||
|
|
||||||
|
async def peer_search(self, node_id: bytes, count=constants.K, max_results=constants.K * 2,
|
||||||
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None
|
||||||
|
) -> typing.List['KademliaPeer']:
|
||||||
|
peers = []
|
||||||
|
async with aclosing(self.get_iterative_node_finder(
|
||||||
|
node_id, shortlist=shortlist, max_results=max_results)) as node_finder:
|
||||||
|
async for iteration_peers in node_finder:
|
||||||
|
peers.extend(iteration_peers)
|
||||||
|
distance = Distance(node_id)
|
||||||
|
peers.sort(key=lambda peer: distance(peer.node_id))
|
||||||
|
return peers[:count]
|
||||||
|
|
||||||
|
async def _accumulate_peers_for_value(self, search_queue: asyncio.Queue, result_queue: asyncio.Queue):
|
||||||
|
tasks = []
|
||||||
|
try:
|
||||||
|
while True:
|
||||||
|
blob_hash = await search_queue.get()
|
||||||
|
tasks.append(self.loop.create_task(self._peers_for_value_producer(blob_hash, result_queue)))
|
||||||
|
finally:
|
||||||
|
for task in tasks:
|
||||||
|
task.cancel()
|
||||||
|
|
||||||
|
async def _peers_for_value_producer(self, blob_hash: str, result_queue: asyncio.Queue):
|
||||||
|
async def put_into_result_queue_after_pong(_peer):
|
||||||
|
try:
|
||||||
|
await self.protocol.get_rpc_peer(_peer).ping()
|
||||||
|
result_queue.put_nowait([_peer])
|
||||||
|
log.debug("pong from %s:%i for %s", _peer.address, _peer.udp_port, blob_hash)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
# prioritize peers who reply to a dht ping first
|
||||||
|
# this minimizes attempting to make tcp connections that won't work later to dead or unreachable peers
|
||||||
|
async with aclosing(self.get_iterative_value_finder(bytes.fromhex(blob_hash))) as value_finder:
|
||||||
|
async for results in value_finder:
|
||||||
|
to_put = []
|
||||||
|
for peer in results:
|
||||||
|
if peer.address == self.protocol.external_ip and self.protocol.peer_port == peer.tcp_port:
|
||||||
|
continue
|
||||||
|
is_good = self.protocol.peer_manager.peer_is_good(peer)
|
||||||
|
if is_good:
|
||||||
|
# the peer has replied recently over UDP, it can probably be reached on the TCP port
|
||||||
|
to_put.append(peer)
|
||||||
|
elif is_good is None:
|
||||||
|
if not peer.udp_port:
|
||||||
|
# TODO: use the same port for TCP and UDP
|
||||||
|
# the udp port must be guessed
|
||||||
|
# default to the ports being the same. if the TCP port appears to be <=0.48.0 default,
|
||||||
|
# including on a network with several nodes, then assume the udp port is proportionately
|
||||||
|
# based on a starting port of 4444
|
||||||
|
udp_port_to_try = peer.tcp_port
|
||||||
|
if 3400 > peer.tcp_port > 3332:
|
||||||
|
udp_port_to_try = (peer.tcp_port - 3333) + 4444
|
||||||
|
self.loop.create_task(put_into_result_queue_after_pong(
|
||||||
|
make_kademlia_peer(peer.node_id, peer.address, udp_port_to_try, peer.tcp_port)
|
||||||
|
))
|
||||||
|
else:
|
||||||
|
self.loop.create_task(put_into_result_queue_after_pong(peer))
|
||||||
|
else:
|
||||||
|
# the peer is known to be bad/unreachable, skip trying to connect to it over TCP
|
||||||
|
log.debug("skip bad peer %s:%i for %s", peer.address, peer.tcp_port, blob_hash)
|
||||||
|
if to_put:
|
||||||
|
result_queue.put_nowait(to_put)
|
||||||
|
|
||||||
|
def accumulate_peers(self, search_queue: asyncio.Queue,
|
||||||
|
peer_queue: typing.Optional[asyncio.Queue] = None
|
||||||
|
) -> typing.Tuple[asyncio.Queue, asyncio.Task]:
|
||||||
|
queue = peer_queue or asyncio.Queue()
|
||||||
|
return queue, self.loop.create_task(self._accumulate_peers_for_value(search_queue, queue))
|
||||||
|
|
||||||
|
|
||||||
|
async def get_kademlia_peers_from_hosts(peer_list: typing.List[typing.Tuple[str, int]]) -> typing.List['KademliaPeer']:
|
||||||
|
peer_address_list = [(await resolve_host(url, port, proto='tcp'), port) for url, port in peer_list]
|
||||||
|
kademlia_peer_list = [make_kademlia_peer(None, address, None, tcp_port=port, allow_localhost=True)
|
||||||
|
for address, port in peer_address_list]
|
||||||
|
return kademlia_peer_list
|
|
@ -1,36 +1,54 @@
|
||||||
import typing
|
import typing
|
||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
import ipaddress
|
from dataclasses import dataclass, field
|
||||||
from binascii import hexlify
|
|
||||||
from functools import lru_cache
|
from functools import lru_cache
|
||||||
|
|
||||||
from lbrynet.dht import constants
|
from prometheus_client import Gauge
|
||||||
from lbrynet.dht.serialization.datagram import make_compact_address, make_compact_ip, decode_compact_address
|
|
||||||
|
|
||||||
|
from lbry.utils import is_valid_public_ipv4 as _is_valid_public_ipv4, LRUCache
|
||||||
|
from lbry.dht import constants
|
||||||
|
from lbry.dht.serialization.datagram import make_compact_address, make_compact_ip, decode_compact_address
|
||||||
|
|
||||||
|
ALLOW_LOCALHOST = False
|
||||||
|
CACHE_SIZE = 16384
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def is_valid_ipv4(address):
|
@lru_cache(CACHE_SIZE)
|
||||||
try:
|
def make_kademlia_peer(node_id: typing.Optional[bytes], address: typing.Optional[str],
|
||||||
ip = ipaddress.ip_address(address)
|
udp_port: typing.Optional[int] = None,
|
||||||
return ip.version == 4
|
tcp_port: typing.Optional[int] = None,
|
||||||
except ipaddress.AddressValueError:
|
allow_localhost: bool = False) -> 'KademliaPeer':
|
||||||
return False
|
return KademliaPeer(address, node_id, udp_port, tcp_port=tcp_port, allow_localhost=allow_localhost)
|
||||||
|
|
||||||
|
|
||||||
|
def is_valid_public_ipv4(address, allow_localhost: bool = False):
|
||||||
|
allow_localhost = bool(allow_localhost or ALLOW_LOCALHOST)
|
||||||
|
return _is_valid_public_ipv4(address, allow_localhost)
|
||||||
|
|
||||||
|
|
||||||
class PeerManager:
|
class PeerManager:
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop):
|
peer_manager_keys_metric = Gauge(
|
||||||
|
"peer_manager_keys", "Number of keys tracked by PeerManager dicts (sum)", namespace="dht_node",
|
||||||
|
labelnames=("scope",)
|
||||||
|
)
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
self._rpc_failures: typing.Dict[
|
self._rpc_failures: typing.Dict[
|
||||||
typing.Tuple[str, int], typing.Tuple[typing.Optional[float], typing.Optional[float]]
|
typing.Tuple[str, int], typing.Tuple[typing.Optional[float], typing.Optional[float]]
|
||||||
] = {}
|
] = LRUCache(CACHE_SIZE)
|
||||||
self._last_replied: typing.Dict[typing.Tuple[str, int], float] = {}
|
self._last_replied: typing.Dict[typing.Tuple[str, int], float] = LRUCache(CACHE_SIZE)
|
||||||
self._last_sent: typing.Dict[typing.Tuple[str, int], float] = {}
|
self._last_sent: typing.Dict[typing.Tuple[str, int], float] = LRUCache(CACHE_SIZE)
|
||||||
self._last_requested: typing.Dict[typing.Tuple[str, int], float] = {}
|
self._last_requested: typing.Dict[typing.Tuple[str, int], float] = LRUCache(CACHE_SIZE)
|
||||||
self._node_id_mapping: typing.Dict[typing.Tuple[str, int], bytes] = {}
|
self._node_id_mapping: typing.Dict[typing.Tuple[str, int], bytes] = LRUCache(CACHE_SIZE)
|
||||||
self._node_id_reverse_mapping: typing.Dict[bytes, typing.Tuple[str, int]] = {}
|
self._node_id_reverse_mapping: typing.Dict[bytes, typing.Tuple[str, int]] = LRUCache(CACHE_SIZE)
|
||||||
self._node_tokens: typing.Dict[bytes, (float, bytes)] = {}
|
self._node_tokens: typing.Dict[bytes, (float, bytes)] = LRUCache(CACHE_SIZE)
|
||||||
|
|
||||||
|
def count_cache_keys(self):
|
||||||
|
return len(self._rpc_failures) + len(self._last_replied) + len(self._last_sent) + len(
|
||||||
|
self._last_requested) + len(self._node_id_mapping) + len(self._node_id_reverse_mapping) + len(
|
||||||
|
self._node_tokens)
|
||||||
|
|
||||||
def reset(self):
|
def reset(self):
|
||||||
for statistic in (self._rpc_failures, self._last_replied, self._last_sent, self._last_requested):
|
for statistic in (self._rpc_failures, self._last_replied, self._last_sent, self._last_requested):
|
||||||
|
@ -61,8 +79,8 @@ class PeerManager:
|
||||||
self._node_tokens[node_id] = (now, token)
|
self._node_tokens[node_id] = (now, token)
|
||||||
|
|
||||||
def get_node_token(self, node_id: bytes) -> typing.Optional[bytes]:
|
def get_node_token(self, node_id: bytes) -> typing.Optional[bytes]:
|
||||||
ts, token = self._node_tokens.get(node_id, (None, None))
|
ts, token = self._node_tokens.get(node_id, (0, None))
|
||||||
if ts and ts > self._loop.time() - constants.token_secret_refresh_interval:
|
if ts and ts > self._loop.time() - constants.TOKEN_SECRET_REFRESH_INTERVAL:
|
||||||
return token
|
return token
|
||||||
|
|
||||||
def get_last_replied(self, address: str, udp_port: int) -> typing.Optional[float]:
|
def get_last_replied(self, address: str, udp_port: int) -> typing.Optional[float]:
|
||||||
|
@ -80,42 +98,44 @@ class PeerManager:
|
||||||
self._node_id_mapping.pop(self._node_id_reverse_mapping.pop(node_id))
|
self._node_id_mapping.pop(self._node_id_reverse_mapping.pop(node_id))
|
||||||
self._node_id_mapping[(address, udp_port)] = node_id
|
self._node_id_mapping[(address, udp_port)] = node_id
|
||||||
self._node_id_reverse_mapping[node_id] = (address, udp_port)
|
self._node_id_reverse_mapping[node_id] = (address, udp_port)
|
||||||
|
self.peer_manager_keys_metric.labels("global").set(self.count_cache_keys())
|
||||||
|
|
||||||
@lru_cache(maxsize=400)
|
def get_node_id_for_endpoint(self, address, port):
|
||||||
def get_kademlia_peer(self, node_id: bytes, address: str, udp_port: int) -> 'KademliaPeer':
|
return self._node_id_mapping.get((address, port))
|
||||||
return KademliaPeer(self._loop, address, node_id, udp_port)
|
|
||||||
|
|
||||||
def prune(self): # TODO: periodically call this
|
def prune(self): # TODO: periodically call this
|
||||||
now = self._loop.time()
|
now = self._loop.time()
|
||||||
to_pop = []
|
to_pop = []
|
||||||
for (address, udp_port), (_, last_failure) in self._rpc_failures.items():
|
for (address, udp_port), (_, last_failure) in self._rpc_failures.items():
|
||||||
if last_failure and last_failure < now - constants.rpc_attempts_pruning_window:
|
if last_failure and last_failure < now - constants.RPC_ATTEMPTS_PRUNING_WINDOW:
|
||||||
to_pop.append((address, udp_port))
|
to_pop.append((address, udp_port))
|
||||||
while to_pop:
|
while to_pop:
|
||||||
del self._rpc_failures[to_pop.pop()]
|
del self._rpc_failures[to_pop.pop()]
|
||||||
to_pop = []
|
to_pop = []
|
||||||
for node_id, (age, token) in self._node_tokens.items():
|
for node_id, (age, token) in self._node_tokens.items(): # pylint: disable=unused-variable
|
||||||
if age < now - constants.token_secret_refresh_interval:
|
if age < now - constants.TOKEN_SECRET_REFRESH_INTERVAL:
|
||||||
to_pop.append(node_id)
|
to_pop.append(node_id)
|
||||||
while to_pop:
|
while to_pop:
|
||||||
del self._node_tokens[to_pop.pop()]
|
del self._node_tokens[to_pop.pop()]
|
||||||
|
|
||||||
def contact_triple_is_good(self, node_id: bytes, address: str, udp_port: int):
|
def contact_triple_is_good(self, node_id: bytes, address: str, udp_port: int): # pylint: disable=too-many-return-statements
|
||||||
"""
|
"""
|
||||||
:return: False if peer is bad, None if peer is unknown, or True if peer is good
|
:return: False if peer is bad, None if peer is unknown, or True if peer is good
|
||||||
"""
|
"""
|
||||||
|
|
||||||
delay = self._loop.time() - constants.check_refresh_interval
|
delay = self._loop.time() - constants.CHECK_REFRESH_INTERVAL
|
||||||
|
|
||||||
# fixme: find a way to re-enable that without breaking other parts
|
# fixme: find a way to re-enable that without breaking other parts
|
||||||
#if node_id not in self._node_id_reverse_mapping or (address, udp_port) not in self._node_id_mapping:
|
# if node_id not in self._node_id_reverse_mapping or (address, udp_port) not in self._node_id_mapping:
|
||||||
# return
|
# return
|
||||||
#addr_tup = (address, udp_port)
|
# addr_tup = (address, udp_port)
|
||||||
#if self._node_id_reverse_mapping[node_id] != addr_tup or self._node_id_mapping[addr_tup] != node_id:
|
# if self._node_id_reverse_mapping[node_id] != addr_tup or self._node_id_mapping[addr_tup] != node_id:
|
||||||
# return
|
# return
|
||||||
previous_failure, most_recent_failure = self._rpc_failures.get((address, udp_port), (None, None))
|
previous_failure, most_recent_failure = self._rpc_failures.get((address, udp_port), (None, None))
|
||||||
last_requested = self._last_requested.get((address, udp_port))
|
last_requested = self._last_requested.get((address, udp_port))
|
||||||
last_replied = self._last_replied.get((address, udp_port))
|
last_replied = self._last_replied.get((address, udp_port))
|
||||||
|
if node_id is None:
|
||||||
|
return None
|
||||||
if most_recent_failure and last_replied:
|
if most_recent_failure and last_replied:
|
||||||
if delay < last_replied > most_recent_failure:
|
if delay < last_replied > most_recent_failure:
|
||||||
return True
|
return True
|
||||||
|
@ -133,49 +153,35 @@ class PeerManager:
|
||||||
def peer_is_good(self, peer: 'KademliaPeer'):
|
def peer_is_good(self, peer: 'KademliaPeer'):
|
||||||
return self.contact_triple_is_good(peer.node_id, peer.address, peer.udp_port)
|
return self.contact_triple_is_good(peer.node_id, peer.address, peer.udp_port)
|
||||||
|
|
||||||
def decode_tcp_peer_from_compact_address(self, compact_address: bytes) -> 'KademliaPeer':
|
|
||||||
node_id, address, tcp_port = decode_compact_address(compact_address)
|
def decode_tcp_peer_from_compact_address(compact_address: bytes) -> 'KademliaPeer': # pylint: disable=no-self-use
|
||||||
return KademliaPeer(self._loop, address, node_id, tcp_port=tcp_port)
|
node_id, address, tcp_port = decode_compact_address(compact_address)
|
||||||
|
return make_kademlia_peer(node_id, address, udp_port=None, tcp_port=tcp_port)
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass(unsafe_hash=True)
|
||||||
class KademliaPeer:
|
class KademliaPeer:
|
||||||
__slots__ = [
|
address: str = field(hash=True)
|
||||||
'loop',
|
_node_id: typing.Optional[bytes] = field(hash=True)
|
||||||
'_node_id',
|
udp_port: typing.Optional[int] = field(hash=True)
|
||||||
'address',
|
tcp_port: typing.Optional[int] = field(compare=False, hash=False)
|
||||||
'udp_port',
|
protocol_version: typing.Optional[int] = field(default=1, compare=False, hash=False)
|
||||||
'tcp_port',
|
allow_localhost: bool = field(default=False, compare=False, hash=False)
|
||||||
'protocol_version',
|
|
||||||
]
|
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, address: str, node_id: typing.Optional[bytes] = None,
|
def __post_init__(self):
|
||||||
udp_port: typing.Optional[int] = None, tcp_port: typing.Optional[int] = None):
|
if self._node_id is not None:
|
||||||
if node_id is not None:
|
if not len(self._node_id) == constants.HASH_LENGTH:
|
||||||
if not len(node_id) == constants.hash_length:
|
raise ValueError("invalid node_id: {}".format(self._node_id.hex()))
|
||||||
raise ValueError("invalid node_id: {}".format(hexlify(node_id).decode()))
|
if self.udp_port is not None and not 1024 <= self.udp_port <= 65535:
|
||||||
if udp_port is not None and not 0 <= udp_port <= 65536:
|
raise ValueError(f"invalid udp port: {self.address}:{self.udp_port}")
|
||||||
raise ValueError("invalid udp port")
|
if self.tcp_port is not None and not 1024 <= self.tcp_port <= 65535:
|
||||||
if tcp_port and not 0 <= tcp_port <= 65536:
|
raise ValueError(f"invalid tcp port: {self.address}:{self.tcp_port}")
|
||||||
raise ValueError("invalid tcp port")
|
if not is_valid_public_ipv4(self.address, self.allow_localhost):
|
||||||
if not is_valid_ipv4(address):
|
raise ValueError(f"invalid ip address: '{self.address}'")
|
||||||
raise ValueError("invalid ip address")
|
|
||||||
self.loop = loop
|
|
||||||
self._node_id = node_id
|
|
||||||
self.address = address
|
|
||||||
self.udp_port = udp_port
|
|
||||||
self.tcp_port = tcp_port
|
|
||||||
self.protocol_version = 1
|
|
||||||
|
|
||||||
def update_tcp_port(self, tcp_port: int):
|
def update_tcp_port(self, tcp_port: int):
|
||||||
self.tcp_port = tcp_port
|
self.tcp_port = tcp_port
|
||||||
|
|
||||||
def update_udp_port(self, udp_port: int):
|
|
||||||
self.udp_port = udp_port
|
|
||||||
|
|
||||||
def set_id(self, node_id):
|
|
||||||
if not self._node_id:
|
|
||||||
self._node_id = node_id
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def node_id(self) -> bytes:
|
def node_id(self) -> bytes:
|
||||||
return self._node_id
|
return self._node_id
|
||||||
|
@ -189,10 +195,5 @@ class KademliaPeer:
|
||||||
def compact_ip(self):
|
def compact_ip(self):
|
||||||
return make_compact_ip(self.address)
|
return make_compact_ip(self.address)
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __str__(self):
|
||||||
if not isinstance(other, KademliaPeer):
|
return f"{self.__class__.__name__}({self.node_id.hex()[:8]}@{self.address}:{self.udp_port}-{self.tcp_port})"
|
||||||
raise TypeError("invalid type to compare with Peer: %s" % str(type(other)))
|
|
||||||
return (self.node_id, self.address, self.udp_port) == (other.node_id, other.address, other.udp_port)
|
|
||||||
|
|
||||||
def __hash__(self):
|
|
||||||
return hash((self.node_id, self.address, self.udp_port))
|
|
|
@ -1,13 +1,13 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import typing
|
import typing
|
||||||
|
|
||||||
from lbrynet.dht import constants
|
from lbry.dht import constants
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbrynet.dht.peer import KademliaPeer, PeerManager
|
from lbry.dht.peer import KademliaPeer, PeerManager
|
||||||
|
|
||||||
|
|
||||||
class DictDataStore:
|
class DictDataStore:
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, peer_manager: 'PeerManager'):
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager'):
|
||||||
# Dictionary format:
|
# Dictionary format:
|
||||||
# { <key>: [(<contact>, <age>), ...] }
|
# { <key>: [(<contact>, <age>), ...] }
|
||||||
self._data_store: typing.Dict[bytes, typing.List[typing.Tuple['KademliaPeer', float]]] = {}
|
self._data_store: typing.Dict[bytes, typing.List[typing.Tuple['KademliaPeer', float]]] = {}
|
||||||
|
@ -16,13 +16,19 @@ class DictDataStore:
|
||||||
self._peer_manager = peer_manager
|
self._peer_manager = peer_manager
|
||||||
self.completed_blobs: typing.Set[str] = set()
|
self.completed_blobs: typing.Set[str] = set()
|
||||||
|
|
||||||
|
def keys(self):
|
||||||
|
return self._data_store.keys()
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return self._data_store.__len__()
|
||||||
|
|
||||||
def removed_expired_peers(self):
|
def removed_expired_peers(self):
|
||||||
now = self.loop.time()
|
now = self.loop.time()
|
||||||
keys = list(self._data_store.keys())
|
keys = list(self._data_store.keys())
|
||||||
for key in keys:
|
for key in keys:
|
||||||
to_remove = []
|
to_remove = []
|
||||||
for (peer, ts) in self._data_store[key]:
|
for (peer, ts) in self._data_store[key]:
|
||||||
if ts + constants.data_expiration < now or self._peer_manager.peer_is_good(peer) is False:
|
if ts + constants.DATA_EXPIRATION < now or self._peer_manager.peer_is_good(peer) is False:
|
||||||
to_remove.append((peer, ts))
|
to_remove.append((peer, ts))
|
||||||
for item in to_remove:
|
for item in to_remove:
|
||||||
self._data_store[key].remove(item)
|
self._data_store[key].remove(item)
|
||||||
|
@ -43,7 +49,7 @@ class DictDataStore:
|
||||||
"""
|
"""
|
||||||
now = self.loop.time()
|
now = self.loop.time()
|
||||||
for (peer, ts) in self._data_store.get(key, []):
|
for (peer, ts) in self._data_store.get(key, []):
|
||||||
if ts + constants.data_expiration > now:
|
if ts + constants.DATA_EXPIRATION > now:
|
||||||
yield peer
|
yield peer
|
||||||
|
|
||||||
def has_peers_for_blob(self, key: bytes) -> bool:
|
def has_peers_for_blob(self, key: bytes) -> bool:
|
||||||
|
@ -53,7 +59,7 @@ class DictDataStore:
|
||||||
now = self.loop.time()
|
now = self.loop.time()
|
||||||
if key in self._data_store:
|
if key in self._data_store:
|
||||||
current = list(filter(lambda x: x[0] == contact, self._data_store[key]))
|
current = list(filter(lambda x: x[0] == contact, self._data_store[key]))
|
||||||
if len(current):
|
if len(current) > 0:
|
||||||
self._data_store[key][self._data_store[key].index(current[0])] = contact, now
|
self._data_store[key][self._data_store[key].index(current[0])] = contact, now
|
||||||
else:
|
else:
|
||||||
self._data_store[key].append((contact, now))
|
self._data_store[key].append((contact, now))
|
||||||
|
@ -65,6 +71,6 @@ class DictDataStore:
|
||||||
|
|
||||||
def get_storing_contacts(self) -> typing.List['KademliaPeer']:
|
def get_storing_contacts(self) -> typing.List['KademliaPeer']:
|
||||||
peers = set()
|
peers = set()
|
||||||
for key, stored in self._data_store.items():
|
for _, stored in self._data_store.items():
|
||||||
peers.update(set(map(lambda tup: tup[0], stored)))
|
peers.update(set(map(lambda tup: tup[0], stored)))
|
||||||
return list(peers)
|
return list(peers)
|
|
@ -1,4 +1,4 @@
|
||||||
from lbrynet.dht import constants
|
from lbry.dht import constants
|
||||||
|
|
||||||
|
|
||||||
class Distance:
|
class Distance:
|
||||||
|
@ -9,17 +9,17 @@ class Distance:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, key: bytes):
|
def __init__(self, key: bytes):
|
||||||
if len(key) != constants.hash_length:
|
if len(key) != constants.HASH_LENGTH:
|
||||||
raise ValueError(f"invalid key length: {len(key)}")
|
raise ValueError(f"invalid key length: {len(key)}")
|
||||||
self.key = key
|
self.key = key
|
||||||
self.val_key_one = int.from_bytes(key, 'big')
|
self.val_key_one = int.from_bytes(key, 'big')
|
||||||
|
|
||||||
def __call__(self, key_two: bytes) -> int:
|
def __call__(self, key_two: bytes) -> int:
|
||||||
if len(key_two) != constants.hash_length:
|
if len(key_two) != constants.HASH_LENGTH:
|
||||||
raise ValueError(f"invalid length of key to compare: {len(key_two)}")
|
raise ValueError(f"invalid length of key to compare: {len(key_two)}")
|
||||||
val_key_two = int.from_bytes(key_two, 'big')
|
val_key_two = int.from_bytes(key_two, 'big')
|
||||||
return self.val_key_one ^ val_key_two
|
return self.val_key_one ^ val_key_two
|
||||||
|
|
||||||
def is_closer(self, a: bytes, b: bytes) -> bool:
|
def is_closer(self, key_a: bytes, key_b: bytes) -> bool:
|
||||||
"""Returns true is `a` is closer to `key` than `b` is"""
|
"""Returns true is `key_a` is closer to `key` than `key_b` is"""
|
||||||
return self(a) < self(b)
|
return self(key_a) < self(key_b)
|
361
lbry/dht/protocol/iterative_find.py
Normal file
361
lbry/dht/protocol/iterative_find.py
Normal file
|
@ -0,0 +1,361 @@
|
||||||
|
import asyncio
|
||||||
|
from itertools import chain
|
||||||
|
from collections import defaultdict, OrderedDict
|
||||||
|
from collections.abc import AsyncIterator
|
||||||
|
import typing
|
||||||
|
import logging
|
||||||
|
from typing import TYPE_CHECKING
|
||||||
|
from lbry.dht import constants
|
||||||
|
from lbry.dht.error import RemoteException, TransportNotConnected
|
||||||
|
from lbry.dht.protocol.distance import Distance
|
||||||
|
from lbry.dht.peer import make_kademlia_peer, decode_tcp_peer_from_compact_address
|
||||||
|
from lbry.dht.serialization.datagram import PAGE_KEY
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from lbry.dht.protocol.protocol import KademliaProtocol
|
||||||
|
from lbry.dht.peer import PeerManager, KademliaPeer
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class FindResponse:
|
||||||
|
@property
|
||||||
|
def found(self) -> bool:
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def get_close_kademlia_peers(self, peer_info) -> typing.Generator[typing.Iterator['KademliaPeer'], None, None]:
|
||||||
|
for contact_triple in self.get_close_triples():
|
||||||
|
node_id, address, udp_port = contact_triple
|
||||||
|
try:
|
||||||
|
yield make_kademlia_peer(node_id, address, udp_port)
|
||||||
|
except ValueError:
|
||||||
|
log.warning("misbehaving peer %s:%i returned peer with reserved ip %s:%i", peer_info.address,
|
||||||
|
peer_info.udp_port, address, udp_port)
|
||||||
|
|
||||||
|
|
||||||
|
class FindNodeResponse(FindResponse):
|
||||||
|
def __init__(self, key: bytes, close_triples: typing.List[typing.Tuple[bytes, str, int]]):
|
||||||
|
self.key = key
|
||||||
|
self.close_triples = close_triples
|
||||||
|
|
||||||
|
@property
|
||||||
|
def found(self) -> bool:
|
||||||
|
return self.key in [triple[0] for triple in self.close_triples]
|
||||||
|
|
||||||
|
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
||||||
|
return self.close_triples
|
||||||
|
|
||||||
|
|
||||||
|
class FindValueResponse(FindResponse):
|
||||||
|
def __init__(self, key: bytes, result_dict: typing.Dict):
|
||||||
|
self.key = key
|
||||||
|
self.token = result_dict[b'token']
|
||||||
|
self.close_triples: typing.List[typing.Tuple[bytes, bytes, int]] = result_dict.get(b'contacts', [])
|
||||||
|
self.found_compact_addresses = result_dict.get(key, [])
|
||||||
|
self.pages = int(result_dict.get(PAGE_KEY, 0))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def found(self) -> bool:
|
||||||
|
return len(self.found_compact_addresses) > 0
|
||||||
|
|
||||||
|
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
||||||
|
return [(node_id, address.decode(), port) for node_id, address, port in self.close_triples]
|
||||||
|
|
||||||
|
|
||||||
|
class IterativeFinder(AsyncIterator):
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop,
|
||||||
|
protocol: 'KademliaProtocol', key: bytes,
|
||||||
|
max_results: typing.Optional[int] = constants.K,
|
||||||
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
||||||
|
if len(key) != constants.HASH_LENGTH:
|
||||||
|
raise ValueError("invalid key length: %i" % len(key))
|
||||||
|
self.loop = loop
|
||||||
|
self.peer_manager = protocol.peer_manager
|
||||||
|
self.protocol = protocol
|
||||||
|
|
||||||
|
self.key = key
|
||||||
|
self.max_results = max(constants.K, max_results)
|
||||||
|
|
||||||
|
self.active: typing.Dict['KademliaPeer', int] = OrderedDict() # peer: distance, sorted
|
||||||
|
self.contacted: typing.Set['KademliaPeer'] = set()
|
||||||
|
self.distance = Distance(key)
|
||||||
|
|
||||||
|
self.iteration_queue = asyncio.Queue()
|
||||||
|
|
||||||
|
self.running_probes: typing.Dict['KademliaPeer', asyncio.Task] = {}
|
||||||
|
self.iteration_count = 0
|
||||||
|
self.running = False
|
||||||
|
self.tasks: typing.List[asyncio.Task] = []
|
||||||
|
for peer in shortlist:
|
||||||
|
if peer.node_id:
|
||||||
|
self._add_active(peer, force=True)
|
||||||
|
else:
|
||||||
|
# seed nodes
|
||||||
|
self._schedule_probe(peer)
|
||||||
|
|
||||||
|
async def send_probe(self, peer: 'KademliaPeer') -> FindResponse:
|
||||||
|
"""
|
||||||
|
Send the rpc request to the peer and return an object with the FindResponse interface
|
||||||
|
"""
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def search_exhausted(self):
|
||||||
|
"""
|
||||||
|
This method ends the iterator due no more peers to contact.
|
||||||
|
Override to provide last time results.
|
||||||
|
"""
|
||||||
|
self.iteration_queue.put_nowait(None)
|
||||||
|
|
||||||
|
def check_result_ready(self, response: FindResponse):
|
||||||
|
"""
|
||||||
|
Called after adding peers from an rpc result to the shortlist.
|
||||||
|
This method is responsible for putting a result for the generator into the Queue
|
||||||
|
"""
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def get_initial_result(self) -> typing.List['KademliaPeer']: #pylint: disable=no-self-use
|
||||||
|
"""
|
||||||
|
Get an initial or cached result to be put into the Queue. Used for findValue requests where the blob
|
||||||
|
has peers in the local data store of blobs announced to us
|
||||||
|
"""
|
||||||
|
return []
|
||||||
|
|
||||||
|
def _add_active(self, peer, force=False):
|
||||||
|
if not force and self.peer_manager.peer_is_good(peer) is False:
|
||||||
|
return
|
||||||
|
if peer in self.contacted:
|
||||||
|
return
|
||||||
|
if peer not in self.active and peer.node_id and peer.node_id != self.protocol.node_id:
|
||||||
|
self.active[peer] = self.distance(peer.node_id)
|
||||||
|
self.active = OrderedDict(sorted(self.active.items(), key=lambda item: item[1]))
|
||||||
|
|
||||||
|
async def _handle_probe_result(self, peer: 'KademliaPeer', response: FindResponse):
|
||||||
|
self._add_active(peer)
|
||||||
|
for new_peer in response.get_close_kademlia_peers(peer):
|
||||||
|
self._add_active(new_peer)
|
||||||
|
self.check_result_ready(response)
|
||||||
|
self._log_state(reason="check result")
|
||||||
|
|
||||||
|
def _reset_closest(self, peer):
|
||||||
|
if peer in self.active:
|
||||||
|
del self.active[peer]
|
||||||
|
|
||||||
|
async def _send_probe(self, peer: 'KademliaPeer'):
|
||||||
|
try:
|
||||||
|
response = await self.send_probe(peer)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
self._reset_closest(peer)
|
||||||
|
return
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
log.debug("%s[%x] cancelled probe",
|
||||||
|
type(self).__name__, id(self))
|
||||||
|
raise
|
||||||
|
except ValueError as err:
|
||||||
|
log.warning(str(err))
|
||||||
|
self._reset_closest(peer)
|
||||||
|
return
|
||||||
|
except TransportNotConnected:
|
||||||
|
await self._aclose(reason="not connected")
|
||||||
|
return
|
||||||
|
except RemoteException:
|
||||||
|
self._reset_closest(peer)
|
||||||
|
return
|
||||||
|
return await self._handle_probe_result(peer, response)
|
||||||
|
|
||||||
|
def _search_round(self):
|
||||||
|
"""
|
||||||
|
Send up to constants.alpha (5) probes to closest active peers
|
||||||
|
"""
|
||||||
|
|
||||||
|
added = 0
|
||||||
|
for index, peer in enumerate(self.active.keys()):
|
||||||
|
if index == 0:
|
||||||
|
log.debug("%s[%x] closest to probe: %s",
|
||||||
|
type(self).__name__, id(self),
|
||||||
|
peer.node_id.hex()[:8])
|
||||||
|
if peer in self.contacted:
|
||||||
|
continue
|
||||||
|
if len(self.running_probes) >= constants.ALPHA:
|
||||||
|
break
|
||||||
|
if index > (constants.K + len(self.running_probes)):
|
||||||
|
break
|
||||||
|
origin_address = (peer.address, peer.udp_port)
|
||||||
|
if peer.node_id == self.protocol.node_id:
|
||||||
|
continue
|
||||||
|
if origin_address == (self.protocol.external_ip, self.protocol.udp_port):
|
||||||
|
continue
|
||||||
|
self._schedule_probe(peer)
|
||||||
|
added += 1
|
||||||
|
log.debug("%s[%x] running %d probes for key %s",
|
||||||
|
type(self).__name__, id(self),
|
||||||
|
len(self.running_probes), self.key.hex()[:8])
|
||||||
|
if not added and not self.running_probes:
|
||||||
|
log.debug("%s[%x] search for %s exhausted",
|
||||||
|
type(self).__name__, id(self),
|
||||||
|
self.key.hex()[:8])
|
||||||
|
self.search_exhausted()
|
||||||
|
|
||||||
|
def _schedule_probe(self, peer: 'KademliaPeer'):
|
||||||
|
self.contacted.add(peer)
|
||||||
|
|
||||||
|
t = self.loop.create_task(self._send_probe(peer))
|
||||||
|
|
||||||
|
def callback(_):
|
||||||
|
self.running_probes.pop(peer, None)
|
||||||
|
if self.running:
|
||||||
|
self._search_round()
|
||||||
|
|
||||||
|
t.add_done_callback(callback)
|
||||||
|
self.running_probes[peer] = t
|
||||||
|
|
||||||
|
def _log_state(self, reason="?"):
|
||||||
|
log.debug("%s[%x] [%s] %s: %i active nodes %i contacted %i produced %i queued",
|
||||||
|
type(self).__name__, id(self), self.key.hex()[:8],
|
||||||
|
reason, len(self.active), len(self.contacted),
|
||||||
|
self.iteration_count, self.iteration_queue.qsize())
|
||||||
|
|
||||||
|
def __aiter__(self):
|
||||||
|
if self.running:
|
||||||
|
raise Exception("already running")
|
||||||
|
self.running = True
|
||||||
|
self.loop.call_soon(self._search_round)
|
||||||
|
return self
|
||||||
|
|
||||||
|
async def __anext__(self) -> typing.List['KademliaPeer']:
|
||||||
|
try:
|
||||||
|
if self.iteration_count == 0:
|
||||||
|
result = self.get_initial_result() or await self.iteration_queue.get()
|
||||||
|
else:
|
||||||
|
result = await self.iteration_queue.get()
|
||||||
|
if not result:
|
||||||
|
raise StopAsyncIteration
|
||||||
|
self.iteration_count += 1
|
||||||
|
return result
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
await self._aclose(reason="cancelled")
|
||||||
|
raise
|
||||||
|
except StopAsyncIteration:
|
||||||
|
await self._aclose(reason="no more results")
|
||||||
|
raise
|
||||||
|
|
||||||
|
async def _aclose(self, reason="?"):
|
||||||
|
log.debug("%s[%x] [%s] shutdown because %s: %i active nodes %i contacted %i produced %i queued",
|
||||||
|
type(self).__name__, id(self), self.key.hex()[:8],
|
||||||
|
reason, len(self.active), len(self.contacted),
|
||||||
|
self.iteration_count, self.iteration_queue.qsize())
|
||||||
|
self.running = False
|
||||||
|
self.iteration_queue.put_nowait(None)
|
||||||
|
for task in chain(self.tasks, self.running_probes.values()):
|
||||||
|
task.cancel()
|
||||||
|
self.tasks.clear()
|
||||||
|
self.running_probes.clear()
|
||||||
|
|
||||||
|
async def aclose(self):
|
||||||
|
if self.running:
|
||||||
|
await self._aclose(reason="aclose")
|
||||||
|
log.debug("%s[%x] [%s] async close completed",
|
||||||
|
type(self).__name__, id(self), self.key.hex()[:8])
|
||||||
|
|
||||||
|
class IterativeNodeFinder(IterativeFinder):
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop,
|
||||||
|
protocol: 'KademliaProtocol', key: bytes,
|
||||||
|
max_results: typing.Optional[int] = constants.K,
|
||||||
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
||||||
|
super().__init__(loop, protocol, key, max_results, shortlist)
|
||||||
|
self.yielded_peers: typing.Set['KademliaPeer'] = set()
|
||||||
|
|
||||||
|
async def send_probe(self, peer: 'KademliaPeer') -> FindNodeResponse:
|
||||||
|
log.debug("probe %s:%d (%s) for NODE %s",
|
||||||
|
peer.address, peer.udp_port, peer.node_id.hex()[:8] if peer.node_id else '', self.key.hex()[:8])
|
||||||
|
response = await self.protocol.get_rpc_peer(peer).find_node(self.key)
|
||||||
|
return FindNodeResponse(self.key, response)
|
||||||
|
|
||||||
|
def search_exhausted(self):
|
||||||
|
self.put_result(self.active.keys(), finish=True)
|
||||||
|
|
||||||
|
def put_result(self, from_iter: typing.Iterable['KademliaPeer'], finish=False):
|
||||||
|
not_yet_yielded = [
|
||||||
|
peer for peer in from_iter
|
||||||
|
if peer not in self.yielded_peers
|
||||||
|
and peer.node_id != self.protocol.node_id
|
||||||
|
and self.peer_manager.peer_is_good(peer) is True # return only peers who answered
|
||||||
|
]
|
||||||
|
not_yet_yielded.sort(key=lambda peer: self.distance(peer.node_id))
|
||||||
|
to_yield = not_yet_yielded[:max(constants.K, self.max_results)]
|
||||||
|
if to_yield:
|
||||||
|
self.yielded_peers.update(to_yield)
|
||||||
|
self.iteration_queue.put_nowait(to_yield)
|
||||||
|
if finish:
|
||||||
|
self.iteration_queue.put_nowait(None)
|
||||||
|
|
||||||
|
def check_result_ready(self, response: FindNodeResponse):
|
||||||
|
found = response.found and self.key != self.protocol.node_id
|
||||||
|
|
||||||
|
if found:
|
||||||
|
log.debug("found")
|
||||||
|
return self.put_result(self.active.keys(), finish=True)
|
||||||
|
|
||||||
|
|
||||||
|
class IterativeValueFinder(IterativeFinder):
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop,
|
||||||
|
protocol: 'KademliaProtocol', key: bytes,
|
||||||
|
max_results: typing.Optional[int] = constants.K,
|
||||||
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
||||||
|
super().__init__(loop, protocol, key, max_results, shortlist)
|
||||||
|
self.blob_peers: typing.Set['KademliaPeer'] = set()
|
||||||
|
# this tracks the index of the most recent page we requested from each peer
|
||||||
|
self.peer_pages: typing.DefaultDict['KademliaPeer', int] = defaultdict(int)
|
||||||
|
# this tracks the set of blob peers returned by each peer
|
||||||
|
self.discovered_peers: typing.Dict['KademliaPeer', typing.Set['KademliaPeer']] = defaultdict(set)
|
||||||
|
|
||||||
|
async def send_probe(self, peer: 'KademliaPeer') -> FindValueResponse:
|
||||||
|
log.debug("probe %s:%d (%s) for VALUE %s",
|
||||||
|
peer.address, peer.udp_port, peer.node_id.hex()[:8], self.key.hex()[:8])
|
||||||
|
page = self.peer_pages[peer]
|
||||||
|
response = await self.protocol.get_rpc_peer(peer).find_value(self.key, page=page)
|
||||||
|
parsed = FindValueResponse(self.key, response)
|
||||||
|
if not parsed.found:
|
||||||
|
return parsed
|
||||||
|
already_known = len(self.discovered_peers[peer])
|
||||||
|
decoded_peers = set()
|
||||||
|
for compact_addr in parsed.found_compact_addresses:
|
||||||
|
try:
|
||||||
|
decoded_peers.add(decode_tcp_peer_from_compact_address(compact_addr))
|
||||||
|
except ValueError:
|
||||||
|
log.warning("misbehaving peer %s:%i returned invalid peer for blob",
|
||||||
|
peer.address, peer.udp_port)
|
||||||
|
self.peer_manager.report_failure(peer.address, peer.udp_port)
|
||||||
|
parsed.found_compact_addresses.clear()
|
||||||
|
return parsed
|
||||||
|
self.discovered_peers[peer].update(decoded_peers)
|
||||||
|
log.debug("probed %s:%i page %i, %i known", peer.address, peer.udp_port, page,
|
||||||
|
already_known + len(parsed.found_compact_addresses))
|
||||||
|
if len(self.discovered_peers[peer]) != already_known + len(parsed.found_compact_addresses):
|
||||||
|
log.warning("misbehaving peer %s:%i returned duplicate peers for blob", peer.address, peer.udp_port)
|
||||||
|
elif len(parsed.found_compact_addresses) >= constants.K and self.peer_pages[peer] < parsed.pages:
|
||||||
|
# the peer returned a full page and indicates it has more
|
||||||
|
self.peer_pages[peer] += 1
|
||||||
|
if peer in self.contacted:
|
||||||
|
# the peer must be removed from self.contacted so that it will be probed for the next page
|
||||||
|
self.contacted.remove(peer)
|
||||||
|
return parsed
|
||||||
|
|
||||||
|
def check_result_ready(self, response: FindValueResponse):
|
||||||
|
if response.found:
|
||||||
|
blob_peers = [decode_tcp_peer_from_compact_address(compact_addr)
|
||||||
|
for compact_addr in response.found_compact_addresses]
|
||||||
|
to_yield = []
|
||||||
|
for blob_peer in blob_peers:
|
||||||
|
if blob_peer not in self.blob_peers:
|
||||||
|
self.blob_peers.add(blob_peer)
|
||||||
|
to_yield.append(blob_peer)
|
||||||
|
if to_yield:
|
||||||
|
self.iteration_queue.put_nowait(to_yield)
|
||||||
|
|
||||||
|
def get_initial_result(self) -> typing.List['KademliaPeer']:
|
||||||
|
if self.protocol.data_store.has_peers_for_blob(self.key):
|
||||||
|
return self.protocol.data_store.get_peers_for_blob(self.key)
|
||||||
|
return []
|
|
@ -3,32 +3,42 @@ import socket
|
||||||
import functools
|
import functools
|
||||||
import hashlib
|
import hashlib
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import time
|
||||||
import typing
|
import typing
|
||||||
import binascii
|
import random
|
||||||
from asyncio.protocols import DatagramProtocol
|
from asyncio.protocols import DatagramProtocol
|
||||||
from asyncio.transports import DatagramTransport
|
from asyncio.transports import DatagramTransport
|
||||||
|
|
||||||
from lbrynet.dht import constants
|
from prometheus_client import Gauge, Counter, Histogram
|
||||||
from lbrynet.dht.serialization.datagram import decode_datagram, ErrorDatagram, ResponseDatagram, RequestDatagram
|
|
||||||
from lbrynet.dht.serialization.datagram import RESPONSE_TYPE, ERROR_TYPE
|
from lbry.dht import constants
|
||||||
from lbrynet.dht.error import RemoteException, TransportNotConnected
|
from lbry.dht.serialization.bencoding import DecodeError
|
||||||
from lbrynet.dht.protocol.routing_table import TreeRoutingTable
|
from lbry.dht.serialization.datagram import decode_datagram, ErrorDatagram, ResponseDatagram, RequestDatagram
|
||||||
from lbrynet.dht.protocol.data_store import DictDataStore
|
from lbry.dht.serialization.datagram import RESPONSE_TYPE, ERROR_TYPE, PAGE_KEY
|
||||||
|
from lbry.dht.error import RemoteException, TransportNotConnected
|
||||||
|
from lbry.dht.protocol.routing_table import TreeRoutingTable
|
||||||
|
from lbry.dht.protocol.data_store import DictDataStore
|
||||||
|
from lbry.dht.peer import make_kademlia_peer
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbrynet.dht.peer import PeerManager, KademliaPeer
|
from lbry.dht.peer import PeerManager, KademliaPeer
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
old_protocol_errors = {
|
OLD_PROTOCOL_ERRORS = {
|
||||||
"findNode() takes exactly 2 arguments (5 given)": "0.19.1",
|
"findNode() takes exactly 2 arguments (5 given)": "0.19.1",
|
||||||
"findValue() takes exactly 2 arguments (5 given)": "0.19.1"
|
"findValue() takes exactly 2 arguments (5 given)": "0.19.1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class KademliaRPC:
|
class KademliaRPC:
|
||||||
def __init__(self, protocol: 'KademliaProtocol', loop: asyncio.BaseEventLoop, peer_port: int = 3333):
|
stored_blob_metric = Gauge(
|
||||||
|
"stored_blobs", "Number of blobs announced by other peers", namespace="dht_node",
|
||||||
|
labelnames=("scope",),
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, protocol: 'KademliaProtocol', loop: asyncio.AbstractEventLoop, peer_port: int = 3333):
|
||||||
self.protocol = protocol
|
self.protocol = protocol
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.peer_port = peer_port
|
self.peer_port = peer_port
|
||||||
|
@ -46,40 +56,45 @@ class KademliaRPC:
|
||||||
return b'pong'
|
return b'pong'
|
||||||
|
|
||||||
def store(self, rpc_contact: 'KademliaPeer', blob_hash: bytes, token: bytes, port: int) -> bytes:
|
def store(self, rpc_contact: 'KademliaPeer', blob_hash: bytes, token: bytes, port: int) -> bytes:
|
||||||
if len(blob_hash) != constants.hash_bits // 8:
|
if len(blob_hash) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid length of blob hash: {len(blob_hash)}")
|
raise ValueError(f"invalid length of blob hash: {len(blob_hash)}")
|
||||||
if not 0 < port < 65535:
|
if not 0 < port < 65535:
|
||||||
raise ValueError(f"invalid tcp port: {port}")
|
raise ValueError(f"invalid tcp port: {port}")
|
||||||
rpc_contact.update_tcp_port(port)
|
rpc_contact.update_tcp_port(port)
|
||||||
if not self.verify_token(token, rpc_contact.compact_ip()):
|
if not self.verify_token(token, rpc_contact.compact_ip()):
|
||||||
if self.loop.time() - self.protocol.started_listening_time < constants.token_secret_refresh_interval:
|
if self.loop.time() - self.protocol.started_listening_time < constants.TOKEN_SECRET_REFRESH_INTERVAL:
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
raise ValueError("Invalid token")
|
raise ValueError("Invalid token")
|
||||||
self.protocol.data_store.add_peer_to_blob(
|
self.protocol.data_store.add_peer_to_blob(
|
||||||
rpc_contact, blob_hash
|
rpc_contact, blob_hash
|
||||||
)
|
)
|
||||||
|
self.stored_blob_metric.labels("global").set(len(self.protocol.data_store))
|
||||||
return b'OK'
|
return b'OK'
|
||||||
|
|
||||||
def find_node(self, rpc_contact: 'KademliaPeer', key: bytes) -> typing.List[typing.Tuple[bytes, str, int]]:
|
def find_node(self, rpc_contact: 'KademliaPeer', key: bytes) -> typing.List[typing.Tuple[bytes, str, int]]:
|
||||||
if len(key) != constants.hash_length:
|
if len(key) != constants.HASH_LENGTH:
|
||||||
raise ValueError("invalid contact node_id length: %i" % len(key))
|
raise ValueError("invalid contact node_id length: %i" % len(key))
|
||||||
|
|
||||||
contacts = self.protocol.routing_table.find_close_peers(key, sender_node_id=rpc_contact.node_id)
|
contacts = self.protocol.routing_table.find_close_peers(key, sender_node_id=rpc_contact.node_id)
|
||||||
contact_triples = []
|
contact_triples = []
|
||||||
for contact in contacts:
|
for contact in contacts[:constants.K * 2]:
|
||||||
contact_triples.append((contact.node_id, contact.address, contact.udp_port))
|
contact_triples.append((contact.node_id, contact.address, contact.udp_port))
|
||||||
return contact_triples
|
return contact_triples
|
||||||
|
|
||||||
def find_value(self, rpc_contact: 'KademliaPeer', key: bytes):
|
def find_value(self, rpc_contact: 'KademliaPeer', key: bytes, page: int = 0):
|
||||||
if len(key) != constants.hash_length:
|
page = page if page > 0 else 0
|
||||||
|
|
||||||
|
if len(key) != constants.HASH_LENGTH:
|
||||||
raise ValueError("invalid blob_exchange hash length: %i" % len(key))
|
raise ValueError("invalid blob_exchange hash length: %i" % len(key))
|
||||||
|
|
||||||
response = {
|
response = {
|
||||||
b'token': self.make_token(rpc_contact.compact_ip()),
|
b'token': self.make_token(rpc_contact.compact_ip()),
|
||||||
b'contacts': self.find_node(rpc_contact, key)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if not page:
|
||||||
|
response[b'contacts'] = self.find_node(rpc_contact, key)[:constants.K]
|
||||||
|
|
||||||
if self.protocol.protocol_version:
|
if self.protocol.protocol_version:
|
||||||
response[b'protocolVersion'] = self.protocol.protocol_version
|
response[b'protocolVersion'] = self.protocol.protocol_version
|
||||||
|
|
||||||
|
@ -90,10 +105,16 @@ class KademliaRPC:
|
||||||
if not rpc_contact.tcp_port or peer.compact_address_tcp() != rpc_contact.compact_address_tcp()
|
if not rpc_contact.tcp_port or peer.compact_address_tcp() != rpc_contact.compact_address_tcp()
|
||||||
]
|
]
|
||||||
# if we don't have k storing peers to return and we have this hash locally, include our contact information
|
# if we don't have k storing peers to return and we have this hash locally, include our contact information
|
||||||
if len(peers) < constants.k and binascii.hexlify(key).decode() in self.protocol.data_store.completed_blobs:
|
if len(peers) < constants.K and key.hex() in self.protocol.data_store.completed_blobs:
|
||||||
peers.append(self.compact_address())
|
peers.append(self.compact_address())
|
||||||
if peers:
|
if not peers:
|
||||||
response[key] = peers
|
response[PAGE_KEY] = 0
|
||||||
|
else:
|
||||||
|
response[PAGE_KEY] = (len(peers) // (constants.K + 1)) + 1 # how many pages of peers we have for the blob
|
||||||
|
if len(peers) > constants.K:
|
||||||
|
random.Random(self.protocol.node_id).shuffle(peers)
|
||||||
|
if page * constants.K < len(peers):
|
||||||
|
response[key] = peers[page * constants.K:page * constants.K + constants.K]
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def refresh_token(self): # TODO: this needs to be called periodically
|
def refresh_token(self): # TODO: this needs to be called periodically
|
||||||
|
@ -121,7 +142,7 @@ class RemoteKademliaRPC:
|
||||||
Encapsulates RPC calls to remote Peers
|
Encapsulates RPC calls to remote Peers
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, peer_tracker: 'PeerManager', protocol: 'KademliaProtocol',
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_tracker: 'PeerManager', protocol: 'KademliaProtocol',
|
||||||
peer: 'KademliaPeer'):
|
peer: 'KademliaPeer'):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.peer_tracker = peer_tracker
|
self.peer_tracker = peer_tracker
|
||||||
|
@ -142,7 +163,7 @@ class RemoteKademliaRPC:
|
||||||
:param blob_hash: blob hash as bytes
|
:param blob_hash: blob hash as bytes
|
||||||
:return: b'OK'
|
:return: b'OK'
|
||||||
"""
|
"""
|
||||||
if len(blob_hash) != constants.hash_bits // 8:
|
if len(blob_hash) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid length of blob hash: {len(blob_hash)}")
|
raise ValueError(f"invalid length of blob hash: {len(blob_hash)}")
|
||||||
if not self.protocol.peer_port or not 0 < self.protocol.peer_port < 65535:
|
if not self.protocol.peer_port or not 0 < self.protocol.peer_port < 65535:
|
||||||
raise ValueError(f"invalid tcp port: {self.protocol.peer_port}")
|
raise ValueError(f"invalid tcp port: {self.protocol.peer_port}")
|
||||||
|
@ -159,14 +180,14 @@ class RemoteKademliaRPC:
|
||||||
"""
|
"""
|
||||||
:return: [(node_id, address, udp_port), ...]
|
:return: [(node_id, address, udp_port), ...]
|
||||||
"""
|
"""
|
||||||
if len(key) != constants.hash_bits // 8:
|
if len(key) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid length of find node key: {len(key)}")
|
raise ValueError(f"invalid length of find node key: {len(key)}")
|
||||||
response = await self.protocol.send_request(
|
response = await self.protocol.send_request(
|
||||||
self.peer, RequestDatagram.make_find_node(self.protocol.node_id, key)
|
self.peer, RequestDatagram.make_find_node(self.protocol.node_id, key)
|
||||||
)
|
)
|
||||||
return [(node_id, address.decode(), udp_port) for node_id, address, udp_port in response.response]
|
return [(node_id, address.decode(), udp_port) for node_id, address, udp_port in response.response]
|
||||||
|
|
||||||
async def find_value(self, key: bytes) -> typing.Union[typing.Dict]:
|
async def find_value(self, key: bytes, page: int = 0) -> typing.Union[typing.Dict]:
|
||||||
"""
|
"""
|
||||||
:return: {
|
:return: {
|
||||||
b'token': <token bytes>,
|
b'token': <token bytes>,
|
||||||
|
@ -174,29 +195,33 @@ class RemoteKademliaRPC:
|
||||||
<key bytes>: [<blob_peer_compact_address, ...]
|
<key bytes>: [<blob_peer_compact_address, ...]
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
if len(key) != constants.hash_bits // 8:
|
if len(key) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid length of find value key: {len(key)}")
|
raise ValueError(f"invalid length of find value key: {len(key)}")
|
||||||
response = await self.protocol.send_request(
|
response = await self.protocol.send_request(
|
||||||
self.peer, RequestDatagram.make_find_value(self.protocol.node_id, key)
|
self.peer, RequestDatagram.make_find_value(self.protocol.node_id, key, page=page)
|
||||||
)
|
)
|
||||||
self.peer_tracker.update_token(self.peer.node_id, response.response[b'token'])
|
self.peer_tracker.update_token(self.peer.node_id, response.response[b'token'])
|
||||||
return response.response
|
return response.response
|
||||||
|
|
||||||
|
|
||||||
class PingQueue:
|
class PingQueue:
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, protocol: 'KademliaProtocol'):
|
def __init__(self, loop: asyncio.AbstractEventLoop, protocol: 'KademliaProtocol'):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
self._protocol = protocol
|
self._protocol = protocol
|
||||||
self._pending_contacts: typing.Dict['KademliaPeer', float] = {}
|
self._pending_contacts: typing.Dict['KademliaPeer', float] = {}
|
||||||
self._process_task: asyncio.Task = None
|
self._process_task: asyncio.Task = None
|
||||||
self._running = False
|
self._running = False
|
||||||
self._running_pings: typing.Set[asyncio.Task] = set()
|
self._running_pings: typing.Set[asyncio.Task] = set()
|
||||||
self._default_delay = constants.maybe_ping_delay
|
self._default_delay = constants.MAYBE_PING_DELAY
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def running(self):
|
def running(self):
|
||||||
return self._running
|
return self._running
|
||||||
|
|
||||||
|
@property
|
||||||
|
def busy(self):
|
||||||
|
return self._running and (any(self._running_pings) or any(self._pending_contacts))
|
||||||
|
|
||||||
def enqueue_maybe_ping(self, *peers: 'KademliaPeer', delay: typing.Optional[float] = None):
|
def enqueue_maybe_ping(self, *peers: 'KademliaPeer', delay: typing.Optional[float] = None):
|
||||||
delay = delay if delay is not None else self._default_delay
|
delay = delay if delay is not None else self._default_delay
|
||||||
now = self._loop.time()
|
now = self._loop.time()
|
||||||
|
@ -208,11 +233,11 @@ class PingQueue:
|
||||||
async def ping_task():
|
async def ping_task():
|
||||||
try:
|
try:
|
||||||
if self._protocol.peer_manager.peer_is_good(peer):
|
if self._protocol.peer_manager.peer_is_good(peer):
|
||||||
if peer not in self._protocol.routing_table.get_peers():
|
if not self._protocol.routing_table.get_peer(peer.node_id):
|
||||||
self._protocol.add_peer(peer)
|
self._protocol.add_peer(peer)
|
||||||
return
|
return
|
||||||
await self._protocol.get_rpc_peer(peer).ping()
|
await self._protocol.get_rpc_peer(peer).ping()
|
||||||
except asyncio.TimeoutError:
|
except (asyncio.TimeoutError, RemoteException):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
task = self._loop.create_task(ping_task())
|
task = self._loop.create_task(ping_task())
|
||||||
|
@ -228,7 +253,7 @@ class PingQueue:
|
||||||
del self._pending_contacts[peer]
|
del self._pending_contacts[peer]
|
||||||
self.maybe_ping(peer)
|
self.maybe_ping(peer)
|
||||||
break
|
break
|
||||||
await asyncio.sleep(1, loop=self._loop)
|
await asyncio.sleep(1)
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
assert not self._running
|
assert not self._running
|
||||||
|
@ -247,9 +272,33 @@ class PingQueue:
|
||||||
|
|
||||||
|
|
||||||
class KademliaProtocol(DatagramProtocol):
|
class KademliaProtocol(DatagramProtocol):
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, peer_manager: 'PeerManager', node_id: bytes, external_ip: str,
|
request_sent_metric = Counter(
|
||||||
udp_port: int, peer_port: int, rpc_timeout: float = constants.rpc_timeout,
|
"request_sent", "Number of requests send from DHT RPC protocol", namespace="dht_node",
|
||||||
split_buckets_under_index: int = constants.split_buckets_under_index):
|
labelnames=("method",),
|
||||||
|
)
|
||||||
|
request_success_metric = Counter(
|
||||||
|
"request_success", "Number of successful requests", namespace="dht_node",
|
||||||
|
labelnames=("method",),
|
||||||
|
)
|
||||||
|
request_error_metric = Counter(
|
||||||
|
"request_error", "Number of errors returned from request to other peers", namespace="dht_node",
|
||||||
|
labelnames=("method",),
|
||||||
|
)
|
||||||
|
HISTOGRAM_BUCKETS = (
|
||||||
|
.005, .01, .025, .05, .075, .1, .25, .5, .75, 1.0, 2.5, 3.0, 3.5, 4.0, 4.50, 5.0, 5.50, 6.0, float('inf')
|
||||||
|
)
|
||||||
|
response_time_metric = Histogram(
|
||||||
|
"response_time", "Response times of DHT RPC requests", namespace="dht_node", buckets=HISTOGRAM_BUCKETS,
|
||||||
|
labelnames=("method",)
|
||||||
|
)
|
||||||
|
received_request_metric = Counter(
|
||||||
|
"received_request", "Number of received DHT RPC requests", namespace="dht_node",
|
||||||
|
labelnames=("method",),
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', node_id: bytes, external_ip: str,
|
||||||
|
udp_port: int, peer_port: int, rpc_timeout: float = constants.RPC_TIMEOUT,
|
||||||
|
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX, is_boostrap_node: bool = False):
|
||||||
self.peer_manager = peer_manager
|
self.peer_manager = peer_manager
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.node_id = node_id
|
self.node_id = node_id
|
||||||
|
@ -259,20 +308,21 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
self.is_seed_node = False
|
self.is_seed_node = False
|
||||||
self.partial_messages: typing.Dict[bytes, typing.Dict[bytes, bytes]] = {}
|
self.partial_messages: typing.Dict[bytes, typing.Dict[bytes, bytes]] = {}
|
||||||
self.sent_messages: typing.Dict[bytes, typing.Tuple['KademliaPeer', asyncio.Future, RequestDatagram]] = {}
|
self.sent_messages: typing.Dict[bytes, typing.Tuple['KademliaPeer', asyncio.Future, RequestDatagram]] = {}
|
||||||
self.protocol_version = constants.protocol_version
|
self.protocol_version = constants.PROTOCOL_VERSION
|
||||||
self.started_listening_time = 0
|
self.started_listening_time = 0
|
||||||
self.transport: DatagramTransport = None
|
self.transport: DatagramTransport = None
|
||||||
self.old_token_secret = constants.generate_id()
|
self.old_token_secret = constants.generate_id()
|
||||||
self.token_secret = constants.generate_id()
|
self.token_secret = constants.generate_id()
|
||||||
self.routing_table = TreeRoutingTable(self.loop, self.peer_manager, self.node_id, split_buckets_under_index)
|
self.routing_table = TreeRoutingTable(
|
||||||
|
self.loop, self.peer_manager, self.node_id, split_buckets_under_index, is_bootstrap_node=is_boostrap_node)
|
||||||
self.data_store = DictDataStore(self.loop, self.peer_manager)
|
self.data_store = DictDataStore(self.loop, self.peer_manager)
|
||||||
self.ping_queue = PingQueue(self.loop, self)
|
self.ping_queue = PingQueue(self.loop, self)
|
||||||
self.node_rpc = KademliaRPC(self, self.loop, self.peer_port)
|
self.node_rpc = KademliaRPC(self, self.loop, self.peer_port)
|
||||||
self.rpc_timeout = rpc_timeout
|
self.rpc_timeout = rpc_timeout
|
||||||
self._split_lock = asyncio.Lock(loop=self.loop)
|
self._split_lock = asyncio.Lock()
|
||||||
self._to_remove: typing.Set['KademliaPeer'] = set()
|
self._to_remove: typing.Set['KademliaPeer'] = set()
|
||||||
self._to_add: typing.Set['KademliaPeer'] = set()
|
self._to_add: typing.Set['KademliaPeer'] = set()
|
||||||
self._wakeup_routing_task = asyncio.Event(loop=self.loop)
|
self._wakeup_routing_task = asyncio.Event()
|
||||||
self.maintaing_routing_task: typing.Optional[asyncio.Task] = None
|
self.maintaing_routing_task: typing.Optional[asyncio.Task] = None
|
||||||
|
|
||||||
@functools.lru_cache(128)
|
@functools.lru_cache(128)
|
||||||
|
@ -311,69 +361,10 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
return args, {}
|
return args, {}
|
||||||
|
|
||||||
async def _add_peer(self, peer: 'KademliaPeer'):
|
async def _add_peer(self, peer: 'KademliaPeer'):
|
||||||
for p in self.routing_table.get_peers():
|
async def probe(some_peer: 'KademliaPeer'):
|
||||||
if (p.address, p.udp_port) == (peer.address, peer.udp_port) and p.node_id != peer.node_id:
|
rpc_peer = self.get_rpc_peer(some_peer)
|
||||||
self.routing_table.remove_peer(p)
|
await rpc_peer.ping()
|
||||||
self.routing_table.join_buckets()
|
return await self.routing_table.add_peer(peer, probe)
|
||||||
bucket_index = self.routing_table.kbucket_index(peer.node_id)
|
|
||||||
if self.routing_table.buckets[bucket_index].add_peer(peer):
|
|
||||||
return True
|
|
||||||
|
|
||||||
# The bucket is full; see if it can be split (by checking if its range includes the host node's node_id)
|
|
||||||
if self.routing_table.should_split(bucket_index, peer.node_id):
|
|
||||||
self.routing_table.split_bucket(bucket_index)
|
|
||||||
# Retry the insertion attempt
|
|
||||||
result = await self._add_peer(peer)
|
|
||||||
self.routing_table.join_buckets()
|
|
||||||
return result
|
|
||||||
else:
|
|
||||||
# We can't split the k-bucket
|
|
||||||
#
|
|
||||||
# The 13 page kademlia paper specifies that the least recently contacted node in the bucket
|
|
||||||
# shall be pinged. If it fails to reply it is replaced with the new contact. If the ping is successful
|
|
||||||
# the new contact is ignored and not added to the bucket (sections 2.2 and 2.4).
|
|
||||||
#
|
|
||||||
# A reasonable extension to this is BEP 0005, which extends the above:
|
|
||||||
#
|
|
||||||
# Not all nodes that we learn about are equal. Some are "good" and some are not.
|
|
||||||
# Many nodes using the DHT are able to send queries and receive responses,
|
|
||||||
# but are not able to respond to queries from other nodes. It is important that
|
|
||||||
# each node's routing table must contain only known good nodes. A good node is
|
|
||||||
# a node has responded to one of our queries within the last 15 minutes. A node
|
|
||||||
# is also good if it has ever responded to one of our queries and has sent us a
|
|
||||||
# query within the last 15 minutes. After 15 minutes of inactivity, a node becomes
|
|
||||||
# questionable. Nodes become bad when they fail to respond to multiple queries
|
|
||||||
# in a row. Nodes that we know are good are given priority over nodes with unknown status.
|
|
||||||
#
|
|
||||||
# When there are bad or questionable nodes in the bucket, the least recent is selected for
|
|
||||||
# potential replacement (BEP 0005). When all nodes in the bucket are fresh, the head (least recent)
|
|
||||||
# contact is selected as described in section 2.2 of the kademlia paper. In both cases the new contact
|
|
||||||
# is ignored if the pinged node replies.
|
|
||||||
|
|
||||||
not_good_contacts = self.routing_table.buckets[bucket_index].get_bad_or_unknown_peers()
|
|
||||||
not_recently_replied = []
|
|
||||||
for p in not_good_contacts:
|
|
||||||
last_replied = self.peer_manager.get_last_replied(p.address, p.udp_port)
|
|
||||||
if not last_replied or last_replied + 60 < self.loop.time():
|
|
||||||
not_recently_replied.append(p)
|
|
||||||
if not_recently_replied:
|
|
||||||
to_replace = not_recently_replied[0]
|
|
||||||
else:
|
|
||||||
to_replace = self.routing_table.buckets[bucket_index].peers[0]
|
|
||||||
last_replied = self.peer_manager.get_last_replied(to_replace.address, to_replace.udp_port)
|
|
||||||
if last_replied and last_replied + 60 > self.loop.time():
|
|
||||||
return False
|
|
||||||
log.debug("pinging %s:%s", to_replace.address, to_replace.udp_port)
|
|
||||||
try:
|
|
||||||
to_replace_rpc = self.get_rpc_peer(to_replace)
|
|
||||||
await to_replace_rpc.ping()
|
|
||||||
return False
|
|
||||||
except asyncio.TimeoutError:
|
|
||||||
log.debug("Replacing dead contact in bucket %i: %s:%i with %s:%i ", bucket_index,
|
|
||||||
to_replace.address, to_replace.udp_port, peer.address, peer.udp_port)
|
|
||||||
if to_replace in self.routing_table.buckets[bucket_index]:
|
|
||||||
self.routing_table.buckets[bucket_index].remove_peer(to_replace)
|
|
||||||
return await self._add_peer(peer)
|
|
||||||
|
|
||||||
def add_peer(self, peer: 'KademliaPeer'):
|
def add_peer(self, peer: 'KademliaPeer'):
|
||||||
if peer.node_id == self.node_id:
|
if peer.node_id == self.node_id:
|
||||||
|
@ -391,39 +382,39 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
async with self._split_lock:
|
async with self._split_lock:
|
||||||
peer = self._to_remove.pop()
|
peer = self._to_remove.pop()
|
||||||
self.routing_table.remove_peer(peer)
|
self.routing_table.remove_peer(peer)
|
||||||
self.routing_table.join_buckets()
|
|
||||||
while self._to_add:
|
while self._to_add:
|
||||||
async with self._split_lock:
|
async with self._split_lock:
|
||||||
await self._add_peer(self._to_add.pop())
|
await self._add_peer(self._to_add.pop())
|
||||||
await asyncio.gather(self._wakeup_routing_task.wait(), asyncio.sleep(.1, loop=self.loop), loop=self.loop)
|
await asyncio.gather(self._wakeup_routing_task.wait(), asyncio.sleep(.1))
|
||||||
self._wakeup_routing_task.clear()
|
self._wakeup_routing_task.clear()
|
||||||
|
|
||||||
def _handle_rpc(self, sender_contact: 'KademliaPeer', message: RequestDatagram):
|
def _handle_rpc(self, sender_contact: 'KademliaPeer', message: RequestDatagram):
|
||||||
assert sender_contact.node_id != self.node_id, (binascii.hexlify(sender_contact.node_id)[:8].decode(),
|
assert sender_contact.node_id != self.node_id, (sender_contact.node_id.hex()[:8],
|
||||||
binascii.hexlify(self.node_id)[:8].decode())
|
self.node_id.hex()[:8])
|
||||||
method = message.method
|
method = message.method
|
||||||
if method not in [b'ping', b'store', b'findNode', b'findValue']:
|
if method not in [b'ping', b'store', b'findNode', b'findValue']:
|
||||||
raise AttributeError('Invalid method: %s' % message.method.decode())
|
raise AttributeError('Invalid method: %s' % message.method.decode())
|
||||||
if message.args and isinstance(message.args[-1], dict) and b'protocolVersion' in message.args[-1]:
|
if message.args and isinstance(message.args[-1], dict) and b'protocolVersion' in message.args[-1]:
|
||||||
# args don't need reformatting
|
# args don't need reformatting
|
||||||
a, kw = tuple(message.args[:-1]), message.args[-1]
|
args, kwargs = tuple(message.args[:-1]), message.args[-1]
|
||||||
else:
|
else:
|
||||||
a, kw = self._migrate_incoming_rpc_args(sender_contact, message.method, *message.args)
|
args, kwargs = self._migrate_incoming_rpc_args(sender_contact, message.method, *message.args)
|
||||||
log.debug("%s:%i RECV CALL %s %s:%i", self.external_ip, self.udp_port, message.method.decode(),
|
log.debug("%s:%i RECV CALL %s %s:%i", self.external_ip, self.udp_port, message.method.decode(),
|
||||||
sender_contact.address, sender_contact.udp_port)
|
sender_contact.address, sender_contact.udp_port)
|
||||||
|
|
||||||
if method == b'ping':
|
if method == b'ping':
|
||||||
result = self.node_rpc.ping()
|
result = self.node_rpc.ping()
|
||||||
elif method == b'store':
|
elif method == b'store':
|
||||||
blob_hash, token, port, original_publisher_id, age = a
|
blob_hash, token, port, original_publisher_id, age = args[:5] # pylint: disable=unused-variable
|
||||||
result = self.node_rpc.store(sender_contact, blob_hash, token, port)
|
result = self.node_rpc.store(sender_contact, blob_hash, token, port)
|
||||||
elif method == b'findNode':
|
|
||||||
key, = a
|
|
||||||
result = self.node_rpc.find_node(sender_contact, key)
|
|
||||||
else:
|
else:
|
||||||
assert method == b'findValue'
|
key = args[0]
|
||||||
key, = a
|
page = kwargs.get(PAGE_KEY, 0)
|
||||||
result = self.node_rpc.find_value(sender_contact, key)
|
if method == b'findNode':
|
||||||
|
result = self.node_rpc.find_node(sender_contact, key)
|
||||||
|
else:
|
||||||
|
assert method == b'findValue'
|
||||||
|
result = self.node_rpc.find_value(sender_contact, key, page)
|
||||||
|
|
||||||
self.send_response(
|
self.send_response(
|
||||||
sender_contact, ResponseDatagram(RESPONSE_TYPE, message.rpc_id, self.node_id, result),
|
sender_contact, ResponseDatagram(RESPONSE_TYPE, message.rpc_id, self.node_id, result),
|
||||||
|
@ -431,11 +422,15 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
|
|
||||||
def handle_request_datagram(self, address: typing.Tuple[str, int], request_datagram: RequestDatagram):
|
def handle_request_datagram(self, address: typing.Tuple[str, int], request_datagram: RequestDatagram):
|
||||||
# This is an RPC method request
|
# This is an RPC method request
|
||||||
|
self.received_request_metric.labels(method=request_datagram.method).inc()
|
||||||
self.peer_manager.report_last_requested(address[0], address[1])
|
self.peer_manager.report_last_requested(address[0], address[1])
|
||||||
try:
|
peer = self.routing_table.get_peer(request_datagram.node_id)
|
||||||
peer = self.routing_table.get_peer(request_datagram.node_id)
|
if not peer:
|
||||||
except IndexError:
|
try:
|
||||||
peer = self.peer_manager.get_kademlia_peer(request_datagram.node_id, address[0], address[1])
|
peer = make_kademlia_peer(request_datagram.node_id, address[0], address[1])
|
||||||
|
except ValueError as err:
|
||||||
|
log.warning("error replying to %s: %s", address[0], str(err))
|
||||||
|
return
|
||||||
try:
|
try:
|
||||||
self._handle_rpc(peer, request_datagram)
|
self._handle_rpc(peer, request_datagram)
|
||||||
# if the contact is not known to be bad (yet) and we haven't yet queried it, send it a ping so that it
|
# if the contact is not known to be bad (yet) and we haven't yet queried it, send it a ping so that it
|
||||||
|
@ -468,26 +463,25 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
def handle_response_datagram(self, address: typing.Tuple[str, int], response_datagram: ResponseDatagram):
|
def handle_response_datagram(self, address: typing.Tuple[str, int], response_datagram: ResponseDatagram):
|
||||||
# Find the message that triggered this response
|
# Find the message that triggered this response
|
||||||
if response_datagram.rpc_id in self.sent_messages:
|
if response_datagram.rpc_id in self.sent_messages:
|
||||||
peer, df, request = self.sent_messages[response_datagram.rpc_id]
|
peer, future, _ = self.sent_messages[response_datagram.rpc_id]
|
||||||
if peer.address != address[0]:
|
if peer.address != address[0]:
|
||||||
df.set_exception(RemoteException(
|
future.set_exception(
|
||||||
f"response from {address[0]}, expected {peer.address}")
|
RemoteException(f"response from {address[0]}, expected {peer.address}")
|
||||||
)
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
# We got a result from the RPC
|
# We got a result from the RPC
|
||||||
if peer.node_id == self.node_id:
|
if peer.node_id == self.node_id:
|
||||||
df.set_exception(RemoteException("node has our node id"))
|
future.set_exception(RemoteException("node has our node id"))
|
||||||
return
|
return
|
||||||
elif response_datagram.node_id == self.node_id:
|
elif response_datagram.node_id == self.node_id:
|
||||||
df.set_exception(RemoteException("incoming message is from our node id"))
|
future.set_exception(RemoteException("incoming message is from our node id"))
|
||||||
return
|
return
|
||||||
peer.set_id(response_datagram.node_id)
|
peer = make_kademlia_peer(response_datagram.node_id, address[0], address[1])
|
||||||
peer.update_udp_port(address[1])
|
|
||||||
self.peer_manager.report_last_replied(address[0], address[1])
|
self.peer_manager.report_last_replied(address[0], address[1])
|
||||||
self.peer_manager.update_contact_triple(peer.node_id, address[0], address[1])
|
self.peer_manager.update_contact_triple(peer.node_id, address[0], address[1])
|
||||||
if not df.cancelled():
|
if not future.cancelled():
|
||||||
df.set_result(response_datagram)
|
future.set_result(response_datagram)
|
||||||
self.add_peer(peer)
|
self.add_peer(peer)
|
||||||
else:
|
else:
|
||||||
log.warning("%s:%i replied, but after we cancelled the request attempt",
|
log.warning("%s:%i replied, but after we cancelled the request attempt",
|
||||||
|
@ -501,11 +495,13 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
# The RPC request raised a remote exception; raise it locally
|
# The RPC request raised a remote exception; raise it locally
|
||||||
remote_exception = RemoteException(f"{error_datagram.exception_type}({error_datagram.response})")
|
remote_exception = RemoteException(f"{error_datagram.exception_type}({error_datagram.response})")
|
||||||
if error_datagram.rpc_id in self.sent_messages:
|
if error_datagram.rpc_id in self.sent_messages:
|
||||||
peer, df, request = self.sent_messages.pop(error_datagram.rpc_id)
|
peer, future, request = self.sent_messages.pop(error_datagram.rpc_id)
|
||||||
if (peer.address, peer.udp_port) != address:
|
if (peer.address, peer.udp_port) != address:
|
||||||
df.set_exception(RemoteException(
|
future.set_exception(
|
||||||
f"response from {address[0]}:{address[1]}, "
|
RemoteException(
|
||||||
f"expected {peer.address}:{peer.udp_port}")
|
f"response from {address[0]}:{address[1]}, "
|
||||||
|
f"expected {peer.address}:{peer.udp_port}"
|
||||||
|
)
|
||||||
)
|
)
|
||||||
return
|
return
|
||||||
error_msg = f"" \
|
error_msg = f"" \
|
||||||
|
@ -514,28 +510,32 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
f"Raised: {str(remote_exception)}"
|
f"Raised: {str(remote_exception)}"
|
||||||
if 'Invalid token' in error_msg:
|
if 'Invalid token' in error_msg:
|
||||||
log.debug(error_msg)
|
log.debug(error_msg)
|
||||||
elif error_datagram.response not in old_protocol_errors:
|
elif error_datagram.response not in OLD_PROTOCOL_ERRORS:
|
||||||
log.warning(error_msg)
|
log.warning(error_msg)
|
||||||
else:
|
else:
|
||||||
log.warning("known dht protocol backwards compatibility error with %s:%i (lbrynet v%s)",
|
log.debug(
|
||||||
peer.address, peer.udp_port, old_protocol_errors[error_datagram.response])
|
"known dht protocol backwards compatibility error with %s:%i (lbrynet v%s)",
|
||||||
df.set_exception(remote_exception)
|
peer.address, peer.udp_port, OLD_PROTOCOL_ERRORS[error_datagram.response]
|
||||||
|
)
|
||||||
|
future.set_exception(remote_exception)
|
||||||
return
|
return
|
||||||
else:
|
else:
|
||||||
if error_datagram.response not in old_protocol_errors:
|
if error_datagram.response not in OLD_PROTOCOL_ERRORS:
|
||||||
msg = f"Received error from {address[0]}:{address[1]}, but it isn't in response to a " \
|
msg = f"Received error from {address[0]}:{address[1]}, but it isn't in response to a " \
|
||||||
f"pending request: {str(remote_exception)}"
|
f"pending request: {str(remote_exception)}"
|
||||||
log.warning(msg)
|
log.warning(msg)
|
||||||
else:
|
else:
|
||||||
log.warning("known dht protocol backwards compatibility error with %s:%i (lbrynet v%s)",
|
log.debug(
|
||||||
address[0], address[1], old_protocol_errors[error_datagram.response])
|
"known dht protocol backwards compatibility error with %s:%i (lbrynet v%s)",
|
||||||
|
address[0], address[1], OLD_PROTOCOL_ERRORS[error_datagram.response]
|
||||||
|
)
|
||||||
|
|
||||||
def datagram_received(self, datagram: bytes, address: typing.Tuple[str, int]) -> None:
|
def datagram_received(self, datagram: bytes, address: typing.Tuple[str, int]) -> None: # pylint: disable=arguments-renamed
|
||||||
try:
|
try:
|
||||||
message = decode_datagram(datagram)
|
message = decode_datagram(datagram)
|
||||||
except (ValueError, TypeError):
|
except (ValueError, TypeError, DecodeError):
|
||||||
self.peer_manager.report_failure(address[0], address[1])
|
self.peer_manager.report_failure(address[0], address[1])
|
||||||
log.warning("Couldn't decode dht datagram from %s: %s", address, binascii.hexlify(datagram).decode())
|
log.warning("Couldn't decode dht datagram from %s: %s", address, datagram.hex())
|
||||||
return
|
return
|
||||||
|
|
||||||
if isinstance(message, RequestDatagram):
|
if isinstance(message, RequestDatagram):
|
||||||
|
@ -550,14 +550,19 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
self._send(peer, request)
|
self._send(peer, request)
|
||||||
response_fut = self.sent_messages[request.rpc_id][1]
|
response_fut = self.sent_messages[request.rpc_id][1]
|
||||||
try:
|
try:
|
||||||
|
self.request_sent_metric.labels(method=request.method).inc()
|
||||||
|
start = time.perf_counter()
|
||||||
response = await asyncio.wait_for(response_fut, self.rpc_timeout)
|
response = await asyncio.wait_for(response_fut, self.rpc_timeout)
|
||||||
|
self.response_time_metric.labels(method=request.method).observe(time.perf_counter() - start)
|
||||||
self.peer_manager.report_last_replied(peer.address, peer.udp_port)
|
self.peer_manager.report_last_replied(peer.address, peer.udp_port)
|
||||||
|
self.request_success_metric.labels(method=request.method).inc()
|
||||||
return response
|
return response
|
||||||
except asyncio.CancelledError:
|
except asyncio.CancelledError:
|
||||||
if not response_fut.done():
|
if not response_fut.done():
|
||||||
response_fut.cancel()
|
response_fut.cancel()
|
||||||
raise
|
raise
|
||||||
except (asyncio.TimeoutError, RemoteException):
|
except (asyncio.TimeoutError, RemoteException):
|
||||||
|
self.request_error_metric.labels(method=request.method).inc()
|
||||||
self.peer_manager.report_failure(peer.address, peer.udp_port)
|
self.peer_manager.report_failure(peer.address, peer.udp_port)
|
||||||
if self.peer_manager.peer_is_good(peer) is False:
|
if self.peer_manager.peer_is_good(peer) is False:
|
||||||
self.remove_peer(peer)
|
self.remove_peer(peer)
|
||||||
|
@ -569,15 +574,18 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
def send_error(self, peer: 'KademliaPeer', error: ErrorDatagram):
|
def send_error(self, peer: 'KademliaPeer', error: ErrorDatagram):
|
||||||
self._send(peer, error)
|
self._send(peer, error)
|
||||||
|
|
||||||
def _send(self, peer: 'KademliaPeer', message: typing.Union[RequestDatagram, ResponseDatagram,
|
def _send(self, peer: 'KademliaPeer', message: typing.Union[RequestDatagram, ResponseDatagram, ErrorDatagram]):
|
||||||
ErrorDatagram]):
|
|
||||||
if not self.transport or self.transport.is_closing():
|
if not self.transport or self.transport.is_closing():
|
||||||
raise TransportNotConnected()
|
raise TransportNotConnected()
|
||||||
|
|
||||||
data = message.bencode()
|
data = message.bencode()
|
||||||
if len(data) > constants.msg_size_limit:
|
if len(data) > constants.MSG_SIZE_LIMIT:
|
||||||
log.exception("unexpected: %i vs %i", len(data), constants.msg_size_limit)
|
log.warning("cannot send datagram larger than %i bytes (packet is %i bytes)",
|
||||||
raise ValueError()
|
constants.MSG_SIZE_LIMIT, len(data))
|
||||||
|
log.debug("Packet is too large to send: %s", data[:3500].hex())
|
||||||
|
raise ValueError(
|
||||||
|
f"cannot send datagram larger than {constants.MSG_SIZE_LIMIT} bytes (packet is {len(data)} bytes)"
|
||||||
|
)
|
||||||
if isinstance(message, (RequestDatagram, ResponseDatagram)):
|
if isinstance(message, (RequestDatagram, ResponseDatagram)):
|
||||||
assert message.node_id == self.node_id, message
|
assert message.node_id == self.node_id, message
|
||||||
if isinstance(message, RequestDatagram):
|
if isinstance(message, RequestDatagram):
|
||||||
|
@ -619,55 +627,40 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
return constants.digest(self.token_secret + compact_ip)
|
return constants.digest(self.token_secret + compact_ip)
|
||||||
|
|
||||||
def verify_token(self, token, compact_ip):
|
def verify_token(self, token, compact_ip):
|
||||||
h = constants.hash_class()
|
h = constants.HASH_CLASS()
|
||||||
h.update(self.token_secret + compact_ip)
|
h.update(self.token_secret + compact_ip)
|
||||||
if self.old_token_secret and not token == h.digest(): # TODO: why should we be accepting the previous token?
|
if self.old_token_secret and not token == h.digest(): # TODO: why should we be accepting the previous token?
|
||||||
h = constants.hash_class()
|
h = constants.HASH_CLASS()
|
||||||
h.update(self.old_token_secret + compact_ip)
|
h.update(self.old_token_secret + compact_ip)
|
||||||
if not token == h.digest():
|
if not token == h.digest():
|
||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
async def store_to_peer(self, hash_value: bytes, peer: 'KademliaPeer') -> typing.Tuple[bytes, bool]:
|
async def store_to_peer(self, hash_value: bytes, peer: 'KademliaPeer', # pylint: disable=too-many-return-statements
|
||||||
|
retry: bool = True) -> typing.Tuple[bytes, bool]:
|
||||||
async def __store():
|
async def __store():
|
||||||
res = await self.get_rpc_peer(peer).store(hash_value)
|
res = await self.get_rpc_peer(peer).store(hash_value)
|
||||||
if res != b"OK":
|
if res != b"OK":
|
||||||
raise ValueError(res)
|
raise ValueError(res)
|
||||||
log.debug("Stored %s to %s", binascii.hexlify(hash_value).decode()[:8], peer)
|
log.debug("Stored %s to %s", hash_value.hex()[:8], peer)
|
||||||
return peer.node_id, True
|
return peer.node_id, True
|
||||||
|
|
||||||
try:
|
try:
|
||||||
return await __store()
|
return await __store()
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
log.debug("Timeout while storing blob_hash %s at %s", binascii.hexlify(hash_value).decode()[:8], peer)
|
log.debug("Timeout while storing blob_hash %s at %s", hash_value.hex()[:8], peer)
|
||||||
|
return peer.node_id, False
|
||||||
except ValueError as err:
|
except ValueError as err:
|
||||||
log.error("Unexpected response: %s" % err)
|
log.error("Unexpected response: %s", err)
|
||||||
except Exception as err:
|
return peer.node_id, False
|
||||||
if 'Invalid token' in str(err):
|
except RemoteException as err:
|
||||||
self.peer_manager.clear_token(peer.node_id)
|
if 'findValue() takes exactly 2 arguments (5 given)' in str(err):
|
||||||
try:
|
log.debug("peer %s:%i is running an incompatible version of lbrynet", peer.address, peer.udp_port)
|
||||||
return await __store()
|
return peer.node_id, False
|
||||||
except:
|
if 'Invalid token' not in str(err):
|
||||||
return peer.node_id, False
|
log.warning("Unexpected error while storing blob_hash: %s", err)
|
||||||
else:
|
return peer.node_id, False
|
||||||
log.exception("Unexpected error while storing blob_hash")
|
self.peer_manager.clear_token(peer.node_id)
|
||||||
return peer.node_id, False
|
if not retry:
|
||||||
|
return peer.node_id, False
|
||||||
def _write(self, data: bytes, address: typing.Tuple[str, int]):
|
return await self.store_to_peer(hash_value, peer, retry=False)
|
||||||
if self.transport:
|
|
||||||
try:
|
|
||||||
self.transport.sendto(data, address)
|
|
||||||
except OSError as err:
|
|
||||||
if err.errno == socket.EWOULDBLOCK:
|
|
||||||
# i'm scared this may swallow important errors, but i get a million of these
|
|
||||||
# on Linux and it doesn't seem to affect anything -grin
|
|
||||||
log.warning("Can't send data to dht: EWOULDBLOCK")
|
|
||||||
# elif err.errno == socket.ENETUNREACH:
|
|
||||||
# # this should probably try to retransmit when the network connection is back
|
|
||||||
# log.error("Network is unreachable")
|
|
||||||
else:
|
|
||||||
log.error("DHT socket error sending %i bytes to %s:%i - %s (code %i)",
|
|
||||||
len(data), address[0], address[1], str(err), err.errno)
|
|
||||||
raise err
|
|
||||||
else:
|
|
||||||
raise TransportNotConnected()
|
|
|
@ -4,19 +4,33 @@ import logging
|
||||||
import typing
|
import typing
|
||||||
import itertools
|
import itertools
|
||||||
|
|
||||||
from lbrynet.dht import constants
|
from prometheus_client import Gauge
|
||||||
from lbrynet.dht.protocol.distance import Distance
|
|
||||||
|
from lbry import utils
|
||||||
|
from lbry.dht import constants
|
||||||
|
from lbry.dht.error import RemoteException
|
||||||
|
from lbry.dht.protocol.distance import Distance
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbrynet.dht.peer import KademliaPeer, PeerManager
|
from lbry.dht.peer import KademliaPeer, PeerManager
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class KBucket:
|
class KBucket:
|
||||||
""" Description - later
|
|
||||||
"""
|
"""
|
||||||
|
Kademlia K-bucket implementation.
|
||||||
|
"""
|
||||||
|
peer_in_routing_table_metric = Gauge(
|
||||||
|
"peers_in_routing_table", "Number of peers on routing table", namespace="dht_node",
|
||||||
|
labelnames=("scope",)
|
||||||
|
)
|
||||||
|
peer_with_x_bit_colliding_metric = Gauge(
|
||||||
|
"peer_x_bit_colliding", "Number of peers with at least X bits colliding with this node id",
|
||||||
|
namespace="dht_node", labelnames=("amount",)
|
||||||
|
)
|
||||||
|
|
||||||
def __init__(self, peer_manager: 'PeerManager', range_min: int, range_max: int, node_id: bytes):
|
def __init__(self, peer_manager: 'PeerManager', range_min: int, range_max: int,
|
||||||
|
node_id: bytes, capacity: int = constants.K):
|
||||||
"""
|
"""
|
||||||
@param range_min: The lower boundary for the range in the n-bit ID
|
@param range_min: The lower boundary for the range in the n-bit ID
|
||||||
space covered by this k-bucket
|
space covered by this k-bucket
|
||||||
|
@ -24,12 +38,12 @@ class KBucket:
|
||||||
covered by this k-bucket
|
covered by this k-bucket
|
||||||
"""
|
"""
|
||||||
self._peer_manager = peer_manager
|
self._peer_manager = peer_manager
|
||||||
self.last_accessed = 0
|
|
||||||
self.range_min = range_min
|
self.range_min = range_min
|
||||||
self.range_max = range_max
|
self.range_max = range_max
|
||||||
self.peers: typing.List['KademliaPeer'] = []
|
self.peers: typing.List['KademliaPeer'] = []
|
||||||
self._node_id = node_id
|
self._node_id = node_id
|
||||||
self._distance_to_self = Distance(node_id)
|
self._distance_to_self = Distance(node_id)
|
||||||
|
self.capacity = capacity
|
||||||
|
|
||||||
def add_peer(self, peer: 'KademliaPeer') -> bool:
|
def add_peer(self, peer: 'KademliaPeer') -> bool:
|
||||||
""" Add contact to _contact list in the right order. This will move the
|
""" Add contact to _contact list in the right order. This will move the
|
||||||
|
@ -50,24 +64,25 @@ class KBucket:
|
||||||
self.peers.append(peer)
|
self.peers.append(peer)
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
for i in range(len(self.peers)):
|
for i, _ in enumerate(self.peers):
|
||||||
p = self.peers[i]
|
local_peer = self.peers[i]
|
||||||
if p.node_id == peer.node_id:
|
if local_peer.node_id == peer.node_id:
|
||||||
self.peers.remove(p)
|
self.peers.remove(local_peer)
|
||||||
self.peers.append(peer)
|
self.peers.append(peer)
|
||||||
return True
|
return True
|
||||||
if len(self.peers) < constants.k:
|
if len(self.peers) < self.capacity:
|
||||||
self.peers.append(peer)
|
self.peers.append(peer)
|
||||||
|
self.peer_in_routing_table_metric.labels("global").inc()
|
||||||
|
bits_colliding = utils.get_colliding_prefix_bits(peer.node_id, self._node_id)
|
||||||
|
self.peer_with_x_bit_colliding_metric.labels(amount=bits_colliding).inc()
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
# raise BucketFull("No space in bucket to insert contact")
|
|
||||||
|
|
||||||
def get_peer(self, node_id: bytes) -> 'KademliaPeer':
|
def get_peer(self, node_id: bytes) -> 'KademliaPeer':
|
||||||
for peer in self.peers:
|
for peer in self.peers:
|
||||||
if peer.node_id == node_id:
|
if peer.node_id == node_id:
|
||||||
return peer
|
return peer
|
||||||
raise IndexError(node_id)
|
|
||||||
|
|
||||||
def get_peers(self, count=-1, exclude_contact=None, sort_distance_to=None) -> typing.List['KademliaPeer']:
|
def get_peers(self, count=-1, exclude_contact=None, sort_distance_to=None) -> typing.List['KademliaPeer']:
|
||||||
""" Returns a list containing up to the first count number of contacts
|
""" Returns a list containing up to the first count number of contacts
|
||||||
|
@ -101,8 +116,8 @@ class KBucket:
|
||||||
current_len = len(peers)
|
current_len = len(peers)
|
||||||
|
|
||||||
# If count greater than k - return only k contacts
|
# If count greater than k - return only k contacts
|
||||||
if count > constants.k:
|
if count > constants.K:
|
||||||
count = constants.k
|
count = constants.K
|
||||||
|
|
||||||
if not current_len:
|
if not current_len:
|
||||||
return peers
|
return peers
|
||||||
|
@ -124,6 +139,9 @@ class KBucket:
|
||||||
|
|
||||||
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
||||||
self.peers.remove(peer)
|
self.peers.remove(peer)
|
||||||
|
self.peer_in_routing_table_metric.labels("global").dec()
|
||||||
|
bits_colliding = utils.get_colliding_prefix_bits(peer.node_id, self._node_id)
|
||||||
|
self.peer_with_x_bit_colliding_metric.labels(amount=bits_colliding).dec()
|
||||||
|
|
||||||
def key_in_range(self, key: bytes) -> bool:
|
def key_in_range(self, key: bytes) -> bool:
|
||||||
""" Tests whether the specified key (i.e. node ID) is in the range
|
""" Tests whether the specified key (i.e. node ID) is in the range
|
||||||
|
@ -161,31 +179,43 @@ class TreeRoutingTable:
|
||||||
version of the Kademlia paper, in section 2.4. It does, however, use the
|
version of the Kademlia paper, in section 2.4. It does, however, use the
|
||||||
ping RPC-based k-bucket eviction algorithm described in section 2.2 of
|
ping RPC-based k-bucket eviction algorithm described in section 2.2 of
|
||||||
that paper.
|
that paper.
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.BaseEventLoop, peer_manager: 'PeerManager', parent_node_id: bytes,
|
BOOTSTRAP MODE: if set to True, we always add all peers. This is so a
|
||||||
split_buckets_under_index: int = constants.split_buckets_under_index):
|
bootstrap node does not get a bias towards its own node id and replies are
|
||||||
|
the best it can provide (joining peer knows its neighbors immediately).
|
||||||
|
Over time, this will need to be optimized so we use the disk as holding
|
||||||
|
everything in memory won't be feasible anymore.
|
||||||
|
See: https://github.com/bittorrent/bootstrap-dht
|
||||||
|
"""
|
||||||
|
bucket_in_routing_table_metric = Gauge(
|
||||||
|
"buckets_in_routing_table", "Number of buckets on routing table", namespace="dht_node",
|
||||||
|
labelnames=("scope",)
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', parent_node_id: bytes,
|
||||||
|
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX, is_bootstrap_node: bool = False):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
self._peer_manager = peer_manager
|
self._peer_manager = peer_manager
|
||||||
self._parent_node_id = parent_node_id
|
self._parent_node_id = parent_node_id
|
||||||
self._split_buckets_under_index = split_buckets_under_index
|
self._split_buckets_under_index = split_buckets_under_index
|
||||||
self.buckets: typing.List[KBucket] = [
|
self.buckets: typing.List[KBucket] = [
|
||||||
KBucket(
|
KBucket(
|
||||||
self._peer_manager, range_min=0, range_max=2 ** constants.hash_bits, node_id=self._parent_node_id
|
self._peer_manager, range_min=0, range_max=2 ** constants.HASH_BITS, node_id=self._parent_node_id,
|
||||||
|
capacity=1 << 32 if is_bootstrap_node else constants.K
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
|
|
||||||
def get_peers(self) -> typing.List['KademliaPeer']:
|
def get_peers(self) -> typing.List['KademliaPeer']:
|
||||||
return list(itertools.chain.from_iterable(map(lambda bucket: bucket.peers, self.buckets)))
|
return list(itertools.chain.from_iterable(map(lambda bucket: bucket.peers, self.buckets)))
|
||||||
|
|
||||||
def should_split(self, bucket_index: int, to_add: bytes) -> bool:
|
def _should_split(self, bucket_index: int, to_add: bytes) -> bool:
|
||||||
# https://stackoverflow.com/questions/32129978/highly-unbalanced-kademlia-routing-table/32187456#32187456
|
# https://stackoverflow.com/questions/32129978/highly-unbalanced-kademlia-routing-table/32187456#32187456
|
||||||
if bucket_index < self._split_buckets_under_index:
|
if bucket_index < self._split_buckets_under_index:
|
||||||
return True
|
return True
|
||||||
contacts = self.get_peers()
|
contacts = self.get_peers()
|
||||||
distance = Distance(self._parent_node_id)
|
distance = Distance(self._parent_node_id)
|
||||||
contacts.sort(key=lambda c: distance(c.node_id))
|
contacts.sort(key=lambda c: distance(c.node_id))
|
||||||
kth_contact = contacts[-1] if len(contacts) < constants.k else contacts[constants.k - 1]
|
kth_contact = contacts[-1] if len(contacts) < constants.K else contacts[constants.K - 1]
|
||||||
return distance(to_add) < distance(kth_contact.node_id)
|
return distance(to_add) < distance(kth_contact.node_id)
|
||||||
|
|
||||||
def find_close_peers(self, key: bytes, count: typing.Optional[int] = None,
|
def find_close_peers(self, key: bytes, count: typing.Optional[int] = None,
|
||||||
|
@ -193,7 +223,7 @@ class TreeRoutingTable:
|
||||||
exclude = [self._parent_node_id]
|
exclude = [self._parent_node_id]
|
||||||
if sender_node_id:
|
if sender_node_id:
|
||||||
exclude.append(sender_node_id)
|
exclude.append(sender_node_id)
|
||||||
count = count or constants.k
|
count = count or constants.K
|
||||||
distance = Distance(key)
|
distance = Distance(key)
|
||||||
contacts = self.get_peers()
|
contacts = self.get_peers()
|
||||||
contacts = [c for c in contacts if c.node_id not in exclude]
|
contacts = [c for c in contacts if c.node_id not in exclude]
|
||||||
|
@ -203,39 +233,32 @@ class TreeRoutingTable:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def get_peer(self, contact_id: bytes) -> 'KademliaPeer':
|
def get_peer(self, contact_id: bytes) -> 'KademliaPeer':
|
||||||
"""
|
return self.buckets[self._kbucket_index(contact_id)].get_peer(contact_id)
|
||||||
@raise IndexError: No contact with the specified contact ID is known
|
|
||||||
by this node
|
|
||||||
"""
|
|
||||||
return self.buckets[self.kbucket_index(contact_id)].get_peer(contact_id)
|
|
||||||
|
|
||||||
def get_refresh_list(self, start_index: int = 0, force: bool = False) -> typing.List[bytes]:
|
def get_refresh_list(self, start_index: int = 0, force: bool = False) -> typing.List[bytes]:
|
||||||
bucket_index = start_index
|
|
||||||
refresh_ids = []
|
refresh_ids = []
|
||||||
now = int(self._loop.time())
|
for offset, _ in enumerate(self.buckets[start_index:]):
|
||||||
for bucket in self.buckets[start_index:]:
|
refresh_ids.append(self._midpoint_id_in_bucket_range(start_index + offset))
|
||||||
if force or now - bucket.last_accessed >= constants.refresh_interval:
|
# if we have 3 or fewer populated buckets get two random ids in the range of each to try and
|
||||||
to_search = self.midpoint_id_in_bucket_range(bucket_index)
|
# populate/split the buckets further
|
||||||
refresh_ids.append(to_search)
|
buckets_with_contacts = self.buckets_with_contacts()
|
||||||
bucket_index += 1
|
if buckets_with_contacts <= 3:
|
||||||
|
for i in range(buckets_with_contacts):
|
||||||
|
refresh_ids.append(self._random_id_in_bucket_range(i))
|
||||||
|
refresh_ids.append(self._random_id_in_bucket_range(i))
|
||||||
return refresh_ids
|
return refresh_ids
|
||||||
|
|
||||||
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
||||||
if not peer.node_id:
|
if not peer.node_id:
|
||||||
return
|
return
|
||||||
bucket_index = self.kbucket_index(peer.node_id)
|
bucket_index = self._kbucket_index(peer.node_id)
|
||||||
try:
|
try:
|
||||||
self.buckets[bucket_index].remove_peer(peer)
|
self.buckets[bucket_index].remove_peer(peer)
|
||||||
|
self._join_buckets()
|
||||||
except ValueError:
|
except ValueError:
|
||||||
return
|
return
|
||||||
|
|
||||||
def touch_kbucket(self, key: bytes) -> None:
|
def _kbucket_index(self, key: bytes) -> int:
|
||||||
self.touch_kbucket_by_index(self.kbucket_index(key))
|
|
||||||
|
|
||||||
def touch_kbucket_by_index(self, bucket_index: int):
|
|
||||||
self.buckets[bucket_index].last_accessed = int(self._loop.time())
|
|
||||||
|
|
||||||
def kbucket_index(self, key: bytes) -> int:
|
|
||||||
i = 0
|
i = 0
|
||||||
for bucket in self.buckets:
|
for bucket in self.buckets:
|
||||||
if bucket.key_in_range(key):
|
if bucket.key_in_range(key):
|
||||||
|
@ -244,19 +267,19 @@ class TreeRoutingTable:
|
||||||
i += 1
|
i += 1
|
||||||
return i
|
return i
|
||||||
|
|
||||||
def random_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
def _random_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
||||||
random_id = int(random.randrange(self.buckets[bucket_index].range_min, self.buckets[bucket_index].range_max))
|
random_id = int(random.randrange(self.buckets[bucket_index].range_min, self.buckets[bucket_index].range_max))
|
||||||
return Distance(
|
return Distance(
|
||||||
self._parent_node_id
|
self._parent_node_id
|
||||||
)(random_id.to_bytes(constants.hash_length, 'big')).to_bytes(constants.hash_length, 'big')
|
)(random_id.to_bytes(constants.HASH_LENGTH, 'big')).to_bytes(constants.HASH_LENGTH, 'big')
|
||||||
|
|
||||||
def midpoint_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
def _midpoint_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
||||||
half = int((self.buckets[bucket_index].range_max - self.buckets[bucket_index].range_min) // 2)
|
half = int((self.buckets[bucket_index].range_max - self.buckets[bucket_index].range_min) // 2)
|
||||||
return Distance(self._parent_node_id)(
|
return Distance(self._parent_node_id)(
|
||||||
int(self.buckets[bucket_index].range_min + half).to_bytes(constants.hash_length, 'big')
|
int(self.buckets[bucket_index].range_min + half).to_bytes(constants.HASH_LENGTH, 'big')
|
||||||
).to_bytes(constants.hash_length, 'big')
|
).to_bytes(constants.HASH_LENGTH, 'big')
|
||||||
|
|
||||||
def split_bucket(self, old_bucket_index: int) -> None:
|
def _split_bucket(self, old_bucket_index: int) -> None:
|
||||||
""" Splits the specified k-bucket into two new buckets which together
|
""" Splits the specified k-bucket into two new buckets which together
|
||||||
cover the same range in the key/ID space
|
cover the same range in the key/ID space
|
||||||
|
|
||||||
|
@ -279,11 +302,12 @@ class TreeRoutingTable:
|
||||||
# ...and remove them from the old bucket
|
# ...and remove them from the old bucket
|
||||||
for contact in new_bucket.peers:
|
for contact in new_bucket.peers:
|
||||||
old_bucket.remove_peer(contact)
|
old_bucket.remove_peer(contact)
|
||||||
|
self.bucket_in_routing_table_metric.labels("global").set(len(self.buckets))
|
||||||
|
|
||||||
def join_buckets(self):
|
def _join_buckets(self):
|
||||||
if len(self.buckets) == 1:
|
if len(self.buckets) == 1:
|
||||||
return
|
return
|
||||||
to_pop = [i for i, bucket in enumerate(self.buckets) if not len(bucket)]
|
to_pop = [i for i, bucket in enumerate(self.buckets) if len(bucket) == 0]
|
||||||
if not to_pop:
|
if not to_pop:
|
||||||
return
|
return
|
||||||
log.info("join buckets %i", len(to_pop))
|
log.info("join buckets %i", len(to_pop))
|
||||||
|
@ -302,18 +326,79 @@ class TreeRoutingTable:
|
||||||
elif can_go_higher:
|
elif can_go_higher:
|
||||||
self.buckets[bucket_index_to_pop + 1].range_min = bucket.range_min
|
self.buckets[bucket_index_to_pop + 1].range_min = bucket.range_min
|
||||||
self.buckets.remove(bucket)
|
self.buckets.remove(bucket)
|
||||||
return self.join_buckets()
|
self.bucket_in_routing_table_metric.labels("global").set(len(self.buckets))
|
||||||
|
return self._join_buckets()
|
||||||
def contact_in_routing_table(self, address_tuple: typing.Tuple[str, int]) -> bool:
|
|
||||||
for bucket in self.buckets:
|
|
||||||
for contact in bucket.get_peers(sort_distance_to=False):
|
|
||||||
if address_tuple[0] == contact.address and address_tuple[1] == contact.udp_port:
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
def buckets_with_contacts(self) -> int:
|
def buckets_with_contacts(self) -> int:
|
||||||
count = 0
|
count = 0
|
||||||
for bucket in self.buckets:
|
for bucket in self.buckets:
|
||||||
if len(bucket):
|
if len(bucket) > 0:
|
||||||
count += 1
|
count += 1
|
||||||
return count
|
return count
|
||||||
|
|
||||||
|
async def add_peer(self, peer: 'KademliaPeer', probe: typing.Callable[['KademliaPeer'], typing.Awaitable]):
|
||||||
|
if not peer.node_id:
|
||||||
|
log.warning("Tried adding a peer with no node id!")
|
||||||
|
return False
|
||||||
|
for my_peer in self.get_peers():
|
||||||
|
if (my_peer.address, my_peer.udp_port) == (peer.address, peer.udp_port) and my_peer.node_id != peer.node_id:
|
||||||
|
self.remove_peer(my_peer)
|
||||||
|
self._join_buckets()
|
||||||
|
bucket_index = self._kbucket_index(peer.node_id)
|
||||||
|
if self.buckets[bucket_index].add_peer(peer):
|
||||||
|
return True
|
||||||
|
|
||||||
|
# The bucket is full; see if it can be split (by checking if its range includes the host node's node_id)
|
||||||
|
if self._should_split(bucket_index, peer.node_id):
|
||||||
|
self._split_bucket(bucket_index)
|
||||||
|
# Retry the insertion attempt
|
||||||
|
result = await self.add_peer(peer, probe)
|
||||||
|
self._join_buckets()
|
||||||
|
return result
|
||||||
|
else:
|
||||||
|
# We can't split the k-bucket
|
||||||
|
#
|
||||||
|
# The 13 page kademlia paper specifies that the least recently contacted node in the bucket
|
||||||
|
# shall be pinged. If it fails to reply it is replaced with the new contact. If the ping is successful
|
||||||
|
# the new contact is ignored and not added to the bucket (sections 2.2 and 2.4).
|
||||||
|
#
|
||||||
|
# A reasonable extension to this is BEP 0005, which extends the above:
|
||||||
|
#
|
||||||
|
# Not all nodes that we learn about are equal. Some are "good" and some are not.
|
||||||
|
# Many nodes using the DHT are able to send queries and receive responses,
|
||||||
|
# but are not able to respond to queries from other nodes. It is important that
|
||||||
|
# each node's routing table must contain only known good nodes. A good node is
|
||||||
|
# a node has responded to one of our queries within the last 15 minutes. A node
|
||||||
|
# is also good if it has ever responded to one of our queries and has sent us a
|
||||||
|
# query within the last 15 minutes. After 15 minutes of inactivity, a node becomes
|
||||||
|
# questionable. Nodes become bad when they fail to respond to multiple queries
|
||||||
|
# in a row. Nodes that we know are good are given priority over nodes with unknown status.
|
||||||
|
#
|
||||||
|
# When there are bad or questionable nodes in the bucket, the least recent is selected for
|
||||||
|
# potential replacement (BEP 0005). When all nodes in the bucket are fresh, the head (least recent)
|
||||||
|
# contact is selected as described in section 2.2 of the kademlia paper. In both cases the new contact
|
||||||
|
# is ignored if the pinged node replies.
|
||||||
|
|
||||||
|
not_good_contacts = self.buckets[bucket_index].get_bad_or_unknown_peers()
|
||||||
|
not_recently_replied = []
|
||||||
|
for my_peer in not_good_contacts:
|
||||||
|
last_replied = self._peer_manager.get_last_replied(my_peer.address, my_peer.udp_port)
|
||||||
|
if not last_replied or last_replied + 60 < self._loop.time():
|
||||||
|
not_recently_replied.append(my_peer)
|
||||||
|
if not_recently_replied:
|
||||||
|
to_replace = not_recently_replied[0]
|
||||||
|
else:
|
||||||
|
to_replace = self.buckets[bucket_index].peers[0]
|
||||||
|
last_replied = self._peer_manager.get_last_replied(to_replace.address, to_replace.udp_port)
|
||||||
|
if last_replied and last_replied + 60 > self._loop.time():
|
||||||
|
return False
|
||||||
|
log.debug("pinging %s:%s", to_replace.address, to_replace.udp_port)
|
||||||
|
try:
|
||||||
|
await probe(to_replace)
|
||||||
|
return False
|
||||||
|
except (asyncio.TimeoutError, RemoteException):
|
||||||
|
log.debug("Replacing dead contact in bucket %i: %s:%i with %s:%i ", bucket_index,
|
||||||
|
to_replace.address, to_replace.udp_port, peer.address, peer.udp_port)
|
||||||
|
if to_replace in self.buckets[bucket_index]:
|
||||||
|
self.buckets[bucket_index].remove_peer(to_replace)
|
||||||
|
return await self.add_peer(peer, probe)
|
|
@ -1,5 +1,5 @@
|
||||||
import typing
|
import typing
|
||||||
from lbrynet.dht.error import DecodeError
|
from lbry.dht.error import DecodeError
|
||||||
|
|
||||||
|
|
||||||
def _bencode(data: typing.Union[int, bytes, bytearray, str, list, tuple, dict]) -> bytes:
|
def _bencode(data: typing.Union[int, bytes, bytearray, str, list, tuple, dict]) -> bytes:
|
||||||
|
@ -52,8 +52,7 @@ def _bdecode(data: bytes, start_index: int = 0) -> typing.Tuple[typing.Union[int
|
||||||
raise DecodeError(err)
|
raise DecodeError(err)
|
||||||
start_index = split_pos + 1
|
start_index = split_pos + 1
|
||||||
end_pos = start_index + length
|
end_pos = start_index + length
|
||||||
b = data[start_index:end_pos]
|
return data[start_index:end_pos], end_pos
|
||||||
return b, end_pos
|
|
||||||
|
|
||||||
|
|
||||||
def bencode(data: typing.Dict) -> bytes:
|
def bencode(data: typing.Dict) -> bytes:
|
||||||
|
@ -63,7 +62,7 @@ def bencode(data: typing.Dict) -> bytes:
|
||||||
|
|
||||||
|
|
||||||
def bdecode(data: bytes, allow_non_dict_return: typing.Optional[bool] = False) -> typing.Dict:
|
def bdecode(data: bytes, allow_non_dict_return: typing.Optional[bool] = False) -> typing.Dict:
|
||||||
assert type(data) == bytes, DecodeError(f"invalid data type: {str(type(data))}")
|
assert isinstance(data, bytes), DecodeError(f"invalid data type: {str(type(data))}")
|
||||||
|
|
||||||
if len(data) == 0:
|
if len(data) == 0:
|
||||||
raise DecodeError('Cannot decode empty string')
|
raise DecodeError('Cannot decode empty string')
|
|
@ -1,12 +1,19 @@
|
||||||
import typing
|
import typing
|
||||||
from functools import reduce
|
from functools import reduce
|
||||||
from lbrynet.dht import constants
|
from lbry.dht import constants
|
||||||
from lbrynet.dht.serialization.bencoding import bencode, bdecode
|
from lbry.dht.serialization.bencoding import bencode, bdecode
|
||||||
|
|
||||||
REQUEST_TYPE = 0
|
REQUEST_TYPE = 0
|
||||||
RESPONSE_TYPE = 1
|
RESPONSE_TYPE = 1
|
||||||
ERROR_TYPE = 2
|
ERROR_TYPE = 2
|
||||||
|
|
||||||
|
OPTIONAL_ARG_OFFSET = 100
|
||||||
|
|
||||||
|
# bencode representation of argument keys
|
||||||
|
PAGE_KEY = b'p'
|
||||||
|
|
||||||
|
OPTIONAL_FIELDS = ()
|
||||||
|
|
||||||
|
|
||||||
class KademliaDatagramBase:
|
class KademliaDatagramBase:
|
||||||
"""
|
"""
|
||||||
|
@ -15,7 +22,7 @@ class KademliaDatagramBase:
|
||||||
these correspond to the packet_type, rpc_id, and node_id args
|
these correspond to the packet_type, rpc_id, and node_id args
|
||||||
"""
|
"""
|
||||||
|
|
||||||
fields = [
|
required_fields = [
|
||||||
'packet_type',
|
'packet_type',
|
||||||
'rpc_id',
|
'rpc_id',
|
||||||
'node_id'
|
'node_id'
|
||||||
|
@ -27,21 +34,26 @@ class KademliaDatagramBase:
|
||||||
self.packet_type = packet_type
|
self.packet_type = packet_type
|
||||||
if self.expected_packet_type != packet_type:
|
if self.expected_packet_type != packet_type:
|
||||||
raise ValueError(f"invalid packet type: {packet_type}, expected {self.expected_packet_type}")
|
raise ValueError(f"invalid packet type: {packet_type}, expected {self.expected_packet_type}")
|
||||||
if len(rpc_id) != constants.rpc_id_length:
|
if len(rpc_id) != constants.RPC_ID_LENGTH:
|
||||||
raise ValueError(f"invalid rpc node_id: {len(rpc_id)} bytes (expected 20)")
|
raise ValueError(f"invalid rpc node_id: {len(rpc_id)} bytes (expected 20)")
|
||||||
if not len(node_id) == constants.hash_length:
|
if not len(node_id) == constants.HASH_LENGTH:
|
||||||
raise ValueError(f"invalid node node_id: {len(node_id)} bytes (expected 48)")
|
raise ValueError(f"invalid node node_id: {len(node_id)} bytes (expected 48)")
|
||||||
self.rpc_id = rpc_id
|
self.rpc_id = rpc_id
|
||||||
self.node_id = node_id
|
self.node_id = node_id
|
||||||
|
|
||||||
def bencode(self) -> bytes:
|
def bencode(self) -> bytes:
|
||||||
return bencode({
|
datagram = {
|
||||||
i: getattr(self, k) for i, k in enumerate(self.fields)
|
i: getattr(self, k) for i, k in enumerate(self.required_fields)
|
||||||
})
|
}
|
||||||
|
for i, k in enumerate(OPTIONAL_FIELDS):
|
||||||
|
value = getattr(self, k, None)
|
||||||
|
if value is not None:
|
||||||
|
datagram[i + OPTIONAL_ARG_OFFSET] = value
|
||||||
|
return bencode(datagram)
|
||||||
|
|
||||||
|
|
||||||
class RequestDatagram(KademliaDatagramBase):
|
class RequestDatagram(KademliaDatagramBase):
|
||||||
fields = [
|
required_fields = [
|
||||||
'packet_type',
|
'packet_type',
|
||||||
'rpc_id',
|
'rpc_id',
|
||||||
'node_id',
|
'node_id',
|
||||||
|
@ -65,18 +77,18 @@ class RequestDatagram(KademliaDatagramBase):
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def make_ping(cls, from_node_id: bytes, rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
def make_ping(cls, from_node_id: bytes, rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
||||||
rpc_id = rpc_id or constants.generate_id()[:constants.rpc_id_length]
|
rpc_id = rpc_id or constants.generate_id()[:constants.RPC_ID_LENGTH]
|
||||||
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'ping')
|
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'ping')
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def make_store(cls, from_node_id: bytes, blob_hash: bytes, token: bytes, port: int,
|
def make_store(cls, from_node_id: bytes, blob_hash: bytes, token: bytes, port: int,
|
||||||
rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
||||||
rpc_id = rpc_id or constants.generate_id()[:constants.rpc_id_length]
|
rpc_id = rpc_id or constants.generate_id()[:constants.RPC_ID_LENGTH]
|
||||||
if len(blob_hash) != constants.hash_bits // 8:
|
if len(blob_hash) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid blob hash length: {len(blob_hash)}")
|
raise ValueError(f"invalid blob hash length: {len(blob_hash)}")
|
||||||
if not 0 < port < 65536:
|
if not 0 < port < 65536:
|
||||||
raise ValueError(f"invalid port: {port}")
|
raise ValueError(f"invalid port: {port}")
|
||||||
if len(token) != constants.hash_bits // 8:
|
if len(token) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid token length: {len(token)}")
|
raise ValueError(f"invalid token length: {len(token)}")
|
||||||
store_args = [blob_hash, token, port, from_node_id, 0]
|
store_args = [blob_hash, token, port, from_node_id, 0]
|
||||||
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'store', store_args)
|
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'store', store_args)
|
||||||
|
@ -84,22 +96,24 @@ class RequestDatagram(KademliaDatagramBase):
|
||||||
@classmethod
|
@classmethod
|
||||||
def make_find_node(cls, from_node_id: bytes, key: bytes,
|
def make_find_node(cls, from_node_id: bytes, key: bytes,
|
||||||
rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
||||||
rpc_id = rpc_id or constants.generate_id()[:constants.rpc_id_length]
|
rpc_id = rpc_id or constants.generate_id()[:constants.RPC_ID_LENGTH]
|
||||||
if len(key) != constants.hash_bits // 8:
|
if len(key) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid key length: {len(key)}")
|
raise ValueError(f"invalid key length: {len(key)}")
|
||||||
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'findNode', [key])
|
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'findNode', [key])
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def make_find_value(cls, from_node_id: bytes, key: bytes,
|
def make_find_value(cls, from_node_id: bytes, key: bytes,
|
||||||
rpc_id: typing.Optional[bytes] = None) -> 'RequestDatagram':
|
rpc_id: typing.Optional[bytes] = None, page: int = 0) -> 'RequestDatagram':
|
||||||
rpc_id = rpc_id or constants.generate_id()[:constants.rpc_id_length]
|
rpc_id = rpc_id or constants.generate_id()[:constants.RPC_ID_LENGTH]
|
||||||
if len(key) != constants.hash_bits // 8:
|
if len(key) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid key length: {len(key)}")
|
raise ValueError(f"invalid key length: {len(key)}")
|
||||||
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'findValue', [key])
|
if page < 0:
|
||||||
|
raise ValueError(f"cannot request a negative page ({page})")
|
||||||
|
return cls(REQUEST_TYPE, rpc_id, from_node_id, b'findValue', [key, {PAGE_KEY: page}])
|
||||||
|
|
||||||
|
|
||||||
class ResponseDatagram(KademliaDatagramBase):
|
class ResponseDatagram(KademliaDatagramBase):
|
||||||
fields = [
|
required_fields = [
|
||||||
'packet_type',
|
'packet_type',
|
||||||
'rpc_id',
|
'rpc_id',
|
||||||
'node_id',
|
'node_id',
|
||||||
|
@ -114,7 +128,7 @@ class ResponseDatagram(KademliaDatagramBase):
|
||||||
|
|
||||||
|
|
||||||
class ErrorDatagram(KademliaDatagramBase):
|
class ErrorDatagram(KademliaDatagramBase):
|
||||||
fields = [
|
required_fields = [
|
||||||
'packet_type',
|
'packet_type',
|
||||||
'rpc_id',
|
'rpc_id',
|
||||||
'node_id',
|
'node_id',
|
||||||
|
@ -130,7 +144,7 @@ class ErrorDatagram(KademliaDatagramBase):
|
||||||
self.response = response.decode()
|
self.response = response.decode()
|
||||||
|
|
||||||
|
|
||||||
def decode_datagram(datagram: bytes) -> typing.Union[RequestDatagram, ResponseDatagram, ErrorDatagram]:
|
def _decode_datagram(datagram: bytes):
|
||||||
msg_types = {
|
msg_types = {
|
||||||
REQUEST_TYPE: RequestDatagram,
|
REQUEST_TYPE: RequestDatagram,
|
||||||
RESPONSE_TYPE: ResponseDatagram,
|
RESPONSE_TYPE: ResponseDatagram,
|
||||||
|
@ -138,23 +152,36 @@ def decode_datagram(datagram: bytes) -> typing.Union[RequestDatagram, ResponseDa
|
||||||
}
|
}
|
||||||
|
|
||||||
primitive: typing.Dict = bdecode(datagram)
|
primitive: typing.Dict = bdecode(datagram)
|
||||||
if primitive[0] in [REQUEST_TYPE, ERROR_TYPE, RESPONSE_TYPE]: # pylint: disable=unsubscriptable-object
|
|
||||||
datagram_type = primitive[0] # pylint: disable=unsubscriptable-object
|
converted = {
|
||||||
|
str(k).encode() if not isinstance(k, bytes) else k: v for k, v in primitive.items()
|
||||||
|
}
|
||||||
|
|
||||||
|
if converted[b'0'] in [REQUEST_TYPE, ERROR_TYPE, RESPONSE_TYPE]: # pylint: disable=unsubscriptable-object
|
||||||
|
datagram_type = converted[b'0'] # pylint: disable=unsubscriptable-object
|
||||||
else:
|
else:
|
||||||
raise ValueError("invalid datagram type")
|
raise ValueError("invalid datagram type")
|
||||||
datagram_class = msg_types[datagram_type]
|
datagram_class = msg_types[datagram_type]
|
||||||
return datagram_class(**{
|
decoded = {
|
||||||
k: primitive[i] # pylint: disable=unsubscriptable-object
|
k: converted[str(i).encode()] # pylint: disable=unsubscriptable-object
|
||||||
for i, k in enumerate(datagram_class.fields)
|
for i, k in enumerate(datagram_class.required_fields)
|
||||||
if i in primitive # pylint: disable=unsupported-membership-test
|
if str(i).encode() in converted # pylint: disable=unsupported-membership-test
|
||||||
}
|
}
|
||||||
)
|
for i, _ in enumerate(OPTIONAL_FIELDS):
|
||||||
|
if str(i + OPTIONAL_ARG_OFFSET).encode() in converted:
|
||||||
|
decoded[i + OPTIONAL_ARG_OFFSET] = converted[str(i + OPTIONAL_ARG_OFFSET).encode()]
|
||||||
|
return decoded, datagram_class
|
||||||
|
|
||||||
|
|
||||||
|
def decode_datagram(datagram: bytes) -> typing.Union[RequestDatagram, ResponseDatagram, ErrorDatagram]:
|
||||||
|
decoded, datagram_class = _decode_datagram(datagram)
|
||||||
|
return datagram_class(**decoded)
|
||||||
|
|
||||||
|
|
||||||
def make_compact_ip(address: str) -> bytearray:
|
def make_compact_ip(address: str) -> bytearray:
|
||||||
compact_ip = reduce(lambda buff, x: buff + bytearray([int(x)]), address.split('.'), bytearray())
|
compact_ip = reduce(lambda buff, x: buff + bytearray([int(x)]), address.split('.'), bytearray())
|
||||||
if len(compact_ip) != 4:
|
if len(compact_ip) != 4:
|
||||||
raise ValueError(f"invalid IPv4 length")
|
raise ValueError("invalid IPv4 length")
|
||||||
return compact_ip
|
return compact_ip
|
||||||
|
|
||||||
|
|
||||||
|
@ -162,8 +189,8 @@ def make_compact_address(node_id: bytes, address: str, port: int) -> bytearray:
|
||||||
compact_ip = make_compact_ip(address)
|
compact_ip = make_compact_ip(address)
|
||||||
if not 0 < port < 65536:
|
if not 0 < port < 65536:
|
||||||
raise ValueError(f'Invalid port: {port}')
|
raise ValueError(f'Invalid port: {port}')
|
||||||
if len(node_id) != constants.hash_bits // 8:
|
if len(node_id) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid node node_id length")
|
raise ValueError("invalid node node_id length")
|
||||||
return compact_ip + port.to_bytes(2, 'big') + node_id
|
return compact_ip + port.to_bytes(2, 'big') + node_id
|
||||||
|
|
||||||
|
|
||||||
|
@ -173,6 +200,6 @@ def decode_compact_address(compact_address: bytes) -> typing.Tuple[bytes, str, i
|
||||||
node_id = compact_address[6:]
|
node_id = compact_address[6:]
|
||||||
if not 0 < port < 65536:
|
if not 0 < port < 65536:
|
||||||
raise ValueError(f'Invalid port: {port}')
|
raise ValueError(f'Invalid port: {port}')
|
||||||
if len(node_id) != constants.hash_bits // 8:
|
if len(node_id) != constants.HASH_BITS // 8:
|
||||||
raise ValueError(f"invalid node node_id length")
|
raise ValueError("invalid node node_id length")
|
||||||
return node_id, address, port
|
return node_id, address, port
|
5
lbry/error/Makefile
Normal file
5
lbry/error/Makefile
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
generate:
|
||||||
|
python generate.py generate > __init__.py
|
||||||
|
|
||||||
|
analyze:
|
||||||
|
python generate.py analyze
|
95
lbry/error/README.md
Normal file
95
lbry/error/README.md
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
# Exceptions
|
||||||
|
|
||||||
|
Exceptions in LBRY are defined and generated from the Markdown table at the end of this README.
|
||||||
|
|
||||||
|
## Guidelines
|
||||||
|
|
||||||
|
When possible, use [built-in Python exceptions](https://docs.python.org/3/library/exceptions.html) or `aiohttp` [general client](https://docs.aiohttp.org/en/latest/client_reference.html#client-exceptions) / [HTTP](https://docs.aiohttp.org/en/latest/web_exceptions.html) exceptions, unless:
|
||||||
|
1. You want to provide a better error message (extend the closest built-in/`aiohttp` exception in this case).
|
||||||
|
2. You need to represent a new situation.
|
||||||
|
|
||||||
|
When defining your own exceptions, consider:
|
||||||
|
1. Extending a built-in Python or `aiohttp` exception.
|
||||||
|
2. Using contextual variables in the error message.
|
||||||
|
|
||||||
|
## Table Column Definitions
|
||||||
|
|
||||||
|
Column | Meaning
|
||||||
|
---|---
|
||||||
|
Code | Codes are used only to define the hierarchy of exceptions and do not end up in the generated output, it is okay to re-number things as necessary at anytime to achieve the desired hierarchy.
|
||||||
|
Name | Becomes the class name of the exception with "Error" appended to the end. Changing names of existing exceptions makes the API backwards incompatible. When extending other exceptions you must specify the full class name, manually adding "Error" as necessary (if extending another SDK exception).
|
||||||
|
Message | User friendly error message explaining the exceptional event. Supports Python formatted strings: any variables used in the string will be generated as arguments in the `__init__` method. Use `--` to provide a doc string after the error message to be added to the class definition.
|
||||||
|
|
||||||
|
## Exceptions Table
|
||||||
|
|
||||||
|
Code | Name | Message
|
||||||
|
---:|---|---
|
||||||
|
**1xx** | UserInput | User input errors.
|
||||||
|
**10x** | Command | Errors preparing to execute commands.
|
||||||
|
101 | CommandDoesNotExist | Command '{command}' does not exist.
|
||||||
|
102 | CommandDeprecated | Command '{command}' is deprecated.
|
||||||
|
103 | CommandInvalidArgument | Invalid argument '{argument}' to command '{command}'.
|
||||||
|
104 | CommandTemporarilyUnavailable | Command '{command}' is temporarily unavailable. -- Such as waiting for required components to start.
|
||||||
|
105 | CommandPermanentlyUnavailable | Command '{command}' is permanently unavailable. -- such as when required component was intentionally configured not to start.
|
||||||
|
**11x** | InputValue(ValueError) | Invalid argument value provided to command.
|
||||||
|
111 | GenericInputValue | The value '{value}' for argument '{argument}' is not valid.
|
||||||
|
112 | InputValueIsNone | None or null is not valid value for argument '{argument}'.
|
||||||
|
113 | ConflictingInputValue | Only '{first_argument}' or '{second_argument}' is allowed, not both.
|
||||||
|
114 | InputStringIsBlank | {argument} cannot be blank.
|
||||||
|
115 | EmptyPublishedFile | Cannot publish empty file: {file_path}
|
||||||
|
116 | MissingPublishedFile | File does not exist: {file_path}
|
||||||
|
117 | InvalidStreamURL | Invalid LBRY stream URL: '{url}' -- When an URL cannot be downloaded, such as '@Channel/' or a collection
|
||||||
|
**2xx** | Configuration | Configuration errors.
|
||||||
|
201 | ConfigWrite | Cannot write configuration file '{path}'. -- When writing the default config fails on startup, such as due to permission issues.
|
||||||
|
202 | ConfigRead | Cannot find provided configuration file '{path}'. -- Can't open the config file user provided via command line args.
|
||||||
|
203 | ConfigParse | Failed to parse the configuration file '{path}'. -- Includes the syntax error / line number to help user fix it.
|
||||||
|
204 | ConfigMissing | Configuration file '{path}' is missing setting that has no default / fallback.
|
||||||
|
205 | ConfigInvalid | Configuration file '{path}' has setting with invalid value.
|
||||||
|
**3xx** | Network | **Networking**
|
||||||
|
301 | NoInternet | No internet connection.
|
||||||
|
302 | NoUPnPSupport | Router does not support UPnP.
|
||||||
|
**4xx** | Wallet | **Wallet Errors**
|
||||||
|
401 | TransactionRejected | Transaction rejected, unknown reason.
|
||||||
|
402 | TransactionFeeTooLow | Fee too low.
|
||||||
|
403 | TransactionInvalidSignature | Invalid signature.
|
||||||
|
404 | InsufficientFunds | Not enough funds to cover this transaction. -- determined by wallet prior to attempting to broadcast a tx; this is different for example from a TX being created and sent but then rejected by lbrycrd for unspendable utxos.
|
||||||
|
405 | ChannelKeyNotFound | Channel signing key not found.
|
||||||
|
406 | ChannelKeyInvalid | Channel signing key is out of date. -- For example, channel was updated but you don't have the updated key.
|
||||||
|
407 | DataDownload | Failed to download blob. *generic*
|
||||||
|
408 | PrivateKeyNotFound | Couldn't find private key for {key} '{value}'.
|
||||||
|
410 | Resolve | Failed to resolve '{url}'.
|
||||||
|
411 | ResolveTimeout | Failed to resolve '{url}' within the timeout.
|
||||||
|
411 | ResolveCensored | Resolve of '{url}' was censored by channel with claim id '{censor_id}'.
|
||||||
|
420 | KeyFeeAboveMaxAllowed | {message}
|
||||||
|
421 | InvalidPassword | Password is invalid.
|
||||||
|
422 | IncompatibleWalletServer | '{server}:{port}' has an incompatibly old version.
|
||||||
|
423 | TooManyClaimSearchParameters | {key} cant have more than {limit} items.
|
||||||
|
424 | AlreadyPurchased | You already have a purchase for claim_id '{claim_id_hex}'. Use --allow-duplicate-purchase flag to override.
|
||||||
|
431 | ServerPaymentInvalidAddress | Invalid address from wallet server: '{address}' - skipping payment round.
|
||||||
|
432 | ServerPaymentWalletLocked | Cannot spend funds with locked wallet, skipping payment round.
|
||||||
|
433 | ServerPaymentFeeAboveMaxAllowed | Daily server fee of {daily_fee} exceeds maximum configured of {max_fee} LBC.
|
||||||
|
434 | WalletNotLoaded | Wallet {wallet_id} is not loaded.
|
||||||
|
435 | WalletAlreadyLoaded | Wallet {wallet_path} is already loaded.
|
||||||
|
436 | WalletNotFound | Wallet not found at {wallet_path}.
|
||||||
|
437 | WalletAlreadyExists | Wallet {wallet_path} already exists, use `wallet_add` to load it.
|
||||||
|
**5xx** | Blob | **Blobs**
|
||||||
|
500 | BlobNotFound | Blob not found.
|
||||||
|
501 | BlobPermissionDenied | Permission denied to read blob.
|
||||||
|
502 | BlobTooBig | Blob is too big.
|
||||||
|
503 | BlobEmpty | Blob is empty.
|
||||||
|
510 | BlobFailedDecryption | Failed to decrypt blob.
|
||||||
|
511 | CorruptBlob | Blobs is corrupted.
|
||||||
|
520 | BlobFailedEncryption | Failed to encrypt blob.
|
||||||
|
531 | DownloadCancelled | Download was canceled.
|
||||||
|
532 | DownloadSDTimeout | Failed to download sd blob {download} within timeout.
|
||||||
|
533 | DownloadDataTimeout | Failed to download data blobs for sd hash {download} within timeout.
|
||||||
|
534 | InvalidStreamDescriptor | {message}
|
||||||
|
535 | InvalidData | {message}
|
||||||
|
536 | InvalidBlobHash | {message}
|
||||||
|
**6xx** | Component | **Components**
|
||||||
|
601 | ComponentStartConditionNotMet | Unresolved dependencies for: {components}
|
||||||
|
602 | ComponentsNotStarted | {message}
|
||||||
|
**7xx** | CurrencyExchange | **Currency Exchange**
|
||||||
|
701 | InvalidExchangeRateResponse | Failed to get exchange rate from {source}: {reason}
|
||||||
|
702 | CurrencyConversion | {message}
|
||||||
|
703 | InvalidCurrency | Invalid currency: {currency} is not a supported currency.
|
494
lbry/error/__init__.py
Normal file
494
lbry/error/__init__.py
Normal file
|
@ -0,0 +1,494 @@
|
||||||
|
from .base import BaseError, claim_id
|
||||||
|
|
||||||
|
|
||||||
|
class UserInputError(BaseError):
|
||||||
|
"""
|
||||||
|
User input errors.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class CommandError(UserInputError):
|
||||||
|
"""
|
||||||
|
Errors preparing to execute commands.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class CommandDoesNotExistError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, command):
|
||||||
|
self.command = command
|
||||||
|
super().__init__(f"Command '{command}' does not exist.")
|
||||||
|
|
||||||
|
|
||||||
|
class CommandDeprecatedError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, command):
|
||||||
|
self.command = command
|
||||||
|
super().__init__(f"Command '{command}' is deprecated.")
|
||||||
|
|
||||||
|
|
||||||
|
class CommandInvalidArgumentError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, argument, command):
|
||||||
|
self.argument = argument
|
||||||
|
self.command = command
|
||||||
|
super().__init__(f"Invalid argument '{argument}' to command '{command}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class CommandTemporarilyUnavailableError(CommandError):
|
||||||
|
"""
|
||||||
|
Such as waiting for required components to start.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, command):
|
||||||
|
self.command = command
|
||||||
|
super().__init__(f"Command '{command}' is temporarily unavailable.")
|
||||||
|
|
||||||
|
|
||||||
|
class CommandPermanentlyUnavailableError(CommandError):
|
||||||
|
"""
|
||||||
|
such as when required component was intentionally configured not to start.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, command):
|
||||||
|
self.command = command
|
||||||
|
super().__init__(f"Command '{command}' is permanently unavailable.")
|
||||||
|
|
||||||
|
|
||||||
|
class InputValueError(UserInputError, ValueError):
|
||||||
|
"""
|
||||||
|
Invalid argument value provided to command.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class GenericInputValueError(InputValueError):
|
||||||
|
|
||||||
|
def __init__(self, value, argument):
|
||||||
|
self.value = value
|
||||||
|
self.argument = argument
|
||||||
|
super().__init__(f"The value '{value}' for argument '{argument}' is not valid.")
|
||||||
|
|
||||||
|
|
||||||
|
class InputValueIsNoneError(InputValueError):
|
||||||
|
|
||||||
|
def __init__(self, argument):
|
||||||
|
self.argument = argument
|
||||||
|
super().__init__(f"None or null is not valid value for argument '{argument}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class ConflictingInputValueError(InputValueError):
|
||||||
|
|
||||||
|
def __init__(self, first_argument, second_argument):
|
||||||
|
self.first_argument = first_argument
|
||||||
|
self.second_argument = second_argument
|
||||||
|
super().__init__(f"Only '{first_argument}' or '{second_argument}' is allowed, not both.")
|
||||||
|
|
||||||
|
|
||||||
|
class InputStringIsBlankError(InputValueError):
|
||||||
|
|
||||||
|
def __init__(self, argument):
|
||||||
|
self.argument = argument
|
||||||
|
super().__init__(f"{argument} cannot be blank.")
|
||||||
|
|
||||||
|
|
||||||
|
class EmptyPublishedFileError(InputValueError):
|
||||||
|
|
||||||
|
def __init__(self, file_path):
|
||||||
|
self.file_path = file_path
|
||||||
|
super().__init__(f"Cannot publish empty file: {file_path}")
|
||||||
|
|
||||||
|
|
||||||
|
class MissingPublishedFileError(InputValueError):
|
||||||
|
|
||||||
|
def __init__(self, file_path):
|
||||||
|
self.file_path = file_path
|
||||||
|
super().__init__(f"File does not exist: {file_path}")
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidStreamURLError(InputValueError):
|
||||||
|
"""
|
||||||
|
When an URL cannot be downloaded, such as '@Channel/' or a collection
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, url):
|
||||||
|
self.url = url
|
||||||
|
super().__init__(f"Invalid LBRY stream URL: '{url}'")
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigurationError(BaseError):
|
||||||
|
"""
|
||||||
|
Configuration errors.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigWriteError(ConfigurationError):
|
||||||
|
"""
|
||||||
|
When writing the default config fails on startup, such as due to permission issues.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, path):
|
||||||
|
self.path = path
|
||||||
|
super().__init__(f"Cannot write configuration file '{path}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigReadError(ConfigurationError):
|
||||||
|
"""
|
||||||
|
Can't open the config file user provided via command line args.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, path):
|
||||||
|
self.path = path
|
||||||
|
super().__init__(f"Cannot find provided configuration file '{path}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigParseError(ConfigurationError):
|
||||||
|
"""
|
||||||
|
Includes the syntax error / line number to help user fix it.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, path):
|
||||||
|
self.path = path
|
||||||
|
super().__init__(f"Failed to parse the configuration file '{path}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigMissingError(ConfigurationError):
|
||||||
|
|
||||||
|
def __init__(self, path):
|
||||||
|
self.path = path
|
||||||
|
super().__init__(f"Configuration file '{path}' is missing setting that has no default / fallback.")
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigInvalidError(ConfigurationError):
|
||||||
|
|
||||||
|
def __init__(self, path):
|
||||||
|
self.path = path
|
||||||
|
super().__init__(f"Configuration file '{path}' has setting with invalid value.")
|
||||||
|
|
||||||
|
|
||||||
|
class NetworkError(BaseError):
|
||||||
|
"""
|
||||||
|
**Networking**
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class NoInternetError(NetworkError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("No internet connection.")
|
||||||
|
|
||||||
|
|
||||||
|
class NoUPnPSupportError(NetworkError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Router does not support UPnP.")
|
||||||
|
|
||||||
|
|
||||||
|
class WalletError(BaseError):
|
||||||
|
"""
|
||||||
|
**Wallet Errors**
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class TransactionRejectedError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Transaction rejected, unknown reason.")
|
||||||
|
|
||||||
|
|
||||||
|
class TransactionFeeTooLowError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Fee too low.")
|
||||||
|
|
||||||
|
|
||||||
|
class TransactionInvalidSignatureError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Invalid signature.")
|
||||||
|
|
||||||
|
|
||||||
|
class InsufficientFundsError(WalletError):
|
||||||
|
"""
|
||||||
|
determined by wallet prior to attempting to broadcast a tx; this is different for example from a TX
|
||||||
|
being created and sent but then rejected by lbrycrd for unspendable utxos.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Not enough funds to cover this transaction.")
|
||||||
|
|
||||||
|
|
||||||
|
class ChannelKeyNotFoundError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Channel signing key not found.")
|
||||||
|
|
||||||
|
|
||||||
|
class ChannelKeyInvalidError(WalletError):
|
||||||
|
"""
|
||||||
|
For example, channel was updated but you don't have the updated key.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Channel signing key is out of date.")
|
||||||
|
|
||||||
|
|
||||||
|
class DataDownloadError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Failed to download blob. *generic*")
|
||||||
|
|
||||||
|
|
||||||
|
class PrivateKeyNotFoundError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, key, value):
|
||||||
|
self.key = key
|
||||||
|
self.value = value
|
||||||
|
super().__init__(f"Couldn't find private key for {key} '{value}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class ResolveError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, url):
|
||||||
|
self.url = url
|
||||||
|
super().__init__(f"Failed to resolve '{url}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class ResolveTimeoutError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, url):
|
||||||
|
self.url = url
|
||||||
|
super().__init__(f"Failed to resolve '{url}' within the timeout.")
|
||||||
|
|
||||||
|
|
||||||
|
class ResolveCensoredError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, url, censor_id, censor_row):
|
||||||
|
self.url = url
|
||||||
|
self.censor_id = censor_id
|
||||||
|
self.censor_row = censor_row
|
||||||
|
super().__init__(f"Resolve of '{url}' was censored by channel with claim id '{censor_id}'.")
|
||||||
|
|
||||||
|
|
||||||
|
class KeyFeeAboveMaxAllowedError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, message):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(f"{message}")
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidPasswordError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Password is invalid.")
|
||||||
|
|
||||||
|
|
||||||
|
class IncompatibleWalletServerError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, server, port):
|
||||||
|
self.server = server
|
||||||
|
self.port = port
|
||||||
|
super().__init__(f"'{server}:{port}' has an incompatibly old version.")
|
||||||
|
|
||||||
|
|
||||||
|
class TooManyClaimSearchParametersError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, key, limit):
|
||||||
|
self.key = key
|
||||||
|
self.limit = limit
|
||||||
|
super().__init__(f"{key} cant have more than {limit} items.")
|
||||||
|
|
||||||
|
|
||||||
|
class AlreadyPurchasedError(WalletError):
|
||||||
|
"""
|
||||||
|
allow-duplicate-purchase flag to override.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, claim_id_hex):
|
||||||
|
self.claim_id_hex = claim_id_hex
|
||||||
|
super().__init__(f"You already have a purchase for claim_id '{claim_id_hex}'. Use")
|
||||||
|
|
||||||
|
|
||||||
|
class ServerPaymentInvalidAddressError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, address):
|
||||||
|
self.address = address
|
||||||
|
super().__init__(f"Invalid address from wallet server: '{address}' - skipping payment round.")
|
||||||
|
|
||||||
|
|
||||||
|
class ServerPaymentWalletLockedError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Cannot spend funds with locked wallet, skipping payment round.")
|
||||||
|
|
||||||
|
|
||||||
|
class ServerPaymentFeeAboveMaxAllowedError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, daily_fee, max_fee):
|
||||||
|
self.daily_fee = daily_fee
|
||||||
|
self.max_fee = max_fee
|
||||||
|
super().__init__(f"Daily server fee of {daily_fee} exceeds maximum configured of {max_fee} LBC.")
|
||||||
|
|
||||||
|
|
||||||
|
class WalletNotLoadedError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, wallet_id):
|
||||||
|
self.wallet_id = wallet_id
|
||||||
|
super().__init__(f"Wallet {wallet_id} is not loaded.")
|
||||||
|
|
||||||
|
|
||||||
|
class WalletAlreadyLoadedError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, wallet_path):
|
||||||
|
self.wallet_path = wallet_path
|
||||||
|
super().__init__(f"Wallet {wallet_path} is already loaded.")
|
||||||
|
|
||||||
|
|
||||||
|
class WalletNotFoundError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, wallet_path):
|
||||||
|
self.wallet_path = wallet_path
|
||||||
|
super().__init__(f"Wallet not found at {wallet_path}.")
|
||||||
|
|
||||||
|
|
||||||
|
class WalletAlreadyExistsError(WalletError):
|
||||||
|
|
||||||
|
def __init__(self, wallet_path):
|
||||||
|
self.wallet_path = wallet_path
|
||||||
|
super().__init__(f"Wallet {wallet_path} already exists, use `wallet_add` to load it.")
|
||||||
|
|
||||||
|
|
||||||
|
class BlobError(BaseError):
|
||||||
|
"""
|
||||||
|
**Blobs**
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class BlobNotFoundError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Blob not found.")
|
||||||
|
|
||||||
|
|
||||||
|
class BlobPermissionDeniedError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Permission denied to read blob.")
|
||||||
|
|
||||||
|
|
||||||
|
class BlobTooBigError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Blob is too big.")
|
||||||
|
|
||||||
|
|
||||||
|
class BlobEmptyError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Blob is empty.")
|
||||||
|
|
||||||
|
|
||||||
|
class BlobFailedDecryptionError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Failed to decrypt blob.")
|
||||||
|
|
||||||
|
|
||||||
|
class CorruptBlobError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Blobs is corrupted.")
|
||||||
|
|
||||||
|
|
||||||
|
class BlobFailedEncryptionError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Failed to encrypt blob.")
|
||||||
|
|
||||||
|
|
||||||
|
class DownloadCancelledError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__("Download was canceled.")
|
||||||
|
|
||||||
|
|
||||||
|
class DownloadSDTimeoutError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self, download):
|
||||||
|
self.download = download
|
||||||
|
super().__init__(f"Failed to download sd blob {download} within timeout.")
|
||||||
|
|
||||||
|
|
||||||
|
class DownloadDataTimeoutError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self, download):
|
||||||
|
self.download = download
|
||||||
|
super().__init__(f"Failed to download data blobs for sd hash {download} within timeout.")
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidStreamDescriptorError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self, message):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(f"{message}")
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidDataError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self, message):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(f"{message}")
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidBlobHashError(BlobError):
|
||||||
|
|
||||||
|
def __init__(self, message):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(f"{message}")
|
||||||
|
|
||||||
|
|
||||||
|
class ComponentError(BaseError):
|
||||||
|
"""
|
||||||
|
**Components**
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class ComponentStartConditionNotMetError(ComponentError):
|
||||||
|
|
||||||
|
def __init__(self, components):
|
||||||
|
self.components = components
|
||||||
|
super().__init__(f"Unresolved dependencies for: {components}")
|
||||||
|
|
||||||
|
|
||||||
|
class ComponentsNotStartedError(ComponentError):
|
||||||
|
|
||||||
|
def __init__(self, message):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(f"{message}")
|
||||||
|
|
||||||
|
|
||||||
|
class CurrencyExchangeError(BaseError):
|
||||||
|
"""
|
||||||
|
**Currency Exchange**
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidExchangeRateResponseError(CurrencyExchangeError):
|
||||||
|
|
||||||
|
def __init__(self, source, reason):
|
||||||
|
self.source = source
|
||||||
|
self.reason = reason
|
||||||
|
super().__init__(f"Failed to get exchange rate from {source}: {reason}")
|
||||||
|
|
||||||
|
|
||||||
|
class CurrencyConversionError(CurrencyExchangeError):
|
||||||
|
|
||||||
|
def __init__(self, message):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(f"{message}")
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidCurrencyError(CurrencyExchangeError):
|
||||||
|
|
||||||
|
def __init__(self, currency):
|
||||||
|
self.currency = currency
|
||||||
|
super().__init__(f"Invalid currency: {currency} is not a supported currency.")
|
9
lbry/error/base.py
Normal file
9
lbry/error/base.py
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
from binascii import hexlify
|
||||||
|
|
||||||
|
|
||||||
|
def claim_id(claim_hash):
|
||||||
|
return hexlify(claim_hash[::-1]).decode()
|
||||||
|
|
||||||
|
|
||||||
|
class BaseError(Exception):
|
||||||
|
pass
|
167
lbry/error/generate.py
Normal file
167
lbry/error/generate.py
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
import re
|
||||||
|
import sys
|
||||||
|
import argparse
|
||||||
|
from pathlib import Path
|
||||||
|
from textwrap import fill, indent
|
||||||
|
|
||||||
|
|
||||||
|
INDENT = ' ' * 4
|
||||||
|
|
||||||
|
CLASS = """
|
||||||
|
|
||||||
|
class {name}({parents}):{doc}
|
||||||
|
"""
|
||||||
|
|
||||||
|
INIT = """
|
||||||
|
def __init__({args}):{fields}
|
||||||
|
super().__init__({format}"{message}")
|
||||||
|
"""
|
||||||
|
|
||||||
|
FUNCTIONS = ['claim_id']
|
||||||
|
|
||||||
|
|
||||||
|
class ErrorClass:
|
||||||
|
|
||||||
|
def __init__(self, hierarchy, name, message):
|
||||||
|
self.hierarchy = hierarchy.replace('**', '')
|
||||||
|
self.other_parents = []
|
||||||
|
if '(' in name:
|
||||||
|
assert ')' in name, f"Missing closing parenthesis in '{name}'."
|
||||||
|
self.other_parents = name[name.find('(')+1:name.find(')')].split(',')
|
||||||
|
name = name[:name.find('(')]
|
||||||
|
self.name = name
|
||||||
|
self.class_name = name+'Error'
|
||||||
|
self.message = message
|
||||||
|
self.comment = ""
|
||||||
|
if '--' in message:
|
||||||
|
self.message, self.comment = message.split('--')
|
||||||
|
self.message = self.message.strip()
|
||||||
|
self.comment = self.comment.strip()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_leaf(self):
|
||||||
|
return 'x' not in self.hierarchy
|
||||||
|
|
||||||
|
@property
|
||||||
|
def code(self):
|
||||||
|
return self.hierarchy.replace('x', '')
|
||||||
|
|
||||||
|
@property
|
||||||
|
def parent_codes(self):
|
||||||
|
return self.hierarchy[0:2], self.hierarchy[0]
|
||||||
|
|
||||||
|
def get_arguments(self):
|
||||||
|
args = ['self']
|
||||||
|
for arg in re.findall('{([a-z0-1_()]+)}', self.message):
|
||||||
|
for func in FUNCTIONS:
|
||||||
|
if arg.startswith(f'{func}('):
|
||||||
|
arg = arg[len(f'{func}('):-1]
|
||||||
|
break
|
||||||
|
args.append(arg)
|
||||||
|
return args
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_fields(args):
|
||||||
|
if len(args) > 1:
|
||||||
|
return ''.join(f'\n{INDENT*2}self.{field} = {field}' for field in args[1:])
|
||||||
|
return ''
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_doc_string(doc):
|
||||||
|
if doc:
|
||||||
|
return f'\n{INDENT}"""\n{indent(fill(doc, 100), INDENT)}\n{INDENT}"""'
|
||||||
|
return ""
|
||||||
|
|
||||||
|
def render(self, out, parent):
|
||||||
|
if not parent:
|
||||||
|
parents = ['BaseError']
|
||||||
|
else:
|
||||||
|
parents = [parent.class_name]
|
||||||
|
parents += self.other_parents
|
||||||
|
args = self.get_arguments()
|
||||||
|
if self.is_leaf:
|
||||||
|
out.write((CLASS + INIT).format(
|
||||||
|
name=self.class_name, parents=', '.join(parents),
|
||||||
|
args=', '.join(args), fields=self.get_fields(args),
|
||||||
|
message=self.message, doc=self.get_doc_string(self.comment), format='f' if len(args) > 1 else ''
|
||||||
|
))
|
||||||
|
else:
|
||||||
|
out.write(CLASS.format(
|
||||||
|
name=self.class_name, parents=', '.join(parents),
|
||||||
|
doc=self.get_doc_string(self.comment or self.message)
|
||||||
|
))
|
||||||
|
|
||||||
|
|
||||||
|
def get_errors():
|
||||||
|
with open('README.md', 'r') as readme:
|
||||||
|
lines = iter(readme.readlines())
|
||||||
|
for line in lines:
|
||||||
|
if line.startswith('## Exceptions Table'):
|
||||||
|
break
|
||||||
|
for line in lines:
|
||||||
|
if line.startswith('---:|'):
|
||||||
|
break
|
||||||
|
for line in lines:
|
||||||
|
if not line:
|
||||||
|
break
|
||||||
|
yield ErrorClass(*[c.strip() for c in line.split('|')])
|
||||||
|
|
||||||
|
|
||||||
|
def find_parent(stack, child):
|
||||||
|
for parent_code in child.parent_codes:
|
||||||
|
parent = stack.get(parent_code)
|
||||||
|
if parent:
|
||||||
|
return parent
|
||||||
|
|
||||||
|
|
||||||
|
def generate(out):
|
||||||
|
out.write(f"from .base import BaseError, {', '.join(FUNCTIONS)}\n")
|
||||||
|
stack = {}
|
||||||
|
for error in get_errors():
|
||||||
|
error.render(out, find_parent(stack, error))
|
||||||
|
if not error.is_leaf:
|
||||||
|
assert error.code not in stack, f"Duplicate code: {error.code}"
|
||||||
|
stack[error.code] = error
|
||||||
|
|
||||||
|
|
||||||
|
def analyze():
|
||||||
|
errors = {e.class_name: [] for e in get_errors() if e.is_leaf}
|
||||||
|
here = Path(__file__).absolute().parents[0]
|
||||||
|
module = here.parent
|
||||||
|
for file_path in module.glob('**/*.py'):
|
||||||
|
if here in file_path.parents:
|
||||||
|
continue
|
||||||
|
with open(file_path) as src_file:
|
||||||
|
src = src_file.read()
|
||||||
|
for error in errors.keys():
|
||||||
|
found = src.count(error)
|
||||||
|
if found > 0:
|
||||||
|
errors[error].append((file_path, found))
|
||||||
|
|
||||||
|
print('Unused Errors:\n')
|
||||||
|
for error, used in errors.items():
|
||||||
|
if used:
|
||||||
|
print(f' - {error}')
|
||||||
|
for use in used:
|
||||||
|
print(f' {use[0].relative_to(module.parent)} {use[1]}')
|
||||||
|
print('')
|
||||||
|
|
||||||
|
print('')
|
||||||
|
print('Unused Errors:')
|
||||||
|
for error, used in errors.items():
|
||||||
|
if not used:
|
||||||
|
print(f' - {error}')
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument("action", choices=['generate', 'analyze'])
|
||||||
|
args = parser.parse_args()
|
||||||
|
if args.action == "analyze":
|
||||||
|
analyze()
|
||||||
|
elif args.action == "generate":
|
||||||
|
generate(sys.stdout)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue