mirror of
https://github.com/langbot-app/LangBot.git
synced 2025-11-26 03:44:58 +08:00
Compare commits
1337 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
8a9000cc67 | ||
|
|
6e3514c0b2 | ||
|
|
2782c8cebe | ||
|
|
13e29a9966 | ||
|
|
601b0a8964 | ||
|
|
7c2ceb0aca | ||
|
|
42fabd5133 | ||
|
|
210a8856e2 | ||
|
|
c531cb11af | ||
|
|
07e073f526 | ||
|
|
c5457374a8 | ||
|
|
5198349591 | ||
|
|
8a4967525a | ||
|
|
30b068c6e2 | ||
|
|
ea3fff59ac | ||
|
|
b09ce8296f | ||
|
|
f9d07779a9 | ||
|
|
51634c1caf | ||
|
|
0e00da6617 | ||
|
|
5ee6baeaaa | ||
|
|
f11a036c60 | ||
|
|
0ac02ff4ce | ||
|
|
99cc50b5cb | ||
|
|
1d8fb02989 | ||
|
|
122cb1188c | ||
|
|
ca36ade288 | ||
|
|
0877046db7 | ||
|
|
ce9615a00e | ||
|
|
dbe5a41395 | ||
|
|
4a4ca54c6e | ||
|
|
47acb63feb | ||
|
|
038c5d41e2 | ||
|
|
011a795895 | ||
|
|
873a0339d8 | ||
|
|
715da548c8 | ||
|
|
5378c6ba35 | ||
|
|
8799f86ea4 | ||
|
|
686be4acbc | ||
|
|
5744eca37a | ||
|
|
70f8ddb1ba | ||
|
|
be1328cee9 | ||
|
|
c0dbf6fd13 | ||
|
|
ffe9c3e0f8 | ||
|
|
e20b79b0ed | ||
|
|
e04d46db2c | ||
|
|
7341435127 | ||
|
|
8b56f94667 | ||
|
|
f5e98d4ebb | ||
|
|
23a0dba470 | ||
|
|
512371cc25 | ||
|
|
cd4a06b692 | ||
|
|
629ebae0e9 | ||
|
|
394d4b3c1b | ||
|
|
432440d6bf | ||
|
|
a0fd152d19 | ||
|
|
1a62e08bab | ||
|
|
edbc59c117 | ||
|
|
cfdd0f8cb2 | ||
|
|
808f30675d | ||
|
|
46072abb41 | ||
|
|
71ffbb9eb5 | ||
|
|
27bbb2297a | ||
|
|
0d235aaef8 | ||
|
|
e22c804deb | ||
|
|
c136e790ef | ||
|
|
3697afd9d6 | ||
|
|
c597c6482a | ||
|
|
dda8c637d8 | ||
|
|
e6d7aaa440 | ||
|
|
028458b33c | ||
|
|
9c7d8099cb | ||
|
|
5640dc332d | ||
|
|
40275c3ef1 | ||
|
|
ebe0b2f335 | ||
|
|
97603e8441 | ||
|
|
72cd444861 | ||
|
|
955b859f2c | ||
|
|
dea5cc9c0c | ||
|
|
d13ab1703e | ||
|
|
61ab6a009b | ||
|
|
a9ae36d362 | ||
|
|
f518395ce5 | ||
|
|
20b17fe378 | ||
|
|
572182180c | ||
|
|
de261099aa | ||
|
|
50f0122955 | ||
|
|
fe9eff923e | ||
|
|
dd36278032 | ||
|
|
a079821976 | ||
|
|
fa233e0a24 | ||
|
|
22306cb4ea | ||
|
|
f2d45a3668 | ||
|
|
db91ff12f7 | ||
|
|
eb841fb73e | ||
|
|
bd0438df76 | ||
|
|
9ca1fc59ef | ||
|
|
84a80a5ec8 | ||
|
|
4b2e248646 | ||
|
|
b90e45590a | ||
|
|
ff93d563a8 | ||
|
|
53228498ed | ||
|
|
8ece82e43a | ||
|
|
8b4684675e | ||
|
|
8cca12fff2 | ||
|
|
a74111612e | ||
|
|
c7799a65c4 | ||
|
|
aabb01c50f | ||
|
|
95e2ada965 | ||
|
|
3fe7d53c76 | ||
|
|
e8634bb1ab | ||
|
|
dbe46b5770 | ||
|
|
6d9fba30b1 | ||
|
|
6a866bf871 | ||
|
|
3c961e4652 | ||
|
|
7abd999420 | ||
|
|
fca8fbb135 | ||
|
|
c67caf18df | ||
|
|
fe956fe4a5 | ||
|
|
0e52f679a2 | ||
|
|
b9500283ec | ||
|
|
441b69b528 | ||
|
|
898bcdc96b | ||
|
|
02bc1fc45e | ||
|
|
5585981dc3 | ||
|
|
a4777f194b | ||
|
|
41aeda8dc0 | ||
|
|
2ed522667e | ||
|
|
1932444666 | ||
|
|
b49b7e963d | ||
|
|
435c11ff27 | ||
|
|
2e93600437 | ||
|
|
faecb70d0f | ||
|
|
92e1ac5c3a | ||
|
|
8963a2117b | ||
|
|
aa300258ab | ||
|
|
48841daff5 | ||
|
|
8878f1ed87 | ||
|
|
f6205d79c0 | ||
|
|
d6d5dac6b3 | ||
|
|
05b979e68a | ||
|
|
9f7d9e4c0d | ||
|
|
98a9fed726 | ||
|
|
720a218259 | ||
|
|
60c0adc6f9 | ||
|
|
bc8c346e68 | ||
|
|
a198b6da0b | ||
|
|
0f3dc35df4 | ||
|
|
7b6e6b046a | ||
|
|
9e503191d6 | ||
|
|
1fd23a0d8d | ||
|
|
3811700a78 | ||
|
|
8762ba3d9c | ||
|
|
c42b5aab5a | ||
|
|
d724899ec0 | ||
|
|
81aacdd76e | ||
|
|
0aa072b4e8 | ||
|
|
6335e9dd8b | ||
|
|
a785289ac9 | ||
|
|
f8bace040c | ||
|
|
d62d597695 | ||
|
|
d938129884 | ||
|
|
327f448321 | ||
|
|
19af3740c1 | ||
|
|
11b1110eed | ||
|
|
682b897e21 | ||
|
|
998ad7623c | ||
|
|
4f1db33abc | ||
|
|
ca6cb60bdd | ||
|
|
133e48a5a9 | ||
|
|
d659d01b1e | ||
|
|
34f73fd84b | ||
|
|
54b87ff79d | ||
|
|
6c2843e7c1 | ||
|
|
6761a31982 | ||
|
|
9401a79b2b | ||
|
|
7a4905d943 | ||
|
|
4db1d2b3a3 | ||
|
|
2ffe2967d6 | ||
|
|
0875c0f266 | ||
|
|
68c7de5199 | ||
|
|
4dfb8597ae | ||
|
|
e21a27ff23 | ||
|
|
91ad7944de | ||
|
|
c86602ebaf | ||
|
|
f75ac292db | ||
|
|
2742c249bf | ||
|
|
36f04849ab | ||
|
|
a60c896e89 | ||
|
|
c442320c7f | ||
|
|
6aeae7e9f5 | ||
|
|
cae79aac48 | ||
|
|
0623f4009a | ||
|
|
06adeb72c4 | ||
|
|
ef044f4fc7 | ||
|
|
7cd4e904ca | ||
|
|
c724494ee7 | ||
|
|
cdb2db348e | ||
|
|
5873d4696f | ||
|
|
613787f49c | ||
|
|
f620874251 | ||
|
|
1f08082a58 | ||
|
|
8f5da1677b | ||
|
|
5439a3a31f | ||
|
|
d92ee23764 | ||
|
|
71ecfc2566 | ||
|
|
c0787e0bb6 | ||
|
|
357da2d236 | ||
|
|
6071241872 | ||
|
|
ab93c67081 | ||
|
|
7af6b833df | ||
|
|
3e4b85aeb5 | ||
|
|
2b6be04c5d | ||
|
|
b2d1c82196 | ||
|
|
a19da7b923 | ||
|
|
4a9a78d07b | ||
|
|
300dbd076f | ||
|
|
ddf52524a8 | ||
|
|
7dcc44b4fc | ||
|
|
75af631c17 | ||
|
|
04dd4fce68 | ||
|
|
2776a95a40 | ||
|
|
dc93b37fd6 | ||
|
|
6502a64cab | ||
|
|
5311e78776 | ||
|
|
35721c1340 | ||
|
|
a76df22cab | ||
|
|
a90f996b24 | ||
|
|
c96d4456ea | ||
|
|
d1df6d993f | ||
|
|
191f8866ae | ||
|
|
e17da4e2ee | ||
|
|
2c3fdb4fdc | ||
|
|
e89c6b68c9 | ||
|
|
51cca31f04 | ||
|
|
e51950aa75 | ||
|
|
4c344e0636 | ||
|
|
90261d1f55 | ||
|
|
fabf93f741 | ||
|
|
ab8ef01c76 | ||
|
|
e463d3a8fe | ||
|
|
a6bc617a3b | ||
|
|
1b1ccdd733 | ||
|
|
8d00e710d5 | ||
|
|
de9e3bdbd5 | ||
|
|
b6e054a73f | ||
|
|
a078b2cf12 | ||
|
|
6f32bf9621 | ||
|
|
ac628b26d9 | ||
|
|
7ba655902b | ||
|
|
05c1fdaa9e | ||
|
|
d7687913a9 | ||
|
|
9e718a2e8a | ||
|
|
cbec2f6d02 | ||
|
|
52eb37d13d | ||
|
|
8e9f43885a | ||
|
|
9eefbcb6f2 | ||
|
|
4d8ebc8c38 | ||
|
|
21cfb6ee6f | ||
|
|
c72ad2b242 | ||
|
|
e83b0a7825 | ||
|
|
a8f2438288 | ||
|
|
d0ceaff6ed | ||
|
|
dbe6272bd8 | ||
|
|
eceaf85807 | ||
|
|
d0606b79b0 | ||
|
|
412f290606 | ||
|
|
21e1acc4f5 | ||
|
|
326aad3c00 | ||
|
|
493c2e9a16 | ||
|
|
51a87e28e2 | ||
|
|
be2ff20f4b | ||
|
|
19c6b2fc32 | ||
|
|
5d249f441b | ||
|
|
852254eaef | ||
|
|
43ea64befa | ||
|
|
0f2cb58897 | ||
|
|
dbece6af7f | ||
|
|
b1e68182bd | ||
|
|
45a64bea78 | ||
|
|
aec8735388 | ||
|
|
1d91faaa49 | ||
|
|
e1e21c0063 | ||
|
|
e775499080 | ||
|
|
735aad5a91 | ||
|
|
fb4e106f69 | ||
|
|
e5659db535 | ||
|
|
5381e09a6c | ||
|
|
21f16ecd68 | ||
|
|
12fc76b326 | ||
|
|
d7f87dd269 | ||
|
|
56227f3713 | ||
|
|
f492fee486 | ||
|
|
41a7814615 | ||
|
|
8644f2c166 | ||
|
|
e4a9365caf | ||
|
|
9fc7af1295 | ||
|
|
d0eeb2b304 | ||
|
|
e4518ebcf1 | ||
|
|
7604cefd0f | ||
|
|
71729d4784 | ||
|
|
1d16bc4968 | ||
|
|
de2bf79004 | ||
|
|
83ed7a9f38 | ||
|
|
c326e72758 | ||
|
|
ac9cef82cc | ||
|
|
ea254d57d2 | ||
|
|
a661f24ae0 | ||
|
|
afabf9256b | ||
|
|
74a8f9c9e2 | ||
|
|
1d11e448f9 | ||
|
|
e3e23cbccb | ||
|
|
79132aa11d | ||
|
|
7bb9e6e951 | ||
|
|
37dc5b4135 | ||
|
|
d588faf470 | ||
|
|
8b51a81158 | ||
|
|
9f125974bf | ||
|
|
d0aed48ca9 | ||
|
|
bf548df6ae | ||
|
|
a3fe105f8e | ||
|
|
5add1d71bc | ||
|
|
7a01cff0c8 | ||
|
|
e8602f7134 | ||
|
|
e9aad2c8d7 | ||
|
|
60d4f3d77c | ||
|
|
9b8c5a3499 | ||
|
|
53dde0607d | ||
|
|
7f034b4ffa | ||
|
|
599ab83100 | ||
|
|
f4a3508ec2 | ||
|
|
44b92909eb | ||
|
|
8ed07b8d1a | ||
|
|
2ff9ced15e | ||
|
|
641b8d71ed | ||
|
|
a31b450f54 | ||
|
|
97bb24c5b9 | ||
|
|
5e5a3639d1 | ||
|
|
0a68a77e28 | ||
|
|
11a0c4142e | ||
|
|
d214d80579 | ||
|
|
ed719fd44e | ||
|
|
5dc6bed0d1 | ||
|
|
b1244a4d4e | ||
|
|
6aa325a4b1 | ||
|
|
88a11561f9 | ||
|
|
fd30022065 | ||
|
|
9486312737 | ||
|
|
e37070a985 | ||
|
|
ffb98ecca2 | ||
|
|
29bd69ef97 | ||
|
|
e46c9530cc | ||
|
|
7ddd303e2d | ||
|
|
66798a1d0f | ||
|
|
bd05afdf14 | ||
|
|
136e48f7ee | ||
|
|
facb5f177a | ||
|
|
10ce31cc46 | ||
|
|
3b4f3c516b | ||
|
|
a1e3981ce4 | ||
|
|
89f26781fe | ||
|
|
914292a80b | ||
|
|
8227e3299b | ||
|
|
07ca48d652 | ||
|
|
243f45c7db | ||
|
|
12cfce3622 | ||
|
|
535c4a8a11 | ||
|
|
6606c671b2 | ||
|
|
242f24840d | ||
|
|
486f636b2d | ||
|
|
b293d7a7cd | ||
|
|
f4fa0b42a6 | ||
|
|
209e89712d | ||
|
|
3314a7a9e9 | ||
|
|
793d64303e | ||
|
|
6642498f00 | ||
|
|
32b400dcb1 | ||
|
|
0dcd2d8179 | ||
|
|
736f8b613c | ||
|
|
9e7d9a937d | ||
|
|
4767983279 | ||
|
|
e37f35d95a | ||
|
|
ad1e609fb9 | ||
|
|
f9bc4a5acd | ||
|
|
2b79185f6a | ||
|
|
840f638472 | ||
|
|
908169a55e | ||
|
|
dbf9f2398e | ||
|
|
2ea3ff0b5c | ||
|
|
91bf72c710 | ||
|
|
baabb70622 | ||
|
|
94ea64a6a9 | ||
|
|
f97896b2c7 | ||
|
|
9027db8587 | ||
|
|
cd46e1c131 | ||
|
|
59211191a4 | ||
|
|
a3ca7e82c7 | ||
|
|
0094056def | ||
|
|
a9f305a1c6 | ||
|
|
e8cc048901 | ||
|
|
05da43f606 | ||
|
|
a81faa7d8e | ||
|
|
18ba7d1da7 | ||
|
|
875adfcbaa | ||
|
|
6e9c213893 | ||
|
|
753066ccb9 | ||
|
|
8b36782c25 | ||
|
|
da9dde6bd2 | ||
|
|
07f6e69b93 | ||
|
|
31a7503df3 | ||
|
|
11db8d8d17 | ||
|
|
93ee8d51bc | ||
|
|
83e80f324e | ||
|
|
c51eac717e | ||
|
|
db7d5dcce3 | ||
|
|
0d25578e22 | ||
|
|
1a457be823 | ||
|
|
20e3edba8f | ||
|
|
036c2182a5 | ||
|
|
6238f430e8 | ||
|
|
9fc891ec01 | ||
|
|
491d977d9e | ||
|
|
9a4bcda9bc | ||
|
|
2c2374a763 | ||
|
|
a76e0b287e | ||
|
|
1d6f1e3c7c | ||
|
|
896fd982a1 | ||
|
|
c031ab20da | ||
|
|
318b6e6bf1 | ||
|
|
ca3999d251 | ||
|
|
658eb278c4 | ||
|
|
bb219889e5 | ||
|
|
3239c9ec3f | ||
|
|
16153dc573 | ||
|
|
e0d9a295ab | ||
|
|
eabdda5eb1 | ||
|
|
43f45f9184 | ||
|
|
7c19785a17 | ||
|
|
78005f8b4e | ||
|
|
0d4784d098 | ||
|
|
805454e037 | ||
|
|
bf383bbf9c | ||
|
|
73ffd67792 | ||
|
|
54bbfc8eda | ||
|
|
a3e234c979 | ||
|
|
9336abff8b | ||
|
|
0fe161cd7f | ||
|
|
7cc55eab3e | ||
|
|
15482e398b | ||
|
|
601fa0ac7f | ||
|
|
2819da5f2f | ||
|
|
3cb3562477 | ||
|
|
cee205994f | ||
|
|
e44df0a3dd | ||
|
|
84a51cb26d | ||
|
|
db02d9c126 | ||
|
|
709b86b724 | ||
|
|
68184b0e47 | ||
|
|
6d2a4c038d | ||
|
|
2f05f5b456 | ||
|
|
d5e3120350 | ||
|
|
a4589327a6 | ||
|
|
c151665419 | ||
|
|
947790e8d1 | ||
|
|
26770439bb | ||
|
|
7da9171dde | ||
|
|
16b386eaf7 | ||
|
|
c330aab48b | ||
|
|
5f998a0852 | ||
|
|
c3dfbb64a6 | ||
|
|
3db52282b8 | ||
|
|
a313ae5f97 | ||
|
|
18cce189a4 | ||
|
|
fb308d576b | ||
|
|
8c976303a4 | ||
|
|
12f1f3609d | ||
|
|
661fdeb6a1 | ||
|
|
d52f9b9543 | ||
|
|
7174742886 | ||
|
|
cd0a8fb24b | ||
|
|
1fbc92bc6d | ||
|
|
231dca956d | ||
|
|
0dd74c825b | ||
|
|
9703fc0366 | ||
|
|
7c3557e943 | ||
|
|
21f153e5c3 | ||
|
|
ea6a0af5a7 | ||
|
|
c53ffaca6c | ||
|
|
3469515e04 | ||
|
|
e8da26cb8a | ||
|
|
1235fc1339 | ||
|
|
47e308b99d | ||
|
|
fdba470e9a | ||
|
|
a1ccceefd2 | ||
|
|
1c4a700d92 | ||
|
|
81c2c3c0e5 | ||
|
|
3c2db5097a | ||
|
|
ce56f79687 | ||
|
|
ee0d6dcdae | ||
|
|
bcf1d92f73 | ||
|
|
ffdec16ce6 | ||
|
|
b2f6e84adc | ||
|
|
f76c457e1f | ||
|
|
80bd0a20df | ||
|
|
efeaf73339 | ||
|
|
91b5100a24 | ||
|
|
d1a06f4730 | ||
|
|
b0b186e951 | ||
|
|
4c8fedef6e | ||
|
|
718c221d01 | ||
|
|
077e77eee5 | ||
|
|
b51ca06c7c | ||
|
|
2f092f4a87 | ||
|
|
f1ff9c05c4 | ||
|
|
c9c8603ccc | ||
|
|
47e281fb61 | ||
|
|
dc625647eb | ||
|
|
66cf1b05be | ||
|
|
622cc89414 | ||
|
|
78d98c40b1 | ||
|
|
1c5f06d9a9 | ||
|
|
998fe5a980 | ||
|
|
8cad4089a7 | ||
|
|
48cc3656bd | ||
|
|
68ddb3a6e1 | ||
|
|
70583f5ba0 | ||
|
|
5bebe01dd0 | ||
|
|
4dd976c9c5 | ||
|
|
221b310485 | ||
|
|
dd1cec70c0 | ||
|
|
7656443b28 | ||
|
|
9d91c13b12 | ||
|
|
7c06141ce2 | ||
|
|
3dc413638b | ||
|
|
bdb8baeddd | ||
|
|
21966bfb69 | ||
|
|
e78c82e999 | ||
|
|
2bdc3468d1 | ||
|
|
987b3dc4ef | ||
|
|
45a10b4ac7 | ||
|
|
b5d33ef629 | ||
|
|
d3629916bf | ||
|
|
c5cb26d295 | ||
|
|
4b2785c5eb | ||
|
|
7ed190e6d2 | ||
|
|
eac041cdd2 | ||
|
|
05527cfc01 | ||
|
|
61e2af4a14 | ||
|
|
79804b6ecd | ||
|
|
76434b2f4e | ||
|
|
ec8bd4922e | ||
|
|
4ffa773fac | ||
|
|
ea8b7bc8aa | ||
|
|
39ce5646f6 | ||
|
|
5092a82739 | ||
|
|
3bba0b6d9a | ||
|
|
7a19dd503d | ||
|
|
9e6a01fefd | ||
|
|
933471b4d9 | ||
|
|
f81808d239 | ||
|
|
96832b6f7d | ||
|
|
e2eb0a84b0 | ||
|
|
c8eb2e3376 | ||
|
|
21fe5822f9 | ||
|
|
d49cc9a7a3 | ||
|
|
910d0bfae1 | ||
|
|
d6761949ca | ||
|
|
6afac1f593 | ||
|
|
4d1a270d22 | ||
|
|
a7888f5536 | ||
|
|
b9049e91cf | ||
|
|
7db56c8e77 | ||
|
|
50563cb957 | ||
|
|
18ae2299a7 | ||
|
|
7463e0aab9 | ||
|
|
c92d47bb95 | ||
|
|
0b1af7df91 | ||
|
|
a9104eb2da | ||
|
|
abbd15d5cc | ||
|
|
aadfa14d59 | ||
|
|
4cd10bbe25 | ||
|
|
1d4a6b71ab | ||
|
|
a7f830dd73 | ||
|
|
bae86ac05c | ||
|
|
a3706bfe21 | ||
|
|
91e23b8c11 | ||
|
|
37ef1c9fab | ||
|
|
6bc6f77af1 | ||
|
|
2c478ccc25 | ||
|
|
404e5492a3 | ||
|
|
d5b5d667a5 | ||
|
|
8807f02f36 | ||
|
|
269e561497 | ||
|
|
527ad81d38 | ||
|
|
972d3c18af | ||
|
|
3cbfc078fc | ||
|
|
fde6822b5c | ||
|
|
930321bcf1 | ||
|
|
c45931363a | ||
|
|
9c6491e5ee | ||
|
|
9bc248f5bc | ||
|
|
becac2fde5 | ||
|
|
1e1a103882 | ||
|
|
e5cffb7c9b | ||
|
|
e2becf7777 | ||
|
|
a6b875a242 | ||
|
|
b5e67f3df8 | ||
|
|
2093fb16a7 | ||
|
|
fc9a9d2386 | ||
|
|
5e69f78f7e | ||
|
|
6919bece77 | ||
|
|
8b003739f1 | ||
|
|
2e9229a6ad | ||
|
|
5a3e7fe8ee | ||
|
|
7b3d7e7bd6 | ||
|
|
fdd7c1864d | ||
|
|
cac5a5adff | ||
|
|
63307633c2 | ||
|
|
387dfa39ff | ||
|
|
1f797f899c | ||
|
|
092bb0a1e2 | ||
|
|
2c3399e237 | ||
|
|
835275b47f | ||
|
|
7b060ce3f9 | ||
|
|
1fb69311b0 | ||
|
|
995d1f61d2 | ||
|
|
80258e9182 | ||
|
|
bd6a32e08e | ||
|
|
5f138de75b | ||
|
|
d0b0f2209a | ||
|
|
0752698c1d | ||
|
|
9855c6b8f5 | ||
|
|
52a7c25540 | ||
|
|
fa823de6b0 | ||
|
|
f53070d8b6 | ||
|
|
7677672691 | ||
|
|
dead8fa168 | ||
|
|
c6347bea45 | ||
|
|
32bd194bfc | ||
|
|
cca48a394d | ||
|
|
a723c8ce37 | ||
|
|
327b2509f6 | ||
|
|
1dae7bd655 | ||
|
|
550a131685 | ||
|
|
0cfb8bb29f | ||
|
|
9c32420a95 | ||
|
|
867093cc88 | ||
|
|
82763f8ec5 | ||
|
|
97449065df | ||
|
|
9489783846 | ||
|
|
f91c9015bc | ||
|
|
302d86056d | ||
|
|
98bebfddaa | ||
|
|
dab20e3187 | ||
|
|
09e72f7c5f | ||
|
|
2028d85f84 | ||
|
|
ed3c0d9014 | ||
|
|
be06150990 | ||
|
|
afb3fb4a31 | ||
|
|
d66577e6c3 | ||
|
|
6a4ea5446a | ||
|
|
74e84c744a | ||
|
|
5ad2446cf3 | ||
|
|
63303bb5c0 | ||
|
|
13393b6624 | ||
|
|
b9fa11c0c3 | ||
|
|
8c6ce1f030 | ||
|
|
1d963d0f0c | ||
|
|
0ee383be27 | ||
|
|
53d09129b4 | ||
|
|
a398c6f311 | ||
|
|
4347ddd42a | ||
|
|
22cb8a6a06 | ||
|
|
7f554fd862 | ||
|
|
a82bfa8a56 | ||
|
|
95784debbf | ||
|
|
2471c5bf0f | ||
|
|
2fe6d731b8 | ||
|
|
ce881372ee | ||
|
|
171ea7c375 | ||
|
|
1e9a6f813f | ||
|
|
39a7f3b2b9 | ||
|
|
8d375a02db | ||
|
|
cac8a0a414 | ||
|
|
c89623967e | ||
|
|
92aa9c1711 | ||
|
|
71f2a58acb | ||
|
|
1f07a8a9e3 | ||
|
|
cacd21bde7 | ||
|
|
a060ec66c3 | ||
|
|
fd10db3c75 | ||
|
|
db4c658980 | ||
|
|
0ee88674f8 | ||
|
|
3540759682 | ||
|
|
44cc8f15b4 | ||
|
|
59f821bf0a | ||
|
|
80858672b0 | ||
|
|
3258d5b255 | ||
|
|
e8c8cc0a9c | ||
|
|
570c19f29f | ||
|
|
ee93fd8636 | ||
|
|
1e6c32ffc7 | ||
|
|
3ef2fb958c | ||
|
|
97edfe7cd7 | ||
|
|
1bdc96f8b2 | ||
|
|
4ef285aee9 | ||
|
|
6ccee3b7cf | ||
|
|
082731ba32 | ||
|
|
0bf85fb644 | ||
|
|
5ce1759dd9 | ||
|
|
1e016dfa24 | ||
|
|
7b3bb53f06 | ||
|
|
53d0059848 | ||
|
|
9a85178a29 | ||
|
|
d74681a128 | ||
|
|
06c8773975 | ||
|
|
ae358dd6d0 | ||
|
|
7174cbf41f | ||
|
|
f73d69e814 | ||
|
|
8af174127d | ||
|
|
991a0aa5f6 | ||
|
|
abc19e78b8 | ||
|
|
836df87e18 | ||
|
|
9cad94e961 | ||
|
|
b9568eb558 | ||
|
|
f951625025 | ||
|
|
c2b3b53c12 | ||
|
|
d95e18c202 | ||
|
|
e705e707e5 | ||
|
|
2fa5d7608f | ||
|
|
f9a3e99795 | ||
|
|
d86ad25f86 | ||
|
|
cf583486e3 | ||
|
|
7366ca59c7 | ||
|
|
12820e6c64 | ||
|
|
71b54fd684 | ||
|
|
aeb1912db6 | ||
|
|
84b2867148 | ||
|
|
5880dacad8 | ||
|
|
b5b67ad958 | ||
|
|
2a913ed24c | ||
|
|
aab56294ba | ||
|
|
26912ef976 | ||
|
|
c1fed3410b | ||
|
|
c853bba4ba | ||
|
|
f340a44abf | ||
|
|
0dec10ddf2 | ||
|
|
7026abe56a | ||
|
|
a9d92115f8 | ||
|
|
6f2d7d96d0 | ||
|
|
532a713355 | ||
|
|
976a9de39c | ||
|
|
32162afa65 | ||
|
|
c1c751a9ab | ||
|
|
b749ba587d | ||
|
|
b2741686fd | ||
|
|
94bf7739a0 | ||
|
|
33d600fb6b | ||
|
|
e2de3d0102 | ||
|
|
6b76adc00e | ||
|
|
61f4cb2f65 | ||
|
|
28bd232dda | ||
|
|
e9e458c877 | ||
|
|
437971ded8 | ||
|
|
3945ac95d1 | ||
|
|
13ab647dc0 | ||
|
|
c75b0ce8fb | ||
|
|
6cc4688660 | ||
|
|
b730f17eb6 | ||
|
|
698782c537 | ||
|
|
2b0faea8ec | ||
|
|
d130c376f4 | ||
|
|
238c55a40e | ||
|
|
b5924bb34f | ||
|
|
1368ee22b2 | ||
|
|
2a0cf57303 | ||
|
|
f10af09bd2 | ||
|
|
850a4eeb7c | ||
|
|
411034902a | ||
|
|
1900ddacbb | ||
|
|
8d084427d2 | ||
|
|
a064c24f60 | ||
|
|
b43882aad0 | ||
|
|
f4ead5ec5c | ||
|
|
ea9ae85428 | ||
|
|
a9a798b19d | ||
|
|
f4ae9df3bf | ||
|
|
f3bcff1261 | ||
|
|
b4bd86549e | ||
|
|
a975718a64 | ||
|
|
3d06a18bcb | ||
|
|
a236089785 | ||
|
|
2f877965cf | ||
|
|
ad5ef95e65 | ||
|
|
8d35ecd711 | ||
|
|
e63c6ac723 | ||
|
|
0984c19fd9 | ||
|
|
a10d3213fd | ||
|
|
f52a0eb02f | ||
|
|
1ea8da69a2 | ||
|
|
5bbc38a7a3 | ||
|
|
aa433bd5ab | ||
|
|
2c5933da0b | ||
|
|
77bc6fbf59 | ||
|
|
701cb7be40 | ||
|
|
ab8d77c968 | ||
|
|
6c03fe678a | ||
|
|
41b30238c3 | ||
|
|
aa768459c0 | ||
|
|
28014512f7 | ||
|
|
f9a99eed66 | ||
|
|
461b574e09 | ||
|
|
36c192ff6b | ||
|
|
101625965c | ||
|
|
83177a3416 | ||
|
|
c3904786e1 | ||
|
|
b31c34905a | ||
|
|
41cbe91870 | ||
|
|
872b16b779 | ||
|
|
9f3cc9c293 | ||
|
|
2d148c4970 | ||
|
|
0869b57741 | ||
|
|
af225aa18f | ||
|
|
06f3c5d32b | ||
|
|
4e71a08b57 | ||
|
|
bf5ebc9245 | ||
|
|
fba81582ab | ||
|
|
b4645168f9 | ||
|
|
d00c68e329 | ||
|
|
cb636b96bf | ||
|
|
12468b5b15 | ||
|
|
6a5414b5fd | ||
|
|
db51fd0ad7 | ||
|
|
256bc4dc1e | ||
|
|
d2bd6e23b6 | ||
|
|
bb12b48887 | ||
|
|
a58e55daf3 | ||
|
|
23a05fe5b0 | ||
|
|
3a63630068 | ||
|
|
565066bbcd | ||
|
|
c10f72cf4c | ||
|
|
af8c21f3d4 | ||
|
|
6f6c3af302 | ||
|
|
61a47808c8 | ||
|
|
e02765bf95 | ||
|
|
b69f193a3e | ||
|
|
7c6526d1ea | ||
|
|
b8776fba65 | ||
|
|
38357dd68d | ||
|
|
d1c2453310 | ||
|
|
ebc1ac50c6 | ||
|
|
892610872f | ||
|
|
a990a40850 | ||
|
|
3f29464dbd | ||
|
|
998d07f3b4 | ||
|
|
949bc6268c | ||
|
|
2c03e5a77e | ||
|
|
aad62dfa6f | ||
|
|
08e27d07ea | ||
|
|
1fddd244e5 | ||
|
|
d85b4b1cf0 | ||
|
|
09fca2c292 | ||
|
|
feda3d18fb | ||
|
|
eb6e5d0756 | ||
|
|
7386daad28 | ||
|
|
3f290b2e1a | ||
|
|
43519ffe80 | ||
|
|
c8bb3d612a | ||
|
|
bc48b7e623 | ||
|
|
d59d5797f6 | ||
|
|
11d3c1e650 | ||
|
|
8cfd9e6694 | ||
|
|
d3f401c54d | ||
|
|
a889170d1a | ||
|
|
459e9f9322 | ||
|
|
707afdcdf9 | ||
|
|
ad1cf379c4 | ||
|
|
582277fe2d | ||
|
|
14b9f814c7 | ||
|
|
b11e5d99b0 | ||
|
|
9590718da4 | ||
|
|
8c2b53cffb | ||
|
|
5a85c073a8 | ||
|
|
2d2fbd0a8b | ||
|
|
1b25a05122 | ||
|
|
709cc1140b | ||
|
|
1730962636 | ||
|
|
a1de4f6f7a | ||
|
|
a5ccda5ed6 | ||
|
|
f035e654ba | ||
|
|
151d3e9f66 | ||
|
|
c79207e197 | ||
|
|
f9d461d9a1 | ||
|
|
3e17bbb90f | ||
|
|
549a7eff7f | ||
|
|
db2e366014 | ||
|
|
26e4215054 | ||
|
|
5f07ff8145 | ||
|
|
e396ba4649 | ||
|
|
d1dff6dedd | ||
|
|
419354cb07 | ||
|
|
7708eaa82c | ||
|
|
9fccf84987 | ||
|
|
0f59788184 | ||
|
|
0ad52bcd3f | ||
|
|
d7d710ec07 | ||
|
|
75a9a3e9af | ||
|
|
70503bedb7 | ||
|
|
7890eac3f8 | ||
|
|
e15f3595b3 | ||
|
|
eebd6a6ba3 | ||
|
|
0407f3e4ac | ||
|
|
5abca84437 | ||
|
|
d2776cc1e6 | ||
|
|
9fe0ee2b77 | ||
|
|
b68daac323 | ||
|
|
665de5dc43 | ||
|
|
e3b280758c | ||
|
|
374ae25d9c | ||
|
|
c86529ac99 | ||
|
|
6309f1fb78 | ||
|
|
c246fb6d8e | ||
|
|
ec6c041bcf | ||
|
|
2da5a9f3c7 | ||
|
|
4e0df52d7c | ||
|
|
71b8bf13e4 | ||
|
|
a8b1e6ce91 | ||
|
|
1419d7611d | ||
|
|
89c83ebf20 | ||
|
|
76d7db88ea | ||
|
|
67a208bc90 | ||
|
|
acbd55ded2 | ||
|
|
11a240a6d1 | ||
|
|
97c85abbe7 | ||
|
|
06a0cd2a3d | ||
|
|
572b215df8 | ||
|
|
2c542bf412 | ||
|
|
1576ba7a01 | ||
|
|
45e4096a12 | ||
|
|
8a1d4fe287 | ||
|
|
98f880ebc2 | ||
|
|
2b852853f3 | ||
|
|
c7a9988033 | ||
|
|
c475eebe1c | ||
|
|
0fe7355ae0 | ||
|
|
57de96e3a2 | ||
|
|
70571cef50 | ||
|
|
0b6deb3340 | ||
|
|
dcda85a825 | ||
|
|
9d3bff018b | ||
|
|
051376e0d2 | ||
|
|
a113785211 | ||
|
|
3f4ed4dc3c | ||
|
|
ac80764fae | ||
|
|
e43afd4891 | ||
|
|
f1aea1d495 | ||
|
|
0e2a5db104 | ||
|
|
3a4c9771fa | ||
|
|
f4f8ef9523 | ||
|
|
b9ace69a72 | ||
|
|
aef0b2a26e | ||
|
|
f7712d71ec | ||
|
|
e94b44e3b8 | ||
|
|
524e863c78 | ||
|
|
bbc80ac901 | ||
|
|
f969ddd6ca | ||
|
|
1cc9781333 | ||
|
|
a609801bae | ||
|
|
d8b606d372 | ||
|
|
572a440e65 | ||
|
|
6e4eeae9b7 | ||
|
|
1a73669df8 | ||
|
|
91ebaf1122 | ||
|
|
46703eb906 | ||
|
|
b9dd9d5193 | ||
|
|
884481a4ec | ||
|
|
9040b37a63 | ||
|
|
99d47b2fa2 | ||
|
|
6575359a94 | ||
|
|
a2fc726372 | ||
|
|
3bfce8ab51 | ||
|
|
ff9a9830f2 | ||
|
|
e2b59e8efe | ||
|
|
04dad9757f | ||
|
|
75ea1080ad | ||
|
|
e25b064319 | ||
|
|
5d0dbc40ce | ||
|
|
beae8de5eb | ||
|
|
c4ff30c722 | ||
|
|
6f4ecb101b | ||
|
|
9f9b0ef846 | ||
|
|
de6957062c | ||
|
|
0a9b43e6fa | ||
|
|
5b0edd9937 | ||
|
|
8a400d202a | ||
|
|
5a1e9f7fb2 | ||
|
|
e03af75cf8 | ||
|
|
0da4919255 | ||
|
|
914e566d1f | ||
|
|
6ec2b653fe | ||
|
|
ba0a088b9c | ||
|
|
478e83bcd9 | ||
|
|
386124a3b9 | ||
|
|
ff5e7c16d1 | ||
|
|
7ff7a66012 | ||
|
|
c99dfb8a86 | ||
|
|
10f9d4c6b3 | ||
|
|
d347813411 | ||
|
|
7a93898b3f | ||
|
|
c057ea900f | ||
|
|
512266e74f | ||
|
|
e36aee11c7 | ||
|
|
97421299f5 | ||
|
|
bc41e5aa80 | ||
|
|
2fa30e7def | ||
|
|
1c6a7d9ba5 | ||
|
|
47435c42a5 | ||
|
|
39a1b421e6 | ||
|
|
b5edf2295b | ||
|
|
fb650a3d7a | ||
|
|
521541f311 | ||
|
|
7020abadbf | ||
|
|
d95fb3b5be | ||
|
|
3e524dc790 | ||
|
|
a64940bff8 | ||
|
|
c739290f0b | ||
|
|
af292fe050 | ||
|
|
634c7fb302 | ||
|
|
33efb94013 | ||
|
|
549e4dc02e | ||
|
|
3d40909c02 | ||
|
|
1aef81e38f | ||
|
|
1b0ae8da58 | ||
|
|
7979a8e97f | ||
|
|
080e53d9a9 | ||
|
|
89bb364b16 | ||
|
|
3586cd941f | ||
|
|
054d0839ac | ||
|
|
dd75f98d85 | ||
|
|
ec23bb5268 | ||
|
|
bc99db4fc1 | ||
|
|
c8275fcfbf | ||
|
|
a345043c30 | ||
|
|
382d37d479 | ||
|
|
32c144a75d | ||
|
|
7ca2aa5e39 | ||
|
|
86cc4a23ac | ||
|
|
08d1e138bd | ||
|
|
a9fe86542f | ||
|
|
4e29776fcd | ||
|
|
ee3eae8f4d | ||
|
|
a84575858a | ||
|
|
ac472291c7 | ||
|
|
f304873c6a | ||
|
|
18caf8face | ||
|
|
d21115aaa8 | ||
|
|
a05ecd2e7f | ||
|
|
32a725126d | ||
|
|
0528690622 | ||
|
|
819339142e | ||
|
|
1d0573e7ff | ||
|
|
00623bc431 | ||
|
|
c872264456 | ||
|
|
1336d3cb9a | ||
|
|
d1459578cd | ||
|
|
8a67fcf40f | ||
|
|
7930370aa9 | ||
|
|
0b854bdcf1 | ||
|
|
cba6aab48d | ||
|
|
12a9ca7a77 | ||
|
|
a6cbd226e1 | ||
|
|
3577e62b41 | ||
|
|
f86e69fcd1 | ||
|
|
292e00b078 | ||
|
|
2a91497bcf | ||
|
|
b0cca0a4c2 | ||
|
|
a2bda85a9c | ||
|
|
20677cff86 | ||
|
|
c8af5d8445 | ||
|
|
2dbe984539 | ||
|
|
6b8fa664f1 | ||
|
|
2b9612e933 | ||
|
|
749d0219fb | ||
|
|
a11a152bd7 | ||
|
|
fc803a3742 | ||
|
|
13a1e15f24 | ||
|
|
3f41b94da5 | ||
|
|
0fb5bfda20 | ||
|
|
dc1fd73ebb | ||
|
|
161b694f71 | ||
|
|
45d1c89e45 | ||
|
|
e26664aa51 | ||
|
|
e29691efbd | ||
|
|
6d45327882 | ||
|
|
fbd41eef49 | ||
|
|
0a30c88322 | ||
|
|
4f5af0e8c8 | ||
|
|
df3f0fd159 | ||
|
|
f2493c79dd | ||
|
|
a86a035b6b | ||
|
|
7995793bfd | ||
|
|
a56b340646 | ||
|
|
7473cdfe16 | ||
|
|
24273ac158 | ||
|
|
fe6275000e | ||
|
|
5fbf369f82 | ||
|
|
4400475ffa | ||
|
|
796eb7c95d | ||
|
|
89a01378e7 | ||
|
|
f4735e5e30 | ||
|
|
f1bb3045aa | ||
|
|
96e474a555 | ||
|
|
833d29b101 | ||
|
|
dce6734ba2 | ||
|
|
0481167dc6 | ||
|
|
a002f93f7b | ||
|
|
3c894fe70e | ||
|
|
8c69b8a1d9 | ||
|
|
a9dae05303 | ||
|
|
ae6994e241 | ||
|
|
caa72fa40c | ||
|
|
46cc9220c3 | ||
|
|
ddb56d7a8e | ||
|
|
a0267416d7 | ||
|
|
56e1ef3602 | ||
|
|
b4fc1057d1 | ||
|
|
06037df607 | ||
|
|
dce134d08d | ||
|
|
cca471d068 | ||
|
|
ddb211b74a | ||
|
|
cef70751ff | ||
|
|
2d2219fc6e | ||
|
|
514a6b4192 | ||
|
|
7a552b3434 | ||
|
|
ecebd1b0e0 | ||
|
|
8dc34d2a88 | ||
|
|
d52644ceec | ||
|
|
3052510591 | ||
|
|
777a5617db | ||
|
|
e17c1087e9 | ||
|
|
633695175a | ||
|
|
9e78bf3d21 | ||
|
|
43aa68a55d | ||
|
|
b8308f8c57 | ||
|
|
466bfbddeb | ||
|
|
b6da07b225 | ||
|
|
2f2159239a | ||
|
|
67d1ca8a65 | ||
|
|
497a393e83 | ||
|
|
782c0e22ea | ||
|
|
2932fc6dfd | ||
|
|
0a9eab2113 | ||
|
|
50a673a8ec | ||
|
|
9e25d0f9e4 | ||
|
|
23cd7be711 | ||
|
|
025b9e33f1 | ||
|
|
bab2f64913 | ||
|
|
b00e09aa9c | ||
|
|
0b109fdc7a | ||
|
|
018fea2ddb | ||
|
|
f8a3cc4352 | ||
|
|
6ab853acc1 | ||
|
|
e825dea02f | ||
|
|
cf8740d16e | ||
|
|
9c4809e26f | ||
|
|
0a232fd9ef | ||
|
|
23016a0791 | ||
|
|
cdcc67ff23 | ||
|
|
92274bfc34 | ||
|
|
2fed6f61ba | ||
|
|
59b2cd26d2 | ||
|
|
f7b87e99d2 | ||
|
|
70bc985145 | ||
|
|
070dbe9108 | ||
|
|
a63fa6d955 | ||
|
|
c7703809b0 | ||
|
|
37eb74338f | ||
|
|
77d5585b7c | ||
|
|
6cab3ef029 | ||
|
|
820a7b78fc | ||
|
|
c51dffef3a | ||
|
|
983bc3da3c | ||
|
|
09be956a58 | ||
|
|
5eded50c53 | ||
|
|
6d8eebd314 | ||
|
|
19a0572b5f | ||
|
|
6272e98474 | ||
|
|
45042fe7d4 | ||
|
|
d85e840126 | ||
|
|
804889f1de | ||
|
|
919c996434 | ||
|
|
00823b3d62 | ||
|
|
af54efd24a | ||
|
|
b1c9b121f6 | ||
|
|
7b5649d153 | ||
|
|
52bf716d84 | ||
|
|
c149dd7b66 | ||
|
|
65d5a1ed63 | ||
|
|
5516754bbb | ||
|
|
08082f2ee3 | ||
|
|
8489266080 | ||
|
|
51c7e0b235 | ||
|
|
628b6b0bb4 | ||
|
|
7e024d860d | ||
|
|
c2f6273f70 | ||
|
|
96e401ec7b | ||
|
|
ae8ac65447 | ||
|
|
2d4f59f36e | ||
|
|
0e85467e02 | ||
|
|
eb41cf5481 | ||
|
|
b970a42d07 | ||
|
|
8c9d123e1c | ||
|
|
ab2a95e347 | ||
|
|
2184c558a4 | ||
|
|
83cb8588fd | ||
|
|
007e82c533 | ||
|
|
499f8580a7 | ||
|
|
a7dc3c5dab | ||
|
|
d01d3a3c53 | ||
|
|
580e062dbf | ||
|
|
c8cee8410c | ||
|
|
6bf331c2e3 | ||
|
|
4c4930737c | ||
|
|
9de01e9525 | ||
|
|
c6a16f5974 | ||
|
|
253ef44d17 | ||
|
|
15a1f00b73 | ||
|
|
b5fa2ea8b8 | ||
|
|
449e024771 | ||
|
|
1bee7a146b | ||
|
|
270a632789 | ||
|
|
418bb05b4c | ||
|
|
052b834151 | ||
|
|
58ee204a75 | ||
|
|
0a02ee8c04 | ||
|
|
950ef4a181 | ||
|
|
7b7cdd8adb | ||
|
|
471768e760 | ||
|
|
c7517d31a4 | ||
|
|
7d10d0398e | ||
|
|
a2bc25c08b | ||
|
|
3cb49fe2d8 | ||
|
|
5b96ac122f | ||
|
|
612033f478 | ||
|
|
48ee940d8e | ||
|
|
e74df0b37d | ||
|
|
640afdc49c | ||
|
|
6b39df5b9b | ||
|
|
e7e698765e | ||
|
|
43fea13dab | ||
|
|
bc899e5bd0 | ||
|
|
160086feb9 | ||
|
|
016391c976 | ||
|
|
91746448a3 | ||
|
|
5cb0543237 | ||
|
|
fac29a24a8 | ||
|
|
4d3a2a21d0 | ||
|
|
6d4f88041c | ||
|
|
18587d3690 | ||
|
|
423090dccd | ||
|
|
78e88baab3 | ||
|
|
6a276767b3 | ||
|
|
2cb26c7c70 | ||
|
|
ff66c88060 | ||
|
|
611e82b8f9 | ||
|
|
59bdee7137 | ||
|
|
e8dbd426ae | ||
|
|
40d6e809a0 | ||
|
|
236c540d18 | ||
|
|
d6ca059f6c | ||
|
|
52c06a60ca | ||
|
|
6353644ec3 | ||
|
|
20df9ded3d | ||
|
|
7569b18a4c | ||
|
|
b9da4f4951 | ||
|
|
89b9e29257 | ||
|
|
d605de9de4 | ||
|
|
d46c94d7c3 | ||
|
|
2db9c00530 | ||
|
|
66d8d159f9 | ||
|
|
9fa1446284 | ||
|
|
b3e4cb48c7 | ||
|
|
0bca7b2247 | ||
|
|
7812e03c9d | ||
|
|
7a852ae5af | ||
|
|
706d9e61c1 | ||
|
|
8f0ed4ff4b | ||
|
|
3415b6f121 | ||
|
|
256ba6fb86 | ||
|
|
d30b2b9afe | ||
|
|
be943ca1fc | ||
|
|
1ddab2a97a | ||
|
|
e15fd4695c | ||
|
|
ffa4b1b4a1 | ||
|
|
f8eee3a2a6 | ||
|
|
eeee7a8343 | ||
|
|
8447b73fcb | ||
|
|
2863945d5f | ||
|
|
cb1f8ca6f7 | ||
|
|
1d9964bcb1 | ||
|
|
15cb8016d3 | ||
|
|
895cc0a2c5 | ||
|
|
20bf349e4e | ||
|
|
e297763da1 | ||
|
|
e471970654 | ||
|
|
12faaaced8 | ||
|
|
083cbc55cc | ||
|
|
8aa7a3273d | ||
|
|
255e2c4385 | ||
|
|
9856306870 | ||
|
|
527ab8b8a7 | ||
|
|
f8e19ba9b3 | ||
|
|
7649dbfbbc | ||
|
|
81e734644d | ||
|
|
ae55cf5b1e | ||
|
|
af539546ef | ||
|
|
0031ce57d0 | ||
|
|
2f48a2ce57 | ||
|
|
6068ab7100 | ||
|
|
29a7dccef4 | ||
|
|
e2073da86e | ||
|
|
ae079526f7 | ||
|
|
947bae8e26 | ||
|
|
a68e29dff6 | ||
|
|
a588d7f960 | ||
|
|
66224e5a32 | ||
|
|
07abad6a14 | ||
|
|
83d02aaaac | ||
|
|
5865ac017c | ||
|
|
4061a92f8e | ||
|
|
d37c31b31c | ||
|
|
973ef0078f | ||
|
|
48dcd257da | ||
|
|
da03911610 | ||
|
|
b6f7f3b73f | ||
|
|
2050d20ea7 | ||
|
|
ac1fb4a63a | ||
|
|
8c67d3c58f |
@@ -1,34 +0,0 @@
|
|||||||
// For format details, see https://aka.ms/devcontainer.json. For config options, see the
|
|
||||||
// README at: https://github.com/devcontainers/templates/tree/main/src/python
|
|
||||||
{
|
|
||||||
"name": "QChatGPT 3.10",
|
|
||||||
// Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile
|
|
||||||
"image": "mcr.microsoft.com/devcontainers/python:0-3.10",
|
|
||||||
|
|
||||||
// Features to add to the dev container. More info: https://containers.dev/features.
|
|
||||||
// "features": {},
|
|
||||||
|
|
||||||
// Use 'forwardPorts' to make a list of ports inside the container available locally.
|
|
||||||
// "forwardPorts": [],
|
|
||||||
|
|
||||||
// Use 'postCreateCommand' to run commands after the container is created.
|
|
||||||
// "postCreateCommand": "pip3 install --user -r requirements.txt",
|
|
||||||
|
|
||||||
// Configure tool-specific properties.
|
|
||||||
// "customizations": {},
|
|
||||||
"customizations": {
|
|
||||||
"codespaces": {
|
|
||||||
"repositories": {
|
|
||||||
"RockChinQ/QChatGPT": {
|
|
||||||
"permissions": "write-all"
|
|
||||||
},
|
|
||||||
"RockChinQ/revLibs": {
|
|
||||||
"permissions": "write-all"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Uncomment to connect as root instead. More info: https://aka.ms/dev-containers-non-root.
|
|
||||||
// "remoteUser": "root"
|
|
||||||
}
|
|
||||||
38
.github/ISSUE_TEMPLATE/bug-report.yml
vendored
38
.github/ISSUE_TEMPLATE/bug-report.yml
vendored
@@ -1,42 +1,30 @@
|
|||||||
name: 漏洞反馈
|
name: 漏洞反馈
|
||||||
description: 报错或漏洞请使用这个模板创建,不使用此模板创建的异常、漏洞相关issue将被直接关闭
|
description: 报错或漏洞请使用这个模板创建,不使用此模板创建的异常、漏洞相关issue将被直接关闭。由于自己操作不当/不甚了解所用技术栈引起的网络连接问题恕无法解决,请勿提 issue。容器间网络连接问题,参考文档 https://docs.langbot.app/deploy/network-details.html
|
||||||
title: "[Bug]: "
|
title: "[Bug]: "
|
||||||
labels: ["bug?"]
|
labels: ["bug?"]
|
||||||
body:
|
body:
|
||||||
- type: dropdown
|
|
||||||
attributes:
|
|
||||||
label: 部署方式
|
|
||||||
description: "主程序使用的部署方式"
|
|
||||||
options:
|
|
||||||
- 手动部署
|
|
||||||
- 安装器部署
|
|
||||||
- 一键安装包部署
|
|
||||||
- Docker部署
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
- type: input
|
- type: input
|
||||||
attributes:
|
attributes:
|
||||||
label: 系统环境
|
label: 运行环境
|
||||||
description: 操作系统、系统架构。
|
description: LangBot 版本、操作系统、系统架构、**Python版本**、**主机地理位置**
|
||||||
placeholder: 例如: CentOS x64、Windows11
|
placeholder: 例如:v3.3.0、CentOS x64 Python 3.10.3、Docker 的系统直接写 Docker 就行
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: Python环境
|
|
||||||
description: 运行程序的Python版本
|
|
||||||
placeholder: 例如: Python 3.10
|
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: textarea
|
- type: textarea
|
||||||
attributes:
|
attributes:
|
||||||
label: 异常情况
|
label: 异常情况
|
||||||
description: 完整描述异常情况,什么时候发生的、发生了什么
|
description: 完整描述异常情况,什么时候发生的、发生了什么。**请附带日志信息。**
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: textarea
|
- type: textarea
|
||||||
attributes:
|
attributes:
|
||||||
label: 报错信息
|
label: 复现步骤
|
||||||
description: 请提供完整的**控制台**报错信息(若有)
|
description: 如何重现这个问题,越详细越好;请贴上所有相关的配置文件和元数据文件(注意隐去敏感信息)
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: textarea
|
||||||
|
attributes:
|
||||||
|
label: 启用的插件
|
||||||
|
description: 有些情况可能和插件功能有关,建议提供插件启用情况。可以使用`!plugin`命令查看已启用的插件
|
||||||
validations:
|
validations:
|
||||||
required: false
|
required: false
|
||||||
|
|||||||
2
.github/ISSUE_TEMPLATE/feature-request.yml
vendored
2
.github/ISSUE_TEMPLATE/feature-request.yml
vendored
@@ -1,6 +1,6 @@
|
|||||||
name: 需求建议
|
name: 需求建议
|
||||||
title: "[Feature]: "
|
title: "[Feature]: "
|
||||||
labels: ["enhancement"]
|
labels: ["改进"]
|
||||||
description: "新功能或现有功能优化请使用这个模板;不符合类别的issue将被直接关闭"
|
description: "新功能或现有功能优化请使用这个模板;不符合类别的issue将被直接关闭"
|
||||||
body:
|
body:
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
|
|||||||
24
.github/ISSUE_TEMPLATE/submit-plugin.yml
vendored
Normal file
24
.github/ISSUE_TEMPLATE/submit-plugin.yml
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
name: 提交新插件
|
||||||
|
title: "[Plugin]: 请求登记新插件"
|
||||||
|
labels: ["独立插件"]
|
||||||
|
description: "本模板供且仅供提交新插件使用"
|
||||||
|
body:
|
||||||
|
- type: input
|
||||||
|
attributes:
|
||||||
|
label: 插件名称
|
||||||
|
description: 填写插件的名称
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: textarea
|
||||||
|
attributes:
|
||||||
|
label: 插件代码库地址
|
||||||
|
description: 仅支持 Github
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: textarea
|
||||||
|
attributes:
|
||||||
|
label: 插件简介
|
||||||
|
description: 插件的简介
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
2
.github/dependabot.yml
vendored
2
.github/dependabot.yml
vendored
@@ -10,6 +10,4 @@ updates:
|
|||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
allow:
|
allow:
|
||||||
- dependency-name: "yiri-mirai"
|
|
||||||
- dependency-name: "dulwich"
|
|
||||||
- dependency-name: "openai"
|
- dependency-name: "openai"
|
||||||
|
|||||||
27
.github/pull_request_template.md
vendored
27
.github/pull_request_template.md
vendored
@@ -2,24 +2,19 @@
|
|||||||
|
|
||||||
实现/解决/优化的内容:
|
实现/解决/优化的内容:
|
||||||
|
|
||||||
### 事务
|
## 检查清单
|
||||||
|
|
||||||
- [ ] 已阅读仓库[贡献指引](../CONTRIBUTING.md)
|
### PR 作者完成
|
||||||
- [ ] 已与维护者在issues或其他平台沟通此PR大致内容
|
|
||||||
|
|
||||||
## 以下内容可在起草PR后、合并PR前逐步完成
|
*请在方括号间写`x`以打勾
|
||||||
|
|
||||||
### 功能
|
- [ ] 阅读仓库[贡献指引](https://github.com/RockChinQ/LangBot/blob/master/CONTRIBUTING.md)了吗?
|
||||||
|
- [ ] 与项目所有者沟通过了吗?
|
||||||
|
- [ ] 我确定已自行测试所作的更改,确保功能符合预期。
|
||||||
|
|
||||||
- [ ] 已编写完善的配置文件字段说明(若有新增)
|
### 项目所有者完成
|
||||||
- [ ] 已编写面向用户的新功能说明(若有必要)
|
|
||||||
- [ ] 已测试新功能或更改
|
|
||||||
|
|
||||||
### 兼容性
|
- [ ] 相关 issues 链接了吗?
|
||||||
|
- [ ] 配置项写好了吗?迁移写好了吗?生效了吗?
|
||||||
- [ ] 已处理版本兼容性
|
- [ ] 依赖写到 requirements.txt 和 core/bootutils/deps.py 了吗
|
||||||
- [ ] 已处理插件兼容问题
|
- [ ] 文档编写了吗?
|
||||||
|
|
||||||
### 风险
|
|
||||||
|
|
||||||
可能导致或已知的问题:
|
|
||||||
29
.github/workflows/build-dev-image.yaml
vendored
Normal file
29
.github/workflows/build-dev-image.yaml
vendored
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
name: Build Dev Image
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-dev-image:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
# 如果是tag则跳过
|
||||||
|
if: ${{ !startsWith(github.ref, 'refs/tags/') }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
persist-credentials: false
|
||||||
|
|
||||||
|
- name: Generate Tag
|
||||||
|
id: generate_tag
|
||||||
|
run: |
|
||||||
|
# 获取分支名称,把/替换为-
|
||||||
|
echo ${{ github.ref }} | sed 's/refs\/heads\///g' | sed 's/\//-/g'
|
||||||
|
echo ::set-output name=tag::$(echo ${{ github.ref }} | sed 's/refs\/heads\///g' | sed 's/\//-/g')
|
||||||
|
- name: Login to Registry
|
||||||
|
run: docker login --username=${{ secrets.DOCKER_USERNAME }} --password ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
- name: Build Docker Image
|
||||||
|
run: |
|
||||||
|
docker buildx create --name mybuilder --use
|
||||||
|
docker build -t rockchin/langbot:${{ steps.generate_tag.outputs.tag }} . --push
|
||||||
45
.github/workflows/build-docker-image.yml
vendored
Normal file
45
.github/workflows/build-docker-image.yml
vendored
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
name: Build Docker Image
|
||||||
|
on:
|
||||||
|
#防止fork乱用action设置只能手动触发构建
|
||||||
|
workflow_dispatch:
|
||||||
|
## 发布release的时候会自动构建
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
jobs:
|
||||||
|
publish-docker-image:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
name: Build image
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
persist-credentials: false
|
||||||
|
|
||||||
|
- name: judge has env GITHUB_REF # 如果没有GITHUB_REF环境变量,则把github.ref变量赋值给GITHUB_REF
|
||||||
|
run: |
|
||||||
|
if [ -z "$GITHUB_REF" ]; then
|
||||||
|
export GITHUB_REF=${{ github.ref }}
|
||||||
|
echo $GITHUB_REF
|
||||||
|
fi
|
||||||
|
- name: Check version
|
||||||
|
id: check_version
|
||||||
|
run: |
|
||||||
|
echo $GITHUB_REF
|
||||||
|
# 如果是tag,则去掉refs/tags/前缀
|
||||||
|
if [[ $GITHUB_REF == refs/tags/* ]]; then
|
||||||
|
echo "It's a tag"
|
||||||
|
echo $GITHUB_REF
|
||||||
|
echo $GITHUB_REF | awk -F '/' '{print $3}'
|
||||||
|
echo ::set-output name=version::$(echo $GITHUB_REF | awk -F '/' '{print $3}')
|
||||||
|
else
|
||||||
|
echo "It's not a tag"
|
||||||
|
echo $GITHUB_REF
|
||||||
|
echo ::set-output name=version::${GITHUB_REF}
|
||||||
|
fi
|
||||||
|
- name: Login to Registry
|
||||||
|
run: docker login --username=${{ secrets.DOCKER_USERNAME }} --password ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
- name: Create Buildx
|
||||||
|
run: docker buildx create --name mybuilder --use
|
||||||
|
- name: Build # image name: rockchin/langbot:<VERSION>
|
||||||
|
run: docker buildx build --platform linux/arm64,linux/amd64 -t rockchin/langbot:${{ steps.check_version.outputs.version }} -t rockchin/langbot:latest . --push
|
||||||
62
.github/workflows/build-release-artifacts.yaml
vendored
Normal file
62
.github/workflows/build-release-artifacts.yaml
vendored
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
name: Build Release Artifacts
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
## 发布release的时候会自动构建
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-artifacts:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
persist-credentials: false
|
||||||
|
|
||||||
|
- name: Check version
|
||||||
|
id: check_version
|
||||||
|
run: |
|
||||||
|
echo $GITHUB_REF
|
||||||
|
# 如果是tag,则去掉refs/tags/前缀
|
||||||
|
if [[ $GITHUB_REF == refs/tags/* ]]; then
|
||||||
|
echo "It's a tag"
|
||||||
|
echo $GITHUB_REF
|
||||||
|
echo $GITHUB_REF | awk -F '/' '{print $3}'
|
||||||
|
echo ::set-output name=version::$(echo $GITHUB_REF | awk -F '/' '{print $3}')
|
||||||
|
else
|
||||||
|
echo "It's not a tag"
|
||||||
|
echo $GITHUB_REF
|
||||||
|
echo ::set-output name=version::${GITHUB_REF}
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Make Temp Directory
|
||||||
|
run: |
|
||||||
|
mkdir -p /tmp/langbot_build_web
|
||||||
|
cp -r . /tmp/langbot_build_web
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '22'
|
||||||
|
- name: Build Web
|
||||||
|
run: |
|
||||||
|
cd /tmp/langbot_build_web/web
|
||||||
|
npm install
|
||||||
|
npm run build
|
||||||
|
- name: Package Output
|
||||||
|
run: |
|
||||||
|
cp -r /tmp/langbot_build_web/web/dist ./web
|
||||||
|
- name: Upload Artifact
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: langbot-${{ steps.check_version.outputs.version }}-all
|
||||||
|
path: .
|
||||||
|
|
||||||
|
- name: Upload To Release
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ secrets.RELEASE_UPLOAD_GITHUB_TOKEN }}
|
||||||
|
run: |
|
||||||
|
# 本目录下所有文件打包成zip
|
||||||
|
zip -r langbot-${{ steps.check_version.outputs.version }}-all.zip .
|
||||||
|
gh release upload ${{ github.event.release.tag_name }} langbot-${{ steps.check_version.outputs.version }}-all.zip
|
||||||
31
.github/workflows/sync-wiki.yml
vendored
31
.github/workflows/sync-wiki.yml
vendored
@@ -1,31 +0,0 @@
|
|||||||
name: Update Wiki
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'res/wiki/**'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-wiki:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
- name: Setup Git
|
|
||||||
run: |
|
|
||||||
git config --global user.name "GitHub Actions"
|
|
||||||
git config --global user.email "github-actions[bot]@users.noreply.github.com"
|
|
||||||
- name: Clone Wiki Repository
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
with:
|
|
||||||
repository: RockChinQ/QChatGPT.wiki
|
|
||||||
path: wiki
|
|
||||||
- name: Copy res/wiki content to wiki
|
|
||||||
run: |
|
|
||||||
cp -r res/wiki/* wiki/
|
|
||||||
- name: Commit and Push Changes
|
|
||||||
run: |
|
|
||||||
cd wiki
|
|
||||||
git add .
|
|
||||||
git commit -m "Update wiki"
|
|
||||||
git push
|
|
||||||
58
.github/workflows/update-cmdpriv-template.yml
vendored
58
.github/workflows/update-cmdpriv-template.yml
vendored
@@ -1,58 +0,0 @@
|
|||||||
name: Update cmdpriv-template
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'pkg/qqbot/cmds/**'
|
|
||||||
pull_request:
|
|
||||||
types: [closed]
|
|
||||||
paths:
|
|
||||||
- 'pkg/qqbot/cmds/**'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-cmdpriv-template:
|
|
||||||
if: github.event.pull_request.merged == true || github.event_name == 'push'
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Set up Python
|
|
||||||
uses: actions/setup-python@v2
|
|
||||||
with:
|
|
||||||
python-version: 3.x
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip
|
|
||||||
python -m pip install --upgrade yiri-mirai openai colorlog func_timeout dulwich Pillow
|
|
||||||
|
|
||||||
- name: Copy Scripts
|
|
||||||
run: |
|
|
||||||
cp res/scripts/generate_cmdpriv_template.py .
|
|
||||||
|
|
||||||
- name: Generate Files
|
|
||||||
run: |
|
|
||||||
python main.py
|
|
||||||
|
|
||||||
- name: Run generate_cmdpriv_template.py
|
|
||||||
run: python3 generate_cmdpriv_template.py
|
|
||||||
|
|
||||||
- name: Check for changes in cmdpriv-template.json
|
|
||||||
id: check_changes
|
|
||||||
run: |
|
|
||||||
if git diff --name-only | grep -q "cmdpriv-template.json"; then
|
|
||||||
echo "::set-output name=changes_detected::true"
|
|
||||||
else
|
|
||||||
echo "::set-output name=changes_detected::false"
|
|
||||||
fi
|
|
||||||
|
|
||||||
- name: Commit changes to cmdpriv-template.json
|
|
||||||
if: steps.check_changes.outputs.changes_detected == 'true'
|
|
||||||
run: |
|
|
||||||
git config --global user.name "GitHub Actions Bot"
|
|
||||||
git config --global user.email "<github-actions@github.com>"
|
|
||||||
git add cmdpriv-template.json
|
|
||||||
git commit -m "Update cmdpriv-template.json"
|
|
||||||
git push
|
|
||||||
53
.github/workflows/update-override-all.yml
vendored
53
.github/workflows/update-override-all.yml
vendored
@@ -1,53 +0,0 @@
|
|||||||
name: Check and Update override_all
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'config-template.py'
|
|
||||||
pull_request:
|
|
||||||
types:
|
|
||||||
- closed
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
paths:
|
|
||||||
- 'config-template.py'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-override-all:
|
|
||||||
name: check and update
|
|
||||||
if: github.event.pull_request.merged == true || github.event_name == 'push'
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Set up Python
|
|
||||||
uses: actions/setup-python@v2
|
|
||||||
with:
|
|
||||||
python-version: 3.x
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip
|
|
||||||
# 在此处添加您的项目所需的其他依赖
|
|
||||||
|
|
||||||
- name: Copy Scripts
|
|
||||||
run: |
|
|
||||||
cp res/scripts/generate_override_all.py .
|
|
||||||
|
|
||||||
- name: Run generate_override_all.py
|
|
||||||
run: python3 generate_override_all.py
|
|
||||||
|
|
||||||
- name: Check for changes in override-all.json
|
|
||||||
id: check_changes
|
|
||||||
run: |
|
|
||||||
git diff --exit-code override-all.json || echo "::set-output name=changes_detected::true"
|
|
||||||
|
|
||||||
- name: Commit and push changes
|
|
||||||
if: steps.check_changes.outputs.changes_detected == 'true'
|
|
||||||
run: |
|
|
||||||
git config --global user.email "github-actions[bot]@users.noreply.github.com"
|
|
||||||
git config --global user.name "GitHub Actions"
|
|
||||||
git add override-all.json
|
|
||||||
git commit -m "Update override-all.json"
|
|
||||||
git push
|
|
||||||
32
.gitignore
vendored
32
.gitignore
vendored
@@ -1,11 +1,11 @@
|
|||||||
config.py
|
/config.py
|
||||||
.idea/
|
.idea/
|
||||||
__pycache__/
|
__pycache__/
|
||||||
database.db
|
database.db
|
||||||
qchatgpt.log
|
langbot.log
|
||||||
/banlist.py
|
/banlist.py
|
||||||
plugins/
|
/plugins/
|
||||||
!plugins/__init__.py
|
!/plugins/__init__.py
|
||||||
/revcfg.py
|
/revcfg.py
|
||||||
prompts/
|
prompts/
|
||||||
logs/
|
logs/
|
||||||
@@ -16,5 +16,27 @@ scenario/
|
|||||||
!scenario/default-template.json
|
!scenario/default-template.json
|
||||||
override.json
|
override.json
|
||||||
cookies.json
|
cookies.json
|
||||||
res/announcement_saved
|
data/labels/announcement_saved.json
|
||||||
cmdpriv.json
|
cmdpriv.json
|
||||||
|
tips.py
|
||||||
|
venv*
|
||||||
|
bin/
|
||||||
|
.vscode
|
||||||
|
test_*
|
||||||
|
venv/
|
||||||
|
hugchat.json
|
||||||
|
qcapi
|
||||||
|
claude.json
|
||||||
|
bard.json
|
||||||
|
/*yaml
|
||||||
|
!components.yaml
|
||||||
|
!/docker-compose.yaml
|
||||||
|
data/labels/instance_id.json
|
||||||
|
.DS_Store
|
||||||
|
/data
|
||||||
|
botpy.log*
|
||||||
|
/poc
|
||||||
|
/libs/wecom_api/test.py
|
||||||
|
/venv
|
||||||
|
/jp-tyo-churros-05.rockchin.top
|
||||||
|
test.py
|
||||||
@@ -17,3 +17,10 @@
|
|||||||
- 解决本项目或衍生项目的issues中亟待解决的问题
|
- 解决本项目或衍生项目的issues中亟待解决的问题
|
||||||
- 阅读并完善本项目文档
|
- 阅读并完善本项目文档
|
||||||
- 在各个社交媒体撰写本项目教程等
|
- 在各个社交媒体撰写本项目教程等
|
||||||
|
|
||||||
|
### 代码规范
|
||||||
|
|
||||||
|
- 代码中的注解`务必`符合Google风格的规范
|
||||||
|
- 模块顶部的引入代码请遵循`系统模块`、`第三方库模块`、`自定义模块`的顺序进行引入
|
||||||
|
- `不要`直接引入模块的特定属性,而是引入这个模块,再通过`xxx.yyy`的形式使用属性
|
||||||
|
- 任何作用域的字段`必须`先声明后使用,并在声明处注明类型提示
|
||||||
|
|||||||
29
Dockerfile
29
Dockerfile
@@ -1,17 +1,22 @@
|
|||||||
FROM python:3.9-slim
|
FROM node:22-alpine AS node
|
||||||
WORKDIR /QChatGPT
|
|
||||||
|
|
||||||
RUN sed -i "s/deb.debian.org/mirrors.tencent.com/g" /etc/apt/sources.list \
|
WORKDIR /app
|
||||||
&& sed -i 's|security.debian.org/debian-security|mirrors.tencent.com/debian-security|g' /etc/apt/sources.list \
|
|
||||||
&& apt-get clean \
|
|
||||||
&& apt-get update \
|
|
||||||
&& apt-get -y upgrade \
|
|
||||||
&& apt-get install -y git \
|
|
||||||
&& apt-get clean \
|
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
COPY . /QChatGPT/
|
COPY web ./web
|
||||||
|
|
||||||
RUN pip install -r requirements.txt -i https://pypi.tuna.tsinghua.edu.cn/simple
|
RUN cd web && npm install && npm run build
|
||||||
|
|
||||||
|
FROM python:3.10.13-slim
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
COPY --from=node /app/web/dist ./web/dist
|
||||||
|
|
||||||
|
RUN apt update \
|
||||||
|
&& apt install gcc -y \
|
||||||
|
&& python -m pip install -r requirements.txt \
|
||||||
|
&& touch /.dockerenv
|
||||||
|
|
||||||
CMD [ "python", "main.py" ]
|
CMD [ "python", "main.py" ]
|
||||||
330
README.md
330
README.md
@@ -1,268 +1,150 @@
|
|||||||
# QChatGPT🤖
|
|
||||||
|
|
||||||
> 2023/3/18 现已支持GPT-4 API(内测),请查看`config-template.py`中的`completion_api_params`
|
<p align="center">
|
||||||
> 2023/3/15 逆向库已支持New Bing,使用方法查看[插件文档](https://github.com/RockChinQ/revLibs)
|
<a href="https://langbot.app">
|
||||||
|
<img src="https://docs.langbot.app/social.png" alt="LangBot"/>
|
||||||
|
</a>
|
||||||
|
|
||||||
- **客官,来都来了,不点个⭐吗?**
|
<div align="center">
|
||||||
- 到[项目Wiki](https://github.com/RockChinQ/QChatGPT/wiki)可了解项目详细信息
|
|
||||||
- 官方交流、答疑群: 656285629
|
|
||||||
- **进群提问前请您`确保`已经找遍文档和issue均无法解决**
|
|
||||||
- 社区群(内有一键部署包、图形化界面等资源): 362515018
|
|
||||||
- QQ频道机器人见[QQChannelChatGPT](https://github.com/Soulter/QQChannelChatGPT)
|
|
||||||
- 欢迎各种形式的贡献,请查看[贡献指引](CONTRIBUTING.md)
|
|
||||||
|
|
||||||
## 🍺模型适配一览
|
<a href="https://trendshift.io/repositories/12901" target="_blank"><img src="https://trendshift.io/api/badge/repositories/12901" alt="RockChinQ%2FLangBot | Trendshift" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
||||||
|
|
||||||
<details>
|
<a href="https://docs.langbot.app">项目主页</a> |
|
||||||
<summary>点击此处展开</summary>
|
<a href="https://docs.langbot.app/insight/intro.html">功能介绍</a> |
|
||||||
|
<a href="https://docs.langbot.app/insight/guide.html">部署文档</a> |
|
||||||
|
<a href="https://docs.langbot.app/usage/faq.html">常见问题</a> |
|
||||||
|
<a href="https://docs.langbot.app/plugin/plugin-intro.html">插件介绍</a> |
|
||||||
|
<a href="https://github.com/RockChinQ/LangBot/issues/new?assignees=&labels=%E7%8B%AC%E7%AB%8B%E6%8F%92%E4%BB%B6&projects=&template=submit-plugin.yml&title=%5BPlugin%5D%3A+%E8%AF%B7%E6%B1%82%E7%99%BB%E8%AE%B0%E6%96%B0%E6%8F%92%E4%BB%B6">提交插件</a>
|
||||||
|
|
||||||
### 文字对话
|
<div align="center">
|
||||||
|
😎高稳定、🧩支持扩展、🦄多模态 - 大模型原生即时通信机器人平台🤖
|
||||||
|
</div>
|
||||||
|
|
||||||
- OpenAI GPT-3.5模型(ChatGPT API), 本项目原生支持, 默认使用
|
<br/>
|
||||||
- OpenAI GPT-3模型, 本项目原生支持, 部署完成后前往`config.py`切换
|
|
||||||
- OpenAI GPT-4模型, 本项目原生支持, 目前需要您的账户通过OpenAI的内测申请, 请前往`config.py`切换
|
|
||||||
- ChatGPT网页版GPT-3.5模型, 由[插件](https://github.com/RockChinQ/revLibs)接入
|
|
||||||
- ChatGPT网页版GPT-4模型, 目前需要ChatGPT Plus订阅, 由[插件](https://github.com/RockChinQ/revLibs)接入
|
|
||||||
- New Bing逆向库, 由[插件](https://github.com/RockChinQ/revLibs)接入
|
|
||||||
|
|
||||||
### 故事续写
|
|
||||||
|
|
||||||
- NovelAI API, 由[插件](https://github.com/dominoar/QCPNovelAi)接入
|
[](https://discord.gg/wdNEHETs87)
|
||||||
|
[](https://qm.qq.com/q/JLi38whHum)
|
||||||
|
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
||||||
|

|
||||||
|
<img src="https://img.shields.io/badge/python-3.10 ~ 3.13 -blue.svg" alt="python">
|
||||||
|
[](https://gitcode.com/RockChinQ/LangBot)
|
||||||
|
|
||||||
### 图片绘制
|
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
||||||
|
|
||||||
- OpenAI DALL·E模型, 本项目原生支持, 使用方法查看[Wiki功能使用页](https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E5%8A%9F%E8%83%BD%E7%82%B9%E5%88%97%E4%B8%BE)
|
</div>
|
||||||
- NovelAI API, 由[插件](https://github.com/dominoar/QCPNovelAi)接入
|
|
||||||
|
|
||||||
### 语音生成
|
</p>
|
||||||
|
|
||||||
- TTS+VITS, 由[插件](https://github.com/dominoar/QChatPlugins)接入
|
## ✨ 特性
|
||||||
- Plachta/VITS-Umamusume-voice-synthesizer, 由[插件](https://github.com/oliverkirk-sudo/chat_voice)接入
|
|
||||||
|
|
||||||
|
- 💬 大模型对话、Agent:支持多种大模型,适配群聊和私聊;具有多轮对话、工具调用、多模态能力,并深度适配 [Dify](https://dify.ai)。目前支持 QQ、QQ频道、企业微信、个人微信、飞书、Discord、Telegram 等平台。
|
||||||
|
- 🛠️ 高稳定性、功能完备:原生支持访问控制、限速、敏感词过滤等机制;配置简单,支持多种部署方式。
|
||||||
|
- 🧩 插件扩展、活跃社区:支持事件驱动、组件扩展等插件机制;适配 Anthropic [MCP 协议](https://modelcontextprotocol.io/);目前已有数十个[插件](https://docs.langbot.app/plugin/plugin-intro.html)
|
||||||
|
- 😻 [New] Web 管理面板:支持通过浏览器管理 LangBot 实例,具体支持功能,查看[文档](https://docs.langbot.app/webui/intro.html)
|
||||||
|
|
||||||
</details>
|
## 📦 开始使用
|
||||||
|
|
||||||
安装[此插件](https://github.com/RockChinQ/Switcher),即可在使用中切换文字模型。
|
> [!IMPORTANT]
|
||||||
|
>
|
||||||
|
> 在您开始任何方式部署之前,请务必阅读[新手指引](https://docs.langbot.app/insight/guide.html)。
|
||||||
|
|
||||||
## ✅功能
|
#### Docker Compose 部署
|
||||||
|
|
||||||
<details>
|
适合熟悉 Docker 的用户,查看文档[Docker 部署](https://docs.langbot.app/deploy/langbot/docker.html)。
|
||||||
<summary>点击此处展开概述</summary>
|
|
||||||
|
|
||||||
<details>
|
#### 宝塔面板部署
|
||||||
<summary>✅支持敏感词过滤,避免账号风险</summary>
|
|
||||||
|
|
||||||
- 难以监测机器人与用户对话时的内容,故引入此功能以减少机器人风险
|
已上架宝塔面板,若您已安装宝塔面板,可以根据[文档](https://docs.langbot.app/deploy/langbot/one-click/bt.html)使用。
|
||||||
- 加入了百度云内容审核,在`config.py`中修改`baidu_check`的值,并填写`baidu_api_key`和`baidu_secret_key`以开启此功能
|
|
||||||
- 编辑`sensitive.json`,并在`config.py`中修改`sensitive_word_filter`的值以开启此功能
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<details>
|
#### Zeabur 云部署
|
||||||
<summary>✅群内多种响应规则,不必at</summary>
|
|
||||||
|
|
||||||
- 默认回复`ai`作为前缀或`@`机器人的消息
|
社区贡献的 Zeabur 模板。
|
||||||
- 详细见`config.py`中的`response_rules`字段
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<details>
|
[](https://zeabur.com/zh-CN/templates/ZKTBDH)
|
||||||
<summary>✅完善的多api-key管理,超额自动切换</summary>
|
|
||||||
|
|
||||||
- 支持配置多个`api-key`,内部统计使用量并在超额时自动切换
|
#### Railway 云部署
|
||||||
- 请在`config.py`中修改`openai_config`的值以设置`api-key`
|
|
||||||
- 可以在`config.py`中修改`api_key_fee_threshold`来自定义切换阈值
|
|
||||||
- 运行期间向机器人说`!usage`以查看当前使用情况
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<details>
|
[](https://railway.app/template/yRrAyL?referralCode=vogKPF)
|
||||||
<summary>✅支持预设指令文字</summary>
|
|
||||||
|
|
||||||
- 支持以自然语言预设文字,自定义机器人人格等信息
|
#### 手动部署
|
||||||
- 详见`config.py`中的`default_prompt`部分
|
|
||||||
- 支持设置多个预设情景,并通过!reset、!default等指令控制,详细请查看[wiki指令](https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E6%9C%BA%E5%99%A8%E4%BA%BA%E6%8C%87%E4%BB%A4)
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<details>
|
直接使用发行版运行,查看文档[手动部署](https://docs.langbot.app/deploy/langbot/manual.html)。
|
||||||
<summary>✅支持对话、绘图等模型,可玩性更高</summary>
|
|
||||||
|
|
||||||
- 现已支持OpenAI的对话`Completion API`和绘图`Image API`
|
## 📸 效果展示
|
||||||
- 向机器人发送指令`!draw <prompt>`即可使用绘图模型
|
|
||||||
</details>
|
|
||||||
<details>
|
|
||||||
<summary>✅支持指令控制热重载、热更新</summary>
|
|
||||||
|
|
||||||
- 允许在运行期间修改`config.py`或其他代码后,以管理员账号向机器人发送指令`!reload`进行热重载,无需重启
|
<img alt="回复效果(带有联网插件)" src="https://docs.langbot.app/QChatGPT-0516.png" width="500px"/>
|
||||||
- 运行期间允许以管理员账号向机器人发送指令`!update`进行热更新,拉取远程最新代码并执行热重载
|
|
||||||
</details>
|
|
||||||
<details>
|
|
||||||
<summary>✅支持插件加载🧩</summary>
|
|
||||||
|
|
||||||
- 自行实现插件加载器及相关支持
|
- WebUI Demo: https://demo.langbot.dev/
|
||||||
- 详细查看[插件使用页](https://github.com/RockChinQ/QChatGPT/wiki/%E6%8F%92%E4%BB%B6%E4%BD%BF%E7%94%A8)
|
- 登录信息:邮箱:`demo@langbot.app` 密码:`langbot123456`
|
||||||
</details>
|
- 注意:仅展示webui效果,公开环境,请不要在其中填入您的任何敏感信息。
|
||||||
<details>
|
|
||||||
<summary>✅私聊、群聊黑名单机制</summary>
|
|
||||||
|
|
||||||
- 支持将人或群聊加入黑名单以忽略其消息
|
## 🔌 组件兼容性
|
||||||
- 详见Wiki`加入黑名单`节
|
|
||||||
</details>
|
|
||||||
<details>
|
|
||||||
<summary>✅长消息处理策略</summary>
|
|
||||||
|
|
||||||
- 支持将长消息转换成图片或消息记录组件,避免消息刷屏
|
### 消息平台
|
||||||
- 请查看`config.py`中`blob_message_strategy`等字段
|
|
||||||
</details>
|
|
||||||
<details>
|
|
||||||
<summary>✅回复速度限制</summary>
|
|
||||||
|
|
||||||
- 支持限制单会话内每分钟可进行的对话次数
|
| 平台 | 状态 | 备注 |
|
||||||
- 具有“等待”和“丢弃”两种策略
|
| --- | --- | --- |
|
||||||
- “等待”策略:在获取到回复后,等待直到此次响应时间达到对话响应时间均值
|
| QQ 个人号 | ✅ | QQ 个人号私聊、群聊 |
|
||||||
- “丢弃”策略:此分钟内对话次数达到限制时,丢弃之后的对话
|
| QQ 官方机器人 | ✅ | QQ 官方机器人,支持频道、私聊、群聊 |
|
||||||
- 详细请查看config.py中的相关配置
|
| 企业微信 | ✅ | |
|
||||||
</details>
|
| 企微对外客服 | ✅ | |
|
||||||
<details>
|
| 个人微信 | ✅ | 使用 [Gewechat](https://github.com/Devo919/Gewechat) 接入 |
|
||||||
<summary>✅支持使用网络代理</summary>
|
| 微信公众号 | ✅ | |
|
||||||
|
| 飞书 | ✅ | |
|
||||||
|
| 钉钉 | ✅ | |
|
||||||
|
| Discord | ✅ | |
|
||||||
|
| Telegram | ✅ | |
|
||||||
|
| Slack | ✅ | |
|
||||||
|
| LINE | 🚧 | |
|
||||||
|
| WhatsApp | 🚧 | |
|
||||||
|
|
||||||
- 目前已支持正向代理访问接口
|
🚧: 正在开发中
|
||||||
- 详细请查看config.py中的`openai_config`的说明
|
|
||||||
</details>
|
|
||||||
</details>
|
|
||||||
|
|
||||||
详情请查看[Wiki功能使用页](https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E5%8A%9F%E8%83%BD%E7%82%B9%E5%88%97%E4%B8%BE)
|
### 大模型能力
|
||||||
|
|
||||||
## 🔩部署
|
| 模型 | 状态 | 备注 |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| [OpenAI](https://platform.openai.com/) | ✅ | 可接入任何 OpenAI 接口格式模型 |
|
||||||
|
| [DeepSeek](https://www.deepseek.com/) | ✅ | |
|
||||||
|
| [Moonshot](https://www.moonshot.cn/) | ✅ | |
|
||||||
|
| [Anthropic](https://www.anthropic.com/) | ✅ | |
|
||||||
|
| [xAI](https://x.ai/) | ✅ | |
|
||||||
|
| [智谱AI](https://open.bigmodel.cn/) | ✅ | |
|
||||||
|
| [Dify](https://dify.ai) | ✅ | LLMOps 平台 |
|
||||||
|
| [Ollama](https://ollama.com/) | ✅ | 本地大模型运行平台 |
|
||||||
|
| [LMStudio](https://lmstudio.ai/) | ✅ | 本地大模型运行平台 |
|
||||||
|
| [GiteeAI](https://ai.gitee.com/) | ✅ | 大模型接口聚合平台 |
|
||||||
|
| [SiliconFlow](https://siliconflow.cn/) | ✅ | 大模型聚合平台 |
|
||||||
|
| [阿里云百炼](https://bailian.console.aliyun.com/) | ✅ | 大模型聚合平台, LLMOps 平台 |
|
||||||
|
| [火山方舟](https://console.volcengine.com/ark/region:ark+cn-beijing/model?vendor=Bytedance&view=LIST_VIEW) | ✅ | 大模型聚合平台, LLMOps 平台 |
|
||||||
|
| [ModelScope](https://modelscope.cn/docs/model-service/API-Inference/intro) | ✅ | 大模型聚合平台 |
|
||||||
|
| [MCP](https://modelcontextprotocol.io/) | ✅ | 支持通过 MCP 协议获取工具 |
|
||||||
|
|
||||||
**部署过程中遇到任何问题,请先在[QChatGPT](https://github.com/RockChinQ/QChatGPT/issues)或[qcg-installer](https://github.com/RockChinQ/qcg-installer/issues)的issue里进行搜索**
|
### TTS
|
||||||
|
|
||||||
### - 注册OpenAI账号
|
| 平台/模型 | 备注 |
|
||||||
|
| --- | --- |
|
||||||
|
| [FishAudio](https://fish.audio/zh-CN/discovery/) | [插件](https://github.com/the-lazy-me/NewChatVoice) |
|
||||||
|
| [海豚 AI](https://www.ttson.cn/?source=thelazy) | [插件](https://github.com/the-lazy-me/NewChatVoice) |
|
||||||
|
| [AzureTTS](https://portal.azure.com/) | [插件](https://github.com/Ingnaryk/LangBot_AzureTTS) |
|
||||||
|
|
||||||
> 若您要直接使用非OpenAI的模型(如New Bing),可跳过此步骤,直接进行之后的部署,完成后按照相关插件的文档进行配置即可
|
### 文生图
|
||||||
|
|
||||||
参考以下文章自行注册
|
| 平台/模型 | 备注 |
|
||||||
|
| --- | --- |
|
||||||
|
| 阿里云百炼 | [插件](https://github.com/Thetail001/LangBot_BailianTextToImagePlugin)
|
||||||
|
|
||||||
> [国内注册ChatGPT的方法(100%可用)](https://www.pythonthree.com/register-openai-chatgpt/)
|
## 😘 社区贡献
|
||||||
> [手把手教你如何注册ChatGPT,超级详细](https://guxiaobei.com/51461)
|
|
||||||
|
|
||||||
注册成功后请前往[个人中心查看](https://beta.openai.com/account/api-keys)api_key
|
感谢以下[代码贡献者](https://github.com/RockChinQ/LangBot/graphs/contributors)和社区里其他成员对 LangBot 的贡献:
|
||||||
完成注册后,使用以下自动化或手动部署步骤
|
|
||||||
|
|
||||||
### - 自动化部署
|
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
||||||
|
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
||||||
|
</a>
|
||||||
|
|
||||||
<details>
|
以及 LangBot 核心团队成员:
|
||||||
<summary>展开查看,以下方式二选一,Linux首选Docker,Windows首选安装器</summary>
|
|
||||||
|
|
||||||
#### Docker方式
|
- [RockChinQ](https://github.com/RockChinQ)
|
||||||
|
- [the-lazy-me](https://github.com/the-lazy-me)
|
||||||
请查看[此文档](res/docs/docker_deploy.md)
|
- [wangcham](https://github.com/wangcham)
|
||||||
由[@mikumifa](https://github.com/mikumifa)贡献
|
- [KaedeSAMA](https://github.com/KaedeSAMA)
|
||||||
|
|
||||||
#### 安装器方式
|
|
||||||
|
|
||||||
使用[此安装器](https://github.com/RockChinQ/qcg-installer)(若无法访问请到[Gitee](https://gitee.com/RockChin/qcg-installer))进行部署
|
|
||||||
|
|
||||||
- 安装器目前仅支持部分平台,请到仓库文档查看,其他平台请手动部署
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
### - 手动部署
|
|
||||||
<details>
|
|
||||||
<summary>手动部署适用于所有平台</summary>
|
|
||||||
|
|
||||||
- 请使用Python 3.9.x以上版本
|
|
||||||
|
|
||||||
#### 配置Mirai
|
|
||||||
|
|
||||||
按照[此教程](https://yiri-mirai.wybxc.cc/tutorials/01/configuration)配置Mirai及YiriMirai
|
|
||||||
启动mirai-console后,使用`login`命令登录QQ账号,保持mirai-console运行状态
|
|
||||||
|
|
||||||
#### 配置主程序
|
|
||||||
|
|
||||||
1. 克隆此项目
|
|
||||||
|
|
||||||
```bash
|
|
||||||
git clone https://github.com/RockChinQ/QChatGPT
|
|
||||||
cd QChatGPT
|
|
||||||
```
|
|
||||||
|
|
||||||
2. 安装依赖
|
|
||||||
|
|
||||||
```bash
|
|
||||||
pip3 install requests yiri-mirai openai colorlog func_timeout dulwich Pillow
|
|
||||||
```
|
|
||||||
|
|
||||||
3. 运行一次主程序,生成配置文件
|
|
||||||
|
|
||||||
```bash
|
|
||||||
python3 main.py
|
|
||||||
```
|
|
||||||
|
|
||||||
4. 编辑配置文件`config.py`
|
|
||||||
|
|
||||||
按照文件内注释填写配置信息
|
|
||||||
|
|
||||||
5. 运行主程序
|
|
||||||
|
|
||||||
```bash
|
|
||||||
python3 main.py
|
|
||||||
```
|
|
||||||
|
|
||||||
无报错信息即为运行成功
|
|
||||||
|
|
||||||
**常见问题**
|
|
||||||
|
|
||||||
- mirai登录提示`QQ版本过低`,见[此issue](https://github.com/RockChinQ/QChatGPT/issues/137)
|
|
||||||
- 如提示安装`uvicorn`或`hypercorn`请*不要*安装,这两个不是必需的,目前存在未知原因bug
|
|
||||||
- 如报错`TypeError: As of 3.10, the *loop* parameter was removed from Lock() since it is no longer necessary`, 请参考 [此处](https://github.com/RockChinQ/QChatGPT/issues/5)
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
## 🚀使用
|
|
||||||
|
|
||||||
**部署完成后必看: [指令说明](https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E6%9C%BA%E5%99%A8%E4%BA%BA%E6%8C%87%E4%BB%A4)**
|
|
||||||
所有功能查看[Wiki功能使用页](https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E4%BD%BF%E7%94%A8%E6%96%B9%E5%BC%8F)
|
|
||||||
|
|
||||||
## 🧩插件生态
|
|
||||||
|
|
||||||
现已支持自行开发插件对功能进行扩展或自定义程序行为
|
|
||||||
详见[Wiki插件使用页](https://github.com/RockChinQ/QChatGPT/wiki/%E6%8F%92%E4%BB%B6%E4%BD%BF%E7%94%A8)
|
|
||||||
开发教程见[Wiki插件开发页](https://github.com/RockChinQ/QChatGPT/wiki/%E6%8F%92%E4%BB%B6%E5%BC%80%E5%8F%91)
|
|
||||||
|
|
||||||
<details>
|
|
||||||
<summary>查看插件列表</summary>
|
|
||||||
|
|
||||||
### 示例插件
|
|
||||||
|
|
||||||
在`tests/plugin_examples`目录下,将其整个目录复制到`plugins`目录下即可使用
|
|
||||||
|
|
||||||
- `cmdcn` - 主程序指令中文形式
|
|
||||||
- `hello_plugin` - 在收到消息`hello`时回复相应消息
|
|
||||||
- `urlikethisijustsix` - 收到冒犯性消息时回复相应消息
|
|
||||||
|
|
||||||
### 更多
|
|
||||||
|
|
||||||
欢迎提交新的插件
|
|
||||||
|
|
||||||
- [revLibs](https://github.com/RockChinQ/revLibs) - 将ChatGPT网页版接入此项目,关于[官方接口和网页版有什么区别](https://github.com/RockChinQ/QChatGPT/wiki/%E5%AE%98%E6%96%B9%E6%8E%A5%E5%8F%A3%E4%B8%8EChatGPT%E7%BD%91%E9%A1%B5%E7%89%88)
|
|
||||||
- [Switcher](https://github.com/RockChinQ/Switcher) - 支持通过指令切换使用的模型
|
|
||||||
- [hello_plugin](https://github.com/RockChinQ/hello_plugin) - `hello_plugin` 的储存库形式,插件开发模板
|
|
||||||
- [dominoar/QChatPlugins](https://github.com/dominoar/QchatPlugins) - dominoar编写的诸多新功能插件(语音输出、Ranimg、屏蔽词规则等)
|
|
||||||
- [dominoar/QCP-NovelAi](https://github.com/dominoar/QCP-NovelAi) - NovelAI 故事叙述与绘画
|
|
||||||
- [oliverkirk-sudo/chat_voice](https://github.com/oliverkirk-sudo/chat_voice) - 文字转语音输出,使用HuggingFace上的[VITS-Umamusume-voice-synthesizer模型](https://huggingface.co/spaces/Plachta/VITS-Umamusume-voice-synthesizer)
|
|
||||||
- [RockChinQ/WaitYiYan](https://github.com/RockChinQ/WaitYiYan) - 实时获取百度`文心一言`等待列表人数
|
|
||||||
- [chordfish-k/QChartGPT_Emoticon_Plugin](https://github.com/chordfish-k/QChartGPT_Emoticon_Plugin) - 使机器人根据回复内容发送表情包
|
|
||||||
- [oliverkirk-sudo/ChatPoeBot](https://github.com/oliverkirk-sudo/ChatPoeBot) - 接入[Poe](https://poe.com/)上的机器人
|
|
||||||
</details>
|
|
||||||
|
|
||||||
## 😘致谢
|
|
||||||
|
|
||||||
- [@the-lazy-me](https://github.com/the-lazy-me) 为本项目制作[视频教程](https://www.bilibili.com/video/BV15v4y1X7aP)
|
|
||||||
- [@mikumifa](https://github.com/mikumifa) 本项目Docker部署仓库开发者
|
|
||||||
- [@dominoar](https://github.com/dominoar) 为本项目开发多种插件
|
|
||||||
- [@万神的星空](https://github.com/qq255204159) 整合包发行
|
|
||||||
- [@ljcduo](https://github.com/ljcduo) GPT-4 API内测账号提供
|
|
||||||
|
|
||||||
以及所有[贡献者](https://github.com/RockChinQ/QChatGPT/graphs/contributors)和其他为本项目提供支持的朋友们。
|
|
||||||
|
|
||||||
<!-- ## 👍赞赏
|
|
||||||
|
|
||||||
<img alt="赞赏码" src="res/mm_reward_qrcode_1672840549070.png" width="400" height="400"/> -->
|
|
||||||
133
README_EN.md
Normal file
133
README_EN.md
Normal file
@@ -0,0 +1,133 @@
|
|||||||
|
<p align="center">
|
||||||
|
<a href="https://langbot.app">
|
||||||
|
<img src="https://docs.langbot.app/social.png" alt="LangBot"/>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
|
||||||
|
<a href="https://trendshift.io/repositories/12901" target="_blank"><img src="https://trendshift.io/api/badge/repositories/12901" alt="RockChinQ%2FLangBot | Trendshift" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
||||||
|
|
||||||
|
<a href="https://docs.langbot.app">Home</a> |
|
||||||
|
<a href="https://docs.langbot.app/insight/intro.html">Features</a> |
|
||||||
|
<a href="https://docs.langbot.app/insight/guide.html">Deployment</a> |
|
||||||
|
<a href="https://docs.langbot.app/usage/faq.html">FAQ</a> |
|
||||||
|
<a href="https://docs.langbot.app/plugin/plugin-intro.html">Plugin</a> |
|
||||||
|
<a href="https://github.com/RockChinQ/LangBot/issues/new?assignees=&labels=%E7%8B%AC%E7%AB%8B%E6%8F%92%E4%BB%B6&projects=&template=submit-plugin.yml&title=%5BPlugin%5D%3A+%E8%AF%B7%E6%B1%82%E7%99%BB%E8%AE%B0%E6%96%B0%E6%8F%92%E4%BB%B6">Submit Plugin</a>
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
😎High Stability, 🧩Extension Supported, 🦄Multi-modal - LLM Native Instant Messaging Bot Platform🤖
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
|
||||||
|
[](https://discord.gg/wdNEHETs87)
|
||||||
|
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
||||||
|
)
|
||||||
|
<img src="https://img.shields.io/badge/python-3.10 ~ 3.13 -blue.svg" alt="python">
|
||||||
|
|
||||||
|
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</p>
|
||||||
|
|
||||||
|
## ✨ Features
|
||||||
|
|
||||||
|
- 💬 Chat with LLM / Agent: Supports multiple LLMs, adapt to group chats and private chats; Supports multi-round conversations, tool calls, and multi-modal capabilities. Deeply integrates with [Dify](https://dify.ai). Currently supports QQ, QQ Channel, WeCom, personal WeChat, Lark, DingTalk, Discord, Telegram, etc.
|
||||||
|
- 🛠️ High Stability, Feature-rich: Native access control, rate limiting, sensitive word filtering, etc. mechanisms; Easy to use, supports multiple deployment methods.
|
||||||
|
- 🧩 Plugin Extension, Active Community: Support event-driven, component extension, etc. plugin mechanisms; Integrate Anthropic [MCP protocol](https://modelcontextprotocol.io/); Currently has dozens of [plugins](https://docs.langbot.app/plugin/plugin-intro.html)
|
||||||
|
- 😻 [New] Web UI: Support management LangBot instance through the browser, for details, see [documentation](https://docs.langbot.app/webui/intro.html)
|
||||||
|
|
||||||
|
## 📦 Getting Started
|
||||||
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
>
|
||||||
|
> - Before you start deploying in any way, please read the [New User Guide](https://docs.langbot.app/insight/guide.html).
|
||||||
|
> - All documentation is in Chinese, we will provide i18n version in the near future.
|
||||||
|
|
||||||
|
#### Docker Compose Deployment
|
||||||
|
|
||||||
|
Suitable for users familiar with Docker, see the [Docker Deployment](https://docs.langbot.app/deploy/langbot/docker.html) documentation.
|
||||||
|
|
||||||
|
#### One-click Deployment on BTPanel
|
||||||
|
|
||||||
|
LangBot has been listed on the BTPanel, if you have installed the BTPanel, you can use the [document](https://docs.langbot.app/deploy/langbot/one-click/bt.html) to use it.
|
||||||
|
|
||||||
|
#### Zeabur Cloud Deployment
|
||||||
|
|
||||||
|
Community contributed Zeabur template.
|
||||||
|
|
||||||
|
[](https://zeabur.com/zh-CN/templates/ZKTBDH)
|
||||||
|
|
||||||
|
#### Railway Cloud Deployment
|
||||||
|
|
||||||
|
[](https://railway.app/template/yRrAyL?referralCode=vogKPF)
|
||||||
|
|
||||||
|
#### Other Deployment Methods
|
||||||
|
|
||||||
|
Directly use the released version to run, see the [Manual Deployment](https://docs.langbot.app/deploy/langbot/manual.html) documentation.
|
||||||
|
|
||||||
|
## 📸 Demo
|
||||||
|
|
||||||
|
<img alt="Reply Effect (with Internet Plugin)" src="https://docs.langbot.app/QChatGPT-0516.png" width="500px"/>
|
||||||
|
|
||||||
|
- WebUI Demo: https://demo.langbot.dev/
|
||||||
|
- Login information: Email: `demo@langbot.app` Password: `langbot123456`
|
||||||
|
- Note: Only the WebUI effect is shown, please do not fill in any sensitive information in the public environment.
|
||||||
|
|
||||||
|
## 🔌 Component Compatibility
|
||||||
|
|
||||||
|
### Message Platform
|
||||||
|
|
||||||
|
| Platform | Status | Remarks |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| Personal QQ | ✅ | |
|
||||||
|
| QQ Official API | ✅ | |
|
||||||
|
| WeCom | ✅ | |
|
||||||
|
| WeComCS | ✅ | |
|
||||||
|
| Personal WeChat | ✅ | Use [Gewechat](https://github.com/Devo919/Gewechat) to access |
|
||||||
|
| Lark | ✅ | |
|
||||||
|
| DingTalk | ✅ | |
|
||||||
|
| Discord | ✅ | |
|
||||||
|
| Telegram | ✅ | |
|
||||||
|
| Slack | ✅ | |
|
||||||
|
| LINE | 🚧 | |
|
||||||
|
| WhatsApp | 🚧 | |
|
||||||
|
|
||||||
|
🚧: In development
|
||||||
|
|
||||||
|
### LLMs
|
||||||
|
|
||||||
|
| LLM | Status | Remarks |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| [OpenAI](https://platform.openai.com/) | ✅ | Available for any OpenAI interface format model |
|
||||||
|
| [DeepSeek](https://www.deepseek.com/) | ✅ | |
|
||||||
|
| [Moonshot](https://www.moonshot.cn/) | ✅ | |
|
||||||
|
| [Anthropic](https://www.anthropic.com/) | ✅ | |
|
||||||
|
| [xAI](https://x.ai/) | ✅ | |
|
||||||
|
| [Zhipu AI](https://open.bigmodel.cn/) | ✅ | |
|
||||||
|
| [Dify](https://dify.ai) | ✅ | LLMOps platform |
|
||||||
|
| [Ollama](https://ollama.com/) | ✅ | Local LLM running platform |
|
||||||
|
| [LMStudio](https://lmstudio.ai/) | ✅ | Local LLM running platform |
|
||||||
|
| [GiteeAI](https://ai.gitee.com/) | ✅ | LLM interface gateway(MaaS) |
|
||||||
|
| [SiliconFlow](https://siliconflow.cn/) | ✅ | LLM gateway(MaaS) |
|
||||||
|
| [Aliyun Bailian](https://bailian.console.aliyun.com/) | ✅ | LLM gateway(MaaS), LLMOps platform |
|
||||||
|
| [Volc Engine Ark](https://console.volcengine.com/ark/region:ark+cn-beijing/model?vendor=Bytedance&view=LIST_VIEW) | ✅ | LLM gateway(MaaS), LLMOps platform |
|
||||||
|
| [ModelScope](https://modelscope.cn/docs/model-service/API-Inference/intro) | ✅ | LLM gateway(MaaS) |
|
||||||
|
| [MCP](https://modelcontextprotocol.io/) | ✅ | Support tool access through MCP protocol |
|
||||||
|
|
||||||
|
## 🤝 Community Contribution
|
||||||
|
|
||||||
|
Thank you for the following [code contributors](https://github.com/RockChinQ/LangBot/graphs/contributors) and other members in the community for their contributions to LangBot:
|
||||||
|
|
||||||
|
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
||||||
|
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
And the core team members of LangBot:
|
||||||
|
|
||||||
|
- [RockChinQ](https://github.com/RockChinQ)
|
||||||
|
- [the-lazy-me](https://github.com/the-lazy-me)
|
||||||
|
- [wangcham](https://github.com/wangcham)
|
||||||
|
- [KaedeSAMA](https://github.com/KaedeSAMA)
|
||||||
132
README_JP.md
Normal file
132
README_JP.md
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
<p align="center">
|
||||||
|
<a href="https://langbot.app">
|
||||||
|
<img src="https://docs.langbot.app/social.png" alt="LangBot"/>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
|
||||||
|
<a href="https://trendshift.io/repositories/12901" target="_blank"><img src="https://trendshift.io/api/badge/repositories/12901" alt="RockChinQ%2FLangBot | Trendshift" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
||||||
|
|
||||||
|
<a href="https://docs.langbot.app">ホーム</a> |
|
||||||
|
<a href="https://docs.langbot.app/insight/intro.html">機能</a> |
|
||||||
|
<a href="https://docs.langbot.app/insight/guide.html">デプロイ</a> |
|
||||||
|
<a href="https://docs.langbot.app/usage/faq.html">FAQ</a> |
|
||||||
|
<a href="https://docs.langbot.app/plugin/plugin-intro.html">プラグイン</a> |
|
||||||
|
<a href="https://github.com/RockChinQ/LangBot/issues/new?assignees=&labels=%E7%8B%AC%E7%AB%8B%E6%8F%92%E4%BB%B6&projects=&template=submit-plugin.yml&title=%5BPlugin%5D%3A+%E8%AF%B7%E6%B1%82%E7%99%BB%E8%AE%B0%E6%96%B0%E6%8F%92%E4%BB%B6">プラグインの提出</a>
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
😎高い安定性、🧩拡張サポート、🦄マルチモーダル - LLMネイティブインスタントメッセージングボットプラットフォーム🤖
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
[](https://discord.gg/wdNEHETs87)
|
||||||
|
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
||||||
|
)
|
||||||
|
<img src="https://img.shields.io/badge/python-3.10 ~ 3.13 -blue.svg" alt="python">
|
||||||
|
|
||||||
|
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</p>
|
||||||
|
|
||||||
|
## ✨ 機能
|
||||||
|
|
||||||
|
- 💬 LLM / エージェントとのチャット: 複数のLLMをサポートし、グループチャットとプライベートチャットに対応。マルチラウンドの会話、ツールの呼び出し、マルチモーダル機能をサポート。 [Dify](https://dify.ai) と深く統合。現在、QQ、QQ チャンネル、WeChat、個人 WeChat、Lark、DingTalk、Discord、Telegram など、複数のプラットフォームをサポートしています。
|
||||||
|
- 🛠️ 高い安定性、豊富な機能: ネイティブのアクセス制御、レート制限、敏感な単語のフィルタリングなどのメカニズムをサポート。使いやすく、複数のデプロイ方法をサポート。
|
||||||
|
- 🧩 プラグイン拡張、活発なコミュニティ: イベント駆動、コンポーネント拡張などのプラグインメカニズムをサポート。適配 Anthropic [MCP プロトコル](https://modelcontextprotocol.io/);豊富なエコシステム、現在数十の[プラグイン](https://docs.langbot.app/plugin/plugin-intro.html)が存在。
|
||||||
|
- 😻 [新機能] Web UI: ブラウザを通じてLangBotインスタンスを管理することをサポート。詳細は[ドキュメント](https://docs.langbot.app/webui/intro.html)を参照。
|
||||||
|
|
||||||
|
## 📦 始め方
|
||||||
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
>
|
||||||
|
> - どのデプロイ方法を始める前に、必ず[新規ユーザーガイド](https://docs.langbot.app/insight/guide.html)をお読みください。
|
||||||
|
> - すべてのドキュメントは中国語で提供されています。近い将来、i18nバージョンを提供する予定です。
|
||||||
|
|
||||||
|
#### Docker Compose デプロイ
|
||||||
|
|
||||||
|
Dockerに慣れているユーザーに適しています。[Dockerデプロイ](https://docs.langbot.app/deploy/langbot/docker.html)のドキュメントを参照してください。
|
||||||
|
|
||||||
|
#### BTPanelでのワンクリックデプロイ
|
||||||
|
|
||||||
|
LangBotはBTPanelにリストされています。BTPanelをインストールしている場合は、[ドキュメント](https://docs.langbot.app/deploy/langbot/one-click/bt.html)を使用して使用できます。
|
||||||
|
|
||||||
|
#### Zeaburクラウドデプロイ
|
||||||
|
|
||||||
|
コミュニティが提供するZeaburテンプレート。
|
||||||
|
|
||||||
|
[](https://zeabur.com/zh-CN/templates/ZKTBDH)
|
||||||
|
|
||||||
|
#### Railwayクラウドデプロイ
|
||||||
|
|
||||||
|
[](https://railway.app/template/yRrAyL?referralCode=vogKPF)
|
||||||
|
|
||||||
|
#### その他のデプロイ方法
|
||||||
|
|
||||||
|
リリースバージョンを直接使用して実行します。[手動デプロイ](https://docs.langbot.app/deploy/langbot/manual.html)のドキュメントを参照してください。
|
||||||
|
|
||||||
|
## 📸 デモ
|
||||||
|
|
||||||
|
<img alt="返信効果(インターネットプラグイン付き)" src="https://docs.langbot.app/QChatGPT-0516.png" width="500px"/>
|
||||||
|
|
||||||
|
- WebUIデモ: https://demo.langbot.dev/
|
||||||
|
- ログイン情報: メール: `demo@langbot.app` パスワード: `langbot123456`
|
||||||
|
- 注意: WebUIの効果のみを示しています。公開環境では、機密情報を入力しないでください。
|
||||||
|
|
||||||
|
## 🔌 コンポーネントの互換性
|
||||||
|
|
||||||
|
### メッセージプラットフォーム
|
||||||
|
|
||||||
|
| プラットフォーム | ステータス | 備考 |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| 個人QQ | ✅ | |
|
||||||
|
| QQ公式API | ✅ | |
|
||||||
|
| WeCom | ✅ | |
|
||||||
|
| WeComCS | ✅ | |
|
||||||
|
| 個人WeChat | ✅ | [Gewechat](https://github.com/Devo919/Gewechat)を使用して接続 |
|
||||||
|
| Lark | ✅ | |
|
||||||
|
| DingTalk | ✅ | |
|
||||||
|
| Discord | ✅ | |
|
||||||
|
| Telegram | ✅ | |
|
||||||
|
| Slack | ✅ | |
|
||||||
|
| LINE | 🚧 | |
|
||||||
|
| WhatsApp | 🚧 | |
|
||||||
|
|
||||||
|
🚧: 開発中
|
||||||
|
|
||||||
|
### LLMs
|
||||||
|
|
||||||
|
| LLM | ステータス | 備考 |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| [OpenAI](https://platform.openai.com/) | ✅ | 任意のOpenAIインターフェース形式モデルに対応 |
|
||||||
|
| [DeepSeek](https://www.deepseek.com/) | ✅ | |
|
||||||
|
| [Moonshot](https://www.moonshot.cn/) | ✅ | |
|
||||||
|
| [Anthropic](https://www.anthropic.com/) | ✅ | |
|
||||||
|
| [xAI](https://x.ai/) | ✅ | |
|
||||||
|
| [Zhipu AI](https://open.bigmodel.cn/) | ✅ | |
|
||||||
|
| [Dify](https://dify.ai) | ✅ | LLMOpsプラットフォーム |
|
||||||
|
| [Ollama](https://ollama.com/) | ✅ | ローカルLLM実行プラットフォーム |
|
||||||
|
| [LMStudio](https://lmstudio.ai/) | ✅ | ローカルLLM実行プラットフォーム |
|
||||||
|
| [GiteeAI](https://ai.gitee.com/) | ✅ | LLMインターフェースゲートウェイ(MaaS) |
|
||||||
|
| [SiliconFlow](https://siliconflow.cn/) | ✅ | LLMゲートウェイ(MaaS) |
|
||||||
|
| [Aliyun Bailian](https://bailian.console.aliyun.com/) | ✅ | LLMゲートウェイ(MaaS), LLMOpsプラットフォーム |
|
||||||
|
| [Volc Engine Ark](https://console.volcengine.com/ark/region:ark+cn-beijing/model?vendor=Bytedance&view=LIST_VIEW) | ✅ | LLMゲートウェイ(MaaS), LLMOpsプラットフォーム |
|
||||||
|
| [ModelScope](https://modelscope.cn/docs/model-service/API-Inference/intro) | ✅ | LLMゲートウェイ(MaaS) |
|
||||||
|
| [MCP](https://modelcontextprotocol.io/) | ✅ | MCPプロトコルをサポート |
|
||||||
|
|
||||||
|
## 🤝 コミュニティ貢献
|
||||||
|
|
||||||
|
LangBot への貢献に対して、以下の [コード貢献者](https://github.com/RockChinQ/LangBot/graphs/contributors) とコミュニティの他のメンバーに感謝します。
|
||||||
|
|
||||||
|
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
||||||
|
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
LangBot の核心チームメンバー:
|
||||||
|
|
||||||
|
- [RockChinQ](https://github.com/RockChinQ)
|
||||||
|
- [the-lazy-me](https://github.com/the-lazy-me)
|
||||||
|
- [wangcham](https://github.com/wangcham)
|
||||||
|
- [KaedeSAMA](https://github.com/KaedeSAMA)
|
||||||
19
components.yaml
Normal file
19
components.yaml
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
apiVersion: v1
|
||||||
|
kind: Blueprint
|
||||||
|
metadata:
|
||||||
|
name: builtin-components
|
||||||
|
label:
|
||||||
|
en_US: Builtin Components
|
||||||
|
zh_CN: 内置组件
|
||||||
|
spec:
|
||||||
|
components:
|
||||||
|
ComponentTemplate:
|
||||||
|
fromFiles:
|
||||||
|
- pkg/platform/adapter.yaml
|
||||||
|
- pkg/provider/modelmgr/requester.yaml
|
||||||
|
MessagePlatformAdapter:
|
||||||
|
fromDirs:
|
||||||
|
- path: pkg/platform/sources/
|
||||||
|
LLMAPIRequester:
|
||||||
|
fromDirs:
|
||||||
|
- path: pkg/provider/modelmgr/requesters/
|
||||||
@@ -1,281 +0,0 @@
|
|||||||
# 配置文件: 注释里标[必需]的参数必须修改, 其他参数根据需要修改, 但请勿删除
|
|
||||||
import logging
|
|
||||||
|
|
||||||
# [必需] Mirai的配置
|
|
||||||
# 请到配置mirai的步骤中的教程查看每个字段的信息
|
|
||||||
# adapter: 选择适配器,目前支持HTTPAdapter和WebSocketAdapter
|
|
||||||
# host: 运行mirai的主机地址
|
|
||||||
# port: 运行mirai的主机端口
|
|
||||||
# verifyKey: mirai-api-http的verifyKey
|
|
||||||
# qq: 机器人的QQ号
|
|
||||||
#
|
|
||||||
# 注意: QQ机器人配置不支持热重载及热更新
|
|
||||||
mirai_http_api_config = {
|
|
||||||
"adapter": "WebSocketAdapter",
|
|
||||||
"host": "localhost",
|
|
||||||
"port": 8080,
|
|
||||||
"verifyKey": "yirimirai",
|
|
||||||
"qq": 1234567890
|
|
||||||
}
|
|
||||||
|
|
||||||
# [必需] OpenAI的配置
|
|
||||||
# api_key: OpenAI的API Key
|
|
||||||
# http_proxy: 请求OpenAI时使用的代理,None为不使用,https和socks5暂不能使用
|
|
||||||
# 若只有一个api-key,请直接修改以下内容中的"openai_api_key"为你的api-key
|
|
||||||
#
|
|
||||||
# 如准备了多个api-key,可以以字典的形式填写,程序会自动选择可用的api-key
|
|
||||||
# 例如
|
|
||||||
# openai_config = {
|
|
||||||
# "api_key": {
|
|
||||||
# "default": "sk-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
# "key1": "sk-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
# "key2": "sk-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
# },
|
|
||||||
# "http_proxy": "http://127.0.0.1:12345"
|
|
||||||
# }
|
|
||||||
#
|
|
||||||
# 现已支持反向代理,可以添加reverse_proxy字段以使用反向代理
|
|
||||||
# 使用反向代理可以在国内使用OpenAI的API,反向代理的配置请参考
|
|
||||||
# https://github.com/Ice-Hazymoon/openai-scf-proxy
|
|
||||||
#
|
|
||||||
# 反向代理填写示例:
|
|
||||||
# openai_config = {
|
|
||||||
# "api_key": {
|
|
||||||
# "default": "sk-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
# "key1": "sk-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
# "key2": "sk-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
# },
|
|
||||||
# "reverse_proxy": "http://example.com:12345/v1"
|
|
||||||
# }
|
|
||||||
openai_config = {
|
|
||||||
"api_key": {
|
|
||||||
"default": "openai_api_key"
|
|
||||||
},
|
|
||||||
"http_proxy": None,
|
|
||||||
"reverse_proxy": None
|
|
||||||
}
|
|
||||||
|
|
||||||
# [必需] 管理员QQ号,用于接收报错等通知及执行管理员级别指令
|
|
||||||
# 支持多个管理员,可以使用list形式设置,例如:
|
|
||||||
# admin_qq = [12345678, 87654321]
|
|
||||||
admin_qq = 0
|
|
||||||
|
|
||||||
# 情景预设(机器人人格)
|
|
||||||
# 每个会话的预设信息,影响所有会话,无视指令重置
|
|
||||||
# 可以通过这个字段指定某些情况的回复,可直接用自然语言描述指令
|
|
||||||
# 例如:
|
|
||||||
# default_prompt = "如果我之后想获取帮助,请你说“输入!help获取帮助”"
|
|
||||||
# 这样用户在不知所措的时候机器人就会提示其输入!help获取帮助
|
|
||||||
# 可参考 https://github.com/PlexPt/awesome-chatgpt-prompts-zh
|
|
||||||
#
|
|
||||||
# 如果需要多个情景预设,并在运行期间方便切换,请使用字典的形式填写,例如
|
|
||||||
# default_prompt = {
|
|
||||||
# "default": "如果我之后想获取帮助,请你说“输入!help获取帮助”",
|
|
||||||
# "linux-terminal": "我想让你充当 Linux 终端。我将输入命令,您将回复终端应显示的内容。",
|
|
||||||
# "en-dict": "我想让你充当英英词典,对于给出的英文单词,你要给出其中文意思以及英文解释,并且给出一个例句,此外不要有其他反馈。",
|
|
||||||
# }
|
|
||||||
#
|
|
||||||
# 在使用期间即可通过指令:
|
|
||||||
# !reset [名称]
|
|
||||||
# 来使用指定的情景预设重置会话
|
|
||||||
# 例如:
|
|
||||||
# !reset linux-terminal
|
|
||||||
# 若不指定名称,则使用默认情景预设
|
|
||||||
#
|
|
||||||
# 也可以使用指令:
|
|
||||||
# !default <名称>
|
|
||||||
# 将指定的情景预设设置为默认情景预设
|
|
||||||
# 例如:
|
|
||||||
# !default linux-terminal
|
|
||||||
# 之后的会话重置时若不指定名称,则使用linux-terminal情景预设
|
|
||||||
#
|
|
||||||
# 还可以加载文件中的预设文字,使用方法请查看:https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E9%A2%84%E8%AE%BE%E6%96%87%E5%AD%97
|
|
||||||
default_prompt = {
|
|
||||||
"default": "如果我之后想获取帮助,请你说“输入!help获取帮助”",
|
|
||||||
}
|
|
||||||
|
|
||||||
# 情景预设格式
|
|
||||||
# 参考值:默认方式:normal | 完整情景:full_scenario
|
|
||||||
# 默认方式 的格式为上述default_prompt中的内容,或prompts目录下的文件名
|
|
||||||
# 完整情景方式 的格式为JSON,在scenario目录下的JSON文件中列出对话的每个回合,编写方法见scenario/default-template.json
|
|
||||||
# 编写方法请查看:https://github.com/RockChinQ/QChatGPT/wiki/%E5%8A%9F%E8%83%BD%E4%BD%BF%E7%94%A8#%E9%A2%84%E8%AE%BE%E6%96%87%E5%AD%97full_scenario%E6%A8%A1%E5%BC%8F
|
|
||||||
preset_mode = "normal"
|
|
||||||
|
|
||||||
# 群内响应规则
|
|
||||||
# 符合此消息的群内消息即使不包含at机器人也会响应
|
|
||||||
# 支持消息前缀匹配及正则表达式匹配
|
|
||||||
# 支持设置是否响应at消息、随机响应概率
|
|
||||||
# 注意:由消息前缀(prefix)匹配的消息中将会删除此前缀,正则表达式(regexp)匹配的消息不会删除匹配的部分
|
|
||||||
# 前缀匹配优先级高于正则表达式匹配
|
|
||||||
# 正则表达式简明教程:https://www.runoob.com/regexp/regexp-tutorial.html
|
|
||||||
response_rules = {
|
|
||||||
"at": True, # 是否响应at机器人的消息
|
|
||||||
"prefix": ["/ai", "!ai", "!ai", "ai"],
|
|
||||||
"regexp": [], # "为什么.*", "怎么?样.*", "怎么.*", "如何.*", "[Hh]ow to.*", "[Ww]hy not.*", "[Ww]hat is.*", ".*怎么办", ".*咋办"
|
|
||||||
"random_rate": 0.0, # 随机响应概率,0.0-1.0,0.0为不随机响应,1.0为响应所有消息, 仅在前几项判断不通过时生效
|
|
||||||
}
|
|
||||||
|
|
||||||
# 消息忽略规则
|
|
||||||
# 适用于私聊及群聊
|
|
||||||
# 符合此规则的消息将不会被响应
|
|
||||||
# 支持消息前缀匹配及正则表达式匹配
|
|
||||||
# 此设置优先级高于response_rules
|
|
||||||
# 用以过滤mirai等其他层级的指令
|
|
||||||
# @see https://github.com/RockChinQ/QChatGPT/issues/165
|
|
||||||
ignore_rules = {
|
|
||||||
"prefix": ["/"],
|
|
||||||
"regexp": []
|
|
||||||
}
|
|
||||||
|
|
||||||
# 是否检查收到的消息中是否包含敏感词
|
|
||||||
# 若收到的消息无法通过下方指定的敏感词检查策略,则发送提示信息
|
|
||||||
income_msg_check = False
|
|
||||||
|
|
||||||
# 敏感词过滤开关,以同样数量的*代替敏感词回复
|
|
||||||
# 请在sensitive.json中添加敏感词
|
|
||||||
sensitive_word_filter = True
|
|
||||||
|
|
||||||
# 是否启用百度云内容安全审核
|
|
||||||
# 注册方式查看 https://cloud.baidu.com/doc/ANTIPORN/s/Wkhu9d5iy
|
|
||||||
baidu_check = False
|
|
||||||
|
|
||||||
# 百度云API_KEY 24位英文数字字符串
|
|
||||||
baidu_api_key = ""
|
|
||||||
|
|
||||||
# 百度云SECRET_KEY 32位的英文数字字符串
|
|
||||||
baidu_secret_key = ""
|
|
||||||
|
|
||||||
# 不合规消息自定义返回
|
|
||||||
inappropriate_message_tips = "[百度云]请珍惜机器人,当前返回内容不合规"
|
|
||||||
|
|
||||||
# 启动时是否发送赞赏码
|
|
||||||
# 仅当使用量已经超过2048字时发送
|
|
||||||
encourage_sponsor_at_start = True
|
|
||||||
|
|
||||||
# 每次向OpenAI接口发送对话记录上下文的字符数
|
|
||||||
# 最大不超过(4096 - max_tokens)个字符,max_tokens为下方completion_api_params中的max_tokens
|
|
||||||
# 注意:较大的prompt_submit_length会导致OpenAI账户额度消耗更快
|
|
||||||
prompt_submit_length = 2048
|
|
||||||
|
|
||||||
# OpenAI补全API的参数
|
|
||||||
# 请在下方填写模型,程序自动选择接口
|
|
||||||
# 现已支持的模型有:
|
|
||||||
#
|
|
||||||
# 'gpt-4'
|
|
||||||
# 'gpt-4-0314'
|
|
||||||
# 'gpt-4-32k'
|
|
||||||
# 'gpt-4-32k-0314'
|
|
||||||
# 'gpt-3.5-turbo'
|
|
||||||
# 'gpt-3.5-turbo-0301'
|
|
||||||
# 'text-davinci-003'
|
|
||||||
# 'text-davinci-002'
|
|
||||||
# 'code-davinci-002'
|
|
||||||
# 'code-cushman-001'
|
|
||||||
# 'text-curie-001'
|
|
||||||
# 'text-babbage-001'
|
|
||||||
# 'text-ada-001'
|
|
||||||
#
|
|
||||||
# 具体请查看OpenAI的文档: https://beta.openai.com/docs/api-reference/completions/create
|
|
||||||
# 请将内容修改到config.py中,请勿修改config-template.py
|
|
||||||
completion_api_params = {
|
|
||||||
"model": "gpt-3.5-turbo",
|
|
||||||
"temperature": 0.9, # 数值越低得到的回答越理性,取值范围[0, 1]
|
|
||||||
"top_p": 1, # 生成的文本的文本与要求的符合度, 取值范围[0, 1]
|
|
||||||
"frequency_penalty": 0.2,
|
|
||||||
"presence_penalty": 1.0,
|
|
||||||
}
|
|
||||||
|
|
||||||
# OpenAI的Image API的参数
|
|
||||||
# 具体请查看OpenAI的文档: https://beta.openai.com/docs/api-reference/images/create
|
|
||||||
image_api_params = {
|
|
||||||
"size": "256x256", # 图片尺寸,支持256x256, 512x512, 1024x1024
|
|
||||||
}
|
|
||||||
|
|
||||||
# 群内回复消息时是否引用原消息
|
|
||||||
quote_origin = True
|
|
||||||
|
|
||||||
# 回复绘图时是否包含图片描述
|
|
||||||
include_image_description = True
|
|
||||||
|
|
||||||
# 消息处理的超时时间,单位为秒
|
|
||||||
process_message_timeout = 30
|
|
||||||
|
|
||||||
# 回复消息时是否显示[GPT]前缀
|
|
||||||
show_prefix = False
|
|
||||||
|
|
||||||
# 应用长消息处理策略的阈值
|
|
||||||
# 当回复消息长度超过此值时,将使用长消息处理策略
|
|
||||||
blob_message_threshold = 256
|
|
||||||
|
|
||||||
# 长消息处理策略
|
|
||||||
# - "image": 将长消息转换为图片发送
|
|
||||||
# - "forward": 将长消息转换为转发消息组件发送
|
|
||||||
blob_message_strategy = "forward"
|
|
||||||
|
|
||||||
# 文字转图片时使用的字体文件路径
|
|
||||||
# 当策略为"image"时生效
|
|
||||||
# 若在Windows系统下,程序会自动使用Windows自带的微软雅黑字体
|
|
||||||
# 若未填写或不存在且不是Windows,将禁用文字转图片功能,改为使用转发消息组件
|
|
||||||
font_path = ""
|
|
||||||
|
|
||||||
# 消息处理超时重试次数
|
|
||||||
retry_times = 3
|
|
||||||
|
|
||||||
# 消息处理出错时是否向用户隐藏错误详细信息
|
|
||||||
# 设置为True时,仅向管理员发送错误详细信息
|
|
||||||
# 设置为False时,向用户及管理员发送错误详细信息
|
|
||||||
hide_exce_info_to_user = False
|
|
||||||
|
|
||||||
# 消息处理出错时向用户发送的提示信息
|
|
||||||
# 仅当hide_exce_info_to_user为True时生效
|
|
||||||
# 设置为空字符串时,不发送提示信息
|
|
||||||
alter_tip_message = '出错了,请稍后再试'
|
|
||||||
|
|
||||||
# 线程池相关配置
|
|
||||||
# 该参数决定机器人可以同时处理几个人的消息,超出线程池数量的请求会被阻塞,不会被丢弃
|
|
||||||
# 如果你不清楚该参数的意义,请不要更改
|
|
||||||
# 程序运行本身线程池,无代码层面修改请勿更改
|
|
||||||
sys_pool_num = 8
|
|
||||||
|
|
||||||
# 执行管理员请求和指令的线程池并行线程数量,一般和管理员数量相等
|
|
||||||
admin_pool_num = 2
|
|
||||||
|
|
||||||
# 执行用户请求和指令的线程池并行线程数量
|
|
||||||
# 如需要更高的并发,可以增大该值
|
|
||||||
user_pool_num = 6
|
|
||||||
|
|
||||||
# 每个会话的过期时间,单位为秒
|
|
||||||
# 默认值20分钟
|
|
||||||
session_expire_time = 60 * 20
|
|
||||||
|
|
||||||
# 会话限速
|
|
||||||
# 单会话内每分钟可进行的对话次数
|
|
||||||
# 若不需要限速,可以设置为一个很大的值
|
|
||||||
# 默认值60次,基本上不会触发限速
|
|
||||||
rate_limitation = 60
|
|
||||||
|
|
||||||
# 会话限速策略
|
|
||||||
# - "wait": 每次对话获取到回复时,等待一定时间再发送回复,保证其不会超过限速均值
|
|
||||||
# - "drop": 此分钟内,若对话次数超过限速次数,则丢弃之后的对话,每自然分钟重置
|
|
||||||
rate_limit_strategy = "wait"
|
|
||||||
|
|
||||||
# drop策略时,超过限速均值时,丢弃的对话的提示信息
|
|
||||||
# 仅当rate_limitation_strategy为"drop"时生效
|
|
||||||
# 若设置为空字符串,则不发送提示信息
|
|
||||||
rate_limit_drop_tip = "本分钟对话次数超过限速次数,此对话被丢弃"
|
|
||||||
|
|
||||||
# 是否在启动时进行依赖库更新
|
|
||||||
upgrade_dependencies = True
|
|
||||||
|
|
||||||
# 是否上报统计信息
|
|
||||||
# 用于统计机器人的使用情况,不会收集任何用户信息
|
|
||||||
# 仅上报时间、字数使用量、绘图使用量,其他信息不会上报
|
|
||||||
report_usage = True
|
|
||||||
|
|
||||||
# 日志级别
|
|
||||||
logging_level = logging.INFO
|
|
||||||
|
|
||||||
# 定制帮助消息
|
|
||||||
help_message = """此机器人通过调用大型语言模型生成回复,不具有情感。
|
|
||||||
你可以用自然语言与其交流,回复的消息中[GPT]开头的为模型生成的语言,[bot]开头的为程序提示。
|
|
||||||
欢迎到github.com/RockChinQ/QChatGPT 给个star"""
|
|
||||||
16
docker-compose.yaml
Normal file
16
docker-compose.yaml
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
version: "3"
|
||||||
|
|
||||||
|
services:
|
||||||
|
langbot:
|
||||||
|
image: rockchin/langbot:latest
|
||||||
|
container_name: langbot
|
||||||
|
volumes:
|
||||||
|
- ./data:/app/data
|
||||||
|
- ./plugins:/app/plugins
|
||||||
|
restart: on-failure
|
||||||
|
environment:
|
||||||
|
- TZ=Asia/Shanghai
|
||||||
|
ports:
|
||||||
|
- 5300:5300 # 供 WebUI 使用
|
||||||
|
- 2280-2290:2280-2290 # 供消息平台适配器方向连接
|
||||||
|
# 根据具体环境配置网络
|
||||||
674
libs/LICENSE
Normal file
674
libs/LICENSE
Normal file
@@ -0,0 +1,674 @@
|
|||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 3, 29 June 2007
|
||||||
|
|
||||||
|
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The GNU General Public License is a free, copyleft license for
|
||||||
|
software and other kinds of works.
|
||||||
|
|
||||||
|
The licenses for most software and other practical works are designed
|
||||||
|
to take away your freedom to share and change the works. By contrast,
|
||||||
|
the GNU General Public License is intended to guarantee your freedom to
|
||||||
|
share and change all versions of a program--to make sure it remains free
|
||||||
|
software for all its users. We, the Free Software Foundation, use the
|
||||||
|
GNU General Public License for most of our software; it applies also to
|
||||||
|
any other work released this way by its authors. You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
them if you wish), that you receive source code or can get it if you
|
||||||
|
want it, that you can change the software or use pieces of it in new
|
||||||
|
free programs, and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to prevent others from denying you
|
||||||
|
these rights or asking you to surrender the rights. Therefore, you have
|
||||||
|
certain responsibilities if you distribute copies of the software, or if
|
||||||
|
you modify it: responsibilities to respect the freedom of others.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must pass on to the recipients the same
|
||||||
|
freedoms that you received. You must make sure that they, too, receive
|
||||||
|
or can get the source code. And you must show them these terms so they
|
||||||
|
know their rights.
|
||||||
|
|
||||||
|
Developers that use the GNU GPL protect your rights with two steps:
|
||||||
|
(1) assert copyright on the software, and (2) offer you this License
|
||||||
|
giving you legal permission to copy, distribute and/or modify it.
|
||||||
|
|
||||||
|
For the developers' and authors' protection, the GPL clearly explains
|
||||||
|
that there is no warranty for this free software. For both users' and
|
||||||
|
authors' sake, the GPL requires that modified versions be marked as
|
||||||
|
changed, so that their problems will not be attributed erroneously to
|
||||||
|
authors of previous versions.
|
||||||
|
|
||||||
|
Some devices are designed to deny users access to install or run
|
||||||
|
modified versions of the software inside them, although the manufacturer
|
||||||
|
can do so. This is fundamentally incompatible with the aim of
|
||||||
|
protecting users' freedom to change the software. The systematic
|
||||||
|
pattern of such abuse occurs in the area of products for individuals to
|
||||||
|
use, which is precisely where it is most unacceptable. Therefore, we
|
||||||
|
have designed this version of the GPL to prohibit the practice for those
|
||||||
|
products. If such problems arise substantially in other domains, we
|
||||||
|
stand ready to extend this provision to those domains in future versions
|
||||||
|
of the GPL, as needed to protect the freedom of users.
|
||||||
|
|
||||||
|
Finally, every program is threatened constantly by software patents.
|
||||||
|
States should not allow patents to restrict development and use of
|
||||||
|
software on general-purpose computers, but in those that do, we wish to
|
||||||
|
avoid the special danger that patents applied to a free program could
|
||||||
|
make it effectively proprietary. To prevent this, the GPL assures that
|
||||||
|
patents cannot be used to render the program non-free.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
0. Definitions.
|
||||||
|
|
||||||
|
"This License" refers to version 3 of the GNU General Public License.
|
||||||
|
|
||||||
|
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||||
|
works, such as semiconductor masks.
|
||||||
|
|
||||||
|
"The Program" refers to any copyrightable work licensed under this
|
||||||
|
License. Each licensee is addressed as "you". "Licensees" and
|
||||||
|
"recipients" may be individuals or organizations.
|
||||||
|
|
||||||
|
To "modify" a work means to copy from or adapt all or part of the work
|
||||||
|
in a fashion requiring copyright permission, other than the making of an
|
||||||
|
exact copy. The resulting work is called a "modified version" of the
|
||||||
|
earlier work or a work "based on" the earlier work.
|
||||||
|
|
||||||
|
A "covered work" means either the unmodified Program or a work based
|
||||||
|
on the Program.
|
||||||
|
|
||||||
|
To "propagate" a work means to do anything with it that, without
|
||||||
|
permission, would make you directly or secondarily liable for
|
||||||
|
infringement under applicable copyright law, except executing it on a
|
||||||
|
computer or modifying a private copy. Propagation includes copying,
|
||||||
|
distribution (with or without modification), making available to the
|
||||||
|
public, and in some countries other activities as well.
|
||||||
|
|
||||||
|
To "convey" a work means any kind of propagation that enables other
|
||||||
|
parties to make or receive copies. Mere interaction with a user through
|
||||||
|
a computer network, with no transfer of a copy, is not conveying.
|
||||||
|
|
||||||
|
An interactive user interface displays "Appropriate Legal Notices"
|
||||||
|
to the extent that it includes a convenient and prominently visible
|
||||||
|
feature that (1) displays an appropriate copyright notice, and (2)
|
||||||
|
tells the user that there is no warranty for the work (except to the
|
||||||
|
extent that warranties are provided), that licensees may convey the
|
||||||
|
work under this License, and how to view a copy of this License. If
|
||||||
|
the interface presents a list of user commands or options, such as a
|
||||||
|
menu, a prominent item in the list meets this criterion.
|
||||||
|
|
||||||
|
1. Source Code.
|
||||||
|
|
||||||
|
The "source code" for a work means the preferred form of the work
|
||||||
|
for making modifications to it. "Object code" means any non-source
|
||||||
|
form of a work.
|
||||||
|
|
||||||
|
A "Standard Interface" means an interface that either is an official
|
||||||
|
standard defined by a recognized standards body, or, in the case of
|
||||||
|
interfaces specified for a particular programming language, one that
|
||||||
|
is widely used among developers working in that language.
|
||||||
|
|
||||||
|
The "System Libraries" of an executable work include anything, other
|
||||||
|
than the work as a whole, that (a) is included in the normal form of
|
||||||
|
packaging a Major Component, but which is not part of that Major
|
||||||
|
Component, and (b) serves only to enable use of the work with that
|
||||||
|
Major Component, or to implement a Standard Interface for which an
|
||||||
|
implementation is available to the public in source code form. A
|
||||||
|
"Major Component", in this context, means a major essential component
|
||||||
|
(kernel, window system, and so on) of the specific operating system
|
||||||
|
(if any) on which the executable work runs, or a compiler used to
|
||||||
|
produce the work, or an object code interpreter used to run it.
|
||||||
|
|
||||||
|
The "Corresponding Source" for a work in object code form means all
|
||||||
|
the source code needed to generate, install, and (for an executable
|
||||||
|
work) run the object code and to modify the work, including scripts to
|
||||||
|
control those activities. However, it does not include the work's
|
||||||
|
System Libraries, or general-purpose tools or generally available free
|
||||||
|
programs which are used unmodified in performing those activities but
|
||||||
|
which are not part of the work. For example, Corresponding Source
|
||||||
|
includes interface definition files associated with source files for
|
||||||
|
the work, and the source code for shared libraries and dynamically
|
||||||
|
linked subprograms that the work is specifically designed to require,
|
||||||
|
such as by intimate data communication or control flow between those
|
||||||
|
subprograms and other parts of the work.
|
||||||
|
|
||||||
|
The Corresponding Source need not include anything that users
|
||||||
|
can regenerate automatically from other parts of the Corresponding
|
||||||
|
Source.
|
||||||
|
|
||||||
|
The Corresponding Source for a work in source code form is that
|
||||||
|
same work.
|
||||||
|
|
||||||
|
2. Basic Permissions.
|
||||||
|
|
||||||
|
All rights granted under this License are granted for the term of
|
||||||
|
copyright on the Program, and are irrevocable provided the stated
|
||||||
|
conditions are met. This License explicitly affirms your unlimited
|
||||||
|
permission to run the unmodified Program. The output from running a
|
||||||
|
covered work is covered by this License only if the output, given its
|
||||||
|
content, constitutes a covered work. This License acknowledges your
|
||||||
|
rights of fair use or other equivalent, as provided by copyright law.
|
||||||
|
|
||||||
|
You may make, run and propagate covered works that you do not
|
||||||
|
convey, without conditions so long as your license otherwise remains
|
||||||
|
in force. You may convey covered works to others for the sole purpose
|
||||||
|
of having them make modifications exclusively for you, or provide you
|
||||||
|
with facilities for running those works, provided that you comply with
|
||||||
|
the terms of this License in conveying all material for which you do
|
||||||
|
not control copyright. Those thus making or running the covered works
|
||||||
|
for you must do so exclusively on your behalf, under your direction
|
||||||
|
and control, on terms that prohibit them from making any copies of
|
||||||
|
your copyrighted material outside their relationship with you.
|
||||||
|
|
||||||
|
Conveying under any other circumstances is permitted solely under
|
||||||
|
the conditions stated below. Sublicensing is not allowed; section 10
|
||||||
|
makes it unnecessary.
|
||||||
|
|
||||||
|
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||||
|
|
||||||
|
No covered work shall be deemed part of an effective technological
|
||||||
|
measure under any applicable law fulfilling obligations under article
|
||||||
|
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||||
|
similar laws prohibiting or restricting circumvention of such
|
||||||
|
measures.
|
||||||
|
|
||||||
|
When you convey a covered work, you waive any legal power to forbid
|
||||||
|
circumvention of technological measures to the extent such circumvention
|
||||||
|
is effected by exercising rights under this License with respect to
|
||||||
|
the covered work, and you disclaim any intention to limit operation or
|
||||||
|
modification of the work as a means of enforcing, against the work's
|
||||||
|
users, your or third parties' legal rights to forbid circumvention of
|
||||||
|
technological measures.
|
||||||
|
|
||||||
|
4. Conveying Verbatim Copies.
|
||||||
|
|
||||||
|
You may convey verbatim copies of the Program's source code as you
|
||||||
|
receive it, in any medium, provided that you conspicuously and
|
||||||
|
appropriately publish on each copy an appropriate copyright notice;
|
||||||
|
keep intact all notices stating that this License and any
|
||||||
|
non-permissive terms added in accord with section 7 apply to the code;
|
||||||
|
keep intact all notices of the absence of any warranty; and give all
|
||||||
|
recipients a copy of this License along with the Program.
|
||||||
|
|
||||||
|
You may charge any price or no price for each copy that you convey,
|
||||||
|
and you may offer support or warranty protection for a fee.
|
||||||
|
|
||||||
|
5. Conveying Modified Source Versions.
|
||||||
|
|
||||||
|
You may convey a work based on the Program, or the modifications to
|
||||||
|
produce it from the Program, in the form of source code under the
|
||||||
|
terms of section 4, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The work must carry prominent notices stating that you modified
|
||||||
|
it, and giving a relevant date.
|
||||||
|
|
||||||
|
b) The work must carry prominent notices stating that it is
|
||||||
|
released under this License and any conditions added under section
|
||||||
|
7. This requirement modifies the requirement in section 4 to
|
||||||
|
"keep intact all notices".
|
||||||
|
|
||||||
|
c) You must license the entire work, as a whole, under this
|
||||||
|
License to anyone who comes into possession of a copy. This
|
||||||
|
License will therefore apply, along with any applicable section 7
|
||||||
|
additional terms, to the whole of the work, and all its parts,
|
||||||
|
regardless of how they are packaged. This License gives no
|
||||||
|
permission to license the work in any other way, but it does not
|
||||||
|
invalidate such permission if you have separately received it.
|
||||||
|
|
||||||
|
d) If the work has interactive user interfaces, each must display
|
||||||
|
Appropriate Legal Notices; however, if the Program has interactive
|
||||||
|
interfaces that do not display Appropriate Legal Notices, your
|
||||||
|
work need not make them do so.
|
||||||
|
|
||||||
|
A compilation of a covered work with other separate and independent
|
||||||
|
works, which are not by their nature extensions of the covered work,
|
||||||
|
and which are not combined with it such as to form a larger program,
|
||||||
|
in or on a volume of a storage or distribution medium, is called an
|
||||||
|
"aggregate" if the compilation and its resulting copyright are not
|
||||||
|
used to limit the access or legal rights of the compilation's users
|
||||||
|
beyond what the individual works permit. Inclusion of a covered work
|
||||||
|
in an aggregate does not cause this License to apply to the other
|
||||||
|
parts of the aggregate.
|
||||||
|
|
||||||
|
6. Conveying Non-Source Forms.
|
||||||
|
|
||||||
|
You may convey a covered work in object code form under the terms
|
||||||
|
of sections 4 and 5, provided that you also convey the
|
||||||
|
machine-readable Corresponding Source under the terms of this License,
|
||||||
|
in one of these ways:
|
||||||
|
|
||||||
|
a) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by the
|
||||||
|
Corresponding Source fixed on a durable physical medium
|
||||||
|
customarily used for software interchange.
|
||||||
|
|
||||||
|
b) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by a
|
||||||
|
written offer, valid for at least three years and valid for as
|
||||||
|
long as you offer spare parts or customer support for that product
|
||||||
|
model, to give anyone who possesses the object code either (1) a
|
||||||
|
copy of the Corresponding Source for all the software in the
|
||||||
|
product that is covered by this License, on a durable physical
|
||||||
|
medium customarily used for software interchange, for a price no
|
||||||
|
more than your reasonable cost of physically performing this
|
||||||
|
conveying of source, or (2) access to copy the
|
||||||
|
Corresponding Source from a network server at no charge.
|
||||||
|
|
||||||
|
c) Convey individual copies of the object code with a copy of the
|
||||||
|
written offer to provide the Corresponding Source. This
|
||||||
|
alternative is allowed only occasionally and noncommercially, and
|
||||||
|
only if you received the object code with such an offer, in accord
|
||||||
|
with subsection 6b.
|
||||||
|
|
||||||
|
d) Convey the object code by offering access from a designated
|
||||||
|
place (gratis or for a charge), and offer equivalent access to the
|
||||||
|
Corresponding Source in the same way through the same place at no
|
||||||
|
further charge. You need not require recipients to copy the
|
||||||
|
Corresponding Source along with the object code. If the place to
|
||||||
|
copy the object code is a network server, the Corresponding Source
|
||||||
|
may be on a different server (operated by you or a third party)
|
||||||
|
that supports equivalent copying facilities, provided you maintain
|
||||||
|
clear directions next to the object code saying where to find the
|
||||||
|
Corresponding Source. Regardless of what server hosts the
|
||||||
|
Corresponding Source, you remain obligated to ensure that it is
|
||||||
|
available for as long as needed to satisfy these requirements.
|
||||||
|
|
||||||
|
e) Convey the object code using peer-to-peer transmission, provided
|
||||||
|
you inform other peers where the object code and Corresponding
|
||||||
|
Source of the work are being offered to the general public at no
|
||||||
|
charge under subsection 6d.
|
||||||
|
|
||||||
|
A separable portion of the object code, whose source code is excluded
|
||||||
|
from the Corresponding Source as a System Library, need not be
|
||||||
|
included in conveying the object code work.
|
||||||
|
|
||||||
|
A "User Product" is either (1) a "consumer product", which means any
|
||||||
|
tangible personal property which is normally used for personal, family,
|
||||||
|
or household purposes, or (2) anything designed or sold for incorporation
|
||||||
|
into a dwelling. In determining whether a product is a consumer product,
|
||||||
|
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||||
|
product received by a particular user, "normally used" refers to a
|
||||||
|
typical or common use of that class of product, regardless of the status
|
||||||
|
of the particular user or of the way in which the particular user
|
||||||
|
actually uses, or expects or is expected to use, the product. A product
|
||||||
|
is a consumer product regardless of whether the product has substantial
|
||||||
|
commercial, industrial or non-consumer uses, unless such uses represent
|
||||||
|
the only significant mode of use of the product.
|
||||||
|
|
||||||
|
"Installation Information" for a User Product means any methods,
|
||||||
|
procedures, authorization keys, or other information required to install
|
||||||
|
and execute modified versions of a covered work in that User Product from
|
||||||
|
a modified version of its Corresponding Source. The information must
|
||||||
|
suffice to ensure that the continued functioning of the modified object
|
||||||
|
code is in no case prevented or interfered with solely because
|
||||||
|
modification has been made.
|
||||||
|
|
||||||
|
If you convey an object code work under this section in, or with, or
|
||||||
|
specifically for use in, a User Product, and the conveying occurs as
|
||||||
|
part of a transaction in which the right of possession and use of the
|
||||||
|
User Product is transferred to the recipient in perpetuity or for a
|
||||||
|
fixed term (regardless of how the transaction is characterized), the
|
||||||
|
Corresponding Source conveyed under this section must be accompanied
|
||||||
|
by the Installation Information. But this requirement does not apply
|
||||||
|
if neither you nor any third party retains the ability to install
|
||||||
|
modified object code on the User Product (for example, the work has
|
||||||
|
been installed in ROM).
|
||||||
|
|
||||||
|
The requirement to provide Installation Information does not include a
|
||||||
|
requirement to continue to provide support service, warranty, or updates
|
||||||
|
for a work that has been modified or installed by the recipient, or for
|
||||||
|
the User Product in which it has been modified or installed. Access to a
|
||||||
|
network may be denied when the modification itself materially and
|
||||||
|
adversely affects the operation of the network or violates the rules and
|
||||||
|
protocols for communication across the network.
|
||||||
|
|
||||||
|
Corresponding Source conveyed, and Installation Information provided,
|
||||||
|
in accord with this section must be in a format that is publicly
|
||||||
|
documented (and with an implementation available to the public in
|
||||||
|
source code form), and must require no special password or key for
|
||||||
|
unpacking, reading or copying.
|
||||||
|
|
||||||
|
7. Additional Terms.
|
||||||
|
|
||||||
|
"Additional permissions" are terms that supplement the terms of this
|
||||||
|
License by making exceptions from one or more of its conditions.
|
||||||
|
Additional permissions that are applicable to the entire Program shall
|
||||||
|
be treated as though they were included in this License, to the extent
|
||||||
|
that they are valid under applicable law. If additional permissions
|
||||||
|
apply only to part of the Program, that part may be used separately
|
||||||
|
under those permissions, but the entire Program remains governed by
|
||||||
|
this License without regard to the additional permissions.
|
||||||
|
|
||||||
|
When you convey a copy of a covered work, you may at your option
|
||||||
|
remove any additional permissions from that copy, or from any part of
|
||||||
|
it. (Additional permissions may be written to require their own
|
||||||
|
removal in certain cases when you modify the work.) You may place
|
||||||
|
additional permissions on material, added by you to a covered work,
|
||||||
|
for which you have or can give appropriate copyright permission.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, for material you
|
||||||
|
add to a covered work, you may (if authorized by the copyright holders of
|
||||||
|
that material) supplement the terms of this License with terms:
|
||||||
|
|
||||||
|
a) Disclaiming warranty or limiting liability differently from the
|
||||||
|
terms of sections 15 and 16 of this License; or
|
||||||
|
|
||||||
|
b) Requiring preservation of specified reasonable legal notices or
|
||||||
|
author attributions in that material or in the Appropriate Legal
|
||||||
|
Notices displayed by works containing it; or
|
||||||
|
|
||||||
|
c) Prohibiting misrepresentation of the origin of that material, or
|
||||||
|
requiring that modified versions of such material be marked in
|
||||||
|
reasonable ways as different from the original version; or
|
||||||
|
|
||||||
|
d) Limiting the use for publicity purposes of names of licensors or
|
||||||
|
authors of the material; or
|
||||||
|
|
||||||
|
e) Declining to grant rights under trademark law for use of some
|
||||||
|
trade names, trademarks, or service marks; or
|
||||||
|
|
||||||
|
f) Requiring indemnification of licensors and authors of that
|
||||||
|
material by anyone who conveys the material (or modified versions of
|
||||||
|
it) with contractual assumptions of liability to the recipient, for
|
||||||
|
any liability that these contractual assumptions directly impose on
|
||||||
|
those licensors and authors.
|
||||||
|
|
||||||
|
All other non-permissive additional terms are considered "further
|
||||||
|
restrictions" within the meaning of section 10. If the Program as you
|
||||||
|
received it, or any part of it, contains a notice stating that it is
|
||||||
|
governed by this License along with a term that is a further
|
||||||
|
restriction, you may remove that term. If a license document contains
|
||||||
|
a further restriction but permits relicensing or conveying under this
|
||||||
|
License, you may add to a covered work material governed by the terms
|
||||||
|
of that license document, provided that the further restriction does
|
||||||
|
not survive such relicensing or conveying.
|
||||||
|
|
||||||
|
If you add terms to a covered work in accord with this section, you
|
||||||
|
must place, in the relevant source files, a statement of the
|
||||||
|
additional terms that apply to those files, or a notice indicating
|
||||||
|
where to find the applicable terms.
|
||||||
|
|
||||||
|
Additional terms, permissive or non-permissive, may be stated in the
|
||||||
|
form of a separately written license, or stated as exceptions;
|
||||||
|
the above requirements apply either way.
|
||||||
|
|
||||||
|
8. Termination.
|
||||||
|
|
||||||
|
You may not propagate or modify a covered work except as expressly
|
||||||
|
provided under this License. Any attempt otherwise to propagate or
|
||||||
|
modify it is void, and will automatically terminate your rights under
|
||||||
|
this License (including any patent licenses granted under the third
|
||||||
|
paragraph of section 11).
|
||||||
|
|
||||||
|
However, if you cease all violation of this License, then your
|
||||||
|
license from a particular copyright holder is reinstated (a)
|
||||||
|
provisionally, unless and until the copyright holder explicitly and
|
||||||
|
finally terminates your license, and (b) permanently, if the copyright
|
||||||
|
holder fails to notify you of the violation by some reasonable means
|
||||||
|
prior to 60 days after the cessation.
|
||||||
|
|
||||||
|
Moreover, your license from a particular copyright holder is
|
||||||
|
reinstated permanently if the copyright holder notifies you of the
|
||||||
|
violation by some reasonable means, this is the first time you have
|
||||||
|
received notice of violation of this License (for any work) from that
|
||||||
|
copyright holder, and you cure the violation prior to 30 days after
|
||||||
|
your receipt of the notice.
|
||||||
|
|
||||||
|
Termination of your rights under this section does not terminate the
|
||||||
|
licenses of parties who have received copies or rights from you under
|
||||||
|
this License. If your rights have been terminated and not permanently
|
||||||
|
reinstated, you do not qualify to receive new licenses for the same
|
||||||
|
material under section 10.
|
||||||
|
|
||||||
|
9. Acceptance Not Required for Having Copies.
|
||||||
|
|
||||||
|
You are not required to accept this License in order to receive or
|
||||||
|
run a copy of the Program. Ancillary propagation of a covered work
|
||||||
|
occurring solely as a consequence of using peer-to-peer transmission
|
||||||
|
to receive a copy likewise does not require acceptance. However,
|
||||||
|
nothing other than this License grants you permission to propagate or
|
||||||
|
modify any covered work. These actions infringe copyright if you do
|
||||||
|
not accept this License. Therefore, by modifying or propagating a
|
||||||
|
covered work, you indicate your acceptance of this License to do so.
|
||||||
|
|
||||||
|
10. Automatic Licensing of Downstream Recipients.
|
||||||
|
|
||||||
|
Each time you convey a covered work, the recipient automatically
|
||||||
|
receives a license from the original licensors, to run, modify and
|
||||||
|
propagate that work, subject to this License. You are not responsible
|
||||||
|
for enforcing compliance by third parties with this License.
|
||||||
|
|
||||||
|
An "entity transaction" is a transaction transferring control of an
|
||||||
|
organization, or substantially all assets of one, or subdividing an
|
||||||
|
organization, or merging organizations. If propagation of a covered
|
||||||
|
work results from an entity transaction, each party to that
|
||||||
|
transaction who receives a copy of the work also receives whatever
|
||||||
|
licenses to the work the party's predecessor in interest had or could
|
||||||
|
give under the previous paragraph, plus a right to possession of the
|
||||||
|
Corresponding Source of the work from the predecessor in interest, if
|
||||||
|
the predecessor has it or can get it with reasonable efforts.
|
||||||
|
|
||||||
|
You may not impose any further restrictions on the exercise of the
|
||||||
|
rights granted or affirmed under this License. For example, you may
|
||||||
|
not impose a license fee, royalty, or other charge for exercise of
|
||||||
|
rights granted under this License, and you may not initiate litigation
|
||||||
|
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||||
|
any patent claim is infringed by making, using, selling, offering for
|
||||||
|
sale, or importing the Program or any portion of it.
|
||||||
|
|
||||||
|
11. Patents.
|
||||||
|
|
||||||
|
A "contributor" is a copyright holder who authorizes use under this
|
||||||
|
License of the Program or a work on which the Program is based. The
|
||||||
|
work thus licensed is called the contributor's "contributor version".
|
||||||
|
|
||||||
|
A contributor's "essential patent claims" are all patent claims
|
||||||
|
owned or controlled by the contributor, whether already acquired or
|
||||||
|
hereafter acquired, that would be infringed by some manner, permitted
|
||||||
|
by this License, of making, using, or selling its contributor version,
|
||||||
|
but do not include claims that would be infringed only as a
|
||||||
|
consequence of further modification of the contributor version. For
|
||||||
|
purposes of this definition, "control" includes the right to grant
|
||||||
|
patent sublicenses in a manner consistent with the requirements of
|
||||||
|
this License.
|
||||||
|
|
||||||
|
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||||
|
patent license under the contributor's essential patent claims, to
|
||||||
|
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||||
|
propagate the contents of its contributor version.
|
||||||
|
|
||||||
|
In the following three paragraphs, a "patent license" is any express
|
||||||
|
agreement or commitment, however denominated, not to enforce a patent
|
||||||
|
(such as an express permission to practice a patent or covenant not to
|
||||||
|
sue for patent infringement). To "grant" such a patent license to a
|
||||||
|
party means to make such an agreement or commitment not to enforce a
|
||||||
|
patent against the party.
|
||||||
|
|
||||||
|
If you convey a covered work, knowingly relying on a patent license,
|
||||||
|
and the Corresponding Source of the work is not available for anyone
|
||||||
|
to copy, free of charge and under the terms of this License, through a
|
||||||
|
publicly available network server or other readily accessible means,
|
||||||
|
then you must either (1) cause the Corresponding Source to be so
|
||||||
|
available, or (2) arrange to deprive yourself of the benefit of the
|
||||||
|
patent license for this particular work, or (3) arrange, in a manner
|
||||||
|
consistent with the requirements of this License, to extend the patent
|
||||||
|
license to downstream recipients. "Knowingly relying" means you have
|
||||||
|
actual knowledge that, but for the patent license, your conveying the
|
||||||
|
covered work in a country, or your recipient's use of the covered work
|
||||||
|
in a country, would infringe one or more identifiable patents in that
|
||||||
|
country that you have reason to believe are valid.
|
||||||
|
|
||||||
|
If, pursuant to or in connection with a single transaction or
|
||||||
|
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||||
|
covered work, and grant a patent license to some of the parties
|
||||||
|
receiving the covered work authorizing them to use, propagate, modify
|
||||||
|
or convey a specific copy of the covered work, then the patent license
|
||||||
|
you grant is automatically extended to all recipients of the covered
|
||||||
|
work and works based on it.
|
||||||
|
|
||||||
|
A patent license is "discriminatory" if it does not include within
|
||||||
|
the scope of its coverage, prohibits the exercise of, or is
|
||||||
|
conditioned on the non-exercise of one or more of the rights that are
|
||||||
|
specifically granted under this License. You may not convey a covered
|
||||||
|
work if you are a party to an arrangement with a third party that is
|
||||||
|
in the business of distributing software, under which you make payment
|
||||||
|
to the third party based on the extent of your activity of conveying
|
||||||
|
the work, and under which the third party grants, to any of the
|
||||||
|
parties who would receive the covered work from you, a discriminatory
|
||||||
|
patent license (a) in connection with copies of the covered work
|
||||||
|
conveyed by you (or copies made from those copies), or (b) primarily
|
||||||
|
for and in connection with specific products or compilations that
|
||||||
|
contain the covered work, unless you entered into that arrangement,
|
||||||
|
or that patent license was granted, prior to 28 March 2007.
|
||||||
|
|
||||||
|
Nothing in this License shall be construed as excluding or limiting
|
||||||
|
any implied license or other defenses to infringement that may
|
||||||
|
otherwise be available to you under applicable patent law.
|
||||||
|
|
||||||
|
12. No Surrender of Others' Freedom.
|
||||||
|
|
||||||
|
If conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot convey a
|
||||||
|
covered work so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you may
|
||||||
|
not convey it at all. For example, if you agree to terms that obligate you
|
||||||
|
to collect a royalty for further conveying from those to whom you convey
|
||||||
|
the Program, the only way you could satisfy both those terms and this
|
||||||
|
License would be to refrain entirely from conveying the Program.
|
||||||
|
|
||||||
|
13. Use with the GNU Affero General Public License.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, you have
|
||||||
|
permission to link or combine any covered work with a work licensed
|
||||||
|
under version 3 of the GNU Affero General Public License into a single
|
||||||
|
combined work, and to convey the resulting work. The terms of this
|
||||||
|
License will continue to apply to the part which is the covered work,
|
||||||
|
but the special requirements of the GNU Affero General Public License,
|
||||||
|
section 13, concerning interaction through a network will apply to the
|
||||||
|
combination as such.
|
||||||
|
|
||||||
|
14. Revised Versions of this License.
|
||||||
|
|
||||||
|
The Free Software Foundation may publish revised and/or new versions of
|
||||||
|
the GNU General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the
|
||||||
|
Program specifies that a certain numbered version of the GNU General
|
||||||
|
Public License "or any later version" applies to it, you have the
|
||||||
|
option of following the terms and conditions either of that numbered
|
||||||
|
version or of any later version published by the Free Software
|
||||||
|
Foundation. If the Program does not specify a version number of the
|
||||||
|
GNU General Public License, you may choose any version ever published
|
||||||
|
by the Free Software Foundation.
|
||||||
|
|
||||||
|
If the Program specifies that a proxy can decide which future
|
||||||
|
versions of the GNU General Public License can be used, that proxy's
|
||||||
|
public statement of acceptance of a version permanently authorizes you
|
||||||
|
to choose that version for the Program.
|
||||||
|
|
||||||
|
Later license versions may give you additional or different
|
||||||
|
permissions. However, no additional obligations are imposed on any
|
||||||
|
author or copyright holder as a result of your choosing to follow a
|
||||||
|
later version.
|
||||||
|
|
||||||
|
15. Disclaimer of Warranty.
|
||||||
|
|
||||||
|
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||||
|
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||||
|
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||||
|
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||||
|
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||||
|
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. Limitation of Liability.
|
||||||
|
|
||||||
|
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||||
|
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||||
|
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||||
|
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||||
|
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||||
|
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||||
|
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||||
|
SUCH DAMAGES.
|
||||||
|
|
||||||
|
17. Interpretation of Sections 15 and 16.
|
||||||
|
|
||||||
|
If the disclaimer of warranty and limitation of liability provided
|
||||||
|
above cannot be given local legal effect according to their terms,
|
||||||
|
reviewing courts shall apply local law that most closely approximates
|
||||||
|
an absolute waiver of all civil liability in connection with the
|
||||||
|
Program, unless a warranty or assumption of liability accompanies a
|
||||||
|
copy of the Program in return for a fee.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
state the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program does terminal interaction, make it output a short
|
||||||
|
notice like this when it starts in an interactive mode:
|
||||||
|
|
||||||
|
<program> Copyright (C) <year> <name of author>
|
||||||
|
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, your program's commands
|
||||||
|
might be different; for a GUI interface, you would use an "about box".
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or school,
|
||||||
|
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||||
|
For more information on this, and how to apply and follow the GNU GPL, see
|
||||||
|
<https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
The GNU General Public License does not permit incorporating your program
|
||||||
|
into proprietary programs. If your program is a subroutine library, you
|
||||||
|
may consider it more useful to permit linking proprietary applications with
|
||||||
|
the library. If this is what you want to do, use the GNU Lesser General
|
||||||
|
Public License instead of this License. But first, please read
|
||||||
|
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
||||||
4
libs/README.md
Normal file
4
libs/README.md
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# LangBot/libs
|
||||||
|
|
||||||
|
LangBot 项目下的 libs 目录下的所有代码均遵循本目录下的许可证约束。
|
||||||
|
您在使用、修改、分发本目录下的代码时,需要遵守其中包含的条款。
|
||||||
3
libs/dify_service_api/README.md
Normal file
3
libs/dify_service_api/README.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# Dify Service API Python SDK
|
||||||
|
|
||||||
|
这个 SDK 尚不完全支持 Dify Service API 的所有功能。
|
||||||
2
libs/dify_service_api/__init__.py
Normal file
2
libs/dify_service_api/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
from .v1 import client
|
||||||
|
from .v1 import errors
|
||||||
44
libs/dify_service_api/test.py
Normal file
44
libs/dify_service_api/test.py
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
from v1 import client
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
import os
|
||||||
|
import json
|
||||||
|
|
||||||
|
|
||||||
|
class TestDifyClient:
|
||||||
|
async def test_chat_messages(self):
|
||||||
|
cln = client.AsyncDifyServiceClient(api_key=os.getenv("DIFY_API_KEY"), base_url=os.getenv("DIFY_BASE_URL"))
|
||||||
|
|
||||||
|
async for chunk in cln.chat_messages(inputs={}, query="调用工具查看现在几点?", user="test"):
|
||||||
|
print(json.dumps(chunk, ensure_ascii=False, indent=4))
|
||||||
|
|
||||||
|
async def test_upload_file(self):
|
||||||
|
cln = client.AsyncDifyServiceClient(api_key=os.getenv("DIFY_API_KEY"), base_url=os.getenv("DIFY_BASE_URL"))
|
||||||
|
|
||||||
|
file_bytes = open("img.png", "rb").read()
|
||||||
|
|
||||||
|
print(type(file_bytes))
|
||||||
|
|
||||||
|
file = ("img2.png", file_bytes, "image/png")
|
||||||
|
|
||||||
|
resp = await cln.upload_file(file=file, user="test")
|
||||||
|
print(json.dumps(resp, ensure_ascii=False, indent=4))
|
||||||
|
|
||||||
|
async def test_workflow_run(self):
|
||||||
|
cln = client.AsyncDifyServiceClient(api_key=os.getenv("DIFY_API_KEY"), base_url=os.getenv("DIFY_BASE_URL"))
|
||||||
|
|
||||||
|
# resp = await cln.workflow_run(inputs={}, user="test")
|
||||||
|
# # print(json.dumps(resp, ensure_ascii=False, indent=4))
|
||||||
|
# print(resp)
|
||||||
|
chunks = []
|
||||||
|
|
||||||
|
ignored_events = ['text_chunk']
|
||||||
|
async for chunk in cln.workflow_run(inputs={}, user="test"):
|
||||||
|
if chunk['event'] in ignored_events:
|
||||||
|
continue
|
||||||
|
chunks.append(chunk)
|
||||||
|
print(json.dumps(chunks, ensure_ascii=False, indent=4))
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
asyncio.run(TestDifyClient().test_chat_messages())
|
||||||
126
libs/dify_service_api/v1/client.py
Normal file
126
libs/dify_service_api/v1/client.py
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import httpx
|
||||||
|
import typing
|
||||||
|
import json
|
||||||
|
|
||||||
|
from .errors import DifyAPIError
|
||||||
|
|
||||||
|
|
||||||
|
class AsyncDifyServiceClient:
|
||||||
|
"""Dify Service API 客户端"""
|
||||||
|
|
||||||
|
api_key: str
|
||||||
|
base_url: str
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
api_key: str,
|
||||||
|
base_url: str = "https://api.dify.ai/v1",
|
||||||
|
) -> None:
|
||||||
|
self.api_key = api_key
|
||||||
|
self.base_url = base_url
|
||||||
|
|
||||||
|
async def chat_messages(
|
||||||
|
self,
|
||||||
|
inputs: dict[str, typing.Any],
|
||||||
|
query: str,
|
||||||
|
user: str,
|
||||||
|
response_mode: str = "streaming", # 当前不支持 blocking
|
||||||
|
conversation_id: str = "",
|
||||||
|
files: list[dict[str, typing.Any]] = [],
|
||||||
|
timeout: float = 30.0,
|
||||||
|
) -> typing.AsyncGenerator[dict[str, typing.Any], None]:
|
||||||
|
"""发送消息"""
|
||||||
|
if response_mode != "streaming":
|
||||||
|
raise DifyAPIError("当前仅支持 streaming 模式")
|
||||||
|
|
||||||
|
async with httpx.AsyncClient(
|
||||||
|
base_url=self.base_url,
|
||||||
|
trust_env=True,
|
||||||
|
timeout=timeout,
|
||||||
|
) as client:
|
||||||
|
async with client.stream(
|
||||||
|
"POST",
|
||||||
|
"/chat-messages",
|
||||||
|
headers={"Authorization": f"Bearer {self.api_key}", "Content-Type": "application/json"},
|
||||||
|
json={
|
||||||
|
"inputs": inputs,
|
||||||
|
"query": query,
|
||||||
|
"user": user,
|
||||||
|
"response_mode": response_mode,
|
||||||
|
"conversation_id": conversation_id,
|
||||||
|
"files": files,
|
||||||
|
},
|
||||||
|
) as r:
|
||||||
|
async for chunk in r.aiter_lines():
|
||||||
|
if r.status_code != 200:
|
||||||
|
raise DifyAPIError(f"{r.status_code} {chunk}")
|
||||||
|
if chunk.strip() == "":
|
||||||
|
continue
|
||||||
|
if chunk.startswith("data:"):
|
||||||
|
yield json.loads(chunk[5:])
|
||||||
|
|
||||||
|
async def workflow_run(
|
||||||
|
self,
|
||||||
|
inputs: dict[str, typing.Any],
|
||||||
|
user: str,
|
||||||
|
response_mode: str = "streaming", # 当前不支持 blocking
|
||||||
|
files: list[dict[str, typing.Any]] = [],
|
||||||
|
timeout: float = 30.0,
|
||||||
|
) -> typing.AsyncGenerator[dict[str, typing.Any], None]:
|
||||||
|
"""运行工作流"""
|
||||||
|
if response_mode != "streaming":
|
||||||
|
raise DifyAPIError("当前仅支持 streaming 模式")
|
||||||
|
|
||||||
|
async with httpx.AsyncClient(
|
||||||
|
base_url=self.base_url,
|
||||||
|
trust_env=True,
|
||||||
|
timeout=timeout,
|
||||||
|
) as client:
|
||||||
|
|
||||||
|
async with client.stream(
|
||||||
|
"POST",
|
||||||
|
"/workflows/run",
|
||||||
|
headers={"Authorization": f"Bearer {self.api_key}", "Content-Type": "application/json"},
|
||||||
|
json={
|
||||||
|
"inputs": inputs,
|
||||||
|
"user": user,
|
||||||
|
"response_mode": response_mode,
|
||||||
|
"files": files,
|
||||||
|
},
|
||||||
|
) as r:
|
||||||
|
async for chunk in r.aiter_lines():
|
||||||
|
if r.status_code != 200:
|
||||||
|
raise DifyAPIError(f"{r.status_code} {chunk}")
|
||||||
|
if chunk.strip() == "":
|
||||||
|
continue
|
||||||
|
if chunk.startswith("data:"):
|
||||||
|
yield json.loads(chunk[5:])
|
||||||
|
|
||||||
|
async def upload_file(
|
||||||
|
self,
|
||||||
|
file: httpx._types.FileTypes,
|
||||||
|
user: str,
|
||||||
|
timeout: float = 30.0,
|
||||||
|
) -> str:
|
||||||
|
"""上传文件"""
|
||||||
|
async with httpx.AsyncClient(
|
||||||
|
base_url=self.base_url,
|
||||||
|
trust_env=True,
|
||||||
|
timeout=timeout,
|
||||||
|
) as client:
|
||||||
|
# multipart/form-data
|
||||||
|
response = await client.post(
|
||||||
|
"/files/upload",
|
||||||
|
headers={"Authorization": f"Bearer {self.api_key}"},
|
||||||
|
files={
|
||||||
|
"file": file,
|
||||||
|
"user": (None, user),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.status_code != 201:
|
||||||
|
raise DifyAPIError(f"{response.status_code} {response.text}")
|
||||||
|
|
||||||
|
return response.json()
|
||||||
17
libs/dify_service_api/v1/client_test.py
Normal file
17
libs/dify_service_api/v1/client_test.py
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
from . import client
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
import os
|
||||||
|
|
||||||
|
|
||||||
|
class TestDifyClient:
|
||||||
|
async def test_chat_messages(self):
|
||||||
|
cln = client.DifyClient(api_key=os.getenv("DIFY_API_KEY"))
|
||||||
|
|
||||||
|
resp = await cln.chat_messages(inputs={}, query="Who are you?", user_id="test")
|
||||||
|
print(resp)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
asyncio.run(TestDifyClient().test_chat_messages())
|
||||||
6
libs/dify_service_api/v1/errors.py
Normal file
6
libs/dify_service_api/v1/errors.py
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
class DifyAPIError(Exception):
|
||||||
|
"""Dify API 请求失败"""
|
||||||
|
|
||||||
|
def __init__(self, message: str):
|
||||||
|
self.message = message
|
||||||
|
super().__init__(self.message)
|
||||||
31
libs/dingtalk_api/EchoHandler.py
Normal file
31
libs/dingtalk_api/EchoHandler.py
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import asyncio
|
||||||
|
import json
|
||||||
|
import dingtalk_stream
|
||||||
|
from dingtalk_stream import AckMessage
|
||||||
|
|
||||||
|
class EchoTextHandler(dingtalk_stream.ChatbotHandler):
|
||||||
|
def __init__(self, client):
|
||||||
|
self.msg_id = ''
|
||||||
|
self.incoming_message = None
|
||||||
|
self.client = client # 用于更新 DingTalkClient 中的 incoming_message
|
||||||
|
|
||||||
|
"""处理钉钉消息"""
|
||||||
|
async def process(self, callback: dingtalk_stream.CallbackMessage):
|
||||||
|
incoming_message = dingtalk_stream.ChatbotMessage.from_dict(callback.data)
|
||||||
|
if incoming_message.message_id != self.msg_id:
|
||||||
|
self.msg_id = incoming_message.message_id
|
||||||
|
|
||||||
|
await self.client.update_incoming_message(incoming_message)
|
||||||
|
|
||||||
|
return AckMessage.STATUS_OK, 'OK'
|
||||||
|
|
||||||
|
async def get_incoming_message(self):
|
||||||
|
"""异步等待消息的到来"""
|
||||||
|
while self.incoming_message is None:
|
||||||
|
await asyncio.sleep(0.1) # 异步等待,避免阻塞
|
||||||
|
|
||||||
|
return self.incoming_message
|
||||||
|
|
||||||
|
async def get_dingtalk_client(client_id, client_secret):
|
||||||
|
from api import DingTalkClient # 延迟导入,避免循环导入
|
||||||
|
return DingTalkClient(client_id, client_secret)
|
||||||
255
libs/dingtalk_api/api.py
Normal file
255
libs/dingtalk_api/api.py
Normal file
@@ -0,0 +1,255 @@
|
|||||||
|
import base64
|
||||||
|
import json
|
||||||
|
import time
|
||||||
|
from typing import Callable
|
||||||
|
import dingtalk_stream
|
||||||
|
from .EchoHandler import EchoTextHandler
|
||||||
|
from .dingtalkevent import DingTalkEvent
|
||||||
|
import httpx
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
|
||||||
|
class DingTalkClient:
|
||||||
|
def __init__(self, client_id: str, client_secret: str,robot_name:str,robot_code:str,markdown_card:bool):
|
||||||
|
"""初始化 WebSocket 连接并自动启动"""
|
||||||
|
self.credential = dingtalk_stream.Credential(client_id, client_secret)
|
||||||
|
self.client = dingtalk_stream.DingTalkStreamClient(self.credential)
|
||||||
|
self.key = client_id
|
||||||
|
self.secret = client_secret
|
||||||
|
# 在 DingTalkClient 中传入自己作为参数,避免循环导入
|
||||||
|
self.EchoTextHandler = EchoTextHandler(self)
|
||||||
|
self.client.register_callback_handler(dingtalk_stream.chatbot.ChatbotMessage.TOPIC, self.EchoTextHandler)
|
||||||
|
self._message_handlers = {
|
||||||
|
"example":[],
|
||||||
|
}
|
||||||
|
self.access_token = ''
|
||||||
|
self.robot_name = robot_name
|
||||||
|
self.robot_code = robot_code
|
||||||
|
self.access_token_expiry_time = ''
|
||||||
|
self.markdown_card = markdown_card
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async def get_access_token(self):
|
||||||
|
url = "https://api.dingtalk.com/v1.0/oauth2/accessToken"
|
||||||
|
headers = {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"appKey": self.key,
|
||||||
|
"appSecret": self.secret
|
||||||
|
}
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
try:
|
||||||
|
response = await client.post(url,json=data,headers=headers)
|
||||||
|
if response.status_code == 200:
|
||||||
|
response_data = response.json()
|
||||||
|
self.access_token = response_data.get("accessToken")
|
||||||
|
expires_in = int(response_data.get("expireIn",7200))
|
||||||
|
self.access_token_expiry_time = time.time() + expires_in - 60
|
||||||
|
except Exception as e:
|
||||||
|
raise Exception(e)
|
||||||
|
|
||||||
|
|
||||||
|
async def is_token_expired(self):
|
||||||
|
"""检查token是否过期"""
|
||||||
|
if self.access_token_expiry_time is None:
|
||||||
|
return True
|
||||||
|
return time.time() > self.access_token_expiry_time
|
||||||
|
|
||||||
|
async def check_access_token(self):
|
||||||
|
if not self.access_token or await self.is_token_expired():
|
||||||
|
return False
|
||||||
|
return bool(self.access_token and self.access_token.strip())
|
||||||
|
|
||||||
|
async def download_image(self,download_code:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
url = 'https://api.dingtalk.com/v1.0/robot/messageFiles/download'
|
||||||
|
params = {
|
||||||
|
"downloadCode":download_code,
|
||||||
|
"robotCode":self.robot_code
|
||||||
|
}
|
||||||
|
headers ={
|
||||||
|
"x-acs-dingtalk-access-token": self.access_token
|
||||||
|
}
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url, headers=headers, json=params)
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
download_url = result.get("downloadUrl")
|
||||||
|
else:
|
||||||
|
raise Exception(f"Error: {response.status_code}, {response.text}")
|
||||||
|
|
||||||
|
if download_url:
|
||||||
|
return await self.download_url_to_base64(download_url)
|
||||||
|
|
||||||
|
async def download_url_to_base64(self,download_url):
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.get(download_url)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
|
||||||
|
file_bytes = response.content
|
||||||
|
base64_str = base64.b64encode(file_bytes).decode('utf-8') # 返回字符串格式
|
||||||
|
return base64_str
|
||||||
|
else:
|
||||||
|
raise Exception("获取文件失败")
|
||||||
|
|
||||||
|
async def get_audio_url(self,download_code:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
url = 'https://api.dingtalk.com/v1.0/robot/messageFiles/download'
|
||||||
|
params = {
|
||||||
|
"downloadCode":download_code,
|
||||||
|
"robotCode":self.robot_code
|
||||||
|
}
|
||||||
|
headers ={
|
||||||
|
"x-acs-dingtalk-access-token": self.access_token
|
||||||
|
}
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url, headers=headers, json=params)
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
download_url = result.get("downloadUrl")
|
||||||
|
if download_url:
|
||||||
|
return await self.download_url_to_base64(download_url)
|
||||||
|
else:
|
||||||
|
raise Exception("获取音频失败")
|
||||||
|
else:
|
||||||
|
raise Exception(f"Error: {response.status_code}, {response.text}")
|
||||||
|
|
||||||
|
async def update_incoming_message(self, message):
|
||||||
|
"""异步更新 DingTalkClient 中的 incoming_message"""
|
||||||
|
message_data = await self.get_message(message)
|
||||||
|
if message_data:
|
||||||
|
event = DingTalkEvent.from_payload(message_data)
|
||||||
|
if event:
|
||||||
|
await self._handle_message(event)
|
||||||
|
|
||||||
|
|
||||||
|
async def send_message(self,content:str,incoming_message):
|
||||||
|
if self.markdown_card:
|
||||||
|
self.EchoTextHandler.reply_markdown(title=self.robot_name+'的回答',text=content,incoming_message=incoming_message)
|
||||||
|
else:
|
||||||
|
self.EchoTextHandler.reply_text(content,incoming_message)
|
||||||
|
|
||||||
|
|
||||||
|
async def get_incoming_message(self):
|
||||||
|
"""获取收到的消息"""
|
||||||
|
return await self.EchoTextHandler.get_incoming_message()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
def decorator(func: Callable[[DingTalkEvent], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def _handle_message(self, event: DingTalkEvent):
|
||||||
|
"""
|
||||||
|
处理消息事件。
|
||||||
|
"""
|
||||||
|
msg_type = event.conversation
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
|
||||||
|
async def get_message(self,incoming_message:dingtalk_stream.chatbot.ChatbotMessage):
|
||||||
|
try:
|
||||||
|
|
||||||
|
# print(json.dumps(incoming_message.to_dict(), indent=4, ensure_ascii=False))
|
||||||
|
message_data = {
|
||||||
|
"IncomingMessage":incoming_message,
|
||||||
|
}
|
||||||
|
if str(incoming_message.conversation_type) == '1':
|
||||||
|
message_data["conversation_type"] = 'FriendMessage'
|
||||||
|
elif str(incoming_message.conversation_type) == '2':
|
||||||
|
message_data["conversation_type"] = 'GroupMessage'
|
||||||
|
|
||||||
|
|
||||||
|
if incoming_message.message_type == 'richText':
|
||||||
|
|
||||||
|
data = incoming_message.rich_text_content.to_dict()
|
||||||
|
for item in data['richText']:
|
||||||
|
if 'text' in item:
|
||||||
|
message_data["Content"] = item['text']
|
||||||
|
if incoming_message.get_image_list()[0]:
|
||||||
|
message_data["Picture"] = await self.download_image(incoming_message.get_image_list()[0])
|
||||||
|
message_data["Type"] = 'text'
|
||||||
|
|
||||||
|
elif incoming_message.message_type == 'text':
|
||||||
|
message_data['Content'] = incoming_message.get_text_list()[0]
|
||||||
|
|
||||||
|
message_data["Type"] = 'text'
|
||||||
|
elif incoming_message.message_type == 'picture':
|
||||||
|
message_data['Picture'] = await self.download_image(incoming_message.get_image_list()[0])
|
||||||
|
|
||||||
|
message_data['Type'] = 'image'
|
||||||
|
elif incoming_message.message_type == 'audio':
|
||||||
|
message_data['Audio'] = await self.get_audio_url(incoming_message.to_dict()['content']['downloadCode'])
|
||||||
|
|
||||||
|
message_data['Type'] = 'audio'
|
||||||
|
|
||||||
|
copy_message_data = message_data.copy()
|
||||||
|
del copy_message_data['IncomingMessage']
|
||||||
|
# print("message_data:", json.dumps(copy_message_data, indent=4, ensure_ascii=False))
|
||||||
|
except Exception:
|
||||||
|
traceback.print_exc()
|
||||||
|
|
||||||
|
return message_data
|
||||||
|
|
||||||
|
async def send_proactive_message_to_one(self,target_id:str,content:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
|
||||||
|
url = 'https://api.dingtalk.com/v1.0/robot/oToMessages/batchSend'
|
||||||
|
|
||||||
|
headers ={
|
||||||
|
"x-acs-dingtalk-access-token":self.access_token,
|
||||||
|
"Content-Type":"application/json",
|
||||||
|
}
|
||||||
|
|
||||||
|
data ={
|
||||||
|
"robotCode":self.robot_code,
|
||||||
|
"userIds":[target_id],
|
||||||
|
"msgKey": "sampleText",
|
||||||
|
"msgParam": json.dumps({"content":content}),
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url,headers=headers,json=data)
|
||||||
|
except Exception:
|
||||||
|
traceback.print_exc()
|
||||||
|
|
||||||
|
|
||||||
|
async def send_proactive_message_to_group(self,target_id:str,content:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
|
||||||
|
url = 'https://api.dingtalk.com/v1.0/robot/groupMessages/send'
|
||||||
|
|
||||||
|
headers ={
|
||||||
|
"x-acs-dingtalk-access-token":self.access_token,
|
||||||
|
"Content-Type":"application/json",
|
||||||
|
}
|
||||||
|
|
||||||
|
data ={
|
||||||
|
"robotCode":self.robot_code,
|
||||||
|
"openConversationId":target_id,
|
||||||
|
"msgKey": "sampleText",
|
||||||
|
"msgParam": json.dumps({"content":content}),
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url,headers=headers,json=data)
|
||||||
|
except Exception:
|
||||||
|
traceback.print_exc()
|
||||||
|
|
||||||
|
async def start(self):
|
||||||
|
"""启动 WebSocket 连接,监听消息"""
|
||||||
|
await self.client.start()
|
||||||
69
libs/dingtalk_api/dingtalkevent.py
Normal file
69
libs/dingtalk_api/dingtalkevent.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
from typing import Dict, Any, Optional
|
||||||
|
import dingtalk_stream
|
||||||
|
|
||||||
|
class DingTalkEvent(dict):
|
||||||
|
@staticmethod
|
||||||
|
def from_payload(payload: Dict[str, Any]) -> Optional["DingTalkEvent"]:
|
||||||
|
try:
|
||||||
|
event = DingTalkEvent(payload)
|
||||||
|
return event
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def content(self):
|
||||||
|
return self.get("Content","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def incoming_message(self) -> Optional["dingtalk_stream.chatbot.ChatbotMessage"]:
|
||||||
|
return self.get("IncomingMessage")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self):
|
||||||
|
return self.get("Type","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def picture(self):
|
||||||
|
return self.get("Picture","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def audio(self):
|
||||||
|
return self.get("Audio","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def conversation(self):
|
||||||
|
return self.get("conversation_type","")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def __getattr__(self, key: str) -> Optional[Any]:
|
||||||
|
"""
|
||||||
|
允许通过属性访问数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[Any]: 字段值。
|
||||||
|
"""
|
||||||
|
return self.get(key)
|
||||||
|
|
||||||
|
def __setattr__(self, key: str, value: Any) -> None:
|
||||||
|
"""
|
||||||
|
允许通过属性设置数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
value (Any): 字段值。
|
||||||
|
"""
|
||||||
|
self[key] = value
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
"""
|
||||||
|
生成事件对象的字符串表示。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 字符串表示。
|
||||||
|
"""
|
||||||
|
return f"<DingTalkEvent {super().__repr__()}>"
|
||||||
340
libs/official_account_api/api.py
Normal file
340
libs/official_account_api/api.py
Normal file
@@ -0,0 +1,340 @@
|
|||||||
|
# 微信公众号的加解密算法与企业微信一样,所以直接使用企业微信的加解密算法文件
|
||||||
|
from collections import deque
|
||||||
|
import time
|
||||||
|
import traceback
|
||||||
|
from ..wecom_api.WXBizMsgCrypt3 import WXBizMsgCrypt
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from quart import Quart,request
|
||||||
|
import hashlib
|
||||||
|
from typing import Callable, Dict, Any
|
||||||
|
from .oaevent import OAEvent
|
||||||
|
import httpx
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
import time
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from pkg.platform.sources import officialaccount as oa
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
xml_template = """
|
||||||
|
<xml>
|
||||||
|
<ToUserName><![CDATA[{to_user}]]></ToUserName>
|
||||||
|
<FromUserName><![CDATA[{from_user}]]></FromUserName>
|
||||||
|
<CreateTime>{create_time}</CreateTime>
|
||||||
|
<MsgType><![CDATA[text]]></MsgType>
|
||||||
|
<Content><![CDATA[{content}]]></Content>
|
||||||
|
</xml>
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class OAClient():
|
||||||
|
|
||||||
|
def __init__(self,token:str,EncodingAESKey:str,AppID:str,Appsecret:str):
|
||||||
|
self.token = token
|
||||||
|
self.aes = EncodingAESKey
|
||||||
|
self.appid = AppID
|
||||||
|
self.appsecret = Appsecret
|
||||||
|
self.base_url = 'https://api.weixin.qq.com'
|
||||||
|
self.access_token = ''
|
||||||
|
self.app = Quart(__name__)
|
||||||
|
self.app.add_url_rule('/callback/command', 'handle_callback', self.handle_callback_request, methods=['GET', 'POST'])
|
||||||
|
self._message_handlers = {
|
||||||
|
"example":[],
|
||||||
|
}
|
||||||
|
self.access_token_expiry_time = None
|
||||||
|
self.msg_id_map = {}
|
||||||
|
self.generated_content = {}
|
||||||
|
|
||||||
|
async def handle_callback_request(self):
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 每隔100毫秒查询是否生成ai回答
|
||||||
|
start_time = time.time()
|
||||||
|
signature = request.args.get("signature", "")
|
||||||
|
timestamp = request.args.get("timestamp", "")
|
||||||
|
nonce = request.args.get("nonce", "")
|
||||||
|
echostr = request.args.get("echostr", "")
|
||||||
|
msg_signature = request.args.get("msg_signature","")
|
||||||
|
if msg_signature is None:
|
||||||
|
raise Exception("msg_signature不在请求体中")
|
||||||
|
|
||||||
|
if request.method == 'GET':
|
||||||
|
# 校验签名
|
||||||
|
check_str = "".join(sorted([self.token, timestamp, nonce]))
|
||||||
|
check_signature = hashlib.sha1(check_str.encode("utf-8")).hexdigest()
|
||||||
|
|
||||||
|
if check_signature == signature:
|
||||||
|
return echostr # 验证成功返回echostr
|
||||||
|
else:
|
||||||
|
raise Exception("拒绝请求")
|
||||||
|
elif request.method == "POST":
|
||||||
|
encryt_msg = await request.data
|
||||||
|
wxcpt = WXBizMsgCrypt(self.token,self.aes,self.appid)
|
||||||
|
ret,xml_msg = wxcpt.DecryptMsg(encryt_msg,msg_signature,timestamp,nonce)
|
||||||
|
xml_msg = xml_msg.decode('utf-8')
|
||||||
|
|
||||||
|
if ret != 0:
|
||||||
|
raise Exception("消息解密失败")
|
||||||
|
|
||||||
|
message_data = await self.get_message(xml_msg)
|
||||||
|
if message_data :
|
||||||
|
event = OAEvent.from_payload(message_data)
|
||||||
|
if event:
|
||||||
|
await self._handle_message(event)
|
||||||
|
|
||||||
|
root = ET.fromstring(xml_msg)
|
||||||
|
from_user = root.find("FromUserName").text # 发送者
|
||||||
|
to_user = root.find("ToUserName").text # 机器人
|
||||||
|
|
||||||
|
timeout = 4.80
|
||||||
|
interval = 0.1
|
||||||
|
while True:
|
||||||
|
content = self.generated_content.pop(message_data["MsgId"], None)
|
||||||
|
if content:
|
||||||
|
response_xml = xml_template.format(
|
||||||
|
to_user=from_user,
|
||||||
|
from_user=to_user,
|
||||||
|
create_time=int(time.time()),
|
||||||
|
content = content
|
||||||
|
)
|
||||||
|
|
||||||
|
return response_xml
|
||||||
|
|
||||||
|
if time.time() - start_time >= timeout:
|
||||||
|
break
|
||||||
|
|
||||||
|
await asyncio.sleep(interval)
|
||||||
|
|
||||||
|
if self.msg_id_map.get(message_data["MsgId"], 1) == 3:
|
||||||
|
|
||||||
|
# response_xml = xml_template.format(
|
||||||
|
# to_user=from_user,
|
||||||
|
# from_user=to_user,
|
||||||
|
# create_time=int(time.time()),
|
||||||
|
# content = "请求失效:暂不支持公众号超过15秒的请求,如有需求,请联系 LangBot 团队。"
|
||||||
|
# )
|
||||||
|
print("请求失效:暂不支持公众号超过15秒的请求,如有需求,请联系 LangBot 团队。")
|
||||||
|
return ''
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
|
||||||
|
|
||||||
|
async def get_message(self, xml_msg: str):
|
||||||
|
|
||||||
|
root = ET.fromstring(xml_msg)
|
||||||
|
|
||||||
|
message_data = {
|
||||||
|
"ToUserName": root.find("ToUserName").text,
|
||||||
|
"FromUserName": root.find("FromUserName").text,
|
||||||
|
"CreateTime": int(root.find("CreateTime").text),
|
||||||
|
"MsgType": root.find("MsgType").text,
|
||||||
|
"Content": root.find("Content").text if root.find("Content") is not None else None,
|
||||||
|
"MsgId": int(root.find("MsgId").text) if root.find("MsgId") is not None else None,
|
||||||
|
}
|
||||||
|
|
||||||
|
return message_data
|
||||||
|
|
||||||
|
|
||||||
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
启动 Quart 应用。
|
||||||
|
"""
|
||||||
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
"""
|
||||||
|
注册消息类型处理器。
|
||||||
|
"""
|
||||||
|
def decorator(func: Callable[[OAEvent], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def _handle_message(self, event: OAEvent):
|
||||||
|
"""
|
||||||
|
处理消息事件。
|
||||||
|
"""
|
||||||
|
message_id = event.message_id
|
||||||
|
if message_id in self.msg_id_map.keys():
|
||||||
|
self.msg_id_map[message_id] += 1
|
||||||
|
return
|
||||||
|
|
||||||
|
self.msg_id_map[message_id] = 1
|
||||||
|
msg_type = event.type
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
async def set_message(self,msg_id:int,content:str):
|
||||||
|
self.generated_content[msg_id] = content
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class OAClientForLongerResponse():
|
||||||
|
|
||||||
|
def __init__(self,token:str,EncodingAESKey:str,AppID:str,Appsecret:str,LoadingMessage:str):
|
||||||
|
self.token = token
|
||||||
|
self.aes = EncodingAESKey
|
||||||
|
self.appid = AppID
|
||||||
|
self.appsecret = Appsecret
|
||||||
|
self.base_url = 'https://api.weixin.qq.com'
|
||||||
|
self.access_token = ''
|
||||||
|
self.app = Quart(__name__)
|
||||||
|
self.app.add_url_rule('/callback/command', 'handle_callback', self.handle_callback_request, methods=['GET', 'POST'])
|
||||||
|
self._message_handlers = {
|
||||||
|
"example":[],
|
||||||
|
}
|
||||||
|
self.access_token_expiry_time = None
|
||||||
|
self.loading_message = LoadingMessage
|
||||||
|
self.msg_queue = {}
|
||||||
|
self.user_msg_queue = {}
|
||||||
|
|
||||||
|
async def handle_callback_request(self):
|
||||||
|
try:
|
||||||
|
start_time = time.time()
|
||||||
|
signature = request.args.get("signature", "")
|
||||||
|
timestamp = request.args.get("timestamp", "")
|
||||||
|
nonce = request.args.get("nonce", "")
|
||||||
|
echostr = request.args.get("echostr", "")
|
||||||
|
msg_signature = request.args.get("msg_signature", "")
|
||||||
|
|
||||||
|
if msg_signature is None:
|
||||||
|
raise Exception("msg_signature不在请求体中")
|
||||||
|
|
||||||
|
if request.method == 'GET':
|
||||||
|
check_str = "".join(sorted([self.token, timestamp, nonce]))
|
||||||
|
check_signature = hashlib.sha1(check_str.encode("utf-8")).hexdigest()
|
||||||
|
return echostr if check_signature == signature else "拒绝请求"
|
||||||
|
|
||||||
|
elif request.method == "POST":
|
||||||
|
encryt_msg = await request.data
|
||||||
|
wxcpt = WXBizMsgCrypt(self.token, self.aes, self.appid)
|
||||||
|
ret, xml_msg = wxcpt.DecryptMsg(encryt_msg, msg_signature, timestamp, nonce)
|
||||||
|
xml_msg = xml_msg.decode('utf-8')
|
||||||
|
|
||||||
|
if ret != 0:
|
||||||
|
raise Exception("消息解密失败")
|
||||||
|
|
||||||
|
# 解析 XML
|
||||||
|
root = ET.fromstring(xml_msg)
|
||||||
|
from_user = root.find("FromUserName").text
|
||||||
|
to_user = root.find("ToUserName").text
|
||||||
|
|
||||||
|
|
||||||
|
if self.msg_queue.get(from_user) and self.msg_queue[from_user][0]["content"]:
|
||||||
|
|
||||||
|
queue_top = self.msg_queue[from_user].pop(0)
|
||||||
|
queue_content = queue_top["content"]
|
||||||
|
|
||||||
|
# 弹出用户消息
|
||||||
|
if self.user_msg_queue.get(from_user) and self.user_msg_queue[from_user]:
|
||||||
|
self.user_msg_queue[from_user].pop(0)
|
||||||
|
|
||||||
|
response_xml = xml_template.format(
|
||||||
|
to_user=from_user,
|
||||||
|
from_user=to_user,
|
||||||
|
create_time=int(time.time()),
|
||||||
|
content=queue_content
|
||||||
|
)
|
||||||
|
return response_xml
|
||||||
|
|
||||||
|
else:
|
||||||
|
response_xml = xml_template.format(
|
||||||
|
to_user=from_user,
|
||||||
|
from_user=to_user,
|
||||||
|
create_time=int(time.time()),
|
||||||
|
content=self.loading_message
|
||||||
|
)
|
||||||
|
|
||||||
|
if self.user_msg_queue.get(from_user) and self.user_msg_queue[from_user][0]["content"]:
|
||||||
|
return response_xml
|
||||||
|
else:
|
||||||
|
message_data = await self.get_message(xml_msg)
|
||||||
|
|
||||||
|
if message_data:
|
||||||
|
event = OAEvent.from_payload(message_data)
|
||||||
|
if event:
|
||||||
|
self.user_msg_queue.setdefault(from_user,[]).append(
|
||||||
|
{
|
||||||
|
"content":event.message,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
await self._handle_message(event)
|
||||||
|
|
||||||
|
return response_xml
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async def get_message(self, xml_msg: str):
|
||||||
|
|
||||||
|
root = ET.fromstring(xml_msg)
|
||||||
|
|
||||||
|
message_data = {
|
||||||
|
"ToUserName": root.find("ToUserName").text,
|
||||||
|
"FromUserName": root.find("FromUserName").text,
|
||||||
|
"CreateTime": int(root.find("CreateTime").text),
|
||||||
|
"MsgType": root.find("MsgType").text,
|
||||||
|
"Content": root.find("Content").text if root.find("Content") is not None else None,
|
||||||
|
"MsgId": int(root.find("MsgId").text) if root.find("MsgId") is not None else None,
|
||||||
|
}
|
||||||
|
|
||||||
|
return message_data
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
启动 Quart 应用。
|
||||||
|
"""
|
||||||
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
"""
|
||||||
|
注册消息类型处理器。
|
||||||
|
"""
|
||||||
|
def decorator(func: Callable[[OAEvent], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def _handle_message(self, event: OAEvent):
|
||||||
|
"""
|
||||||
|
处理消息事件。
|
||||||
|
"""
|
||||||
|
|
||||||
|
msg_type = event.type
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
async def set_message(self,from_user:int,message_id:int,content:str):
|
||||||
|
if from_user not in self.msg_queue:
|
||||||
|
self.msg_queue[from_user] = []
|
||||||
|
|
||||||
|
self.msg_queue[from_user].append(
|
||||||
|
{
|
||||||
|
"msg_id":message_id,
|
||||||
|
"content":content,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
167
libs/official_account_api/oaevent.py
Normal file
167
libs/official_account_api/oaevent.py
Normal file
@@ -0,0 +1,167 @@
|
|||||||
|
from typing import Dict, Any, Optional
|
||||||
|
|
||||||
|
|
||||||
|
class OAEvent(dict):
|
||||||
|
"""
|
||||||
|
封装从微信公众号收到的事件数据对象(字典),提供属性以获取其中的字段。
|
||||||
|
|
||||||
|
除 `type` 和 `detail_type` 属性对于任何事件都有效外,其它属性是否存在(若不存在则返回 `None`)依事件类型不同而不同。
|
||||||
|
"""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def from_payload(payload: Dict[str, Any]) -> Optional["OAEvent"]:
|
||||||
|
"""
|
||||||
|
从微信公众号事件数据构造 `WecomEvent` 对象。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
payload (Dict[str, Any]): 解密后的微信事件数据。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[OAEvent]: 如果事件数据合法,则返回 OAEvent 对象;否则返回 None。
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
event = OAEvent(payload)
|
||||||
|
_ = event.type, event.detail_type # 确保必须字段存在
|
||||||
|
return event
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self) -> str:
|
||||||
|
"""
|
||||||
|
事件类型,例如 "message"、"event"、"text" 等。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件类型。
|
||||||
|
"""
|
||||||
|
return self.get("MsgType", "")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def picurl(self) -> str:
|
||||||
|
"""
|
||||||
|
图片链接
|
||||||
|
"""
|
||||||
|
return self.get("PicUrl","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def detail_type(self) -> str:
|
||||||
|
"""
|
||||||
|
事件详细类型,依 `type` 的不同而不同。例如:
|
||||||
|
- 消息事件: "text", "image", "voice", 等
|
||||||
|
- 事件通知: "subscribe", "unsubscribe", "click", 等
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件详细类型。
|
||||||
|
"""
|
||||||
|
if self.type == "event":
|
||||||
|
return self.get("Event", "")
|
||||||
|
return self.type
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self) -> str:
|
||||||
|
"""
|
||||||
|
事件名,对于消息事件是 `type.detail_type`,对于其他事件是 `event_type`。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件名。
|
||||||
|
"""
|
||||||
|
return f"{self.type}.{self.detail_type}"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
发送方账号
|
||||||
|
"""
|
||||||
|
return self.get("FromUserName")
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def receiver_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
接收者 ID,例如机器人自身的公众号微信 ID。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 接收者 ID。
|
||||||
|
"""
|
||||||
|
return self.get("ToUserName")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
消息 ID,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 消息 ID。
|
||||||
|
"""
|
||||||
|
return self.get("MsgId")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
消息内容,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 消息内容。
|
||||||
|
"""
|
||||||
|
return self.get("Content")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def media_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
媒体文件 ID,仅在图片、语音等消息类型中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 媒体文件 ID。
|
||||||
|
"""
|
||||||
|
return self.get("MediaId")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def timestamp(self) -> Optional[int]:
|
||||||
|
"""
|
||||||
|
事件发生的时间戳。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[int]: 时间戳。
|
||||||
|
"""
|
||||||
|
return self.get("CreateTime")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def event_key(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
事件的 Key 值,例如点击菜单时的 `EventKey`。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 事件 Key。
|
||||||
|
"""
|
||||||
|
return self.get("EventKey")
|
||||||
|
|
||||||
|
def __getattr__(self, key: str) -> Optional[Any]:
|
||||||
|
"""
|
||||||
|
允许通过属性访问数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[Any]: 字段值。
|
||||||
|
"""
|
||||||
|
return self.get(key)
|
||||||
|
|
||||||
|
def __setattr__(self, key: str, value: Any) -> None:
|
||||||
|
"""
|
||||||
|
允许通过属性设置数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
value (Any): 字段值。
|
||||||
|
"""
|
||||||
|
self[key] = value
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
"""
|
||||||
|
生成事件对象的字符串表示。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 字符串表示。
|
||||||
|
"""
|
||||||
|
return f"<WecomEvent {super().__repr__()}>"
|
||||||
274
libs/qq_official_api/api.py
Normal file
274
libs/qq_official_api/api.py
Normal file
@@ -0,0 +1,274 @@
|
|||||||
|
import time
|
||||||
|
from quart import request
|
||||||
|
import base64
|
||||||
|
import binascii
|
||||||
|
import httpx
|
||||||
|
from quart import Quart
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from typing import Callable, Dict, Any
|
||||||
|
from pkg.platform.types import events as platform_events, message as platform_message
|
||||||
|
import aiofiles
|
||||||
|
from .qqofficialevent import QQOfficialEvent
|
||||||
|
import json
|
||||||
|
import hmac
|
||||||
|
import base64
|
||||||
|
import hashlib
|
||||||
|
import traceback
|
||||||
|
from cryptography.hazmat.primitives.asymmetric import ed25519
|
||||||
|
from .qqofficialevent import QQOfficialEvent
|
||||||
|
|
||||||
|
def handle_validation(body: dict, bot_secret: str):
|
||||||
|
|
||||||
|
# bot正确的secert是32位的,此处仅为了适配演示demo
|
||||||
|
while len(bot_secret) < 32:
|
||||||
|
bot_secret = bot_secret * 2
|
||||||
|
bot_secret = bot_secret[:32]
|
||||||
|
# 实际使用场景中以上三行内容可清除
|
||||||
|
|
||||||
|
seed_bytes = bot_secret.encode()
|
||||||
|
|
||||||
|
signing_key = ed25519.Ed25519PrivateKey.from_private_bytes(seed_bytes)
|
||||||
|
|
||||||
|
msg = body['d']['event_ts'] + body['d']['plain_token']
|
||||||
|
msg_bytes = msg.encode()
|
||||||
|
|
||||||
|
signature = signing_key.sign(msg_bytes)
|
||||||
|
|
||||||
|
signature_hex = signature.hex()
|
||||||
|
|
||||||
|
response = {
|
||||||
|
"plain_token": body['d']['plain_token'],
|
||||||
|
"signature": signature_hex
|
||||||
|
}
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
class QQOfficialClient:
|
||||||
|
def __init__(self, secret: str, token: str, app_id: str):
|
||||||
|
self.app = Quart(__name__)
|
||||||
|
self.app.add_url_rule(
|
||||||
|
"/callback/command",
|
||||||
|
"handle_callback",
|
||||||
|
self.handle_callback_request,
|
||||||
|
methods=["GET", "POST"],
|
||||||
|
)
|
||||||
|
self.secret = secret
|
||||||
|
self.token = token
|
||||||
|
self.app_id = app_id
|
||||||
|
self._message_handlers = {
|
||||||
|
}
|
||||||
|
self.base_url = "https://api.sgroup.qq.com"
|
||||||
|
self.access_token = ""
|
||||||
|
self.access_token_expiry_time = None
|
||||||
|
|
||||||
|
async def check_access_token(self):
|
||||||
|
"""检查access_token是否存在"""
|
||||||
|
if not self.access_token or await self.is_token_expired():
|
||||||
|
return False
|
||||||
|
return bool(self.access_token and self.access_token.strip())
|
||||||
|
|
||||||
|
async def get_access_token(self):
|
||||||
|
"""获取access_token"""
|
||||||
|
url = "https://bots.qq.com/app/getAppAccessToken"
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"appId":self.app_id,
|
||||||
|
"clientSecret":self.secret,
|
||||||
|
}
|
||||||
|
headers = {
|
||||||
|
"content-type":"application/json",
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
response = await client.post(url,json=params,headers=headers)
|
||||||
|
if response.status_code == 200:
|
||||||
|
response_data = response.json()
|
||||||
|
access_token = response_data.get("access_token")
|
||||||
|
expires_in = int(response_data.get("expires_in",7200))
|
||||||
|
self.access_token_expiry_time = time.time() + expires_in - 60
|
||||||
|
if access_token:
|
||||||
|
self.access_token = access_token
|
||||||
|
except Exception as e:
|
||||||
|
raise Exception(f"获取access_token失败: {e}")
|
||||||
|
|
||||||
|
|
||||||
|
async def handle_callback_request(self):
|
||||||
|
"""处理回调请求"""
|
||||||
|
try:
|
||||||
|
# 读取请求数据
|
||||||
|
body = await request.get_data()
|
||||||
|
payload = json.loads(body)
|
||||||
|
|
||||||
|
|
||||||
|
# 验证是否为回调验证请求
|
||||||
|
if payload.get("op") == 13:
|
||||||
|
# 生成签名
|
||||||
|
response = handle_validation(payload, self.secret)
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
if payload.get("op") == 0:
|
||||||
|
message_data = await self.get_message(payload)
|
||||||
|
if message_data:
|
||||||
|
event = QQOfficialEvent.from_payload(message_data)
|
||||||
|
await self._handle_message(event)
|
||||||
|
|
||||||
|
return {"code": 0, "message": "success"}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
return {"error": str(e)}, 400
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
||||||
|
"""启动 Quart 应用"""
|
||||||
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
"""注册消息类型处理器"""
|
||||||
|
|
||||||
|
def decorator(func: Callable[[platform_events.Event], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def _handle_message(self, event:QQOfficialEvent):
|
||||||
|
"""处理消息事件"""
|
||||||
|
msg_type = event.t
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
|
||||||
|
async def get_message(self,msg:dict) -> Dict[str,Any]:
|
||||||
|
"""获取消息"""
|
||||||
|
message_data = {
|
||||||
|
"t": msg.get("t",{}),
|
||||||
|
"user_openid": msg.get("d",{}).get("author",{}).get("user_openid",{}),
|
||||||
|
"timestamp": msg.get("d",{}).get("timestamp",{}),
|
||||||
|
"d_author_id": msg.get("d",{}).get("author",{}).get("id",{}),
|
||||||
|
"content": msg.get("d",{}).get("content",{}),
|
||||||
|
"d_id": msg.get("d",{}).get("id",{}),
|
||||||
|
"id": msg.get("id",{}),
|
||||||
|
"channel_id": msg.get("d",{}).get("channel_id",{}),
|
||||||
|
"username": msg.get("d",{}).get("author",{}).get("username",{}),
|
||||||
|
"guild_id": msg.get("d",{}).get("guild_id",{}),
|
||||||
|
"member_openid": msg.get("d",{}).get("author",{}).get("openid",{}),
|
||||||
|
"group_openid": msg.get("d",{}).get("group_openid",{})
|
||||||
|
}
|
||||||
|
attachments = msg.get("d", {}).get("attachments", [])
|
||||||
|
image_attachments = [attachment['url'] for attachment in attachments if await self.is_image(attachment)]
|
||||||
|
image_attachments_type = [attachment['content_type'] for attachment in attachments if await self.is_image(attachment)]
|
||||||
|
if image_attachments:
|
||||||
|
message_data["image_attachments"] = image_attachments[0]
|
||||||
|
message_data["content_type"] = image_attachments_type[0]
|
||||||
|
else:
|
||||||
|
|
||||||
|
message_data["image_attachments"] = None
|
||||||
|
|
||||||
|
return message_data
|
||||||
|
|
||||||
|
|
||||||
|
async def is_image(self,attachment:dict) -> bool:
|
||||||
|
"""判断是否为图片附件"""
|
||||||
|
content_type = attachment.get("content_type","")
|
||||||
|
return content_type.startswith("image/")
|
||||||
|
|
||||||
|
|
||||||
|
async def send_private_text_msg(self,user_openid:str,content:str,msg_id:str):
|
||||||
|
"""发送私聊消息"""
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
|
||||||
|
url = self.base_url + "/v2/users/" + user_openid + "/messages"
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
headers = {
|
||||||
|
"Authorization": f"QQBot {self.access_token}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"content": content,
|
||||||
|
"msg_type": 0,
|
||||||
|
"msg_id": msg_id,
|
||||||
|
}
|
||||||
|
response = await client.post(url,headers=headers,json=data)
|
||||||
|
if response.status_code == 200:
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
raise ValueError(response)
|
||||||
|
|
||||||
|
|
||||||
|
async def send_group_text_msg(self,group_openid:str,content:str,msg_id:str):
|
||||||
|
"""发送群聊消息"""
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
|
||||||
|
url = self.base_url + "/v2/groups/" + group_openid + "/messages"
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
headers = {
|
||||||
|
"Authorization": f"QQBot {self.access_token}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
data = {
|
||||||
|
"content": content,
|
||||||
|
"msg_type": 0,
|
||||||
|
"msg_id": msg_id,
|
||||||
|
}
|
||||||
|
response = await client.post(url,headers=headers,json=data)
|
||||||
|
if response.status_code == 200:
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
raise Exception(response.read().decode())
|
||||||
|
|
||||||
|
async def send_channle_group_text_msg(self,channel_id:str,content:str,msg_id:str):
|
||||||
|
"""发送频道群聊消息"""
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
|
||||||
|
url = self.base_url + "/channels/" + channel_id + "/messages"
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
headers = {
|
||||||
|
"Authorization": f"QQBot {self.access_token}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
params = {
|
||||||
|
"content": content,
|
||||||
|
"msg_type": 0,
|
||||||
|
"msg_id": msg_id,
|
||||||
|
}
|
||||||
|
response = await client.post(url,headers=headers,json=params)
|
||||||
|
if response.status_code == 200:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
raise Exception(response)
|
||||||
|
|
||||||
|
async def send_channle_private_text_msg(self,guild_id:str,content:str,msg_id:str):
|
||||||
|
"""发送频道私聊消息"""
|
||||||
|
if not await self.check_access_token():
|
||||||
|
await self.get_access_token()
|
||||||
|
|
||||||
|
url = self.base_url + "/dms/" + guild_id + "/messages"
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
headers = {
|
||||||
|
"Authorization": f"QQBot {self.access_token}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
params = {
|
||||||
|
"content": content,
|
||||||
|
"msg_type": 0,
|
||||||
|
"msg_id": msg_id,
|
||||||
|
}
|
||||||
|
response = await client.post(url,headers=headers,json=params)
|
||||||
|
if response.status_code == 200:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
raise Exception(response)
|
||||||
|
|
||||||
|
async def is_token_expired(self):
|
||||||
|
"""检查token是否过期"""
|
||||||
|
if self.access_token_expiry_time is None:
|
||||||
|
return True
|
||||||
|
return time.time() > self.access_token_expiry_time
|
||||||
114
libs/qq_official_api/qqofficialevent.py
Normal file
114
libs/qq_official_api/qqofficialevent.py
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
from typing import Dict, Any, Optional
|
||||||
|
|
||||||
|
class QQOfficialEvent(dict):
|
||||||
|
@staticmethod
|
||||||
|
def from_payload(payload: Dict[str, Any]) -> Optional["QQOfficialEvent"]:
|
||||||
|
try:
|
||||||
|
event = QQOfficialEvent(payload)
|
||||||
|
return event
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def t(self) -> str:
|
||||||
|
"""
|
||||||
|
事件类型
|
||||||
|
"""
|
||||||
|
return self.get("t", "")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user_openid(self) -> str:
|
||||||
|
"""
|
||||||
|
用户openid
|
||||||
|
"""
|
||||||
|
return self.get("user_openid",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def timestamp(self) -> str:
|
||||||
|
"""
|
||||||
|
时间戳
|
||||||
|
"""
|
||||||
|
return self.get("timestamp",{})
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def d_author_id(self) -> str:
|
||||||
|
"""
|
||||||
|
作者id
|
||||||
|
"""
|
||||||
|
return self.get("id",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def content(self) -> str:
|
||||||
|
"""
|
||||||
|
内容
|
||||||
|
"""
|
||||||
|
return self.get("content",'')
|
||||||
|
|
||||||
|
@property
|
||||||
|
def d_id(self) -> str:
|
||||||
|
"""
|
||||||
|
d_id
|
||||||
|
"""
|
||||||
|
return self.get("d_id",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def id(self) -> str:
|
||||||
|
"""
|
||||||
|
消息id,msg_id
|
||||||
|
"""
|
||||||
|
return self.get("id",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channel_id(self) -> str:
|
||||||
|
"""
|
||||||
|
频道id
|
||||||
|
"""
|
||||||
|
return self.get("channel_id",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def username(self) -> str:
|
||||||
|
"""
|
||||||
|
用户名
|
||||||
|
"""
|
||||||
|
return self.get("username",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def guild_id(self) -> str:
|
||||||
|
"""
|
||||||
|
频道id
|
||||||
|
"""
|
||||||
|
return self.get("guild_id",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def member_openid(self) -> str:
|
||||||
|
"""
|
||||||
|
成员openid
|
||||||
|
"""
|
||||||
|
return self.get("openid",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def attachments(self) -> str:
|
||||||
|
"""
|
||||||
|
附件url
|
||||||
|
"""
|
||||||
|
url = self.get("image_attachments", "")
|
||||||
|
if url and not url.startswith("https://"):
|
||||||
|
url = "https://" + url
|
||||||
|
return url
|
||||||
|
|
||||||
|
@property
|
||||||
|
def group_openid(self) -> str:
|
||||||
|
"""
|
||||||
|
群组id
|
||||||
|
"""
|
||||||
|
return self.get("group_openid",{})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def content_type(self) -> str:
|
||||||
|
"""
|
||||||
|
文件类型
|
||||||
|
"""
|
||||||
|
return self.get("content_type","")
|
||||||
|
|
||||||
111
libs/slack_api/api.py
Normal file
111
libs/slack_api/api.py
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
import json
|
||||||
|
from quart import Quart, jsonify,request
|
||||||
|
from slack_sdk.web.async_client import AsyncWebClient
|
||||||
|
from .slackevent import SlackEvent
|
||||||
|
from typing import Callable, Dict, Any
|
||||||
|
from pkg.platform.types import events as platform_events, message as platform_message
|
||||||
|
|
||||||
|
class SlackClient():
|
||||||
|
|
||||||
|
def __init__(self,bot_token:str,signing_secret:str):
|
||||||
|
|
||||||
|
self.bot_token = bot_token
|
||||||
|
self.signing_secret = signing_secret
|
||||||
|
self.app = Quart(__name__)
|
||||||
|
self.client = AsyncWebClient(self.bot_token)
|
||||||
|
self.app.add_url_rule('/callback/command', 'handle_callback', self.handle_callback_request, methods=['GET', 'POST'])
|
||||||
|
self._message_handlers = {
|
||||||
|
"example":[],
|
||||||
|
}
|
||||||
|
self.bot_user_id = None # 避免机器人回复自己的消息
|
||||||
|
|
||||||
|
async def handle_callback_request(self):
|
||||||
|
try:
|
||||||
|
body = await request.get_data()
|
||||||
|
data = json.loads(body)
|
||||||
|
if 'type' in data:
|
||||||
|
if data['type'] == 'url_verification':
|
||||||
|
return data['challenge']
|
||||||
|
|
||||||
|
bot_user_id = data.get("event",{}).get("bot_id","")
|
||||||
|
|
||||||
|
if self.bot_user_id and bot_user_id == self.bot_user_id:
|
||||||
|
return jsonify({'status': 'ok'})
|
||||||
|
|
||||||
|
|
||||||
|
# 处理私信
|
||||||
|
if data and data.get("event", {}).get("channel_type") in ["im"]:
|
||||||
|
event = SlackEvent.from_payload(data)
|
||||||
|
await self._handle_message(event)
|
||||||
|
return jsonify({'status': 'ok'})
|
||||||
|
|
||||||
|
#处理群聊
|
||||||
|
if data.get("event",{}).get("type") == 'app_mention':
|
||||||
|
data.setdefault("event", {})["channel_type"] = "channel"
|
||||||
|
event = SlackEvent.from_payload(data)
|
||||||
|
await self._handle_message(event)
|
||||||
|
return jsonify({'status':'ok'})
|
||||||
|
|
||||||
|
return jsonify({'status': 'ok'})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
raise(e)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async def _handle_message(self, event: SlackEvent):
|
||||||
|
"""
|
||||||
|
处理消息事件。
|
||||||
|
"""
|
||||||
|
msg_type = event.type
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
"""注册消息类型处理器"""
|
||||||
|
def decorator(func: Callable[[platform_events.Event], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def send_message_to_channel(self,text:str,channel_id:str):
|
||||||
|
try:
|
||||||
|
response = await self.client.chat_postMessage(
|
||||||
|
channel=channel_id,
|
||||||
|
text=text
|
||||||
|
)
|
||||||
|
if self.bot_user_id is None and response.get("ok"):
|
||||||
|
self.bot_user_id = response["message"]["bot_id"]
|
||||||
|
return
|
||||||
|
except Exception as e:
|
||||||
|
raise e
|
||||||
|
|
||||||
|
async def send_message_to_one(self,text:str,user_id:str):
|
||||||
|
try:
|
||||||
|
response = await self.client.chat_postMessage(
|
||||||
|
channel = '@'+user_id,
|
||||||
|
text= text
|
||||||
|
)
|
||||||
|
if self.bot_user_id is None and response.get("ok"):
|
||||||
|
self.bot_user_id = response["message"]["bot_id"]
|
||||||
|
|
||||||
|
return
|
||||||
|
except Exception as e:
|
||||||
|
raise e
|
||||||
|
|
||||||
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
启动 Quart 应用。
|
||||||
|
"""
|
||||||
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
91
libs/slack_api/slackevent.py
Normal file
91
libs/slack_api/slackevent.py
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
from typing import Dict, Any, Optional
|
||||||
|
|
||||||
|
class SlackEvent(dict):
|
||||||
|
@staticmethod
|
||||||
|
def from_payload(payload: Dict[str, Any]) -> Optional["SlackEvent"]:
|
||||||
|
try:
|
||||||
|
event = SlackEvent(payload)
|
||||||
|
return event
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def text(self) -> str:
|
||||||
|
|
||||||
|
if self.get("event", {}).get("channel_type") == "im":
|
||||||
|
blocks = self.get("event", {}).get("blocks", [])
|
||||||
|
if not blocks:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
elements = blocks[0].get("elements", [])
|
||||||
|
if not elements:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
elements = elements[0].get("elements", [])
|
||||||
|
text = ""
|
||||||
|
|
||||||
|
for el in elements:
|
||||||
|
if el.get("type") == "text":
|
||||||
|
text += el.get("text", "")
|
||||||
|
elif el.get("type") == "link":
|
||||||
|
text += el.get("url", "")
|
||||||
|
|
||||||
|
return text
|
||||||
|
|
||||||
|
|
||||||
|
if self.get("event",{}).get("channel_type") == 'channel':
|
||||||
|
message_text = ""
|
||||||
|
for block in self.get("event", {}).get("blocks", []):
|
||||||
|
if block.get("type") == "rich_text":
|
||||||
|
for element in block.get("elements", []):
|
||||||
|
if element.get("type") == "rich_text_section":
|
||||||
|
parts = []
|
||||||
|
for el in element.get("elements", []):
|
||||||
|
if el.get("type") == "text":
|
||||||
|
parts.append(el["text"])
|
||||||
|
elif el.get("type") == "link":
|
||||||
|
parts.append(el["url"])
|
||||||
|
message_text = "".join(parts)
|
||||||
|
|
||||||
|
return message_text
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user_id(self) -> Optional[str]:
|
||||||
|
return self.get("event", {}).get("user","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channel_id(self) -> Optional[str]:
|
||||||
|
return self.get("event", {}).get("channel","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self) -> str:
|
||||||
|
""" message对应私聊,app_mention对应频道at """
|
||||||
|
return self.get("event", {}).get("channel_type", "")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message_id(self) -> str:
|
||||||
|
return self.get("event_id","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def pic_url(self) -> str:
|
||||||
|
"""提取 Slack 事件中的图片 URL"""
|
||||||
|
files = self.get("event", {}).get("files", [])
|
||||||
|
if files:
|
||||||
|
return files[0].get("url_private", "")
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def sender_name(self) -> str:
|
||||||
|
return self.get("event", {}).get("user","")
|
||||||
|
|
||||||
|
def __getattr__(self, key: str) -> Optional[Any]:
|
||||||
|
return self.get(key)
|
||||||
|
|
||||||
|
def __setattr__(self, key: str, value: Any) -> None:
|
||||||
|
self[key] = value
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return f"<SlackEvent {super().__repr__()}>"
|
||||||
279
libs/wecom_api/WXBizMsgCrypt3.py
Normal file
279
libs/wecom_api/WXBizMsgCrypt3.py
Normal file
@@ -0,0 +1,279 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# -*- encoding:utf-8 -*-
|
||||||
|
|
||||||
|
""" 对企业微信发送给企业后台的消息加解密示例代码.
|
||||||
|
@copyright: Copyright (c) 1998-2014 Tencent Inc.
|
||||||
|
|
||||||
|
"""
|
||||||
|
# ------------------------------------------------------------------------
|
||||||
|
import logging
|
||||||
|
import base64
|
||||||
|
import random
|
||||||
|
import hashlib
|
||||||
|
import time
|
||||||
|
import struct
|
||||||
|
from Crypto.Cipher import AES
|
||||||
|
import xml.etree.cElementTree as ET
|
||||||
|
import socket
|
||||||
|
|
||||||
|
from . import ierror
|
||||||
|
|
||||||
|
|
||||||
|
"""
|
||||||
|
Crypto.Cipher包已不再维护,开发者可以通过以下命令下载安装最新版的加解密工具包
|
||||||
|
pip install pycryptodome
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class FormatException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def throw_exception(message, exception_class=FormatException):
|
||||||
|
"""my define raise exception function"""
|
||||||
|
raise exception_class(message)
|
||||||
|
|
||||||
|
|
||||||
|
class SHA1:
|
||||||
|
"""计算企业微信的消息签名接口"""
|
||||||
|
|
||||||
|
def getSHA1(self, token, timestamp, nonce, encrypt):
|
||||||
|
"""用SHA1算法生成安全签名
|
||||||
|
@param token: 票据
|
||||||
|
@param timestamp: 时间戳
|
||||||
|
@param encrypt: 密文
|
||||||
|
@param nonce: 随机字符串
|
||||||
|
@return: 安全签名
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
sortlist = [token, timestamp, nonce, encrypt]
|
||||||
|
sortlist.sort()
|
||||||
|
sha = hashlib.sha1()
|
||||||
|
sha.update("".join(sortlist).encode())
|
||||||
|
return ierror.WXBizMsgCrypt_OK, sha.hexdigest()
|
||||||
|
except Exception as e:
|
||||||
|
logger = logging.getLogger()
|
||||||
|
logger.error(e)
|
||||||
|
return ierror.WXBizMsgCrypt_ComputeSignature_Error, None
|
||||||
|
|
||||||
|
|
||||||
|
class XMLParse:
|
||||||
|
"""提供提取消息格式中的密文及生成回复消息格式的接口"""
|
||||||
|
|
||||||
|
# xml消息模板
|
||||||
|
AES_TEXT_RESPONSE_TEMPLATE = """<xml>
|
||||||
|
<Encrypt><![CDATA[%(msg_encrypt)s]]></Encrypt>
|
||||||
|
<MsgSignature><![CDATA[%(msg_signaturet)s]]></MsgSignature>
|
||||||
|
<TimeStamp>%(timestamp)s</TimeStamp>
|
||||||
|
<Nonce><![CDATA[%(nonce)s]]></Nonce>
|
||||||
|
</xml>"""
|
||||||
|
|
||||||
|
def extract(self, xmltext):
|
||||||
|
"""提取出xml数据包中的加密消息
|
||||||
|
@param xmltext: 待提取的xml字符串
|
||||||
|
@return: 提取出的加密消息字符串
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
xml_tree = ET.fromstring(xmltext)
|
||||||
|
encrypt = xml_tree.find("Encrypt")
|
||||||
|
return ierror.WXBizMsgCrypt_OK, encrypt.text
|
||||||
|
except Exception as e:
|
||||||
|
logger = logging.getLogger()
|
||||||
|
logger.error(e)
|
||||||
|
return ierror.WXBizMsgCrypt_ParseXml_Error, None
|
||||||
|
|
||||||
|
def generate(self, encrypt, signature, timestamp, nonce):
|
||||||
|
"""生成xml消息
|
||||||
|
@param encrypt: 加密后的消息密文
|
||||||
|
@param signature: 安全签名
|
||||||
|
@param timestamp: 时间戳
|
||||||
|
@param nonce: 随机字符串
|
||||||
|
@return: 生成的xml字符串
|
||||||
|
"""
|
||||||
|
resp_dict = {
|
||||||
|
'msg_encrypt': encrypt,
|
||||||
|
'msg_signaturet': signature,
|
||||||
|
'timestamp': timestamp,
|
||||||
|
'nonce': nonce,
|
||||||
|
}
|
||||||
|
resp_xml = self.AES_TEXT_RESPONSE_TEMPLATE % resp_dict
|
||||||
|
return resp_xml
|
||||||
|
|
||||||
|
|
||||||
|
class PKCS7Encoder():
|
||||||
|
"""提供基于PKCS7算法的加解密接口"""
|
||||||
|
|
||||||
|
block_size = 32
|
||||||
|
|
||||||
|
def encode(self, text):
|
||||||
|
""" 对需要加密的明文进行填充补位
|
||||||
|
@param text: 需要进行填充补位操作的明文
|
||||||
|
@return: 补齐明文字符串
|
||||||
|
"""
|
||||||
|
text_length = len(text)
|
||||||
|
# 计算需要填充的位数
|
||||||
|
amount_to_pad = self.block_size - (text_length % self.block_size)
|
||||||
|
if amount_to_pad == 0:
|
||||||
|
amount_to_pad = self.block_size
|
||||||
|
# 获得补位所用的字符
|
||||||
|
pad = chr(amount_to_pad)
|
||||||
|
return text + (pad * amount_to_pad).encode()
|
||||||
|
|
||||||
|
def decode(self, decrypted):
|
||||||
|
"""删除解密后明文的补位字符
|
||||||
|
@param decrypted: 解密后的明文
|
||||||
|
@return: 删除补位字符后的明文
|
||||||
|
"""
|
||||||
|
pad = ord(decrypted[-1])
|
||||||
|
if pad < 1 or pad > 32:
|
||||||
|
pad = 0
|
||||||
|
return decrypted[:-pad]
|
||||||
|
|
||||||
|
|
||||||
|
class Prpcrypt(object):
|
||||||
|
"""提供接收和推送给企业微信消息的加解密接口"""
|
||||||
|
|
||||||
|
def __init__(self, key):
|
||||||
|
|
||||||
|
# self.key = base64.b64decode(key+"=")
|
||||||
|
self.key = key
|
||||||
|
# 设置加解密模式为AES的CBC模式
|
||||||
|
self.mode = AES.MODE_CBC
|
||||||
|
|
||||||
|
def encrypt(self, text, receiveid):
|
||||||
|
"""对明文进行加密
|
||||||
|
@param text: 需要加密的明文
|
||||||
|
@return: 加密得到的字符串
|
||||||
|
"""
|
||||||
|
# 16位随机字符串添加到明文开头
|
||||||
|
text = text.encode()
|
||||||
|
text = self.get_random_str() + struct.pack("I", socket.htonl(len(text))) + text + receiveid.encode()
|
||||||
|
|
||||||
|
# 使用自定义的填充方式对明文进行补位填充
|
||||||
|
pkcs7 = PKCS7Encoder()
|
||||||
|
text = pkcs7.encode(text)
|
||||||
|
# 加密
|
||||||
|
cryptor = AES.new(self.key, self.mode, self.key[:16])
|
||||||
|
try:
|
||||||
|
ciphertext = cryptor.encrypt(text)
|
||||||
|
# 使用BASE64对加密后的字符串进行编码
|
||||||
|
return ierror.WXBizMsgCrypt_OK, base64.b64encode(ciphertext)
|
||||||
|
except Exception as e:
|
||||||
|
logger = logging.getLogger()
|
||||||
|
logger.error(e)
|
||||||
|
return ierror.WXBizMsgCrypt_EncryptAES_Error, None
|
||||||
|
|
||||||
|
def decrypt(self, text, receiveid):
|
||||||
|
"""对解密后的明文进行补位删除
|
||||||
|
@param text: 密文
|
||||||
|
@return: 删除填充补位后的明文
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
cryptor = AES.new(self.key, self.mode, self.key[:16])
|
||||||
|
# 使用BASE64对密文进行解码,然后AES-CBC解密
|
||||||
|
plain_text = cryptor.decrypt(base64.b64decode(text))
|
||||||
|
except Exception as e:
|
||||||
|
logger = logging.getLogger()
|
||||||
|
logger.error(e)
|
||||||
|
return ierror.WXBizMsgCrypt_DecryptAES_Error, None
|
||||||
|
try:
|
||||||
|
pad = plain_text[-1]
|
||||||
|
# 去掉补位字符串
|
||||||
|
# pkcs7 = PKCS7Encoder()
|
||||||
|
# plain_text = pkcs7.encode(plain_text)
|
||||||
|
# 去除16位随机字符串
|
||||||
|
content = plain_text[16:-pad]
|
||||||
|
xml_len = socket.ntohl(struct.unpack("I", content[: 4])[0])
|
||||||
|
xml_content = content[4: xml_len + 4]
|
||||||
|
from_receiveid = content[xml_len + 4:]
|
||||||
|
except Exception as e:
|
||||||
|
logger = logging.getLogger()
|
||||||
|
logger.error(e)
|
||||||
|
return ierror.WXBizMsgCrypt_IllegalBuffer, None
|
||||||
|
|
||||||
|
if from_receiveid.decode('utf8') != receiveid:
|
||||||
|
return ierror.WXBizMsgCrypt_ValidateCorpid_Error, None
|
||||||
|
return 0, xml_content
|
||||||
|
|
||||||
|
def get_random_str(self):
|
||||||
|
""" 随机生成16位字符串
|
||||||
|
@return: 16位字符串
|
||||||
|
"""
|
||||||
|
return str(random.randint(1000000000000000, 9999999999999999)).encode()
|
||||||
|
|
||||||
|
|
||||||
|
class WXBizMsgCrypt(object):
|
||||||
|
# 构造函数
|
||||||
|
def __init__(self, sToken, sEncodingAESKey, sReceiveId):
|
||||||
|
try:
|
||||||
|
self.key = base64.b64decode(sEncodingAESKey + "=")
|
||||||
|
assert len(self.key) == 32
|
||||||
|
except:
|
||||||
|
throw_exception("[error]: EncodingAESKey unvalid !", FormatException)
|
||||||
|
# return ierror.WXBizMsgCrypt_IllegalAesKey,None
|
||||||
|
self.m_sToken = sToken
|
||||||
|
self.m_sReceiveId = sReceiveId
|
||||||
|
|
||||||
|
# 验证URL
|
||||||
|
# @param sMsgSignature: 签名串,对应URL参数的msg_signature
|
||||||
|
# @param sTimeStamp: 时间戳,对应URL参数的timestamp
|
||||||
|
# @param sNonce: 随机串,对应URL参数的nonce
|
||||||
|
# @param sEchoStr: 随机串,对应URL参数的echostr
|
||||||
|
# @param sReplyEchoStr: 解密之后的echostr,当return返回0时有效
|
||||||
|
# @return:成功0,失败返回对应的错误码
|
||||||
|
|
||||||
|
def VerifyURL(self, sMsgSignature, sTimeStamp, sNonce, sEchoStr):
|
||||||
|
sha1 = SHA1()
|
||||||
|
ret, signature = sha1.getSHA1(self.m_sToken, sTimeStamp, sNonce, sEchoStr)
|
||||||
|
if ret != 0:
|
||||||
|
return ret, None
|
||||||
|
if not signature == sMsgSignature:
|
||||||
|
return ierror.WXBizMsgCrypt_ValidateSignature_Error, None
|
||||||
|
pc = Prpcrypt(self.key)
|
||||||
|
ret, sReplyEchoStr = pc.decrypt(sEchoStr, self.m_sReceiveId)
|
||||||
|
return ret, sReplyEchoStr
|
||||||
|
|
||||||
|
def EncryptMsg(self, sReplyMsg, sNonce, timestamp=None):
|
||||||
|
# 将企业回复用户的消息加密打包
|
||||||
|
# @param sReplyMsg: 企业号待回复用户的消息,xml格式的字符串
|
||||||
|
# @param sTimeStamp: 时间戳,可以自己生成,也可以用URL参数的timestamp,如为None则自动用当前时间
|
||||||
|
# @param sNonce: 随机串,可以自己生成,也可以用URL参数的nonce
|
||||||
|
# sEncryptMsg: 加密后的可以直接回复用户的密文,包括msg_signature, timestamp, nonce, encrypt的xml格式的字符串,
|
||||||
|
# return:成功0,sEncryptMsg,失败返回对应的错误码None
|
||||||
|
pc = Prpcrypt(self.key)
|
||||||
|
ret, encrypt = pc.encrypt(sReplyMsg, self.m_sReceiveId)
|
||||||
|
encrypt = encrypt.decode('utf8')
|
||||||
|
if ret != 0:
|
||||||
|
return ret, None
|
||||||
|
if timestamp is None:
|
||||||
|
timestamp = str(int(time.time()))
|
||||||
|
# 生成安全签名
|
||||||
|
sha1 = SHA1()
|
||||||
|
ret, signature = sha1.getSHA1(self.m_sToken, timestamp, sNonce, encrypt)
|
||||||
|
if ret != 0:
|
||||||
|
return ret, None
|
||||||
|
xmlParse = XMLParse()
|
||||||
|
return ret, xmlParse.generate(encrypt, signature, timestamp, sNonce)
|
||||||
|
|
||||||
|
def DecryptMsg(self, sPostData, sMsgSignature, sTimeStamp, sNonce):
|
||||||
|
# 检验消息的真实性,并且获取解密后的明文
|
||||||
|
# @param sMsgSignature: 签名串,对应URL参数的msg_signature
|
||||||
|
# @param sTimeStamp: 时间戳,对应URL参数的timestamp
|
||||||
|
# @param sNonce: 随机串,对应URL参数的nonce
|
||||||
|
# @param sPostData: 密文,对应POST请求的数据
|
||||||
|
# xml_content: 解密后的原文,当return返回0时有效
|
||||||
|
# @return: 成功0,失败返回对应的错误码
|
||||||
|
# 验证安全签名
|
||||||
|
xmlParse = XMLParse()
|
||||||
|
ret, encrypt = xmlParse.extract(sPostData)
|
||||||
|
if ret != 0:
|
||||||
|
return ret, None
|
||||||
|
sha1 = SHA1()
|
||||||
|
ret, signature = sha1.getSHA1(self.m_sToken, sTimeStamp, sNonce, encrypt)
|
||||||
|
if ret != 0:
|
||||||
|
return ret, None
|
||||||
|
if not signature == sMsgSignature:
|
||||||
|
return ierror.WXBizMsgCrypt_ValidateSignature_Error, None
|
||||||
|
pc = Prpcrypt(self.key)
|
||||||
|
ret, xml_content = pc.decrypt(encrypt, self.m_sReceiveId)
|
||||||
|
return ret, xml_content
|
||||||
318
libs/wecom_api/api.py
Normal file
318
libs/wecom_api/api.py
Normal file
@@ -0,0 +1,318 @@
|
|||||||
|
from quart import request
|
||||||
|
from .WXBizMsgCrypt3 import WXBizMsgCrypt
|
||||||
|
import base64
|
||||||
|
import binascii
|
||||||
|
import httpx
|
||||||
|
from quart import Quart
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from typing import Callable, Dict, Any
|
||||||
|
from .wecomevent import WecomEvent
|
||||||
|
from pkg.platform.types import events as platform_events, message as platform_message
|
||||||
|
import aiofiles
|
||||||
|
|
||||||
|
|
||||||
|
class WecomClient():
|
||||||
|
def __init__(self,corpid:str,secret:str,token:str,EncodingAESKey:str,contacts_secret:str):
|
||||||
|
self.corpid = corpid
|
||||||
|
self.secret = secret
|
||||||
|
self.access_token_for_contacts =''
|
||||||
|
self.token = token
|
||||||
|
self.aes = EncodingAESKey
|
||||||
|
self.base_url = 'https://qyapi.weixin.qq.com/cgi-bin'
|
||||||
|
self.access_token = ''
|
||||||
|
self.secret_for_contacts = contacts_secret
|
||||||
|
self.app = Quart(__name__)
|
||||||
|
self.wxcpt = WXBizMsgCrypt(self.token, self.aes, self.corpid)
|
||||||
|
self.app.add_url_rule('/callback/command', 'handle_callback', self.handle_callback_request, methods=['GET', 'POST'])
|
||||||
|
self._message_handlers = {
|
||||||
|
"example":[],
|
||||||
|
}
|
||||||
|
|
||||||
|
#access——token操作
|
||||||
|
async def check_access_token(self):
|
||||||
|
return bool(self.access_token and self.access_token.strip())
|
||||||
|
|
||||||
|
async def check_access_token_for_contacts(self):
|
||||||
|
return bool(self.access_token_for_contacts and self.access_token_for_contacts.strip())
|
||||||
|
|
||||||
|
async def get_access_token(self,secret):
|
||||||
|
url = f'https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid={self.corpid}&corpsecret={secret}'
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.get(url)
|
||||||
|
data = response.json()
|
||||||
|
if 'access_token' in data:
|
||||||
|
return data['access_token']
|
||||||
|
else:
|
||||||
|
raise Exception(f"未获取access token: {data}")
|
||||||
|
|
||||||
|
async def get_users(self):
|
||||||
|
if not self.check_access_token_for_contacts():
|
||||||
|
self.access_token_for_contacts = await self.get_access_token(self.secret_for_contacts)
|
||||||
|
|
||||||
|
url = self.base_url+'/user/list_id?access_token='+self.access_token_for_contacts
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"cursor":"",
|
||||||
|
"limit":10000,
|
||||||
|
}
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] == 0:
|
||||||
|
dept_users = data['dept_user']
|
||||||
|
userid = []
|
||||||
|
for user in dept_users:
|
||||||
|
userid.append(user["userid"])
|
||||||
|
return userid
|
||||||
|
else:
|
||||||
|
raise Exception("未获取用户")
|
||||||
|
|
||||||
|
async def send_to_all(self,content:str,agent_id:int):
|
||||||
|
if not self.check_access_token_for_contacts():
|
||||||
|
self.access_token_for_contacts = await self.get_access_token(self.secret_for_contacts)
|
||||||
|
|
||||||
|
url = self.base_url+'/message/send?access_token='+self.access_token_for_contacts
|
||||||
|
user_ids = await self.get_users()
|
||||||
|
user_ids_string = "|".join(user_ids)
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"touser" : user_ids_string,
|
||||||
|
"msgtype" : "text",
|
||||||
|
"agentid" : agent_id,
|
||||||
|
"text" : {
|
||||||
|
"content" : content,
|
||||||
|
},
|
||||||
|
"safe":0,
|
||||||
|
"enable_id_trans": 0,
|
||||||
|
"enable_duplicate_check": 0,
|
||||||
|
"duplicate_check_interval": 1800
|
||||||
|
}
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] != 0:
|
||||||
|
raise Exception("Failed to send message: "+str(data))
|
||||||
|
|
||||||
|
async def send_image(self,user_id:str,agent_id:int,media_id:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
url = self.base_url+'/media/upload?access_token='+self.access_token
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"touser" : user_id,
|
||||||
|
"toparty" : "",
|
||||||
|
"totag":"",
|
||||||
|
"agentid" : agent_id,
|
||||||
|
"msgtype" : "image",
|
||||||
|
"image" : {
|
||||||
|
"media_id" : media_id,
|
||||||
|
},
|
||||||
|
"safe":0,
|
||||||
|
"enable_id_trans": 0,
|
||||||
|
"enable_duplicate_check": 0,
|
||||||
|
"duplicate_check_interval": 1800
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
except Exception as e:
|
||||||
|
raise Exception("Failed to send image: "+str(e))
|
||||||
|
|
||||||
|
# 企业微信错误码40014和42001,代表accesstoken问题
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
return await self.send_image(user_id,agent_id,media_id)
|
||||||
|
|
||||||
|
if data['errcode'] != 0:
|
||||||
|
raise Exception("Failed to send image: "+str(data))
|
||||||
|
|
||||||
|
async def send_private_msg(self,user_id:str, agent_id:int,content:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
|
||||||
|
url = self.base_url+'/message/send?access_token='+self.access_token
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params={
|
||||||
|
"touser" : user_id,
|
||||||
|
"msgtype" : "text",
|
||||||
|
"agentid" : agent_id,
|
||||||
|
"text" : {
|
||||||
|
"content" : content,
|
||||||
|
},
|
||||||
|
"safe":0,
|
||||||
|
"enable_id_trans": 0,
|
||||||
|
"enable_duplicate_check": 0,
|
||||||
|
"duplicate_check_interval": 1800
|
||||||
|
}
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
return await self.send_private_msg(user_id,agent_id,content)
|
||||||
|
if data['errcode'] != 0:
|
||||||
|
raise Exception("Failed to send message: "+str(data))
|
||||||
|
|
||||||
|
async def handle_callback_request(self):
|
||||||
|
"""
|
||||||
|
处理回调请求,包括 GET 验证和 POST 消息接收。
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
|
||||||
|
msg_signature = request.args.get("msg_signature")
|
||||||
|
timestamp = request.args.get("timestamp")
|
||||||
|
nonce = request.args.get("nonce")
|
||||||
|
|
||||||
|
if request.method == "GET":
|
||||||
|
echostr = request.args.get("echostr")
|
||||||
|
ret, reply_echo_str = self.wxcpt.VerifyURL(msg_signature, timestamp, nonce, echostr)
|
||||||
|
if ret != 0:
|
||||||
|
raise Exception(f"验证失败,错误码: {ret}")
|
||||||
|
return reply_echo_str
|
||||||
|
|
||||||
|
elif request.method == "POST":
|
||||||
|
encrypt_msg = await request.data
|
||||||
|
ret, xml_msg = self.wxcpt.DecryptMsg(encrypt_msg, msg_signature, timestamp, nonce)
|
||||||
|
if ret != 0:
|
||||||
|
raise Exception(f"消息解密失败,错误码: {ret}")
|
||||||
|
|
||||||
|
# 解析消息并处理
|
||||||
|
message_data = await self.get_message(xml_msg)
|
||||||
|
if message_data:
|
||||||
|
event = WecomEvent.from_payload(message_data) # 转换为 WecomEvent 对象
|
||||||
|
if event:
|
||||||
|
await self._handle_message(event)
|
||||||
|
|
||||||
|
return "success"
|
||||||
|
except Exception as e:
|
||||||
|
return f"Error processing request: {str(e)}", 400
|
||||||
|
|
||||||
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
启动 Quart 应用。
|
||||||
|
"""
|
||||||
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
"""
|
||||||
|
注册消息类型处理器。
|
||||||
|
"""
|
||||||
|
def decorator(func: Callable[[WecomEvent], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def _handle_message(self, event: WecomEvent):
|
||||||
|
"""
|
||||||
|
处理消息事件。
|
||||||
|
"""
|
||||||
|
msg_type = event.type
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
async def get_message(self, xml_msg: str) -> Dict[str, Any]:
|
||||||
|
"""
|
||||||
|
解析微信返回的 XML 消息并转换为字典。
|
||||||
|
"""
|
||||||
|
root = ET.fromstring(xml_msg)
|
||||||
|
message_data = {
|
||||||
|
"ToUserName": root.find("ToUserName").text,
|
||||||
|
"FromUserName": root.find("FromUserName").text,
|
||||||
|
"CreateTime": int(root.find("CreateTime").text),
|
||||||
|
"MsgType": root.find("MsgType").text,
|
||||||
|
"Content": root.find("Content").text if root.find("Content") is not None else None,
|
||||||
|
"MsgId": int(root.find("MsgId").text) if root.find("MsgId") is not None else None,
|
||||||
|
"AgentID": int(root.find("AgentID").text) if root.find("AgentID") is not None else None,
|
||||||
|
}
|
||||||
|
if message_data["MsgType"] == "image":
|
||||||
|
message_data["MediaId"] = root.find("MediaId").text if root.find("MediaId") is not None else None
|
||||||
|
message_data["PicUrl"] = root.find("PicUrl").text if root.find("PicUrl") is not None else None
|
||||||
|
|
||||||
|
return message_data
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
async def get_image_type(image_bytes: bytes) -> str:
|
||||||
|
"""
|
||||||
|
通过图片的magic numbers判断图片类型
|
||||||
|
"""
|
||||||
|
magic_numbers = {
|
||||||
|
b'\xFF\xD8\xFF': 'jpg',
|
||||||
|
b'\x89\x50\x4E\x47': 'png',
|
||||||
|
b'\x47\x49\x46': 'gif',
|
||||||
|
b'\x42\x4D': 'bmp',
|
||||||
|
b'\x00\x00\x01\x00': 'ico'
|
||||||
|
}
|
||||||
|
|
||||||
|
for magic, ext in magic_numbers.items():
|
||||||
|
if image_bytes.startswith(magic):
|
||||||
|
return ext
|
||||||
|
return 'jpg' # 默认返回jpg
|
||||||
|
|
||||||
|
|
||||||
|
async def upload_to_work(self, image: platform_message.Image):
|
||||||
|
"""
|
||||||
|
获取 media_id
|
||||||
|
"""
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
|
||||||
|
url = self.base_url + '/media/upload?access_token=' + self.access_token + '&type=file'
|
||||||
|
file_bytes = None
|
||||||
|
file_name = "uploaded_file.txt"
|
||||||
|
|
||||||
|
# 获取文件的二进制数据
|
||||||
|
if image.path:
|
||||||
|
async with aiofiles.open(image.path, 'rb') as f:
|
||||||
|
file_bytes = await f.read()
|
||||||
|
file_name = image.path.split('/')[-1]
|
||||||
|
elif image.url:
|
||||||
|
file_bytes = await self.download_image_to_bytes(image.url)
|
||||||
|
file_name = image.url.split('/')[-1]
|
||||||
|
elif image.base64:
|
||||||
|
try:
|
||||||
|
base64_data = image.base64
|
||||||
|
if ',' in base64_data:
|
||||||
|
base64_data = base64_data.split(',', 1)[1]
|
||||||
|
padding = 4 - (len(base64_data) % 4) if len(base64_data) % 4 else 0
|
||||||
|
padded_base64 = base64_data + '=' * padding
|
||||||
|
file_bytes = base64.b64decode(padded_base64)
|
||||||
|
except binascii.Error as e:
|
||||||
|
raise ValueError(f"Invalid base64 string: {str(e)}")
|
||||||
|
else:
|
||||||
|
raise ValueError("image对象出错")
|
||||||
|
|
||||||
|
# 设置 multipart/form-data 格式的文件
|
||||||
|
boundary = "-------------------------acebdf13572468"
|
||||||
|
headers = {
|
||||||
|
'Content-Type': f'multipart/form-data; boundary={boundary}'
|
||||||
|
}
|
||||||
|
body = (
|
||||||
|
f"--{boundary}\r\n"
|
||||||
|
f"Content-Disposition: form-data; name=\"media\"; filename=\"{file_name}\"; filelength={len(file_bytes)}\r\n"
|
||||||
|
f"Content-Type: application/octet-stream\r\n\r\n"
|
||||||
|
).encode('utf-8') + file_bytes + f"\r\n--{boundary}--\r\n".encode('utf-8')
|
||||||
|
|
||||||
|
# 上传文件
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url, headers=headers, content=body)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
media_id = await self.upload_to_work(image)
|
||||||
|
if data.get('errcode', 0) != 0:
|
||||||
|
raise Exception("failed to upload file")
|
||||||
|
|
||||||
|
media_id = data.get('media_id')
|
||||||
|
return media_id
|
||||||
|
|
||||||
|
async def download_image_to_bytes(self,url:str) -> bytes:
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.get(url)
|
||||||
|
response.raise_for_status()
|
||||||
|
return response.content
|
||||||
|
|
||||||
|
#进行media_id的获取
|
||||||
|
async def get_media_id(self, image: platform_message.Image):
|
||||||
|
|
||||||
|
media_id = await self.upload_to_work(image=image)
|
||||||
|
return media_id
|
||||||
20
libs/wecom_api/ierror.py
Normal file
20
libs/wecom_api/ierror.py
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#########################################################################
|
||||||
|
# Author: jonyqin
|
||||||
|
# Created Time: Thu 11 Sep 2014 01:53:58 PM CST
|
||||||
|
# File Name: ierror.py
|
||||||
|
# Description:定义错误码含义
|
||||||
|
#########################################################################
|
||||||
|
WXBizMsgCrypt_OK = 0
|
||||||
|
WXBizMsgCrypt_ValidateSignature_Error = -40001
|
||||||
|
WXBizMsgCrypt_ParseXml_Error = -40002
|
||||||
|
WXBizMsgCrypt_ComputeSignature_Error = -40003
|
||||||
|
WXBizMsgCrypt_IllegalAesKey = -40004
|
||||||
|
WXBizMsgCrypt_ValidateCorpid_Error = -40005
|
||||||
|
WXBizMsgCrypt_EncryptAES_Error = -40006
|
||||||
|
WXBizMsgCrypt_DecryptAES_Error = -40007
|
||||||
|
WXBizMsgCrypt_IllegalBuffer = -40008
|
||||||
|
WXBizMsgCrypt_EncodeBase64_Error = -40009
|
||||||
|
WXBizMsgCrypt_DecodeBase64_Error = -40010
|
||||||
|
WXBizMsgCrypt_GenReturnXml_Error = -40011
|
||||||
179
libs/wecom_api/wecomevent.py
Normal file
179
libs/wecom_api/wecomevent.py
Normal file
@@ -0,0 +1,179 @@
|
|||||||
|
from typing import Dict, Any, Optional
|
||||||
|
|
||||||
|
|
||||||
|
class WecomEvent(dict):
|
||||||
|
"""
|
||||||
|
封装从企业微信收到的事件数据对象(字典),提供属性以获取其中的字段。
|
||||||
|
|
||||||
|
除 `type` 和 `detail_type` 属性对于任何事件都有效外,其它属性是否存在(若不存在则返回 `None`)依事件类型不同而不同。
|
||||||
|
"""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def from_payload(payload: Dict[str, Any]) -> Optional["WecomEvent"]:
|
||||||
|
"""
|
||||||
|
从企业微信事件数据构造 `WecomEvent` 对象。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
payload (Dict[str, Any]): 解密后的企业微信事件数据。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[WecomEvent]: 如果事件数据合法,则返回 WecomEvent 对象;否则返回 None。
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
event = WecomEvent(payload)
|
||||||
|
_ = event.type, event.detail_type # 确保必须字段存在
|
||||||
|
return event
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self) -> str:
|
||||||
|
"""
|
||||||
|
事件类型,例如 "message"、"event"、"text" 等。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件类型。
|
||||||
|
"""
|
||||||
|
return self.get("MsgType", "")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def picurl(self) -> str:
|
||||||
|
"""
|
||||||
|
图片链接
|
||||||
|
"""
|
||||||
|
return self.get("PicUrl")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def detail_type(self) -> str:
|
||||||
|
"""
|
||||||
|
事件详细类型,依 `type` 的不同而不同。例如:
|
||||||
|
- 消息事件: "text", "image", "voice", 等
|
||||||
|
- 事件通知: "subscribe", "unsubscribe", "click", 等
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件详细类型。
|
||||||
|
"""
|
||||||
|
if self.type == "event":
|
||||||
|
return self.get("Event", "")
|
||||||
|
return self.type
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self) -> str:
|
||||||
|
"""
|
||||||
|
事件名,对于消息事件是 `type.detail_type`,对于其他事件是 `event_type`。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件名。
|
||||||
|
"""
|
||||||
|
return f"{self.type}.{self.detail_type}"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
用户 ID,例如消息的发送者或事件的触发者。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 用户 ID。
|
||||||
|
"""
|
||||||
|
return self.get("FromUserName")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def agent_id(self) -> Optional[int]:
|
||||||
|
"""
|
||||||
|
机器人 ID,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[int]: 机器人 ID。
|
||||||
|
"""
|
||||||
|
return self.get("AgentID")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def receiver_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
接收者 ID,例如机器人自身的企业微信 ID。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 接收者 ID。
|
||||||
|
"""
|
||||||
|
return self.get("ToUserName")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
消息 ID,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 消息 ID。
|
||||||
|
"""
|
||||||
|
return self.get("MsgId")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
消息内容,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 消息内容。
|
||||||
|
"""
|
||||||
|
return self.get("Content")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def media_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
媒体文件 ID,仅在图片、语音等消息类型中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 媒体文件 ID。
|
||||||
|
"""
|
||||||
|
return self.get("MediaId")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def timestamp(self) -> Optional[int]:
|
||||||
|
"""
|
||||||
|
事件发生的时间戳。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[int]: 时间戳。
|
||||||
|
"""
|
||||||
|
return self.get("CreateTime")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def event_key(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
事件的 Key 值,例如点击菜单时的 `EventKey`。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 事件 Key。
|
||||||
|
"""
|
||||||
|
return self.get("EventKey")
|
||||||
|
|
||||||
|
def __getattr__(self, key: str) -> Optional[Any]:
|
||||||
|
"""
|
||||||
|
允许通过属性访问数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[Any]: 字段值。
|
||||||
|
"""
|
||||||
|
return self.get(key)
|
||||||
|
|
||||||
|
def __setattr__(self, key: str, value: Any) -> None:
|
||||||
|
"""
|
||||||
|
允许通过属性设置数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
value (Any): 字段值。
|
||||||
|
"""
|
||||||
|
self[key] = value
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
"""
|
||||||
|
生成事件对象的字符串表示。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 字符串表示。
|
||||||
|
"""
|
||||||
|
return f"<WecomEvent {super().__repr__()}>"
|
||||||
337
libs/wecom_customer_service_api/api.py
Normal file
337
libs/wecom_customer_service_api/api.py
Normal file
@@ -0,0 +1,337 @@
|
|||||||
|
from quart import request
|
||||||
|
from ..wecom_api.WXBizMsgCrypt3 import WXBizMsgCrypt
|
||||||
|
import base64
|
||||||
|
import binascii
|
||||||
|
import httpx
|
||||||
|
import traceback
|
||||||
|
from quart import Quart
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from typing import Callable, Dict, Any
|
||||||
|
from .wecomcsevent import WecomCSEvent
|
||||||
|
from pkg.platform.types import events as platform_events, message as platform_message
|
||||||
|
import aiofiles
|
||||||
|
|
||||||
|
|
||||||
|
class WecomCSClient():
|
||||||
|
def __init__(self,corpid:str,secret:str,token:str,EncodingAESKey:str):
|
||||||
|
self.corpid = corpid
|
||||||
|
self.secret = secret
|
||||||
|
self.access_token_for_contacts =''
|
||||||
|
self.token = token
|
||||||
|
self.aes = EncodingAESKey
|
||||||
|
self.base_url = 'https://qyapi.weixin.qq.com/cgi-bin'
|
||||||
|
self.access_token = ''
|
||||||
|
self.app = Quart(__name__)
|
||||||
|
self.wxcpt = WXBizMsgCrypt(self.token, self.aes, self.corpid)
|
||||||
|
self.app.add_url_rule('/callback/command', 'handle_callback', self.handle_callback_request, methods=['GET', 'POST'])
|
||||||
|
self._message_handlers = {
|
||||||
|
"example":[],
|
||||||
|
}
|
||||||
|
|
||||||
|
async def get_pic_url(self, media_id: str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
|
||||||
|
url = f"{self.base_url}/media/get?access_token={self.access_token}&media_id={media_id}"
|
||||||
|
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.get(url)
|
||||||
|
if response.headers.get("Content-Type", "").startswith("application/json"):
|
||||||
|
data = response.json()
|
||||||
|
if data.get('errcode') in [40014, 42001]:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
return await self.get_pic_url(media_id)
|
||||||
|
else:
|
||||||
|
raise Exception("Failed to get image: " + str(data))
|
||||||
|
|
||||||
|
# 否则是图片,转成 base64
|
||||||
|
image_bytes = response.content
|
||||||
|
content_type = response.headers.get("Content-Type", "")
|
||||||
|
base64_str = base64.b64encode(image_bytes).decode("utf-8")
|
||||||
|
base64_str = f"data:{content_type};base64,{base64_str}"
|
||||||
|
return base64_str
|
||||||
|
|
||||||
|
|
||||||
|
#access——token操作
|
||||||
|
async def check_access_token(self):
|
||||||
|
return bool(self.access_token and self.access_token.strip())
|
||||||
|
|
||||||
|
async def check_access_token_for_contacts(self):
|
||||||
|
return bool(self.access_token_for_contacts and self.access_token_for_contacts.strip())
|
||||||
|
|
||||||
|
async def get_access_token(self,secret):
|
||||||
|
url = f'https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid={self.corpid}&corpsecret={secret}'
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.get(url)
|
||||||
|
data = response.json()
|
||||||
|
if 'access_token' in data:
|
||||||
|
return data['access_token']
|
||||||
|
else:
|
||||||
|
raise Exception(f"未获取access token: {data}")
|
||||||
|
|
||||||
|
async def get_detailed_message_list(self,xml_msg:str):
|
||||||
|
# 在本方法中解析消息,并且获得消息的具体内容
|
||||||
|
root = ET.fromstring(xml_msg)
|
||||||
|
token = root.find("Token").text
|
||||||
|
open_kfid = root.find("OpenKfId").text
|
||||||
|
|
||||||
|
# if open_kfid in self.openkfid_list:
|
||||||
|
# return None
|
||||||
|
# else:
|
||||||
|
# self.openkfid_list.append(open_kfid)
|
||||||
|
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
|
||||||
|
url = self.base_url+'/kf/sync_msg?access_token='+ self.access_token
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"token": token,
|
||||||
|
"voice_format": 0,
|
||||||
|
"open_kfid": open_kfid,
|
||||||
|
}
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] != 0:
|
||||||
|
raise Exception("Failed to get message")
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
return await self.get_detailed_message_list(xml_msg)
|
||||||
|
last_msg_data = data['msg_list'][-1]
|
||||||
|
open_kfid = last_msg_data.get("open_kfid")
|
||||||
|
# 进行获取图片操作
|
||||||
|
if last_msg_data.get("msgtype") == "image":
|
||||||
|
media_id = last_msg_data.get("image").get("media_id")
|
||||||
|
picurl = await self.get_pic_url(media_id)
|
||||||
|
last_msg_data["picurl"] = picurl
|
||||||
|
# await self.change_service_status(userid=external_userid,openkfid=open_kfid,servicer=servicer)
|
||||||
|
return last_msg_data
|
||||||
|
|
||||||
|
|
||||||
|
async def change_service_status(self,userid:str,openkfid:str,servicer:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
url = self.base_url+"/kf/service_state/get?access_token="+self.access_token
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"open_kfid" : openkfid,
|
||||||
|
"external_userid" : userid,
|
||||||
|
"service_state" : 1,
|
||||||
|
"servicer_userid" : servicer,
|
||||||
|
}
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
return await self.change_service_status(userid,openkfid)
|
||||||
|
if data['errcode'] != 0:
|
||||||
|
raise Exception("Failed to change service status: "+str(data))
|
||||||
|
|
||||||
|
|
||||||
|
async def send_image(self,user_id:str,agent_id:int,media_id:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
url = self.base_url+'/media/upload?access_token='+self.access_token
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
params = {
|
||||||
|
"touser" : user_id,
|
||||||
|
"toparty" : "",
|
||||||
|
"totag":"",
|
||||||
|
"agentid" : agent_id,
|
||||||
|
"msgtype" : "image",
|
||||||
|
"image" : {
|
||||||
|
"media_id" : media_id,
|
||||||
|
},
|
||||||
|
"safe":0,
|
||||||
|
"enable_id_trans": 0,
|
||||||
|
"enable_duplicate_check": 0,
|
||||||
|
"duplicate_check_interval": 1800
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
response = await client.post(url,json=params)
|
||||||
|
data = response.json()
|
||||||
|
except Exception as e:
|
||||||
|
raise Exception("Failed to send image: "+str(e))
|
||||||
|
|
||||||
|
# 企业微信错误码40014和42001,代表accesstoken问题
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
return await self.send_image(user_id,agent_id,media_id)
|
||||||
|
|
||||||
|
if data['errcode'] != 0:
|
||||||
|
raise Exception("Failed to send image: "+str(data))
|
||||||
|
|
||||||
|
|
||||||
|
async def send_text_msg(self, open_kfid: str, external_userid: str, msgid: str,content:str):
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
|
||||||
|
url = f"https://qyapi.weixin.qq.com/cgi-bin/kf/send_msg?access_token={self.access_token}"
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
"touser": external_userid,
|
||||||
|
"open_kfid": open_kfid,
|
||||||
|
"msgid": msgid,
|
||||||
|
"msgtype": "text",
|
||||||
|
"text": {
|
||||||
|
"content": content,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url, json=payload)
|
||||||
|
|
||||||
|
data = response.json()
|
||||||
|
if data.get("errcode") != 0:
|
||||||
|
raise Exception(f"消息发送失败: {data}")
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
async def handle_callback_request(self):
|
||||||
|
"""
|
||||||
|
处理回调请求,包括 GET 验证和 POST 消息接收。
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
|
||||||
|
msg_signature = request.args.get("msg_signature")
|
||||||
|
timestamp = request.args.get("timestamp")
|
||||||
|
nonce = request.args.get("nonce")
|
||||||
|
|
||||||
|
if request.method == "GET":
|
||||||
|
echostr = request.args.get("echostr")
|
||||||
|
ret, reply_echo_str = self.wxcpt.VerifyURL(msg_signature, timestamp, nonce, echostr)
|
||||||
|
if ret != 0:
|
||||||
|
raise Exception(f"验证失败,错误码: {ret}")
|
||||||
|
return reply_echo_str
|
||||||
|
|
||||||
|
elif request.method == "POST":
|
||||||
|
encrypt_msg = await request.data
|
||||||
|
ret, xml_msg = self.wxcpt.DecryptMsg(encrypt_msg, msg_signature, timestamp, nonce)
|
||||||
|
if ret != 0:
|
||||||
|
raise Exception(f"消息解密失败,错误码: {ret}")
|
||||||
|
|
||||||
|
# 解析消息并处理
|
||||||
|
message_data = await self.get_detailed_message_list(xml_msg)
|
||||||
|
if message_data is not None:
|
||||||
|
event = WecomCSEvent.from_payload(message_data)
|
||||||
|
if event:
|
||||||
|
await self._handle_message(event)
|
||||||
|
|
||||||
|
return "success"
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
return f"Error processing request: {str(e)}", 400
|
||||||
|
|
||||||
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
启动 Quart 应用。
|
||||||
|
"""
|
||||||
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
||||||
|
|
||||||
|
def on_message(self, msg_type: str):
|
||||||
|
"""
|
||||||
|
注册消息类型处理器。
|
||||||
|
"""
|
||||||
|
def decorator(func: Callable[[WecomCSEvent], None]):
|
||||||
|
if msg_type not in self._message_handlers:
|
||||||
|
self._message_handlers[msg_type] = []
|
||||||
|
self._message_handlers[msg_type].append(func)
|
||||||
|
return func
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
async def _handle_message(self, event: WecomCSEvent):
|
||||||
|
"""
|
||||||
|
处理消息事件。
|
||||||
|
"""
|
||||||
|
msg_type = event.type
|
||||||
|
if msg_type in self._message_handlers:
|
||||||
|
for handler in self._message_handlers[msg_type]:
|
||||||
|
await handler(event)
|
||||||
|
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
async def get_image_type(image_bytes: bytes) -> str:
|
||||||
|
"""
|
||||||
|
通过图片的magic numbers判断图片类型
|
||||||
|
"""
|
||||||
|
magic_numbers = {
|
||||||
|
b'\xFF\xD8\xFF': 'jpg',
|
||||||
|
b'\x89\x50\x4E\x47': 'png',
|
||||||
|
b'\x47\x49\x46': 'gif',
|
||||||
|
b'\x42\x4D': 'bmp',
|
||||||
|
b'\x00\x00\x01\x00': 'ico'
|
||||||
|
}
|
||||||
|
|
||||||
|
for magic, ext in magic_numbers.items():
|
||||||
|
if image_bytes.startswith(magic):
|
||||||
|
return ext
|
||||||
|
return 'jpg' # 默认返回jpg
|
||||||
|
|
||||||
|
|
||||||
|
async def upload_to_work(self, image: platform_message.Image):
|
||||||
|
"""
|
||||||
|
获取 media_id
|
||||||
|
"""
|
||||||
|
if not await self.check_access_token():
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
|
||||||
|
url = self.base_url + '/media/upload?access_token=' + self.access_token + '&type=file'
|
||||||
|
file_bytes = None
|
||||||
|
file_name = "uploaded_file.txt"
|
||||||
|
|
||||||
|
# 获取文件的二进制数据
|
||||||
|
if image.path:
|
||||||
|
async with aiofiles.open(image.path, 'rb') as f:
|
||||||
|
file_bytes = await f.read()
|
||||||
|
file_name = image.path.split('/')[-1]
|
||||||
|
elif image.url:
|
||||||
|
file_bytes = await self.download_image_to_bytes(image.url)
|
||||||
|
file_name = image.url.split('/')[-1]
|
||||||
|
elif image.base64:
|
||||||
|
try:
|
||||||
|
base64_data = image.base64
|
||||||
|
if ',' in base64_data:
|
||||||
|
base64_data = base64_data.split(',', 1)[1]
|
||||||
|
padding = 4 - (len(base64_data) % 4) if len(base64_data) % 4 else 0
|
||||||
|
padded_base64 = base64_data + '=' * padding
|
||||||
|
file_bytes = base64.b64decode(padded_base64)
|
||||||
|
except binascii.Error as e:
|
||||||
|
raise ValueError(f"Invalid base64 string: {str(e)}")
|
||||||
|
else:
|
||||||
|
raise ValueError("image对象出错")
|
||||||
|
|
||||||
|
# 设置 multipart/form-data 格式的文件
|
||||||
|
boundary = "-------------------------acebdf13572468"
|
||||||
|
headers = {
|
||||||
|
'Content-Type': f'multipart/form-data; boundary={boundary}'
|
||||||
|
}
|
||||||
|
body = (
|
||||||
|
f"--{boundary}\r\n"
|
||||||
|
f"Content-Disposition: form-data; name=\"media\"; filename=\"{file_name}\"; filelength={len(file_bytes)}\r\n"
|
||||||
|
f"Content-Type: application/octet-stream\r\n\r\n"
|
||||||
|
).encode('utf-8') + file_bytes + f"\r\n--{boundary}--\r\n".encode('utf-8')
|
||||||
|
|
||||||
|
# 上传文件
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.post(url, headers=headers, content=body)
|
||||||
|
data = response.json()
|
||||||
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
||||||
|
self.access_token = await self.get_access_token(self.secret)
|
||||||
|
media_id = await self.upload_to_work(image)
|
||||||
|
if data.get('errcode', 0) != 0:
|
||||||
|
raise Exception("failed to upload file")
|
||||||
|
|
||||||
|
media_id = data.get('media_id')
|
||||||
|
return media_id
|
||||||
|
|
||||||
|
async def download_image_to_bytes(self,url:str) -> bytes:
|
||||||
|
async with httpx.AsyncClient() as client:
|
||||||
|
response = await client.get(url)
|
||||||
|
response.raise_for_status()
|
||||||
|
return response.content
|
||||||
|
|
||||||
|
#进行media_id的获取
|
||||||
|
async def get_media_id(self, image: platform_message.Image):
|
||||||
|
|
||||||
|
media_id = await self.upload_to_work(image=image)
|
||||||
|
return media_id
|
||||||
134
libs/wecom_customer_service_api/wecomcsevent.py
Normal file
134
libs/wecom_customer_service_api/wecomcsevent.py
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
from typing import Dict, Any, Optional
|
||||||
|
|
||||||
|
|
||||||
|
class WecomCSEvent(dict):
|
||||||
|
"""
|
||||||
|
封装从企业微信收到的事件数据对象(字典),提供属性以获取其中的字段。
|
||||||
|
|
||||||
|
除 `type` 和 `detail_type` 属性对于任何事件都有效外,其它属性是否存在(若不存在则返回 `None`)依事件类型不同而不同。
|
||||||
|
"""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def from_payload(payload: Dict[str, Any]) -> Optional["WecomCSEvent"]:
|
||||||
|
"""
|
||||||
|
从企业微信(客服会话)事件数据构造 `WecomEvent` 对象。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
payload (Dict[str, Any]): 解密后的企业微信事件数据。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[WecomEvent]: 如果事件数据合法,则返回 WecomEvent 对象;否则返回 None。
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
event = WecomCSEvent(payload)
|
||||||
|
_ = event.type,
|
||||||
|
return event
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self) -> str:
|
||||||
|
"""
|
||||||
|
事件类型,例如 "message"、"event"、"text" 等。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 事件类型。
|
||||||
|
"""
|
||||||
|
return self.get("msgtype", "")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
用户 ID,例如消息的发送者或事件的触发者。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 用户 ID。
|
||||||
|
"""
|
||||||
|
return self.get("external_userid")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def receiver_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
接收者 ID,例如机器人自身的企业微信 ID。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 接收者 ID。
|
||||||
|
"""
|
||||||
|
return self.get("open_kfid","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def picurl(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
图片 URL,仅在图片消息中存在。
|
||||||
|
base64格式
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 图片 URL。
|
||||||
|
"""
|
||||||
|
|
||||||
|
return self.get("picurl","")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message_id(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
消息 ID,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 消息 ID。
|
||||||
|
"""
|
||||||
|
return self.get("msgid")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def message(self) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
消息内容,仅在消息类型事件中存在。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[str]: 消息内容。
|
||||||
|
"""
|
||||||
|
if self.get("msgtype") == 'text':
|
||||||
|
return self.get("text").get("content","")
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
@property
|
||||||
|
def timestamp(self) -> Optional[int]:
|
||||||
|
"""
|
||||||
|
事件发生的时间戳。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[int]: 时间戳。
|
||||||
|
"""
|
||||||
|
return self.get("send_time")
|
||||||
|
|
||||||
|
|
||||||
|
def __getattr__(self, key: str) -> Optional[Any]:
|
||||||
|
"""
|
||||||
|
允许通过属性访问数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[Any]: 字段值。
|
||||||
|
"""
|
||||||
|
return self.get(key)
|
||||||
|
|
||||||
|
def __setattr__(self, key: str, value: Any) -> None:
|
||||||
|
"""
|
||||||
|
允许通过属性设置数据中的任意字段。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
key (str): 字段名。
|
||||||
|
value (Any): 字段值。
|
||||||
|
"""
|
||||||
|
self[key] = value
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
"""
|
||||||
|
生成事件对象的字符串表示。
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: 字符串表示。
|
||||||
|
"""
|
||||||
|
return f"<WecomEvent {super().__repr__()}>"
|
||||||
493
main.py
493
main.py
@@ -1,450 +1,87 @@
|
|||||||
import importlib
|
# LangBot 终端启动入口
|
||||||
import json
|
# 在此层级解决依赖项检查。
|
||||||
import os
|
# LangBot/main.py
|
||||||
import shutil
|
|
||||||
import threading
|
|
||||||
import time
|
|
||||||
|
|
||||||
import logging
|
asciiart = r"""
|
||||||
import sys
|
_ ___ _
|
||||||
import traceback
|
| | __ _ _ _ __ _| _ ) ___| |_
|
||||||
|
| |__/ _` | ' \/ _` | _ \/ _ \ _|
|
||||||
|
|____\__,_|_||_\__, |___/\___/\__|
|
||||||
|
|___/
|
||||||
|
|
||||||
sys.path.append(".")
|
⭐️开源地址: https://github.com/RockChinQ/LangBot
|
||||||
|
📖文档地址: https://docs.langbot.app
|
||||||
|
"""
|
||||||
|
|
||||||
try:
|
|
||||||
import colorlog
|
|
||||||
except ImportError:
|
|
||||||
# 尝试安装
|
|
||||||
import pkg.utils.pkgmgr as pkgmgr
|
|
||||||
try:
|
|
||||||
pkgmgr.install_requirements("requirements.txt")
|
|
||||||
import colorlog
|
|
||||||
except ImportError:
|
|
||||||
print("依赖不满足,请查看 https://github.com/RockChinQ/qcg-installer/issues/15")
|
|
||||||
sys.exit(1)
|
|
||||||
import colorlog
|
|
||||||
|
|
||||||
import requests
|
import asyncio
|
||||||
import websockets.exceptions
|
|
||||||
from urllib3.exceptions import InsecureRequestWarning
|
|
||||||
import pkg.utils.context
|
|
||||||
|
|
||||||
|
|
||||||
log_colors_config = {
|
async def main_entry(loop: asyncio.AbstractEventLoop):
|
||||||
'DEBUG': 'green', # cyan white
|
print(asciiart)
|
||||||
'INFO': 'white',
|
|
||||||
'WARNING': 'yellow',
|
|
||||||
'ERROR': 'red',
|
|
||||||
'CRITICAL': 'cyan',
|
|
||||||
}
|
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
def init_db():
|
# 检查依赖
|
||||||
import pkg.database.manager
|
|
||||||
database = pkg.database.manager.DatabaseManager()
|
|
||||||
|
|
||||||
database.initialize_database()
|
from pkg.core.bootutils import deps
|
||||||
|
|
||||||
|
missing_deps = await deps.check_deps()
|
||||||
|
|
||||||
def ensure_dependencies():
|
if missing_deps:
|
||||||
import pkg.utils.pkgmgr as pkgmgr
|
print("以下依赖包未安装,将自动安装,请完成后重启程序:")
|
||||||
pkgmgr.run_pip(["install", "openai", "Pillow", "--upgrade",
|
for dep in missing_deps:
|
||||||
"-i", "https://pypi.douban.com/simple/",
|
print("-", dep)
|
||||||
"--trusted-host", "pypi.douban.com"])
|
await deps.install_deps(missing_deps)
|
||||||
|
print("已自动安装缺失的依赖包,请重启程序。")
|
||||||
|
|
||||||
known_exception_caught = False
|
|
||||||
|
|
||||||
log_file_name = "qchatgpt.log"
|
|
||||||
|
|
||||||
|
|
||||||
def init_runtime_log_file():
|
|
||||||
"""为此次运行生成日志文件
|
|
||||||
格式: qchatgpt-yyyy-MM-dd-HH-mm-ss.log
|
|
||||||
"""
|
|
||||||
global log_file_name
|
|
||||||
|
|
||||||
# 检查logs目录是否存在
|
|
||||||
if not os.path.exists("logs"):
|
|
||||||
os.mkdir("logs")
|
|
||||||
|
|
||||||
# 检查本目录是否有qchatgpt.log,若有,移动到logs目录
|
|
||||||
if os.path.exists("qchatgpt.log"):
|
|
||||||
shutil.move("qchatgpt.log", "logs/qchatgpt.legacy.log")
|
|
||||||
|
|
||||||
log_file_name = "logs/qchatgpt-%s.log" % time.strftime("%Y-%m-%d-%H-%M-%S", time.localtime())
|
|
||||||
|
|
||||||
|
|
||||||
def reset_logging():
|
|
||||||
global log_file_name
|
|
||||||
|
|
||||||
import config
|
|
||||||
|
|
||||||
if pkg.utils.context.context['logger_handler'] is not None:
|
|
||||||
logging.getLogger().removeHandler(pkg.utils.context.context['logger_handler'])
|
|
||||||
|
|
||||||
for handler in logging.getLogger().handlers:
|
|
||||||
logging.getLogger().removeHandler(handler)
|
|
||||||
|
|
||||||
logging.basicConfig(level=config.logging_level, # 设置日志输出格式
|
|
||||||
filename=log_file_name, # log日志输出的文件位置和文件名
|
|
||||||
format="[%(asctime)s.%(msecs)03d] %(filename)s (%(lineno)d) - [%(levelname)s] : %(message)s",
|
|
||||||
# 日志输出的格式
|
|
||||||
# -8表示占位符,让输出左对齐,输出长度都为8位
|
|
||||||
datefmt="%Y-%m-%d %H:%M:%S" # 时间输出的格式
|
|
||||||
)
|
|
||||||
sh = logging.StreamHandler()
|
|
||||||
sh.setLevel(config.logging_level)
|
|
||||||
sh.setFormatter(colorlog.ColoredFormatter(
|
|
||||||
fmt="%(log_color)s[%(asctime)s.%(msecs)03d] %(filename)s (%(lineno)d) - [%(levelname)s] : "
|
|
||||||
"%(message)s",
|
|
||||||
datefmt="%Y-%m-%d %H:%M:%S",
|
|
||||||
log_colors=log_colors_config
|
|
||||||
))
|
|
||||||
logging.getLogger().addHandler(sh)
|
|
||||||
pkg.utils.context.context['logger_handler'] = sh
|
|
||||||
return sh
|
|
||||||
|
|
||||||
|
|
||||||
# 临时函数,用于加载config和上下文,未来统一放在config类
|
|
||||||
def load_config():
|
|
||||||
# 完整性校验
|
|
||||||
is_integrity = True
|
|
||||||
config_template = importlib.import_module('config-template')
|
|
||||||
config = importlib.import_module('config')
|
|
||||||
for key in dir(config_template):
|
|
||||||
if not key.startswith("__") and not hasattr(config, key):
|
|
||||||
setattr(config, key, getattr(config_template, key))
|
|
||||||
logging.warning("[{}]不存在".format(key))
|
|
||||||
is_integrity = False
|
|
||||||
|
|
||||||
if not is_integrity:
|
|
||||||
logging.warning("配置文件不完整,请依据config-template.py检查config.py")
|
|
||||||
|
|
||||||
# 检查override.json覆盖
|
|
||||||
if os.path.exists("override.json"):
|
|
||||||
override_json = json.load(open("override.json", "r", encoding="utf-8"))
|
|
||||||
for key in override_json:
|
|
||||||
if hasattr(config, key):
|
|
||||||
setattr(config, key, override_json[key])
|
|
||||||
logging.info("覆写配置[{}]为[{}]".format(key, override_json[key]))
|
|
||||||
else:
|
|
||||||
logging.error("无法覆写配置[{}]为[{}],该配置不存在,请检查override.json是否正确".format(key, override_json[key]))
|
|
||||||
|
|
||||||
if not is_integrity:
|
|
||||||
logging.warning("以上配置已被设为默认值,将在5秒后继续启动... ")
|
|
||||||
time.sleep(5)
|
|
||||||
|
|
||||||
# 存进上下文
|
|
||||||
pkg.utils.context.set_config(config)
|
|
||||||
|
|
||||||
|
|
||||||
def start(first_time_init=False):
|
|
||||||
"""启动流程,reload之后会被执行"""
|
|
||||||
|
|
||||||
global known_exception_caught
|
|
||||||
import pkg.utils.context
|
|
||||||
|
|
||||||
config = pkg.utils.context.get_config()
|
|
||||||
# 更新openai库到最新版本
|
|
||||||
if not hasattr(config, 'upgrade_dependencies') or config.upgrade_dependencies:
|
|
||||||
print("正在更新依赖库,请等待...")
|
|
||||||
if not hasattr(config, 'upgrade_dependencies'):
|
|
||||||
print("这个操作不是必须的,如果不想更新,请在config.py中添加upgrade_dependencies=False")
|
|
||||||
else:
|
|
||||||
print("这个操作不是必须的,如果不想更新,请在config.py中将upgrade_dependencies设置为False")
|
|
||||||
try:
|
|
||||||
ensure_dependencies()
|
|
||||||
except Exception as e:
|
|
||||||
print("更新openai库失败:{}, 请忽略或自行更新".format(e))
|
|
||||||
|
|
||||||
known_exception_caught = False
|
|
||||||
try:
|
|
||||||
|
|
||||||
sh = reset_logging()
|
|
||||||
pkg.utils.context.context['logger_handler'] = sh
|
|
||||||
|
|
||||||
# 检查是否设置了管理员
|
|
||||||
if not (hasattr(config, 'admin_qq') and config.admin_qq != 0):
|
|
||||||
# logging.warning("未设置管理员QQ,管理员权限指令及运行告警将无法使用,如需设置请修改config.py中的admin_qq字段")
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
config.admin_qq = int(input("未设置管理员QQ,管理员权限指令及运行告警将无法使用,请输入管理员QQ号: "))
|
|
||||||
# 写入到文件
|
|
||||||
|
|
||||||
# 读取文件
|
|
||||||
config_file_str = ""
|
|
||||||
with open("config.py", "r", encoding="utf-8") as f:
|
|
||||||
config_file_str = f.read()
|
|
||||||
# 替换
|
|
||||||
config_file_str = config_file_str.replace("admin_qq = 0", "admin_qq = " + str(config.admin_qq))
|
|
||||||
# 写入
|
|
||||||
with open("config.py", "w", encoding="utf-8") as f:
|
|
||||||
f.write(config_file_str)
|
|
||||||
|
|
||||||
print("管理员QQ已设置,如需修改请修改config.py中的admin_qq字段")
|
|
||||||
time.sleep(4)
|
|
||||||
break
|
|
||||||
except ValueError:
|
|
||||||
print("请输入数字")
|
|
||||||
|
|
||||||
import pkg.openai.manager
|
|
||||||
import pkg.database.manager
|
|
||||||
import pkg.openai.session
|
|
||||||
import pkg.qqbot.manager
|
|
||||||
import pkg.openai.dprompt
|
|
||||||
import pkg.qqbot.cmds.aamgr
|
|
||||||
|
|
||||||
try:
|
|
||||||
pkg.openai.dprompt.register_all()
|
|
||||||
pkg.qqbot.cmds.aamgr.register_all()
|
|
||||||
pkg.qqbot.cmds.aamgr.apply_privileges()
|
|
||||||
except Exception as e:
|
|
||||||
logging.error(e)
|
|
||||||
traceback.print_exc()
|
|
||||||
|
|
||||||
# 配置openai api_base
|
|
||||||
if "reverse_proxy" in config.openai_config and config.openai_config["reverse_proxy"] is not None:
|
|
||||||
import openai
|
|
||||||
openai.api_base = config.openai_config["reverse_proxy"]
|
|
||||||
|
|
||||||
# 主启动流程
|
|
||||||
database = pkg.database.manager.DatabaseManager()
|
|
||||||
|
|
||||||
database.initialize_database()
|
|
||||||
|
|
||||||
openai_interact = pkg.openai.manager.OpenAIInteract(config.openai_config['api_key'])
|
|
||||||
|
|
||||||
# 加载所有未超时的session
|
|
||||||
pkg.openai.session.load_sessions()
|
|
||||||
|
|
||||||
# 初始化qq机器人
|
|
||||||
qqbot = pkg.qqbot.manager.QQBotManager(mirai_http_api_config=config.mirai_http_api_config,
|
|
||||||
timeout=config.process_message_timeout, retry=config.retry_times,
|
|
||||||
first_time_init=first_time_init)
|
|
||||||
|
|
||||||
# 加载插件
|
|
||||||
import pkg.plugin.host
|
|
||||||
pkg.plugin.host.load_plugins()
|
|
||||||
|
|
||||||
pkg.plugin.host.initialize_plugins()
|
|
||||||
|
|
||||||
if first_time_init: # 不是热重载之后的启动,则启动新的bot线程
|
|
||||||
|
|
||||||
import mirai.exceptions
|
|
||||||
|
|
||||||
def run_bot_wrapper():
|
|
||||||
global known_exception_caught
|
|
||||||
try:
|
|
||||||
qqbot.bot.run()
|
|
||||||
except TypeError as e:
|
|
||||||
if str(e).__contains__("argument 'debug'"):
|
|
||||||
logging.error(
|
|
||||||
"连接bot失败:{}, 解决方案: https://github.com/RockChinQ/QChatGPT/issues/82".format(e))
|
|
||||||
known_exception_caught = True
|
|
||||||
elif str(e).__contains__("As of 3.10, the *loop*"):
|
|
||||||
logging.error(
|
|
||||||
"Websockets版本过低:{}, 解决方案: https://github.com/RockChinQ/QChatGPT/issues/5".format(e))
|
|
||||||
known_exception_caught = True
|
|
||||||
|
|
||||||
except websockets.exceptions.InvalidStatus as e:
|
|
||||||
logging.error(
|
|
||||||
"mirai-api-http端口无法使用:{}, 解决方案: https://github.com/RockChinQ/QChatGPT/issues/22".format(
|
|
||||||
e))
|
|
||||||
known_exception_caught = True
|
|
||||||
except mirai.exceptions.NetworkError as e:
|
|
||||||
logging.error("连接mirai-api-http失败:{}, 请检查是否已按照文档启动mirai".format(e))
|
|
||||||
known_exception_caught = True
|
|
||||||
except Exception as e:
|
|
||||||
if str(e).__contains__("404"):
|
|
||||||
logging.error(
|
|
||||||
"mirai-api-http端口无法使用:{}, 解决方案: https://github.com/RockChinQ/QChatGPT/issues/22".format(
|
|
||||||
e))
|
|
||||||
known_exception_caught = True
|
|
||||||
elif str(e).__contains__("signal only works in main thread"):
|
|
||||||
logging.error(
|
|
||||||
"hypercorn异常:{}, 解决方案: https://github.com/RockChinQ/QChatGPT/issues/86".format(
|
|
||||||
e))
|
|
||||||
known_exception_caught = True
|
|
||||||
elif str(e).__contains__("did not receive a valid HTTP"):
|
|
||||||
logging.error(
|
|
||||||
"mirai-api-http端口无法使用:{}, 解决方案: https://github.com/RockChinQ/QChatGPT/issues/22".format(
|
|
||||||
e))
|
|
||||||
else:
|
|
||||||
logging.error(
|
|
||||||
"捕捉到未知异常:{}, 请前往 https://github.com/RockChinQ/QChatGPT/issues 查找或提issue".format(e))
|
|
||||||
known_exception_caught = True
|
|
||||||
raise e
|
|
||||||
finally:
|
|
||||||
time.sleep(12)
|
|
||||||
threading.Thread(
|
|
||||||
target=run_bot_wrapper
|
|
||||||
).start()
|
|
||||||
finally:
|
|
||||||
# 判断若是Windows,输出选择模式可能会暂停程序的警告
|
|
||||||
if os.name == 'nt':
|
|
||||||
time.sleep(2)
|
|
||||||
logging.info("您正在使用Windows系统,若命令行窗口处于“选择”模式,程序可能会被暂停,此时请右键点击窗口空白区域使其取消选择模式。")
|
|
||||||
|
|
||||||
time.sleep(12)
|
|
||||||
|
|
||||||
if first_time_init:
|
|
||||||
if not known_exception_caught:
|
|
||||||
logging.info("QQ: {}, MAH: {}".format(config.mirai_http_api_config['qq'], config.mirai_http_api_config['host']+":"+str(config.mirai_http_api_config['port'])))
|
|
||||||
logging.info('程序启动完成,如长时间未显示 ”成功登录到账号xxxxx“ ,并且不回复消息,请查看 '
|
|
||||||
'https://github.com/RockChinQ/QChatGPT/issues/37')
|
|
||||||
else:
|
|
||||||
sys.exit(1)
|
|
||||||
else:
|
|
||||||
logging.info('热重载完成')
|
|
||||||
|
|
||||||
# 发送赞赏码
|
|
||||||
if config.encourage_sponsor_at_start \
|
|
||||||
and pkg.utils.context.get_openai_manager().audit_mgr.get_total_text_length() >= 2048:
|
|
||||||
|
|
||||||
logging.info("发送赞赏码")
|
|
||||||
from mirai import MessageChain, Plain, Image
|
|
||||||
import pkg.utils.constants
|
|
||||||
message_chain = MessageChain([
|
|
||||||
Plain("自2022年12月初以来,开发者已经花费了大量时间和精力来维护本项目,如果您觉得本项目对您有帮助,欢迎赞赏开发者,"
|
|
||||||
"以支持项目稳定运行😘"),
|
|
||||||
Image(base64=pkg.utils.constants.alipay_qr_b64),
|
|
||||||
Image(base64=pkg.utils.constants.wechat_qr_b64),
|
|
||||||
Plain("BTC: 3N4Azee63vbBB9boGv9Rjf4N5SocMe5eCq\nXMR: 89LS21EKQuDGkyQoe2nDupiuWXk4TVD6FALvSKv5owfmeJEPFpHeMsZLYtLiJ6GxLrhsRe5gMs6MyMSDn4GNQAse2Mae4KE\n\n"),
|
|
||||||
Plain("(本消息仅在启动时发送至管理员,如果您不想再看到此消息,请在config.py中将encourage_sponsor_at_start设置为False)")
|
|
||||||
])
|
|
||||||
pkg.utils.context.get_qqbot_manager().notify_admin_message_chain(message_chain)
|
|
||||||
|
|
||||||
time.sleep(5)
|
|
||||||
import pkg.utils.updater
|
|
||||||
try:
|
|
||||||
if pkg.utils.updater.is_new_version_available():
|
|
||||||
logging.info("新版本可用,请发送 !update 进行自动更新\n更新日志:\n{}".format("\n".join(pkg.utils.updater.get_rls_notes())))
|
|
||||||
else:
|
|
||||||
logging.info("当前已是最新版本")
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
logging.warning("检查更新失败:{}".format(e))
|
|
||||||
|
|
||||||
try:
|
|
||||||
import pkg.utils.announcement as announcement
|
|
||||||
new_announcement = announcement.fetch_new()
|
|
||||||
if new_announcement != "":
|
|
||||||
logging.critical("[公告] {}".format(new_announcement))
|
|
||||||
except Exception as e:
|
|
||||||
logging.warning("获取公告失败:{}".format(e))
|
|
||||||
|
|
||||||
return qqbot
|
|
||||||
|
|
||||||
def stop():
|
|
||||||
import pkg.qqbot.manager
|
|
||||||
import pkg.openai.session
|
|
||||||
try:
|
|
||||||
import pkg.plugin.host
|
|
||||||
pkg.plugin.host.unload_plugins()
|
|
||||||
|
|
||||||
qqbot_inst = pkg.utils.context.get_qqbot_manager()
|
|
||||||
assert isinstance(qqbot_inst, pkg.qqbot.manager.QQBotManager)
|
|
||||||
|
|
||||||
for session in pkg.openai.session.sessions:
|
|
||||||
logging.info('持久化session: %s', session)
|
|
||||||
pkg.openai.session.sessions[session].persistence()
|
|
||||||
pkg.utils.context.get_database_manager().close()
|
|
||||||
except Exception as e:
|
|
||||||
if not isinstance(e, KeyboardInterrupt):
|
|
||||||
raise e
|
|
||||||
|
|
||||||
|
|
||||||
def check_file():
|
|
||||||
# 配置文件存在性校验
|
|
||||||
if not os.path.exists('config.py'):
|
|
||||||
shutil.copy('config-template.py', 'config.py')
|
|
||||||
print('请先在config.py中填写配置')
|
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
# 检查是否有banlist.py,如果没有就把banlist-template.py复制一份
|
# 检查pydantic版本,如果没有 pydantic.v1,则把 pydantic 映射为 v1
|
||||||
if not os.path.exists('banlist.py'):
|
import pydantic.version
|
||||||
shutil.copy('res/templates/banlist-template.py', 'banlist.py')
|
if pydantic.version.VERSION < '2.0':
|
||||||
|
import pydantic
|
||||||
|
sys.modules['pydantic.v1'] = pydantic
|
||||||
|
|
||||||
# 检查是否有sensitive.json
|
# 检查配置文件
|
||||||
if not os.path.exists("sensitive.json"):
|
|
||||||
shutil.copy("res/templates/sensitive-template.json", "sensitive.json")
|
|
||||||
|
|
||||||
# 检查是否有scenario/default.json
|
from pkg.core.bootutils import files
|
||||||
if not os.path.exists("scenario/default.json"):
|
|
||||||
shutil.copy("scenario/default-template.json", "scenario/default.json")
|
|
||||||
|
|
||||||
# 检查cmdpriv.json
|
generated_files = await files.generate_files()
|
||||||
if not os.path.exists("cmdpriv.json"):
|
|
||||||
shutil.copy("res/templates/cmdpriv-template.json", "cmdpriv.json")
|
|
||||||
|
|
||||||
# 检查temp目录
|
if generated_files:
|
||||||
if not os.path.exists("temp/"):
|
print("以下文件不存在,已自动生成:")
|
||||||
os.mkdir("temp/")
|
for file in generated_files:
|
||||||
|
print("-", file)
|
||||||
|
|
||||||
# 检查并创建plugins、prompts目录
|
from pkg.core import boot
|
||||||
check_path = ["plugins", "prompts"]
|
await boot.main(loop)
|
||||||
for path in check_path:
|
|
||||||
if not os.path.exists(path):
|
|
||||||
os.mkdir(path)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
# 初始化相关文件
|
|
||||||
check_file()
|
|
||||||
|
|
||||||
# 初始化logging
|
|
||||||
init_runtime_log_file()
|
|
||||||
pkg.utils.context.context['logger_handler'] = reset_logging()
|
|
||||||
|
|
||||||
# 加载配置
|
|
||||||
load_config()
|
|
||||||
config = pkg.utils.context.get_config()
|
|
||||||
|
|
||||||
# 配置线程池
|
|
||||||
from pkg.utils import ThreadCtl
|
|
||||||
thread_ctl = ThreadCtl(
|
|
||||||
sys_pool_num=config.sys_pool_num,
|
|
||||||
admin_pool_num=config.admin_pool_num,
|
|
||||||
user_pool_num=config.user_pool_num
|
|
||||||
)
|
|
||||||
# 存进上下文
|
|
||||||
pkg.utils.context.set_thread_ctl(thread_ctl)
|
|
||||||
|
|
||||||
# 启动指令处理
|
|
||||||
if len(sys.argv) > 1 and sys.argv[1] == 'init_db':
|
|
||||||
init_db()
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
elif len(sys.argv) > 1 and sys.argv[1] == 'update':
|
|
||||||
print("正在进行程序更新...")
|
|
||||||
import pkg.utils.updater as updater
|
|
||||||
updater.update_all(cli=True)
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
# 关闭urllib的http警告
|
|
||||||
requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
|
|
||||||
|
|
||||||
pkg.utils.context.get_thread_ctl().submit_sys_task(
|
|
||||||
start,
|
|
||||||
True
|
|
||||||
)
|
|
||||||
|
|
||||||
# 主线程循环
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
time.sleep(0xFF)
|
|
||||||
except:
|
|
||||||
stop()
|
|
||||||
pkg.utils.context.get_thread_ctl().shutdown()
|
|
||||||
import platform
|
|
||||||
if platform.system() == 'Windows':
|
|
||||||
cmd = "taskkill /F /PID {}".format(os.getpid())
|
|
||||||
elif platform.system() in ['Linux', 'Darwin']:
|
|
||||||
cmd = "kill -9 {}".format(os.getpid())
|
|
||||||
os.system(cmd)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
main()
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
# 必须大于 3.10.1
|
||||||
|
if sys.version_info < (3, 10, 1):
|
||||||
|
print("需要 Python 3.10.1 及以上版本,当前 Python 版本为:", sys.version)
|
||||||
|
input("按任意键退出...")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
# 检查本目录是否有main.py,且包含LangBot字符串
|
||||||
|
invalid_pwd = False
|
||||||
|
|
||||||
|
if not os.path.exists('main.py'):
|
||||||
|
invalid_pwd = True
|
||||||
|
else:
|
||||||
|
with open('main.py', 'r', encoding='utf-8') as f:
|
||||||
|
content = f.read()
|
||||||
|
if "LangBot/main.py" not in content:
|
||||||
|
invalid_pwd = True
|
||||||
|
if invalid_pwd:
|
||||||
|
print("请在 LangBot 项目根目录下以命令形式运行此程序。")
|
||||||
|
input("按任意键退出...")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
loop = asyncio.new_event_loop()
|
||||||
|
|
||||||
|
loop.run_until_complete(main_entry(loop))
|
||||||
|
|||||||
@@ -1,78 +0,0 @@
|
|||||||
{
|
|
||||||
"comment": "这是override.json支持的字段全集, 关于override.json机制, 请查看https://github.com/RockChinQ/QChatGPT/pull/271",
|
|
||||||
"mirai_http_api_config": {
|
|
||||||
"adapter": "WebSocketAdapter",
|
|
||||||
"host": "localhost",
|
|
||||||
"port": 8080,
|
|
||||||
"verifyKey": "yirimirai",
|
|
||||||
"qq": 1234567890
|
|
||||||
},
|
|
||||||
"openai_config": {
|
|
||||||
"api_key": {
|
|
||||||
"default": "openai_api_key"
|
|
||||||
},
|
|
||||||
"http_proxy": null,
|
|
||||||
"reverse_proxy": null
|
|
||||||
},
|
|
||||||
"admin_qq": 0,
|
|
||||||
"default_prompt": {
|
|
||||||
"default": "如果我之后想获取帮助,请你说“输入!help获取帮助”"
|
|
||||||
},
|
|
||||||
"preset_mode": "normal",
|
|
||||||
"response_rules": {
|
|
||||||
"at": true,
|
|
||||||
"prefix": [
|
|
||||||
"/ai",
|
|
||||||
"!ai",
|
|
||||||
"!ai",
|
|
||||||
"ai"
|
|
||||||
],
|
|
||||||
"regexp": [],
|
|
||||||
"random_rate": 0.0
|
|
||||||
},
|
|
||||||
"ignore_rules": {
|
|
||||||
"prefix": [
|
|
||||||
"/"
|
|
||||||
],
|
|
||||||
"regexp": []
|
|
||||||
},
|
|
||||||
"income_msg_check": false,
|
|
||||||
"sensitive_word_filter": true,
|
|
||||||
"baidu_check": false,
|
|
||||||
"baidu_api_key": "",
|
|
||||||
"baidu_secret_key": "",
|
|
||||||
"inappropriate_message_tips": "[百度云]请珍惜机器人,当前返回内容不合规",
|
|
||||||
"encourage_sponsor_at_start": true,
|
|
||||||
"prompt_submit_length": 2048,
|
|
||||||
"completion_api_params": {
|
|
||||||
"model": "gpt-3.5-turbo",
|
|
||||||
"temperature": 0.9,
|
|
||||||
"top_p": 1,
|
|
||||||
"frequency_penalty": 0.2,
|
|
||||||
"presence_penalty": 1.0
|
|
||||||
},
|
|
||||||
"image_api_params": {
|
|
||||||
"size": "256x256"
|
|
||||||
},
|
|
||||||
"quote_origin": true,
|
|
||||||
"include_image_description": true,
|
|
||||||
"process_message_timeout": 30,
|
|
||||||
"show_prefix": false,
|
|
||||||
"blob_message_threshold": 256,
|
|
||||||
"blob_message_strategy": "forward",
|
|
||||||
"font_path": "",
|
|
||||||
"retry_times": 3,
|
|
||||||
"hide_exce_info_to_user": false,
|
|
||||||
"alter_tip_message": "出错了,请稍后再试",
|
|
||||||
"sys_pool_num": 8,
|
|
||||||
"admin_pool_num": 2,
|
|
||||||
"user_pool_num": 6,
|
|
||||||
"session_expire_time": 1200,
|
|
||||||
"rate_limitation": 60,
|
|
||||||
"rate_limit_strategy": "wait",
|
|
||||||
"rate_limit_drop_tip": "本分钟对话次数超过限速次数,此对话被丢弃",
|
|
||||||
"upgrade_dependencies": true,
|
|
||||||
"report_usage": true,
|
|
||||||
"logging_level": 20,
|
|
||||||
"help_message": "此机器人通过调用大型语言模型生成回复,不具有情感。\n你可以用自然语言与其交流,回复的消息中[GPT]开头的为模型生成的语言,[bot]开头的为程序提示。\n欢迎到github.com/RockChinQ/QChatGPT 给个star"
|
|
||||||
}
|
|
||||||
107
pkg/api/http/controller/group.py
Normal file
107
pkg/api/http/controller/group.py
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import abc
|
||||||
|
import typing
|
||||||
|
import enum
|
||||||
|
import quart
|
||||||
|
from quart.typing import RouteCallable
|
||||||
|
|
||||||
|
from ....core import app
|
||||||
|
|
||||||
|
|
||||||
|
preregistered_groups: list[type[RouterGroup]] = []
|
||||||
|
"""RouterGroup 的预注册列表"""
|
||||||
|
|
||||||
|
def group_class(name: str, path: str) -> None:
|
||||||
|
"""注册一个 RouterGroup"""
|
||||||
|
|
||||||
|
def decorator(cls: typing.Type[RouterGroup]) -> typing.Type[RouterGroup]:
|
||||||
|
cls.name = name
|
||||||
|
cls.path = path
|
||||||
|
preregistered_groups.append(cls)
|
||||||
|
return cls
|
||||||
|
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
|
||||||
|
class AuthType(enum.Enum):
|
||||||
|
"""认证类型"""
|
||||||
|
NONE = 'none'
|
||||||
|
USER_TOKEN = 'user-token'
|
||||||
|
|
||||||
|
|
||||||
|
class RouterGroup(abc.ABC):
|
||||||
|
|
||||||
|
name: str
|
||||||
|
|
||||||
|
path: str
|
||||||
|
|
||||||
|
ap: app.Application
|
||||||
|
|
||||||
|
quart_app: quart.Quart
|
||||||
|
|
||||||
|
def __init__(self, ap: app.Application, quart_app: quart.Quart) -> None:
|
||||||
|
self.ap = ap
|
||||||
|
self.quart_app = quart_app
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def route(self, rule: str, auth_type: AuthType = AuthType.USER_TOKEN, **options: typing.Any) -> typing.Callable[[RouteCallable], RouteCallable]: # decorator
|
||||||
|
"""注册一个路由"""
|
||||||
|
def decorator(f: RouteCallable) -> RouteCallable:
|
||||||
|
nonlocal rule
|
||||||
|
rule = self.path + rule
|
||||||
|
|
||||||
|
async def handler_error(*args, **kwargs):
|
||||||
|
|
||||||
|
if auth_type == AuthType.USER_TOKEN:
|
||||||
|
# 从Authorization头中获取token
|
||||||
|
token = quart.request.headers.get('Authorization', '').replace('Bearer ', '')
|
||||||
|
|
||||||
|
if not token:
|
||||||
|
return self.http_status(401, -1, '未提供有效的用户令牌')
|
||||||
|
|
||||||
|
try:
|
||||||
|
user_email = await self.ap.user_service.verify_jwt_token(token)
|
||||||
|
|
||||||
|
# 检查f是否接受user_email参数
|
||||||
|
if 'user_email' in f.__code__.co_varnames:
|
||||||
|
kwargs['user_email'] = user_email
|
||||||
|
except Exception as e:
|
||||||
|
return self.http_status(401, -1, str(e))
|
||||||
|
|
||||||
|
try:
|
||||||
|
return await f(*args, **kwargs)
|
||||||
|
except Exception as e: # 自动 500
|
||||||
|
return self.http_status(500, -2, str(e))
|
||||||
|
|
||||||
|
new_f = handler_error
|
||||||
|
new_f.__name__ = (self.name + rule).replace('/', '__')
|
||||||
|
new_f.__doc__ = f.__doc__
|
||||||
|
|
||||||
|
self.quart_app.route(rule, **options)(new_f)
|
||||||
|
return f
|
||||||
|
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
def success(self, data: typing.Any = None) -> quart.Response:
|
||||||
|
"""返回一个 200 响应"""
|
||||||
|
return quart.jsonify({
|
||||||
|
'code': 0,
|
||||||
|
'msg': 'ok',
|
||||||
|
'data': data,
|
||||||
|
})
|
||||||
|
|
||||||
|
def fail(self, code: int, msg: str) -> quart.Response:
|
||||||
|
"""返回一个异常响应"""
|
||||||
|
|
||||||
|
return quart.jsonify({
|
||||||
|
'code': code,
|
||||||
|
'msg': msg,
|
||||||
|
})
|
||||||
|
|
||||||
|
def http_status(self, status: int, code: int, msg: str) -> quart.Response:
|
||||||
|
"""返回一个指定状态码的响应"""
|
||||||
|
return self.fail(code, msg), status
|
||||||
32
pkg/api/http/controller/groups/logs.py
Normal file
32
pkg/api/http/controller/groups/logs.py
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
import quart
|
||||||
|
|
||||||
|
from .....core import app
|
||||||
|
from .. import group
|
||||||
|
|
||||||
|
|
||||||
|
@group.group_class('logs', '/api/v1/logs')
|
||||||
|
class LogsRouterGroup(group.RouterGroup):
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
@self.route('', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
|
||||||
|
start_page_number = int(quart.request.args.get('start_page_number', 0))
|
||||||
|
start_offset = int(quart.request.args.get('start_offset', 0))
|
||||||
|
|
||||||
|
logs_str, end_page_number, end_offset = self.ap.log_cache.get_log_by_pointer(
|
||||||
|
start_page_number=start_page_number,
|
||||||
|
start_offset=start_offset
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.success(
|
||||||
|
data={
|
||||||
|
"logs": logs_str,
|
||||||
|
"end_page_number": end_page_number,
|
||||||
|
"end_offset": end_offset
|
||||||
|
}
|
||||||
|
)
|
||||||
84
pkg/api/http/controller/groups/plugins.py
Normal file
84
pkg/api/http/controller/groups/plugins.py
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
import quart
|
||||||
|
|
||||||
|
from .....core import app, taskmgr
|
||||||
|
from .. import group
|
||||||
|
|
||||||
|
|
||||||
|
@group.group_class('plugins', '/api/v1/plugins')
|
||||||
|
class PluginsRouterGroup(group.RouterGroup):
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
@self.route('', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
plugins = self.ap.plugin_mgr.plugins()
|
||||||
|
|
||||||
|
plugins_data = [plugin.model_dump() for plugin in plugins]
|
||||||
|
|
||||||
|
return self.success(data={
|
||||||
|
'plugins': plugins_data
|
||||||
|
})
|
||||||
|
|
||||||
|
@self.route('/<author>/<plugin_name>/toggle', methods=['PUT'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _(author: str, plugin_name: str) -> str:
|
||||||
|
data = await quart.request.json
|
||||||
|
target_enabled = data.get('target_enabled')
|
||||||
|
await self.ap.plugin_mgr.update_plugin_switch(plugin_name, target_enabled)
|
||||||
|
return self.success()
|
||||||
|
|
||||||
|
@self.route('/<author>/<plugin_name>/update', methods=['POST'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _(author: str, plugin_name: str) -> str:
|
||||||
|
ctx = taskmgr.TaskContext.new()
|
||||||
|
wrapper = self.ap.task_mgr.create_user_task(
|
||||||
|
self.ap.plugin_mgr.update_plugin(plugin_name, task_context=ctx),
|
||||||
|
kind="plugin-operation",
|
||||||
|
name=f"plugin-update-{plugin_name}",
|
||||||
|
label=f"更新插件 {plugin_name}",
|
||||||
|
context=ctx
|
||||||
|
)
|
||||||
|
return self.success(data={
|
||||||
|
'task_id': wrapper.id
|
||||||
|
})
|
||||||
|
|
||||||
|
@self.route('/<author>/<plugin_name>', methods=['DELETE'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _(author: str, plugin_name: str) -> str:
|
||||||
|
ctx = taskmgr.TaskContext.new()
|
||||||
|
wrapper = self.ap.task_mgr.create_user_task(
|
||||||
|
self.ap.plugin_mgr.uninstall_plugin(plugin_name, task_context=ctx),
|
||||||
|
kind="plugin-operation",
|
||||||
|
name=f'plugin-remove-{plugin_name}',
|
||||||
|
label=f'删除插件 {plugin_name}',
|
||||||
|
context=ctx
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.success(data={
|
||||||
|
'task_id': wrapper.id
|
||||||
|
})
|
||||||
|
|
||||||
|
@self.route('/reorder', methods=['PUT'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
data = await quart.request.json
|
||||||
|
await self.ap.plugin_mgr.reorder_plugins(data.get('plugins'))
|
||||||
|
return self.success()
|
||||||
|
|
||||||
|
@self.route('/install/github', methods=['POST'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
data = await quart.request.json
|
||||||
|
|
||||||
|
ctx = taskmgr.TaskContext.new()
|
||||||
|
short_source_str = data['source'][-8:]
|
||||||
|
wrapper = self.ap.task_mgr.create_user_task(
|
||||||
|
self.ap.plugin_mgr.install_plugin(data['source'], task_context=ctx),
|
||||||
|
kind="plugin-operation",
|
||||||
|
name=f'plugin-install-github',
|
||||||
|
label=f'安装插件 ...{short_source_str}',
|
||||||
|
context=ctx
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.success(data={
|
||||||
|
'task_id': wrapper.id
|
||||||
|
})
|
||||||
62
pkg/api/http/controller/groups/settings.py
Normal file
62
pkg/api/http/controller/groups/settings.py
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
import quart
|
||||||
|
|
||||||
|
from .....core import app
|
||||||
|
from .. import group
|
||||||
|
|
||||||
|
|
||||||
|
@group.group_class('settings', '/api/v1/settings')
|
||||||
|
class SettingsRouterGroup(group.RouterGroup):
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
|
||||||
|
@self.route('', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
return self.success(
|
||||||
|
data={
|
||||||
|
"managers": [
|
||||||
|
{
|
||||||
|
"name": m.name,
|
||||||
|
"description": m.description,
|
||||||
|
}
|
||||||
|
for m in self.ap.settings_mgr.get_manager_list()
|
||||||
|
]
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@self.route('/<manager_name>', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _(manager_name: str) -> str:
|
||||||
|
|
||||||
|
manager = self.ap.settings_mgr.get_manager(manager_name)
|
||||||
|
|
||||||
|
if manager is None:
|
||||||
|
return self.fail(1, '配置管理器不存在')
|
||||||
|
|
||||||
|
return self.success(
|
||||||
|
data={
|
||||||
|
"manager": {
|
||||||
|
"name": manager.name,
|
||||||
|
"description": manager.description,
|
||||||
|
"schema": manager.schema,
|
||||||
|
"file": manager.file.config_file_name,
|
||||||
|
"data": manager.data,
|
||||||
|
"doc_link": manager.doc_link
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@self.route('/<manager_name>/data', methods=['PUT'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _(manager_name: str) -> str:
|
||||||
|
data = await quart.request.json
|
||||||
|
manager = self.ap.settings_mgr.get_manager(manager_name)
|
||||||
|
|
||||||
|
if manager is None:
|
||||||
|
return self.fail(code=1, msg='配置管理器不存在')
|
||||||
|
|
||||||
|
# manager.data = data['data']
|
||||||
|
for k, v in data['data'].items():
|
||||||
|
manager.data[k] = v
|
||||||
|
|
||||||
|
await manager.dump_config()
|
||||||
|
return self.success(data={
|
||||||
|
"data": manager.data
|
||||||
|
})
|
||||||
23
pkg/api/http/controller/groups/stats.py
Normal file
23
pkg/api/http/controller/groups/stats.py
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import quart
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from .....core import app, taskmgr
|
||||||
|
from .. import group
|
||||||
|
|
||||||
|
|
||||||
|
@group.group_class('stats', '/api/v1/stats')
|
||||||
|
class StatsRouterGroup(group.RouterGroup):
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
@self.route('/basic', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
|
||||||
|
conv_count = 0
|
||||||
|
for session in self.ap.sess_mgr.session_list:
|
||||||
|
conv_count += len(session.conversations if session.conversations is not None else [])
|
||||||
|
|
||||||
|
return self.success(data={
|
||||||
|
'active_session_count': len(self.ap.sess_mgr.session_list),
|
||||||
|
'conversation_count': conv_count,
|
||||||
|
'query_count': self.ap.query_pool.query_id_counter,
|
||||||
|
})
|
||||||
63
pkg/api/http/controller/groups/system.py
Normal file
63
pkg/api/http/controller/groups/system.py
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
import quart
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from .....core import app, taskmgr
|
||||||
|
from .. import group
|
||||||
|
from .....utils import constants
|
||||||
|
|
||||||
|
|
||||||
|
@group.group_class('system', '/api/v1/system')
|
||||||
|
class SystemRouterGroup(group.RouterGroup):
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
@self.route('/info', methods=['GET'], auth_type=group.AuthType.NONE)
|
||||||
|
async def _() -> str:
|
||||||
|
return self.success(
|
||||||
|
data={
|
||||||
|
"version": constants.semantic_version,
|
||||||
|
"debug": constants.debug_mode,
|
||||||
|
"enabled_platform_count": len(self.ap.platform_mgr.adapters)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@self.route('/tasks', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
task_type = quart.request.args.get("type")
|
||||||
|
|
||||||
|
if task_type == '':
|
||||||
|
task_type = None
|
||||||
|
|
||||||
|
return self.success(
|
||||||
|
data=self.ap.task_mgr.get_tasks_dict(task_type)
|
||||||
|
)
|
||||||
|
|
||||||
|
@self.route('/tasks/<task_id>', methods=['GET'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _(task_id: str) -> str:
|
||||||
|
task = self.ap.task_mgr.get_task_by_id(int(task_id))
|
||||||
|
|
||||||
|
if task is None:
|
||||||
|
return self.http_status(404, 404, "Task not found")
|
||||||
|
|
||||||
|
return self.success(data=task.to_dict())
|
||||||
|
|
||||||
|
@self.route('/reload', methods=['POST'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
json_data = await quart.request.json
|
||||||
|
|
||||||
|
scope = json_data.get("scope")
|
||||||
|
|
||||||
|
await self.ap.reload(
|
||||||
|
scope=scope
|
||||||
|
)
|
||||||
|
return self.success()
|
||||||
|
|
||||||
|
@self.route('/_debug/exec', methods=['POST'], auth_type=group.AuthType.USER_TOKEN)
|
||||||
|
async def _() -> str:
|
||||||
|
if not constants.debug_mode:
|
||||||
|
return self.http_status(403, 403, "Forbidden")
|
||||||
|
|
||||||
|
py_code = await quart.request.data
|
||||||
|
|
||||||
|
ap = self.ap
|
||||||
|
|
||||||
|
return self.success(data=exec(py_code, {"ap": ap}))
|
||||||
47
pkg/api/http/controller/groups/user.py
Normal file
47
pkg/api/http/controller/groups/user.py
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import quart
|
||||||
|
import sqlalchemy
|
||||||
|
import argon2
|
||||||
|
|
||||||
|
from .. import group
|
||||||
|
from .....persistence.entities import user
|
||||||
|
|
||||||
|
|
||||||
|
@group.group_class('user', '/api/v1/user')
|
||||||
|
class UserRouterGroup(group.RouterGroup):
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
@self.route('/init', methods=['GET', 'POST'], auth_type=group.AuthType.NONE)
|
||||||
|
async def _() -> str:
|
||||||
|
if quart.request.method == 'GET':
|
||||||
|
return self.success(data={
|
||||||
|
'initialized': await self.ap.user_service.is_initialized()
|
||||||
|
})
|
||||||
|
|
||||||
|
if await self.ap.user_service.is_initialized():
|
||||||
|
return self.fail(1, '系统已初始化')
|
||||||
|
|
||||||
|
json_data = await quart.request.json
|
||||||
|
|
||||||
|
user_email = json_data['user']
|
||||||
|
password = json_data['password']
|
||||||
|
|
||||||
|
await self.ap.user_service.create_user(user_email, password)
|
||||||
|
|
||||||
|
return self.success()
|
||||||
|
|
||||||
|
@self.route('/auth', methods=['POST'], auth_type=group.AuthType.NONE)
|
||||||
|
async def _() -> str:
|
||||||
|
json_data = await quart.request.json
|
||||||
|
|
||||||
|
try:
|
||||||
|
token = await self.ap.user_service.authenticate(json_data['user'], json_data['password'])
|
||||||
|
except argon2.exceptions.VerifyMismatchError:
|
||||||
|
return self.fail(1, '用户名或密码错误')
|
||||||
|
|
||||||
|
return self.success(data={
|
||||||
|
'token': token
|
||||||
|
})
|
||||||
|
|
||||||
|
@self.route('/check-token', methods=['GET'])
|
||||||
|
async def _() -> str:
|
||||||
|
return self.success()
|
||||||
107
pkg/api/http/controller/main.py
Normal file
107
pkg/api/http/controller/main.py
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
import os
|
||||||
|
|
||||||
|
import quart
|
||||||
|
import quart_cors
|
||||||
|
|
||||||
|
from ....core import app, entities as core_entities
|
||||||
|
from .groups import logs, system, settings, plugins, stats, user
|
||||||
|
from . import group
|
||||||
|
|
||||||
|
|
||||||
|
class HTTPController:
|
||||||
|
|
||||||
|
ap: app.Application
|
||||||
|
|
||||||
|
quart_app: quart.Quart
|
||||||
|
|
||||||
|
def __init__(self, ap: app.Application) -> None:
|
||||||
|
self.ap = ap
|
||||||
|
self.quart_app = quart.Quart(__name__)
|
||||||
|
quart_cors.cors(self.quart_app, allow_origin="*")
|
||||||
|
|
||||||
|
async def initialize(self) -> None:
|
||||||
|
await self.register_routes()
|
||||||
|
|
||||||
|
async def run(self) -> None:
|
||||||
|
if self.ap.system_cfg.data["http-api"]["enable"]:
|
||||||
|
|
||||||
|
async def shutdown_trigger_placeholder():
|
||||||
|
while True:
|
||||||
|
await asyncio.sleep(1)
|
||||||
|
|
||||||
|
async def exception_handler(*args, **kwargs):
|
||||||
|
try:
|
||||||
|
await self.quart_app.run_task(
|
||||||
|
*args, **kwargs
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
self.ap.logger.error(f"启动 HTTP 服务失败: {e}")
|
||||||
|
|
||||||
|
self.ap.task_mgr.create_task(
|
||||||
|
exception_handler(
|
||||||
|
host=self.ap.system_cfg.data["http-api"]["host"],
|
||||||
|
port=self.ap.system_cfg.data["http-api"]["port"],
|
||||||
|
shutdown_trigger=shutdown_trigger_placeholder,
|
||||||
|
),
|
||||||
|
name="http-api-quart",
|
||||||
|
scopes=[core_entities.LifecycleControlScope.APPLICATION],
|
||||||
|
)
|
||||||
|
|
||||||
|
# await asyncio.sleep(5)
|
||||||
|
|
||||||
|
async def register_routes(self) -> None:
|
||||||
|
|
||||||
|
@self.quart_app.route("/healthz")
|
||||||
|
async def healthz():
|
||||||
|
return {"code": 0, "msg": "ok"}
|
||||||
|
|
||||||
|
for g in group.preregistered_groups:
|
||||||
|
ginst = g(self.ap, self.quart_app)
|
||||||
|
await ginst.initialize()
|
||||||
|
|
||||||
|
frontend_path = "web/dist"
|
||||||
|
|
||||||
|
@self.quart_app.route("/")
|
||||||
|
async def index():
|
||||||
|
return await quart.send_from_directory(
|
||||||
|
frontend_path,
|
||||||
|
"index.html",
|
||||||
|
mimetype="text/html"
|
||||||
|
)
|
||||||
|
|
||||||
|
@self.quart_app.route("/<path:path>")
|
||||||
|
async def static_file(path: str):
|
||||||
|
|
||||||
|
mimetype = None
|
||||||
|
|
||||||
|
if path.endswith(".html"):
|
||||||
|
mimetype = "text/html"
|
||||||
|
elif path.endswith(".js"):
|
||||||
|
mimetype = "application/javascript"
|
||||||
|
elif path.endswith(".css"):
|
||||||
|
mimetype = "text/css"
|
||||||
|
elif path.endswith(".png"):
|
||||||
|
mimetype = "image/png"
|
||||||
|
elif path.endswith(".jpg"):
|
||||||
|
mimetype = "image/jpeg"
|
||||||
|
elif path.endswith(".jpeg"):
|
||||||
|
mimetype = "image/jpeg"
|
||||||
|
elif path.endswith(".gif"):
|
||||||
|
mimetype = "image/gif"
|
||||||
|
elif path.endswith(".svg"):
|
||||||
|
mimetype = "image/svg+xml"
|
||||||
|
elif path.endswith(".ico"):
|
||||||
|
mimetype = "image/x-icon"
|
||||||
|
elif path.endswith(".json"):
|
||||||
|
mimetype = "application/json"
|
||||||
|
elif path.endswith(".txt"):
|
||||||
|
mimetype = "text/plain"
|
||||||
|
|
||||||
|
return await quart.send_from_directory(
|
||||||
|
frontend_path,
|
||||||
|
path,
|
||||||
|
mimetype=mimetype
|
||||||
|
)
|
||||||
73
pkg/api/http/service/user.py
Normal file
73
pkg/api/http/service/user.py
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import sqlalchemy
|
||||||
|
import argon2
|
||||||
|
import jwt
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
from ....core import app
|
||||||
|
from ....persistence.entities import user
|
||||||
|
from ....utils import constants
|
||||||
|
|
||||||
|
|
||||||
|
class UserService:
|
||||||
|
|
||||||
|
ap: app.Application
|
||||||
|
|
||||||
|
def __init__(self, ap: app.Application) -> None:
|
||||||
|
self.ap = ap
|
||||||
|
|
||||||
|
async def is_initialized(self) -> bool:
|
||||||
|
result = await self.ap.persistence_mgr.execute_async(
|
||||||
|
sqlalchemy.select(user.User).limit(1)
|
||||||
|
)
|
||||||
|
|
||||||
|
result_list = result.all()
|
||||||
|
return result_list is not None and len(result_list) > 0
|
||||||
|
|
||||||
|
async def create_user(self, user_email: str, password: str) -> None:
|
||||||
|
ph = argon2.PasswordHasher()
|
||||||
|
|
||||||
|
hashed_password = ph.hash(password)
|
||||||
|
|
||||||
|
await self.ap.persistence_mgr.execute_async(
|
||||||
|
sqlalchemy.insert(user.User).values(
|
||||||
|
user=user_email,
|
||||||
|
password=hashed_password
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
async def authenticate(self, user_email: str, password: str) -> str | None:
|
||||||
|
result = await self.ap.persistence_mgr.execute_async(
|
||||||
|
sqlalchemy.select(user.User).where(user.User.user == user_email)
|
||||||
|
)
|
||||||
|
|
||||||
|
result_list = result.all()
|
||||||
|
|
||||||
|
if result_list is None or len(result_list) == 0:
|
||||||
|
raise ValueError('用户不存在')
|
||||||
|
|
||||||
|
user_obj = result_list[0]
|
||||||
|
|
||||||
|
ph = argon2.PasswordHasher()
|
||||||
|
|
||||||
|
ph.verify(user_obj.password, password)
|
||||||
|
|
||||||
|
return await self.generate_jwt_token(user_email)
|
||||||
|
|
||||||
|
async def generate_jwt_token(self, user_email: str) -> str:
|
||||||
|
jwt_secret = self.ap.instance_secret_meta.data['jwt_secret']
|
||||||
|
jwt_expire = self.ap.system_cfg.data['http-api']['jwt-expire']
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
'user': user_email,
|
||||||
|
'iss': 'LangBot-'+constants.edition,
|
||||||
|
'exp': datetime.datetime.now() + datetime.timedelta(seconds=jwt_expire)
|
||||||
|
}
|
||||||
|
|
||||||
|
return jwt.encode(payload, jwt_secret, algorithm='HS256')
|
||||||
|
|
||||||
|
async def verify_jwt_token(self, token: str) -> str:
|
||||||
|
jwt_secret = self.ap.instance_secret_meta.data['jwt_secret']
|
||||||
|
|
||||||
|
return jwt.decode(token, jwt_secret, algorithms=['HS256'])['user']
|
||||||
88
pkg/audit/center/apigroup.py
Normal file
88
pkg/audit/center/apigroup.py
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import abc
|
||||||
|
import uuid
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
import aiohttp
|
||||||
|
import requests
|
||||||
|
|
||||||
|
from ...core import app, entities as core_entities
|
||||||
|
|
||||||
|
|
||||||
|
class APIGroup(metaclass=abc.ABCMeta):
|
||||||
|
"""API 组抽象类"""
|
||||||
|
|
||||||
|
_basic_info: dict = None
|
||||||
|
_runtime_info: dict = None
|
||||||
|
|
||||||
|
prefix = None
|
||||||
|
|
||||||
|
ap: app.Application
|
||||||
|
|
||||||
|
def __init__(self, prefix: str, ap: app.Application):
|
||||||
|
self.prefix = prefix
|
||||||
|
self.ap = ap
|
||||||
|
|
||||||
|
async def _do(
|
||||||
|
self,
|
||||||
|
method: str,
|
||||||
|
path: str,
|
||||||
|
data: dict = None,
|
||||||
|
params: dict = None,
|
||||||
|
headers: dict = {},
|
||||||
|
**kwargs,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
执行请求
|
||||||
|
"""
|
||||||
|
self._runtime_info["account_id"] = "-1"
|
||||||
|
|
||||||
|
url = self.prefix + path
|
||||||
|
data = json.dumps(data)
|
||||||
|
headers["Content-Type"] = "application/json"
|
||||||
|
|
||||||
|
try:
|
||||||
|
async with aiohttp.ClientSession() as session:
|
||||||
|
async with session.request(
|
||||||
|
method, url, data=data, params=params, headers=headers, **kwargs
|
||||||
|
) as resp:
|
||||||
|
self.ap.logger.debug("data: %s", data)
|
||||||
|
self.ap.logger.debug("ret: %s", await resp.text())
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.ap.logger.debug(f"上报失败: {e}")
|
||||||
|
|
||||||
|
async def do(
|
||||||
|
self,
|
||||||
|
method: str,
|
||||||
|
path: str,
|
||||||
|
data: dict = None,
|
||||||
|
params: dict = None,
|
||||||
|
headers: dict = {},
|
||||||
|
**kwargs,
|
||||||
|
) -> asyncio.Task:
|
||||||
|
"""执行请求"""
|
||||||
|
|
||||||
|
return self.ap.task_mgr.create_task(
|
||||||
|
self._do(method, path, data, params, headers, **kwargs),
|
||||||
|
kind="telemetry-operation",
|
||||||
|
name=f"{method} {path}",
|
||||||
|
scopes=[core_entities.LifecycleControlScope.APPLICATION],
|
||||||
|
).task
|
||||||
|
|
||||||
|
def gen_rid(self):
|
||||||
|
"""生成一个请求 ID"""
|
||||||
|
return str(uuid.uuid4())
|
||||||
|
|
||||||
|
def basic_info(self):
|
||||||
|
"""获取基本信息"""
|
||||||
|
basic_info = APIGroup._basic_info.copy()
|
||||||
|
basic_info["rid"] = self.gen_rid()
|
||||||
|
return basic_info
|
||||||
|
|
||||||
|
def runtime_info(self):
|
||||||
|
"""获取运行时信息"""
|
||||||
|
return APIGroup._runtime_info
|
||||||
55
pkg/audit/center/groups/main.py
Normal file
55
pkg/audit/center/groups/main.py
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from .. import apigroup
|
||||||
|
from ....core import app
|
||||||
|
|
||||||
|
|
||||||
|
class V2MainDataAPI(apigroup.APIGroup):
|
||||||
|
"""主程序相关 数据API"""
|
||||||
|
|
||||||
|
def __init__(self, prefix: str, ap: app.Application):
|
||||||
|
self.ap = ap
|
||||||
|
super().__init__(prefix+"/main", ap)
|
||||||
|
|
||||||
|
async def do(self, *args, **kwargs):
|
||||||
|
if not self.ap.system_cfg.data['report-usage']:
|
||||||
|
return None
|
||||||
|
return await super().do(*args, **kwargs)
|
||||||
|
|
||||||
|
async def post_update_record(
|
||||||
|
self,
|
||||||
|
spent_seconds: int,
|
||||||
|
infer_reason: str,
|
||||||
|
old_version: str,
|
||||||
|
new_version: str,
|
||||||
|
):
|
||||||
|
"""提交更新记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/update",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"update_info": {
|
||||||
|
"spent_seconds": spent_seconds,
|
||||||
|
"infer_reason": infer_reason,
|
||||||
|
"old_version": old_version,
|
||||||
|
"new_version": new_version,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
async def post_announcement_showed(
|
||||||
|
self,
|
||||||
|
ids: list[int],
|
||||||
|
):
|
||||||
|
"""提交公告已阅"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/announcement",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"announcement_info": {
|
||||||
|
"ids": ids,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
65
pkg/audit/center/groups/plugin.py
Normal file
65
pkg/audit/center/groups/plugin.py
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from ....core import app
|
||||||
|
from .. import apigroup
|
||||||
|
|
||||||
|
|
||||||
|
class V2PluginDataAPI(apigroup.APIGroup):
|
||||||
|
"""插件数据相关 API"""
|
||||||
|
|
||||||
|
def __init__(self, prefix: str, ap: app.Application):
|
||||||
|
self.ap = ap
|
||||||
|
super().__init__(prefix+"/plugin", ap)
|
||||||
|
|
||||||
|
async def do(self, *args, **kwargs):
|
||||||
|
if not self.ap.system_cfg.data['report-usage']:
|
||||||
|
return None
|
||||||
|
return await super().do(*args, **kwargs)
|
||||||
|
|
||||||
|
async def post_install_record(
|
||||||
|
self,
|
||||||
|
plugin: dict
|
||||||
|
):
|
||||||
|
"""提交插件安装记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/install",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"plugin": plugin,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
async def post_remove_record(
|
||||||
|
self,
|
||||||
|
plugin: dict
|
||||||
|
):
|
||||||
|
"""提交插件卸载记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/remove",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"plugin": plugin,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
async def post_update_record(
|
||||||
|
self,
|
||||||
|
plugin: dict,
|
||||||
|
old_version: str,
|
||||||
|
new_version: str,
|
||||||
|
):
|
||||||
|
"""提交插件更新记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/update",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"plugin": plugin,
|
||||||
|
"update_info": {
|
||||||
|
"old_version": old_version,
|
||||||
|
"new_version": new_version,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
88
pkg/audit/center/groups/usage.py
Normal file
88
pkg/audit/center/groups/usage.py
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from .. import apigroup
|
||||||
|
from ....core import app
|
||||||
|
|
||||||
|
|
||||||
|
class V2UsageDataAPI(apigroup.APIGroup):
|
||||||
|
"""使用量数据相关 API"""
|
||||||
|
|
||||||
|
def __init__(self, prefix: str, ap: app.Application):
|
||||||
|
self.ap = ap
|
||||||
|
super().__init__(prefix+"/usage", ap)
|
||||||
|
|
||||||
|
async def do(self, *args, **kwargs):
|
||||||
|
if not self.ap.system_cfg.data['report-usage']:
|
||||||
|
return None
|
||||||
|
return await super().do(*args, **kwargs)
|
||||||
|
|
||||||
|
async def post_query_record(
|
||||||
|
self,
|
||||||
|
session_type: str,
|
||||||
|
session_id: str,
|
||||||
|
query_ability_provider: str,
|
||||||
|
usage: int,
|
||||||
|
model_name: str,
|
||||||
|
response_seconds: int,
|
||||||
|
retry_times: int,
|
||||||
|
):
|
||||||
|
"""提交请求记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/query",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"runtime": self.runtime_info(),
|
||||||
|
"session_info": {
|
||||||
|
"type": session_type,
|
||||||
|
"id": session_id,
|
||||||
|
},
|
||||||
|
"query_info": {
|
||||||
|
"ability_provider": query_ability_provider,
|
||||||
|
"usage": usage,
|
||||||
|
"model_name": model_name,
|
||||||
|
"response_seconds": response_seconds,
|
||||||
|
"retry_times": retry_times,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
async def post_event_record(
|
||||||
|
self,
|
||||||
|
plugins: list[dict],
|
||||||
|
event_name: str,
|
||||||
|
):
|
||||||
|
"""提交事件触发记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/event",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"runtime": self.runtime_info(),
|
||||||
|
"plugins": plugins,
|
||||||
|
"event_info": {
|
||||||
|
"name": event_name,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
async def post_function_record(
|
||||||
|
self,
|
||||||
|
plugin: dict,
|
||||||
|
function_name: str,
|
||||||
|
function_description: str,
|
||||||
|
):
|
||||||
|
"""提交内容函数使用记录"""
|
||||||
|
return await self.do(
|
||||||
|
"POST",
|
||||||
|
"/function",
|
||||||
|
data={
|
||||||
|
"basic": self.basic_info(),
|
||||||
|
"plugin": plugin,
|
||||||
|
"function_info": {
|
||||||
|
"name": function_name,
|
||||||
|
"description": function_description,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
35
pkg/audit/center/v2.py
Normal file
35
pkg/audit/center/v2.py
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from . import apigroup
|
||||||
|
from .groups import main
|
||||||
|
from .groups import usage
|
||||||
|
from .groups import plugin
|
||||||
|
from ...core import app
|
||||||
|
|
||||||
|
|
||||||
|
class V2CenterAPI:
|
||||||
|
"""中央服务器 v2 API 交互类"""
|
||||||
|
|
||||||
|
main: main.V2MainDataAPI = None
|
||||||
|
"""主 API 组"""
|
||||||
|
|
||||||
|
usage: usage.V2UsageDataAPI = None
|
||||||
|
"""使用量 API 组"""
|
||||||
|
|
||||||
|
plugin: plugin.V2PluginDataAPI = None
|
||||||
|
"""插件 API 组"""
|
||||||
|
|
||||||
|
def __init__(self, ap: app.Application, backend_url: str, basic_info: dict = None, runtime_info: dict = None):
|
||||||
|
"""初始化"""
|
||||||
|
|
||||||
|
logging.debug("basic_info: %s, runtime_info: %s", basic_info, runtime_info)
|
||||||
|
|
||||||
|
apigroup.APIGroup._basic_info = basic_info
|
||||||
|
apigroup.APIGroup._runtime_info = runtime_info
|
||||||
|
|
||||||
|
self.main = main.V2MainDataAPI(backend_url, ap)
|
||||||
|
self.usage = usage.V2UsageDataAPI(backend_url, ap)
|
||||||
|
self.plugin = plugin.V2PluginDataAPI(backend_url, ap)
|
||||||
|
|
||||||
@@ -1,133 +0,0 @@
|
|||||||
"""
|
|
||||||
使用量统计以及数据上报功能实现
|
|
||||||
"""
|
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import json
|
|
||||||
import logging
|
|
||||||
|
|
||||||
import requests
|
|
||||||
|
|
||||||
import pkg.utils.context
|
|
||||||
import pkg.utils.updater
|
|
||||||
|
|
||||||
|
|
||||||
class DataGatherer:
|
|
||||||
"""数据收集器"""
|
|
||||||
|
|
||||||
usage = {}
|
|
||||||
"""各api-key的使用量
|
|
||||||
|
|
||||||
以key值md5为key,{
|
|
||||||
"text": {
|
|
||||||
"text-davinci-003": 文字量:int,
|
|
||||||
},
|
|
||||||
"image": {
|
|
||||||
"256x256": 图片数量:int,
|
|
||||||
}
|
|
||||||
}为值的字典"""
|
|
||||||
|
|
||||||
version_str = "undetermined"
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.load_from_db()
|
|
||||||
try:
|
|
||||||
self.version_str = pkg.utils.updater.get_current_tag() # 从updater模块获取版本号
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def report_to_server(self, subservice_name: str, count: int):
|
|
||||||
"""向中央服务器报告使用量
|
|
||||||
|
|
||||||
只会报告此次请求的使用量,不会报告总量。
|
|
||||||
不包含除版本号、使用类型、使用量以外的任何信息,仅供开发者分析使用情况。
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
config = pkg.utils.context.get_config()
|
|
||||||
if not config.report_usage:
|
|
||||||
return
|
|
||||||
res = requests.get("http://reports.rockchin.top:18989/usage?service_name=qchatgpt.{}&version={}&count={}".format(subservice_name, self.version_str, count))
|
|
||||||
if res.status_code != 200 or res.text != "ok":
|
|
||||||
logging.warning("report to server failed, status_code: {}, text: {}".format(res.status_code, res.text))
|
|
||||||
except:
|
|
||||||
return
|
|
||||||
|
|
||||||
def get_usage(self, key_md5):
|
|
||||||
return self.usage[key_md5] if key_md5 in self.usage else {}
|
|
||||||
|
|
||||||
def report_text_model_usage(self, model, total_tokens):
|
|
||||||
"""调用方报告文字模型请求文字使用量"""
|
|
||||||
|
|
||||||
key_md5 = pkg.utils.context.get_openai_manager().key_mgr.get_using_key_md5() # 以key的md5进行储存
|
|
||||||
|
|
||||||
if key_md5 not in self.usage:
|
|
||||||
self.usage[key_md5] = {}
|
|
||||||
|
|
||||||
if "text" not in self.usage[key_md5]:
|
|
||||||
self.usage[key_md5]["text"] = {}
|
|
||||||
|
|
||||||
if model not in self.usage[key_md5]["text"]:
|
|
||||||
self.usage[key_md5]["text"][model] = 0
|
|
||||||
|
|
||||||
length = total_tokens
|
|
||||||
self.usage[key_md5]["text"][model] += length
|
|
||||||
self.dump_to_db()
|
|
||||||
|
|
||||||
self.report_to_server("text", length)
|
|
||||||
|
|
||||||
def report_image_model_usage(self, size):
|
|
||||||
"""调用方报告图片模型请求图片使用量"""
|
|
||||||
|
|
||||||
key_md5 = pkg.utils.context.get_openai_manager().key_mgr.get_using_key_md5()
|
|
||||||
|
|
||||||
if key_md5 not in self.usage:
|
|
||||||
self.usage[key_md5] = {}
|
|
||||||
|
|
||||||
if "image" not in self.usage[key_md5]:
|
|
||||||
self.usage[key_md5]["image"] = {}
|
|
||||||
|
|
||||||
if size not in self.usage[key_md5]["image"]:
|
|
||||||
self.usage[key_md5]["image"][size] = 0
|
|
||||||
|
|
||||||
self.usage[key_md5]["image"][size] += 1
|
|
||||||
self.dump_to_db()
|
|
||||||
|
|
||||||
self.report_to_server("image", 1)
|
|
||||||
|
|
||||||
def get_text_length_of_key(self, key):
|
|
||||||
"""获取指定api-key (明文) 的文字总使用量(本地记录)"""
|
|
||||||
key_md5 = hashlib.md5(key.encode('utf-8')).hexdigest()
|
|
||||||
if key_md5 not in self.usage:
|
|
||||||
return 0
|
|
||||||
if "text" not in self.usage[key_md5]:
|
|
||||||
return 0
|
|
||||||
# 遍历其中所有模型,求和
|
|
||||||
return sum(self.usage[key_md5]["text"].values())
|
|
||||||
|
|
||||||
def get_image_count_of_key(self, key):
|
|
||||||
"""获取指定api-key (明文) 的图片总使用量(本地记录)"""
|
|
||||||
|
|
||||||
key_md5 = hashlib.md5(key.encode('utf-8')).hexdigest()
|
|
||||||
if key_md5 not in self.usage:
|
|
||||||
return 0
|
|
||||||
if "image" not in self.usage[key_md5]:
|
|
||||||
return 0
|
|
||||||
# 遍历其中所有模型,求和
|
|
||||||
return sum(self.usage[key_md5]["image"].values())
|
|
||||||
|
|
||||||
def get_total_text_length(self):
|
|
||||||
"""获取所有api-key的文字总使用量(本地记录)"""
|
|
||||||
total = 0
|
|
||||||
for key in self.usage:
|
|
||||||
if "text" not in self.usage[key]:
|
|
||||||
continue
|
|
||||||
total += sum(self.usage[key]["text"].values())
|
|
||||||
return total
|
|
||||||
|
|
||||||
def dump_to_db(self):
|
|
||||||
pkg.utils.context.get_database_manager().dump_usage_json(self.usage)
|
|
||||||
|
|
||||||
def load_from_db(self):
|
|
||||||
json_str = pkg.utils.context.get_database_manager().load_usage_json()
|
|
||||||
if json_str is not None:
|
|
||||||
self.usage = json.loads(json_str)
|
|
||||||
85
pkg/audit/identifier.py
Normal file
85
pkg/audit/identifier.py
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
# 实例 识别码 控制
|
||||||
|
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
import json
|
||||||
|
import time
|
||||||
|
|
||||||
|
|
||||||
|
identifier = {
|
||||||
|
'host_id': '',
|
||||||
|
'instance_id': '',
|
||||||
|
'host_create_ts': 0,
|
||||||
|
'instance_create_ts': 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
HOST_ID_FILE = os.path.expanduser('~/.langbot/host_id.json')
|
||||||
|
INSTANCE_ID_FILE = 'data/labels/instance_id.json'
|
||||||
|
|
||||||
|
def init():
|
||||||
|
global identifier
|
||||||
|
|
||||||
|
if not os.path.exists(os.path.expanduser('~/.langbot')):
|
||||||
|
os.mkdir(os.path.expanduser('~/.langbot'))
|
||||||
|
|
||||||
|
if not os.path.exists(HOST_ID_FILE):
|
||||||
|
new_host_id = 'host_'+str(uuid.uuid4())
|
||||||
|
new_host_create_ts = int(time.time())
|
||||||
|
|
||||||
|
with open(HOST_ID_FILE, 'w') as f:
|
||||||
|
json.dump({
|
||||||
|
'host_id': new_host_id,
|
||||||
|
'host_create_ts': new_host_create_ts
|
||||||
|
}, f)
|
||||||
|
|
||||||
|
identifier['host_id'] = new_host_id
|
||||||
|
identifier['host_create_ts'] = new_host_create_ts
|
||||||
|
else:
|
||||||
|
loaded_host_id = ''
|
||||||
|
loaded_host_create_ts = 0
|
||||||
|
|
||||||
|
with open(HOST_ID_FILE, 'r') as f:
|
||||||
|
file_content = json.load(f)
|
||||||
|
loaded_host_id = file_content['host_id']
|
||||||
|
loaded_host_create_ts = file_content['host_create_ts']
|
||||||
|
|
||||||
|
identifier['host_id'] = loaded_host_id
|
||||||
|
identifier['host_create_ts'] = loaded_host_create_ts
|
||||||
|
|
||||||
|
# 检查实例 id
|
||||||
|
if os.path.exists(INSTANCE_ID_FILE):
|
||||||
|
instance_id = {}
|
||||||
|
with open(INSTANCE_ID_FILE, 'r') as f:
|
||||||
|
instance_id = json.load(f)
|
||||||
|
|
||||||
|
if instance_id['host_id'] != identifier['host_id']: # 如果实例 id 不是当前主机的,删除
|
||||||
|
os.remove(INSTANCE_ID_FILE)
|
||||||
|
|
||||||
|
if not os.path.exists(INSTANCE_ID_FILE):
|
||||||
|
new_instance_id = 'instance_'+str(uuid.uuid4())
|
||||||
|
new_instance_create_ts = int(time.time())
|
||||||
|
|
||||||
|
with open(INSTANCE_ID_FILE, 'w') as f:
|
||||||
|
json.dump({
|
||||||
|
'host_id': identifier['host_id'],
|
||||||
|
'instance_id': new_instance_id,
|
||||||
|
'instance_create_ts': new_instance_create_ts
|
||||||
|
}, f)
|
||||||
|
|
||||||
|
identifier['instance_id'] = new_instance_id
|
||||||
|
identifier['instance_create_ts'] = new_instance_create_ts
|
||||||
|
else:
|
||||||
|
loaded_instance_id = ''
|
||||||
|
loaded_instance_create_ts = 0
|
||||||
|
|
||||||
|
with open(INSTANCE_ID_FILE, 'r') as f:
|
||||||
|
file_content = json.load(f)
|
||||||
|
loaded_instance_id = file_content['instance_id']
|
||||||
|
loaded_instance_create_ts = file_content['instance_create_ts']
|
||||||
|
|
||||||
|
identifier['instance_id'] = loaded_instance_id
|
||||||
|
identifier['instance_create_ts'] = loaded_instance_create_ts
|
||||||
|
|
||||||
|
def print_out():
|
||||||
|
global identifier
|
||||||
|
print(identifier)
|
||||||
0
pkg/command/__init__.py
Normal file
0
pkg/command/__init__.py
Normal file
129
pkg/command/cmdmgr.py
Normal file
129
pkg/command/cmdmgr.py
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from ..core import app, entities as core_entities
|
||||||
|
from ..provider import entities as llm_entities
|
||||||
|
from . import entities, operator, errors
|
||||||
|
from ..config import manager as cfg_mgr
|
||||||
|
|
||||||
|
# 引入所有算子以便注册
|
||||||
|
from .operators import func, plugin, default, reset, list as list_cmd, last, next, delc, resend, prompt, cmd, help, version, update, ollama, model
|
||||||
|
|
||||||
|
|
||||||
|
class CommandManager:
|
||||||
|
"""命令管理器
|
||||||
|
"""
|
||||||
|
|
||||||
|
ap: app.Application
|
||||||
|
|
||||||
|
cmd_list: list[operator.CommandOperator]
|
||||||
|
"""
|
||||||
|
运行时命令列表,扁平存储,各个对象包含对应的子节点引用
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, ap: app.Application):
|
||||||
|
self.ap = ap
|
||||||
|
|
||||||
|
async def initialize(self):
|
||||||
|
|
||||||
|
# 设置各个类的路径
|
||||||
|
def set_path(cls: operator.CommandOperator, ancestors: list[str]):
|
||||||
|
cls.path = '.'.join(ancestors + [cls.name])
|
||||||
|
for op in operator.preregistered_operators:
|
||||||
|
if op.parent_class == cls:
|
||||||
|
set_path(op, ancestors + [cls.name])
|
||||||
|
|
||||||
|
for cls in operator.preregistered_operators:
|
||||||
|
if cls.parent_class is None:
|
||||||
|
set_path(cls, [])
|
||||||
|
|
||||||
|
# 应用命令权限配置
|
||||||
|
for cls in operator.preregistered_operators:
|
||||||
|
if cls.path in self.ap.command_cfg.data['privilege']:
|
||||||
|
cls.lowest_privilege = self.ap.command_cfg.data['privilege'][cls.path]
|
||||||
|
|
||||||
|
# 实例化所有类
|
||||||
|
self.cmd_list = [cls(self.ap) for cls in operator.preregistered_operators]
|
||||||
|
|
||||||
|
# 设置所有类的子节点
|
||||||
|
for cmd in self.cmd_list:
|
||||||
|
cmd.children = [child for child in self.cmd_list if child.parent_class == cmd.__class__]
|
||||||
|
|
||||||
|
# 初始化所有类
|
||||||
|
for cmd in self.cmd_list:
|
||||||
|
await cmd.initialize()
|
||||||
|
|
||||||
|
async def _execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext,
|
||||||
|
operator_list: list[operator.CommandOperator],
|
||||||
|
operator: operator.CommandOperator = None
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
"""执行命令
|
||||||
|
"""
|
||||||
|
|
||||||
|
found = False
|
||||||
|
if len(context.crt_params) > 0: # 查找下一个参数是否对应此节点的某个子节点名
|
||||||
|
for oper in operator_list:
|
||||||
|
if (context.crt_params[0] == oper.name \
|
||||||
|
or context.crt_params[0] in oper.alias) \
|
||||||
|
and (oper.parent_class is None or oper.parent_class == operator.__class__):
|
||||||
|
found = True
|
||||||
|
|
||||||
|
context.crt_command = context.crt_params[0]
|
||||||
|
context.crt_params = context.crt_params[1:]
|
||||||
|
|
||||||
|
async for ret in self._execute(
|
||||||
|
context,
|
||||||
|
oper.children,
|
||||||
|
oper
|
||||||
|
):
|
||||||
|
yield ret
|
||||||
|
break
|
||||||
|
|
||||||
|
if not found: # 如果下一个参数未在此节点的子节点中找到,则执行此节点或者报错
|
||||||
|
if operator is None:
|
||||||
|
yield entities.CommandReturn(
|
||||||
|
error=errors.CommandNotFoundError(context.crt_params[0])
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if operator.lowest_privilege > context.privilege:
|
||||||
|
yield entities.CommandReturn(
|
||||||
|
error=errors.CommandPrivilegeError(operator.name)
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
async for ret in operator.execute(context):
|
||||||
|
yield ret
|
||||||
|
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
command_text: str,
|
||||||
|
query: core_entities.Query,
|
||||||
|
session: core_entities.Session
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
"""执行命令
|
||||||
|
"""
|
||||||
|
|
||||||
|
privilege = 1
|
||||||
|
|
||||||
|
if f'{query.launcher_type.value}_{query.launcher_id}' in self.ap.system_cfg.data['admin-sessions']:
|
||||||
|
privilege = 2
|
||||||
|
|
||||||
|
ctx = entities.ExecuteContext(
|
||||||
|
query=query,
|
||||||
|
session=session,
|
||||||
|
command_text=command_text,
|
||||||
|
command='',
|
||||||
|
crt_command='',
|
||||||
|
params=command_text.split(' '),
|
||||||
|
crt_params=command_text.split(' '),
|
||||||
|
privilege=privilege
|
||||||
|
)
|
||||||
|
|
||||||
|
async for ret in self._execute(
|
||||||
|
ctx,
|
||||||
|
self.cmd_list
|
||||||
|
):
|
||||||
|
yield ret
|
||||||
76
pkg/command/entities.py
Normal file
76
pkg/command/entities.py
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
import pydantic.v1 as pydantic
|
||||||
|
|
||||||
|
from ..core import app, entities as core_entities
|
||||||
|
from . import errors, operator
|
||||||
|
from ..platform.types import message as platform_message
|
||||||
|
|
||||||
|
|
||||||
|
class CommandReturn(pydantic.BaseModel):
|
||||||
|
"""命令返回值
|
||||||
|
"""
|
||||||
|
|
||||||
|
text: typing.Optional[str] = None
|
||||||
|
"""文本
|
||||||
|
"""
|
||||||
|
|
||||||
|
image: typing.Optional[platform_message.Image] = None
|
||||||
|
"""弃用"""
|
||||||
|
|
||||||
|
image_url: typing.Optional[str] = None
|
||||||
|
"""图片链接
|
||||||
|
"""
|
||||||
|
|
||||||
|
error: typing.Optional[errors.CommandError]= None
|
||||||
|
"""错误
|
||||||
|
"""
|
||||||
|
|
||||||
|
class Config:
|
||||||
|
arbitrary_types_allowed = True
|
||||||
|
|
||||||
|
|
||||||
|
class ExecuteContext(pydantic.BaseModel):
|
||||||
|
"""单次命令执行上下文
|
||||||
|
"""
|
||||||
|
|
||||||
|
query: core_entities.Query
|
||||||
|
"""本次消息的请求对象"""
|
||||||
|
|
||||||
|
session: core_entities.Session
|
||||||
|
"""本次消息所属的会话对象"""
|
||||||
|
|
||||||
|
command_text: str
|
||||||
|
"""命令完整文本"""
|
||||||
|
|
||||||
|
command: str
|
||||||
|
"""命令名称"""
|
||||||
|
|
||||||
|
crt_command: str
|
||||||
|
"""当前命令
|
||||||
|
|
||||||
|
多级命令中crt_command为当前命令,command为根命令。
|
||||||
|
例如:!plugin on Webwlkr
|
||||||
|
处理到plugin时,command为plugin,crt_command为plugin
|
||||||
|
处理到on时,command为plugin,crt_command为on
|
||||||
|
"""
|
||||||
|
|
||||||
|
params: list[str]
|
||||||
|
"""命令参数
|
||||||
|
|
||||||
|
整个命令以空格分割后的参数列表
|
||||||
|
"""
|
||||||
|
|
||||||
|
crt_params: list[str]
|
||||||
|
"""当前命令参数
|
||||||
|
|
||||||
|
多级命令中crt_params为当前命令参数,params为根命令参数。
|
||||||
|
例如:!plugin on Webwlkr
|
||||||
|
处理到plugin时,params为['on', 'Webwlkr'],crt_params为['on', 'Webwlkr']
|
||||||
|
处理到on时,params为['on', 'Webwlkr'],crt_params为['Webwlkr']
|
||||||
|
"""
|
||||||
|
|
||||||
|
privilege: int
|
||||||
|
"""发起人权限"""
|
||||||
33
pkg/command/errors.py
Normal file
33
pkg/command/errors.py
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
|
||||||
|
|
||||||
|
class CommandError(Exception):
|
||||||
|
|
||||||
|
def __init__(self, message: str = None):
|
||||||
|
self.message = message
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.message
|
||||||
|
|
||||||
|
|
||||||
|
class CommandNotFoundError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, message: str = None):
|
||||||
|
super().__init__("未知命令: "+message)
|
||||||
|
|
||||||
|
|
||||||
|
class CommandPrivilegeError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, message: str = None):
|
||||||
|
super().__init__("权限不足: "+message)
|
||||||
|
|
||||||
|
|
||||||
|
class ParamNotEnoughError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, message: str = None):
|
||||||
|
super().__init__("参数不足: "+message)
|
||||||
|
|
||||||
|
|
||||||
|
class CommandOperationError(CommandError):
|
||||||
|
|
||||||
|
def __init__(self, message: str = None):
|
||||||
|
super().__init__("操作失败: "+message)
|
||||||
113
pkg/command/operator.py
Normal file
113
pkg/command/operator.py
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import abc
|
||||||
|
|
||||||
|
from ..core import app, entities as core_entities
|
||||||
|
from . import entities
|
||||||
|
|
||||||
|
|
||||||
|
preregistered_operators: list[typing.Type[CommandOperator]] = []
|
||||||
|
"""预注册命令算子列表。在初始化时,所有算子类会被注册到此列表中。"""
|
||||||
|
|
||||||
|
|
||||||
|
def operator_class(
|
||||||
|
name: str,
|
||||||
|
help: str = "",
|
||||||
|
usage: str = None,
|
||||||
|
alias: list[str] = [],
|
||||||
|
privilege: int=1, # 1为普通用户,2为管理员
|
||||||
|
parent_class: typing.Type[CommandOperator] = None
|
||||||
|
) -> typing.Callable[[typing.Type[CommandOperator]], typing.Type[CommandOperator]]:
|
||||||
|
"""命令类装饰器
|
||||||
|
|
||||||
|
Args:
|
||||||
|
name (str): 名称
|
||||||
|
help (str, optional): 帮助信息. Defaults to "".
|
||||||
|
usage (str, optional): 使用说明. Defaults to None.
|
||||||
|
alias (list[str], optional): 别名. Defaults to [].
|
||||||
|
privilege (int, optional): 权限,1为普通用户可用,2为仅管理员可用. Defaults to 1.
|
||||||
|
parent_class (typing.Type[CommandOperator], optional): 父节点,若为None则为顶级命令. Defaults to None.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
typing.Callable[[typing.Type[CommandOperator]], typing.Type[CommandOperator]]: 装饰器
|
||||||
|
"""
|
||||||
|
|
||||||
|
def decorator(cls: typing.Type[CommandOperator]) -> typing.Type[CommandOperator]:
|
||||||
|
assert issubclass(cls, CommandOperator)
|
||||||
|
|
||||||
|
cls.name = name
|
||||||
|
cls.alias = alias
|
||||||
|
cls.help = help
|
||||||
|
cls.usage = usage
|
||||||
|
cls.parent_class = parent_class
|
||||||
|
cls.lowest_privilege = privilege
|
||||||
|
|
||||||
|
preregistered_operators.append(cls)
|
||||||
|
|
||||||
|
return cls
|
||||||
|
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
|
||||||
|
class CommandOperator(metaclass=abc.ABCMeta):
|
||||||
|
"""命令算子抽象类
|
||||||
|
|
||||||
|
以下的参数均不需要在子类中设置,只需要在使用装饰器注册类时作为参数传递即可。
|
||||||
|
命令支持级联,即一个命令可以有多个子命令,子命令可以有子命令,以此类推。
|
||||||
|
处理命令时,若有子命令,会以当前参数列表的第一个参数去匹配子命令,若匹配成功,则转移到子命令中执行。
|
||||||
|
若没有匹配成功或没有子命令,则执行当前命令。
|
||||||
|
"""
|
||||||
|
|
||||||
|
ap: app.Application
|
||||||
|
|
||||||
|
name: str
|
||||||
|
"""名称,搜索到时若符合则使用"""
|
||||||
|
|
||||||
|
path: str
|
||||||
|
"""路径,所有父节点的name的连接,用于定义命令权限,由管理器在初始化时自动设置。
|
||||||
|
"""
|
||||||
|
|
||||||
|
alias: list[str]
|
||||||
|
"""同name"""
|
||||||
|
|
||||||
|
help: str
|
||||||
|
"""此节点的帮助信息"""
|
||||||
|
|
||||||
|
usage: str = None
|
||||||
|
"""用法"""
|
||||||
|
|
||||||
|
parent_class: typing.Union[typing.Type[CommandOperator], None] = None
|
||||||
|
"""父节点类。标记以供管理器在初始化时编织父子关系。"""
|
||||||
|
|
||||||
|
lowest_privilege: int = 0
|
||||||
|
"""最低权限。若权限低于此值,则不予执行。"""
|
||||||
|
|
||||||
|
children: list[CommandOperator]
|
||||||
|
"""子节点。解析命令时,若节点有子节点,则以下一个参数去匹配子节点,
|
||||||
|
若有匹配中的,转移到子节点中执行,若没有匹配中的或没有子节点,执行此节点。"""
|
||||||
|
|
||||||
|
def __init__(self, ap: app.Application):
|
||||||
|
self.ap = ap
|
||||||
|
self.children = []
|
||||||
|
|
||||||
|
async def initialize(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
"""实现此方法以执行命令
|
||||||
|
|
||||||
|
支持多次yield以返回多个结果。
|
||||||
|
例如:一个安装插件的命令,可能会有下载、解压、安装等多个步骤,每个步骤都可以返回一个结果。
|
||||||
|
|
||||||
|
Args:
|
||||||
|
context (entities.ExecuteContext): 命令执行上下文
|
||||||
|
|
||||||
|
Yields:
|
||||||
|
entities.CommandReturn: 命令返回封装
|
||||||
|
"""
|
||||||
|
pass
|
||||||
0
pkg/command/operators/__init__.py
Normal file
0
pkg/command/operators/__init__.py
Normal file
50
pkg/command/operators/cmd.py
Normal file
50
pkg/command/operators/cmd.py
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="cmd",
|
||||||
|
help='显示命令列表',
|
||||||
|
usage='!cmd\n!cmd <命令名称>'
|
||||||
|
)
|
||||||
|
class CmdOperator(operator.CommandOperator):
|
||||||
|
"""命令列表
|
||||||
|
"""
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
"""执行
|
||||||
|
"""
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
reply_str = "当前所有命令: \n\n"
|
||||||
|
|
||||||
|
for cmd in self.ap.cmd_mgr.cmd_list:
|
||||||
|
if cmd.parent_class is None:
|
||||||
|
reply_str += f"{cmd.name}: {cmd.help}\n"
|
||||||
|
|
||||||
|
reply_str += "\n使用 !cmd <命令名称> 查看命令的详细帮助"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str.strip())
|
||||||
|
|
||||||
|
else:
|
||||||
|
cmd_name = context.crt_params[0]
|
||||||
|
|
||||||
|
cmd = None
|
||||||
|
|
||||||
|
for _cmd in self.ap.cmd_mgr.cmd_list:
|
||||||
|
if (cmd_name == _cmd.name or cmd_name in _cmd.alias) and (_cmd.parent_class is None):
|
||||||
|
cmd = _cmd
|
||||||
|
break
|
||||||
|
|
||||||
|
if cmd is None:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandNotFoundError(cmd_name))
|
||||||
|
else:
|
||||||
|
reply_str = f"{cmd.name}: {cmd.help}\n\n"
|
||||||
|
reply_str += f"使用方法: \n{cmd.usage}"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str.strip())
|
||||||
62
pkg/command/operators/default.py
Normal file
62
pkg/command/operators/default.py
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="default",
|
||||||
|
help="操作情景预设",
|
||||||
|
usage='!default\n!default set <指定情景预设为默认>'
|
||||||
|
)
|
||||||
|
class DefaultOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
reply_str = "当前所有情景预设: \n\n"
|
||||||
|
|
||||||
|
for prompt in self.ap.prompt_mgr.get_all_prompts():
|
||||||
|
|
||||||
|
content = ""
|
||||||
|
for msg in prompt.messages:
|
||||||
|
content += f" {msg.readable_str()}\n"
|
||||||
|
|
||||||
|
reply_str += f"名称: {prompt.name}\n内容: \n{content}\n\n"
|
||||||
|
|
||||||
|
reply_str += f"当前会话使用的是: {context.session.use_prompt_name}"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str.strip())
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="set",
|
||||||
|
help="设置当前会话默认情景预设",
|
||||||
|
parent_class=DefaultOperator
|
||||||
|
)
|
||||||
|
class DefaultSetOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.ParamNotEnoughError('请提供情景预设名称'))
|
||||||
|
else:
|
||||||
|
prompt_name = context.crt_params[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
prompt = await self.ap.prompt_mgr.get_prompt_by_prefix(prompt_name)
|
||||||
|
if prompt is None:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("设置当前会话默认情景预设失败: 未找到情景预设 {}".format(prompt_name)))
|
||||||
|
else:
|
||||||
|
context.session.use_prompt_name = prompt.name
|
||||||
|
yield entities.CommandReturn(text=f"已设置当前会话默认情景预设为 {prompt_name}, !reset 后生效")
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("设置当前会话默认情景预设失败: "+str(e)))
|
||||||
62
pkg/command/operators/delc.py
Normal file
62
pkg/command/operators/delc.py
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="del",
|
||||||
|
help="删除当前会话的历史记录",
|
||||||
|
usage='!del <序号>\n!del all'
|
||||||
|
)
|
||||||
|
class DelOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if context.session.conversations:
|
||||||
|
delete_index = 0
|
||||||
|
if len(context.crt_params) > 0:
|
||||||
|
try:
|
||||||
|
delete_index = int(context.crt_params[0])
|
||||||
|
except:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('索引必须是整数'))
|
||||||
|
return
|
||||||
|
|
||||||
|
if delete_index < 0 or delete_index >= len(context.session.conversations):
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('索引超出范围'))
|
||||||
|
return
|
||||||
|
|
||||||
|
# 倒序
|
||||||
|
to_delete_index = len(context.session.conversations)-1-delete_index
|
||||||
|
|
||||||
|
if context.session.conversations[to_delete_index] == context.session.using_conversation:
|
||||||
|
context.session.using_conversation = None
|
||||||
|
|
||||||
|
del context.session.conversations[to_delete_index]
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=f"已删除对话: {delete_index}")
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('当前没有对话'))
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="all",
|
||||||
|
help="删除此会话的所有历史记录",
|
||||||
|
parent_class=DelOperator
|
||||||
|
)
|
||||||
|
class DelAllOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
context.session.conversations = []
|
||||||
|
context.session.using_conversation = None
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text="已删除所有对话")
|
||||||
29
pkg/command/operators/func.py
Normal file
29
pkg/command/operators/func.py
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
from typing import AsyncGenerator
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr
|
||||||
|
from ...plugin import context as plugin_context
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(name="func", help="查看所有已注册的内容函数", usage='!func')
|
||||||
|
class FuncOperator(operator.CommandOperator):
|
||||||
|
async def execute(
|
||||||
|
self, context: entities.ExecuteContext
|
||||||
|
) -> AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
reply_str = "当前已启用的内容函数: \n\n"
|
||||||
|
|
||||||
|
index = 1
|
||||||
|
|
||||||
|
all_functions = await self.ap.tool_mgr.get_all_functions(
|
||||||
|
plugin_enabled=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
for func in all_functions:
|
||||||
|
reply_str += "{}. {}:\n{}\n\n".format(
|
||||||
|
index,
|
||||||
|
func.name,
|
||||||
|
func.description,
|
||||||
|
)
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str)
|
||||||
23
pkg/command/operators/help.py
Normal file
23
pkg/command/operators/help.py
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name='help',
|
||||||
|
help='显示帮助',
|
||||||
|
usage='!help\n!help <命令名称>'
|
||||||
|
)
|
||||||
|
class HelpOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
help = self.ap.system_cfg.data['help-message']
|
||||||
|
|
||||||
|
help += '\n发送命令 !cmd 可查看命令列表'
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=help)
|
||||||
36
pkg/command/operators/last.py
Normal file
36
pkg/command/operators/last.py
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="last",
|
||||||
|
help="切换到前一个对话",
|
||||||
|
usage='!last'
|
||||||
|
)
|
||||||
|
class LastOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if context.session.conversations:
|
||||||
|
# 找到当前会话的上一个会话
|
||||||
|
for index in range(len(context.session.conversations)-1, -1, -1):
|
||||||
|
if context.session.conversations[index] == context.session.using_conversation:
|
||||||
|
if index == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('已经是第一个对话了'))
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
context.session.using_conversation = context.session.conversations[index-1]
|
||||||
|
time_str = context.session.using_conversation.create_time.strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=f"已切换到上一个对话: {index} {time_str}: {context.session.using_conversation.messages[0].readable_str()}")
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('当前没有对话'))
|
||||||
56
pkg/command/operators/list.py
Normal file
56
pkg/command/operators/list.py
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="list",
|
||||||
|
help="列出此会话中的所有历史对话",
|
||||||
|
usage='!list\n!list <页码>'
|
||||||
|
)
|
||||||
|
class ListOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
page = 0
|
||||||
|
|
||||||
|
if len(context.crt_params) > 0:
|
||||||
|
try:
|
||||||
|
page = int(context.crt_params[0]-1)
|
||||||
|
except:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('页码应为整数'))
|
||||||
|
return
|
||||||
|
|
||||||
|
record_per_page = 10
|
||||||
|
|
||||||
|
content = ''
|
||||||
|
|
||||||
|
index = 0
|
||||||
|
|
||||||
|
using_conv_index = 0
|
||||||
|
|
||||||
|
for conv in context.session.conversations[::-1]:
|
||||||
|
time_str = conv.create_time.strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
if conv == context.session.using_conversation:
|
||||||
|
using_conv_index = index
|
||||||
|
|
||||||
|
if index >= page * record_per_page and index < (page + 1) * record_per_page:
|
||||||
|
content += f"{index} {time_str}: {conv.messages[0].readable_str() if len(conv.messages) > 0 else '无内容'}\n"
|
||||||
|
index += 1
|
||||||
|
|
||||||
|
if content == '':
|
||||||
|
content = '无'
|
||||||
|
else:
|
||||||
|
if context.session.using_conversation is None:
|
||||||
|
content += "\n当前处于新会话"
|
||||||
|
else:
|
||||||
|
content += f"\n当前会话: {using_conv_index} {context.session.using_conversation.create_time.strftime('%Y-%m-%d %H:%M:%S')}: {context.session.using_conversation.messages[0].readable_str() if len(context.session.using_conversation.messages) > 0 else '无内容'}"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=f"第 {page + 1} 页 (时间倒序):\n{content}")
|
||||||
86
pkg/command/operators/model.py
Normal file
86
pkg/command/operators/model.py
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="model",
|
||||||
|
help='显示和切换模型列表',
|
||||||
|
usage='!model\n!model show <模型名>\n!model set <模型名>',
|
||||||
|
privilege=2
|
||||||
|
)
|
||||||
|
class ModelOperator(operator.CommandOperator):
|
||||||
|
"""Model命令"""
|
||||||
|
|
||||||
|
async def execute(self, context: entities.ExecuteContext) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
content = '模型列表:\n'
|
||||||
|
|
||||||
|
model_list = self.ap.model_mgr.model_list
|
||||||
|
|
||||||
|
for model in model_list:
|
||||||
|
content += f"\n名称: {model.name}\n"
|
||||||
|
content += f"请求器: {model.requester.name}\n"
|
||||||
|
|
||||||
|
content += f"\n当前对话使用模型: {context.query.use_model.name}\n"
|
||||||
|
content += f"新对话默认使用模型: {self.ap.provider_cfg.data.get('model')}\n"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=content.strip())
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="show",
|
||||||
|
help='显示模型详情',
|
||||||
|
privilege=2,
|
||||||
|
parent_class=ModelOperator
|
||||||
|
)
|
||||||
|
class ModelShowOperator(operator.CommandOperator):
|
||||||
|
"""Model Show命令"""
|
||||||
|
|
||||||
|
async def execute(self, context: entities.ExecuteContext) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
model_name = context.crt_params[0]
|
||||||
|
|
||||||
|
model = None
|
||||||
|
for _model in self.ap.model_mgr.model_list:
|
||||||
|
if model_name == _model.name:
|
||||||
|
model = _model
|
||||||
|
break
|
||||||
|
|
||||||
|
if model is None:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError(f"未找到模型 {model_name}"))
|
||||||
|
else:
|
||||||
|
content = f"模型详情\n"
|
||||||
|
content += f"名称: {model.name}\n"
|
||||||
|
if model.model_name is not None:
|
||||||
|
content += f"请求模型名称: {model.model_name}\n"
|
||||||
|
content += f"请求器: {model.requester.name}\n"
|
||||||
|
content += f"密钥组: {model.token_mgr.provider}\n"
|
||||||
|
content += f"支持视觉: {model.vision_supported}\n"
|
||||||
|
content += f"支持工具: {model.tool_call_supported}\n"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=content.strip())
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="set",
|
||||||
|
help='设置默认使用模型',
|
||||||
|
privilege=2,
|
||||||
|
parent_class=ModelOperator
|
||||||
|
)
|
||||||
|
class ModelSetOperator(operator.CommandOperator):
|
||||||
|
"""Model Set命令"""
|
||||||
|
|
||||||
|
async def execute(self, context: entities.ExecuteContext) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
model_name = context.crt_params[0]
|
||||||
|
|
||||||
|
model = None
|
||||||
|
for _model in self.ap.model_mgr.model_list:
|
||||||
|
if model_name == _model.name:
|
||||||
|
model = _model
|
||||||
|
break
|
||||||
|
|
||||||
|
if model is None:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError(f"未找到模型 {model_name}"))
|
||||||
|
else:
|
||||||
|
self.ap.provider_cfg.data['model'] = model_name
|
||||||
|
await self.ap.provider_cfg.dump_config()
|
||||||
|
yield entities.CommandReturn(text=f"已设置当前使用模型为 {model_name},重置会话以生效")
|
||||||
35
pkg/command/operators/next.py
Normal file
35
pkg/command/operators/next.py
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="next",
|
||||||
|
help="切换到后一个对话",
|
||||||
|
usage='!next'
|
||||||
|
)
|
||||||
|
class NextOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if context.session.conversations:
|
||||||
|
# 找到当前会话的下一个会话
|
||||||
|
for index in range(len(context.session.conversations)):
|
||||||
|
if context.session.conversations[index] == context.session.using_conversation:
|
||||||
|
if index == len(context.session.conversations)-1:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('已经是最后一个对话了'))
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
context.session.using_conversation = context.session.conversations[index+1]
|
||||||
|
time_str = context.session.using_conversation.create_time.strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=f"已切换到后一个对话: {index} {time_str}: {context.session.using_conversation.messages[0].content}")
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('当前没有对话'))
|
||||||
121
pkg/command/operators/ollama.py
Normal file
121
pkg/command/operators/ollama.py
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import json
|
||||||
|
import typing
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
import ollama
|
||||||
|
from .. import operator, entities, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="ollama",
|
||||||
|
help="ollama平台操作",
|
||||||
|
usage="!ollama\n!ollama show <模型名>\n!ollama pull <模型名>\n!ollama del <模型名>"
|
||||||
|
)
|
||||||
|
class OllamaOperator(operator.CommandOperator):
|
||||||
|
async def execute(
|
||||||
|
self, context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
try:
|
||||||
|
content: str = '模型列表:\n'
|
||||||
|
model_list: list = ollama.list().get('models', [])
|
||||||
|
for model in model_list:
|
||||||
|
content += f"名称: {model['name']}\n"
|
||||||
|
content += f"修改时间: {model['modified_at']}\n"
|
||||||
|
content += f"大小: {bytes_to_mb(model['size'])}MB\n\n"
|
||||||
|
yield entities.CommandReturn(text=f"{content.strip()}")
|
||||||
|
except ollama.ResponseError as e:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError(f"无法获取模型列表,请确认 Ollama 服务正常"))
|
||||||
|
|
||||||
|
|
||||||
|
def bytes_to_mb(num_bytes):
|
||||||
|
mb: float = num_bytes / 1024 / 1024
|
||||||
|
return format(mb, '.2f')
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="show",
|
||||||
|
help="ollama模型详情",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=OllamaOperator
|
||||||
|
)
|
||||||
|
class OllamaShowOperator(operator.CommandOperator):
|
||||||
|
async def execute(
|
||||||
|
self, context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
content: str = '模型详情:\n'
|
||||||
|
try:
|
||||||
|
show: dict = ollama.show(model=context.crt_params[0])
|
||||||
|
model_info: dict = show.get('model_info', {})
|
||||||
|
ignore_show: str = 'too long to show...'
|
||||||
|
|
||||||
|
for key in ['license', 'modelfile']:
|
||||||
|
show[key] = ignore_show
|
||||||
|
|
||||||
|
for key in ['tokenizer.chat_template.rag', 'tokenizer.chat_template.tool_use']:
|
||||||
|
model_info[key] = ignore_show
|
||||||
|
|
||||||
|
content += json.dumps(show, indent=4)
|
||||||
|
yield entities.CommandReturn(text=content.strip())
|
||||||
|
except ollama.ResponseError as e:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError(f"无法获取模型详情,请确认 Ollama 服务正常"))
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="pull",
|
||||||
|
help="ollama模型拉取",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=OllamaOperator
|
||||||
|
)
|
||||||
|
class OllamaPullOperator(operator.CommandOperator):
|
||||||
|
async def execute(
|
||||||
|
self, context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
try:
|
||||||
|
model_list: list = ollama.list().get('models', [])
|
||||||
|
if context.crt_params[0] in [model['name'] for model in model_list]:
|
||||||
|
yield entities.CommandReturn(text="模型已存在")
|
||||||
|
return
|
||||||
|
except ollama.ResponseError as e:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError(f"无法获取模型列表,请确认 Ollama 服务正常"))
|
||||||
|
return
|
||||||
|
|
||||||
|
on_progress: bool = False
|
||||||
|
progress_count: int = 0
|
||||||
|
try:
|
||||||
|
for resp in ollama.pull(model=context.crt_params[0], stream=True):
|
||||||
|
total: typing.Any = resp.get('total')
|
||||||
|
if not on_progress:
|
||||||
|
if total is not None:
|
||||||
|
on_progress = True
|
||||||
|
yield entities.CommandReturn(text=resp.get('status'))
|
||||||
|
else:
|
||||||
|
if total is None:
|
||||||
|
on_progress = False
|
||||||
|
|
||||||
|
completed: typing.Any = resp.get('completed')
|
||||||
|
if isinstance(completed, int) and isinstance(total, int):
|
||||||
|
percentage_completed = (completed / total) * 100
|
||||||
|
if percentage_completed > progress_count:
|
||||||
|
progress_count += 10
|
||||||
|
yield entities.CommandReturn(
|
||||||
|
text=f"下载进度: {completed}/{total} ({percentage_completed:.2f}%)")
|
||||||
|
except ollama.ResponseError as e:
|
||||||
|
yield entities.CommandReturn(text=f"拉取失败: {e.error}")
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="del",
|
||||||
|
help="ollama模型删除",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=OllamaOperator
|
||||||
|
)
|
||||||
|
class OllamaDelOperator(operator.CommandOperator):
|
||||||
|
async def execute(
|
||||||
|
self, context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
try:
|
||||||
|
ret: str = ollama.delete(model=context.crt_params[0])['status']
|
||||||
|
except ollama.ResponseError as e:
|
||||||
|
ret = f"{e.error}"
|
||||||
|
yield entities.CommandReturn(text=ret)
|
||||||
219
pkg/command/operators/plugin.py
Normal file
219
pkg/command/operators/plugin.py
Normal file
@@ -0,0 +1,219 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
import typing
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
from ...core import app
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="plugin",
|
||||||
|
help="插件操作",
|
||||||
|
usage="!plugin\n!plugin get <插件仓库地址>\n!plugin update\n!plugin del <插件名>\n!plugin on <插件名>\n!plugin off <插件名>"
|
||||||
|
)
|
||||||
|
class PluginOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
plugin_list = self.ap.plugin_mgr.plugins()
|
||||||
|
reply_str = "所有插件({}):\n".format(len(plugin_list))
|
||||||
|
idx = 0
|
||||||
|
for plugin in plugin_list:
|
||||||
|
reply_str += "\n#{} {} {}\n{}\nv{}\n作者: {}\n"\
|
||||||
|
.format((idx+1), plugin.plugin_name,
|
||||||
|
"[已禁用]" if not plugin.enabled else "",
|
||||||
|
plugin.plugin_description,
|
||||||
|
plugin.plugin_version, plugin.plugin_author)
|
||||||
|
|
||||||
|
# TODO 从元数据调远程地址
|
||||||
|
|
||||||
|
idx += 1
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str)
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="get",
|
||||||
|
help="安装插件",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=PluginOperator
|
||||||
|
)
|
||||||
|
class PluginGetOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.ParamNotEnoughError('请提供插件仓库地址'))
|
||||||
|
else:
|
||||||
|
repo = context.crt_params[0]
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text="正在安装插件...")
|
||||||
|
|
||||||
|
try:
|
||||||
|
await self.ap.plugin_mgr.install_plugin(repo)
|
||||||
|
yield entities.CommandReturn(text="插件安装成功,请重启程序以加载插件")
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件安装失败: "+str(e)))
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="update",
|
||||||
|
help="更新插件",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=PluginOperator
|
||||||
|
)
|
||||||
|
class PluginUpdateOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.ParamNotEnoughError('请提供插件名称'))
|
||||||
|
else:
|
||||||
|
plugin_name = context.crt_params[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
plugin_container = self.ap.plugin_mgr.get_plugin_by_name(plugin_name)
|
||||||
|
|
||||||
|
if plugin_container is not None:
|
||||||
|
yield entities.CommandReturn(text="正在更新插件...")
|
||||||
|
await self.ap.plugin_mgr.update_plugin(plugin_name)
|
||||||
|
yield entities.CommandReturn(text="插件更新成功,请重启程序以加载插件")
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件更新失败: 未找到插件"))
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件更新失败: "+str(e)))
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="all",
|
||||||
|
help="更新所有插件",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=PluginUpdateOperator
|
||||||
|
)
|
||||||
|
class PluginUpdateAllOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
try:
|
||||||
|
plugins = [
|
||||||
|
p.plugin_name
|
||||||
|
for p in self.ap.plugin_mgr.plugins()
|
||||||
|
]
|
||||||
|
|
||||||
|
if plugins:
|
||||||
|
yield entities.CommandReturn(text="正在更新插件...")
|
||||||
|
updated = []
|
||||||
|
try:
|
||||||
|
for plugin_name in plugins:
|
||||||
|
await self.ap.plugin_mgr.update_plugin(plugin_name)
|
||||||
|
updated.append(plugin_name)
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件更新失败: "+str(e)))
|
||||||
|
yield entities.CommandReturn(text="已更新插件: {}".format(", ".join(updated)))
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(text="没有可更新的插件")
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件更新失败: "+str(e)))
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="del",
|
||||||
|
help="删除插件",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=PluginOperator
|
||||||
|
)
|
||||||
|
class PluginDelOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.ParamNotEnoughError('请提供插件名称'))
|
||||||
|
else:
|
||||||
|
plugin_name = context.crt_params[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
plugin_container = self.ap.plugin_mgr.get_plugin_by_name(plugin_name)
|
||||||
|
|
||||||
|
if plugin_container is not None:
|
||||||
|
yield entities.CommandReturn(text="正在删除插件...")
|
||||||
|
await self.ap.plugin_mgr.uninstall_plugin(plugin_name)
|
||||||
|
yield entities.CommandReturn(text="插件删除成功,请重启程序以加载插件")
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件删除失败: 未找到插件"))
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件删除失败: "+str(e)))
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="on",
|
||||||
|
help="启用插件",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=PluginOperator
|
||||||
|
)
|
||||||
|
class PluginEnableOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.ParamNotEnoughError('请提供插件名称'))
|
||||||
|
else:
|
||||||
|
plugin_name = context.crt_params[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
if await self.ap.plugin_mgr.update_plugin_switch(plugin_name, True):
|
||||||
|
yield entities.CommandReturn(text="已启用插件: {}".format(plugin_name))
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件状态修改失败: 未找到插件 {}".format(plugin_name)))
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件状态修改失败: "+str(e)))
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="off",
|
||||||
|
help="禁用插件",
|
||||||
|
privilege=2,
|
||||||
|
parent_class=PluginOperator
|
||||||
|
)
|
||||||
|
class PluginDisableOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
if len(context.crt_params) == 0:
|
||||||
|
yield entities.CommandReturn(error=errors.ParamNotEnoughError('请提供插件名称'))
|
||||||
|
else:
|
||||||
|
plugin_name = context.crt_params[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
if await self.ap.plugin_mgr.update_plugin_switch(plugin_name, False):
|
||||||
|
yield entities.CommandReturn(text="已禁用插件: {}".format(plugin_name))
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件状态修改失败: 未找到插件 {}".format(plugin_name)))
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("插件状态修改失败: "+str(e)))
|
||||||
29
pkg/command/operators/prompt.py
Normal file
29
pkg/command/operators/prompt.py
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="prompt",
|
||||||
|
help="查看当前对话的前文",
|
||||||
|
usage='!prompt'
|
||||||
|
)
|
||||||
|
class PromptOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
"""执行
|
||||||
|
"""
|
||||||
|
if context.session.using_conversation is None:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandOperationError('当前没有对话'))
|
||||||
|
else:
|
||||||
|
reply_str = '当前对话所有内容:\n\n'
|
||||||
|
|
||||||
|
for msg in context.session.using_conversation.messages:
|
||||||
|
reply_str += f"{msg.role}: {msg.content}\n"
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str)
|
||||||
34
pkg/command/operators/resend.py
Normal file
34
pkg/command/operators/resend.py
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="resend",
|
||||||
|
help="重发当前会话的最后一条消息",
|
||||||
|
usage='!resend'
|
||||||
|
)
|
||||||
|
class ResendOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
# 回滚到最后一条用户message前
|
||||||
|
if context.session.using_conversation is None:
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("当前没有对话"))
|
||||||
|
else:
|
||||||
|
conv_msg = context.session.using_conversation.messages
|
||||||
|
|
||||||
|
# 倒序一直删到最后一条用户message
|
||||||
|
while len(conv_msg) > 0 and conv_msg[-1].role != 'user':
|
||||||
|
conv_msg.pop()
|
||||||
|
|
||||||
|
if len(conv_msg) > 0:
|
||||||
|
# 删除最后一条用户message
|
||||||
|
conv_msg.pop()
|
||||||
|
|
||||||
|
# 不重发了,提示用户已删除就行了
|
||||||
|
yield entities.CommandReturn(text="已删除最后一次请求记录")
|
||||||
23
pkg/command/operators/reset.py
Normal file
23
pkg/command/operators/reset.py
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="reset",
|
||||||
|
help="重置当前会话",
|
||||||
|
usage='!reset'
|
||||||
|
)
|
||||||
|
class ResetOperator(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
"""执行
|
||||||
|
"""
|
||||||
|
context.session.using_conversation = None
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text="已重置当前会话")
|
||||||
30
pkg/command/operators/update.py
Normal file
30
pkg/command/operators/update.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
from .. import operator, entities, cmdmgr, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="update",
|
||||||
|
help="更新程序",
|
||||||
|
usage='!update',
|
||||||
|
privilege=2
|
||||||
|
)
|
||||||
|
class UpdateCommand(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
|
||||||
|
try:
|
||||||
|
yield entities.CommandReturn(text="正在进行更新...")
|
||||||
|
if await self.ap.ver_mgr.update_all():
|
||||||
|
yield entities.CommandReturn(text="更新完成,请重启程序以应用更新")
|
||||||
|
else:
|
||||||
|
yield entities.CommandReturn(text="当前已是最新版本")
|
||||||
|
except Exception as e:
|
||||||
|
traceback.print_exc()
|
||||||
|
yield entities.CommandReturn(error=errors.CommandError("更新失败: "+str(e)))
|
||||||
27
pkg/command/operators/version.py
Normal file
27
pkg/command/operators/version.py
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import typing
|
||||||
|
|
||||||
|
from .. import operator, cmdmgr, entities, errors
|
||||||
|
|
||||||
|
|
||||||
|
@operator.operator_class(
|
||||||
|
name="version",
|
||||||
|
help="显示版本信息",
|
||||||
|
usage='!version'
|
||||||
|
)
|
||||||
|
class VersionCommand(operator.CommandOperator):
|
||||||
|
|
||||||
|
async def execute(
|
||||||
|
self,
|
||||||
|
context: entities.ExecuteContext
|
||||||
|
) -> typing.AsyncGenerator[entities.CommandReturn, None]:
|
||||||
|
reply_str = f"当前版本: \n{self.ap.ver_mgr.get_current_version()}"
|
||||||
|
|
||||||
|
try:
|
||||||
|
if await self.ap.ver_mgr.is_new_version_available():
|
||||||
|
reply_str += "\n\n有新版本可用。"
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
yield entities.CommandReturn(text=reply_str.strip())
|
||||||
0
pkg/config/__init__.py
Normal file
0
pkg/config/__init__.py
Normal file
0
pkg/config/impls/__init__.py
Normal file
0
pkg/config/impls/__init__.py
Normal file
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user