mirror of
https://github.com/Mrs4s/go-cqhttp.git
synced 2025-06-30 03:43:25 +00:00
Compare commits
490 Commits
v0.9.15
...
v0.9.33-fi
Author | SHA1 | Date | |
---|---|---|---|
d63d10b69e | |||
dfeadef1a2 | |||
b23f68e746 | |||
c5a5f71664 | |||
6973a10b4b | |||
3468c38fbc | |||
d76bcfb1f0 | |||
716c38bab4 | |||
03518b4823 | |||
09eeff3ce6 | |||
6599bf6aa6 | |||
ffb2faeae7 | |||
71e3e97aeb | |||
dcb708e3e3 | |||
b9fb833af4 | |||
b93df0250d | |||
33d64b6e27 | |||
7f831cf856 | |||
7a713c9327 | |||
28d536a3af | |||
375fdc8dca | |||
d1c6053df8 | |||
143d5ef8ab | |||
439c3422fb | |||
2c7b56a79c | |||
1549ef4a32 | |||
4ad684061b | |||
c7e6457e9e | |||
9d5842f09b | |||
b6f3c300ab | |||
2057e769da | |||
0ea85e2fab | |||
36b081eeec | |||
2a044a62d2 | |||
e0ffb98665 | |||
861bcf30b1 | |||
56d7815f6a | |||
59934982b9 | |||
09c4801b56 | |||
f1f58b4072 | |||
86c2f3cb11 | |||
03c3340954 | |||
64b1a223cc | |||
b2be595fb8 | |||
22225f2b74 | |||
eb3d7434cc | |||
8f7f207891 | |||
83e0aaa0f8 | |||
8f76558317 | |||
d94d389843 | |||
df0d5af07a | |||
277629e784 | |||
d66a99c5a3 | |||
55895e0de1 | |||
39da163641 | |||
1dbd72c138 | |||
31756e9e1d | |||
5ab713e7d3 | |||
b6eb559172 | |||
da2e34aaa5 | |||
beb1a2883a | |||
8f34003c6f | |||
7f3e58f97a | |||
0fc004077b | |||
e9a03b68b5 | |||
e9b39f3379 | |||
e053da62e1 | |||
f87b5c8215 | |||
28f1594539 | |||
990e7744a3 | |||
8900839018 | |||
4a3288cb8d | |||
0f0f74ece8 | |||
79ba9474a3 | |||
cb9d872071 | |||
4c3105fa24 | |||
a8176e69e6 | |||
afa356e551 | |||
a52ae1c828 | |||
b8ec05e1c1 | |||
2cbf23bceb | |||
3e5c2bc8e3 | |||
69a2360d02 | |||
5b0ede5346 | |||
80af002668 | |||
b3152bb514 | |||
9a018e2a47 | |||
998b8cb146 | |||
7e3f94ad2e | |||
f9dc22f2e5 | |||
0ce54ac21b | |||
06ac2fcb0c | |||
30f1f3199a | |||
535b4ee641 | |||
0ed6522535 | |||
3326660880 | |||
171aba527e | |||
d2bdf47bf8 | |||
4c2b56457e | |||
8b5d63e02c | |||
7dd0001dc8 | |||
7f9e4e6a20 | |||
f59ce1480e | |||
c2c7b96f1b | |||
c49c68891b | |||
5b394b7a78 | |||
0cc3d90581 | |||
d906bbf0ff | |||
e2d2461595 | |||
6bb1f1603e | |||
5e02883028 | |||
1f5c9acefb | |||
36b235871f | |||
f675a70af3 | |||
26afca1555 | |||
76b793f119 | |||
491bd2276e | |||
3d81777ed1 | |||
7b1f0d72eb | |||
a0219d76ea | |||
08b55473aa | |||
66d76aa1f1 | |||
151e44628c | |||
d32f427328 | |||
e911123a30 | |||
cc72332455 | |||
84b4889def | |||
6ad0d68978 | |||
1b63a15bbe | |||
ce0a5b0271 | |||
f96abc5e26 | |||
2cdb341db4 | |||
c2fd0f1bb2 | |||
8ab874ca2b | |||
8e61060ef0 | |||
508117d30b | |||
1c965ab9ac | |||
9db1dcc76a | |||
fffed72d44 | |||
dd5fdb0735 | |||
c35f46e033 | |||
98b9be575e | |||
b99986d112 | |||
2302cf6263 | |||
6a20a86e49 | |||
9b36645b73 | |||
d1372332f3 | |||
2978116c89 | |||
8c82082991 | |||
08394ae87d | |||
7df17f7ff2 | |||
9bd41c7792 | |||
18f8a12b8c | |||
755a794150 | |||
0ab52da0f0 | |||
2d768c3c1f | |||
1feb44d6ba | |||
06e40a940e | |||
95376a8a63 | |||
c619ee7feb | |||
a53e549e6c | |||
a6c13b68cd | |||
7659a214d6 | |||
b7d3aec9b7 | |||
584159d285 | |||
6d2f464bee | |||
e719c86731 | |||
c984f22d4f | |||
aaa73ef5f5 | |||
f9c0b94af6 | |||
4c9f60da09 | |||
a72a688a62 | |||
1d7f1cc5d5 | |||
24937f2386 | |||
73bd756c20 | |||
9d58c56b35 | |||
6879197107 | |||
a979c46563 | |||
e8513da090 | |||
2a5f78499b | |||
b167231787 | |||
7ae9c2d220 | |||
31f24525f0 | |||
51101f02cd | |||
4d404eacd9 | |||
ff29e16220 | |||
d4811d53e7 | |||
b19b114d3a | |||
95c399a003 | |||
e3f0dbc4ac | |||
b0d5589dcd | |||
b4d29e270a | |||
a5958e0877 | |||
cc1093ff57 | |||
a712db5d50 | |||
030eb6b7c4 | |||
6b706ca3ff | |||
bbf2025350 | |||
e386a52fed | |||
4c8ae2f08b | |||
d70e5d91b3 | |||
c2fae1e6f5 | |||
1ce73cb36a | |||
dfbea6ac28 | |||
29d001d60c | |||
93f3786059 | |||
e59d0a1bfd | |||
5c1ad4ad8e | |||
9cd044c402 | |||
add9d3fc20 | |||
0d1dfeefa1 | |||
d438543746 | |||
79b0c95f09 | |||
f5f3a314eb | |||
de234c7721 | |||
a6a1de0a00 | |||
062a898383 | |||
83ce4e58a5 | |||
eef140e922 | |||
6581394ae1 | |||
2e59cb8a17 | |||
b73162238f | |||
1a3cba5425 | |||
9ecef96205 | |||
43ecc25989 | |||
9b41cd8ea2 | |||
e0ab2417c4 | |||
c7ab32fead | |||
8f5b8375a7 | |||
9049018d2f | |||
fffe64651a | |||
a9d53f038f | |||
cec81b7ad8 | |||
dddb3d2299 | |||
f6d37d460c | |||
aca6a3cf28 | |||
d390dd539b | |||
3269387163 | |||
cd09a68e0d | |||
d2d408a8e9 | |||
995737b0b8 | |||
72d62ad7c3 | |||
387d33a01b | |||
65811d4e22 | |||
01dd760b5c | |||
c31169b99b | |||
b5b6117487 | |||
4662f1e170 | |||
ea65e545ad | |||
d67621487d | |||
c4e54446c5 | |||
1d1925208b | |||
741a91cf70 | |||
fef9395890 | |||
50bfc08974 | |||
2005cbccb0 | |||
69cadad155 | |||
a4c6c6d7f3 | |||
6f251e3dec | |||
6101c4d8a9 | |||
9b77f09d2f | |||
e9db10c9b7 | |||
e87de2f6b3 | |||
a1a3e26b0b | |||
92c224be5b | |||
c691aa70ea | |||
ec45d04c81 | |||
30c4d508bc | |||
b0e00fe052 | |||
09aa852308 | |||
0cfaa4c7ad | |||
faba3e042f | |||
b33bb37da9 | |||
50ebfe4766 | |||
e4a458e7c8 | |||
8c63a420a3 | |||
b1fccbc0a3 | |||
1a2fa65347 | |||
5ddb246ab2 | |||
86da65771d | |||
eca396afb3 | |||
e5718fc47c | |||
4b12bbc1e4 | |||
a26681a4d9 | |||
a417ff0881 | |||
845b483242 | |||
da8b0931f1 | |||
3dc436b534 | |||
6a0f44f6cc | |||
5501c6192f | |||
ca62f6f591 | |||
ac5803a539 | |||
828ca2ec3c | |||
514f93b527 | |||
4d260be76a | |||
793cf2e2b9 | |||
2a56475daf | |||
f9e2130b9f | |||
e2073f5371 | |||
b1fcfc6f97 | |||
6a14998928 | |||
24dad3d6f9 | |||
b104b21ab9 | |||
5024cc08a7 | |||
3c73769db3 | |||
d79cb0b536 | |||
2b00b49755 | |||
a77977b12e | |||
6565a19aaa | |||
a9001d5c9e | |||
a991add799 | |||
42434956d9 | |||
49e3995129 | |||
8c1770318c | |||
7d3265b6d4 | |||
4c65859608 | |||
0100b85fb1 | |||
3bd557d7a0 | |||
05d8e773f2 | |||
b2f46b4bbe | |||
9fa1b7727e | |||
9e43d9c2bb | |||
ad5e87c4c7 | |||
ff8fd92a26 | |||
0f96950757 | |||
993f05bfa6 | |||
d309a10fba | |||
755949fb33 | |||
7f103a7f9a | |||
f487d12b83 | |||
df1cb3de87 | |||
7d62db2ec0 | |||
9b5358351f | |||
a048353067 | |||
a455aeea39 | |||
710a8588b7 | |||
d25674a9ef | |||
0133944c0b | |||
bcdbefb355 | |||
fc18b84d4b | |||
e55b7d846f | |||
1f7c4cab50 | |||
d35ad66adf | |||
047b5208f4 | |||
804bd9a711 | |||
f569225305 | |||
9c65ff4dcd | |||
d0398f30c6 | |||
517e2ea3d4 | |||
33120b2496 | |||
38b6fab496 | |||
304b38f6af | |||
e8fd57d477 | |||
4d376c0518 | |||
027aeca622 | |||
cefbfb64ce | |||
04bcd2e270 | |||
0e7c30e854 | |||
0894f2ac44 | |||
50191fcedf | |||
41614b9d16 | |||
11592c4483 | |||
a9d08e5d92 | |||
1c4cb96ce0 | |||
46747354c8 | |||
01988cd3d7 | |||
0e95977ff3 | |||
be17920559 | |||
fe7a1f5028 | |||
174ebfae9d | |||
02355db0f1 | |||
bda02d895a | |||
5fe15163f8 | |||
174bb0bbe1 | |||
e319f2645e | |||
52c911056e | |||
ac97c04cef | |||
801fa9a204 | |||
17f8232a1c | |||
cb9436601f | |||
03cc0dba95 | |||
248ba84a0c | |||
8874ed0392 | |||
9acf598209 | |||
15085d0a6b | |||
4e2fb91d8f | |||
37ea92b928 | |||
fc1680aa2e | |||
6b256c8b82 | |||
65acc888fd | |||
84a826815d | |||
8ae2c1c9be | |||
8fa804a2c9 | |||
d6961fc69b | |||
b4229c5b0e | |||
8a8ce4049d | |||
7d1ed2577e | |||
b3c706cb70 | |||
382daf48a5 | |||
2b92035fdc | |||
5868b68437 | |||
2949f68532 | |||
d7abe6f7e3 | |||
2259e7ccba | |||
5e6b3e09f7 | |||
9f060ce2ed | |||
3709f5c7b9 | |||
fa0a2d80f8 | |||
894a5d9641 | |||
30ff055126 | |||
f8d74a6ac9 | |||
d625c79f7c | |||
c500bfc55c | |||
c3da6f29e2 | |||
196cffb099 | |||
cb9599271a | |||
5ec4e3331d | |||
739d486b09 | |||
322b70c6b5 | |||
470efa07ab | |||
b7572f8d2c | |||
c9a914b5d5 | |||
51696e8054 | |||
a6bcd96415 | |||
08694f5ae8 | |||
8c71dbff68 | |||
568b0e479f | |||
d1da08a376 | |||
5768c61bc7 | |||
864120d3fe | |||
99b414530a | |||
4820eb2fec | |||
00d80d5dfc | |||
22d9ddb4ea | |||
fa33dbdd4e | |||
be8b68c8e9 | |||
cef129eff7 | |||
1fac06f58a | |||
1a1f860dbe | |||
f325b26e1a | |||
b545e05a9d | |||
b290a0a596 | |||
9af61a336b | |||
774a1e32da | |||
50eee15a67 | |||
edf6180e1c | |||
3c04573e82 | |||
5e49820319 | |||
1418e36bab | |||
611d16d79e | |||
c6f701e8c8 | |||
a49a57b964 | |||
bb03315930 | |||
5683db6324 | |||
f1ffa17d20 | |||
69675aa0f9 | |||
782a3e2a39 | |||
1967e687af | |||
23d436972c | |||
129622dd24 | |||
19104d00a3 | |||
6c0e78b60c | |||
a075f41a51 | |||
cdbf903361 | |||
906247fcf0 | |||
72a7430841 | |||
5c4f586c36 | |||
5d2e202b0c | |||
c27ebadbc4 | |||
d5a8f3ead2 | |||
03bfd9dd3d | |||
0b02178402 | |||
3819f2c2da | |||
b297aa2e13 | |||
c8160550ec | |||
44b26b70a3 | |||
97f7de85a2 | |||
ae58d26047 | |||
47cb3a5235 | |||
4941f0c3f8 | |||
bc47267c3c | |||
8449770b6d | |||
a38c74a767 | |||
4ab1812640 | |||
c87836355f | |||
8524fbf1ce | |||
92fb69beef | |||
dd03efc9c8 | |||
fdd517ee8c | |||
2d010326c7 |
5
.dockerignore
Normal file
5
.dockerignore
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
.gitlab-ci.yml
|
||||||
|
.dockerignore
|
||||||
|
Dockerfile
|
||||||
|
README.md
|
||||||
|
LICENSE
|
20
.github/ISSUE_TEMPLATE/bug--.md
vendored
Normal file
20
.github/ISSUE_TEMPLATE/bug--.md
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
---
|
||||||
|
name: Bug汇报
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title: ''
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**环境信息**
|
||||||
|
请根据实际使用环境修改以下信息
|
||||||
|
go-cqhttp版本: v0.9.10
|
||||||
|
运行环境: windows_amd64
|
||||||
|
连接方式: 反向WS
|
||||||
|
|
||||||
|
**bug内容**
|
||||||
|
请在这里详细描述bug的内容
|
||||||
|
|
||||||
|
**复现方法**
|
||||||
|
请在这里分步骤的描述如何复现这个bug
|
8
.github/workflows/ci.yml
vendored
8
.github/workflows/ci.yml
vendored
@ -14,21 +14,23 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
# build and publish in parallel: linux/386, linux/amd64, windows/386, windows/amd64, darwin/386, darwin/amd64
|
# build and publish in parallel: linux/386, linux/amd64, windows/386, windows/amd64, darwin/amd64
|
||||||
goos: [linux, windows, darwin]
|
goos: [linux, windows, darwin]
|
||||||
goarch: ["386", amd64, arm]
|
goarch: ["386", amd64, arm]
|
||||||
exclude:
|
exclude:
|
||||||
- goos: darwin
|
- goos: darwin
|
||||||
goarch: arm
|
goarch: arm
|
||||||
|
- goos: darwin
|
||||||
|
goarch: "386"
|
||||||
fail-fast: true
|
fail-fast: true
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
|
|
||||||
- name: Setup Go environment
|
- name: Setup Go environment
|
||||||
uses: actions/setup-go@v2.1.1
|
uses: actions/setup-go@v2.1.3
|
||||||
with:
|
with:
|
||||||
go-version: 1.14
|
go-version: 1.15
|
||||||
|
|
||||||
- name: Build binary file
|
- name: Build binary file
|
||||||
env:
|
env:
|
||||||
|
10
.github/workflows/release.yml
vendored
10
.github/workflows/release.yml
vendored
@ -14,14 +14,20 @@ jobs:
|
|||||||
exclude:
|
exclude:
|
||||||
- goos: darwin
|
- goos: darwin
|
||||||
goarch: arm
|
goarch: arm
|
||||||
|
- goos: darwin
|
||||||
|
goarch: "386"
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
- uses: wangyoucao577/go-release-action@master
|
- name: Set RELEASE_VERSION env
|
||||||
|
run: echo "RELEASE_VERSION=${GITHUB_REF:10}" >> $GITHUB_ENV
|
||||||
|
- uses: pcrbot/go-release-action@master
|
||||||
env:
|
env:
|
||||||
CGO_ENABLED: 0
|
CGO_ENABLED: 0
|
||||||
with:
|
with:
|
||||||
github_token: ${{ secrets.GITHUB_TOKEN }}
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
goos: ${{ matrix.goos }}
|
goos: ${{ matrix.goos }}
|
||||||
goarch: ${{ matrix.goarch }}
|
goarch: ${{ matrix.goarch }}
|
||||||
ldflags: "-w -s"
|
goversion: "https://golang.org/dl/go1.15.3.linux-amd64.tar.gz"
|
||||||
|
ldflags: -w -s -X "github.com/Mrs4s/go-cqhttp/coolq.Version=${{ env.RELEASE_VERSION }}"
|
||||||
|
|
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
vendor/
|
||||||
|
.idea
|
22
Dockerfile
Normal file
22
Dockerfile
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
FROM golang:1.15.5-alpine AS builder
|
||||||
|
|
||||||
|
RUN go env -w GO111MODULE=auto \
|
||||||
|
&& go env -w CGO_ENABLED=0 \
|
||||||
|
&& go env -w GOPROXY=https://goproxy.cn,direct
|
||||||
|
|
||||||
|
WORKDIR /build
|
||||||
|
|
||||||
|
COPY ./ .
|
||||||
|
|
||||||
|
RUN set -ex \
|
||||||
|
&& cd /build \
|
||||||
|
&& go build -ldflags "-s -w -extldflags '-static'" -o cqhttp
|
||||||
|
|
||||||
|
FROM alpine:latest
|
||||||
|
|
||||||
|
COPY --from=builder /build/cqhttp /usr/bin/cqhttp
|
||||||
|
RUN chmod +x /usr/bin/cqhttp
|
||||||
|
|
||||||
|
WORKDIR /data
|
||||||
|
|
||||||
|
ENTRYPOINT [ "/usr/bin/cqhttp" ]
|
99
README.md
99
README.md
@ -20,7 +20,7 @@
|
|||||||
- [x] 消息撤回事件
|
- [x] 消息撤回事件
|
||||||
- [x] 解析/发送 回复消息
|
- [x] 解析/发送 回复消息
|
||||||
- [x] 解析/发送 合并转发
|
- [x] 解析/发送 合并转发
|
||||||
- [ ] 使用代理请求网络图片
|
- [x] 使用代理请求网络图片
|
||||||
|
|
||||||
#### 实现
|
#### 实现
|
||||||
<details>
|
<details>
|
||||||
@ -28,12 +28,17 @@
|
|||||||
|
|
||||||
- [CQ:image]
|
- [CQ:image]
|
||||||
- [CQ:record]
|
- [CQ:record]
|
||||||
|
- [CQ:video]
|
||||||
- [CQ:face]
|
- [CQ:face]
|
||||||
- [CQ:at]
|
- [CQ:at]
|
||||||
- [CQ:share]
|
- [CQ:share]
|
||||||
- [CQ:reply]
|
- [CQ:reply]
|
||||||
- [CQ:forward]
|
- [CQ:forward]
|
||||||
- [CQ:node]
|
- [CQ:node]
|
||||||
|
- [CQ:gift]
|
||||||
|
- [CQ:redbag]
|
||||||
|
- [CQ:tts]
|
||||||
|
- [CQ:music]
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
@ -41,33 +46,34 @@
|
|||||||
<summary>已实现API</summary>
|
<summary>已实现API</summary>
|
||||||
|
|
||||||
##### 注意: 部分API实现与CQHTTP原版略有差异,请参考文档
|
##### 注意: 部分API实现与CQHTTP原版略有差异,请参考文档
|
||||||
| API | 功能 |
|
| API | 功能 |
|
||||||
| ------------------------ | ------------------------------------------------------------ |
|
| ------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
|
||||||
| /get_login_info | [获取登录号信息](https://cqhttp.cc/docs/4.15/#/API?id=get_login_info-获取登录号信息) |
|
| /get_login_info | [获取登录号信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_login_info-%E8%8E%B7%E5%8F%96%E7%99%BB%E5%BD%95%E5%8F%B7%E4%BF%A1%E6%81%AF) |
|
||||||
| /get_friend_list | [获取好友列表](https://cqhttp.cc/docs/4.15/#/API?id=get_friend_list-获取好友列表) |
|
| /get_friend_list | [获取好友列表](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_friend_list-%E8%8E%B7%E5%8F%96%E5%A5%BD%E5%8F%8B%E5%88%97%E8%A1%A8) |
|
||||||
| /get_group_list | [获取群列表](https://cqhttp.cc/docs/4.15/#/API?id=get_group_list-获取群列表) |
|
| /get_group_list | [获取群列表](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_group_list-%E8%8E%B7%E5%8F%96%E7%BE%A4%E5%88%97%E8%A1%A8) |
|
||||||
| /get_group_info | [获取群信息](https://cqhttp.cc/docs/4.15/#/API?id=get_group_info-获取群信息) |
|
| /get_group_info | [获取群信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_group_info-%E8%8E%B7%E5%8F%96%E7%BE%A4%E4%BF%A1%E6%81%AF) |
|
||||||
| /get_group_member_list | [获取群成员列表](https://cqhttp.cc/docs/4.15/#/API?id=get_group_member_list-获取群成员列表) |
|
| /get_group_member_list | [获取群成员列表](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_group_member_list-%E8%8E%B7%E5%8F%96%E7%BE%A4%E6%88%90%E5%91%98%E5%88%97%E8%A1%A8) |
|
||||||
| /get_group_member_info | [获取群成员信息](https://cqhttp.cc/docs/4.15/#/API?id=get_group_member_info-获取群成员信息) |
|
| /get_group_member_info | [获取群成员信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_group_member_info-%E8%8E%B7%E5%8F%96%E7%BE%A4%E6%88%90%E5%91%98%E4%BF%A1%E6%81%AF) |
|
||||||
| /send_msg | [发送消息](https://cqhttp.cc/docs/4.15/#/API?id=send_msg-发送消息) |
|
| /send_msg | [发送消息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#send_msg-%E5%8F%91%E9%80%81%E6%B6%88%E6%81%AF) |
|
||||||
| /send_group_msg | [发送群消息](https://cqhttp.cc/docs/4.15/#/API?id=send_group_msg-发送群消息) |
|
| /send_group_msg | [发送群消息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#send_group_msg-%E5%8F%91%E9%80%81%E7%BE%A4%E6%B6%88%E6%81%AF) |
|
||||||
| /send_private_msg | [发送私聊消息](https://cqhttp.cc/docs/4.15/#/API?id=send_private_msg-发送私聊消息) |
|
| /send_private_msg | [发送私聊消息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#send_private_msg-%E5%8F%91%E9%80%81%E7%A7%81%E8%81%8A%E6%B6%88%E6%81%AF) |
|
||||||
| /delete_msg | [撤回信息](https://cqhttp.cc/docs/4.15/#/API?id=delete_msg-撤回消息) |
|
| /delete_msg | [撤回信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#delete_msg-%E6%92%A4%E5%9B%9E%E6%B6%88%E6%81%AF) |
|
||||||
| /set_friend_add_request | [处理加好友请求](https://cqhttp.cc/docs/4.15/#/API?id=set_friend_add_request-处理加好友请求) |
|
| /set_friend_add_request | [处理加好友请求](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_friend_add_request-%E5%A4%84%E7%90%86%E5%8A%A0%E5%A5%BD%E5%8F%8B%E8%AF%B7%E6%B1%82) |
|
||||||
| /set_group_add_request | [处理加群请求/邀请](https://cqhttp.cc/docs/4.15/#/API?id=set_group_add_request-处理加群请求/邀请) |
|
| /set_group_add_request | [处理加群请求/邀请](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_add_request-%E5%A4%84%E7%90%86%E5%8A%A0%E7%BE%A4%E8%AF%B7%E6%B1%82%E9%82%80%E8%AF%B7) |
|
||||||
| /set_group_card | [设置群名片(群备注)](https://cqhttp.cc/docs/4.15/#/API?id=set_group_card-设置群名片(群备注)) |
|
| /set_group_card | [设置群名片(群备注)](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_card-%E8%AE%BE%E7%BD%AE%E7%BE%A4%E5%90%8D%E7%89%87%E7%BE%A4%E5%A4%87%E6%B3%A8) |
|
||||||
| /set_group_special_title | [设置群组专属头衔](https://cqhttp.cc/docs/4.15/#/API?id=set_group_special_title-设置群组专属头衔) |
|
| /set_group_special_title | [设置群组专属头衔](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_special_title-%E8%AE%BE%E7%BD%AE%E7%BE%A4%E7%BB%84%E4%B8%93%E5%B1%9E%E5%A4%B4%E8%A1%94) |
|
||||||
| /set_group_kick | [群组T人](https://cqhttp.cc/docs/4.15/#/API?id=set_group_kick-群组踢人) |
|
| /set_group_kick | [群组T人](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_kick-%E7%BE%A4%E7%BB%84%E8%B8%A2%E4%BA%BA) |
|
||||||
| /set_group_ban | [群组单人禁言](https://cqhttp.cc/docs/4.15/#/API?id=set_group_ban-群组单人禁言) |
|
| /set_group_ban | [群组单人禁言](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_ban-%E7%BE%A4%E7%BB%84%E5%8D%95%E4%BA%BA%E7%A6%81%E8%A8%80) |
|
||||||
| /set_group_whole_ban | [群组全员禁言](https://cqhttp.cc/docs/4.15/#/API?id=set_group_whole_ban-群组全员禁言) |
|
| /set_group_whole_ban | [群组全员禁言](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_whole_ban-%E7%BE%A4%E7%BB%84%E5%85%A8%E5%91%98%E7%A6%81%E8%A8%80) |
|
||||||
| /set_group_leave | [退出群组](https://cqhttp.cc/docs/4.15/#/API?id=set_group_leave-退出群组) |
|
| /set_group_leave | [退出群组](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_leave-%E9%80%80%E5%87%BA%E7%BE%A4%E7%BB%84) |
|
||||||
| /set_group_name | 设置群组名(拓展API) |
|
| /set_group_name | [设置群组名](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_name-%E8%AE%BE%E7%BD%AE%E7%BE%A4%E5%90%8D) |
|
||||||
| /get_image | 获取图片信息(拓展API) |
|
| /set_restart | [重启go-cqhttp](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_restart-%E9%87%8D%E5%90%AF-onebot-%E5%AE%9E%E7%8E%B0) |
|
||||||
| /get_group_msg | 获取群组消息(拓展API) |
|
| /get_image | [获取图片信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_image-%E8%8E%B7%E5%8F%96%E5%9B%BE%E7%89%87) |
|
||||||
| /can_send_image | [检查是否可以发送图片](https://cqhttp.cc/docs/4.15/#/API?id=can_send_image-检查是否可以发送图片) |
|
| /get_msg | [获取消息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_msg-%E8%8E%B7%E5%8F%96%E6%B6%88%E6%81%AF) |
|
||||||
| /can_send_record | [检查是否可以发送语音](https://cqhttp.cc/docs/4.15/#/API?id=can_send_record-检查是否可以发送语音) |
|
| /can_send_image | [检查是否可以发送图片](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#can_send_image-%E6%A3%80%E6%9F%A5%E6%98%AF%E5%90%A6%E5%8F%AF%E4%BB%A5%E5%8F%91%E9%80%81%E5%9B%BE%E7%89%87) |
|
||||||
| /get_status | [获取插件运行状态](https://cqhttp.cc/docs/4.15/#/API?id=get_status-获取插件运行状态) |
|
| /can_send_record | [检查是否可以发送语音](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#can_send_record-%E6%A3%80%E6%9F%A5%E6%98%AF%E5%90%A6%E5%8F%AF%E4%BB%A5%E5%8F%91%E9%80%81%E8%AF%AD%E9%9F%B3) |
|
||||||
| /get_version_info | [获取 酷Q 及 CQHTTP插件的版本信息](https://cqhttp.cc/docs/4.15/#/API?id=get_version_info-获取-酷q-及-cqhttp-插件的版本信息) |
|
| /get_status | [获取插件运行状态](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_status-%E8%8E%B7%E5%8F%96%E8%BF%90%E8%A1%8C%E7%8A%B6%E6%80%81) |
|
||||||
|
| /get_version_info | [获取 酷Q 及 CQHTTP插件的版本信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_version_info-%E8%8E%B7%E5%8F%96%E7%89%88%E6%9C%AC%E4%BF%A1%E6%81%AF) |
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
@ -75,22 +81,33 @@
|
|||||||
<summary>已实现Event</summary>
|
<summary>已实现Event</summary>
|
||||||
|
|
||||||
##### 注意: 部分Event数据与CQHTTP原版略有差异,请参考文档
|
##### 注意: 部分Event数据与CQHTTP原版略有差异,请参考文档
|
||||||
| Event |
|
| Event |
|
||||||
| ------------------------------------------------------------ |
|
| ---------------------------------------------------------------------------------------------------------------------------------------------------- |
|
||||||
| [私聊信息](https://cqhttp.cc/docs/4.15/#/Post?id=私聊消息) |
|
| [私聊信息](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/message.md#%E7%A7%81%E8%81%8A%E6%B6%88%E6%81%AF) |
|
||||||
| [群消息](https://cqhttp.cc/docs/4.15/#/Post?id=群消息) |
|
| [群消息](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/message.md#%E7%BE%A4%E6%B6%88%E6%81%AF) |
|
||||||
| [群消息撤回(拓展Event)](docs/cqhttp.md#群消息撤回) |
|
| [群消息撤回(拓展Event)](docs/cqhttp.md#群消息撤回) |
|
||||||
| [好友消息撤回(拓展Event)](docs/cqhttp.md#好友消息撤回) |
|
| [好友消息撤回(拓展Event)](docs/cqhttp.md#好友消息撤回) |
|
||||||
| [群管理员变动](https://cqhttp.cc/docs/4.15/#/Post?id=群管理员变动) |
|
| [群内提示事件(拓展Event)(龙王等事件)](docs/cqhttp.md#群内戳一戳) |
|
||||||
| [群成员减少](https://cqhttp.cc/docs/4.15/#/Post?id=群成员减少) |
|
| [群管理员变动](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/notice.md#%E7%BE%A4%E7%AE%A1%E7%90%86%E5%91%98%E5%8F%98%E5%8A%A8) |
|
||||||
| [群成员增加](https://cqhttp.cc/docs/4.15/#/Post?id=群成员增加) |
|
| [群成员减少](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/notice.md#%E7%BE%A4%E6%88%90%E5%91%98%E5%87%8F%E5%B0%91) |
|
||||||
| [群禁言](https://cqhttp.cc/docs/4.15/#/Post?id=群禁言) |
|
| [群成员增加](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/notice.md#%E7%BE%A4%E6%88%90%E5%91%98%E5%A2%9E%E5%8A%A0) |
|
||||||
| [群文件上传](https://cqhttp.cc/docs/4.15/#/Post?id=群文件上传)|
|
| [群禁言](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/notice.md#%E7%BE%A4%E7%A6%81%E8%A8%80) |
|
||||||
| [加好友请求](https://cqhttp.cc/docs/4.15/#/Post?id=加好友请求) |
|
| [群文件上传](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/notice.md#%E7%BE%A4%E6%96%87%E4%BB%B6%E4%B8%8A%E4%BC%A0) |
|
||||||
| [加群请求/邀请](https://cqhttp.cc/docs/4.15/#/Post?id=加群请求/邀请) |
|
| [加好友请求](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/request.md#%E5%8A%A0%E5%A5%BD%E5%8F%8B%E8%AF%B7%E6%B1%82) |
|
||||||
|
| [加群请求/邀请](https://github.com/howmanybots/onebot/blob/master/v11/specs/event/request.md#%E5%8A%A0%E7%BE%A4%E8%AF%B7%E6%B1%82%E9%82%80%E8%AF%B7) |
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
# 关于ISSUE
|
||||||
|
|
||||||
|
以下ISSUE会被直接关闭
|
||||||
|
- 提交BUG不使用Template
|
||||||
|
- 询问已知问题
|
||||||
|
- 提问找不到重点
|
||||||
|
- 重复提问
|
||||||
|
|
||||||
|
> 请注意, 开发者并没有义务回复您的问题. 您应该具备基本的提问技巧。
|
||||||
|
|
||||||
# 性能
|
# 性能
|
||||||
|
|
||||||
在关闭数据库的情况下, 加载25个好友128个群运行24小时后内存使用为10MB左右. 开启数据库后内存使用将根据消息量增加10-20MB, 如果系统内存小于128M建议关闭数据库使用.
|
在关闭数据库的情况下, 加载25个好友128个群运行24小时后内存使用为10MB左右. 开启数据库后内存使用将根据消息量增加10-20MB, 如果系统内存小于128M建议关闭数据库使用.
|
||||||
|
551
coolq/api.go
551
coolq/api.go
@ -1,20 +1,24 @@
|
|||||||
package coolq
|
package coolq
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"runtime"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/Mrs4s/MiraiGo/binary"
|
"github.com/Mrs4s/MiraiGo/binary"
|
||||||
"github.com/Mrs4s/MiraiGo/client"
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
"github.com/Mrs4s/MiraiGo/message"
|
"github.com/Mrs4s/MiraiGo/message"
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"github.com/tidwall/gjson"
|
"github.com/tidwall/gjson"
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"runtime"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var Version = "unknown"
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=get_login_info-%E8%8E%B7%E5%8F%96%E7%99%BB%E5%BD%95%E5%8F%B7%E4%BF%A1%E6%81%AF
|
// https://cqhttp.cc/docs/4.15/#/API?id=get_login_info-%E8%8E%B7%E5%8F%96%E7%99%BB%E5%BD%95%E5%8F%B7%E4%BF%A1%E6%81%AF
|
||||||
func (bot *CQBot) CQGetLoginInfo() MSG {
|
func (bot *CQBot) CQGetLoginInfo() MSG {
|
||||||
return OK(MSG{"user_id": bot.Client.Uin, "nickname": bot.Client.Nickname})
|
return OK(MSG{"user_id": bot.Client.Uin, "nickname": bot.Client.Nickname})
|
||||||
@ -22,7 +26,7 @@ func (bot *CQBot) CQGetLoginInfo() MSG {
|
|||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=get_friend_list-%E8%8E%B7%E5%8F%96%E5%A5%BD%E5%8F%8B%E5%88%97%E8%A1%A8
|
// https://cqhttp.cc/docs/4.15/#/API?id=get_friend_list-%E8%8E%B7%E5%8F%96%E5%A5%BD%E5%8F%8B%E5%88%97%E8%A1%A8
|
||||||
func (bot *CQBot) CQGetFriendList() MSG {
|
func (bot *CQBot) CQGetFriendList() MSG {
|
||||||
var fs []MSG
|
fs := make([]MSG, 0)
|
||||||
for _, f := range bot.Client.FriendList {
|
for _, f := range bot.Client.FriendList {
|
||||||
fs = append(fs, MSG{
|
fs = append(fs, MSG{
|
||||||
"nickname": f.Nickname,
|
"nickname": f.Nickname,
|
||||||
@ -34,8 +38,11 @@ func (bot *CQBot) CQGetFriendList() MSG {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=get_group_list-%E8%8E%B7%E5%8F%96%E7%BE%A4%E5%88%97%E8%A1%A8
|
// https://cqhttp.cc/docs/4.15/#/API?id=get_group_list-%E8%8E%B7%E5%8F%96%E7%BE%A4%E5%88%97%E8%A1%A8
|
||||||
func (bot *CQBot) CQGetGroupList() MSG {
|
func (bot *CQBot) CQGetGroupList(noCache bool) MSG {
|
||||||
var gs []MSG
|
gs := make([]MSG, 0)
|
||||||
|
if noCache {
|
||||||
|
_ = bot.Client.ReloadGroupList()
|
||||||
|
}
|
||||||
for _, g := range bot.Client.GroupList {
|
for _, g := range bot.Client.GroupList {
|
||||||
gs = append(gs, MSG{
|
gs = append(gs, MSG{
|
||||||
"group_id": g.Code,
|
"group_id": g.Code,
|
||||||
@ -51,7 +58,7 @@ func (bot *CQBot) CQGetGroupList() MSG {
|
|||||||
func (bot *CQBot) CQGetGroupInfo(groupId int64) MSG {
|
func (bot *CQBot) CQGetGroupInfo(groupId int64) MSG {
|
||||||
group := bot.Client.FindGroup(groupId)
|
group := bot.Client.FindGroup(groupId)
|
||||||
if group == nil {
|
if group == nil {
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
return OK(MSG{
|
return OK(MSG{
|
||||||
"group_id": group.Code,
|
"group_id": group.Code,
|
||||||
@ -62,12 +69,20 @@ func (bot *CQBot) CQGetGroupInfo(groupId int64) MSG {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=get_group_member_list-%E8%8E%B7%E5%8F%96%E7%BE%A4%E6%88%90%E5%91%98%E5%88%97%E8%A1%A8
|
// https://cqhttp.cc/docs/4.15/#/API?id=get_group_member_list-%E8%8E%B7%E5%8F%96%E7%BE%A4%E6%88%90%E5%91%98%E5%88%97%E8%A1%A8
|
||||||
func (bot *CQBot) CQGetGroupMemberList(groupId int64) MSG {
|
func (bot *CQBot) CQGetGroupMemberList(groupId int64, noCache bool) MSG {
|
||||||
group := bot.Client.FindGroup(groupId)
|
group := bot.Client.FindGroup(groupId)
|
||||||
if group == nil {
|
if group == nil {
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
var members []MSG
|
if noCache {
|
||||||
|
t, err := bot.Client.GetGroupMembers(group)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("刷新群 %v 成员列表失败: %v", groupId, err)
|
||||||
|
return Failed(100, "GET_MEMBERS_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
group.Members = t
|
||||||
|
}
|
||||||
|
members := make([]MSG, 0)
|
||||||
for _, m := range group.Members {
|
for _, m := range group.Members {
|
||||||
members = append(members, convertGroupMemberInfo(groupId, m))
|
members = append(members, convertGroupMemberInfo(groupId, m))
|
||||||
}
|
}
|
||||||
@ -75,36 +90,107 @@ func (bot *CQBot) CQGetGroupMemberList(groupId int64) MSG {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=get_group_member_info-%E8%8E%B7%E5%8F%96%E7%BE%A4%E6%88%90%E5%91%98%E4%BF%A1%E6%81%AF
|
// https://cqhttp.cc/docs/4.15/#/API?id=get_group_member_info-%E8%8E%B7%E5%8F%96%E7%BE%A4%E6%88%90%E5%91%98%E4%BF%A1%E6%81%AF
|
||||||
func (bot *CQBot) CQGetGroupMemberInfo(groupId, userId int64, noCache bool) MSG {
|
func (bot *CQBot) CQGetGroupMemberInfo(groupId, userId int64) MSG {
|
||||||
group := bot.Client.FindGroup(groupId)
|
group := bot.Client.FindGroup(groupId)
|
||||||
if group == nil {
|
if group == nil {
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
|
||||||
if noCache {
|
|
||||||
t, err := bot.Client.GetGroupMembers(group)
|
|
||||||
if err != nil {
|
|
||||||
log.Warnf("刷新群 %v 成员列表失败: %v", groupId, err)
|
|
||||||
return Failed(100)
|
|
||||||
}
|
|
||||||
group.Members = t
|
|
||||||
}
|
}
|
||||||
member := group.FindMember(userId)
|
member := group.FindMember(userId)
|
||||||
if member == nil {
|
if member == nil {
|
||||||
return Failed(102)
|
return Failed(100, "MEMBER_NOT_FOUND", "群员不存在")
|
||||||
}
|
}
|
||||||
return OK(convertGroupMemberInfo(groupId, member))
|
return OK(convertGroupMemberInfo(groupId, member))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetGroupFileSystemInfo(groupId int64) MSG {
|
||||||
|
fs, err := bot.Client.GetGroupFileSystem(groupId)
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("获取群 %v 文件系统信息失败: %v", groupId, err)
|
||||||
|
return Failed(100, "FILE_SYSTEM_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
return OK(fs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetGroupRootFiles(groupId int64) MSG {
|
||||||
|
fs, err := bot.Client.GetGroupFileSystem(groupId)
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("获取群 %v 文件系统信息失败: %v", groupId, err)
|
||||||
|
return Failed(100, "FILE_SYSTEM_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
files, folders, err := fs.Root()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("获取群 %v 根目录文件失败: %v", groupId, err)
|
||||||
|
return Failed(100, "FILE_SYSTEM_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
return OK(MSG{
|
||||||
|
"files": files,
|
||||||
|
"folders": folders,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetGroupFilesByFolderId(groupId int64, folderId string) MSG {
|
||||||
|
fs, err := bot.Client.GetGroupFileSystem(groupId)
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("获取群 %v 文件系统信息失败: %v", groupId, err)
|
||||||
|
return Failed(100, "FILE_SYSTEM_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
files, folders, err := fs.GetFilesByFolder(folderId)
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("获取群 %v 根目录 %v 子文件失败: %v", groupId, folderId, err)
|
||||||
|
return Failed(100, "FILE_SYSTEM_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
return OK(MSG{
|
||||||
|
"files": files,
|
||||||
|
"folders": folders,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetGroupFileUrl(groupId int64, fileId string, busId int32) MSG {
|
||||||
|
url := bot.Client.GetGroupFileUrl(groupId, fileId, busId)
|
||||||
|
if url == "" {
|
||||||
|
return Failed(100, "FILE_SYSTEM_API_ERROR")
|
||||||
|
}
|
||||||
|
return OK(MSG{
|
||||||
|
"url": url,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetWordSlices(content string) MSG {
|
||||||
|
slices, err := bot.Client.GetWordSegmentation(content)
|
||||||
|
if err != nil {
|
||||||
|
return Failed(100, "WORD_SEGMENTATION_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
for i := 0; i < len(slices); i++ {
|
||||||
|
slices[i] = strings.ReplaceAll(slices[i], "\u0000", "")
|
||||||
|
}
|
||||||
|
return OK(MSG{"slices": slices})
|
||||||
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=send_group_msg-%E5%8F%91%E9%80%81%E7%BE%A4%E6%B6%88%E6%81%AF
|
// https://cqhttp.cc/docs/4.15/#/API?id=send_group_msg-%E5%8F%91%E9%80%81%E7%BE%A4%E6%B6%88%E6%81%AF
|
||||||
func (bot *CQBot) CQSendGroupMessage(groupId int64, i interface{}) MSG {
|
func (bot *CQBot) CQSendGroupMessage(groupId int64, i interface{}, autoEscape bool) MSG {
|
||||||
var str string
|
var str string
|
||||||
|
fixAt := func(elem []message.IMessageElement) {
|
||||||
|
for _, e := range elem {
|
||||||
|
if at, ok := e.(*message.AtElement); ok && at.Target != 0 {
|
||||||
|
at.Display = "@" + func() string {
|
||||||
|
mem := bot.Client.FindGroup(groupId).FindMember(at.Target)
|
||||||
|
if mem != nil {
|
||||||
|
return mem.DisplayName()
|
||||||
|
}
|
||||||
|
return strconv.FormatInt(at.Target, 10)
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
if m, ok := i.(gjson.Result); ok {
|
if m, ok := i.(gjson.Result); ok {
|
||||||
if m.Type == gjson.JSON {
|
if m.Type == gjson.JSON {
|
||||||
elem := bot.ConvertObjectMessage(m, true)
|
elem := bot.ConvertObjectMessage(m, true)
|
||||||
|
fixAt(elem)
|
||||||
mid := bot.SendGroupMessage(groupId, &message.SendingMessage{Elements: elem})
|
mid := bot.SendGroupMessage(groupId, &message.SendingMessage{Elements: elem})
|
||||||
if mid == -1 {
|
if mid == -1 {
|
||||||
return Failed(100)
|
return Failed(100, "SEND_MSG_API_ERROR", "请参考输出")
|
||||||
}
|
}
|
||||||
|
log.Infof("发送群 %v(%v) 的消息: %v (%v)", groupId, groupId, limitedString(m.String()), mid)
|
||||||
return OK(MSG{"message_id": mid})
|
return OK(MSG{"message_id": mid})
|
||||||
}
|
}
|
||||||
str = func() string {
|
str = func() string {
|
||||||
@ -113,24 +199,25 @@ func (bot *CQBot) CQSendGroupMessage(groupId int64, i interface{}) MSG {
|
|||||||
}
|
}
|
||||||
return m.Raw
|
return m.Raw
|
||||||
}()
|
}()
|
||||||
}
|
} else if s, ok := i.(string); ok {
|
||||||
if s, ok := i.(string); ok {
|
|
||||||
str = s
|
str = s
|
||||||
}
|
}
|
||||||
if str == "" {
|
if str == "" {
|
||||||
return Failed(100)
|
log.Warnf("群消息发送失败: 信息为空. MSG: %v", i)
|
||||||
|
return Failed(100, "EMPTY_MSG_ERROR", "消息为空")
|
||||||
}
|
}
|
||||||
elem := bot.ConvertStringMessage(str, true)
|
var elem []message.IMessageElement
|
||||||
// fix at display
|
if autoEscape {
|
||||||
for _, e := range elem {
|
elem = append(elem, message.NewText(str))
|
||||||
if at, ok := e.(*message.AtElement); ok && at.Target != 0 {
|
} else {
|
||||||
at.Display = "@" + bot.Client.FindGroup(groupId).FindMember(at.Target).DisplayName()
|
elem = bot.ConvertStringMessage(str, true)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
fixAt(elem)
|
||||||
mid := bot.SendGroupMessage(groupId, &message.SendingMessage{Elements: elem})
|
mid := bot.SendGroupMessage(groupId, &message.SendingMessage{Elements: elem})
|
||||||
if mid == -1 {
|
if mid == -1 {
|
||||||
return Failed(100)
|
return Failed(100, "SEND_MSG_API_ERROR", "请参考输出")
|
||||||
}
|
}
|
||||||
|
log.Infof("发送群 %v(%v) 的消息: %v (%v)", groupId, groupId, limitedString(str), mid)
|
||||||
return OK(MSG{"message_id": mid})
|
return OK(MSG{"message_id": mid})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -155,7 +242,7 @@ func (bot *CQBot) CQSendGroupForwardMessage(groupId int64, m gjson.Result) MSG {
|
|||||||
ts.Add(time.Second)
|
ts.Add(time.Second)
|
||||||
if e.Get("data.id").Exists() {
|
if e.Get("data.id").Exists() {
|
||||||
i, _ := strconv.Atoi(e.Get("data.id").Str)
|
i, _ := strconv.Atoi(e.Get("data.id").Str)
|
||||||
m := bot.GetGroupMessage(int32(i))
|
m := bot.GetMessage(int32(i))
|
||||||
if m != nil {
|
if m != nil {
|
||||||
sender := m["sender"].(message.Sender)
|
sender := m["sender"].(message.Sender)
|
||||||
nodes = append(nodes, &message.ForwardNode{
|
nodes = append(nodes, &message.ForwardNode{
|
||||||
@ -178,11 +265,24 @@ func (bot *CQBot) CQSendGroupForwardMessage(groupId int64, m gjson.Result) MSG {
|
|||||||
name := e.Get("data.name").Str
|
name := e.Get("data.name").Str
|
||||||
content := bot.ConvertObjectMessage(e.Get("data.content"), true)
|
content := bot.ConvertObjectMessage(e.Get("data.content"), true)
|
||||||
if uin != 0 && name != "" && len(content) > 0 {
|
if uin != 0 && name != "" && len(content) > 0 {
|
||||||
|
var newElem []message.IMessageElement
|
||||||
|
for _, elem := range content {
|
||||||
|
if img, ok := elem.(*message.ImageElement); ok {
|
||||||
|
gm, err := bot.Client.UploadGroupImage(groupId, img.Data)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("警告:群 %v 图片上传失败: %v", groupId, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
newElem = append(newElem, gm)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
newElem = append(newElem, elem)
|
||||||
|
}
|
||||||
nodes = append(nodes, &message.ForwardNode{
|
nodes = append(nodes, &message.ForwardNode{
|
||||||
SenderId: uin,
|
SenderId: uin,
|
||||||
SenderName: name,
|
SenderName: name,
|
||||||
Time: int32(ts.Unix()),
|
Time: int32(ts.Unix()),
|
||||||
Message: content,
|
Message: newElem,
|
||||||
})
|
})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -205,15 +305,16 @@ func (bot *CQBot) CQSendGroupForwardMessage(groupId int64, m gjson.Result) MSG {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=send_private_msg-%E5%8F%91%E9%80%81%E7%A7%81%E8%81%8A%E6%B6%88%E6%81%AF
|
// https://cqhttp.cc/docs/4.15/#/API?id=send_private_msg-%E5%8F%91%E9%80%81%E7%A7%81%E8%81%8A%E6%B6%88%E6%81%AF
|
||||||
func (bot *CQBot) CQSendPrivateMessage(userId int64, i interface{}) MSG {
|
func (bot *CQBot) CQSendPrivateMessage(userId int64, i interface{}, autoEscape bool) MSG {
|
||||||
var str string
|
var str string
|
||||||
if m, ok := i.(gjson.Result); ok {
|
if m, ok := i.(gjson.Result); ok {
|
||||||
if m.Type == gjson.JSON {
|
if m.Type == gjson.JSON {
|
||||||
elem := bot.ConvertObjectMessage(m, true)
|
elem := bot.ConvertObjectMessage(m, true)
|
||||||
mid := bot.SendPrivateMessage(userId, &message.SendingMessage{Elements: elem})
|
mid := bot.SendPrivateMessage(userId, &message.SendingMessage{Elements: elem})
|
||||||
if mid == -1 {
|
if mid == -1 {
|
||||||
return Failed(100)
|
return Failed(100, "SEND_MSG_API_ERROR", "请参考输出")
|
||||||
}
|
}
|
||||||
|
log.Infof("发送好友 %v(%v) 的消息: %v (%v)", userId, userId, limitedString(m.String()), mid)
|
||||||
return OK(MSG{"message_id": mid})
|
return OK(MSG{"message_id": mid})
|
||||||
}
|
}
|
||||||
str = func() string {
|
str = func() string {
|
||||||
@ -222,18 +323,23 @@ func (bot *CQBot) CQSendPrivateMessage(userId int64, i interface{}) MSG {
|
|||||||
}
|
}
|
||||||
return m.Raw
|
return m.Raw
|
||||||
}()
|
}()
|
||||||
}
|
} else if s, ok := i.(string); ok {
|
||||||
if s, ok := i.(string); ok {
|
|
||||||
str = s
|
str = s
|
||||||
}
|
}
|
||||||
if str == "" {
|
if str == "" {
|
||||||
return Failed(100)
|
return Failed(100, "EMPTY_MSG_ERROR", "消息为空")
|
||||||
|
}
|
||||||
|
var elem []message.IMessageElement
|
||||||
|
if autoEscape {
|
||||||
|
elem = append(elem, message.NewText(str))
|
||||||
|
} else {
|
||||||
|
elem = bot.ConvertStringMessage(str, false)
|
||||||
}
|
}
|
||||||
elem := bot.ConvertStringMessage(str, false)
|
|
||||||
mid := bot.SendPrivateMessage(userId, &message.SendingMessage{Elements: elem})
|
mid := bot.SendPrivateMessage(userId, &message.SendingMessage{Elements: elem})
|
||||||
if mid == -1 {
|
if mid == -1 {
|
||||||
return Failed(100)
|
return Failed(100, "SEND_MSG_API_ERROR", "请参考输出")
|
||||||
}
|
}
|
||||||
|
log.Infof("发送好友 %v(%v) 的消息: %v (%v)", userId, userId, limitedString(str), mid)
|
||||||
return OK(MSG{"message_id": mid})
|
return OK(MSG{"message_id": mid})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -245,7 +351,7 @@ func (bot *CQBot) CQSetGroupCard(groupId, userId int64, card string) MSG {
|
|||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_special_title-%E8%AE%BE%E7%BD%AE%E7%BE%A4%E7%BB%84%E4%B8%93%E5%B1%9E%E5%A4%B4%E8%A1%94
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_special_title-%E8%AE%BE%E7%BD%AE%E7%BE%A4%E7%BB%84%E4%B8%93%E5%B1%9E%E5%A4%B4%E8%A1%94
|
||||||
@ -256,7 +362,7 @@ func (bot *CQBot) CQSetGroupSpecialTitle(groupId, userId int64, title string) MS
|
|||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) CQSetGroupName(groupId int64, name string) MSG {
|
func (bot *CQBot) CQSetGroupName(groupId int64, name string) MSG {
|
||||||
@ -264,18 +370,26 @@ func (bot *CQBot) CQSetGroupName(groupId int64, name string) MSG {
|
|||||||
g.UpdateName(name)
|
g.UpdateName(name)
|
||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQSetGroupMemo(groupId int64, msg string) MSG {
|
||||||
|
if g := bot.Client.FindGroup(groupId); g != nil {
|
||||||
|
g.UpdateMemo(msg)
|
||||||
|
return OK(nil)
|
||||||
|
}
|
||||||
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_kick-%E7%BE%A4%E7%BB%84%E8%B8%A2%E4%BA%BA
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_kick-%E7%BE%A4%E7%BB%84%E8%B8%A2%E4%BA%BA
|
||||||
func (bot *CQBot) CQSetGroupKick(groupId, userId int64, msg string) MSG {
|
func (bot *CQBot) CQSetGroupKick(groupId, userId int64, msg string, block bool) MSG {
|
||||||
if g := bot.Client.FindGroup(groupId); g != nil {
|
if g := bot.Client.FindGroup(groupId); g != nil {
|
||||||
if m := g.FindMember(userId); m != nil {
|
if m := g.FindMember(userId); m != nil {
|
||||||
m.Kick(msg)
|
m.Kick(msg, block)
|
||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_ban-%E7%BE%A4%E7%BB%84%E5%8D%95%E4%BA%BA%E7%A6%81%E8%A8%80
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_ban-%E7%BE%A4%E7%BB%84%E5%8D%95%E4%BA%BA%E7%A6%81%E8%A8%80
|
||||||
@ -286,7 +400,7 @@ func (bot *CQBot) CQSetGroupBan(groupId, userId int64, duration uint32) MSG {
|
|||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_whole_ban-%E7%BE%A4%E7%BB%84%E5%85%A8%E5%91%98%E7%A6%81%E8%A8%80
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_whole_ban-%E7%BE%A4%E7%BB%84%E5%85%A8%E5%91%98%E7%A6%81%E8%A8%80
|
||||||
@ -295,7 +409,7 @@ func (bot *CQBot) CQSetGroupWholeBan(groupId int64, enable bool) MSG {
|
|||||||
g.MuteAll(enable)
|
g.MuteAll(enable)
|
||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_leave-%E9%80%80%E5%87%BA%E7%BE%A4%E7%BB%84
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_leave-%E9%80%80%E5%87%BA%E7%BE%A4%E7%BB%84
|
||||||
@ -304,14 +418,14 @@ func (bot *CQBot) CQSetGroupLeave(groupId int64) MSG {
|
|||||||
g.Quit()
|
g.Quit()
|
||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
return Failed(100)
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_friend_add_request-%E5%A4%84%E7%90%86%E5%8A%A0%E5%A5%BD%E5%8F%8B%E8%AF%B7%E6%B1%82
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_friend_add_request-%E5%A4%84%E7%90%86%E5%8A%A0%E5%A5%BD%E5%8F%8B%E8%AF%B7%E6%B1%82
|
||||||
func (bot *CQBot) CQProcessFriendRequest(flag string, approve bool) MSG {
|
func (bot *CQBot) CQProcessFriendRequest(flag string, approve bool) MSG {
|
||||||
req, ok := bot.friendReqCache.Load(flag)
|
req, ok := bot.friendReqCache.Load(flag)
|
||||||
if !ok {
|
if !ok {
|
||||||
return Failed(100)
|
return Failed(100, "FLAG_NOT_FOUND", "FLAG不存在")
|
||||||
}
|
}
|
||||||
if approve {
|
if approve {
|
||||||
req.(*client.NewFriendRequest).Accept()
|
req.(*client.NewFriendRequest).Accept()
|
||||||
@ -322,43 +436,184 @@ func (bot *CQBot) CQProcessFriendRequest(flag string, approve bool) MSG {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_add_request-%E5%A4%84%E7%90%86%E5%8A%A0%E7%BE%A4%E8%AF%B7%E6%B1%82%EF%BC%8F%E9%82%80%E8%AF%B7
|
// https://cqhttp.cc/docs/4.15/#/API?id=set_group_add_request-%E5%A4%84%E7%90%86%E5%8A%A0%E7%BE%A4%E8%AF%B7%E6%B1%82%EF%BC%8F%E9%82%80%E8%AF%B7
|
||||||
func (bot *CQBot) CQProcessGroupRequest(flag, subType string, approve bool) MSG {
|
func (bot *CQBot) CQProcessGroupRequest(flag, subType, reason string, approve bool) MSG {
|
||||||
|
msgs, err := bot.Client.GetGroupSystemMessages()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("获取群系统消息失败: %v", err)
|
||||||
|
return Failed(100, "SYSTEM_MSG_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
if subType == "add" {
|
if subType == "add" {
|
||||||
req, ok := bot.joinReqCache.Load(flag)
|
for _, req := range msgs.JoinRequests {
|
||||||
if !ok {
|
if strconv.FormatInt(req.RequestId, 10) == flag {
|
||||||
return Failed(100)
|
if req.Checked {
|
||||||
|
log.Errorf("处理群系统消息失败: 无法操作已处理的消息.")
|
||||||
|
return Failed(100, "FLAG_HAS_BEEN_CHECKED", "消息已被处理")
|
||||||
|
}
|
||||||
|
if approve {
|
||||||
|
req.Accept()
|
||||||
|
} else {
|
||||||
|
req.Reject(false, reason)
|
||||||
|
}
|
||||||
|
return OK(nil)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
bot.joinReqCache.Delete(flag)
|
} else {
|
||||||
if approve {
|
for _, req := range msgs.InvitedRequests {
|
||||||
req.(*client.UserJoinGroupRequest).Accept()
|
if strconv.FormatInt(req.RequestId, 10) == flag {
|
||||||
} else {
|
if req.Checked {
|
||||||
req.(*client.UserJoinGroupRequest).Reject()
|
log.Errorf("处理群系统消息失败: 无法操作已处理的消息.")
|
||||||
|
return Failed(100, "FLAG_HAS_BEEN_CHECKED", "消息已被处理")
|
||||||
|
}
|
||||||
|
if approve {
|
||||||
|
req.Accept()
|
||||||
|
} else {
|
||||||
|
req.Reject(false, reason)
|
||||||
|
}
|
||||||
|
return OK(nil)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return OK(nil)
|
|
||||||
}
|
}
|
||||||
req, ok := bot.invitedReqCache.Load(flag)
|
log.Errorf("处理群系统消息失败: 消息 %v 不存在.", flag)
|
||||||
if ok {
|
return Failed(100, "FLAG_NOT_FOUND", "FLAG不存在")
|
||||||
bot.invitedReqCache.Delete(flag)
|
|
||||||
if approve {
|
|
||||||
req.(*client.GroupInvitedRequest).Accept()
|
|
||||||
} else {
|
|
||||||
req.(*client.GroupInvitedRequest).Reject()
|
|
||||||
}
|
|
||||||
return OK(nil)
|
|
||||||
}
|
|
||||||
return Failed(100)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=delete_msg-%E6%92%A4%E5%9B%9E%E6%B6%88%E6%81%AF
|
// https://cqhttp.cc/docs/4.15/#/API?id=delete_msg-%E6%92%A4%E5%9B%9E%E6%B6%88%E6%81%AF
|
||||||
func (bot *CQBot) CQDeleteMessage(messageId int32) MSG {
|
func (bot *CQBot) CQDeleteMessage(messageId int32) MSG {
|
||||||
msg := bot.GetGroupMessage(messageId)
|
msg := bot.GetMessage(messageId)
|
||||||
if msg == nil {
|
if msg == nil {
|
||||||
return Failed(100)
|
return Failed(100, "MESSAGE_NOT_FOUND", "消息不存在")
|
||||||
|
}
|
||||||
|
if _, ok := msg["group"]; ok {
|
||||||
|
if err := bot.Client.RecallGroupMessage(msg["group"].(int64), msg["message-id"].(int32), msg["internal-id"].(int32)); err != nil {
|
||||||
|
log.Warnf("撤回 %v 失败: %v", messageId, err)
|
||||||
|
return Failed(100, "RECALL_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if msg["sender"].(message.Sender).Uin != bot.Client.Uin {
|
||||||
|
log.Warnf("撤回 %v 失败: 好友会话无法撤回对方消息.", messageId)
|
||||||
|
return Failed(100, "CANNOT_RECALL_FRIEND_MSG", "无法撤回对方消息")
|
||||||
|
}
|
||||||
|
if err := bot.Client.RecallPrivateMessage(msg["target"].(int64), int64(msg["time"].(int32)), msg["message-id"].(int32), msg["internal-id"].(int32)); err != nil {
|
||||||
|
log.Warnf("撤回 %v 失败: %v", messageId, err)
|
||||||
|
return Failed(100, "RECALL_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
bot.Client.RecallGroupMessage(msg["group"].(int64), msg["message-id"].(int32), msg["internal-id"].(int32))
|
|
||||||
return OK(nil)
|
return OK(nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#set_group_admin-%E7%BE%A4%E7%BB%84%E8%AE%BE%E7%BD%AE%E7%AE%A1%E7%90%86%E5%91%98
|
||||||
|
func (bot *CQBot) CQSetGroupAdmin(groupId, userId int64, enable bool) MSG {
|
||||||
|
group := bot.Client.FindGroup(groupId)
|
||||||
|
if group == nil || group.OwnerUin != bot.Client.Uin {
|
||||||
|
return Failed(100, "PERMISSION_DENIED", "群不存在或权限不足")
|
||||||
|
}
|
||||||
|
mem := group.FindMember(userId)
|
||||||
|
if mem == nil {
|
||||||
|
return Failed(100, "GROUP_MEMBER_NOT_FOUND", "群成员不存在")
|
||||||
|
}
|
||||||
|
mem.SetAdmin(enable)
|
||||||
|
t, err := bot.Client.GetGroupMembers(group)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("刷新群 %v 成员列表失败: %v", groupId, err)
|
||||||
|
return Failed(100, "GET_MEMBERS_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
group.Members = t
|
||||||
|
return OK(nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetVipInfo(userId int64) MSG {
|
||||||
|
vip, err := bot.Client.GetVipInfo(userId)
|
||||||
|
if err != nil {
|
||||||
|
return Failed(100, "VIP_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
msg := MSG{
|
||||||
|
"user_id": vip.Uin,
|
||||||
|
"nickname": vip.Name,
|
||||||
|
"level": vip.Level,
|
||||||
|
"level_speed": vip.LevelSpeed,
|
||||||
|
"vip_level": vip.VipLevel,
|
||||||
|
"vip_growth_speed": vip.VipGrowthSpeed,
|
||||||
|
"vip_growth_total": vip.VipGrowthTotal,
|
||||||
|
}
|
||||||
|
return OK(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_group_honor_info-%E8%8E%B7%E5%8F%96%E7%BE%A4%E8%8D%A3%E8%AA%89%E4%BF%A1%E6%81%AF
|
||||||
|
func (bot *CQBot) CQGetGroupHonorInfo(groupId int64, t string) MSG {
|
||||||
|
msg := MSG{"group_id": groupId}
|
||||||
|
convertMem := func(memList []client.HonorMemberInfo) (ret []MSG) {
|
||||||
|
for _, mem := range memList {
|
||||||
|
ret = append(ret, MSG{
|
||||||
|
"user_id": mem.Uin,
|
||||||
|
"nickname": mem.Name,
|
||||||
|
"avatar": mem.Avatar,
|
||||||
|
"description": mem.Desc,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if t == "talkative" || t == "all" {
|
||||||
|
if honor, err := bot.Client.GetGroupHonorInfo(groupId, client.Talkative); err == nil {
|
||||||
|
if honor.CurrentTalkative.Uin != 0 {
|
||||||
|
msg["current_talkative"] = MSG{
|
||||||
|
"user_id": honor.CurrentTalkative.Uin,
|
||||||
|
"nickname": honor.CurrentTalkative.Name,
|
||||||
|
"avatar": honor.CurrentTalkative.Avatar,
|
||||||
|
"day_count": honor.CurrentTalkative.DayCount,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
msg["talkative_list"] = convertMem(honor.TalkativeList)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if t == "performer" || t == "all" {
|
||||||
|
if honor, err := bot.Client.GetGroupHonorInfo(groupId, client.Performer); err == nil {
|
||||||
|
msg["performer_lis"] = convertMem(honor.ActorList)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if t == "legend" || t == "all" {
|
||||||
|
if honor, err := bot.Client.GetGroupHonorInfo(groupId, client.Legend); err == nil {
|
||||||
|
msg["legend_list"] = convertMem(honor.LegendList)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if t == "strong_newbie" || t == "all" {
|
||||||
|
if honor, err := bot.Client.GetGroupHonorInfo(groupId, client.StrongNewbie); err == nil {
|
||||||
|
msg["strong_newbie_list"] = convertMem(honor.StrongNewbieList)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if t == "emotion" || t == "all" {
|
||||||
|
if honor, err := bot.Client.GetGroupHonorInfo(groupId, client.Emotion); err == nil {
|
||||||
|
msg["emotion_list"] = convertMem(honor.EmotionList)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return OK(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_stranger_info-%E8%8E%B7%E5%8F%96%E9%99%8C%E7%94%9F%E4%BA%BA%E4%BF%A1%E6%81%AF
|
||||||
|
func (bot *CQBot) CQGetStrangerInfo(userId int64) MSG {
|
||||||
|
info, err := bot.Client.GetSummaryInfo(userId)
|
||||||
|
if err != nil {
|
||||||
|
return Failed(100, "SUMMARY_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
return OK(MSG{
|
||||||
|
"user_id": info.Uin,
|
||||||
|
"nickname": info.Nickname,
|
||||||
|
"sex": func() string {
|
||||||
|
if info.Sex == 1 {
|
||||||
|
return "female"
|
||||||
|
}
|
||||||
|
return "male"
|
||||||
|
}(),
|
||||||
|
"age": info.Age,
|
||||||
|
"level": info.Level,
|
||||||
|
"login_days": info.LoginDays,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// https://cqhttp.cc/docs/4.15/#/API?id=-handle_quick_operation-%E5%AF%B9%E4%BA%8B%E4%BB%B6%E6%89%A7%E8%A1%8C%E5%BF%AB%E9%80%9F%E6%93%8D%E4%BD%9C
|
// https://cqhttp.cc/docs/4.15/#/API?id=-handle_quick_operation-%E5%AF%B9%E4%BA%8B%E4%BB%B6%E6%89%A7%E8%A1%8C%E5%BF%AB%E9%80%9F%E6%93%8D%E4%BD%9C
|
||||||
// https://github.com/richardchien/coolq-http-api/blob/master/src/cqhttp/plugins/web/http.cpp#L376
|
// https://github.com/richardchien/coolq-http-api/blob/master/src/cqhttp/plugins/web/http.cpp#L376
|
||||||
func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) MSG {
|
func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) MSG {
|
||||||
@ -368,6 +623,7 @@ func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) MSG {
|
|||||||
msgType := context.Get("message_type").Str
|
msgType := context.Get("message_type").Str
|
||||||
reply := operation.Get("reply")
|
reply := operation.Get("reply")
|
||||||
if reply.Exists() {
|
if reply.Exists() {
|
||||||
|
autoEscape := global.EnsureBool(operation.Get("auto_escape"), false)
|
||||||
/*
|
/*
|
||||||
at := true
|
at := true
|
||||||
if operation.Get("at_sender").Exists() {
|
if operation.Get("at_sender").Exists() {
|
||||||
@ -376,10 +632,10 @@ func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) MSG {
|
|||||||
*/
|
*/
|
||||||
// TODO: 处理at字段
|
// TODO: 处理at字段
|
||||||
if msgType == "group" {
|
if msgType == "group" {
|
||||||
bot.CQSendGroupMessage(context.Get("group_id").Int(), reply)
|
bot.CQSendGroupMessage(context.Get("group_id").Int(), reply, autoEscape)
|
||||||
}
|
}
|
||||||
if msgType == "private" {
|
if msgType == "private" {
|
||||||
bot.CQSendPrivateMessage(context.Get("user_id").Int(), reply)
|
bot.CQSendPrivateMessage(context.Get("user_id").Int(), reply, autoEscape)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if msgType == "group" {
|
if msgType == "group" {
|
||||||
@ -389,7 +645,7 @@ func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) MSG {
|
|||||||
bot.CQDeleteMessage(int32(context.Get("message_id").Int()))
|
bot.CQDeleteMessage(int32(context.Get("message_id").Int()))
|
||||||
}
|
}
|
||||||
if operation.Get("kick").Bool() && !isAnonymous {
|
if operation.Get("kick").Bool() && !isAnonymous {
|
||||||
bot.CQSetGroupKick(context.Get("group_id").Int(), context.Get("user_id").Int(), "")
|
bot.CQSetGroupKick(context.Get("group_id").Int(), context.Get("user_id").Int(), "", operation.Get("reject_add_request").Bool())
|
||||||
}
|
}
|
||||||
if operation.Get("ban").Bool() {
|
if operation.Get("ban").Bool() {
|
||||||
var duration uint32 = 30 * 60
|
var duration uint32 = 30 * 60
|
||||||
@ -404,12 +660,12 @@ func (bot *CQBot) CQHandleQuickOperation(context, operation gjson.Result) MSG {
|
|||||||
}
|
}
|
||||||
case "request":
|
case "request":
|
||||||
reqType := context.Get("request_type").Str
|
reqType := context.Get("request_type").Str
|
||||||
if context.Get("approve").Bool() {
|
if operation.Get("approve").Exists() {
|
||||||
if reqType == "friend" {
|
if reqType == "friend" {
|
||||||
bot.CQProcessFriendRequest(context.Get("flag").Str, true)
|
bot.CQProcessFriendRequest(context.Get("flag").Str, operation.Get("approve").Bool())
|
||||||
}
|
}
|
||||||
if reqType == "group" {
|
if reqType == "group" {
|
||||||
bot.CQProcessGroupRequest(context.Get("flag").Str, context.Get("sub_type").Str, true)
|
bot.CQProcessGroupRequest(context.Get("flag").Str, context.Get("sub_type").Str, operation.Get("reason").Str, operation.Get("approve").Bool())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -423,30 +679,39 @@ func (bot *CQBot) CQGetImage(file string) MSG {
|
|||||||
if b, err := ioutil.ReadFile(path.Join(global.IMAGE_PATH, file)); err == nil {
|
if b, err := ioutil.ReadFile(path.Join(global.IMAGE_PATH, file)); err == nil {
|
||||||
r := binary.NewReader(b)
|
r := binary.NewReader(b)
|
||||||
r.ReadBytes(16)
|
r.ReadBytes(16)
|
||||||
return OK(MSG{
|
msg := MSG{
|
||||||
"size": r.ReadInt32(),
|
"size": r.ReadInt32(),
|
||||||
"filename": r.ReadString(),
|
"filename": r.ReadString(),
|
||||||
"url": r.ReadString(),
|
"url": r.ReadString(),
|
||||||
})
|
}
|
||||||
|
local := path.Join(global.CACHE_PATH, file+"."+path.Ext(msg["filename"].(string)))
|
||||||
|
if !global.PathExists(local) {
|
||||||
|
if data, err := global.GetBytes(msg["url"].(string)); err == nil {
|
||||||
|
_ = ioutil.WriteFile(local, data, 0644)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
msg["file"] = local
|
||||||
|
return OK(msg)
|
||||||
|
} else {
|
||||||
|
return Failed(100, "LOAD_FILE_ERROR", err.Error())
|
||||||
}
|
}
|
||||||
return Failed(100)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) CQGetForwardMessage(resId string) MSG {
|
func (bot *CQBot) CQGetForwardMessage(resId string) MSG {
|
||||||
m := bot.Client.GetForwardMessage(resId)
|
m := bot.Client.GetForwardMessage(resId)
|
||||||
if m == nil {
|
if m == nil {
|
||||||
return Failed(100)
|
return Failed(100, "MSG_NOT_FOUND", "消息不存在")
|
||||||
}
|
}
|
||||||
var r []MSG
|
r := make([]MSG, 0)
|
||||||
for _, n := range m.Nodes {
|
for _, n := range m.Nodes {
|
||||||
checkImage(n.Message)
|
bot.checkMedia(n.Message)
|
||||||
r = append(r, MSG{
|
r = append(r, MSG{
|
||||||
"sender": MSG{
|
"sender": MSG{
|
||||||
"user_id": n.SenderId,
|
"user_id": n.SenderId,
|
||||||
"nickname": n.SenderName,
|
"nickname": n.SenderName,
|
||||||
},
|
},
|
||||||
"time": n.Time,
|
"time": n.Time,
|
||||||
"content": ToStringMessage(n.Message, 0, false),
|
"content": ToFormattedMessage(n.Message, 0, false),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return OK(MSG{
|
return OK(MSG{
|
||||||
@ -454,32 +719,83 @@ func (bot *CQBot) CQGetForwardMessage(resId string) MSG {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) CQGetGroupMessage(messageId int32) MSG {
|
func (bot *CQBot) CQGetMessage(messageId int32) MSG {
|
||||||
msg := bot.GetGroupMessage(messageId)
|
msg := bot.GetMessage(messageId)
|
||||||
if msg == nil {
|
if msg == nil {
|
||||||
return Failed(100)
|
return Failed(100, "MSG_NOT_FOUND", "消息不存在")
|
||||||
}
|
}
|
||||||
sender := msg["sender"].(message.Sender)
|
sender := msg["sender"].(message.Sender)
|
||||||
|
gid, isGroup := msg["group"]
|
||||||
|
raw := msg["message"].(string)
|
||||||
return OK(MSG{
|
return OK(MSG{
|
||||||
"message_id": messageId,
|
"message_id": messageId,
|
||||||
"real_id": msg["message-id"],
|
"real_id": msg["message-id"],
|
||||||
|
"group": isGroup,
|
||||||
|
"group_id": gid,
|
||||||
"sender": MSG{
|
"sender": MSG{
|
||||||
"user_id": sender.Uin,
|
"user_id": sender.Uin,
|
||||||
"nickname": sender.Nickname,
|
"nickname": sender.Nickname,
|
||||||
},
|
},
|
||||||
"time": msg["time"],
|
"time": msg["time"],
|
||||||
"content": msg["message"],
|
"message": ToFormattedMessage(bot.ConvertStringMessage(raw, isGroup), func() int64 {
|
||||||
|
if isGroup {
|
||||||
|
return gid.(int64)
|
||||||
|
}
|
||||||
|
return sender.Uin
|
||||||
|
}(), false),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQGetGroupSystemMessages() MSG {
|
||||||
|
msg, err := bot.Client.GetGroupSystemMessages()
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("获取群系统消息失败: %v", err)
|
||||||
|
return Failed(100, "SYSTEM_MSG_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
return OK(msg)
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) CQCanSendImage() MSG {
|
func (bot *CQBot) CQCanSendImage() MSG {
|
||||||
return OK(MSG{"yes": true})
|
return OK(MSG{"yes": true})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) CQCanSendRecord() MSG {
|
func (bot *CQBot) CQCanSendRecord() MSG {
|
||||||
return OK(MSG{"yes": false})
|
return OK(MSG{"yes": true})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQOcrImage(imageId string) MSG {
|
||||||
|
img, err := bot.makeImageElem(map[string]string{"file": imageId}, true)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("load image error: %v", err)
|
||||||
|
return Failed(100, "LOAD_FILE_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
rsp, err := bot.Client.ImageOcr(img)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("ocr image error: %v", err)
|
||||||
|
return Failed(100, "OCR_API_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
return OK(rsp)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQReloadEventFilter() MSG {
|
||||||
|
global.BootFilter()
|
||||||
|
return OK(nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) CQSetGroupPortrait(groupId int64, file, cache string) MSG {
|
||||||
|
if g := bot.Client.FindGroup(groupId); g != nil {
|
||||||
|
img, err := global.FindFile(file, cache, global.IMAGE_PATH)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("set group portrait error: %v", err)
|
||||||
|
return Failed(100, "LOAD_FILE_ERROR", err.Error())
|
||||||
|
}
|
||||||
|
g.UpdateGroupHeadPortrait(img)
|
||||||
|
return OK(nil)
|
||||||
|
}
|
||||||
|
return Failed(100, "GROUP_NOT_FOUND", "群聊不存在")
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://github.com/howmanybots/onebot/blob/master/v11/specs/api/public.md#get_status-%E8%8E%B7%E5%8F%96%E8%BF%90%E8%A1%8C%E7%8A%B6%E6%80%81
|
||||||
func (bot *CQBot) CQGetStatus() MSG {
|
func (bot *CQBot) CQGetStatus() MSG {
|
||||||
return OK(MSG{
|
return OK(MSG{
|
||||||
"app_initialized": true,
|
"app_initialized": true,
|
||||||
@ -487,7 +803,8 @@ func (bot *CQBot) CQGetStatus() MSG {
|
|||||||
"plugins_good": nil,
|
"plugins_good": nil,
|
||||||
"app_good": true,
|
"app_good": true,
|
||||||
"online": bot.Client.Online,
|
"online": bot.Client.Online,
|
||||||
"good": true,
|
"good": bot.Client.Online,
|
||||||
|
"stat": bot.Client.GetStatistics(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -502,6 +819,21 @@ func (bot *CQBot) CQGetVersionInfo() MSG {
|
|||||||
"plugin_build_configuration": "release",
|
"plugin_build_configuration": "release",
|
||||||
"runtime_version": runtime.Version(),
|
"runtime_version": runtime.Version(),
|
||||||
"runtime_os": runtime.GOOS,
|
"runtime_os": runtime.GOOS,
|
||||||
|
"version": Version,
|
||||||
|
"protocol": func() int {
|
||||||
|
switch client.SystemDeviceInfo.Protocol {
|
||||||
|
case client.IPad:
|
||||||
|
return 0
|
||||||
|
case client.AndroidPhone:
|
||||||
|
return 1
|
||||||
|
case client.AndroidWatch:
|
||||||
|
return 2
|
||||||
|
case client.MacOS:
|
||||||
|
return 3
|
||||||
|
default:
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
}(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -509,8 +841,16 @@ func OK(data interface{}) MSG {
|
|||||||
return MSG{"data": data, "retcode": 0, "status": "ok"}
|
return MSG{"data": data, "retcode": 0, "status": "ok"}
|
||||||
}
|
}
|
||||||
|
|
||||||
func Failed(code int) MSG {
|
func Failed(code int, msg ...string) MSG {
|
||||||
return MSG{"data": nil, "retcode": code, "status": "failed"}
|
m := ""
|
||||||
|
w := ""
|
||||||
|
if len(msg) > 0 {
|
||||||
|
m = msg[0]
|
||||||
|
}
|
||||||
|
if len(msg) > 1 {
|
||||||
|
w = msg[1]
|
||||||
|
}
|
||||||
|
return MSG{"data": nil, "retcode": code, "msg": m, "wording": w, "status": "failed"}
|
||||||
}
|
}
|
||||||
|
|
||||||
func convertGroupMemberInfo(groupId int64, m *client.GroupMemberInfo) MSG {
|
func convertGroupMemberInfo(groupId int64, m *client.GroupMemberInfo) MSG {
|
||||||
@ -541,3 +881,12 @@ func convertGroupMemberInfo(groupId int64, m *client.GroupMemberInfo) MSG {
|
|||||||
"card_changeable": false,
|
"card_changeable": false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func limitedString(str string) string {
|
||||||
|
if strings.Count(str, "") <= 10 {
|
||||||
|
return str
|
||||||
|
}
|
||||||
|
limited := []rune(str)
|
||||||
|
limited = limited[:10]
|
||||||
|
return string(limited) + " ..."
|
||||||
|
}
|
||||||
|
319
coolq/bot.go
319
coolq/bot.go
@ -3,42 +3,49 @@ package coolq
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/gob"
|
"encoding/gob"
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"hash/crc32"
|
||||||
|
"path"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
|
|
||||||
"github.com/Mrs4s/MiraiGo/binary"
|
"github.com/Mrs4s/MiraiGo/binary"
|
||||||
"github.com/Mrs4s/MiraiGo/client"
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
"github.com/Mrs4s/MiraiGo/message"
|
"github.com/Mrs4s/MiraiGo/message"
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
|
||||||
|
jsoniter "github.com/json-iterator/go"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"github.com/xujiajun/nutsdb"
|
"github.com/tidwall/gjson"
|
||||||
"hash/crc32"
|
|
||||||
"path"
|
|
||||||
"sync"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
type CQBot struct {
|
type CQBot struct {
|
||||||
Client *client.QQClient
|
Client *client.QQClient
|
||||||
|
|
||||||
events []func(MSG)
|
events []func(MSG)
|
||||||
db *nutsdb.DB
|
db *leveldb.DB
|
||||||
friendReqCache sync.Map
|
friendReqCache sync.Map
|
||||||
invitedReqCache sync.Map
|
tempMsgCache sync.Map
|
||||||
joinReqCache sync.Map
|
oneWayMsgCache sync.Map
|
||||||
}
|
}
|
||||||
|
|
||||||
type MSG map[string]interface{}
|
type MSG map[string]interface{}
|
||||||
|
|
||||||
|
var ForceFragmented = false
|
||||||
|
|
||||||
func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
||||||
bot := &CQBot{
|
bot := &CQBot{
|
||||||
Client: cli,
|
Client: cli,
|
||||||
}
|
}
|
||||||
if conf.EnableDB {
|
if conf.EnableDB {
|
||||||
opt := nutsdb.DefaultOptions
|
p := path.Join("data", "leveldb")
|
||||||
opt.Dir = path.Join("data", "db")
|
db, err := leveldb.OpenFile(p, nil)
|
||||||
opt.EntryIdxMode = nutsdb.HintBPTSparseIdxMode
|
|
||||||
db, err := nutsdb.Open(opt)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("打开数据库失败, 如果频繁遇到此问题请清理 data/db 文件夹或关闭数据库功能。")
|
log.Fatalf("打开数据库失败, 如果频繁遇到此问题请清理 data/leveldb 文件夹或关闭数据库功能。")
|
||||||
}
|
}
|
||||||
bot.db = db
|
bot.db = db
|
||||||
gob.Register(message.Sender{})
|
gob.Register(message.Sender{})
|
||||||
@ -51,15 +58,41 @@ func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
|||||||
bot.Client.OnTempMessage(bot.tempMessageEvent)
|
bot.Client.OnTempMessage(bot.tempMessageEvent)
|
||||||
bot.Client.OnGroupMuted(bot.groupMutedEvent)
|
bot.Client.OnGroupMuted(bot.groupMutedEvent)
|
||||||
bot.Client.OnGroupMessageRecalled(bot.groupRecallEvent)
|
bot.Client.OnGroupMessageRecalled(bot.groupRecallEvent)
|
||||||
|
bot.Client.OnGroupNotify(bot.groupNotifyEvent)
|
||||||
|
bot.Client.OnFriendNotify(bot.friendNotifyEvent)
|
||||||
bot.Client.OnFriendMessageRecalled(bot.friendRecallEvent)
|
bot.Client.OnFriendMessageRecalled(bot.friendRecallEvent)
|
||||||
|
bot.Client.OnReceivedOfflineFile(bot.offlineFileEvent)
|
||||||
bot.Client.OnJoinGroup(bot.joinGroupEvent)
|
bot.Client.OnJoinGroup(bot.joinGroupEvent)
|
||||||
bot.Client.OnLeaveGroup(bot.leaveGroupEvent)
|
bot.Client.OnLeaveGroup(bot.leaveGroupEvent)
|
||||||
bot.Client.OnGroupMemberJoined(bot.memberJoinEvent)
|
bot.Client.OnGroupMemberJoined(bot.memberJoinEvent)
|
||||||
bot.Client.OnGroupMemberLeaved(bot.memberLeaveEvent)
|
bot.Client.OnGroupMemberLeaved(bot.memberLeaveEvent)
|
||||||
bot.Client.OnGroupMemberPermissionChanged(bot.memberPermissionChangedEvent)
|
bot.Client.OnGroupMemberPermissionChanged(bot.memberPermissionChangedEvent)
|
||||||
|
bot.Client.OnGroupMemberCardUpdated(bot.memberCardUpdatedEvent)
|
||||||
bot.Client.OnNewFriendRequest(bot.friendRequestEvent)
|
bot.Client.OnNewFriendRequest(bot.friendRequestEvent)
|
||||||
|
bot.Client.OnNewFriendAdded(bot.friendAddedEvent)
|
||||||
bot.Client.OnGroupInvited(bot.groupInvitedEvent)
|
bot.Client.OnGroupInvited(bot.groupInvitedEvent)
|
||||||
bot.Client.OnUserWantJoinGroup(bot.groupJoinReqEvent)
|
bot.Client.OnUserWantJoinGroup(bot.groupJoinReqEvent)
|
||||||
|
go func() {
|
||||||
|
i := conf.HeartbeatInterval
|
||||||
|
if i < 0 {
|
||||||
|
log.Warn("警告: 心跳功能已关闭,若非预期,请检查配置文件。")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if i == 0 {
|
||||||
|
i = 5
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
time.Sleep(time.Second * i)
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
"self_id": bot.Client.Uin,
|
||||||
|
"post_type": "meta_event",
|
||||||
|
"meta_event_type": "heartbeat",
|
||||||
|
"status": nil,
|
||||||
|
"interval": 1000 * i,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}()
|
||||||
return bot
|
return bot
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,22 +100,19 @@ func (bot *CQBot) OnEventPush(f func(m MSG)) {
|
|||||||
bot.events = append(bot.events, f)
|
bot.events = append(bot.events, f)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) GetGroupMessage(mid int32) MSG {
|
func (bot *CQBot) GetMessage(mid int32) MSG {
|
||||||
if bot.db != nil {
|
if bot.db != nil {
|
||||||
m := MSG{}
|
m := MSG{}
|
||||||
err := bot.db.View(func(tx *nutsdb.Tx) error {
|
data, err := bot.db.Get(binary.ToBytes(mid), nil)
|
||||||
e, err := tx.Get("group-messages", binary.ToBytes(mid))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
buff := new(bytes.Buffer)
|
|
||||||
buff.Write(binary.GZipUncompress(e.Value))
|
|
||||||
return gob.NewDecoder(buff).Decode(&m)
|
|
||||||
})
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
return m
|
buff := new(bytes.Buffer)
|
||||||
|
buff.Write(binary.GZipUncompress(data))
|
||||||
|
err = gob.NewDecoder(buff).Decode(&m)
|
||||||
|
if err == nil {
|
||||||
|
return m
|
||||||
|
}
|
||||||
}
|
}
|
||||||
log.Warnf("获取信息时出现错误: %v", err)
|
log.Warnf("获取信息时出现错误: %v id: %v", err, mid)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -100,7 +130,7 @@ func (bot *CQBot) SendGroupMessage(groupId int64, m *message.SendingMessage) int
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*message.VoiceElement); ok {
|
if i, ok := elem.(*message.VoiceElement); ok {
|
||||||
gv, err := bot.Client.UploadGroupPtt(groupId, i.Data, int32(len(i.Data)))
|
gv, err := bot.Client.UploadGroupPtt(groupId, i.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("警告: 群 %v 消息语音上传失败: %v", groupId, err)
|
log.Warnf("警告: 群 %v 消息语音上传失败: %v", groupId, err)
|
||||||
continue
|
continue
|
||||||
@ -108,10 +138,92 @@ func (bot *CQBot) SendGroupMessage(groupId int64, m *message.SendingMessage) int
|
|||||||
newElem = append(newElem, gv)
|
newElem = append(newElem, gv)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if i, ok := elem.(*PokeElement); ok {
|
||||||
|
if group := bot.Client.FindGroup(groupId); group != nil {
|
||||||
|
if mem := group.FindMember(i.Target); mem != nil {
|
||||||
|
mem.Poke()
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*GiftElement); ok {
|
||||||
|
bot.Client.SendGroupGift(uint64(groupId), uint64(i.Target), i.GiftId)
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*QQMusicElement); ok {
|
||||||
|
var msgStyle uint32 = 4
|
||||||
|
if i.MusicUrl == "" {
|
||||||
|
msgStyle = 0 // fix vip song
|
||||||
|
}
|
||||||
|
ret, err := bot.Client.SendGroupRichMessage(groupId, 100497308, 1, msgStyle, client.RichClientInfo{
|
||||||
|
Platform: 1,
|
||||||
|
SdkVersion: "0.0.0",
|
||||||
|
PackageName: "com.tencent.qqmusic",
|
||||||
|
Signature: "cbd27cd7c861227d013a25b2d10f0799",
|
||||||
|
}, &message.RichMessage{
|
||||||
|
Title: i.Title,
|
||||||
|
Summary: i.Summary,
|
||||||
|
Url: i.Url,
|
||||||
|
PictureUrl: i.PictureUrl,
|
||||||
|
MusicUrl: i.MusicUrl,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("警告: 群 %v 富文本消息发送失败: %v", groupId, err)
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return bot.InsertGroupMessage(ret)
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*CloudMusicElement); ok {
|
||||||
|
ret, err := bot.Client.SendGroupRichMessage(groupId, 100495085, 1, 4, client.RichClientInfo{
|
||||||
|
Platform: 1,
|
||||||
|
SdkVersion: "0.0.0",
|
||||||
|
PackageName: "com.netease.cloudmusic",
|
||||||
|
Signature: "da6b069da1e2982db3e386233f68d76d",
|
||||||
|
}, &message.RichMessage{
|
||||||
|
Title: i.Title,
|
||||||
|
Summary: i.Summary,
|
||||||
|
Url: i.Url,
|
||||||
|
PictureUrl: i.PictureUrl,
|
||||||
|
MusicUrl: i.MusicUrl,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("警告: 群 %v 富文本消息发送失败: %v", groupId, err)
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return bot.InsertGroupMessage(ret)
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*MiguMusicElement); ok {
|
||||||
|
ret, err := bot.Client.SendGroupRichMessage(groupId, 1101053067, 1, 4, client.RichClientInfo{
|
||||||
|
Platform: 1,
|
||||||
|
SdkVersion: "0.0.0",
|
||||||
|
PackageName: "cmccwm.mobilemusic",
|
||||||
|
Signature: "6cdc72a439cef99a3418d2a78aa28c73",
|
||||||
|
}, &message.RichMessage{
|
||||||
|
Title: i.Title,
|
||||||
|
Summary: i.Summary,
|
||||||
|
Url: i.Url,
|
||||||
|
PictureUrl: i.PictureUrl,
|
||||||
|
MusicUrl: i.MusicUrl,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("警告: 群 %v 富文本消息发送失败: %v", groupId, err)
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return bot.InsertGroupMessage(ret)
|
||||||
|
}
|
||||||
newElem = append(newElem, elem)
|
newElem = append(newElem, elem)
|
||||||
}
|
}
|
||||||
|
if len(newElem) == 0 {
|
||||||
|
log.Warnf("群消息发送失败: 消息为空.")
|
||||||
|
return -1
|
||||||
|
}
|
||||||
m.Elements = newElem
|
m.Elements = newElem
|
||||||
ret := bot.Client.SendGroupMessage(groupId, m)
|
bot.checkMedia(newElem)
|
||||||
|
ret := bot.Client.SendGroupMessage(groupId, m, ForceFragmented)
|
||||||
|
if ret == nil || ret.Id == -1 {
|
||||||
|
log.Warnf("群消息发送失败: 账号可能被风控.")
|
||||||
|
return -1
|
||||||
|
}
|
||||||
return bot.InsertGroupMessage(ret)
|
return bot.InsertGroupMessage(ret)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -121,17 +233,103 @@ func (bot *CQBot) SendPrivateMessage(target int64, m *message.SendingMessage) in
|
|||||||
if i, ok := elem.(*message.ImageElement); ok {
|
if i, ok := elem.(*message.ImageElement); ok {
|
||||||
fm, err := bot.Client.UploadPrivateImage(target, i.Data)
|
fm, err := bot.Client.UploadPrivateImage(target, i.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("警告: 好友 %v 消息图片上传失败.", target)
|
log.Warnf("警告: 私聊 %v 消息图片上传失败.", target)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
newElem = append(newElem, fm)
|
newElem = append(newElem, fm)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if i, ok := elem.(*PokeElement); ok {
|
||||||
|
bot.Client.SendFriendPoke(i.Target)
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*message.VoiceElement); ok {
|
||||||
|
fv, err := bot.Client.UploadPrivatePtt(target, i.Data)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("警告: 好友 %v 消息语音上传失败: %v", target, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
newElem = append(newElem, fv)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*QQMusicElement); ok {
|
||||||
|
var msgStyle uint32 = 4
|
||||||
|
if i.MusicUrl == "" {
|
||||||
|
msgStyle = 0 // fix vip song
|
||||||
|
}
|
||||||
|
bot.Client.SendFriendRichMessage(target, 100497308, 1, msgStyle, client.RichClientInfo{
|
||||||
|
Platform: 1,
|
||||||
|
SdkVersion: "0.0.0",
|
||||||
|
PackageName: "com.tencent.qqmusic",
|
||||||
|
Signature: "cbd27cd7c861227d013a25b2d10f0799",
|
||||||
|
}, &message.RichMessage{
|
||||||
|
Title: i.Title,
|
||||||
|
Summary: i.Summary,
|
||||||
|
Url: i.Url,
|
||||||
|
PictureUrl: i.PictureUrl,
|
||||||
|
MusicUrl: i.MusicUrl,
|
||||||
|
})
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*CloudMusicElement); ok {
|
||||||
|
bot.Client.SendFriendRichMessage(target, 100495085, 1, 4, client.RichClientInfo{
|
||||||
|
Platform: 1,
|
||||||
|
SdkVersion: "0.0.0",
|
||||||
|
PackageName: "com.netease.cloudmusic",
|
||||||
|
Signature: "da6b069da1e2982db3e386233f68d76d",
|
||||||
|
}, &message.RichMessage{
|
||||||
|
Title: i.Title,
|
||||||
|
Summary: i.Summary,
|
||||||
|
Url: i.Url,
|
||||||
|
PictureUrl: i.PictureUrl,
|
||||||
|
MusicUrl: i.MusicUrl,
|
||||||
|
})
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*MiguMusicElement); ok {
|
||||||
|
bot.Client.SendFriendRichMessage(target, 1101053067, 1, 4, client.RichClientInfo{
|
||||||
|
Platform: 1,
|
||||||
|
SdkVersion: "0.0.0",
|
||||||
|
PackageName: "cmccwm.mobilemusic",
|
||||||
|
Signature: "6cdc72a439cef99a3418d2a78aa28c73",
|
||||||
|
}, &message.RichMessage{
|
||||||
|
Title: i.Title,
|
||||||
|
Summary: i.Summary,
|
||||||
|
Url: i.Url,
|
||||||
|
PictureUrl: i.PictureUrl,
|
||||||
|
MusicUrl: i.MusicUrl,
|
||||||
|
})
|
||||||
|
return 0
|
||||||
|
}
|
||||||
newElem = append(newElem, elem)
|
newElem = append(newElem, elem)
|
||||||
}
|
}
|
||||||
|
if len(newElem) == 0 {
|
||||||
|
log.Warnf("好友消息发送失败: 消息为空.")
|
||||||
|
return -1
|
||||||
|
}
|
||||||
m.Elements = newElem
|
m.Elements = newElem
|
||||||
ret := bot.Client.SendPrivateMessage(target, m)
|
bot.checkMedia(newElem)
|
||||||
return ToGlobalId(target, ret.Id)
|
var id int32 = -1
|
||||||
|
if bot.Client.FindFriend(target) != nil { // 双向好友
|
||||||
|
msg := bot.Client.SendPrivateMessage(target, m)
|
||||||
|
if msg != nil {
|
||||||
|
id = bot.InsertPrivateMessage(msg)
|
||||||
|
}
|
||||||
|
} else if code, ok := bot.tempMsgCache.Load(target); ok { // 临时会话
|
||||||
|
msg := bot.Client.SendTempMessage(code.(int64), target, m)
|
||||||
|
if msg != nil {
|
||||||
|
id = msg.Id
|
||||||
|
}
|
||||||
|
} else if _, ok := bot.oneWayMsgCache.Load(target); ok { // 单向好友
|
||||||
|
msg := bot.Client.SendPrivateMessage(target, m)
|
||||||
|
if msg != nil {
|
||||||
|
id = bot.InsertPrivateMessage(msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if id == -1 {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return id
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
||||||
@ -146,14 +344,36 @@ func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
|||||||
}
|
}
|
||||||
id := ToGlobalId(m.GroupCode, m.Id)
|
id := ToGlobalId(m.GroupCode, m.Id)
|
||||||
if bot.db != nil {
|
if bot.db != nil {
|
||||||
err := bot.db.Update(func(tx *nutsdb.Tx) error {
|
buf := new(bytes.Buffer)
|
||||||
buf := new(bytes.Buffer)
|
if err := gob.NewEncoder(buf).Encode(val); err != nil {
|
||||||
if err := gob.NewEncoder(buf).Encode(val); err != nil {
|
log.Warnf("记录聊天数据时出现错误: %v", err)
|
||||||
return err
|
return -1
|
||||||
}
|
}
|
||||||
return tx.Put("group-messages", binary.ToBytes(id), binary.GZipCompress(buf.Bytes()), 0)
|
if err := bot.db.Put(binary.ToBytes(id), binary.GZipCompress(buf.Bytes()), nil); err != nil {
|
||||||
})
|
log.Warnf("记录聊天数据时出现错误: %v", err)
|
||||||
if err != nil {
|
return -1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return id
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 {
|
||||||
|
val := MSG{
|
||||||
|
"message-id": m.Id,
|
||||||
|
"internal-id": m.InternalId,
|
||||||
|
"target": m.Target,
|
||||||
|
"sender": m.Sender,
|
||||||
|
"time": m.Time,
|
||||||
|
"message": ToStringMessage(m.Elements, m.Sender.Uin, true),
|
||||||
|
}
|
||||||
|
id := ToGlobalId(m.Sender.Uin, m.Id)
|
||||||
|
if bot.db != nil {
|
||||||
|
buf := new(bytes.Buffer)
|
||||||
|
if err := gob.NewEncoder(buf).Encode(val); err != nil {
|
||||||
|
log.Warnf("记录聊天数据时出现错误: %v", err)
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
if err := bot.db.Put(binary.ToBytes(id), binary.GZipCompress(buf.Bytes()), nil); err != nil {
|
||||||
log.Warnf("记录聊天数据时出现错误: %v", err)
|
log.Warnf("记录聊天数据时出现错误: %v", err)
|
||||||
return -1
|
return -1
|
||||||
}
|
}
|
||||||
@ -172,8 +392,22 @@ func (bot *CQBot) Release() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) dispatchEventMessage(m MSG) {
|
func (bot *CQBot) dispatchEventMessage(m MSG) {
|
||||||
|
payload := gjson.Parse(m.ToJson())
|
||||||
|
filter := global.EventFilter
|
||||||
|
if filter != nil && (*filter).Eval(payload) == false {
|
||||||
|
log.Debug("Event filtered!")
|
||||||
|
return
|
||||||
|
}
|
||||||
for _, f := range bot.events {
|
for _, f := range bot.events {
|
||||||
f(m)
|
fn := f
|
||||||
|
go func() {
|
||||||
|
start := time.Now()
|
||||||
|
fn(m)
|
||||||
|
end := time.Now()
|
||||||
|
if end.Sub(start) > time.Second*5 {
|
||||||
|
log.Debugf("警告: 事件处理耗时超过 5 秒 (%v), 请检查应用是否有堵塞.", end.Sub(start))
|
||||||
|
}
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -182,6 +416,9 @@ func formatGroupName(group *client.GroupInfo) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func formatMemberName(mem *client.GroupMemberInfo) string {
|
func formatMemberName(mem *client.GroupMemberInfo) string {
|
||||||
|
if mem == nil {
|
||||||
|
return "未知"
|
||||||
|
}
|
||||||
return fmt.Sprintf("%s(%d)", mem.DisplayName(), mem.Uin)
|
return fmt.Sprintf("%s(%d)", mem.DisplayName(), mem.Uin)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
915
coolq/cqcode.go
915
coolq/cqcode.go
File diff suppressed because it is too large
Load Diff
287
coolq/event.go
287
coolq/event.go
@ -2,28 +2,52 @@ package coolq
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
|
"io/ioutil"
|
||||||
|
"path"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/Mrs4s/MiraiGo/binary"
|
"github.com/Mrs4s/MiraiGo/binary"
|
||||||
"github.com/Mrs4s/MiraiGo/client"
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
"github.com/Mrs4s/MiraiGo/message"
|
"github.com/Mrs4s/MiraiGo/message"
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"io/ioutil"
|
|
||||||
"path"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var format = "string"
|
||||||
|
|
||||||
|
func SetMessageFormat(f string) {
|
||||||
|
format = f
|
||||||
|
}
|
||||||
|
|
||||||
|
func ToFormattedMessage(e []message.IMessageElement, code int64, raw ...bool) (r interface{}) {
|
||||||
|
if format == "string" {
|
||||||
|
r = ToStringMessage(e, code, raw...)
|
||||||
|
} else if format == "array" {
|
||||||
|
r = ToArrayMessage(e, code, raw...)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) privateMessageEvent(c *client.QQClient, m *message.PrivateMessage) {
|
func (bot *CQBot) privateMessageEvent(c *client.QQClient, m *message.PrivateMessage) {
|
||||||
checkImage(m.Elements)
|
bot.checkMedia(m.Elements)
|
||||||
cqm := ToStringMessage(m.Elements, 0, true)
|
cqm := ToStringMessage(m.Elements, m.Sender.Uin, true)
|
||||||
log.Infof("收到好友 %v(%v) 的消息: %v", m.Sender.DisplayName(), m.Sender.Uin, cqm)
|
if !m.Sender.IsFriend {
|
||||||
|
bot.oneWayMsgCache.Store(m.Sender.Uin, "")
|
||||||
|
}
|
||||||
|
id := m.Id
|
||||||
|
if bot.db != nil {
|
||||||
|
id = bot.InsertPrivateMessage(m)
|
||||||
|
}
|
||||||
|
log.Infof("收到好友 %v(%v) 的消息: %v (%v)", m.Sender.DisplayName(), m.Sender.Uin, cqm, id)
|
||||||
fm := MSG{
|
fm := MSG{
|
||||||
"post_type": "message",
|
"post_type": "message",
|
||||||
"message_type": "private",
|
"message_type": "private",
|
||||||
"sub_type": "friend",
|
"sub_type": "friend",
|
||||||
"message_id": ToGlobalId(m.Sender.Uin, m.Id),
|
"message_id": id,
|
||||||
"user_id": m.Sender.Uin,
|
"user_id": m.Sender.Uin,
|
||||||
"message": ToStringMessage(m.Elements, 0, false),
|
"message": ToFormattedMessage(m.Elements, m.Sender.Uin, false),
|
||||||
"raw_message": cqm,
|
"raw_message": cqm,
|
||||||
"font": 0,
|
"font": 0,
|
||||||
"self_id": c.Uin,
|
"self_id": c.Uin,
|
||||||
@ -39,7 +63,7 @@ func (bot *CQBot) privateMessageEvent(c *client.QQClient, m *message.PrivateMess
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) groupMessageEvent(c *client.QQClient, m *message.GroupMessage) {
|
func (bot *CQBot) groupMessageEvent(c *client.QQClient, m *message.GroupMessage) {
|
||||||
checkImage(m.Elements)
|
bot.checkMedia(m.Elements)
|
||||||
for _, elem := range m.Elements {
|
for _, elem := range m.Elements {
|
||||||
if file, ok := elem.(*message.GroupFileElement); ok {
|
if file, ok := elem.(*message.GroupFileElement); ok {
|
||||||
log.Infof("群 %v(%v) 内 %v(%v) 上传了文件: %v", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, file.Name)
|
log.Infof("群 %v(%v) 内 %v(%v) 上传了文件: %v", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, file.Name)
|
||||||
@ -71,7 +95,7 @@ func (bot *CQBot) groupMessageEvent(c *client.QQClient, m *message.GroupMessage)
|
|||||||
"anonymous": nil,
|
"anonymous": nil,
|
||||||
"font": 0,
|
"font": 0,
|
||||||
"group_id": m.GroupCode,
|
"group_id": m.GroupCode,
|
||||||
"message": ToStringMessage(m.Elements, m.GroupCode, false),
|
"message": ToFormattedMessage(m.Elements, m.GroupCode, false),
|
||||||
"message_id": id,
|
"message_id": id,
|
||||||
"message_type": "group",
|
"message_type": "group",
|
||||||
"post_type": "message",
|
"post_type": "message",
|
||||||
@ -117,8 +141,9 @@ func (bot *CQBot) groupMessageEvent(c *client.QQClient, m *message.GroupMessage)
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) tempMessageEvent(c *client.QQClient, m *message.TempMessage) {
|
func (bot *CQBot) tempMessageEvent(c *client.QQClient, m *message.TempMessage) {
|
||||||
checkImage(m.Elements)
|
bot.checkMedia(m.Elements)
|
||||||
cqm := ToStringMessage(m.Elements, 0, true)
|
cqm := ToStringMessage(m.Elements, 0, true)
|
||||||
|
bot.tempMsgCache.Store(m.Sender.Uin, m.GroupCode)
|
||||||
log.Infof("收到来自群 %v(%v) 内 %v(%v) 的临时会话消息: %v", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, cqm)
|
log.Infof("收到来自群 %v(%v) 内 %v(%v) 的临时会话消息: %v", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, cqm)
|
||||||
tm := MSG{
|
tm := MSG{
|
||||||
"post_type": "message",
|
"post_type": "message",
|
||||||
@ -126,7 +151,7 @@ func (bot *CQBot) tempMessageEvent(c *client.QQClient, m *message.TempMessage) {
|
|||||||
"sub_type": "group",
|
"sub_type": "group",
|
||||||
"message_id": m.Id,
|
"message_id": m.Id,
|
||||||
"user_id": m.Sender.Uin,
|
"user_id": m.Sender.Uin,
|
||||||
"message": ToStringMessage(m.Elements, 0, false),
|
"message": ToFormattedMessage(m.Elements, 0, false),
|
||||||
"raw_message": cqm,
|
"raw_message": cqm,
|
||||||
"font": 0,
|
"font": 0,
|
||||||
"self_id": c.Uin,
|
"self_id": c.Uin,
|
||||||
@ -143,13 +168,24 @@ func (bot *CQBot) tempMessageEvent(c *client.QQClient, m *message.TempMessage) {
|
|||||||
|
|
||||||
func (bot *CQBot) groupMutedEvent(c *client.QQClient, e *client.GroupMuteEvent) {
|
func (bot *CQBot) groupMutedEvent(c *client.QQClient, e *client.GroupMuteEvent) {
|
||||||
g := c.FindGroup(e.GroupCode)
|
g := c.FindGroup(e.GroupCode)
|
||||||
if e.Time > 0 {
|
if e.TargetUin == 0 {
|
||||||
log.Infof("群 %v 内 %v 被 %v 禁言了 %v秒.",
|
if e.Time != 0 {
|
||||||
formatGroupName(g), formatMemberName(g.FindMember(e.TargetUin)), formatMemberName(g.FindMember(e.OperatorUin)), e.Time)
|
log.Infof("群 %v 被 %v 开启全员禁言.",
|
||||||
|
formatGroupName(g), formatMemberName(g.FindMember(e.OperatorUin)))
|
||||||
|
} else {
|
||||||
|
log.Infof("群 %v 被 %v 解除全员禁言.",
|
||||||
|
formatGroupName(g), formatMemberName(g.FindMember(e.OperatorUin)))
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
log.Infof("群 %v 内 %v 被 %v 解除禁言.",
|
if e.Time > 0 {
|
||||||
formatGroupName(g), formatMemberName(g.FindMember(e.TargetUin)), formatMemberName(g.FindMember(e.OperatorUin)))
|
log.Infof("群 %v 内 %v 被 %v 禁言了 %v 秒.",
|
||||||
|
formatGroupName(g), formatMemberName(g.FindMember(e.TargetUin)), formatMemberName(g.FindMember(e.OperatorUin)), e.Time)
|
||||||
|
} else {
|
||||||
|
log.Infof("群 %v 内 %v 被 %v 解除禁言.",
|
||||||
|
formatGroupName(g), formatMemberName(g.FindMember(e.TargetUin)), formatMemberName(g.FindMember(e.OperatorUin)))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bot.dispatchEventMessage(MSG{
|
bot.dispatchEventMessage(MSG{
|
||||||
"post_type": "notice",
|
"post_type": "notice",
|
||||||
"duration": e.Time,
|
"duration": e.Time,
|
||||||
@ -160,10 +196,10 @@ func (bot *CQBot) groupMutedEvent(c *client.QQClient, e *client.GroupMuteEvent)
|
|||||||
"user_id": e.TargetUin,
|
"user_id": e.TargetUin,
|
||||||
"time": time.Now().Unix(),
|
"time": time.Now().Unix(),
|
||||||
"sub_type": func() string {
|
"sub_type": func() string {
|
||||||
if e.Time > 0 {
|
if e.Time == 0 {
|
||||||
return "ban"
|
return "lift_ban"
|
||||||
}
|
}
|
||||||
return "lift_ban"
|
return "ban"
|
||||||
}(),
|
}(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -185,20 +221,121 @@ func (bot *CQBot) groupRecallEvent(c *client.QQClient, e *client.GroupMessageRec
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) groupNotifyEvent(c *client.QQClient, e client.INotifyEvent) {
|
||||||
|
group := c.FindGroup(e.From())
|
||||||
|
switch notify := e.(type) {
|
||||||
|
case *client.GroupPokeNotifyEvent:
|
||||||
|
sender := group.FindMember(notify.Sender)
|
||||||
|
receiver := group.FindMember(notify.Receiver)
|
||||||
|
log.Infof("群 %v 内 %v 戳了戳 %v", formatGroupName(group), formatMemberName(sender), formatMemberName(receiver))
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"group_id": group.Code,
|
||||||
|
"notice_type": "notify",
|
||||||
|
"sub_type": "poke",
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"user_id": notify.Sender,
|
||||||
|
"sender_id": notify.Sender,
|
||||||
|
"target_id": notify.Receiver,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
})
|
||||||
|
case *client.GroupRedBagLuckyKingNotifyEvent:
|
||||||
|
sender := group.FindMember(notify.Sender)
|
||||||
|
luckyKing := group.FindMember(notify.LuckyKing)
|
||||||
|
log.Infof("群 %v 内 %v 的红包被抢完, %v 是运气王", formatGroupName(group), formatMemberName(sender), formatMemberName(luckyKing))
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"group_id": group.Code,
|
||||||
|
"notice_type": "notify",
|
||||||
|
"sub_type": "lucky_king",
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"user_id": notify.Sender,
|
||||||
|
"sender_id": notify.Sender,
|
||||||
|
"target_id": notify.LuckyKing,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
})
|
||||||
|
case *client.MemberHonorChangedNotifyEvent:
|
||||||
|
log.Info(notify.Content())
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"group_id": group.Code,
|
||||||
|
"notice_type": "notify",
|
||||||
|
"sub_type": "honor",
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"user_id": notify.Uin,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
"honor_type": func() string {
|
||||||
|
switch notify.Honor {
|
||||||
|
case client.Talkative:
|
||||||
|
return "talkative"
|
||||||
|
case client.Performer:
|
||||||
|
return "performer"
|
||||||
|
case client.Emotion:
|
||||||
|
return "emotion"
|
||||||
|
default:
|
||||||
|
return "ERROR"
|
||||||
|
}
|
||||||
|
}(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) friendNotifyEvent(c *client.QQClient, e client.INotifyEvent) {
|
||||||
|
friend := c.FindFriend(e.From())
|
||||||
|
switch notify := e.(type) {
|
||||||
|
case *client.FriendPokeNotifyEvent:
|
||||||
|
log.Infof("好友 %v 戳了戳你.", friend.Nickname)
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"notice_type": "notify",
|
||||||
|
"sub_type": "poke",
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"user_id": notify.Sender,
|
||||||
|
"sender_id": notify.Sender,
|
||||||
|
"target_id": notify.Receiver,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) friendRecallEvent(c *client.QQClient, e *client.FriendMessageRecalledEvent) {
|
func (bot *CQBot) friendRecallEvent(c *client.QQClient, e *client.FriendMessageRecalledEvent) {
|
||||||
f := c.FindFriend(e.FriendUin)
|
f := c.FindFriend(e.FriendUin)
|
||||||
gid := ToGlobalId(e.FriendUin, e.MessageId)
|
gid := ToGlobalId(e.FriendUin, e.MessageId)
|
||||||
log.Infof("好友 %v(%v) 撤回了消息: %v", f.Nickname, f.Uin, gid)
|
if f != nil {
|
||||||
|
log.Infof("好友 %v(%v) 撤回了消息: %v", f.Nickname, f.Uin, gid)
|
||||||
|
} else {
|
||||||
|
log.Infof("好友 %v 撤回了消息: %v", e.FriendUin, gid)
|
||||||
|
}
|
||||||
bot.dispatchEventMessage(MSG{
|
bot.dispatchEventMessage(MSG{
|
||||||
"post_type": "notice",
|
"post_type": "notice",
|
||||||
"notice_type": "friend_recall",
|
"notice_type": "friend_recall",
|
||||||
"self_id": c.Uin,
|
"self_id": c.Uin,
|
||||||
"user_id": f.Uin,
|
"user_id": e.FriendUin,
|
||||||
"time": e.Time,
|
"time": e.Time,
|
||||||
"message_id": gid,
|
"message_id": gid,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) offlineFileEvent(c *client.QQClient, e *client.OfflineFileEvent) {
|
||||||
|
f := c.FindFriend(e.Sender)
|
||||||
|
if f == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Infof("好友 %v(%v) 发送了离线文件 %v", f.Nickname, f.Uin, e.FileName)
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"notice_type": "offline_file",
|
||||||
|
"user_id": e.Sender,
|
||||||
|
"file": MSG{
|
||||||
|
"name": e.FileName,
|
||||||
|
"size": e.FileSize,
|
||||||
|
"url": e.DownloadUrl,
|
||||||
|
},
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) joinGroupEvent(c *client.QQClient, group *client.GroupInfo) {
|
func (bot *CQBot) joinGroupEvent(c *client.QQClient, group *client.GroupInfo) {
|
||||||
log.Infof("Bot进入了群 %v.", formatGroupName(group))
|
log.Infof("Bot进入了群 %v.", formatGroupName(group))
|
||||||
bot.dispatchEventMessage(bot.groupIncrease(group.Code, 0, c.Uin))
|
bot.dispatchEventMessage(bot.groupIncrease(group.Code, 0, c.Uin))
|
||||||
@ -231,6 +368,20 @@ func (bot *CQBot) memberPermissionChangedEvent(c *client.QQClient, e *client.Mem
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) memberCardUpdatedEvent(c *client.QQClient, e *client.MemberCardUpdatedEvent) {
|
||||||
|
log.Infof("群 %v 的 %v 更新了名片 %v -> %v", formatGroupName(e.Group), formatMemberName(e.Member), e.OldCard, e.Member.CardName)
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"notice_type": "group_card",
|
||||||
|
"group_id": e.Group.Code,
|
||||||
|
"user_id": e.Member.Uin,
|
||||||
|
"card_new": e.Member.CardName,
|
||||||
|
"card_old": e.OldCard,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
"self_id": c.Uin,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) memberJoinEvent(c *client.QQClient, e *client.MemberJoinGroupEvent) {
|
func (bot *CQBot) memberJoinEvent(c *client.QQClient, e *client.MemberJoinGroupEvent) {
|
||||||
log.Infof("新成员 %v 进入了群 %v.", formatMemberName(e.Member), formatGroupName(e.Group))
|
log.Infof("新成员 %v 进入了群 %v.", formatMemberName(e.Member), formatGroupName(e.Group))
|
||||||
bot.dispatchEventMessage(bot.groupIncrease(e.Group.Code, 0, e.Member.Uin))
|
bot.dispatchEventMessage(bot.groupIncrease(e.Group.Code, 0, e.Member.Uin))
|
||||||
@ -260,10 +411,21 @@ func (bot *CQBot) friendRequestEvent(c *client.QQClient, e *client.NewFriendRequ
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) friendAddedEvent(c *client.QQClient, e *client.NewFriendEvent) {
|
||||||
|
log.Infof("添加了新好友: %v(%v)", e.Friend.Nickname, e.Friend.Uin)
|
||||||
|
bot.tempMsgCache.Delete(e.Friend.Uin)
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"notice_type": "friend_add",
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"user_id": e.Friend.Uin,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) groupInvitedEvent(c *client.QQClient, e *client.GroupInvitedRequest) {
|
func (bot *CQBot) groupInvitedEvent(c *client.QQClient, e *client.GroupInvitedRequest) {
|
||||||
log.Infof("收到来自群 %v(%v) 内用户 %v(%v) 的加群邀请.", e.GroupName, e.GroupCode, e.InvitorNick, e.InvitorUin)
|
log.Infof("收到来自群 %v(%v) 内用户 %v(%v) 的加群邀请.", e.GroupName, e.GroupCode, e.InvitorNick, e.InvitorUin)
|
||||||
flag := strconv.FormatInt(e.RequestId, 10)
|
flag := strconv.FormatInt(e.RequestId, 10)
|
||||||
bot.invitedReqCache.Store(flag, e)
|
|
||||||
bot.dispatchEventMessage(MSG{
|
bot.dispatchEventMessage(MSG{
|
||||||
"post_type": "request",
|
"post_type": "request",
|
||||||
"request_type": "group",
|
"request_type": "group",
|
||||||
@ -278,16 +440,15 @@ func (bot *CQBot) groupInvitedEvent(c *client.QQClient, e *client.GroupInvitedRe
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) groupJoinReqEvent(c *client.QQClient, e *client.UserJoinGroupRequest) {
|
func (bot *CQBot) groupJoinReqEvent(c *client.QQClient, e *client.UserJoinGroupRequest) {
|
||||||
log.Infof("群 %v(%v) 收到来自用户 %v(%v) 的加群请求.", e.GroupName, e.GroupName, e.RequesterNick, e.RequesterUin)
|
log.Infof("群 %v(%v) 收到来自用户 %v(%v) 的加群请求.", e.GroupName, e.GroupCode, e.RequesterNick, e.RequesterUin)
|
||||||
flag := strconv.FormatInt(e.RequestId, 10)
|
flag := strconv.FormatInt(e.RequestId, 10)
|
||||||
bot.joinReqCache.Store(flag, e)
|
|
||||||
bot.dispatchEventMessage(MSG{
|
bot.dispatchEventMessage(MSG{
|
||||||
"post_type": "request",
|
"post_type": "request",
|
||||||
"request_type": "group",
|
"request_type": "group",
|
||||||
"sub_type": "add",
|
"sub_type": "add",
|
||||||
"group_id": e.GroupCode,
|
"group_id": e.GroupCode,
|
||||||
"user_id": e.RequesterUin,
|
"user_id": e.RequesterUin,
|
||||||
"comment": "",
|
"comment": e.Message,
|
||||||
"flag": flag,
|
"flag": flag,
|
||||||
"time": time.Now().Unix(),
|
"time": time.Now().Unix(),
|
||||||
"self_id": c.Uin,
|
"self_id": c.Uin,
|
||||||
@ -333,9 +494,10 @@ func (bot *CQBot) groupDecrease(groupCode, userUin int64, operator *client.Group
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkImage(e []message.IMessageElement) {
|
func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
||||||
for _, elem := range e {
|
for _, elem := range e {
|
||||||
if i, ok := elem.(*message.ImageElement); ok {
|
switch i := elem.(type) {
|
||||||
|
case *message.ImageElement:
|
||||||
filename := hex.EncodeToString(i.Md5) + ".image"
|
filename := hex.EncodeToString(i.Md5) + ".image"
|
||||||
if !global.PathExists(path.Join(global.IMAGE_PATH, filename)) {
|
if !global.PathExists(path.Join(global.IMAGE_PATH, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
@ -343,9 +505,74 @@ func checkImage(e []message.IMessageElement) {
|
|||||||
w.WriteUInt32(uint32(i.Size))
|
w.WriteUInt32(uint32(i.Size))
|
||||||
w.WriteString(i.Filename)
|
w.WriteString(i.Filename)
|
||||||
w.WriteString(i.Url)
|
w.WriteString(i.Url)
|
||||||
}), 0777)
|
}), 0644)
|
||||||
}
|
}
|
||||||
i.Filename = filename
|
i.Filename = filename
|
||||||
|
case *message.GroupImageElement:
|
||||||
|
filename := hex.EncodeToString(i.Md5) + ".image"
|
||||||
|
if !global.PathExists(path.Join(global.IMAGE_PATH, filename)) {
|
||||||
|
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
|
w.Write(i.Md5)
|
||||||
|
w.WriteUInt32(uint32(i.Size))
|
||||||
|
w.WriteString(filename)
|
||||||
|
w.WriteString(i.Url)
|
||||||
|
}), 0644)
|
||||||
|
}
|
||||||
|
case *message.FriendImageElement:
|
||||||
|
filename := hex.EncodeToString(i.Md5) + ".image"
|
||||||
|
if !global.PathExists(path.Join(global.IMAGE_PATH, filename)) {
|
||||||
|
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
|
w.Write(i.Md5)
|
||||||
|
w.WriteUInt32(uint32(0)) // 发送时会调用url, 大概没事
|
||||||
|
w.WriteString(filename)
|
||||||
|
w.WriteString(i.Url)
|
||||||
|
}), 0644)
|
||||||
|
}
|
||||||
|
case *message.GroupFlashImgElement:
|
||||||
|
filename := hex.EncodeToString(i.Md5) + ".image"
|
||||||
|
if !global.PathExists(path.Join(global.IMAGE_PATH, filename)) {
|
||||||
|
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
|
w.Write(i.Md5)
|
||||||
|
w.WriteUInt32(uint32(i.Size))
|
||||||
|
w.WriteString(i.Filename)
|
||||||
|
w.WriteString("")
|
||||||
|
}), 0644)
|
||||||
|
}
|
||||||
|
i.Filename = filename
|
||||||
|
case *message.FriendFlashImgElement:
|
||||||
|
filename := hex.EncodeToString(i.Md5) + ".image"
|
||||||
|
if !global.PathExists(path.Join(global.IMAGE_PATH, filename)) {
|
||||||
|
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
|
w.Write(i.Md5)
|
||||||
|
w.WriteUInt32(uint32(i.Size))
|
||||||
|
w.WriteString(i.Filename)
|
||||||
|
w.WriteString("")
|
||||||
|
}), 0644)
|
||||||
|
}
|
||||||
|
i.Filename = filename
|
||||||
|
case *message.VoiceElement:
|
||||||
|
i.Name = strings.ReplaceAll(i.Name, "{", "")
|
||||||
|
i.Name = strings.ReplaceAll(i.Name, "}", "")
|
||||||
|
if !global.PathExists(path.Join(global.VOICE_PATH, i.Name)) {
|
||||||
|
b, err := global.GetBytes(i.Url)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("语音文件 %v 下载失败: %v", i.Name, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
_ = ioutil.WriteFile(path.Join(global.VOICE_PATH, i.Name), b, 0644)
|
||||||
|
}
|
||||||
|
case *message.ShortVideoElement:
|
||||||
|
filename := hex.EncodeToString(i.Md5) + ".video"
|
||||||
|
if !global.PathExists(path.Join(global.VIDEO_PATH, filename)) {
|
||||||
|
_ = ioutil.WriteFile(path.Join(global.VIDEO_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
|
w.Write(i.Md5)
|
||||||
|
w.WriteUInt32(uint32(i.Size))
|
||||||
|
w.WriteString(i.Name)
|
||||||
|
w.Write(i.Uuid)
|
||||||
|
}), 0644)
|
||||||
|
}
|
||||||
|
i.Name = filename
|
||||||
|
i.Url = bot.Client.GetShortVideoUrl(i.Uuid, i.Md5)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
132
docs/EventFilter.md
Normal file
132
docs/EventFilter.md
Normal file
@ -0,0 +1,132 @@
|
|||||||
|
# 事件过滤器
|
||||||
|
|
||||||
|
在go-cqhttp同级目录下新建`filter.json`文件即可开启事件过滤器,启动时会读取该文件中定义的过滤规则(使用 JSON 编写),若文件不存在,或过滤规则语法错误,则会暂停所有上报。
|
||||||
|
事件过滤器会处理所有事件(包括心跳事件在内的元事件),请谨慎使用!!
|
||||||
|
|
||||||
|
## 示例
|
||||||
|
|
||||||
|
这节首先给出一些示例,演示过滤器的基本用法,下一节将给出具体语法说明。
|
||||||
|
|
||||||
|
### 只上报以「!!」开头的消息
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"raw_message": {
|
||||||
|
".regex": "^!!"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 只上报群组的非匿名消息
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"message_type": "group",
|
||||||
|
"anonymous": {
|
||||||
|
".eq": null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 只上报私聊或特定群组的非匿名消息
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
".or": [
|
||||||
|
{
|
||||||
|
"message_type": "private"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"message_type": "group",
|
||||||
|
"group_id": {
|
||||||
|
".in": [
|
||||||
|
123456
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"anonymous": {
|
||||||
|
".eq": null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 只上报群组 11111、22222、33333 中不是用户 12345 发送的消息,以及用户 66666 发送的所有消息
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
".or": [
|
||||||
|
{
|
||||||
|
"group_id": {
|
||||||
|
".in": [11111, 22222, 33333]
|
||||||
|
},
|
||||||
|
"user_id": {
|
||||||
|
".neq": 12345
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"user_id": 66666
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 一个更复杂的例子
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
".or": [
|
||||||
|
{
|
||||||
|
"message_type": "private",
|
||||||
|
"user_id": {
|
||||||
|
".not": {
|
||||||
|
".in": [11111, 22222, 33333]
|
||||||
|
},
|
||||||
|
".neq": 44444
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"message_type": {
|
||||||
|
".regex": "group|discuss"
|
||||||
|
},
|
||||||
|
".or": [
|
||||||
|
{
|
||||||
|
"group_id": 12345
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"raw_message": {
|
||||||
|
".contains": "通知"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 语法说明
|
||||||
|
|
||||||
|
过滤规则最外层是一个 JSON 对象,其中的键,如果以 `.`(点号)开头,则表示运算符,其值为运算符的参数,如果不以 `.` 开头,则表示对事件数据对象中相应键的过滤。过滤规则中任何一个对象,只有在它的所有项都匹配的情况下,才会让事件通过(等价于一个 `and` 运算);其中,不以 `.` 开头的键,若其值不是对象,则只有在这个值和事件数据相应值相等的情况下,才会通过(等价于一个 `eq` 运算符)。
|
||||||
|
|
||||||
|
下面列出所有运算符(「要求的参数类型」是指运算符的键所对应的值的类型,「可作用于的类型」是指在过滤时事件对象相应值的类型):
|
||||||
|
|
||||||
|
| 运算符 | 要求的参数类型 | 可作用于的类型 |
|
||||||
|
| ----------- | -------------------------- | ----------------------------------------------------- |
|
||||||
|
| `.not` | object | 任何 |
|
||||||
|
| `.and` | object | 若参数中全为运算符,则任何;若不全为运算符,则 object |
|
||||||
|
| `.or` | array(数组元素为 object) | 任何 |
|
||||||
|
| `.eq` | 任何 | 任何 |
|
||||||
|
| `.neq` | 任何 | 任何 |
|
||||||
|
| `.in` | string/array | 若参数为 string,则 string;若参数为 array,则任何 |
|
||||||
|
| `.contains` | string | string |
|
||||||
|
| `.regex` | string | string |
|
||||||
|
|
||||||
|
|
||||||
|
## 过滤时的事件数据对象
|
||||||
|
|
||||||
|
过滤器在go-cqhttp构建好事件数据后运行,各事件的数据字段见[OneBot标准]( https://github.com/howmanybots/onebot/blob/master/v11/specs/event/README.md )。
|
||||||
|
|
||||||
|
这里有几点需要注意:
|
||||||
|
|
||||||
|
- `message` 字段在运行过滤器时是消息段数组的形式(见 [消息格式]( https://github.com/howmanybots/onebot/blob/master/v11/specs/message/array.md ))
|
||||||
|
- `raw_message` 字段为未经**CQ码**处理的原始消息字符串,这意味着其中可能会出现形如 `[CQ:face,id=123]` 的 CQ 码
|
5
docs/QA.md
Normal file
5
docs/QA.md
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# 常见问题
|
||||||
|
|
||||||
|
### Q: 为什么挂一段时间后就会出现 `消息发送失败,账号可能被风控`?
|
||||||
|
|
||||||
|
### A: 如果你刚开始使用 go-cqhttp 建议挂机3-7天,即可解除风控
|
251
docs/adminApi.md
Normal file
251
docs/adminApi.md
Normal file
@ -0,0 +1,251 @@
|
|||||||
|
# 管理 API
|
||||||
|
|
||||||
|
> 支持跨域
|
||||||
|
|
||||||
|
## 公共参数
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------------ | ------ | --------------------------- |
|
||||||
|
| access_token | string | 校验口令,config.hjson中配置 |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## admin/do_restart
|
||||||
|
|
||||||
|
### 热重启
|
||||||
|
|
||||||
|
> 热重启
|
||||||
|
|
||||||
|
> ps: 目前不支持ws部分的修改生效
|
||||||
|
|
||||||
|
method:`POST/GET`
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ---- | ---- |
|
||||||
|
| 无 | | |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### admin/get_web_write
|
||||||
|
|
||||||
|
> 拉取验证码/设备锁
|
||||||
|
|
||||||
|
method: `GET`
|
||||||
|
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ---- | ---- |
|
||||||
|
| 无 | | |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {"ispic": true,"picbase64":"xxxxx"}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| -------- | ------ | --------------------------------------------------- |
|
||||||
|
| ispic | bool | 是否是验证码类型 true是,false为不是(比如设备锁 |
|
||||||
|
| picbas64 | string | 验证码的base64编码内容,加上头,放入img标签即可显示 |
|
||||||
|
|
||||||
|
### admin/do_web_write
|
||||||
|
|
||||||
|
> web输入验证码/设备锁确认
|
||||||
|
|
||||||
|
method: `POST` formdata
|
||||||
|
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ---------- |
|
||||||
|
| input | string | 输入的内容 |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### admin/do_restart_docker
|
||||||
|
|
||||||
|
> 冷重启
|
||||||
|
|
||||||
|
> 注意:此api 会直接结束掉进程,需要依赖docker/supervisor等进程管理工具来自动拉起
|
||||||
|
|
||||||
|
method: `POST`
|
||||||
|
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ---- | ---- |
|
||||||
|
| 无 | | |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
### admin/do_process_restart
|
||||||
|
|
||||||
|
> 冷重启
|
||||||
|
|
||||||
|
method: `POST`
|
||||||
|
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ---- | ---- |
|
||||||
|
| 无 | | |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
### admin/do_config_base
|
||||||
|
|
||||||
|
> 基础配置
|
||||||
|
|
||||||
|
method: `POST` formdata
|
||||||
|
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------------ | ------ | ------------------------------------- |
|
||||||
|
| uin | string | qq号 |
|
||||||
|
| password | string | qq密码 |
|
||||||
|
| enable_db | string | 是否启动数据库,填 'true' 或者 'false' |
|
||||||
|
| access_token | string | 授权 token |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### admin/do_config_http
|
||||||
|
|
||||||
|
> http服务配置
|
||||||
|
|
||||||
|
method: `POST` formdata
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ----------- | ------ | --------------------------------------------- |
|
||||||
|
| port | string | 服务端口 |
|
||||||
|
| host | string | 服务监听地址 |
|
||||||
|
| enable | string | 是否启用 ,填 'true' 或者 'false' |
|
||||||
|
| timeout | string | http请求超时时间 |
|
||||||
|
| post_url | string | post上报地址 不需要就填空字符串,或者不填 |
|
||||||
|
| post_secret | string | post上报的secret 不需要就填空字符串,或者不填 |
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### admin/do_config_ws
|
||||||
|
|
||||||
|
> 正向ws设置
|
||||||
|
|
||||||
|
method: `POST` formdata
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | -------------------------------- |
|
||||||
|
| port | string | 服务端口 |
|
||||||
|
| host | string | 服务监听地址 |
|
||||||
|
| enable | string | 是否启用 ,填 'true' 或者 'false' |
|
||||||
|
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
### admin/do_config_reverse
|
||||||
|
|
||||||
|
> 反向ws配置
|
||||||
|
|
||||||
|
method: `POST` formdata
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | -------------------------------- |
|
||||||
|
| port | string | 服务端口 |
|
||||||
|
| host | string | 服务监听地址 |
|
||||||
|
| enable | string | 是否启用 ,填 'true' 或者 'false' |
|
||||||
|
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
### admin/do_config_json
|
||||||
|
|
||||||
|
> 直接修改 config.hjson配置
|
||||||
|
|
||||||
|
method: `POST` formdata
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ----------------------------------- |
|
||||||
|
| json | string | 完整的config.hjson的配合,json字符串 |
|
||||||
|
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
### admin/get_config_json
|
||||||
|
|
||||||
|
> 获取当前 config.hjson配置
|
||||||
|
|
||||||
|
method: `GET`
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ---- | ---- |
|
||||||
|
| 无 | | |
|
||||||
|
|
||||||
|
|
||||||
|
返回:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{"data": {"config":"xxxx"}, "retcode": 0, "status": "ok"}
|
||||||
|
```
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ----------------------------------- |
|
||||||
|
| config | string | 完整的config.hjson的配合,json字符串 |
|
||||||
|
|
102
docs/config.md
102
docs/config.md
@ -1,6 +1,6 @@
|
|||||||
# 配置
|
# 配置
|
||||||
|
|
||||||
go-cqhttp 包含 `config.json` 和 `device.json` 两个配置文件, 其中 `config.json` 为运行配置 `device.json` 为虚拟设备信息.
|
go-cqhttp 包含 `config.hjson` 和 `device.json` 两个配置文件, 其中 `config.json` 为运行配置 `device.json` 为虚拟设备信息.
|
||||||
|
|
||||||
## 从原CQHTTP导入配置
|
## 从原CQHTTP导入配置
|
||||||
|
|
||||||
@ -18,15 +18,31 @@ go-cqhttp 支持导入CQHTTP的配置文件, 具体步骤为:
|
|||||||
{
|
{
|
||||||
"uin": 0,
|
"uin": 0,
|
||||||
"password": "",
|
"password": "",
|
||||||
|
"encrypt_password": false,
|
||||||
|
"password_encrypted": "",
|
||||||
"enable_db": true,
|
"enable_db": true,
|
||||||
"access_token": "",
|
"access_token": "",
|
||||||
"relogin": false,
|
"relogin": {
|
||||||
"relogin_delay": 0,
|
"enabled": true,
|
||||||
|
"relogin_delay": 3,
|
||||||
|
"max_relogin_times": 0
|
||||||
|
},
|
||||||
|
"_rate_limit": {
|
||||||
|
"enabled": false,
|
||||||
|
"frequency": 1,
|
||||||
|
"bucket_size": 1
|
||||||
|
},
|
||||||
|
"post_message_format": "string",
|
||||||
|
"ignore_invalid_cqcode": false,
|
||||||
|
"force_fragmented": true,
|
||||||
|
"heartbeat_interval": 5,
|
||||||
|
"use_sso_address": false,
|
||||||
"http_config": {
|
"http_config": {
|
||||||
"enabled": true,
|
"enabled": true,
|
||||||
"host": "0.0.0.0",
|
"host": "0.0.0.0",
|
||||||
"port": 5700,
|
"port": 5700,
|
||||||
"post_urls": []
|
"timeout": 5,
|
||||||
|
"post_urls": {"url:port": "secret"}
|
||||||
},
|
},
|
||||||
"ws_config": {
|
"ws_config": {
|
||||||
"enabled": true,
|
"enabled": true,
|
||||||
@ -45,16 +61,72 @@ go-cqhttp 支持导入CQHTTP的配置文件, 具体步骤为:
|
|||||||
}
|
}
|
||||||
````
|
````
|
||||||
|
|
||||||
| 字段 | 类型 | 说明 |
|
| 字段 | 类型 | 说明 |
|
||||||
| ------------------ | -------- | ------------------------------------------------------------------- |
|
| --------------------- | -------- | ---------------------------------------------------------------------------------------- |
|
||||||
| uin | int64 | 登录用QQ号 |
|
| uin | int64 | 登录用QQ号 |
|
||||||
| password | string | 登录用密码 |
|
| password | string | 登录用密码 |
|
||||||
| enable_db | bool | 是否开启内置数据库, 关闭后将无法使用 **回复/撤回** 等上下文相关接口 |
|
| encrypt_password | bool | 是否对密码进行加密. |
|
||||||
| access_token | string | 同CQHTTP的 `access_token` 用于身份验证 |
|
| password_encrypted | string | 加密后的密码(请勿修改) |
|
||||||
| relogin | bool | 是否自动重新登录 |
|
| enable_db | bool | 是否开启内置数据库, 关闭后将无法使用 **回复/撤回** 等上下文相关接口 |
|
||||||
| relogin_delay | int | 重登录延时(秒) |
|
| access_token | string | 同CQHTTP的 `access_token` 用于身份验证 |
|
||||||
| http_config | object | HTTP API配置 |
|
| relogin | bool | 是否自动重新登录 |
|
||||||
| ws_config | object | Websocket API 配置 |
|
| relogin_delay | int | 重登录延时(秒) |
|
||||||
| ws_reverse_servers | object[] | 反向 Websocket API 配置 |
|
| max_relogin_times | uint | 最大重登录次数,若0则不设置上限 |
|
||||||
|
| _rate_limit | bool | 是否启用API调用限速 |
|
||||||
|
| frequency | float64 | 1s内能调用API的次数 |
|
||||||
|
| bucket_size | int | 令牌桶的大小,默认为1,修改此值可允许一定程度内连续调用api |
|
||||||
|
| post_message_format | string | 上报信息类型 |
|
||||||
|
| ignore_invalid_cqcode | bool | 是否忽略错误的CQ码 |
|
||||||
|
| force_fragmented | bool | 是否强制分片发送群长消息 |
|
||||||
|
| use_sso_address | bool | 是否使用服务器下发的地址 |
|
||||||
|
| heartbeat_interval | int64 | 心跳间隔时间,单位秒。小于0则关闭心跳,等于0使用默认值(5秒) |
|
||||||
|
| http_config | object | HTTP API配置 |
|
||||||
|
| ws_config | object | Websocket API 配置 |
|
||||||
|
| ws_reverse_servers | object[] | 反向 Websocket API 配置 |
|
||||||
|
| log_level | string | 指定日志收集级别,将收集的日志单独存放到固定文件中,便于查看日志线索 当前支持 warn,error |
|
||||||
|
|
||||||
|
> 注: 开启密码加密后程序将在每次启动时要求输入解密密钥, 密钥错误会导致登录时提示密码错误.
|
||||||
|
> 解密后密码将储存在内存中,用于自动重连等功能. 所以此加密并不能防止内存读取.
|
||||||
|
> 解密密钥在使用完成后并不会留存在内存中, 所以可用相对简单的字符串作为密钥
|
||||||
|
|
||||||
|
> 注2: 分片发送为原酷Q发送长消息的老方案, 发送速度更优/兼容性更好,但在有发言频率限制的群里,可能无法发送。关闭后将优先使用新方案, 能发送更长的消息, 但发送速度更慢,在部分老客户端将无法解析.
|
||||||
|
|
||||||
|
> 注3:关闭心跳服务可能引起断线,请谨慎关闭
|
||||||
|
|
||||||
|
## 设备信息
|
||||||
|
|
||||||
|
默认生成的设备信息如下所示:
|
||||||
|
|
||||||
|
``` json
|
||||||
|
{
|
||||||
|
"protocol": 0,
|
||||||
|
"display": "xxx",
|
||||||
|
"finger_print": "xxx",
|
||||||
|
"boot_id": "xxx",
|
||||||
|
"proc_version": "xxx",
|
||||||
|
"imei": "xxx"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
在大部分情况下 我们只需要关心 `protocol` 字段:
|
||||||
|
|
||||||
|
| 值 | 类型 | 限制 |
|
||||||
|
| --- | ------------- | ---------------------------------------------------------------- |
|
||||||
|
| 0 | iPad | 无 |
|
||||||
|
| 1 | Android Phone | 无法接收新版表情如 `/吃瓜 /汪汪`, 会自动转换为字符串 |
|
||||||
|
| 2 | Android Watch | 除`Android Phone`的限制外, 无法接收 `notify` 事件、无法接收口令红包、无法接收撤回消息 |
|
||||||
|
| 3 | MacOS | 无 |
|
||||||
|
|
||||||
|
> 注意, 根据协议的不同, 各类消息有所限制
|
||||||
|
|
||||||
|
## 自定义服务器IP
|
||||||
|
|
||||||
|
> 某些海外服务器使用默认地址可能会存在链路问题,此功能可以指定 go-cqhttp 连接哪些地址以达到最优化.
|
||||||
|
|
||||||
|
将文件 `address.txt` 创建到 `go-cqhttp` 工作目录, 并键入 `IP:PORT` 以换行符为分割即可.
|
||||||
|
|
||||||
|
示例:
|
||||||
|
````
|
||||||
|
1.1.1.1:53
|
||||||
|
1.1.2.2:8899
|
||||||
|
````
|
569
docs/cqhttp.md
569
docs/cqhttp.md
@ -4,6 +4,28 @@
|
|||||||
|
|
||||||
## CQCode
|
## CQCode
|
||||||
|
|
||||||
|
### 图片
|
||||||
|
|
||||||
|
| 参数名 | 可能的值 | 说明 |
|
||||||
|
| ------- | --------------- | --------------------------------------------------------------- |
|
||||||
|
| `file` | - | 图片文件名 |
|
||||||
|
| `type` | `flash`,`show` | 图片类型,`flash` 表示闪照,`show` 表示秀图,默认普通图片 |
|
||||||
|
| `url` | - | 图片 URL |
|
||||||
|
| `cache` | `0` `1` | 只在通过网络 URL 发送时有效,表示是否使用已缓存的文件,默认 `1` |
|
||||||
|
| `id` | - | 发送秀图时的特效id,默认为40000 |
|
||||||
|
|
||||||
|
可用的特效ID:
|
||||||
|
|
||||||
|
| id | 类型 |
|
||||||
|
| ----- | ---- |
|
||||||
|
| 40000 | 普通 |
|
||||||
|
| 40001 | 幻影 |
|
||||||
|
| 40002 | 抖动 |
|
||||||
|
| 40003 | 生日 |
|
||||||
|
| 40004 | 爱你 |
|
||||||
|
| 40005 | 征友 |
|
||||||
|
|
||||||
|
|
||||||
### 回复
|
### 回复
|
||||||
|
|
||||||
Type : `reply`
|
Type : `reply`
|
||||||
@ -18,6 +40,74 @@ Type : `reply`
|
|||||||
|
|
||||||
示例: `[CQ:reply,id=123456]`
|
示例: `[CQ:reply,id=123456]`
|
||||||
|
|
||||||
|
### 红包
|
||||||
|
|
||||||
|
Type: `redbag`
|
||||||
|
|
||||||
|
范围: **接收**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ----------- |
|
||||||
|
| title | string | 祝福语/口令 |
|
||||||
|
|
||||||
|
示例: `[CQ:redbag,title=恭喜发财]`
|
||||||
|
|
||||||
|
### 戳一戳
|
||||||
|
|
||||||
|
> 注意:发送戳一戳消息无法撤回,返回的 `message id` 恒定为 `0`
|
||||||
|
|
||||||
|
Type: `poke`
|
||||||
|
|
||||||
|
范围: **发送(仅群聊)**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ----- | ------------ |
|
||||||
|
| qq | int64 | 需要戳的成员 |
|
||||||
|
|
||||||
|
示例: `[CQ:poke,qq=123456]`
|
||||||
|
|
||||||
|
### 礼物
|
||||||
|
|
||||||
|
> 注意:仅支持免费礼物,发送群礼物消息无法撤回,返回的 `message id` 恒定为 `0`
|
||||||
|
|
||||||
|
Type: `gift`
|
||||||
|
|
||||||
|
范围: **发送(仅群聊,接收的时候不是CQ码)**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ----- | -------------- |
|
||||||
|
| qq | int64 | 接收礼物的成员 |
|
||||||
|
| id | int | 礼物的类型 |
|
||||||
|
|
||||||
|
目前支持的礼物ID:
|
||||||
|
|
||||||
|
| id | 类型 |
|
||||||
|
| --- | ---------- |
|
||||||
|
| 0 | 甜Wink |
|
||||||
|
| 1 | 快乐肥宅水 |
|
||||||
|
| 2 | 幸运手链 |
|
||||||
|
| 3 | 卡布奇诺 |
|
||||||
|
| 4 | 猫咪手表 |
|
||||||
|
| 5 | 绒绒手套 |
|
||||||
|
| 6 | 彩虹糖果 |
|
||||||
|
| 7 | 坚强 |
|
||||||
|
| 8 | 告白话筒 |
|
||||||
|
| 9 | 牵你的手 |
|
||||||
|
| 10 | 可爱猫咪 |
|
||||||
|
| 11 | 神秘面具 |
|
||||||
|
| 12 | 我超忙的 |
|
||||||
|
| 13 | 爱心口罩 |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
示例: `[CQ:gift,qq=123456,id=8]`
|
||||||
|
|
||||||
### 合并转发
|
### 合并转发
|
||||||
|
|
||||||
Type: `forward`
|
Type: `forward`
|
||||||
@ -26,8 +116,8 @@ Type: `forward`
|
|||||||
|
|
||||||
参数:
|
参数:
|
||||||
|
|
||||||
| 参数名 | 类型 | 说明 |
|
| 参数名 | 类型 | 说明 |
|
||||||
| ------ | ------ | ------------------------------------------------------------ |
|
| ------ | ------ | ------------------------------------------------------------- |
|
||||||
| id | string | 合并转发ID, 需要通过 `/get_forward_msg` API获取转发的具体内容 |
|
| id | string | 合并转发ID, 需要通过 `/get_forward_msg` API获取转发的具体内容 |
|
||||||
|
|
||||||
示例: `[CQ:forward,id=xxxx]`
|
示例: `[CQ:forward,id=xxxx]`
|
||||||
@ -40,12 +130,12 @@ Type: `node`
|
|||||||
|
|
||||||
参数:
|
参数:
|
||||||
|
|
||||||
| 参数名 | 类型 | 说明 | 特殊说明 |
|
| 参数名 | 类型 | 说明 | 特殊说明 |
|
||||||
| ------- | ------- | -------------- | ------------------------------------------------------------ |
|
| ------- | ------- | -------------- | -------------------------------------------------------------------------------------- |
|
||||||
| id | int32 | 转发消息id | 直接引用他人的消息合并转发, 实际查看顺序为原消息发送顺序 **与下面的自定义消息二选一** |
|
| id | int32 | 转发消息id | 直接引用他人的消息合并转发, 实际查看顺序为原消息发送顺序 **与下面的自定义消息二选一** |
|
||||||
| name | string | 发送者显示名字 | 用于自定义消息 (自定义消息并合并转发,实际查看顺序为自定义消息段顺序) |
|
| name | string | 发送者显示名字 | 用于自定义消息 (自定义消息并合并转发,实际查看顺序为自定义消息段顺序) |
|
||||||
| uin | int64 | 发送者QQ号 | 用于自定义消息 |
|
| uin | int64 | 发送者QQ号 | 用于自定义消息 |
|
||||||
| content | message | 具体消息 | 用于自定义消息 **不支持转发套娃,不支持引用回复** |
|
| content | message | 具体消息 | 用于自定义消息 **不支持转发套娃,不支持引用回复** |
|
||||||
|
|
||||||
特殊说明: **需要使用单独的API `/send_group_forward_msg` 发送,并且由于消息段较为复杂,仅支持Array形式入参。 如果引用消息和自定义消息同时出现,实际查看顺序将取消息段顺序. 另外按 [CQHTTP](https://cqhttp.cc/docs/4.15/#/Message?id=格式) 文档说明, `data` 应全为字符串, 但由于需要接收`message` 类型的消息, 所以 *仅限此Type的content字段* 支持Array套娃**
|
特殊说明: **需要使用单独的API `/send_group_forward_msg` 发送,并且由于消息段较为复杂,仅支持Array形式入参。 如果引用消息和自定义消息同时出现,实际查看顺序将取消息段顺序. 另外按 [CQHTTP](https://cqhttp.cc/docs/4.15/#/Message?id=格式) 文档说明, `data` 应全为字符串, 但由于需要接收`message` 类型的消息, 所以 *仅限此Type的content字段* 支持Array套娃**
|
||||||
|
|
||||||
@ -119,7 +209,127 @@ Type: `node`
|
|||||||
]
|
]
|
||||||
````
|
````
|
||||||
|
|
||||||
|
### xml支持
|
||||||
|
|
||||||
|
Type: `xml`
|
||||||
|
|
||||||
|
范围: **发送/接收**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ----------------------------------------- |
|
||||||
|
| data | string | xml内容,xml中的value部分,记得实体化处理 |
|
||||||
|
| resid | int32 | 可以不填 |
|
||||||
|
|
||||||
|
示例: `[CQ:xml,data=xxxx]`
|
||||||
|
|
||||||
|
#### 一些xml样例
|
||||||
|
|
||||||
|
#### ps:重要:xml中的value部分,记得html实体化处理后,再打加入到cq码中
|
||||||
|
|
||||||
|
#### qq音乐
|
||||||
|
|
||||||
|
```xml
|
||||||
|
<?xml version='1.0' encoding='UTF-8' standalone='yes' ?><msg serviceID="2" templateID="1" action="web" brief="[分享] 十年" sourceMsgId="0" url="https://i.y.qq.com/v8/playsong.html?_wv=1&songid=4830342&souce=qqshare&source=qqshare&ADTAG=qqshare" flag="0" adverSign="0" multiMsgFlag="0" ><item layout="2"><audio cover="http://imgcache.qq.com/music/photo/album_500/26/500_albumpic_89526_0.jpg" src="http://ws.stream.qqmusic.qq.com/C400003mAan70zUy5O.m4a?guid=1535153710&vkey=D5315B8C0603653592AD4879A8A3742177F59D582A7A86546E24DD7F282C3ACF81526C76E293E57EA1E42CF19881C561275D919233333ADE&uin=&fromtag=3" /><title>十年</title><summary>陈奕迅</summary></item><source name="QQ音乐" icon="https://i.gtimg.cn/open/app_icon/01/07/98/56/1101079856_100_m.png" url="http://web.p.qq.com/qqmpmobile/aio/app.html?id=1101079856" action="app" a_actionData="com.tencent.qqmusic" i_actionData="tencent1101079856://" appid="1101079856" /></msg>
|
||||||
|
```
|
||||||
|
#### 网易音乐
|
||||||
|
```xml
|
||||||
|
<?xml version='1.0' encoding='UTF-8' standalone='yes' ?><msg serviceID="2" templateID="1" action="web" brief="[分享] 十年" sourceMsgId="0" url="http://music.163.com/m/song/409650368" flag="0" adverSign="0" multiMsgFlag="0" ><item layout="2"><audio cover="http://p2.music.126.net/g-Qgb9ibk9Wp_0HWra0xQQ==/16636710440565853.jpg?param=90y90" src="https://music.163.com/song/media/outer/url?id=409650368.mp3" /><title>十年</title><summary>黄梦之</summary></item><source name="网易云音乐" icon="https://pic.rmb.bdstatic.com/911423bee2bef937975b29b265d737b3.png" url="http://web.p.qq.com/qqmpmobile/aio/app.html?id=1101079856" action="app" a_actionData="com.netease.cloudmusic" i_actionData="tencent100495085://" appid="100495085" /></msg>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 卡片消息1
|
||||||
|
```xml
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
|
||||||
|
<msg serviceID="1">
|
||||||
|
<item><title>生死8秒!女司机高速急刹,他一个操作救下一车性命</title></item>
|
||||||
|
<source name="官方认证消息" icon="https://qzs.qq.com/ac/qzone_v5/client/auth_icon.png" action="" appid="-1" />
|
||||||
|
</msg>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 卡片消息2
|
||||||
|
```xml
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
|
||||||
|
<msg serviceID="1">
|
||||||
|
<item layout="4">
|
||||||
|
<title>test title</title>
|
||||||
|
<picture cover="http://url.cn/5CEwIUy"/>
|
||||||
|
</item>
|
||||||
|
</msg>
|
||||||
|
```
|
||||||
|
|
||||||
|
### json消息支持
|
||||||
|
|
||||||
|
Type: `json`
|
||||||
|
|
||||||
|
范围: **发送/接收**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ----------------------------------------------- |
|
||||||
|
| data | string | json内容,json的所有字符串记得实体化处理 |
|
||||||
|
| resid | int32 | 默认不填为0,走小程序通道,填了走富文本通道发送 |
|
||||||
|
|
||||||
|
json中的字符串需要进行转义:
|
||||||
|
|
||||||
|
>","=>`,`、
|
||||||
|
|
||||||
|
>"&"=> `&`、
|
||||||
|
|
||||||
|
>"["=>`[`、
|
||||||
|
|
||||||
|
>"]"=>`]`、
|
||||||
|
|
||||||
|
否则无法正确得到解析
|
||||||
|
|
||||||
|
示例json 的cq码:
|
||||||
|
```test
|
||||||
|
[CQ:json,data={"app":"com.tencent.miniapp","desc":"","view":"notification","ver":"0.0.0.1","prompt":"[应用]","appID":"","sourceName":"","actionData":"","actionData_A":"","sourceUrl":"","meta":{"notification":{"appInfo":{"appName":"全国疫情数据统计","appType":4,"appid":1109659848,"iconUrl":"http:\/\/gchat.qpic.cn\/gchatpic_new\/719328335\/-2010394141-6383A777BEB79B70B31CE250142D740F\/0"},"data":[{"title":"确诊","value":"80932"},{"title":"今日确诊","value":"28"},{"title":"疑似","value":"72"},{"title":"今日疑似","value":"5"},{"title":"治愈","value":"60197"},{"title":"今日治愈","value":"1513"},{"title":"死亡","value":"3140"},{"title":"今**亡","value":"17"}],"title":"中国加油,武汉加油","button":[{"name":"病毒:SARS-CoV-2,其导致疾病命名 COVID-19","action":""},{"name":"传染源:新冠肺炎的患者。无症状感染者也可能成为传染源。","action":""}],"emphasis_keyword":""}},"text":"","sourceAd":""}]
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### cardimage 一种xml的图片消息(装逼大图)
|
||||||
|
|
||||||
|
ps: xml 接口的消息都存在风控风险,请自行兼容发送失败后的处理(可以失败后走普通图片模式)
|
||||||
|
|
||||||
|
Type: `cardimage`
|
||||||
|
|
||||||
|
范围: **发送**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| --------- | ------ | ------------------------------------- |
|
||||||
|
| file | string | 和image的file字段对齐,支持也是一样的 |
|
||||||
|
| minwidth | int64 | 默认不填为400,最小width |
|
||||||
|
| minheight | int64 | 默认不填为400,最小height |
|
||||||
|
| maxwidth | int64 | 默认不填为500,最大width |
|
||||||
|
| maxheight | int64 | 默认不填为1000,最大height |
|
||||||
|
| source | string | 分享来源的名称,可以留空 |
|
||||||
|
| icon | string | 分享来源的icon图标url,可以留空 |
|
||||||
|
|
||||||
|
|
||||||
|
示例cardimage 的cq码:
|
||||||
|
```test
|
||||||
|
[CQ:cardimage,file=https://i.pixiv.cat/img-master/img/2020/03/25/00/00/08/80334602_p0_master1200.jpg]
|
||||||
|
```
|
||||||
|
|
||||||
|
### 文本转语音
|
||||||
|
|
||||||
|
> 注意:通过TX的TTS接口,采用的音源与登录账号的性别有关
|
||||||
|
|
||||||
|
Type: `tts`
|
||||||
|
|
||||||
|
范围: **发送(仅群聊)**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | ---- |
|
||||||
|
| text | string | 内容 |
|
||||||
|
|
||||||
|
示例: `[CQ:tts,text=这是一条测试消息]`
|
||||||
|
|
||||||
## API
|
## API
|
||||||
|
|
||||||
@ -129,10 +339,32 @@ Type: `node`
|
|||||||
|
|
||||||
**参数**
|
**参数**
|
||||||
|
|
||||||
| 字段 | 类型 | 说明 |
|
| 字段 | 类型 | 说明 |
|
||||||
| -------- | ------ | ---- |
|
| ---------- | ------ | ---- |
|
||||||
| group_id | int64 | 群号 |
|
| group_id | int64 | 群号 |
|
||||||
| name | string | 新名 |
|
| group_name | string | 新名 |
|
||||||
|
|
||||||
|
### 设置群头像
|
||||||
|
|
||||||
|
终结点: `/set_group_portrait`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| -------- | ------ | ------------------------ |
|
||||||
|
| group_id | int64 | 群号 |
|
||||||
|
| file | string | 图片文件名 |
|
||||||
|
| cache | int | 表示是否使用已缓存的文件 |
|
||||||
|
|
||||||
|
[1]`file` 参数支持以下几种格式:
|
||||||
|
|
||||||
|
- 绝对路径,例如 `file:///C:\\Users\Richard\Pictures\1.png`,格式使用 [`file` URI](https://tools.ietf.org/html/rfc8089)
|
||||||
|
- 网络 URL,例如 `http://i1.piimg.com/567571/fdd6e7b6d93f1ef0.jpg`
|
||||||
|
- Base64 编码,例如 `base64://iVBORw0KGgoAAAANSUhEUgAAABQAAAAVCAIAAADJt1n/AAAAKElEQVQ4EWPk5+RmIBcwkasRpG9UM4mhNxpgowFGMARGEwnBIEJVAAAdBgBNAZf+QAAAAABJRU5ErkJggg==`
|
||||||
|
|
||||||
|
[2]`cache`参数: 通过网络 URL 发送时有效,`1`表示使用缓存,`0`关闭关闭缓存,默认 为`1`
|
||||||
|
|
||||||
|
[3] 目前这个API在登录一段时间后因cookie失效而失效,请考虑后使用
|
||||||
|
|
||||||
### 获取图片信息
|
### 获取图片信息
|
||||||
|
|
||||||
@ -154,9 +386,9 @@ Type: `node`
|
|||||||
| `filename` | string | 图片文件原名 |
|
| `filename` | string | 图片文件原名 |
|
||||||
| `url` | string | 图片下载地址 |
|
| `url` | string | 图片下载地址 |
|
||||||
|
|
||||||
### 获取群消息
|
### 获取消息
|
||||||
|
|
||||||
终结点: `/get_group_msg`
|
终结点: `/get_msg`
|
||||||
|
|
||||||
参数
|
参数
|
||||||
|
|
||||||
@ -172,7 +404,7 @@ Type: `node`
|
|||||||
| `real_id` | int32 | 消息真实id |
|
| `real_id` | int32 | 消息真实id |
|
||||||
| `sender` | object | 发送者 |
|
| `sender` | object | 发送者 |
|
||||||
| `time` | int32 | 发送时间 |
|
| `time` | int32 | 发送时间 |
|
||||||
| `content` | message | 消息内容 |
|
| `message` | message | 消息内容 |
|
||||||
|
|
||||||
### 获取合并转发内容
|
### 获取合并转发内容
|
||||||
|
|
||||||
@ -230,7 +462,222 @@ Type: `node`
|
|||||||
| `group_id` | int64 | 群号 |
|
| `group_id` | int64 | 群号 |
|
||||||
| `messages` | forward node[] | 自定义转发消息, 具体看CQCode |
|
| `messages` | forward node[] | 自定义转发消息, 具体看CQCode |
|
||||||
|
|
||||||
###
|
### 获取中文分词
|
||||||
|
|
||||||
|
终结点: `/.get_word_slices`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| --------- | ------ | ---- |
|
||||||
|
| `content` | string | 内容 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| -------- | -------- | ---- |
|
||||||
|
| `slices` | string[] | 词组 |
|
||||||
|
|
||||||
|
### 图片OCR
|
||||||
|
|
||||||
|
> 注意: 目前图片OCR接口仅支持接受的图片
|
||||||
|
|
||||||
|
终结点: `/.ocr_image`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------- | ------ | ------ |
|
||||||
|
| `image` | string | 图片ID |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | --------------- | ------- |
|
||||||
|
| `texts` | TextDetection[] | OCR结果 |
|
||||||
|
| `language` | string | 语言 |
|
||||||
|
|
||||||
|
**TextDetection**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------- | ------- | ------ |
|
||||||
|
| `text` | string | 文本 |
|
||||||
|
| `confidence` | int32 | 置信度 |
|
||||||
|
| `coordinates` | vector2 | 坐标 |
|
||||||
|
|
||||||
|
|
||||||
|
### 获取群系统消息
|
||||||
|
|
||||||
|
终结点: `/get_group_system_msg`
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------------ | ---------------- | ------------ |
|
||||||
|
| `invited_requests` | InvitedRequest[] | 邀请消息列表 |
|
||||||
|
| `join_requests` | JoinRequest[] | 进群消息列表 |
|
||||||
|
|
||||||
|
> 注意: 如果列表不存在任何消息, 将返回 `null`
|
||||||
|
|
||||||
|
**InvitedRequest**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| -------------- | ------ | ----------------- |
|
||||||
|
| `request_id` | int64 | 请求ID |
|
||||||
|
| `invitor_uin` | int64 | 邀请者 |
|
||||||
|
| `invitor_nick` | string | 邀请者昵称 |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
| `group_name` | string | 群名 |
|
||||||
|
| `checked` | bool | 是否已被处理 |
|
||||||
|
| `actor` | int64 | 处理者, 未处理为0 |
|
||||||
|
|
||||||
|
**JoinRequest**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------------- | ------ | ----------------- |
|
||||||
|
| `request_id` | int64 | 请求ID |
|
||||||
|
| `requester_uin` | int64 | 请求者ID |
|
||||||
|
| `requester_nick` | string | 请求者昵称 |
|
||||||
|
| `message` | string | 验证消息 |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
| `group_name` | string | 群名 |
|
||||||
|
| `checked` | bool | 是否已被处理 |
|
||||||
|
| `actor` | int64 | 处理者, 未处理为0 |
|
||||||
|
|
||||||
|
### 获取群文件系统信息
|
||||||
|
|
||||||
|
终结点: `/get_group_file_system_info`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ----- | ---- |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------- | ----- | ---------- |
|
||||||
|
| `file_count` | int32 | 文件总数 |
|
||||||
|
| `limit_count` | int32 | 文件上限 |
|
||||||
|
| `used_space` | int64 | 已使用空间 |
|
||||||
|
| `total_space` | int64 | 空间上限 |
|
||||||
|
|
||||||
|
### 获取群根目录文件列表
|
||||||
|
|
||||||
|
> `File` 和 `Folder` 对象信息请参考最下方
|
||||||
|
|
||||||
|
终结点: `/get_group_root_files`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ----- | ---- |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| --------- | -------- | ---------- |
|
||||||
|
| `files` | File[] | 文件列表 |
|
||||||
|
| `folders` | Folder[] | 文件夹列表 |
|
||||||
|
|
||||||
|
### 获取群子目录文件列表
|
||||||
|
|
||||||
|
> `File` 和 `Folder` 对象信息请参考最下方
|
||||||
|
|
||||||
|
终结点: `/get_group_files_by_folder`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ----------- | ------ | --------------------------- |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
| `folder_id` | string | 文件夹ID 参考 `Folder` 对象 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| --------- | -------- | ---------- |
|
||||||
|
| `files` | File[] | 文件列表 |
|
||||||
|
| `folders` | Folder[] | 文件夹列表 |
|
||||||
|
|
||||||
|
### 获取群文件资源链接
|
||||||
|
|
||||||
|
> `File` 和 `Folder` 对象信息请参考最下方
|
||||||
|
|
||||||
|
终结点: `/get_group_file_url`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ------ | ------------------------- |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
| `file_id` | string | 文件ID 参考 `File` 对象 |
|
||||||
|
| `busid` | int32 | 文件类型 参考 `File` 对象 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ----- | ------ | ------------ |
|
||||||
|
| `url` | string | 文件下载链接 |
|
||||||
|
|
||||||
|
**File**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------------- | ------ | ---------------------- |
|
||||||
|
| `file_id` | string | 文件ID |
|
||||||
|
| `file_name` | string | 文件名 |
|
||||||
|
| `busid` | int32 | 文件类型 |
|
||||||
|
| `file_size` | int64 | 文件大小 |
|
||||||
|
| `upload_time` | int64 | 上传时间 |
|
||||||
|
| `dead_time` | int64 | 过期时间,永久文件恒为0 |
|
||||||
|
| `modify_time` | int64 | 最后修改时间 |
|
||||||
|
| `download_times` | int32 | 下载次数 |
|
||||||
|
| `uploader` | int64 | 上传者ID |
|
||||||
|
| `uploader_name` | string | 上传者名字 |
|
||||||
|
|
||||||
|
**Folder**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------------ | ------ | ---------- |
|
||||||
|
| `folder_id` | string | 文件夹ID |
|
||||||
|
| `folder_name` | string | 文件名 |
|
||||||
|
| `create_time` | int64 | 创建时间 |
|
||||||
|
| `creator` | int64 | 创建者 |
|
||||||
|
| `creator_name` | string | 创建者名字 |
|
||||||
|
| `total_file_count` | int32 | 子文件数量 |
|
||||||
|
|
||||||
|
### 获取状态
|
||||||
|
|
||||||
|
终结点: `/get_status`
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| --------------- | ---------- | ------------------------------- |
|
||||||
|
| app_initialized | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
||||||
|
| app_enabled | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
||||||
|
| plugins_good | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
||||||
|
| app_good | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
||||||
|
| online | bool | 表示BOT是否在线 |
|
||||||
|
| goold | bool | 同 `online` |
|
||||||
|
| stat | Statistics | 运行统计 |
|
||||||
|
|
||||||
|
**Statistics**
|
||||||
|
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------------- | ------ | ---------------- |
|
||||||
|
| packet_received | uint64 | 收到的数据包总数 |
|
||||||
|
| packet_sent | uint64 | 发送的数据包总数 |
|
||||||
|
| packet_lost | uint32 | 数据包丢失总数 |
|
||||||
|
| message_received | uint64 | 接受信息总数 |
|
||||||
|
| message_sent | uint64 | 发送信息总数 |
|
||||||
|
| disconnect_times | uint32 | TCP链接断开次数 |
|
||||||
|
| lost_times | uint32 | 账号掉线次数 |
|
||||||
|
|
||||||
|
> 注意: 所有统计信息都将在重启后重制
|
||||||
|
|
||||||
## 事件
|
## 事件
|
||||||
|
|
||||||
@ -251,10 +698,90 @@ Type: `node`
|
|||||||
|
|
||||||
**上报数据**
|
**上报数据**
|
||||||
|
|
||||||
| 字段 | 类型 | 可能的值 | 说明 |
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
| ------------- | ------ | -------------- | -------------- |
|
| ------------- | ------ | --------------- | -------------- |
|
||||||
| `post_type` | string | `notice` | 上报类型 |
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
| `notice_type` | string | `friend_recall`| 消息类型 |
|
| `notice_type` | string | `friend_recall` | 消息类型 |
|
||||||
| `user_id` | int64 | | 好友id |
|
| `user_id` | int64 | | 好友id |
|
||||||
| `message_id` | int64 | | 被撤回的消息id |
|
| `message_id` | int64 | | 被撤回的消息id |
|
||||||
|
|
||||||
|
#### 群内戳一戳
|
||||||
|
|
||||||
|
> 注意:此事件无法在平板和手表协议上触发
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | -------- | -------- |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `notify` | 消息类型 |
|
||||||
|
| `group_id` | int64 | | 群号 |
|
||||||
|
| `sub_type` | string | `poke` | 提示类型 |
|
||||||
|
| `user_id` | int64 | | 发送者id |
|
||||||
|
| `target_id` | int64 | | 被戳者id |
|
||||||
|
|
||||||
|
#### 群红包运气王提示
|
||||||
|
|
||||||
|
> 注意:此事件无法在平板和手表协议上触发
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | ------------ | ------------ |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `notify` | 消息类型 |
|
||||||
|
| `group_id` | int64 | | 群号 |
|
||||||
|
| `sub_type` | string | `lucky_king` | 提示类型 |
|
||||||
|
| `user_id` | int64 | | 红包发送者id |
|
||||||
|
| `target_id` | int64 | | 运气王id |
|
||||||
|
|
||||||
|
#### 群成员荣誉变更提示
|
||||||
|
|
||||||
|
> 注意:此事件无法在平板和手表协议上触发
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | -------------------------------------------------------- | -------- |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `notify` | 消息类型 |
|
||||||
|
| `group_id` | int64 | | 群号 |
|
||||||
|
| `sub_type` | string | `honor` | 提示类型 |
|
||||||
|
| `user_id` | int64 | | 成员id |
|
||||||
|
| `honor_type` | string | `talkative:龙王` `performer:群聊之火` `emotion:快乐源泉` | 荣誉类型 |
|
||||||
|
|
||||||
|
#### 群成员名片更新
|
||||||
|
|
||||||
|
> 注意: 此事件不保证时效性,仅在收到消息时校验卡片
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | ------------ | -------- |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `group_card` | 消息类型 |
|
||||||
|
| `group_id` | int64 | | 群号 |
|
||||||
|
| `user_id` | int64 | | 成员id |
|
||||||
|
| `card_new` | int64 | | 新名片 |
|
||||||
|
| `card_old` | int64 | | 旧名片 |
|
||||||
|
|
||||||
|
> PS: 当名片为空时 `card_xx` 字段为空字符串, 并不是昵称
|
||||||
|
|
||||||
|
#### 接收到离线文件
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | -------------- | -------- |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `offline_file` | 消息类型 |
|
||||||
|
| `user_id` | int64 | | 发送者id |
|
||||||
|
| `file` | object | | 文件数据 |
|
||||||
|
|
||||||
|
**file object**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------ | ------ | -------- | -------- |
|
||||||
|
| `name` | string | | 文件名 |
|
||||||
|
| `size` | int64 | | 文件大小 |
|
||||||
|
| `url` | string | | 下载链接 |
|
@ -5,7 +5,7 @@ go-cqhttp 默认生成的文件树如下所示:
|
|||||||
````
|
````
|
||||||
.
|
.
|
||||||
├── go-cqhttp
|
├── go-cqhttp
|
||||||
├── config.json
|
├── config.hjson
|
||||||
├── device.json
|
├── device.json
|
||||||
├── logs
|
├── logs
|
||||||
│ └── xx-xx-xx.log
|
│ └── xx-xx-xx.log
|
||||||
@ -18,7 +18,7 @@ go-cqhttp 默认生成的文件树如下所示:
|
|||||||
| 文件 | 用途 |
|
| 文件 | 用途 |
|
||||||
| ----------- | ------------------- |
|
| ----------- | ------------------- |
|
||||||
| go-cqhttp | go-cqhttp可执行文件 |
|
| go-cqhttp | go-cqhttp可执行文件 |
|
||||||
| config.json | 运行配置文件 |
|
| config.hjson | 运行配置文件 |
|
||||||
| device.json | 虚拟设备配置文件 |
|
| device.json | 虚拟设备配置文件 |
|
||||||
| logs | 日志存放目录 |
|
| logs | 日志存放目录 |
|
||||||
| data | 数据目录 |
|
| data | 数据目录 |
|
||||||
|
@ -1,3 +1,147 @@
|
|||||||
# 开始
|
# 开始
|
||||||
|
|
||||||
欢迎来到 go-cqhttp 文档
|
欢迎来到 go-cqhttp 文档 目前还在咕
|
||||||
|
|
||||||
|
# 基础教程
|
||||||
|
## 下载
|
||||||
|
从[release](https://github.com/Mrs4s/go-cqhttp/releases)界面下载最新版本的go-cqhttp
|
||||||
|
|
||||||
|
- Windows下32位文件为 `go-cqhttp-v*-windows-386.zip`
|
||||||
|
- Windows下64位文件为 `go-cqhttp-v*-windows-amd64.zip`
|
||||||
|
- Windows下arm用(如使用高通CPU的笔记本)文件为 `go-cqhttp-v*-windows-arm.zip`
|
||||||
|
- Linux下32位文件为 `go-cqhttp-v*-linux-386.tar.gz`
|
||||||
|
- Linux下64位文件为 `go-cqhttp-v*-linux-amd64.tar.gz`
|
||||||
|
- Linux下arm用(如树莓派)文件为 `go-cqhttp-v*-linux-arm.tar.gz`
|
||||||
|
- MD5文件为 `*.md5` ,用于校验文件完整性
|
||||||
|
- 如果没有你所使用的系统版本或者希望自己构建,请移步[进阶指南-如何自己构建](#如何自己构建)
|
||||||
|
|
||||||
|
## 解压
|
||||||
|
|
||||||
|
- Windows下请使用自己熟悉的解压软件自行解压
|
||||||
|
- Linux下在命令行中输入 `tar -xzvf [文件名]`
|
||||||
|
|
||||||
|
## 使用
|
||||||
|
|
||||||
|
### Windows
|
||||||
|
|
||||||
|
#### 标准方法
|
||||||
|
|
||||||
|
1. 双击`go-cqhttp.exe`此时将提示
|
||||||
|
```
|
||||||
|
[WARNING]: 尝试加载配置文件 config.hjson 失败: 文件不存在
|
||||||
|
[INFO]: 默认配置文件已生成,请编辑 config.hjson 后重启程序.
|
||||||
|
```
|
||||||
|
2. 参照[config.md](https://github.com/Mrs4s/go-cqhttp/blob/master/docs/config.md)和你所用到的插件的 `README` 填入参数
|
||||||
|
3. 再次双击`go-cqhttp.exe`
|
||||||
|
```
|
||||||
|
[INFO]: 登录成功 欢迎使用: balabala
|
||||||
|
```
|
||||||
|
|
||||||
|
如出现需要认证的信息,请自行认证设备。
|
||||||
|
|
||||||
|
此时,基础配置完成
|
||||||
|
|
||||||
|
#### 懒人法
|
||||||
|
|
||||||
|
1. [下载包含Windows.bat的zip](https://github.com/fkx4-p/go-cqhttp-lazy/archive/master.zip)
|
||||||
|
2. 解压
|
||||||
|
3. 将`Windows.bat`复制/剪切到**go-cqhttp**文件夹
|
||||||
|
4. 双击运行
|
||||||
|
|
||||||
|
效果如下
|
||||||
|
|
||||||
|
```
|
||||||
|
QQ account:
|
||||||
|
[QQ账号]
|
||||||
|
QQ password:
|
||||||
|
[QQ密码]
|
||||||
|
enable http?(Y/n)
|
||||||
|
[是否开启http(y/n),默认开启]
|
||||||
|
enable ws?(Y/n)
|
||||||
|
[是否开启websocket(y/n),默认开启]
|
||||||
|
请按任意键继续. . .
|
||||||
|
```
|
||||||
|
|
||||||
|
5. 双击`go-cqhttp.exe`
|
||||||
|
```
|
||||||
|
[INFO]: 登录成功 欢迎使用: balabala
|
||||||
|
```
|
||||||
|
|
||||||
|
如出现需要认证的信息,请自行认证设备。
|
||||||
|
|
||||||
|
此时,基础配置完成
|
||||||
|
|
||||||
|
### Linux
|
||||||
|
|
||||||
|
#### 标准方法
|
||||||
|
|
||||||
|
1. 打开一个命令行/ssh
|
||||||
|
2. `cd`到解压目录
|
||||||
|
3. 输入 `./go-cqhttp`,`Enter`运行 ,此时将提示
|
||||||
|
```
|
||||||
|
[WARNING]: 尝试加载配置文件 config.hjson 失败: 文件不存在
|
||||||
|
[INFO]: 默认配置文件已生成,请编辑 config.hjson 后重启程序.
|
||||||
|
```
|
||||||
|
|
||||||
|
4. 参照[config.md](https://github.com/Mrs4s/go-cqhttp/blob/master/docs/config.md)和你所用到的插件的 `README` 填入参数
|
||||||
|
5. 再次输入 `./go-cqhttp`,`Enter`运行
|
||||||
|
```
|
||||||
|
[INFO]: 登录成功 欢迎使用: balabala
|
||||||
|
```
|
||||||
|
|
||||||
|
如出现需要认证的信息,请自行认证设备。
|
||||||
|
|
||||||
|
此时,基础配置完成
|
||||||
|
|
||||||
|
#### 懒人法
|
||||||
|
|
||||||
|
暂时咕咕咕了
|
||||||
|
|
||||||
|
## 验证http是否成功配置
|
||||||
|
|
||||||
|
此时,如果在本地开启的服务器,可以在浏览器输入`http://127.0.0.1:5700/send_private_msg?user_id=[接收者qq号]&message=[发送的信息]`来发送一条测试信息
|
||||||
|
|
||||||
|
如果出现`{"data":{"message_id":balabala},"retcode":0,"status":"ok"}`则证明已经成功配置HTTP
|
||||||
|
|
||||||
|
*注:请 连 中括号 也替换掉,就像这样:*`http://127.0.0.1:5700/send_private_msg?user_id=10001&message=ffeecoishp`
|
||||||
|
|
||||||
|
# 进阶指南
|
||||||
|
|
||||||
|
## 如何自己构建
|
||||||
|
|
||||||
|
1. [下载源码](https://github.com/Mrs4s/go-cqhttp/archive/master.zip)并解压 || 使用`git clone https://github.com/Mrs4s/go-cqhttp.git`来拉取
|
||||||
|
|
||||||
|
2. [下载golang binary release](https://golang.google.cn/dl/)并安装或者[自己构建golang](https://golang.google.cn/doc/install/source)
|
||||||
|
|
||||||
|
3. 在`cmd`或Linux命令行中,`cd`到目录中
|
||||||
|
|
||||||
|
4. 输入`go build -ldflags "-s -w -extldflags '-static'"`,`Enter`运行
|
||||||
|
|
||||||
|
*注:可以使用*`go env -w GOPROXY=https://goproxy.cn,direct`*来加速国内依赖安装速度*
|
||||||
|
|
||||||
|
## 更新
|
||||||
|
|
||||||
|
### 方法一
|
||||||
|
|
||||||
|
从[release](https://github.com/Mrs4s/go-cqhttp/releases)界面下载最新版本的go-cqhttp
|
||||||
|
并替换之前的版本
|
||||||
|
|
||||||
|
### 方法二
|
||||||
|
|
||||||
|
使用更新参数,在命令行中打开go-cqhttp所在目录
|
||||||
|
#### windows
|
||||||
|
输入指令
|
||||||
|
`go-cqhttp.exe update`
|
||||||
|
|
||||||
|
如果在国内连接github下载速度可能很慢,可以使用镜像源下载
|
||||||
|
|
||||||
|
`go-cqhttp.exe update https://github.rc1844.workers.dev`
|
||||||
|
|
||||||
|
几个可用的镜像源
|
||||||
|
- `https://hub.fastgit.org`
|
||||||
|
- `https://github.com.cnpmjs.org`
|
||||||
|
- `https://github.bajins.com`
|
||||||
|
- `https://github.rc1844.workers.dev`
|
||||||
|
|
||||||
|
#### linux
|
||||||
|
方法与windows基本一致,将 `go-cqhttp.exe` 替换为 `./go-cqhttp`即可
|
||||||
|
45
global/codec.go
Normal file
45
global/codec.go
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
package global
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/md5"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"path"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/wdvxdr1123/go-silk/silk"
|
||||||
|
)
|
||||||
|
|
||||||
|
var codec silk.Encoder
|
||||||
|
var useCodec = true
|
||||||
|
var once sync.Once
|
||||||
|
|
||||||
|
func InitCodec() {
|
||||||
|
once.Do(func() {
|
||||||
|
log.Info("正在加载silk编码器...")
|
||||||
|
err := codec.Init("data/cache", "codec")
|
||||||
|
if err != nil {
|
||||||
|
log.Error(err)
|
||||||
|
useCodec = false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func Encoder(data []byte) ([]byte, error) {
|
||||||
|
if useCodec == false {
|
||||||
|
return nil, errors.New("no silk encoder")
|
||||||
|
}
|
||||||
|
h := md5.New()
|
||||||
|
h.Write(data)
|
||||||
|
tempName := fmt.Sprintf("%x", h.Sum(nil))
|
||||||
|
if silkPath := path.Join("data/cache", tempName+".silk"); PathExists(silkPath) {
|
||||||
|
return ioutil.ReadFile(silkPath)
|
||||||
|
}
|
||||||
|
slk, err := codec.EncodeToSilk(data, tempName, true)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return slk, nil
|
||||||
|
}
|
223
global/config.go
223
global/config.go
@ -1,21 +1,168 @@
|
|||||||
package global
|
package global
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"os"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/hjson/hjson-go"
|
||||||
|
jsoniter "github.com/json-iterator/go"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
|
var DefaultConfigWithComments = `
|
||||||
|
/*
|
||||||
|
go-cqhttp 默认配置文件
|
||||||
|
*/
|
||||||
|
|
||||||
|
{
|
||||||
|
// QQ号
|
||||||
|
uin: 0
|
||||||
|
// QQ密码
|
||||||
|
password: ""
|
||||||
|
// 是否启用密码加密
|
||||||
|
encrypt_password: false
|
||||||
|
// 加密后的密码, 如未启用密码加密将为空, 请勿随意修改.
|
||||||
|
password_encrypted: ""
|
||||||
|
// 是否启用内置数据库
|
||||||
|
// 启用将会增加10-20MB的内存占用和一定的磁盘空间
|
||||||
|
// 关闭将无法使用 撤回 回复 get_msg 等上下文相关功能
|
||||||
|
enable_db: true
|
||||||
|
// 访问密钥, 强烈推荐在公网的服务器设置
|
||||||
|
access_token: ""
|
||||||
|
// 重连设置
|
||||||
|
relogin: {
|
||||||
|
// 是否启用自动重连
|
||||||
|
// 如不启用掉线后将不会自动重连
|
||||||
|
enabled: true
|
||||||
|
// 重连延迟, 单位秒
|
||||||
|
relogin_delay: 3
|
||||||
|
// 最大重连次数, 0为无限制
|
||||||
|
max_relogin_times: 0
|
||||||
|
}
|
||||||
|
// API限速设置
|
||||||
|
// 该设置为全局生效
|
||||||
|
// 原 cqhttp 虽然启用了 rate_limit 后缀, 但是基本没插件适配
|
||||||
|
// 目前该限速设置为令牌桶算法, 请参考:
|
||||||
|
// https://baike.baidu.com/item/%E4%BB%A4%E7%89%8C%E6%A1%B6%E7%AE%97%E6%B3%95/6597000?fr=aladdin
|
||||||
|
_rate_limit: {
|
||||||
|
// 是否启用限速
|
||||||
|
enabled: false
|
||||||
|
// 令牌回复频率, 单位秒
|
||||||
|
frequency: 1
|
||||||
|
// 令牌桶大小
|
||||||
|
bucket_size: 1
|
||||||
|
}
|
||||||
|
// 是否忽略无效的CQ码
|
||||||
|
// 如果为假将原样发送
|
||||||
|
ignore_invalid_cqcode: false
|
||||||
|
// 是否强制分片发送消息
|
||||||
|
// 分片发送将会带来更快的速度
|
||||||
|
// 但是兼容性会有些问题
|
||||||
|
force_fragmented: false
|
||||||
|
// 心跳频率, 单位秒
|
||||||
|
// -1 为关闭心跳
|
||||||
|
heartbeat_interval: 0
|
||||||
|
// HTTP设置
|
||||||
|
http_config: {
|
||||||
|
// 是否启用正向HTTP服务器
|
||||||
|
enabled: true
|
||||||
|
// 服务端监听地址
|
||||||
|
host: 0.0.0.0
|
||||||
|
// 服务端监听端口
|
||||||
|
port: 5700
|
||||||
|
// 反向HTTP超时时间, 单位秒
|
||||||
|
// 最小值为5,小于5将会忽略本项设置
|
||||||
|
timeout: 0
|
||||||
|
// 反向HTTP POST地址列表
|
||||||
|
// 格式:
|
||||||
|
// {
|
||||||
|
// 地址: secret
|
||||||
|
// }
|
||||||
|
post_urls: {}
|
||||||
|
}
|
||||||
|
// 正向WS设置
|
||||||
|
ws_config: {
|
||||||
|
// 是否启用正向WS服务器
|
||||||
|
enabled: true
|
||||||
|
// 正向WS服务器监听地址
|
||||||
|
host: 0.0.0.0
|
||||||
|
// 正向WS服务器监听端口
|
||||||
|
port: 6700
|
||||||
|
}
|
||||||
|
// 反向WS设置
|
||||||
|
ws_reverse_servers: [
|
||||||
|
// 可以添加多个反向WS推送
|
||||||
|
{
|
||||||
|
// 是否启用该推送
|
||||||
|
enabled: false
|
||||||
|
// 反向WS Universal 地址
|
||||||
|
// 注意 设置了此项地址后下面两项将会被忽略
|
||||||
|
// 留空请使用 ""
|
||||||
|
reverse_url: ws://you_websocket_universal.server
|
||||||
|
// 反向WS API 地址
|
||||||
|
reverse_api_url: ws://you_websocket_api.server
|
||||||
|
// 反向WS Event 地址
|
||||||
|
reverse_event_url: ws://you_websocket_event.server
|
||||||
|
// 重连间隔 单位毫秒
|
||||||
|
reverse_reconnect_interval: 3000
|
||||||
|
}
|
||||||
|
]
|
||||||
|
// 上报数据类型
|
||||||
|
// 可选: string array
|
||||||
|
post_message_format: string
|
||||||
|
// 是否使用服务器下发的新地址进行重连
|
||||||
|
// 注意, 此设置可能导致在海外服务器上连接情况更差
|
||||||
|
use_sso_address: false
|
||||||
|
// 是否启用 DEBUG
|
||||||
|
debug: false
|
||||||
|
// 日志等级
|
||||||
|
log_level: ""
|
||||||
|
// WebUi 设置
|
||||||
|
web_ui: {
|
||||||
|
// 是否启用 WebUi
|
||||||
|
enabled: true
|
||||||
|
// 监听地址
|
||||||
|
host: 127.0.0.1
|
||||||
|
// 监听端口
|
||||||
|
web_ui_port: 9999
|
||||||
|
// 是否接收来自web的输入
|
||||||
|
web_input: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
type JsonConfig struct {
|
type JsonConfig struct {
|
||||||
Uin int64 `json:"uin"`
|
Uin int64 `json:"uin"`
|
||||||
Password string `json:"password"`
|
Password string `json:"password"`
|
||||||
EnableDB bool `json:"enable_db"`
|
EncryptPassword bool `json:"encrypt_password"`
|
||||||
AccessToken string `json:"access_token"`
|
PasswordEncrypted string `json:"password_encrypted"`
|
||||||
ReLogin bool `json:"relogin"`
|
EnableDB bool `json:"enable_db"`
|
||||||
ReLoginDelay int `json:"relogin_delay"`
|
AccessToken string `json:"access_token"`
|
||||||
HttpConfig *GoCQHttpConfig `json:"http_config"`
|
ReLogin struct {
|
||||||
WSConfig *GoCQWebsocketConfig `json:"ws_config"`
|
Enabled bool `json:"enabled"`
|
||||||
ReverseServers []*GoCQReverseWebsocketConfig `json:"ws_reverse_servers"`
|
ReLoginDelay int `json:"relogin_delay"`
|
||||||
Debug bool `json:"debug"`
|
MaxReloginTimes uint `json:"max_relogin_times"`
|
||||||
|
} `json:"relogin"`
|
||||||
|
RateLimit struct {
|
||||||
|
Enabled bool `json:"enabled"`
|
||||||
|
Frequency float64 `json:"frequency"`
|
||||||
|
BucketSize int `json:"bucket_size"`
|
||||||
|
} `json:"_rate_limit"`
|
||||||
|
IgnoreInvalidCQCode bool `json:"ignore_invalid_cqcode"`
|
||||||
|
ForceFragmented bool `json:"force_fragmented"`
|
||||||
|
ProxyRewrite string `json:"proxy_rewrite"`
|
||||||
|
HeartbeatInterval time.Duration `json:"heartbeat_interval"`
|
||||||
|
HttpConfig *GoCQHttpConfig `json:"http_config"`
|
||||||
|
WSConfig *GoCQWebsocketConfig `json:"ws_config"`
|
||||||
|
ReverseServers []*GoCQReverseWebsocketConfig `json:"ws_reverse_servers"`
|
||||||
|
PostMessageFormat string `json:"post_message_format"`
|
||||||
|
UseSSOAddress bool `json:"use_sso_address"`
|
||||||
|
Debug bool `json:"debug"`
|
||||||
|
LogLevel string `json:"log_level"`
|
||||||
|
WebUi *GoCqWebUi `json:"web_ui"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type CQHttpApiConfig struct {
|
type CQHttpApiConfig struct {
|
||||||
@ -41,6 +188,7 @@ type GoCQHttpConfig struct {
|
|||||||
Enabled bool `json:"enabled"`
|
Enabled bool `json:"enabled"`
|
||||||
Host string `json:"host"`
|
Host string `json:"host"`
|
||||||
Port uint16 `json:"port"`
|
Port uint16 `json:"port"`
|
||||||
|
Timeout int32 `json:"timeout"`
|
||||||
PostUrls map[string]string `json:"post_urls"`
|
PostUrls map[string]string `json:"post_urls"`
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -58,9 +206,36 @@ type GoCQReverseWebsocketConfig struct {
|
|||||||
ReverseReconnectInterval uint16 `json:"reverse_reconnect_interval"`
|
ReverseReconnectInterval uint16 `json:"reverse_reconnect_interval"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type GoCqWebUi struct {
|
||||||
|
Enabled bool `json:"enabled"`
|
||||||
|
Host string `json:"host"`
|
||||||
|
WebUiPort uint64 `json:"web_ui_port"`
|
||||||
|
WebInput bool `json:"web_input"`
|
||||||
|
}
|
||||||
|
|
||||||
func DefaultConfig() *JsonConfig {
|
func DefaultConfig() *JsonConfig {
|
||||||
return &JsonConfig{
|
return &JsonConfig{
|
||||||
EnableDB: true,
|
EnableDB: true,
|
||||||
|
ReLogin: struct {
|
||||||
|
Enabled bool `json:"enabled"`
|
||||||
|
ReLoginDelay int `json:"relogin_delay"`
|
||||||
|
MaxReloginTimes uint `json:"max_relogin_times"`
|
||||||
|
}{
|
||||||
|
Enabled: true,
|
||||||
|
ReLoginDelay: 3,
|
||||||
|
MaxReloginTimes: 0,
|
||||||
|
},
|
||||||
|
RateLimit: struct {
|
||||||
|
Enabled bool `json:"enabled"`
|
||||||
|
Frequency float64 `json:"frequency"`
|
||||||
|
BucketSize int `json:"bucket_size"`
|
||||||
|
}{
|
||||||
|
Enabled: false,
|
||||||
|
Frequency: 1,
|
||||||
|
BucketSize: 1,
|
||||||
|
},
|
||||||
|
PostMessageFormat: "string",
|
||||||
|
ForceFragmented: false,
|
||||||
HttpConfig: &GoCQHttpConfig{
|
HttpConfig: &GoCQHttpConfig{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
Host: "0.0.0.0",
|
Host: "0.0.0.0",
|
||||||
@ -81,6 +256,12 @@ func DefaultConfig() *JsonConfig {
|
|||||||
ReverseReconnectInterval: 3000,
|
ReverseReconnectInterval: 3000,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
WebUi: &GoCqWebUi{
|
||||||
|
Enabled: true,
|
||||||
|
Host: "127.0.0.1",
|
||||||
|
WebInput: false,
|
||||||
|
WebUiPort: 9999,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -89,20 +270,30 @@ func Load(p string) *JsonConfig {
|
|||||||
log.Warnf("尝试加载配置文件 %v 失败: 文件不存在", p)
|
log.Warnf("尝试加载配置文件 %v 失败: 文件不存在", p)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
c := JsonConfig{}
|
var dat map[string]interface{}
|
||||||
err := json.Unmarshal([]byte(ReadAllText(p)), &c)
|
var c = JsonConfig{}
|
||||||
|
err := hjson.Unmarshal([]byte(ReadAllText(p)), &dat)
|
||||||
|
if err == nil {
|
||||||
|
b, _ := json.Marshal(dat)
|
||||||
|
err = json.Unmarshal(b, &c)
|
||||||
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("尝试加载配置文件 %v 时出现错误: %v", p, err)
|
log.Warnf("尝试加载配置文件 %v 时出现错误: %v", p, err)
|
||||||
|
log.Infoln("原文件已备份")
|
||||||
|
_ = os.Rename(p, p+".backup"+strconv.FormatInt(time.Now().Unix(), 10))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return &c
|
return &c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *JsonConfig) Save(p string) error {
|
func (c *JsonConfig) Save(p string) error {
|
||||||
data, err := json.MarshalIndent(c, "", "\t")
|
data, err := hjson.MarshalWithOptions(c, hjson.EncoderOptions{
|
||||||
|
Eol: "\n",
|
||||||
|
BracesSameLine: true,
|
||||||
|
IndentBy: " ",
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
WriteAllText(p, string(data))
|
return WriteAllText(p, string(data))
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
252
global/filter.go
Normal file
252
global/filter.go
Normal file
@ -0,0 +1,252 @@
|
|||||||
|
package global
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Filter interface {
|
||||||
|
Eval(payload gjson.Result) bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type OperationNode struct {
|
||||||
|
key string
|
||||||
|
filter Filter
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotOperator struct {
|
||||||
|
operand_ Filter
|
||||||
|
}
|
||||||
|
|
||||||
|
func notOperatorConstruct(argument gjson.Result) *NotOperator {
|
||||||
|
if !argument.IsObject() {
|
||||||
|
panic("the argument of 'not' operator must be an object")
|
||||||
|
}
|
||||||
|
op := new(NotOperator)
|
||||||
|
op.operand_ = Generate("and", argument)
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (notOperator NotOperator) Eval(payload gjson.Result) bool {
|
||||||
|
return !(notOperator.operand_).Eval(payload)
|
||||||
|
}
|
||||||
|
|
||||||
|
type AndOperator struct {
|
||||||
|
operands []OperationNode
|
||||||
|
}
|
||||||
|
|
||||||
|
func andOperatorConstruct(argument gjson.Result) *AndOperator {
|
||||||
|
if !argument.IsObject() {
|
||||||
|
panic("the argument of 'and' operator must be an object")
|
||||||
|
}
|
||||||
|
op := new(AndOperator)
|
||||||
|
argument.ForEach(func(key, value gjson.Result) bool {
|
||||||
|
if key.Str[0] == '.' {
|
||||||
|
// is an operator
|
||||||
|
// ".foo": {
|
||||||
|
// "bar": "baz"
|
||||||
|
// }
|
||||||
|
opKey := key.Str[1:]
|
||||||
|
op.operands = append(op.operands, OperationNode{"", Generate(opKey, value)})
|
||||||
|
} else if value.IsObject() {
|
||||||
|
// is an normal key with an object as the value
|
||||||
|
// "foo": {
|
||||||
|
// ".bar": "baz"
|
||||||
|
// }
|
||||||
|
opKey := key.String()
|
||||||
|
op.operands = append(op.operands, OperationNode{opKey, Generate("and", value)})
|
||||||
|
} else {
|
||||||
|
// is an normal key with a non-object as the value
|
||||||
|
// "foo": "bar"
|
||||||
|
opKey := key.String()
|
||||||
|
op.operands = append(op.operands, OperationNode{opKey, Generate("eq", value)})
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (andOperator *AndOperator) Eval(payload gjson.Result) bool {
|
||||||
|
res := true
|
||||||
|
for _, operand := range andOperator.operands {
|
||||||
|
|
||||||
|
if len(operand.key) == 0 {
|
||||||
|
// is an operator
|
||||||
|
res = res && operand.filter.Eval(payload)
|
||||||
|
} else {
|
||||||
|
// is an normal key
|
||||||
|
val := payload.Get(operand.key)
|
||||||
|
res = res && operand.filter.Eval(val)
|
||||||
|
}
|
||||||
|
|
||||||
|
if res == false {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
type OrOperator struct {
|
||||||
|
operands []Filter
|
||||||
|
}
|
||||||
|
|
||||||
|
func orOperatorConstruct(argument gjson.Result) *OrOperator {
|
||||||
|
if !argument.IsArray() {
|
||||||
|
panic("the argument of 'or' operator must be an array")
|
||||||
|
}
|
||||||
|
op := new(OrOperator)
|
||||||
|
argument.ForEach(func(_, value gjson.Result) bool {
|
||||||
|
op.operands = append(op.operands, Generate("and", value))
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (orOperator OrOperator) Eval(payload gjson.Result) bool {
|
||||||
|
res := false
|
||||||
|
for _, operand := range orOperator.operands {
|
||||||
|
res = res || operand.Eval(payload)
|
||||||
|
|
||||||
|
if res == true {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
type EqualOperator struct {
|
||||||
|
value gjson.Result
|
||||||
|
}
|
||||||
|
|
||||||
|
func equalOperatorConstruct(argument gjson.Result) *EqualOperator {
|
||||||
|
op := new(EqualOperator)
|
||||||
|
op.value = argument
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (equalOperator EqualOperator) Eval(payload gjson.Result) bool {
|
||||||
|
return payload.String() == equalOperator.value.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotEqualOperator struct {
|
||||||
|
value gjson.Result
|
||||||
|
}
|
||||||
|
|
||||||
|
func notEqualOperatorConstruct(argument gjson.Result) *NotEqualOperator {
|
||||||
|
op := new(NotEqualOperator)
|
||||||
|
op.value = argument
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (notEqualOperator NotEqualOperator) Eval(payload gjson.Result) bool {
|
||||||
|
return !(payload.String() == notEqualOperator.value.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
type InOperator struct {
|
||||||
|
operand gjson.Result
|
||||||
|
}
|
||||||
|
|
||||||
|
func inOperatorConstruct(argument gjson.Result) *InOperator {
|
||||||
|
if argument.IsObject() {
|
||||||
|
panic("the argument of 'in' operator must be an array or a string")
|
||||||
|
}
|
||||||
|
op := new(InOperator)
|
||||||
|
op.operand = argument
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (inOperator InOperator) Eval(payload gjson.Result) bool {
|
||||||
|
if inOperator.operand.IsArray() {
|
||||||
|
res := false
|
||||||
|
inOperator.operand.ForEach(func(key, value gjson.Result) bool {
|
||||||
|
res = res || value.String() == payload.String()
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
return strings.Contains(inOperator.operand.String(), payload.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
type ContainsOperator struct {
|
||||||
|
operand string
|
||||||
|
}
|
||||||
|
|
||||||
|
func containsOperatorConstruct(argument gjson.Result) *ContainsOperator {
|
||||||
|
if argument.IsArray() || argument.IsObject() {
|
||||||
|
panic("the argument of 'contains' operator must be a string")
|
||||||
|
}
|
||||||
|
op := new(ContainsOperator)
|
||||||
|
op.operand = argument.String()
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (containsOperator ContainsOperator) Eval(payload gjson.Result) bool {
|
||||||
|
if payload.IsObject() || payload.IsArray() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return strings.Contains(payload.String(), containsOperator.operand)
|
||||||
|
}
|
||||||
|
|
||||||
|
type RegexOperator struct {
|
||||||
|
regex string
|
||||||
|
}
|
||||||
|
|
||||||
|
func regexOperatorConstruct(argument gjson.Result) *RegexOperator {
|
||||||
|
if argument.IsArray() || argument.IsObject() {
|
||||||
|
panic("the argument of 'regex' operator must be a string")
|
||||||
|
}
|
||||||
|
op := new(RegexOperator)
|
||||||
|
op.regex = argument.String()
|
||||||
|
return op
|
||||||
|
}
|
||||||
|
|
||||||
|
func (containsOperator RegexOperator) Eval(payload gjson.Result) bool {
|
||||||
|
matched, _ := regexp.MatchString(containsOperator.regex, payload.String())
|
||||||
|
return matched
|
||||||
|
}
|
||||||
|
|
||||||
|
func Generate(opName string, argument gjson.Result) Filter {
|
||||||
|
switch opName {
|
||||||
|
case "not":
|
||||||
|
return notOperatorConstruct(argument)
|
||||||
|
case "and":
|
||||||
|
return andOperatorConstruct(argument)
|
||||||
|
case "or":
|
||||||
|
return orOperatorConstruct(argument)
|
||||||
|
case "neq":
|
||||||
|
return notEqualOperatorConstruct(argument)
|
||||||
|
case "eq":
|
||||||
|
return equalOperatorConstruct(argument)
|
||||||
|
case "in":
|
||||||
|
return inOperatorConstruct(argument)
|
||||||
|
case "contains":
|
||||||
|
return containsOperatorConstruct(argument)
|
||||||
|
case "regex":
|
||||||
|
return regexOperatorConstruct(argument)
|
||||||
|
default:
|
||||||
|
panic("the operator " + opName + " is not supported")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var EventFilter = new(Filter)
|
||||||
|
|
||||||
|
func BootFilter() {
|
||||||
|
defer func() {
|
||||||
|
if e := recover(); e != nil {
|
||||||
|
log.Warnf("事件过滤器启动失败: %v", e)
|
||||||
|
EventFilter = nil
|
||||||
|
} else {
|
||||||
|
log.Info("事件过滤器启动成功.")
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
f, err := ioutil.ReadFile("filter.json")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
} else {
|
||||||
|
*EventFilter = Generate("and", gjson.ParseBytes(f))
|
||||||
|
}
|
||||||
|
}
|
126
global/fs.go
126
global/fs.go
@ -1,13 +1,34 @@
|
|||||||
package global
|
package global
|
||||||
|
|
||||||
import (
|
import (
|
||||||
log "github.com/sirupsen/logrus"
|
"bytes"
|
||||||
|
"crypto/md5"
|
||||||
|
"encoding/base64"
|
||||||
|
"encoding/hex"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"net"
|
||||||
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"runtime"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/dustin/go-humanize"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
var IMAGE_PATH = path.Join("data", "images")
|
var (
|
||||||
|
IMAGE_PATH = path.Join("data", "images")
|
||||||
|
VOICE_PATH = path.Join("data", "voices")
|
||||||
|
VIDEO_PATH = path.Join("data", "videos")
|
||||||
|
CACHE_PATH = path.Join("data", "cache")
|
||||||
|
|
||||||
|
HEADER_AMR = []byte("#!AMR")
|
||||||
|
HEADER_SILK = []byte("\x02#!SILK_V3")
|
||||||
|
)
|
||||||
|
|
||||||
func PathExists(path string) bool {
|
func PathExists(path string) bool {
|
||||||
_, err := os.Stat(path)
|
_, err := os.Stat(path)
|
||||||
@ -22,8 +43,8 @@ func ReadAllText(path string) string {
|
|||||||
return string(b)
|
return string(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
func WriteAllText(path, text string) {
|
func WriteAllText(path, text string) error {
|
||||||
_ = ioutil.WriteFile(path, []byte(text), 0777)
|
return ioutil.WriteFile(path, []byte(text), 0644)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Check(err error) {
|
func Check(err error) {
|
||||||
@ -32,9 +53,96 @@ func Check(err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func IsAMR(b []byte) bool {
|
func IsAMRorSILK(b []byte) bool {
|
||||||
if len(b) <= 6 {
|
return bytes.HasPrefix(b, HEADER_AMR) || bytes.HasPrefix(b, HEADER_SILK)
|
||||||
return false
|
}
|
||||||
}
|
|
||||||
return b[0] == 0x23 && b[1] == 0x21 && b[2] == 0x41 && b[3] == 0x4D && b[4] == 0x52 // amr file header
|
func FindFile(f, cache, PATH string) (data []byte, err error) {
|
||||||
|
data, err = nil, errors.New("syntax error")
|
||||||
|
if strings.HasPrefix(f, "http") || strings.HasPrefix(f, "https") {
|
||||||
|
if cache == "" {
|
||||||
|
cache = "1"
|
||||||
|
}
|
||||||
|
hash := md5.Sum([]byte(f))
|
||||||
|
cacheFile := path.Join(CACHE_PATH, hex.EncodeToString(hash[:])+".cache")
|
||||||
|
if PathExists(cacheFile) && cache == "1" {
|
||||||
|
return ioutil.ReadFile(cacheFile)
|
||||||
|
}
|
||||||
|
data, err = GetBytes(f)
|
||||||
|
_ = ioutil.WriteFile(cacheFile, data, 0644)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else if strings.HasPrefix(f, "base64") {
|
||||||
|
data, err = base64.StdEncoding.DecodeString(strings.ReplaceAll(f, "base64://", ""))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else if strings.HasPrefix(f, "file") {
|
||||||
|
var fu *url.URL
|
||||||
|
fu, err = url.Parse(f)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(fu.Path, "/") && runtime.GOOS == `windows` {
|
||||||
|
fu.Path = fu.Path[1:]
|
||||||
|
}
|
||||||
|
data, err = ioutil.ReadFile(fu.Path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else if PathExists(path.Join(PATH, f)) {
|
||||||
|
data, err = ioutil.ReadFile(path.Join(PATH, f))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func DelFile(path string) bool {
|
||||||
|
err := os.Remove(path)
|
||||||
|
if err != nil {
|
||||||
|
// 删除失败
|
||||||
|
log.Error(err)
|
||||||
|
return false
|
||||||
|
} else {
|
||||||
|
// 删除成功
|
||||||
|
log.Info(path + "删除成功")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadAddrFile(path string) []*net.TCPAddr {
|
||||||
|
d, err := ioutil.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
str := string(d)
|
||||||
|
lines := strings.Split(str, "\n")
|
||||||
|
var ret []*net.TCPAddr
|
||||||
|
for _, l := range lines {
|
||||||
|
ip := strings.Split(strings.TrimSpace(l), ":")
|
||||||
|
if len(ip) == 2 {
|
||||||
|
port, _ := strconv.Atoi(ip[1])
|
||||||
|
ret = append(ret, &net.TCPAddr{IP: net.ParseIP(ip[0]), Port: port})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
type WriteCounter struct {
|
||||||
|
Total uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (wc *WriteCounter) Write(p []byte) (int, error) {
|
||||||
|
n := len(p)
|
||||||
|
wc.Total += uint64(n)
|
||||||
|
wc.PrintProgress()
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (wc WriteCounter) PrintProgress() {
|
||||||
|
fmt.Printf("\r%s", strings.Repeat(" ", 35))
|
||||||
|
fmt.Printf("\rDownloading... %s complete", humanize.Bytes(wc.Total))
|
||||||
}
|
}
|
||||||
|
@ -3,18 +3,48 @@ package global
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"compress/gzip"
|
"compress/gzip"
|
||||||
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var client = &http.Client{
|
||||||
|
Timeout: time.Second * 15,
|
||||||
|
Transport: &http.Transport{
|
||||||
|
Proxy: func(request *http.Request) (u *url.URL, e error) {
|
||||||
|
if Proxy == "" {
|
||||||
|
return http.ProxyFromEnvironment(request)
|
||||||
|
}
|
||||||
|
return url.Parse(Proxy)
|
||||||
|
},
|
||||||
|
DialContext: (&net.Dialer{
|
||||||
|
Timeout: 30 * time.Second,
|
||||||
|
KeepAlive: 30 * time.Second,
|
||||||
|
DualStack: true,
|
||||||
|
}).DialContext,
|
||||||
|
ForceAttemptHTTP2: true,
|
||||||
|
MaxIdleConns: 100,
|
||||||
|
IdleConnTimeout: 90 * time.Second,
|
||||||
|
TLSHandshakeTimeout: 10 * time.Second,
|
||||||
|
ExpectContinueTimeout: 1 * time.Second,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var Proxy string
|
||||||
|
|
||||||
func GetBytes(url string) ([]byte, error) {
|
func GetBytes(url string) ([]byte, error) {
|
||||||
req, err := http.NewRequest("GET", url, nil)
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
req.Header["User-Agent"] = []string{"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.116 Safari/537.36 Edg/83.0.478.61"}
|
req.Header["User-Agent"] = []string{"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.116 Safari/537.36 Edg/83.0.478.61"}
|
||||||
resp, err := http.DefaultClient.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -32,3 +62,19 @@ func GetBytes(url string) ([]byte, error) {
|
|||||||
}
|
}
|
||||||
return body, nil
|
return body, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func QQMusicSongInfo(id string) (gjson.Result, error) {
|
||||||
|
d, err := GetBytes(`https://u.y.qq.com/cgi-bin/musicu.fcg?format=json&inCharset=utf8&outCharset=utf-8¬ice=0&platform=yqq.json&needNewCode=0&data={%22comm%22:{%22ct%22:24,%22cv%22:0},%22songinfo%22:{%22method%22:%22get_song_detail_yqq%22,%22param%22:{%22song_type%22:0,%22song_mid%22:%22%22,%22song_id%22:` + id + `},%22module%22:%22music.pf_song_detail_svr%22}}`)
|
||||||
|
if err != nil {
|
||||||
|
return gjson.Result{}, err
|
||||||
|
}
|
||||||
|
return gjson.ParseBytes(d).Get("songinfo.data"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func NeteaseMusicSongInfo(id string) (gjson.Result, error) {
|
||||||
|
d, err := GetBytes(fmt.Sprintf("http://music.163.com/api/song/detail/?id=%s&ids=%%5B%s%%5D", id, id))
|
||||||
|
if err != nil {
|
||||||
|
return gjson.Result{}, err
|
||||||
|
}
|
||||||
|
return gjson.ParseBytes(d).Get("songs.0"), nil
|
||||||
|
}
|
||||||
|
73
global/param.go
Normal file
73
global/param.go
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
package global
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
|
)
|
||||||
|
|
||||||
|
var trueSet = map[string]struct{}{
|
||||||
|
"true": {},
|
||||||
|
"yes": {},
|
||||||
|
"1": {},
|
||||||
|
}
|
||||||
|
|
||||||
|
var falseSet = map[string]struct{}{
|
||||||
|
"false": {},
|
||||||
|
"no": {},
|
||||||
|
"0": {},
|
||||||
|
}
|
||||||
|
|
||||||
|
func EnsureBool(p interface{}, defaultVal bool) bool {
|
||||||
|
var str string
|
||||||
|
if b, ok := p.(bool); ok {
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
if j, ok := p.(gjson.Result); ok {
|
||||||
|
if !j.Exists() {
|
||||||
|
return defaultVal
|
||||||
|
}
|
||||||
|
if j.Type == gjson.True {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if j.Type == gjson.False {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if j.Type != gjson.String {
|
||||||
|
return defaultVal
|
||||||
|
}
|
||||||
|
str = j.Str
|
||||||
|
} else if s, ok := p.(string); ok {
|
||||||
|
str = s
|
||||||
|
}
|
||||||
|
str = strings.ToLower(str)
|
||||||
|
if _, ok := trueSet[str]; ok {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if _, ok := falseSet[str]; ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return defaultVal
|
||||||
|
}
|
||||||
|
|
||||||
|
// VersionNameCompare 检查版本名是否需要更新, 仅适用于 go-cqhttp 的版本命名规则
|
||||||
|
// 例: v0.9.29-fix2 == v0.9.29-fix2 -> false
|
||||||
|
// v0.9.29-fix1 < v0.9.29-fix2 -> true
|
||||||
|
// v0.9.29-fix2 > v0.9.29-fix1 -> false
|
||||||
|
// v0.9.29-fix2 < v0.9.30 -> true
|
||||||
|
func VersionNameCompare(current, remote string) bool {
|
||||||
|
sp := regexp.MustCompile(`[0-9]\d*`)
|
||||||
|
cur := sp.FindAllStringSubmatch(current, -1)
|
||||||
|
re := sp.FindAllStringSubmatch(remote, -1)
|
||||||
|
for i := 0; i < int(math.Min(float64(len(cur)), float64(len(re)))); i++ {
|
||||||
|
curSub, _ := strconv.Atoi(cur[i][0])
|
||||||
|
reSub, _ := strconv.Atoi(re[i][0])
|
||||||
|
if curSub < reSub {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return len(cur) < len(re)
|
||||||
|
}
|
21
global/ratelimit.go
Normal file
21
global/ratelimit.go
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
package global
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"golang.org/x/time/rate"
|
||||||
|
)
|
||||||
|
|
||||||
|
var limiter *rate.Limiter
|
||||||
|
var limitEnable = false
|
||||||
|
|
||||||
|
func RateLimit(ctx context.Context) {
|
||||||
|
if limitEnable {
|
||||||
|
_ = limiter.Wait(ctx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func InitLimiter(r float64, b int) {
|
||||||
|
limitEnable = true
|
||||||
|
limiter = rate.NewLimiter(rate.Limit(r), b)
|
||||||
|
}
|
28
go.mod
28
go.mod
@ -1,20 +1,28 @@
|
|||||||
module github.com/Mrs4s/go-cqhttp
|
module github.com/Mrs4s/go-cqhttp
|
||||||
|
|
||||||
go 1.14
|
go 1.15
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200808044635-cd20d7d43dbb
|
github.com/Mrs4s/MiraiGo v0.0.0-20201202165542-3344bb0c87fa
|
||||||
|
github.com/dustin/go-humanize v1.0.0
|
||||||
|
github.com/getlantern/go-update v0.0.0-20190510022740-79c495ab728c
|
||||||
|
github.com/getlantern/golog v0.0.0-20201105130739-9586b8bde3a9 // indirect
|
||||||
github.com/gin-gonic/gin v1.6.3
|
github.com/gin-gonic/gin v1.6.3
|
||||||
github.com/gorilla/websocket v1.4.2
|
github.com/gorilla/websocket v1.4.2
|
||||||
github.com/guonaihong/gout v0.1.1
|
github.com/guonaihong/gout v0.1.3
|
||||||
github.com/lestrrat-go/file-rotatelogs v2.3.0+incompatible
|
github.com/hjson/hjson-go v3.1.0+incompatible
|
||||||
github.com/lestrrat-go/strftime v1.0.1 // indirect
|
github.com/json-iterator/go v1.1.10
|
||||||
|
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0 // indirect
|
||||||
|
github.com/kr/binarydist v0.1.0 // indirect
|
||||||
|
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible
|
||||||
|
github.com/lestrrat-go/strftime v1.0.3 // indirect
|
||||||
github.com/pkg/errors v0.9.1 // indirect
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
github.com/sirupsen/logrus v1.6.0
|
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5
|
||||||
|
github.com/sirupsen/logrus v1.7.0
|
||||||
|
github.com/syndtr/goleveldb v1.0.0
|
||||||
github.com/t-tomalak/logrus-easy-formatter v0.0.0-20190827215021-c074f06c5816
|
github.com/t-tomalak/logrus-easy-formatter v0.0.0-20190827215021-c074f06c5816
|
||||||
github.com/tidwall/gjson v1.6.0
|
github.com/tidwall/gjson v1.6.3
|
||||||
github.com/xujiajun/nutsdb v0.5.0
|
github.com/wdvxdr1123/go-silk v0.0.0-20201007123416-b982fd3d91d6
|
||||||
github.com/yinghau76/go-ascii-art v0.0.0-20190517192627-e7f465a30189
|
github.com/yinghau76/go-ascii-art v0.0.0-20190517192627-e7f465a30189
|
||||||
golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa
|
golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e
|
||||||
golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae // indirect
|
|
||||||
)
|
)
|
||||||
|
104
go.sum
104
go.sum
@ -1,19 +1,32 @@
|
|||||||
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200807030850-ed30f7ad5934 h1:LoNjIsnyEQFGP9IchIQ65yHRCfNKSru3BAOguRepkCM=
|
github.com/Mrs4s/MiraiGo v0.0.0-20201202140458-0eb4eb738d31 h1:jClElKWovoOzDYmVd16UQc8638d0FvyKCtVj7qF+ej4=
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200807030850-ed30f7ad5934/go.mod h1:0je03wji/tSw4bUH4QCF2Z4/EjyNWjSJTyy5tliX6EM=
|
github.com/Mrs4s/MiraiGo v0.0.0-20201202140458-0eb4eb738d31/go.mod h1:J1zaJWyeX7hQIPpOobqb8opxTNPbguotudPPrHJMoDM=
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200808003732-2a32e623270d h1:K9jHdcO13mLqQB0xm0/ZlY852FoVQJ/WSDwfdmfhDlU=
|
github.com/Mrs4s/MiraiGo v0.0.0-20201202165542-3344bb0c87fa h1:2+TW0hS+hbdmfD5wXir629LUZgZVZwax4iLpdvb8j+s=
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200808003732-2a32e623270d/go.mod h1:0je03wji/tSw4bUH4QCF2Z4/EjyNWjSJTyy5tliX6EM=
|
github.com/Mrs4s/MiraiGo v0.0.0-20201202165542-3344bb0c87fa/go.mod h1:J1zaJWyeX7hQIPpOobqb8opxTNPbguotudPPrHJMoDM=
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200808044635-cd20d7d43dbb h1:XLe/UreYJRT65GStA3+irRL1Ao0pHZwBtCmTc+4prwA=
|
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20200808044635-cd20d7d43dbb/go.mod h1:0je03wji/tSw4bUH4QCF2Z4/EjyNWjSJTyy5tliX6EM=
|
|
||||||
github.com/bwmarrin/snowflake v0.3.0 h1:xm67bEhkKh6ij1790JB83OujPR5CzNe8QuQqAgISZN0=
|
|
||||||
github.com/bwmarrin/snowflake v0.3.0/go.mod h1:NdZxfVWX+oR6y2K0o6qAYv6gIOP9rjG0/E9WsDpxqwE=
|
|
||||||
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
|
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
|
||||||
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
|
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo=
|
||||||
|
github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
|
||||||
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
||||||
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
|
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
||||||
|
github.com/getlantern/context v0.0.0-20190109183933-c447772a6520 h1:NRUJuo3v3WGC/g5YiyF790gut6oQr5f3FBI88Wv0dx4=
|
||||||
|
github.com/getlantern/context v0.0.0-20190109183933-c447772a6520/go.mod h1:L+mq6/vvYHKjCX2oez0CgEAJmbq1fbb/oNJIWQkBybY=
|
||||||
|
github.com/getlantern/errors v1.0.1 h1:XukU2whlh7OdpxnkXhNH9VTLVz0EVPGKDV5K0oWhvzw=
|
||||||
|
github.com/getlantern/errors v1.0.1/go.mod h1:l+xpFBrCtDLpK9qNjxs+cHU6+BAdlBaxHqikB6Lku3A=
|
||||||
|
github.com/getlantern/go-update v0.0.0-20190510022740-79c495ab728c h1:mP9bsvdddRSMwqO+lmNuSrsH7nD2nBIz7af+e/4je4c=
|
||||||
|
github.com/getlantern/go-update v0.0.0-20190510022740-79c495ab728c/go.mod h1:goroSTghTcnjKaR2C8ovKWy1lEvRNfqHrW/kRJNMek0=
|
||||||
|
github.com/getlantern/golog v0.0.0-20201105130739-9586b8bde3a9 h1:8MYJU90rB1bsavemKSAuDKBjtAKo5xq95bEPOnzV7CE=
|
||||||
|
github.com/getlantern/golog v0.0.0-20201105130739-9586b8bde3a9/go.mod h1:ZyIjgH/1wTCl+B+7yH1DqrWp6MPJqESmwmEQ89ZfhvA=
|
||||||
|
github.com/getlantern/hex v0.0.0-20190417191902-c6586a6fe0b7 h1:micT5vkcr9tOVk1FiH8SWKID8ultN44Z+yzd2y/Vyb0=
|
||||||
|
github.com/getlantern/hex v0.0.0-20190417191902-c6586a6fe0b7/go.mod h1:dD3CgOrwlzca8ed61CsZouQS5h5jIzkK9ZWrTcf0s+o=
|
||||||
|
github.com/getlantern/hidden v0.0.0-20190325191715-f02dbb02be55 h1:XYzSdCbkzOC0FDNrgJqGRo8PCMFOBFL9py72DRs7bmc=
|
||||||
|
github.com/getlantern/hidden v0.0.0-20190325191715-f02dbb02be55/go.mod h1:6mmzY2kW1TOOrVy+r41Za2MxXM+hhqTtY3oBKd2AgFA=
|
||||||
|
github.com/getlantern/ops v0.0.0-20190325191751-d70cb0d6f85f h1:wrYrQttPS8FHIRSlsrcuKazukx/xqO/PpLZzZXsF+EA=
|
||||||
|
github.com/getlantern/ops v0.0.0-20190325191751-d70cb0d6f85f/go.mod h1:D5ao98qkA6pxftxoqzibIBBrLSUli+kYnJqrgBf9cIA=
|
||||||
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
||||||
github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
|
github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
|
||||||
github.com/gin-gonic/gin v1.5.0/go.mod h1:Nd6IXA8m5kNZdNEHMBd93KT+mdY3+bewLgRvmCsR2Do=
|
github.com/gin-gonic/gin v1.5.0/go.mod h1:Nd6IXA8m5kNZdNEHMBd93KT+mdY3+bewLgRvmCsR2Do=
|
||||||
@ -28,6 +41,8 @@ github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD87
|
|||||||
github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA=
|
github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA=
|
||||||
github.com/go-playground/validator/v10 v10.2.0 h1:KgJ0snyC2R9VXYN2rneOtQcw5aHQB1Vv0sFl1UcHBOY=
|
github.com/go-playground/validator/v10 v10.2.0 h1:KgJ0snyC2R9VXYN2rneOtQcw5aHQB1Vv0sFl1UcHBOY=
|
||||||
github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI=
|
github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI=
|
||||||
|
github.com/go-stack/stack v1.8.0 h1:5SgMzNM5HxrEjV0ww2lTmX6E2Izsfxas4+YHWRs3Lsk=
|
||||||
|
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
||||||
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
||||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
@ -39,8 +54,10 @@ github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrU
|
|||||||
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
|
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
|
||||||
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
|
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
|
||||||
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
|
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
|
||||||
github.com/golang/protobuf v1.4.2 h1:+Z5KGCizgyZCbGh1KZqA0fcLLkwbsjIzS4aV2v7wJX0=
|
github.com/golang/protobuf v1.4.3 h1:JjCZWpVbqXDqFVmTfYWEVTMIYrL/NPdPSCHPJ0T/raM=
|
||||||
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
||||||
|
github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db h1:woRePGFeVFfLKN/pOkfl+p/TAqKOfFu+7KPlMVpok/w=
|
||||||
|
github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
|
||||||
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
||||||
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
||||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
||||||
@ -50,60 +67,73 @@ github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/
|
|||||||
github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
github.com/gorilla/websocket v1.4.2 h1:+/TMaTYc4QFitKJxsQ7Yye35DkWvkdLcvGKqM+x0Ufc=
|
github.com/gorilla/websocket v1.4.2 h1:+/TMaTYc4QFitKJxsQ7Yye35DkWvkdLcvGKqM+x0Ufc=
|
||||||
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
|
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
|
||||||
github.com/guonaihong/gout v0.1.1 h1:2i3eqQ1KUhTlj7AFeIHqVUFku5QwUhwE2wNgYTVpbxQ=
|
github.com/guonaihong/gout v0.1.3 h1:BIiV6nnsA+R6dIB1P33uhCM8+TVAG3zHrXGZad7hDc8=
|
||||||
github.com/guonaihong/gout v0.1.1/go.mod h1:vXvv5Kxr70eM5wrp4F0+t9lnLWmq+YPW2GByll2f/EA=
|
github.com/guonaihong/gout v0.1.3/go.mod h1:vXvv5Kxr70eM5wrp4F0+t9lnLWmq+YPW2GByll2f/EA=
|
||||||
|
github.com/hjson/hjson-go v3.1.0+incompatible h1:DY/9yE8ey8Zv22bY+mHV1uk2yRy0h8tKhZ77hEdi0Aw=
|
||||||
|
github.com/hjson/hjson-go v3.1.0+incompatible/go.mod h1:qsetwF8NlsTsOTwZTApNlTCerV+b2GjYRRcIk4JMFio=
|
||||||
|
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
||||||
github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
||||||
|
github.com/json-iterator/go v1.1.10 h1:Kz6Cvnvv2wGdaG/V8yMvfkmNiXq9Ya2KUv4rouJJr68=
|
||||||
|
github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
||||||
|
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0 h1:iQTw/8FWTuc7uiaSepXwyf3o52HaUYcV+Tu66S3F5GA=
|
||||||
|
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0/go.mod h1:1NbS8ALrpOvjt0rHPNLyCIeMtbizbir8U//inJ+zuB8=
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.3 h1:CE8S1cTafDpPvMhIxNJKvHsGVBgn1xWYf1NbHQhywc8=
|
github.com/kr/binarydist v0.1.0 h1:6kAoLA9FMMnNGSehX0s1PdjbEaACznAv/W219j2uvyo=
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
github.com/kr/binarydist v0.1.0/go.mod h1:DY7S//GCoz1BCd0B0EVrinCKAZN3pXe+MDaIZbXQVgM=
|
||||||
github.com/leodido/go-urn v1.1.0/go.mod h1:+cyI34gQWZcE1eQU7NVgKkkzdXDQHr1dBMtdAPozLkw=
|
github.com/leodido/go-urn v1.1.0/go.mod h1:+cyI34gQWZcE1eQU7NVgKkkzdXDQHr1dBMtdAPozLkw=
|
||||||
github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
|
github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
|
||||||
github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
|
github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
|
||||||
github.com/lestrrat-go/envload v0.0.0-20180220234015-a3eb8ddeffcc/go.mod h1:kopuH9ugFRkIXf3YoqHKyrJ9YfUFsckUU9S7B+XP+is=
|
github.com/lestrrat-go/envload v0.0.0-20180220234015-a3eb8ddeffcc/go.mod h1:kopuH9ugFRkIXf3YoqHKyrJ9YfUFsckUU9S7B+XP+is=
|
||||||
github.com/lestrrat-go/file-rotatelogs v2.3.0+incompatible h1:4mNlp+/SvALIPFpbXV3kxNJJno9iKFWGxSDE13Kl66Q=
|
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible h1:Y6sqxHMyB1D2YSzWkLibYKgg+SwmyFU9dF2hn6MdTj4=
|
||||||
github.com/lestrrat-go/file-rotatelogs v2.3.0+incompatible/go.mod h1:ZQnN8lSECaebrkQytbHj4xNgtg8CR7RYXnPok8e0EHA=
|
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible/go.mod h1:ZQnN8lSECaebrkQytbHj4xNgtg8CR7RYXnPok8e0EHA=
|
||||||
github.com/lestrrat-go/strftime v1.0.1 h1:o7qz5pmLzPDLyGW4lG6JvTKPUfTFXwe+vOamIYWtnVU=
|
github.com/lestrrat-go/strftime v1.0.3 h1:qqOPU7y+TM8Y803I8fG9c/DyKG3xH/xkng6keC1015Q=
|
||||||
github.com/lestrrat-go/strftime v1.0.1/go.mod h1:E1nN3pCbtMSu1yjSVeyuRFVm/U0xoR76fd03sz+Qz4g=
|
github.com/lestrrat-go/strftime v1.0.3/go.mod h1:E1nN3pCbtMSu1yjSVeyuRFVm/U0xoR76fd03sz+Qz4g=
|
||||||
github.com/mattn/go-isatty v0.0.9/go.mod h1:YNRxwqDuOph6SZLI9vUUz6OYw3QyUt7WiY2yME+cCiQ=
|
github.com/mattn/go-isatty v0.0.9/go.mod h1:YNRxwqDuOph6SZLI9vUUz6OYw3QyUt7WiY2yME+cCiQ=
|
||||||
github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY=
|
github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY=
|
||||||
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
|
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742 h1:Esafd1046DLDQ0W1YjYsBW+p8U2u7vzgW2SQVmlNazg=
|
||||||
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
||||||
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ=
|
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ=
|
||||||
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8=
|
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8=
|
||||||
|
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
|
github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
|
github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
||||||
|
github.com/oxtoacart/bpool v0.0.0-20190530202638-03653db5a59c h1:rp5dCmg/yLR3mgFuSOe4oEnDDmGLROTvMragMUXpTQw=
|
||||||
|
github.com/oxtoacart/bpool v0.0.0-20190530202638-03653db5a59c/go.mod h1:X07ZCGwUbLaax7L0S3Tw4hpejzu63ZrrQiUe6W0hcy0=
|
||||||
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
||||||
|
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5 h1:mZHayPoR0lNmnHyvtYjDeq0zlVHn9K/ZXoy17ylucdo=
|
||||||
|
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5/go.mod h1:GEXHk5HgEKCvEIIrSpFI3ozzG5xOKA2DVlEX/gGnewM=
|
||||||
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
||||||
github.com/sirupsen/logrus v1.6.0 h1:UBcNElsrwanuuMsnGSlYmtmgbb23qDR5dG+6X6Oo89I=
|
github.com/sirupsen/logrus v1.7.0 h1:ShrD1U9pZB12TX0cVy0DtePoCH97K8EtX+mg7ZARUtM=
|
||||||
github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
|
github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
||||||
|
github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE=
|
||||||
|
github.com/syndtr/goleveldb v1.0.0/go.mod h1:ZVVdQEZoIme9iO1Ch2Jdy24qqXrMMOU6lpPAyBWyWuQ=
|
||||||
github.com/t-tomalak/logrus-easy-formatter v0.0.0-20190827215021-c074f06c5816 h1:J6v8awz+me+xeb/cUTotKgceAYouhIB3pjzgRd6IlGk=
|
github.com/t-tomalak/logrus-easy-formatter v0.0.0-20190827215021-c074f06c5816 h1:J6v8awz+me+xeb/cUTotKgceAYouhIB3pjzgRd6IlGk=
|
||||||
github.com/t-tomalak/logrus-easy-formatter v0.0.0-20190827215021-c074f06c5816/go.mod h1:tzym/CEb5jnFI+Q0k4Qq3+LvRF4gO3E2pxS8fHP8jcA=
|
github.com/t-tomalak/logrus-easy-formatter v0.0.0-20190827215021-c074f06c5816/go.mod h1:tzym/CEb5jnFI+Q0k4Qq3+LvRF4gO3E2pxS8fHP8jcA=
|
||||||
github.com/tidwall/gjson v1.6.0 h1:9VEQWz6LLMUsUl6PueE49ir4Ka6CzLymOAZDxpFsTDc=
|
github.com/tidwall/gjson v1.6.3 h1:aHoiiem0dr7GHkW001T1SMTJ7X5PvyekH5WX0whWGnI=
|
||||||
github.com/tidwall/gjson v1.6.0/go.mod h1:P256ACg0Mn+j1RXIDXoss50DeIABTYK1PULOJHhxOls=
|
github.com/tidwall/gjson v1.6.3/go.mod h1:BaHyNc5bjzYkPqgLq7mdVzeiRtULKULXLgZFKsxEHI0=
|
||||||
github.com/tidwall/match v1.0.1 h1:PnKP62LPNxHKTwvHHZZzdOAOCtsJTjo6dZLCwpKm5xc=
|
github.com/tidwall/match v1.0.1 h1:PnKP62LPNxHKTwvHHZZzdOAOCtsJTjo6dZLCwpKm5xc=
|
||||||
github.com/tidwall/match v1.0.1/go.mod h1:LujAq0jyVjBy028G1WhWfIzbpQfMO8bBZ6Tyb0+pL9E=
|
github.com/tidwall/match v1.0.1/go.mod h1:LujAq0jyVjBy028G1WhWfIzbpQfMO8bBZ6Tyb0+pL9E=
|
||||||
github.com/tidwall/pretty v1.0.0 h1:HsD+QiTn7sK6flMKIvNmpqz1qrpP3Ps6jOKIKMooyg4=
|
github.com/tidwall/pretty v1.0.2 h1:Z7S3cePv9Jwm1KwS0513MRaoUe3S01WPbLNV40pwWZU=
|
||||||
github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
|
github.com/tidwall/pretty v1.0.2/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
|
||||||
github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo=
|
github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo=
|
||||||
github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
|
github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
|
||||||
github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
|
github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
|
||||||
github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
|
github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
|
||||||
github.com/xujiajun/gorouter v1.2.0/go.mod h1:yJrIta+bTNpBM/2UT8hLOaEAFckO+m/qmR3luMIQygM=
|
github.com/wdvxdr1123/go-silk v0.0.0-20201007123416-b982fd3d91d6 h1:lX18MCdNzT2zIi7K02x4C5cPkDXpL+wCb1YTAMXjLWQ=
|
||||||
github.com/xujiajun/mmap-go v1.0.1 h1:7Se7ss1fLPPRW+ePgqGpCkfGIZzJV6JPq9Wq9iv/WHc=
|
github.com/wdvxdr1123/go-silk v0.0.0-20201007123416-b982fd3d91d6/go.mod h1:5q9LFlBr+yX/J8Jd/9wHdXwkkjFkNyQIS7kX2Lgx/Zs=
|
||||||
github.com/xujiajun/mmap-go v1.0.1/go.mod h1:CNN6Sw4SL69Sui00p0zEzcZKbt+5HtEnYUsc6BKKRMg=
|
|
||||||
github.com/xujiajun/nutsdb v0.5.0 h1:j/jM3Zw7Chg8WK7bAcKR0Xr7Mal47U1oJAMgySfDn9E=
|
|
||||||
github.com/xujiajun/nutsdb v0.5.0/go.mod h1:owdwN0tW084RxEodABLbO7h4Z2s9WiAjZGZFhRh0/1Q=
|
|
||||||
github.com/xujiajun/utils v0.0.0-20190123093513-8bf096c4f53b h1:jKG9OiL4T4xQN3IUrhUpc1tG+HfDXppkgVcrAiiaI/0=
|
|
||||||
github.com/xujiajun/utils v0.0.0-20190123093513-8bf096c4f53b/go.mod h1:AZd87GYJlUzl82Yab2kTjx1EyXSQCAfZDhpTo1SQC4k=
|
|
||||||
github.com/yinghau76/go-ascii-art v0.0.0-20190517192627-e7f465a30189 h1:4UJw9if55Fu3HOwbfcaQlJ27p3oeJU2JZqoeT3ITJQk=
|
github.com/yinghau76/go-ascii-art v0.0.0-20190517192627-e7f465a30189 h1:4UJw9if55Fu3HOwbfcaQlJ27p3oeJU2JZqoeT3ITJQk=
|
||||||
github.com/yinghau76/go-ascii-art v0.0.0-20190517192627-e7f465a30189/go.mod h1:rIrm5geMiBhPQkdfUm8gDFi/WiHneOp1i9KjmJqc+9I=
|
github.com/yinghau76/go-ascii-art v0.0.0-20190517192627-e7f465a30189/go.mod h1:rIrm5geMiBhPQkdfUm8gDFi/WiHneOp1i9KjmJqc+9I=
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
@ -113,6 +143,7 @@ golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvx
|
|||||||
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
||||||
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa h1:F+8P+gmewFQYRk6JoLQLwjBCTu3mcIURZfNkVweuRKA=
|
golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa h1:F+8P+gmewFQYRk6JoLQLwjBCTu3mcIURZfNkVweuRKA=
|
||||||
@ -122,15 +153,17 @@ golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJ
|
|||||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20181221143128-b4a75ba826a6/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20200116001909-b77594299b42 h1:vEOn+mP2zCOVzKckCZy6YsCtDblrpj/w7B9nxGNELpg=
|
||||||
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae h1:Ih9Yo4hSPImZOpfGuA4bR/ORKTAbhZo2AbWNRCnevdo=
|
|
||||||
golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
||||||
|
golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e h1:EHBhcS0mlXEAVwNyO2dLfjToGsyY4j24pTs2ScHnX7s=
|
||||||
|
golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
||||||
@ -156,8 +189,11 @@ google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpAD
|
|||||||
google.golang.org/protobuf v1.25.0 h1:Ejskq+SyPohKW+1uil0JJMtmHCgJPJ/qWTxr8qp+R4c=
|
google.golang.org/protobuf v1.25.0 h1:Ejskq+SyPohKW+1uil0JJMtmHCgJPJ/qWTxr8qp+R4c=
|
||||||
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
|
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
|
||||||
gopkg.in/go-playground/assert.v1 v1.2.1/go.mod h1:9RXL0bg/zibRAgZUYszZSwO/z8Y/a8bDuhia5mkpMnE=
|
gopkg.in/go-playground/assert.v1 v1.2.1/go.mod h1:9RXL0bg/zibRAgZUYszZSwO/z8Y/a8bDuhia5mkpMnE=
|
||||||
gopkg.in/go-playground/validator.v9 v9.29.1/go.mod h1:+c9/zcJMFNgbLvly1L1V+PpxWdVbfP1avr/N00E2vyQ=
|
gopkg.in/go-playground/validator.v9 v9.29.1/go.mod h1:+c9/zcJMFNgbLvly1L1V+PpxWdVbfP1avr/N00E2vyQ=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
||||||
|
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
||||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
|
466
main.go
466
main.go
@ -2,28 +2,40 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"bytes"
|
"crypto/md5"
|
||||||
"encoding/json"
|
"encoding/base64"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/Mrs4s/MiraiGo/client"
|
|
||||||
"github.com/Mrs4s/go-cqhttp/coolq"
|
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
|
||||||
"github.com/Mrs4s/go-cqhttp/server"
|
|
||||||
rotatelogs "github.com/lestrrat-go/file-rotatelogs"
|
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
easy "github.com/t-tomalak/logrus-easy-formatter"
|
|
||||||
asciiart "github.com/yinghau76/go-ascii-art"
|
|
||||||
"image"
|
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"os/exec"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
"path"
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
"runtime"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/go-cqhttp/server"
|
||||||
|
"github.com/guonaihong/gout"
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/MiraiGo/binary"
|
||||||
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
"github.com/getlantern/go-update"
|
||||||
|
jsoniter "github.com/json-iterator/go"
|
||||||
|
rotatelogs "github.com/lestrrat-go/file-rotatelogs"
|
||||||
|
"github.com/rifflock/lfshook"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
easy "github.com/t-tomalak/logrus-easy-formatter"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
log.SetFormatter(&easy.Formatter{
|
log.SetFormatter(&easy.Formatter{
|
||||||
TimestampFormat: "2006-01-02 15:04:05",
|
TimestampFormat: "2006-01-02 15:04:05",
|
||||||
@ -33,17 +45,29 @@ func init() {
|
|||||||
if err == nil {
|
if err == nil {
|
||||||
log.SetOutput(io.MultiWriter(os.Stderr, w))
|
log.SetOutput(io.MultiWriter(os.Stderr, w))
|
||||||
}
|
}
|
||||||
if !global.PathExists("data") {
|
if !global.PathExists(global.IMAGE_PATH) {
|
||||||
if err := os.Mkdir("data", 0777); err != nil {
|
if err := os.MkdirAll(global.IMAGE_PATH, 0755); err != nil {
|
||||||
log.Fatalf("创建数据文件夹失败: %v", err)
|
|
||||||
}
|
|
||||||
if err := os.Mkdir(path.Join("data", "images"), 0777); err != nil {
|
|
||||||
log.Fatalf("创建图片缓存文件夹失败: %v", err)
|
log.Fatalf("创建图片缓存文件夹失败: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if !global.PathExists(global.VOICE_PATH) {
|
||||||
|
if err := os.MkdirAll(global.VOICE_PATH, 0755); err != nil {
|
||||||
|
log.Fatalf("创建语音缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !global.PathExists(global.VIDEO_PATH) {
|
||||||
|
if err := os.MkdirAll(global.VIDEO_PATH, 0755); err != nil {
|
||||||
|
log.Fatalf("创建视频缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !global.PathExists(global.CACHE_PATH) {
|
||||||
|
if err := os.MkdirAll(global.CACHE_PATH, 0755); err != nil {
|
||||||
|
log.Fatalf("创建发送图片缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
if global.PathExists("cqhttp.json") {
|
if global.PathExists("cqhttp.json") {
|
||||||
log.Info("发现 cqhttp.json 将在五秒后尝试导入配置,按 Ctrl+C 取消.")
|
log.Info("发现 cqhttp.json 将在五秒后尝试导入配置,按 Ctrl+C 取消.")
|
||||||
log.Warn("警告: 该操作会删除 cqhttp.json 并覆盖 config.json 文件.")
|
log.Warn("警告: 该操作会删除 cqhttp.json 并覆盖 config.hjson 文件.")
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
conf := global.CQHttpApiConfig{}
|
conf := global.CQHttpApiConfig{}
|
||||||
if err := json.Unmarshal([]byte(global.ReadAllText("cqhttp.json")), &conf); err != nil {
|
if err := json.Unmarshal([]byte(global.ReadAllText("cqhttp.json")), &conf); err != nil {
|
||||||
@ -65,8 +89,8 @@ func init() {
|
|||||||
goConf.ReverseServers[0].ReverseEventUrl = conf.WSReverseEventUrl
|
goConf.ReverseServers[0].ReverseEventUrl = conf.WSReverseEventUrl
|
||||||
goConf.ReverseServers[0].ReverseReconnectInterval = conf.WSReverseReconnectInterval
|
goConf.ReverseServers[0].ReverseReconnectInterval = conf.WSReverseReconnectInterval
|
||||||
}
|
}
|
||||||
if err := goConf.Save("config.json"); err != nil {
|
if err := goConf.Save("config.hjson"); err != nil {
|
||||||
log.Fatalf("保存 config.json 时出现错误: %v", err)
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
}
|
}
|
||||||
_ = os.Remove("cqhttp.json")
|
_ = os.Remove("cqhttp.json")
|
||||||
}
|
}
|
||||||
@ -74,9 +98,35 @@ func init() {
|
|||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
console := bufio.NewReader(os.Stdin)
|
console := bufio.NewReader(os.Stdin)
|
||||||
|
var strKey string
|
||||||
|
var isFastStart bool = false
|
||||||
|
arg := os.Args
|
||||||
|
if len(arg) > 1 {
|
||||||
|
for i := range arg {
|
||||||
|
switch arg[i] {
|
||||||
|
case "update":
|
||||||
|
if len(arg) > i+1 {
|
||||||
|
selfUpdate(arg[i+1])
|
||||||
|
} else {
|
||||||
|
selfUpdate("")
|
||||||
|
}
|
||||||
|
case "key":
|
||||||
|
if len(arg) > i+1 {
|
||||||
|
b := []byte(arg[i+1])
|
||||||
|
b = append(b, 13, 10)
|
||||||
|
strKey = string(b[:])
|
||||||
|
}
|
||||||
|
case "faststart":
|
||||||
|
isFastStart = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var conf *global.JsonConfig
|
var conf *global.JsonConfig
|
||||||
if global.PathExists("config.json") || os.Getenv("UIN") == "" {
|
if global.PathExists("config.json") {
|
||||||
conf = global.Load("config.json")
|
conf = global.Load("config.json")
|
||||||
|
_ = conf.Save("config.hjson")
|
||||||
|
_ = os.Remove("config.json")
|
||||||
} else if os.Getenv("UIN") != "" {
|
} else if os.Getenv("UIN") != "" {
|
||||||
log.Infof("将从环境变量加载配置.")
|
log.Infof("将从环境变量加载配置.")
|
||||||
uin, _ := strconv.ParseInt(os.Getenv("UIN"), 10, 64)
|
uin, _ := strconv.ParseInt(os.Getenv("UIN"), 10, 64)
|
||||||
@ -96,35 +146,79 @@ func main() {
|
|||||||
Host: "0.0.0.0",
|
Host: "0.0.0.0",
|
||||||
Port: 6700,
|
Port: 6700,
|
||||||
},
|
},
|
||||||
Debug: os.Getenv("DEBUG") == "true",
|
PostMessageFormat: "string",
|
||||||
|
Debug: os.Getenv("DEBUG") == "true",
|
||||||
}
|
}
|
||||||
if post != "" {
|
if post != "" {
|
||||||
conf.HttpConfig.PostUrls[post] = os.Getenv("HTTP_SECRET")
|
conf.HttpConfig.PostUrls[post] = os.Getenv("HTTP_SECRET")
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
conf = global.Load("config.hjson")
|
||||||
}
|
}
|
||||||
if conf == nil {
|
if conf == nil {
|
||||||
err := global.DefaultConfig().Save("config.json")
|
err := global.WriteAllText("config.hjson", global.DefaultConfigWithComments)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("创建默认配置文件时出现错误: %v", err)
|
log.Fatalf("创建默认配置文件时出现错误: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
log.Infof("默认配置文件已生成, 请编辑 config.json 后重启程序.")
|
log.Infof("默认配置文件已生成, 请编辑 config.hjson 后重启程序.")
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if conf.Uin == 0 || conf.Password == "" {
|
if conf.Uin == 0 || (conf.Password == "" && conf.PasswordEncrypted == "") {
|
||||||
log.Warnf("请修改 config.json 以添加账号密码.")
|
log.Warnf("请修改 config.hjson 以添加账号密码.")
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// log classified by level
|
||||||
|
// Collect all records up to the specified level (default level: warn)
|
||||||
|
logLevel := conf.LogLevel
|
||||||
|
if logLevel != "" {
|
||||||
|
date := time.Now().Format("2006-01-02")
|
||||||
|
var logPathMap lfshook.PathMap
|
||||||
|
switch conf.LogLevel {
|
||||||
|
case "warn":
|
||||||
|
logPathMap = lfshook.PathMap{
|
||||||
|
log.WarnLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
log.ErrorLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
log.FatalLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
log.PanicLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
}
|
||||||
|
case "error":
|
||||||
|
logPathMap = lfshook.PathMap{
|
||||||
|
log.ErrorLevel: path.Join("logs", date+"-error.log"),
|
||||||
|
log.FatalLevel: path.Join("logs", date+"-error.log"),
|
||||||
|
log.PanicLevel: path.Join("logs", date+"-error.log"),
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
logPathMap = lfshook.PathMap{
|
||||||
|
log.WarnLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
log.ErrorLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
log.FatalLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
log.PanicLevel: path.Join("logs", date+"-warn.log"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.AddHook(lfshook.NewHook(
|
||||||
|
logPathMap,
|
||||||
|
&easy.Formatter{
|
||||||
|
TimestampFormat: "2006-01-02 15:04:05",
|
||||||
|
LogFormat: "[%time%] [%lvl%]: %msg% \n",
|
||||||
|
},
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info("当前版本:", coolq.Version)
|
||||||
if conf.Debug {
|
if conf.Debug {
|
||||||
log.SetLevel(log.DebugLevel)
|
log.SetLevel(log.DebugLevel)
|
||||||
log.Warnf("已开启Debug模式.")
|
log.Warnf("已开启Debug模式.")
|
||||||
|
log.Debugf("开发交流群: 192548878")
|
||||||
}
|
}
|
||||||
if !global.PathExists("device.json") {
|
if !global.PathExists("device.json") {
|
||||||
log.Warn("虚拟设备信息不存在, 将自动生成随机设备.")
|
log.Warn("虚拟设备信息不存在, 将自动生成随机设备.")
|
||||||
client.GenRandomDevice()
|
client.GenRandomDevice()
|
||||||
_ = ioutil.WriteFile("device.json", client.SystemDeviceInfo.ToJson(), 0777)
|
_ = ioutil.WriteFile("device.json", client.SystemDeviceInfo.ToJson(), 0644)
|
||||||
log.Info("已生成设备信息并保存到 device.json 文件.")
|
log.Info("已生成设备信息并保存到 device.json 文件.")
|
||||||
} else {
|
} else {
|
||||||
log.Info("将使用 device.json 内的设备信息运行Bot.")
|
log.Info("将使用 device.json 内的设备信息运行Bot.")
|
||||||
@ -132,81 +226,259 @@ func main() {
|
|||||||
log.Fatalf("加载设备信息失败: %v", err)
|
log.Fatalf("加载设备信息失败: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
log.Info("Bot将在5秒后登录并开始信息处理, 按 Ctrl+C 取消.")
|
if conf.EncryptPassword && conf.PasswordEncrypted == "" {
|
||||||
time.Sleep(time.Second * 5)
|
log.Infof("密码加密已启用, 请输入Key对密码进行加密: (Enter 提交)")
|
||||||
log.Info("开始尝试登录并同步消息...")
|
strKey, _ := console.ReadString('\n')
|
||||||
cli := client.NewClient(conf.Uin, conf.Password)
|
key := md5.Sum([]byte(strKey))
|
||||||
rsp, err := cli.Login()
|
if encrypted := EncryptPwd(conf.Password, key[:]); encrypted != "" {
|
||||||
for {
|
conf.Password = ""
|
||||||
global.Check(err)
|
conf.PasswordEncrypted = encrypted
|
||||||
if !rsp.Success {
|
_ = conf.Save("config.hjson")
|
||||||
switch rsp.Error {
|
} else {
|
||||||
case client.NeedCaptcha:
|
log.Warnf("加密时出现问题.")
|
||||||
img, _, _ := image.Decode(bytes.NewReader(rsp.CaptchaImage))
|
|
||||||
fmt.Println(asciiart.New("image", img).Art)
|
|
||||||
log.Warn("请输入验证码: (Enter 提交)")
|
|
||||||
text, _ := console.ReadString('\n')
|
|
||||||
rsp, err = cli.SubmitCaptcha(strings.ReplaceAll(text, "\n", ""), rsp.CaptchaSign)
|
|
||||||
continue
|
|
||||||
case client.UnsafeDeviceError:
|
|
||||||
log.Warnf("账号已开启设备锁,请前往 -> %v <- 验证并重启Bot.", rsp.VerifyUrl)
|
|
||||||
log.Infof(" 按 Enter 继续....")
|
|
||||||
_, _ = console.ReadString('\n')
|
|
||||||
return
|
|
||||||
case client.OtherLoginError, client.UnknownLoginError:
|
|
||||||
log.Fatalf("登录失败: %v", rsp.ErrorMessage)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
log.Infof("登录成功 欢迎使用: %v", cli.Nickname)
|
|
||||||
time.Sleep(time.Second)
|
|
||||||
log.Info("开始加载好友列表...")
|
|
||||||
global.Check(cli.ReloadFriendList())
|
|
||||||
log.Infof("共加载 %v 个好友.", len(cli.FriendList))
|
|
||||||
log.Infof("开始加载群列表...")
|
|
||||||
global.Check(cli.ReloadGroupList())
|
|
||||||
log.Infof("共加载 %v 个群.", len(cli.GroupList))
|
|
||||||
b := coolq.NewQQBot(cli, conf)
|
|
||||||
if conf.HttpConfig != nil && conf.HttpConfig.Enabled {
|
|
||||||
server.HttpServer.Run(fmt.Sprintf("%s:%d", conf.HttpConfig.Host, conf.HttpConfig.Port), conf.AccessToken, b)
|
|
||||||
for k, v := range conf.HttpConfig.PostUrls {
|
|
||||||
server.NewHttpClient().Run(k, v, b)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if conf.WSConfig != nil && conf.WSConfig.Enabled {
|
if conf.PasswordEncrypted != "" {
|
||||||
server.WebsocketServer.Run(fmt.Sprintf("%s:%d", conf.WSConfig.Host, conf.WSConfig.Port), conf.AccessToken, b)
|
if strKey == "" {
|
||||||
}
|
log.Infof("密码加密已启用, 请输入Key对密码进行解密以继续: (Enter 提交)")
|
||||||
for _, rc := range conf.ReverseServers {
|
cancel := make(chan struct{}, 1)
|
||||||
server.NewWebsocketClient(rc, conf.AccessToken, b).Run()
|
go func() {
|
||||||
}
|
select {
|
||||||
log.Info("资源初始化完成, 开始处理信息.")
|
case <-cancel:
|
||||||
log.Info("アトリは、高性能ですから!")
|
return
|
||||||
cli.OnDisconnected(func(bot *client.QQClient, e *client.ClientDisconnectedEvent) {
|
case <-time.After(time.Second * 45):
|
||||||
if conf.ReLogin {
|
log.Infof("解密key输入超时")
|
||||||
log.Warnf("Bot已离线 (%v),将在 %v 秒后尝试重连.", e.Message, conf.ReLoginDelay)
|
time.Sleep(3 * time.Second)
|
||||||
time.Sleep(time.Second * time.Duration(conf.ReLoginDelay))
|
os.Exit(0)
|
||||||
rsp, err := cli.Login()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("重连失败: %v", err)
|
|
||||||
}
|
|
||||||
if !rsp.Success {
|
|
||||||
switch rsp.Error {
|
|
||||||
case client.NeedCaptcha:
|
|
||||||
log.Fatalf("重连失败: 需要验证码. (验证码处理正在开发中)")
|
|
||||||
case client.UnsafeDeviceError:
|
|
||||||
log.Fatalf("重连失败: 设备锁")
|
|
||||||
default:
|
|
||||||
log.Fatalf("重连失败: %v", rsp.ErrorMessage)
|
|
||||||
}
|
}
|
||||||
|
}()
|
||||||
|
strKey, _ = console.ReadString('\n')
|
||||||
|
cancel <- struct{}{}
|
||||||
|
} else {
|
||||||
|
log.Infof("密码加密已启用, 使用运行时传递的参数进行解密,按 Ctrl+C 取消.")
|
||||||
|
}
|
||||||
|
key := md5.Sum([]byte(strKey))
|
||||||
|
conf.Password = DecryptPwd(conf.PasswordEncrypted, key[:])
|
||||||
|
}
|
||||||
|
if !isFastStart {
|
||||||
|
log.Info("Bot将在5秒后登录并开始信息处理, 按 Ctrl+C 取消.")
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
}
|
||||||
|
log.Info("开始尝试登录并同步消息...")
|
||||||
|
log.Infof("使用协议: %v", func() string {
|
||||||
|
switch client.SystemDeviceInfo.Protocol {
|
||||||
|
case client.IPad:
|
||||||
|
return "iPad"
|
||||||
|
case client.AndroidPhone:
|
||||||
|
return "Android Phone"
|
||||||
|
case client.AndroidWatch:
|
||||||
|
return "Android Watch"
|
||||||
|
case client.MacOS:
|
||||||
|
return "MacOS"
|
||||||
|
}
|
||||||
|
return "未知"
|
||||||
|
}())
|
||||||
|
cli := client.NewClient(conf.Uin, conf.Password)
|
||||||
|
cli.OnLog(func(c *client.QQClient, e *client.LogEvent) {
|
||||||
|
switch e.Type {
|
||||||
|
case "INFO":
|
||||||
|
log.Info("Protocol -> " + e.Message)
|
||||||
|
case "ERROR":
|
||||||
|
log.Error("Protocol -> " + e.Message)
|
||||||
|
case "DEBUG":
|
||||||
|
log.Debug("Protocol -> " + e.Message)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
if global.PathExists("address.txt") {
|
||||||
|
log.Infof("检测到 address.txt 文件. 将覆盖目标IP.")
|
||||||
|
addr := global.ReadAddrFile("address.txt")
|
||||||
|
if len(addr) > 0 {
|
||||||
|
cli.SetCustomServer(addr)
|
||||||
|
}
|
||||||
|
log.Infof("读取到 %v 个自定义地址.", len(addr))
|
||||||
|
}
|
||||||
|
cli.OnServerUpdated(func(bot *client.QQClient, e *client.ServerUpdatedEvent) bool {
|
||||||
|
if !conf.UseSSOAddress {
|
||||||
|
log.Infof("收到服务器地址更新通知, 根据配置文件已忽略.")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
log.Infof("收到服务器地址更新通知, 将在下一次重连时应用. ")
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
if conf.WebUi == nil {
|
||||||
|
conf.WebUi = &global.GoCqWebUi{
|
||||||
|
Enabled: true,
|
||||||
|
WebInput: false,
|
||||||
|
Host: "0.0.0.0",
|
||||||
|
WebUiPort: 9999,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if conf.WebUi.WebUiPort <= 0 {
|
||||||
|
conf.WebUi.WebUiPort = 9999
|
||||||
|
}
|
||||||
|
if conf.WebUi.Host == "" {
|
||||||
|
conf.WebUi.Host = "127.0.0.1"
|
||||||
|
}
|
||||||
|
global.Proxy = conf.ProxyRewrite
|
||||||
|
b := server.WebServer.Run(fmt.Sprintf("%s:%d", conf.WebUi.Host, conf.WebUi.WebUiPort), cli)
|
||||||
|
c := server.Console
|
||||||
|
r := server.Restart
|
||||||
|
go checkUpdate()
|
||||||
|
signal.Notify(c, os.Interrupt, os.Kill)
|
||||||
|
select {
|
||||||
|
case <-c:
|
||||||
|
b.Release()
|
||||||
|
case <-r:
|
||||||
|
log.Info("正在重启中...")
|
||||||
|
defer b.Release()
|
||||||
|
restart(arg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func EncryptPwd(pwd string, key []byte) string {
|
||||||
|
tea := binary.NewTeaCipher(key)
|
||||||
|
if tea == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return base64.StdEncoding.EncodeToString(tea.Encrypt([]byte(pwd)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func DecryptPwd(ePwd string, key []byte) string {
|
||||||
|
defer func() {
|
||||||
|
if pan := recover(); pan != nil {
|
||||||
|
log.Fatalf("密码解密失败: %v", pan)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
encrypted, err := base64.StdEncoding.DecodeString(ePwd)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
tea := binary.NewTeaCipher(key)
|
||||||
|
if tea == nil {
|
||||||
|
panic("密钥错误")
|
||||||
|
}
|
||||||
|
return string(tea.Decrypt(encrypted))
|
||||||
|
}
|
||||||
|
|
||||||
|
func checkUpdate() {
|
||||||
|
log.Infof("正在检查更新.")
|
||||||
|
if coolq.Version == "unknown" {
|
||||||
|
log.Warnf("检查更新失败: 使用的 Actions 测试版或自编译版本.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
var res string
|
||||||
|
if err := gout.GET("https://api.github.com/repos/Mrs4s/go-cqhttp/releases").BindBody(&res).Do(); err != nil {
|
||||||
|
log.Warnf("检查更新失败: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
detail := gjson.Parse(res)
|
||||||
|
if len(detail.Array()) < 1 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
info := detail.Array()[0]
|
||||||
|
if global.VersionNameCompare(coolq.Version, info.Get("tag_name").Str) {
|
||||||
|
log.Infof("当前有更新的 go-cqhttp 可供更新, 请前往 https://github.com/Mrs4s/go-cqhttp/releases 下载.")
|
||||||
|
log.Infof("当前版本: %v 最新版本: %v", coolq.Version, info.Get("tag_name").Str)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Infof("检查更新完成. 当前已运行最新版本.")
|
||||||
|
}
|
||||||
|
|
||||||
|
func selfUpdate(imageUrl string) {
|
||||||
|
console := bufio.NewReader(os.Stdin)
|
||||||
|
readLine := func() (str string) {
|
||||||
|
str, _ = console.ReadString('\n')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Infof("正在检查更新.")
|
||||||
|
var res string
|
||||||
|
if err := gout.GET("https://api.github.com/repos/Mrs4s/go-cqhttp/releases").BindBody(&res).Do(); err != nil {
|
||||||
|
log.Warnf("检查更新失败: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
detail := gjson.Parse(res)
|
||||||
|
if len(detail.Array()) < 1 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
info := detail.Array()[0]
|
||||||
|
version := info.Get("tag_name").Str
|
||||||
|
if coolq.Version != version {
|
||||||
|
log.Info("当前最新版本为 ", version)
|
||||||
|
log.Warn("是否更新(y/N): ")
|
||||||
|
r := strings.TrimSpace(readLine())
|
||||||
|
|
||||||
|
doUpdate := func() {
|
||||||
|
log.Info("正在更新,请稍等...")
|
||||||
|
url := fmt.Sprintf(
|
||||||
|
"%v/Mrs4s/go-cqhttp/releases/download/%v/go-cqhttp-%v-%v-%v",
|
||||||
|
func() string {
|
||||||
|
if imageUrl != "" {
|
||||||
|
return imageUrl
|
||||||
|
}
|
||||||
|
return "https://github.com"
|
||||||
|
}(),
|
||||||
|
version,
|
||||||
|
version,
|
||||||
|
runtime.GOOS,
|
||||||
|
runtime.GOARCH,
|
||||||
|
)
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
url = url + ".exe"
|
||||||
}
|
}
|
||||||
|
resp, err := http.Get(url)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
log.Error("更新失败!")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
wc := global.WriteCounter{}
|
||||||
|
err, _ = update.New().FromStream(io.TeeReader(resp.Body, &wc))
|
||||||
|
fmt.Println()
|
||||||
|
if err != nil {
|
||||||
|
log.Error("更新失败!")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Info("更新完成!")
|
||||||
|
}
|
||||||
|
|
||||||
|
if r == "y" || r == "Y" {
|
||||||
|
doUpdate()
|
||||||
|
} else {
|
||||||
|
log.Warn("已取消更新!")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Info("按 Enter 继续....")
|
||||||
|
readLine()
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func restart(Args []string) {
|
||||||
|
cmd := &exec.Cmd{}
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
file, err := exec.LookPath(Args[0])
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("重启失败:%s", err.Error())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
b.Release()
|
path, err := filepath.Abs(file)
|
||||||
log.Fatalf("Bot已离线:%v", e.Message)
|
if err != nil {
|
||||||
})
|
log.Errorf("重启失败:%s", err.Error())
|
||||||
c := make(chan os.Signal, 1)
|
}
|
||||||
signal.Notify(c, os.Interrupt, os.Kill)
|
Args = append([]string{"/c", "start ", path, "faststart"}, Args[1:]...)
|
||||||
<-c
|
cmd = &exec.Cmd{
|
||||||
b.Release()
|
Path: "cmd.exe",
|
||||||
|
Args: Args,
|
||||||
|
Stderr: os.Stderr,
|
||||||
|
Stdout: os.Stdout,
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Args = append(Args, "faststart")
|
||||||
|
cmd = &exec.Cmd{
|
||||||
|
Path: Args[0],
|
||||||
|
Args: Args,
|
||||||
|
Stderr: os.Stderr,
|
||||||
|
Stdout: os.Stdout,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cmd.Start()
|
||||||
}
|
}
|
||||||
|
587
server/apiAdmin.go
Normal file
587
server/apiAdmin.go
Normal file
@ -0,0 +1,587 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"bytes"
|
||||||
|
"encoding/base64"
|
||||||
|
"fmt"
|
||||||
|
"image"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"os/signal"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
jsoniter "github.com/json-iterator/go"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
|
asciiart "github.com/yinghau76/go-ascii-art"
|
||||||
|
)
|
||||||
|
|
||||||
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
|
var WebInput = make(chan string, 1) //长度1,用于阻塞
|
||||||
|
|
||||||
|
var Console = make(chan os.Signal, 1)
|
||||||
|
|
||||||
|
var Restart = make(chan struct{}, 1)
|
||||||
|
|
||||||
|
var JsonConfig *global.JsonConfig
|
||||||
|
|
||||||
|
type webServer struct {
|
||||||
|
engine *gin.Engine
|
||||||
|
bot *coolq.CQBot
|
||||||
|
Cli *client.QQClient
|
||||||
|
Conf *global.JsonConfig //old config
|
||||||
|
Console *bufio.Reader
|
||||||
|
}
|
||||||
|
|
||||||
|
var WebServer = &webServer{}
|
||||||
|
|
||||||
|
// admin 子站的 路由映射
|
||||||
|
var HttpuriAdmin = map[string]func(s *webServer, c *gin.Context){
|
||||||
|
"do_restart": AdminDoRestart, //热重启
|
||||||
|
"do_process_restart": AdminProcessRestart, //进程重启
|
||||||
|
"get_web_write": AdminWebWrite, //获取是否验证码输入
|
||||||
|
"do_web_write": AdminDoWebWrite, //web上进行输入操作
|
||||||
|
"do_restart_docker": AdminDoRestartDocker, //直接停止(依赖supervisord/docker)重新拉起
|
||||||
|
"do_config_base": AdminDoConfigBase, //修改config.json中的基础部分
|
||||||
|
"do_config_http": AdminDoConfigHttp, //修改config.json的http部分
|
||||||
|
"do_config_ws": AdminDoConfigWs, //修改config.json的正向ws部分
|
||||||
|
"do_config_reverse": AdminDoConfigReverse, //修改config.json 中的反向ws部分
|
||||||
|
"do_config_json": AdminDoConfigJson, //直接修改 config.json配置
|
||||||
|
"get_config_json": AdminGetConfigJson, //拉取 当前的config.json配置
|
||||||
|
}
|
||||||
|
|
||||||
|
func Failed(code int, msg string) coolq.MSG {
|
||||||
|
return coolq.MSG{"data": nil, "retcode": code, "status": "failed", "msg": msg}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *webServer) Run(addr string, cli *client.QQClient) *coolq.CQBot {
|
||||||
|
s.Cli = cli
|
||||||
|
s.Conf = GetConf()
|
||||||
|
JsonConfig = s.Conf
|
||||||
|
gin.SetMode(gin.ReleaseMode)
|
||||||
|
s.engine = gin.New()
|
||||||
|
|
||||||
|
s.engine.Use(AuthMiddleWare())
|
||||||
|
|
||||||
|
//通用路由
|
||||||
|
s.engine.Any("/admin/:action", s.admin)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
//开启端口监听
|
||||||
|
if s.Conf.WebUi != nil && s.Conf.WebUi.Enabled {
|
||||||
|
log.Infof("Admin API 服务器已启动: %v", addr)
|
||||||
|
err := s.engine.Run(addr)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(err)
|
||||||
|
log.Infof("请检查端口是否被占用.")
|
||||||
|
c := make(chan os.Signal, 1)
|
||||||
|
signal.Notify(c, os.Interrupt, os.Kill)
|
||||||
|
<-c
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
//关闭端口监听
|
||||||
|
c := make(chan os.Signal, 1)
|
||||||
|
signal.Notify(c, os.Interrupt, os.Kill)
|
||||||
|
<-c
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
s.Dologin()
|
||||||
|
s.UpServer()
|
||||||
|
b := s.bot //外部引入 bot对象,用于操作bot
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *webServer) Dologin() {
|
||||||
|
s.Console = bufio.NewReader(os.Stdin)
|
||||||
|
readLine := func() (str string) {
|
||||||
|
str, _ = s.Console.ReadString('\n')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
conf := GetConf()
|
||||||
|
cli := s.Cli
|
||||||
|
cli.AllowSlider = true
|
||||||
|
rsp, err := cli.Login()
|
||||||
|
count := 0
|
||||||
|
for {
|
||||||
|
global.Check(err)
|
||||||
|
var text string
|
||||||
|
if !rsp.Success {
|
||||||
|
switch rsp.Error {
|
||||||
|
case client.SliderNeededError:
|
||||||
|
if client.SystemDeviceInfo.Protocol == client.AndroidPhone {
|
||||||
|
log.Warnf("警告: Android Phone 强制要求暂不支持的滑条验证码, 请开启设备锁或切换到Watch协议验证通过后再使用.")
|
||||||
|
log.Infof("按 Enter 继续....")
|
||||||
|
readLine()
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
cli.AllowSlider = false
|
||||||
|
cli.Disconnect()
|
||||||
|
rsp, err = cli.Login()
|
||||||
|
continue
|
||||||
|
case client.NeedCaptcha:
|
||||||
|
_ = ioutil.WriteFile("captcha.jpg", rsp.CaptchaImage, 0644)
|
||||||
|
img, _, _ := image.Decode(bytes.NewReader(rsp.CaptchaImage))
|
||||||
|
fmt.Println(asciiart.New("image", img).Art)
|
||||||
|
if conf.WebUi != nil && conf.WebUi.WebInput {
|
||||||
|
log.Warnf("请输入验证码 (captcha.jpg): (http://%s:%d/admin/do_web_write 输入)", conf.WebUi.Host, conf.WebUi.WebUiPort)
|
||||||
|
text = <-WebInput
|
||||||
|
} else {
|
||||||
|
log.Warn("请输入验证码 (captcha.jpg): (Enter 提交)")
|
||||||
|
text = readLine()
|
||||||
|
}
|
||||||
|
rsp, err = cli.SubmitCaptcha(strings.ReplaceAll(text, "\n", ""), rsp.CaptchaSign)
|
||||||
|
global.DelFile("captcha.jpg")
|
||||||
|
continue
|
||||||
|
case client.SMSNeededError:
|
||||||
|
log.Warnf("账号已开启设备锁, 按下 Enter 向手机 %v 发送短信验证码.", rsp.SMSPhone)
|
||||||
|
readLine()
|
||||||
|
if !cli.RequestSMS() {
|
||||||
|
log.Warnf("发送验证码失败,可能是请求过于频繁.")
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
log.Warn("请输入短信验证码: (Enter 提交)")
|
||||||
|
text = readLine()
|
||||||
|
rsp, err = cli.SubmitSMS(strings.ReplaceAll(strings.ReplaceAll(text, "\n", ""), "\r", ""))
|
||||||
|
continue
|
||||||
|
case client.SMSOrVerifyNeededError:
|
||||||
|
log.Warnf("账号已开启设备锁,请选择验证方式:")
|
||||||
|
log.Warnf("1. 向手机 %v 发送短信验证码", rsp.SMSPhone)
|
||||||
|
log.Warnf("2. 使用手机QQ扫码验证.")
|
||||||
|
log.Warn("请输入(1 - 2): ")
|
||||||
|
text = readLine()
|
||||||
|
if strings.Contains(text, "1") {
|
||||||
|
if !cli.RequestSMS() {
|
||||||
|
log.Warnf("发送验证码失败,可能是请求过于频繁.")
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
log.Warn("请输入短信验证码: (Enter 提交)")
|
||||||
|
text = readLine()
|
||||||
|
rsp, err = cli.SubmitSMS(strings.ReplaceAll(strings.ReplaceAll(text, "\n", ""), "\r", ""))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
log.Warnf("请前往 -> %v <- 验证并重启Bot.", rsp.VerifyUrl)
|
||||||
|
log.Infof("按 Enter 继续....")
|
||||||
|
readLine()
|
||||||
|
os.Exit(0)
|
||||||
|
return
|
||||||
|
case client.UnsafeDeviceError:
|
||||||
|
log.Warnf("账号已开启设备锁,请前往 -> %v <- 验证并重启Bot.", rsp.VerifyUrl)
|
||||||
|
if conf.WebUi != nil && conf.WebUi.WebInput {
|
||||||
|
log.Infof(" (http://%s:%d/admin/do_web_write 确认后继续)....", conf.WebUi.Host, conf.WebUi.WebUiPort)
|
||||||
|
text = <-WebInput
|
||||||
|
} else {
|
||||||
|
log.Infof("按 Enter 继续....")
|
||||||
|
readLine()
|
||||||
|
}
|
||||||
|
log.Info(text)
|
||||||
|
os.Exit(0)
|
||||||
|
return
|
||||||
|
case client.OtherLoginError, client.UnknownLoginError:
|
||||||
|
msg := rsp.ErrorMessage
|
||||||
|
if strings.Contains(msg, "版本") {
|
||||||
|
msg = "密码错误或账号被冻结"
|
||||||
|
}
|
||||||
|
if strings.Contains(msg, "上网环境") && count < 5 {
|
||||||
|
cli.Disconnect()
|
||||||
|
rsp, err = cli.Login()
|
||||||
|
count++
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
log.Warnf("登录失败: %v", msg)
|
||||||
|
log.Infof("按 Enter 继续....")
|
||||||
|
readLine()
|
||||||
|
os.Exit(0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
log.Infof("登录成功 欢迎使用: %v", cli.Nickname)
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
log.Info("开始加载好友列表...")
|
||||||
|
global.Check(cli.ReloadFriendList())
|
||||||
|
log.Infof("共加载 %v 个好友.", len(cli.FriendList))
|
||||||
|
log.Infof("开始加载群列表...")
|
||||||
|
global.Check(cli.ReloadGroupList())
|
||||||
|
log.Infof("共加载 %v 个群.", len(cli.GroupList))
|
||||||
|
s.bot = coolq.NewQQBot(cli, conf)
|
||||||
|
if conf.PostMessageFormat != "string" && conf.PostMessageFormat != "array" {
|
||||||
|
log.Warnf("post_message_format 配置错误, 将自动使用 string")
|
||||||
|
coolq.SetMessageFormat("string")
|
||||||
|
} else {
|
||||||
|
coolq.SetMessageFormat(conf.PostMessageFormat)
|
||||||
|
}
|
||||||
|
if conf.RateLimit.Enabled {
|
||||||
|
global.InitLimiter(conf.RateLimit.Frequency, conf.RateLimit.BucketSize)
|
||||||
|
}
|
||||||
|
log.Info("正在加载事件过滤器.")
|
||||||
|
global.BootFilter()
|
||||||
|
global.InitCodec()
|
||||||
|
coolq.IgnoreInvalidCQCode = conf.IgnoreInvalidCQCode
|
||||||
|
coolq.ForceFragmented = conf.ForceFragmented
|
||||||
|
log.Info("资源初始化完成, 开始处理信息.")
|
||||||
|
log.Info("アトリは、高性能ですから!")
|
||||||
|
cli.OnDisconnected(func(bot *client.QQClient, e *client.ClientDisconnectedEvent) {
|
||||||
|
if conf.ReLogin.Enabled {
|
||||||
|
var times uint = 1
|
||||||
|
for {
|
||||||
|
if cli.Online {
|
||||||
|
log.Warn("Bot已登录")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if times > conf.ReLogin.MaxReloginTimes && conf.ReLogin.MaxReloginTimes != 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
log.Warnf("Bot已离线 (%v),将在 %v 秒后尝试重连. 重连次数:%v",
|
||||||
|
e.Message, conf.ReLogin.ReLoginDelay, times)
|
||||||
|
times++
|
||||||
|
time.Sleep(time.Second * time.Duration(conf.ReLogin.ReLoginDelay))
|
||||||
|
rsp, err := cli.Login()
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("重连失败: %v", err)
|
||||||
|
cli.Disconnect()
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if !rsp.Success {
|
||||||
|
switch rsp.Error {
|
||||||
|
case client.NeedCaptcha:
|
||||||
|
log.Fatalf("重连失败: 需要验证码. (验证码处理正在开发中)")
|
||||||
|
case client.UnsafeDeviceError:
|
||||||
|
log.Fatalf("重连失败: 设备锁")
|
||||||
|
default:
|
||||||
|
log.Errorf("重连失败: %v", rsp.ErrorMessage)
|
||||||
|
cli.Disconnect()
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Info("重连成功")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Fatal("重连失败: 重连次数达到设置的上限值")
|
||||||
|
}
|
||||||
|
s.bot.Release()
|
||||||
|
log.Fatalf("Bot已离线:%v", e.Message)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *webServer) admin(c *gin.Context) {
|
||||||
|
action := c.Param("action")
|
||||||
|
log.Debugf("WebServer接收到cgi调用: %v", action)
|
||||||
|
if f, ok := HttpuriAdmin[action]; ok {
|
||||||
|
f(s, c)
|
||||||
|
} else {
|
||||||
|
c.JSON(200, coolq.Failed(404))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取当前配置文件信息
|
||||||
|
func GetConf() *global.JsonConfig {
|
||||||
|
if JsonConfig != nil {
|
||||||
|
return JsonConfig
|
||||||
|
}
|
||||||
|
conf := global.Load("config.hjson")
|
||||||
|
return conf
|
||||||
|
}
|
||||||
|
|
||||||
|
// admin 控制器 登录验证
|
||||||
|
func AuthMiddleWare() gin.HandlerFunc {
|
||||||
|
return func(c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
//处理跨域问题
|
||||||
|
c.Header("Access-Control-Allow-Origin", "*")
|
||||||
|
c.Header("Access-Control-Allow-Headers", "Content-Type,AccessToken,X-CSRF-Token, Authorization, Token")
|
||||||
|
c.Header("Access-Control-Allow-Methods", "POST, GET, OPTIONS, PUT, PATCH, DELETE")
|
||||||
|
c.Header("Access-Control-Expose-Headers", "Content-Length, Access-Control-Allow-Origin, Access-Control-Allow-Headers, Content-Type")
|
||||||
|
c.Header("Access-Control-Allow-Credentials", "true")
|
||||||
|
// 放行所有OPTIONS方法,因为有的模板是要请求两次的
|
||||||
|
if c.Request.Method == "OPTIONS" {
|
||||||
|
c.AbortWithStatus(http.StatusNoContent)
|
||||||
|
}
|
||||||
|
// 处理请求
|
||||||
|
if c.Request.Method != "GET" && c.Request.Method != "POST" {
|
||||||
|
log.Warnf("已拒绝客户端 %v 的请求: 方法错误", c.Request.RemoteAddr)
|
||||||
|
c.Status(404)
|
||||||
|
c.Abort()
|
||||||
|
}
|
||||||
|
if c.Request.Method == "POST" && strings.Contains(c.Request.Header.Get("Content-Type"), "application/json") {
|
||||||
|
d, err := c.GetRawData()
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("获取请求 %v 的Body时出现错误: %v", c.Request.RequestURI, err)
|
||||||
|
c.Status(400)
|
||||||
|
c.Abort()
|
||||||
|
}
|
||||||
|
if !gjson.ValidBytes(d) {
|
||||||
|
log.Warnf("已拒绝客户端 %v 的请求: 非法Json", c.Request.RemoteAddr)
|
||||||
|
c.Status(400)
|
||||||
|
c.Abort()
|
||||||
|
}
|
||||||
|
c.Set("json_body", gjson.ParseBytes(d))
|
||||||
|
}
|
||||||
|
authToken := conf.AccessToken
|
||||||
|
if auth := c.Request.Header.Get("Authorization"); auth != "" {
|
||||||
|
if strings.SplitN(auth, " ", 2)[1] != authToken {
|
||||||
|
c.AbortWithStatus(401)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
} else if c.Query("access_token") != authToken {
|
||||||
|
c.AbortWithStatus(401)
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
c.Next()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *webServer) DoReLogin() { // TODO: 协议层的 ReLogin
|
||||||
|
JsonConfig = nil
|
||||||
|
conf := GetConf()
|
||||||
|
OldConf := s.Conf
|
||||||
|
cli := client.NewClient(conf.Uin, conf.Password)
|
||||||
|
log.Info("开始尝试登录并同步消息...")
|
||||||
|
log.Infof("使用协议: %v", func() string {
|
||||||
|
switch client.SystemDeviceInfo.Protocol {
|
||||||
|
case client.IPad:
|
||||||
|
return "iPad"
|
||||||
|
case client.AndroidPhone:
|
||||||
|
return "Android Phone"
|
||||||
|
case client.AndroidWatch:
|
||||||
|
return "Android Watch"
|
||||||
|
case client.MacOS:
|
||||||
|
return "MacOS"
|
||||||
|
}
|
||||||
|
return "未知"
|
||||||
|
}())
|
||||||
|
cli.OnLog(func(c *client.QQClient, e *client.LogEvent) {
|
||||||
|
switch e.Type {
|
||||||
|
case "INFO":
|
||||||
|
log.Info("Protocol -> " + e.Message)
|
||||||
|
case "ERROR":
|
||||||
|
log.Error("Protocol -> " + e.Message)
|
||||||
|
case "DEBUG":
|
||||||
|
log.Debug("Protocol -> " + e.Message)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
cli.OnServerUpdated(func(bot *client.QQClient, e *client.ServerUpdatedEvent) bool {
|
||||||
|
if !conf.UseSSOAddress {
|
||||||
|
log.Infof("收到服务器地址更新通知, 根据配置文件已忽略.")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
log.Infof("收到服务器地址更新通知, 将在下一次重连时应用. ")
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
s.Cli = cli
|
||||||
|
s.Dologin()
|
||||||
|
//关闭之前的 server
|
||||||
|
if OldConf.HttpConfig != nil && OldConf.HttpConfig.Enabled {
|
||||||
|
HttpServer.ShutDown()
|
||||||
|
}
|
||||||
|
//if OldConf.WSConfig != nil && OldConf.WSConfig.Enabled {
|
||||||
|
// server.WsShutdown()
|
||||||
|
//}
|
||||||
|
//s.UpServer()
|
||||||
|
s.ReloadServer()
|
||||||
|
s.Conf = conf
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *webServer) UpServer() {
|
||||||
|
conf := GetConf()
|
||||||
|
if conf.HttpConfig != nil && conf.HttpConfig.Enabled {
|
||||||
|
go HttpServer.Run(fmt.Sprintf("%s:%d", conf.HttpConfig.Host, conf.HttpConfig.Port), conf.AccessToken, s.bot)
|
||||||
|
for k, v := range conf.HttpConfig.PostUrls {
|
||||||
|
NewHttpClient().Run(k, v, conf.HttpConfig.Timeout, s.bot)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if conf.WSConfig != nil && conf.WSConfig.Enabled {
|
||||||
|
go WebsocketServer.Run(fmt.Sprintf("%s:%d", conf.WSConfig.Host, conf.WSConfig.Port), conf.AccessToken, s.bot)
|
||||||
|
}
|
||||||
|
for _, rc := range conf.ReverseServers {
|
||||||
|
go NewWebsocketClient(rc, conf.AccessToken, s.bot).Run()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 暂不支持ws服务的重启
|
||||||
|
func (s *webServer) ReloadServer() {
|
||||||
|
conf := GetConf()
|
||||||
|
if conf.HttpConfig != nil && conf.HttpConfig.Enabled {
|
||||||
|
go HttpServer.Run(fmt.Sprintf("%s:%d", conf.HttpConfig.Host, conf.HttpConfig.Port), conf.AccessToken, s.bot)
|
||||||
|
for k, v := range conf.HttpConfig.PostUrls {
|
||||||
|
NewHttpClient().Run(k, v, conf.HttpConfig.Timeout, s.bot)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, rc := range conf.ReverseServers {
|
||||||
|
go NewWebsocketClient(rc, conf.AccessToken, s.bot).Run()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 热重启
|
||||||
|
func AdminDoRestart(s *webServer, c *gin.Context) {
|
||||||
|
s.bot.Release()
|
||||||
|
s.bot = nil
|
||||||
|
s.Cli = nil
|
||||||
|
s.DoReLogin()
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 进程重启
|
||||||
|
func AdminProcessRestart(s *webServer, c *gin.Context) {
|
||||||
|
Restart <- struct{}{}
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
|
||||||
|
// 冷重启
|
||||||
|
func AdminDoRestartDocker(s *webServer, c *gin.Context) {
|
||||||
|
Console <- os.Kill
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// web输入 html 页面
|
||||||
|
func AdminWebWrite(s *webServer, c *gin.Context) {
|
||||||
|
pic := global.ReadAllText("captcha.jpg")
|
||||||
|
var picbase64 string
|
||||||
|
var ispic = false
|
||||||
|
if pic != "" {
|
||||||
|
input := []byte(pic)
|
||||||
|
// base64编码
|
||||||
|
picbase64 = base64.StdEncoding.EncodeToString(input)
|
||||||
|
ispic = true
|
||||||
|
}
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{
|
||||||
|
"ispic": ispic, //为空则为 设备锁 或者没有需要输入
|
||||||
|
"picbase64": picbase64, //web上显示图片
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
// web输入 处理
|
||||||
|
func AdminDoWebWrite(s *webServer, c *gin.Context) {
|
||||||
|
input := c.PostForm("input")
|
||||||
|
WebInput <- input
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
|
||||||
|
// 普通配置修改
|
||||||
|
func AdminDoConfigBase(s *webServer, c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
conf.Uin, _ = strconv.ParseInt(c.PostForm("uin"), 10, 64)
|
||||||
|
conf.Password = c.PostForm("password")
|
||||||
|
if c.PostForm("enable_db") == "true" {
|
||||||
|
conf.EnableDB = true
|
||||||
|
} else {
|
||||||
|
conf.EnableDB = false
|
||||||
|
}
|
||||||
|
conf.AccessToken = c.PostForm("access_token")
|
||||||
|
if err := conf.Save("config.hjson"); err != nil {
|
||||||
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
|
} else {
|
||||||
|
JsonConfig = nil
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// http配置修改
|
||||||
|
func AdminDoConfigHttp(s *webServer, c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
p, _ := strconv.ParseUint(c.PostForm("port"), 10, 16)
|
||||||
|
conf.HttpConfig.Port = uint16(p)
|
||||||
|
conf.HttpConfig.Host = c.PostForm("host")
|
||||||
|
if c.PostForm("enable") == "true" {
|
||||||
|
conf.HttpConfig.Enabled = true
|
||||||
|
} else {
|
||||||
|
conf.HttpConfig.Enabled = false
|
||||||
|
}
|
||||||
|
t, _ := strconv.ParseInt(c.PostForm("timeout"), 10, 32)
|
||||||
|
conf.HttpConfig.Timeout = int32(t)
|
||||||
|
if c.PostForm("post_url") != "" {
|
||||||
|
conf.HttpConfig.PostUrls[c.PostForm("post_url")] = c.PostForm("post_secret")
|
||||||
|
}
|
||||||
|
if err := conf.Save("config.hjson"); err != nil {
|
||||||
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
|
} else {
|
||||||
|
JsonConfig = nil
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ws配置修改
|
||||||
|
func AdminDoConfigWs(s *webServer, c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
p, _ := strconv.ParseUint(c.PostForm("port"), 10, 16)
|
||||||
|
conf.WSConfig.Port = uint16(p)
|
||||||
|
conf.WSConfig.Host = c.PostForm("host")
|
||||||
|
if c.PostForm("enable") == "true" {
|
||||||
|
conf.WSConfig.Enabled = true
|
||||||
|
} else {
|
||||||
|
conf.WSConfig.Enabled = false
|
||||||
|
}
|
||||||
|
if err := conf.Save("config.hjson"); err != nil {
|
||||||
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
|
} else {
|
||||||
|
JsonConfig = nil
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 反向ws配置修改
|
||||||
|
func AdminDoConfigReverse(s *webServer, c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
conf.ReverseServers[0].ReverseApiUrl = c.PostForm("reverse_api_url")
|
||||||
|
conf.ReverseServers[0].ReverseUrl = c.PostForm("reverse_url")
|
||||||
|
conf.ReverseServers[0].ReverseEventUrl = c.PostForm("reverse_event_url")
|
||||||
|
t, _ := strconv.ParseUint(c.PostForm("reverse_reconnect_interval"), 10, 16)
|
||||||
|
conf.ReverseServers[0].ReverseReconnectInterval = uint16(t)
|
||||||
|
if c.PostForm("enable") == "true" {
|
||||||
|
conf.ReverseServers[0].Enabled = true
|
||||||
|
} else {
|
||||||
|
conf.ReverseServers[0].Enabled = false
|
||||||
|
}
|
||||||
|
if err := conf.Save("config.hjson"); err != nil {
|
||||||
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
|
} else {
|
||||||
|
JsonConfig = nil
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// config.json配置修改
|
||||||
|
func AdminDoConfigJson(s *webServer, c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
Json := c.PostForm("json")
|
||||||
|
err := json.Unmarshal([]byte(Json), &conf)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("尝试加载配置文件 %v 时出现错误: %v", "config.hjson", err)
|
||||||
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := conf.Save("config.hjson"); err != nil {
|
||||||
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
|
} else {
|
||||||
|
JsonConfig = nil
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 拉取config.json配置
|
||||||
|
func AdminGetConfigJson(s *webServer, c *gin.Context) {
|
||||||
|
conf := GetConf()
|
||||||
|
c.JSON(200, coolq.OK(coolq.MSG{"config": conf}))
|
||||||
|
|
||||||
|
}
|
418
server/http.go
418
server/http.go
@ -1,14 +1,20 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"crypto/hmac"
|
"crypto/hmac"
|
||||||
"crypto/sha1"
|
"crypto/sha1"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/guonaihong/gout/dataflow"
|
||||||
|
|
||||||
"github.com/Mrs4s/go-cqhttp/coolq"
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/guonaihong/gout"
|
"github.com/guonaihong/gout"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
@ -18,12 +24,14 @@ import (
|
|||||||
type httpServer struct {
|
type httpServer struct {
|
||||||
engine *gin.Engine
|
engine *gin.Engine
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
|
Http *http.Server
|
||||||
}
|
}
|
||||||
|
|
||||||
type httpClient struct {
|
type httpClient struct {
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
secret string
|
secret string
|
||||||
addr string
|
addr string
|
||||||
|
timeout int32
|
||||||
}
|
}
|
||||||
|
|
||||||
var HttpServer = &httpServer{}
|
var HttpServer = &httpServer{}
|
||||||
@ -38,7 +46,7 @@ func (s *httpServer) Run(addr, authToken string, bot *coolq.CQBot) {
|
|||||||
c.Status(404)
|
c.Status(404)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if c.Request.Method == "POST" && c.Request.Header.Get("Content-Type") == "application/json" {
|
if c.Request.Method == "POST" && strings.Contains(c.Request.Header.Get("Content-Type"), "application/json") {
|
||||||
d, err := c.GetRawData()
|
d, err := c.GetRawData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("获取请求 %v 的Body时出现错误: %v", c.Request.RequestURI, err)
|
log.Warnf("获取请求 %v 的Body时出现错误: %v", c.Request.RequestURI, err)
|
||||||
@ -71,91 +79,27 @@ func (s *httpServer) Run(addr, authToken string, bot *coolq.CQBot) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
s.engine.Any("/get_login_info", s.GetLoginInfo)
|
s.engine.Any("/:action", s.HandleActions)
|
||||||
s.engine.Any("/get_login_info_async", s.GetLoginInfo)
|
|
||||||
|
|
||||||
s.engine.Any("/get_friend_list", s.GetFriendList)
|
|
||||||
s.engine.Any("/get_friend_list_async", s.GetFriendList)
|
|
||||||
|
|
||||||
s.engine.Any("/get_group_list", s.GetGroupList)
|
|
||||||
s.engine.Any("/get_group_list_async", s.GetGroupList)
|
|
||||||
|
|
||||||
s.engine.Any("/get_group_info", s.GetGroupInfo)
|
|
||||||
s.engine.Any("/get_group_info_async", s.GetGroupInfo)
|
|
||||||
|
|
||||||
s.engine.Any("/get_group_member_list", s.GetGroupMemberList)
|
|
||||||
s.engine.Any("/get_group_member_list_async", s.GetGroupMemberList)
|
|
||||||
|
|
||||||
s.engine.Any("/get_group_member_info", s.GetGroupMemberInfo)
|
|
||||||
s.engine.Any("/get_group_member_info_async", s.GetGroupMemberInfo)
|
|
||||||
|
|
||||||
s.engine.Any("/send_msg", s.SendMessage)
|
|
||||||
s.engine.Any("/send_msg_async", s.SendMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/send_private_msg", s.SendPrivateMessage)
|
|
||||||
s.engine.Any("/send_private_msg_async", s.SendPrivateMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/send_group_msg", s.SendGroupMessage)
|
|
||||||
s.engine.Any("/send_group_msg_async", s.SendGroupMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/send_group_forward_msg", s.SendGroupForwardMessage)
|
|
||||||
s.engine.Any("/send_group_forward_msg_async", s.SendGroupForwardMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/delete_msg", s.DeleteMessage)
|
|
||||||
s.engine.Any("/delete_msg_async", s.DeleteMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/set_friend_add_request", s.ProcessFriendRequest)
|
|
||||||
s.engine.Any("/set_friend_add_request_async", s.ProcessFriendRequest)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_add_request", s.ProcessGroupRequest)
|
|
||||||
s.engine.Any("/set_group_add_request_async", s.ProcessGroupRequest)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_card", s.SetGroupCard)
|
|
||||||
s.engine.Any("/set_group_card_async", s.SetGroupCard)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_special_title", s.SetSpecialTitle)
|
|
||||||
s.engine.Any("/set_group_special_title_async", s.SetSpecialTitle)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_kick", s.SetGroupKick)
|
|
||||||
s.engine.Any("/set_group_kick_async", s.SetGroupKick)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_ban", s.SetGroupBan)
|
|
||||||
s.engine.Any("/set_group_ban_async", s.SetGroupBan)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_whole_ban", s.SetWholeBan)
|
|
||||||
s.engine.Any("/set_group_whole_ban_async", s.SetWholeBan)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_name", s.SetGroupName)
|
|
||||||
s.engine.Any("/set_group_name_async", s.SetGroupName)
|
|
||||||
|
|
||||||
s.engine.Any("/set_group_leave", s.SetGroupLeave)
|
|
||||||
s.engine.Any("/set_group_leave_async", s.SetGroupLeave)
|
|
||||||
|
|
||||||
s.engine.Any("/get_image", s.GetImage)
|
|
||||||
s.engine.Any("/get_image_async", s.GetImage)
|
|
||||||
|
|
||||||
s.engine.Any("/get_forward_msg", s.GetForwardMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/get_group_msg", s.GetGroupMessage)
|
|
||||||
s.engine.Any("/get_group_msg_async", s.GetGroupMessage)
|
|
||||||
|
|
||||||
s.engine.Any("/can_send_image", s.CanSendImage)
|
|
||||||
s.engine.Any("/can_send_image_async", s.CanSendImage)
|
|
||||||
|
|
||||||
s.engine.Any("/can_send_record", s.CanSendRecord)
|
|
||||||
s.engine.Any("/can_send_record_async", s.CanSendRecord)
|
|
||||||
|
|
||||||
s.engine.Any("/get_status", s.GetStatus)
|
|
||||||
s.engine.Any("/get_status_async", s.GetStatus)
|
|
||||||
|
|
||||||
s.engine.Any("/get_version_info", s.GetVersionInfo)
|
|
||||||
s.engine.Any("/get_version_info_async", s.GetVersionInfo)
|
|
||||||
|
|
||||||
s.engine.Any("/.handle_quick_operation", s.HandleQuickOperation)
|
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
log.Infof("CQ HTTP 服务器已启动: %v", addr)
|
log.Infof("CQ HTTP 服务器已启动: %v", addr)
|
||||||
log.Fatal(s.engine.Run(addr))
|
s.Http = &http.Server{
|
||||||
|
Addr: addr,
|
||||||
|
Handler: s.engine,
|
||||||
|
}
|
||||||
|
if err := s.Http.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
||||||
|
log.Error(err)
|
||||||
|
log.Infof("请检查端口是否被占用.")
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
//err := s.engine.Run(addr)
|
||||||
|
//if err != nil {
|
||||||
|
// log.Error(err)
|
||||||
|
// log.Infof("请检查端口是否被占用.")
|
||||||
|
// time.Sleep(time.Second * 5)
|
||||||
|
// os.Exit(1)
|
||||||
|
//}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,10 +107,14 @@ func NewHttpClient() *httpClient {
|
|||||||
return &httpClient{}
|
return &httpClient{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *httpClient) Run(addr, secret string, bot *coolq.CQBot) {
|
func (c *httpClient) Run(addr, secret string, timeout int32, bot *coolq.CQBot) {
|
||||||
c.bot = bot
|
c.bot = bot
|
||||||
c.secret = secret
|
c.secret = secret
|
||||||
c.addr = addr
|
c.addr = addr
|
||||||
|
c.timeout = timeout
|
||||||
|
if c.timeout < 5 {
|
||||||
|
c.timeout = 5
|
||||||
|
}
|
||||||
bot.OnEventPush(c.onBotPushEvent)
|
bot.OnEventPush(c.onBotPushEvent)
|
||||||
log.Infof("HTTP POST上报器已启动: %v", addr)
|
log.Infof("HTTP POST上报器已启动: %v", addr)
|
||||||
}
|
}
|
||||||
@ -184,175 +132,274 @@ func (c *httpClient) onBotPushEvent(m coolq.MSG) {
|
|||||||
h["X-Signature"] = "sha1=" + hex.EncodeToString(mac.Sum(nil))
|
h["X-Signature"] = "sha1=" + hex.EncodeToString(mac.Sum(nil))
|
||||||
}
|
}
|
||||||
return h
|
return h
|
||||||
}()).SetTimeout(time.Second * 5).Do()
|
}()).SetTimeout(time.Second * time.Duration(c.timeout)).F().Retry().Attempt(5).
|
||||||
|
WaitTime(time.Millisecond * 500).MaxWaitTime(time.Second * 5).
|
||||||
|
Func(func(con *dataflow.Context) error {
|
||||||
|
if con.Error != nil {
|
||||||
|
log.Warnf("上报Event到 HTTP 服务器 %v 时出现错误: %v 将重试.", c.addr, con.Error)
|
||||||
|
return con.Error
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}).Do()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("上报Event数据到 %v 失败: %v", c.addr, err)
|
log.Warnf("上报Event数据 %v 到 %v 失败: %v", m.ToJson(), c.addr, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
log.Debugf("上报Event数据 %v 到 %v", m.ToJson(), c.addr)
|
||||||
if gjson.Valid(res) {
|
if gjson.Valid(res) {
|
||||||
c.bot.CQHandleQuickOperation(gjson.Parse(m.ToJson()), gjson.Parse(res))
|
c.bot.CQHandleQuickOperation(gjson.Parse(m.ToJson()), gjson.Parse(res))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetLoginInfo(c *gin.Context) {
|
func (s *httpServer) HandleActions(c *gin.Context) {
|
||||||
|
global.RateLimit(context.Background())
|
||||||
|
action := strings.ReplaceAll(c.Param("action"), "_async", "")
|
||||||
|
log.Debugf("HTTPServer接收到API调用: %v", action)
|
||||||
|
if f, ok := httpApi[action]; ok {
|
||||||
|
f(s, c)
|
||||||
|
} else {
|
||||||
|
c.JSON(200, coolq.Failed(404))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetLoginInfo(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQGetLoginInfo())
|
c.JSON(200, s.bot.CQGetLoginInfo())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetFriendList(c *gin.Context) {
|
func GetFriendList(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQGetFriendList())
|
c.JSON(200, s.bot.CQGetFriendList())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetGroupList(c *gin.Context) {
|
func GetGroupList(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQGetGroupList())
|
nc := getParamOrDefault(c, "no_cache", "false")
|
||||||
|
c.JSON(200, s.bot.CQGetGroupList(nc == "true"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetGroupInfo(c *gin.Context) {
|
func GetGroupInfo(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQGetGroupInfo(gid))
|
c.JSON(200, s.bot.CQGetGroupInfo(gid))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetGroupMemberList(c *gin.Context) {
|
func GetGroupMemberList(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQGetGroupMemberList(gid))
|
nc := getParamOrDefault(c, "no_cache", "false")
|
||||||
|
c.JSON(200, s.bot.CQGetGroupMemberList(gid, nc == "true"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetGroupMemberInfo(c *gin.Context) {
|
func GetGroupMemberInfo(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
nc := getParamOrDefault(c, "no_cache", "false")
|
c.JSON(200, s.bot.CQGetGroupMemberInfo(gid, uid))
|
||||||
c.JSON(200, s.bot.CQGetGroupMemberInfo(gid, uid, nc == "true"))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SendMessage(c *gin.Context) {
|
func GetGroupFileSystemInfo(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQGetGroupFileSystemInfo(gid))
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetGroupRootFiles(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQGetGroupRootFiles(gid))
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetGroupFilesByFolderId(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
folderId := getParam(c, "folder_id")
|
||||||
|
c.JSON(200, s.bot.CQGetGroupFilesByFolderId(gid, folderId))
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetGroupFileUrl(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
fid := getParam(c, "file_id")
|
||||||
|
busid, _ := strconv.ParseInt(getParam(c, "busid"), 10, 32)
|
||||||
|
c.JSON(200, s.bot.CQGetGroupFileUrl(gid, fid, int32(busid)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func SendMessage(s *httpServer, c *gin.Context) {
|
||||||
|
if getParam(c, "message_type") == "private" {
|
||||||
|
SendPrivateMessage(s, c)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if getParam(c, "message_type") == "group" {
|
||||||
|
SendGroupMessage(s, c)
|
||||||
|
return
|
||||||
|
}
|
||||||
if getParam(c, "group_id") != "" {
|
if getParam(c, "group_id") != "" {
|
||||||
s.SendGroupMessage(c)
|
SendGroupMessage(s, c)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if getParam(c, "user_id") != "" {
|
if getParam(c, "user_id") != "" {
|
||||||
s.SendPrivateMessage(c)
|
SendPrivateMessage(s, c)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SendPrivateMessage(c *gin.Context) {
|
func SendPrivateMessage(s *httpServer, c *gin.Context) {
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
msg := getParam(c, "message")
|
msg, t := getParamWithType(c, "message")
|
||||||
if gjson.Valid(msg) {
|
autoEscape := global.EnsureBool(getParam(c, "auto_escape"), false)
|
||||||
c.JSON(200, s.bot.CQSendPrivateMessage(uid, gjson.Parse(msg)))
|
if t == gjson.JSON {
|
||||||
|
c.JSON(200, s.bot.CQSendPrivateMessage(uid, gjson.Parse(msg), autoEscape))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
c.JSON(200, s.bot.CQSendPrivateMessage(uid, msg))
|
c.JSON(200, s.bot.CQSendPrivateMessage(uid, msg, autoEscape))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SendGroupMessage(c *gin.Context) {
|
func SendGroupMessage(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
msg := getParam(c, "message")
|
msg, t := getParamWithType(c, "message")
|
||||||
if gjson.Valid(msg) {
|
autoEscape := global.EnsureBool(getParam(c, "auto_escape"), false)
|
||||||
c.JSON(200, s.bot.CQSendGroupMessage(gid, gjson.Parse(msg)))
|
if t == gjson.JSON {
|
||||||
|
c.JSON(200, s.bot.CQSendGroupMessage(gid, gjson.Parse(msg), autoEscape))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
c.JSON(200, s.bot.CQSendGroupMessage(gid, msg))
|
c.JSON(200, s.bot.CQSendGroupMessage(gid, msg, autoEscape))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SendGroupForwardMessage(c *gin.Context) {
|
func SendGroupForwardMessage(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
msg := getParam(c, "messages")
|
msg := getParam(c, "messages")
|
||||||
c.JSON(200, s.bot.CQSendGroupForwardMessage(gid, gjson.Parse(msg)))
|
c.JSON(200, s.bot.CQSendGroupForwardMessage(gid, gjson.Parse(msg)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetImage(c *gin.Context) {
|
func GetImage(s *httpServer, c *gin.Context) {
|
||||||
file := getParam(c, "file")
|
file := getParam(c, "file")
|
||||||
c.JSON(200, s.bot.CQGetImage(file))
|
c.JSON(200, s.bot.CQGetImage(file))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetGroupMessage(c *gin.Context) {
|
func GetMessage(s *httpServer, c *gin.Context) {
|
||||||
mid, _ := strconv.ParseInt(getParam(c, "message_id"), 10, 32)
|
mid, _ := strconv.ParseInt(getParam(c, "message_id"), 10, 32)
|
||||||
c.JSON(200, s.bot.CQGetGroupMessage(int32(mid)))
|
c.JSON(200, s.bot.CQGetMessage(int32(mid)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) ProcessFriendRequest(c *gin.Context) {
|
func GetGroupHonorInfo(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQGetGroupHonorInfo(gid, getParam(c, "type")))
|
||||||
|
}
|
||||||
|
|
||||||
|
func ProcessFriendRequest(s *httpServer, c *gin.Context) {
|
||||||
flag := getParam(c, "flag")
|
flag := getParam(c, "flag")
|
||||||
approve := getParamOrDefault(c, "approve", "true")
|
approve := getParamOrDefault(c, "approve", "true")
|
||||||
c.JSON(200, s.bot.CQProcessFriendRequest(flag, approve == "true"))
|
c.JSON(200, s.bot.CQProcessFriendRequest(flag, approve == "true"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) ProcessGroupRequest(c *gin.Context) {
|
func ProcessGroupRequest(s *httpServer, c *gin.Context) {
|
||||||
flag := getParam(c, "flag")
|
flag := getParam(c, "flag")
|
||||||
subType := getParam(c, "sub_type")
|
subType := getParam(c, "sub_type")
|
||||||
if subType == "" {
|
if subType == "" {
|
||||||
subType = getParam(c, "type")
|
subType = getParam(c, "type")
|
||||||
}
|
}
|
||||||
approve := getParamOrDefault(c, "approve", "true")
|
approve := getParamOrDefault(c, "approve", "true")
|
||||||
c.JSON(200, s.bot.CQProcessGroupRequest(flag, subType, approve == "true"))
|
c.JSON(200, s.bot.CQProcessGroupRequest(flag, subType, getParam(c, "reason"), approve == "true"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetGroupCard(c *gin.Context) {
|
func SetGroupCard(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQSetGroupCard(gid, uid, getParam(c, "card")))
|
c.JSON(200, s.bot.CQSetGroupCard(gid, uid, getParam(c, "card")))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetSpecialTitle(c *gin.Context) {
|
func SetSpecialTitle(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQSetGroupSpecialTitle(gid, uid, getParam(c, "special_title")))
|
c.JSON(200, s.bot.CQSetGroupSpecialTitle(gid, uid, getParam(c, "special_title")))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetGroupKick(c *gin.Context) {
|
func SetGroupKick(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
msg := getParam(c, "message")
|
msg := getParam(c, "message")
|
||||||
c.JSON(200, s.bot.CQSetGroupKick(gid, uid, msg))
|
block := getParamOrDefault(c, "reject_add_request", "false")
|
||||||
|
c.JSON(200, s.bot.CQSetGroupKick(gid, uid, msg, block == "true"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetGroupBan(c *gin.Context) {
|
func SetGroupBan(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
i, _ := strconv.ParseInt(getParamOrDefault(c, "duration", "1800"), 10, 64)
|
i, _ := strconv.ParseInt(getParamOrDefault(c, "duration", "1800"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQSetGroupBan(gid, uid, uint32(i)))
|
c.JSON(200, s.bot.CQSetGroupBan(gid, uid, uint32(i)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetWholeBan(c *gin.Context) {
|
func SetWholeBan(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQSetGroupWholeBan(gid, getParamOrDefault(c, "enable", "true") == "true"))
|
c.JSON(200, s.bot.CQSetGroupWholeBan(gid, getParamOrDefault(c, "enable", "true") == "true"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetGroupName(c *gin.Context) {
|
func SetGroupName(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQSetGroupName(gid, getParam(c, "name")))
|
c.JSON(200, s.bot.CQSetGroupName(gid, getParam(c, "group_name")))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) SetGroupLeave(c *gin.Context) {
|
func SetGroupAdmin(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQSetGroupAdmin(gid, uid, getParamOrDefault(c, "enable", "true") == "true"))
|
||||||
|
}
|
||||||
|
|
||||||
|
func SendGroupNotice(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQSetGroupMemo(gid, getParam(c, "content")))
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetGroupLeave(s *httpServer, c *gin.Context) {
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
c.JSON(200, s.bot.CQSetGroupLeave(gid))
|
c.JSON(200, s.bot.CQSetGroupLeave(gid))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetForwardMessage(c *gin.Context) {
|
func SetRestart(s *httpServer, c *gin.Context) {
|
||||||
|
delay, _ := strconv.ParseInt(getParam(c, "delay"), 10, 64)
|
||||||
|
c.JSON(200, coolq.MSG{"data": nil, "retcode": 0, "status": "async"})
|
||||||
|
go func(delay int64) {
|
||||||
|
time.Sleep(time.Duration(delay) * time.Millisecond)
|
||||||
|
Restart <- struct{}{}
|
||||||
|
}(delay)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetForwardMessage(s *httpServer, c *gin.Context) {
|
||||||
resId := getParam(c, "message_id")
|
resId := getParam(c, "message_id")
|
||||||
c.JSON(200, s.bot.CQGetForwardMessage(resId))
|
c.JSON(200, s.bot.CQGetForwardMessage(resId))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) DeleteMessage(c *gin.Context) {
|
func GetGroupSystemMessage(s *httpServer, c *gin.Context) {
|
||||||
|
c.JSON(200, s.bot.CQGetGroupSystemMessages())
|
||||||
|
}
|
||||||
|
|
||||||
|
func DeleteMessage(s *httpServer, c *gin.Context) {
|
||||||
mid, _ := strconv.ParseInt(getParam(c, "message_id"), 10, 32)
|
mid, _ := strconv.ParseInt(getParam(c, "message_id"), 10, 32)
|
||||||
c.JSON(200, s.bot.CQDeleteMessage(int32(mid)))
|
c.JSON(200, s.bot.CQDeleteMessage(int32(mid)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) CanSendImage(c *gin.Context) {
|
func CanSendImage(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQCanSendImage())
|
c.JSON(200, s.bot.CQCanSendImage())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) CanSendRecord(c *gin.Context) {
|
func CanSendRecord(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQCanSendRecord())
|
c.JSON(200, s.bot.CQCanSendRecord())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetStatus(c *gin.Context) {
|
func GetStatus(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQGetStatus())
|
c.JSON(200, s.bot.CQGetStatus())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) GetVersionInfo(c *gin.Context) {
|
func GetVersionInfo(s *httpServer, c *gin.Context) {
|
||||||
c.JSON(200, s.bot.CQGetVersionInfo())
|
c.JSON(200, s.bot.CQGetVersionInfo())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) HandleQuickOperation(c *gin.Context) {
|
func ReloadEventFilter(s *httpServer, c *gin.Context) {
|
||||||
|
c.JSON(200, s.bot.CQReloadEventFilter())
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetVipInfo(s *httpServer, c *gin.Context) {
|
||||||
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQGetVipInfo(uid))
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetStrangerInfo(s *httpServer, c *gin.Context) {
|
||||||
|
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
||||||
|
c.JSON(200, s.bot.CQGetStrangerInfo(uid))
|
||||||
|
}
|
||||||
|
|
||||||
|
func HandleQuickOperation(s *httpServer, c *gin.Context) {
|
||||||
if c.Request.Method != "POST" {
|
if c.Request.Method != "POST" {
|
||||||
c.AbortWithStatus(404)
|
c.AbortWithStatus(404)
|
||||||
return
|
return
|
||||||
@ -363,6 +410,23 @@ func (s *httpServer) HandleQuickOperation(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func OcrImage(s *httpServer, c *gin.Context) {
|
||||||
|
img := getParam(c, "image")
|
||||||
|
c.JSON(200, s.bot.CQOcrImage(img))
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetWordSlices(s *httpServer, c *gin.Context) {
|
||||||
|
content := getParam(c, "content")
|
||||||
|
c.JSON(200, s.bot.CQGetWordSlices(content))
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetGroupPortrait(s *httpServer, c *gin.Context) {
|
||||||
|
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
||||||
|
file := getParam(c, "file")
|
||||||
|
cache := getParam(c, "cache")
|
||||||
|
c.JSON(200, s.bot.CQSetGroupPortrait(gid, file, cache))
|
||||||
|
}
|
||||||
|
|
||||||
func getParamOrDefault(c *gin.Context, k, def string) string {
|
func getParamOrDefault(c *gin.Context, k, def string) string {
|
||||||
r := getParam(c, k)
|
r := getParam(c, k)
|
||||||
if r != "" {
|
if r != "" {
|
||||||
@ -372,36 +436,100 @@ func getParamOrDefault(c *gin.Context, k, def string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func getParam(c *gin.Context, k string) string {
|
func getParam(c *gin.Context, k string) string {
|
||||||
|
p, _ := getParamWithType(c, k)
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
func getParamWithType(c *gin.Context, k string) (string, gjson.Type) {
|
||||||
if q := c.Query(k); q != "" {
|
if q := c.Query(k); q != "" {
|
||||||
return q
|
return q, gjson.Null
|
||||||
}
|
}
|
||||||
if c.Request.Method == "POST" {
|
if c.Request.Method == "POST" {
|
||||||
if h := c.Request.Header.Get("Content-Type"); h != "" {
|
if h := c.Request.Header.Get("Content-Type"); h != "" {
|
||||||
if h == "application/x-www-form-urlencoded" {
|
if strings.Contains(h, "application/x-www-form-urlencoded") {
|
||||||
if p, ok := c.GetPostForm(k); ok {
|
if p, ok := c.GetPostForm(k); ok {
|
||||||
return p
|
return p, gjson.Null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if h == "application/json" {
|
if strings.Contains(h, "application/json") {
|
||||||
if obj, ok := c.Get("json_body"); ok {
|
if obj, ok := c.Get("json_body"); ok {
|
||||||
res := obj.(gjson.Result).Get(k)
|
res := obj.(gjson.Result).Get(k)
|
||||||
if res.Exists() {
|
if res.Exists() {
|
||||||
switch res.Type {
|
switch res.Type {
|
||||||
case gjson.JSON:
|
case gjson.JSON:
|
||||||
return res.Raw
|
return res.Raw, gjson.JSON
|
||||||
case gjson.String:
|
case gjson.String:
|
||||||
return res.Str
|
return res.Str, gjson.String
|
||||||
case gjson.Number:
|
case gjson.Number:
|
||||||
return strconv.FormatInt(res.Int(), 10) // 似乎没有需要接受 float 类型的api
|
return strconv.FormatInt(res.Int(), 10), gjson.Number // 似乎没有需要接受 float 类型的api
|
||||||
case gjson.True:
|
case gjson.True:
|
||||||
return "true"
|
return "true", gjson.True
|
||||||
case gjson.False:
|
case gjson.False:
|
||||||
return "false"
|
return "false", gjson.False
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ""
|
return "", gjson.Null
|
||||||
|
}
|
||||||
|
|
||||||
|
var httpApi = map[string]func(s *httpServer, c *gin.Context){
|
||||||
|
"get_login_info": GetLoginInfo,
|
||||||
|
"get_friend_list": GetFriendList,
|
||||||
|
"get_group_list": GetGroupList,
|
||||||
|
"get_group_info": GetGroupInfo,
|
||||||
|
"get_group_member_list": GetGroupMemberList,
|
||||||
|
"get_group_member_info": GetGroupMemberInfo,
|
||||||
|
"get_group_file_system_info": GetGroupFileSystemInfo,
|
||||||
|
"get_group_root_files": GetGroupRootFiles,
|
||||||
|
"get_group_files_by_folder": GetGroupFilesByFolderId,
|
||||||
|
"get_group_file_url": GetGroupFileUrl,
|
||||||
|
"send_msg": SendMessage,
|
||||||
|
"send_group_msg": SendGroupMessage,
|
||||||
|
"send_group_forward_msg": SendGroupForwardMessage,
|
||||||
|
"send_private_msg": SendPrivateMessage,
|
||||||
|
"delete_msg": DeleteMessage,
|
||||||
|
"set_friend_add_request": ProcessFriendRequest,
|
||||||
|
"set_group_add_request": ProcessGroupRequest,
|
||||||
|
"set_group_card": SetGroupCard,
|
||||||
|
"set_group_special_title": SetSpecialTitle,
|
||||||
|
"set_group_kick": SetGroupKick,
|
||||||
|
"set_group_ban": SetGroupBan,
|
||||||
|
"set_group_whole_ban": SetWholeBan,
|
||||||
|
"set_group_name": SetGroupName,
|
||||||
|
"set_group_admin": SetGroupAdmin,
|
||||||
|
"set_restart": SetRestart,
|
||||||
|
"_send_group_notice": SendGroupNotice,
|
||||||
|
"set_group_leave": SetGroupLeave,
|
||||||
|
"get_image": GetImage,
|
||||||
|
"get_forward_msg": GetForwardMessage,
|
||||||
|
"get_msg": GetMessage,
|
||||||
|
"get_group_system_msg": GetGroupSystemMessage,
|
||||||
|
"get_group_honor_info": GetGroupHonorInfo,
|
||||||
|
"can_send_image": CanSendImage,
|
||||||
|
"can_send_record": CanSendRecord,
|
||||||
|
"get_status": GetStatus,
|
||||||
|
"get_version_info": GetVersionInfo,
|
||||||
|
"_get_vip_info": GetVipInfo,
|
||||||
|
"get_stranger_info": GetStrangerInfo,
|
||||||
|
"reload_event_filter": ReloadEventFilter,
|
||||||
|
"set_group_portrait": SetGroupPortrait,
|
||||||
|
".handle_quick_operation": HandleQuickOperation,
|
||||||
|
".ocr_image": OcrImage,
|
||||||
|
".get_word_slices": GetWordSlices,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *httpServer) ShutDown() {
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||||||
|
defer cancel()
|
||||||
|
if err := s.Http.Shutdown(ctx); err != nil {
|
||||||
|
log.Fatal("http Server Shutdown:", err)
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
log.Println("timeout of 5 seconds.")
|
||||||
|
}
|
||||||
|
log.Println("http Server exiting")
|
||||||
}
|
}
|
||||||
|
@ -1,26 +1,27 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/Mrs4s/go-cqhttp/coolq"
|
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
|
||||||
"github.com/gorilla/websocket"
|
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
"github.com/tidwall/gjson"
|
|
||||||
wsc "golang.org/x/net/websocket"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
"github.com/gorilla/websocket"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
)
|
)
|
||||||
|
|
||||||
type websocketServer struct {
|
type websocketServer struct {
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
token string
|
token string
|
||||||
eventConn []*websocket.Conn
|
eventConn []*websocketConn
|
||||||
pushLock *sync.Mutex
|
eventConnMutex sync.Mutex
|
||||||
handshake string
|
handshake string
|
||||||
}
|
}
|
||||||
|
|
||||||
type websocketClient struct {
|
type websocketClient struct {
|
||||||
@ -28,9 +29,13 @@ type websocketClient struct {
|
|||||||
token string
|
token string
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
|
|
||||||
pushLock *sync.Mutex
|
universalConn *websocketConn
|
||||||
universalConn *wsc.Conn
|
eventConn *websocketConn
|
||||||
eventConn *wsc.Conn
|
}
|
||||||
|
|
||||||
|
type websocketConn struct {
|
||||||
|
*websocket.Conn
|
||||||
|
sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
var WebsocketServer = &websocketServer{}
|
var WebsocketServer = &websocketServer{}
|
||||||
@ -42,7 +47,6 @@ var upgrader = websocket.Upgrader{
|
|||||||
|
|
||||||
func (s *websocketServer) Run(addr, authToken string, b *coolq.CQBot) {
|
func (s *websocketServer) Run(addr, authToken string, b *coolq.CQBot) {
|
||||||
s.token = authToken
|
s.token = authToken
|
||||||
s.pushLock = new(sync.Mutex)
|
|
||||||
s.bot = b
|
s.bot = b
|
||||||
s.handshake = fmt.Sprintf(`{"_post_method":2,"meta_event_type":"lifecycle","post_type":"meta_event","self_id":%d,"sub_type":"connect","time":%d}`,
|
s.handshake = fmt.Sprintf(`{"_post_method":2,"meta_event_type":"lifecycle","post_type":"meta_event","self_id":%d,"sub_type":"connect","time":%d}`,
|
||||||
s.bot.Client.Uin, time.Now().Unix())
|
s.bot.Client.Uin, time.Now().Unix())
|
||||||
@ -57,39 +61,37 @@ func (s *websocketServer) Run(addr, authToken string, b *coolq.CQBot) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func NewWebsocketClient(conf *global.GoCQReverseWebsocketConfig, authToken string, b *coolq.CQBot) *websocketClient {
|
func NewWebsocketClient(conf *global.GoCQReverseWebsocketConfig, authToken string, b *coolq.CQBot) *websocketClient {
|
||||||
return &websocketClient{conf: conf, token: authToken, bot: b, pushLock: new(sync.Mutex)}
|
return &websocketClient{conf: conf, token: authToken, bot: b}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) Run() {
|
func (c *websocketClient) Run() {
|
||||||
if !c.conf.Enabled {
|
if !c.conf.Enabled {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if c.conf.ReverseApiUrl != "" {
|
|
||||||
c.connectApi()
|
|
||||||
}
|
|
||||||
if c.conf.ReverseEventUrl != "" {
|
|
||||||
c.connectEvent()
|
|
||||||
}
|
|
||||||
if c.conf.ReverseUrl != "" {
|
if c.conf.ReverseUrl != "" {
|
||||||
c.connectUniversal()
|
c.connectUniversal()
|
||||||
|
} else {
|
||||||
|
if c.conf.ReverseApiUrl != "" {
|
||||||
|
c.connectApi()
|
||||||
|
}
|
||||||
|
if c.conf.ReverseEventUrl != "" {
|
||||||
|
c.connectEvent()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
c.bot.OnEventPush(c.onBotPushEvent)
|
c.bot.OnEventPush(c.onBotPushEvent)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) connectApi() {
|
func (c *websocketClient) connectApi() {
|
||||||
log.Infof("开始尝试连接到反向Websocket API服务器: %v", c.conf.ReverseApiUrl)
|
log.Infof("开始尝试连接到反向Websocket API服务器: %v", c.conf.ReverseApiUrl)
|
||||||
wsConf, err := wsc.NewConfig(c.conf.ReverseApiUrl, c.conf.ReverseApiUrl)
|
header := http.Header{
|
||||||
if err != nil {
|
"X-Client-Role": []string{"API"},
|
||||||
log.Warnf("连接到反向Websocket API服务器 %v 时出现致命错误: %v", c.conf.ReverseApiUrl, err)
|
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
||||||
return
|
"User-Agent": []string{"CQHttp/4.15.0"},
|
||||||
}
|
}
|
||||||
wsConf.Header["X-Client-Role"] = []string{"API"}
|
|
||||||
wsConf.Header["X-Self-ID"] = []string{strconv.FormatInt(c.bot.Client.Uin, 10)}
|
|
||||||
wsConf.Header["User-Agent"] = []string{"CQHttp/4.15.0"}
|
|
||||||
if c.token != "" {
|
if c.token != "" {
|
||||||
wsConf.Header["Authorization"] = []string{"Token " + c.token}
|
header["Authorization"] = []string{"Token " + c.token}
|
||||||
}
|
}
|
||||||
conn, err := wsc.DialConfig(wsConf)
|
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseApiUrl, header)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("连接到反向Websocket API服务器 %v 时出现错误: %v", c.conf.ReverseApiUrl, err)
|
log.Warnf("连接到反向Websocket API服务器 %v 时出现错误: %v", c.conf.ReverseApiUrl, err)
|
||||||
if c.conf.ReverseReconnectInterval != 0 {
|
if c.conf.ReverseReconnectInterval != 0 {
|
||||||
@ -99,49 +101,52 @@ func (c *websocketClient) connectApi() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
log.Infof("已连接到反向Websocket API服务器 %v", c.conf.ReverseApiUrl)
|
log.Infof("已连接到反向Websocket API服务器 %v", c.conf.ReverseApiUrl)
|
||||||
go c.listenApi(conn, false)
|
wrappedConn := &websocketConn{Conn: conn}
|
||||||
|
go c.listenApi(wrappedConn, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) connectEvent() {
|
func (c *websocketClient) connectEvent() {
|
||||||
log.Infof("开始尝试连接到反向Websocket Event服务器: %v", c.conf.ReverseEventUrl)
|
log.Infof("开始尝试连接到反向Websocket Event服务器: %v", c.conf.ReverseEventUrl)
|
||||||
wsConf, err := wsc.NewConfig(c.conf.ReverseEventUrl, c.conf.ReverseEventUrl)
|
header := http.Header{
|
||||||
if err != nil {
|
"X-Client-Role": []string{"Event"},
|
||||||
log.Warnf("连接到反向Websocket Event服务器 %v 时出现致命错误: %v", c.conf.ReverseApiUrl, err)
|
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
||||||
return
|
"User-Agent": []string{"CQHttp/4.15.0"},
|
||||||
}
|
}
|
||||||
wsConf.Header["X-Client-Role"] = []string{"Event"}
|
|
||||||
wsConf.Header["X-Self-ID"] = []string{strconv.FormatInt(c.bot.Client.Uin, 10)}
|
|
||||||
wsConf.Header["User-Agent"] = []string{"CQHttp/4.15.0"}
|
|
||||||
if c.token != "" {
|
if c.token != "" {
|
||||||
wsConf.Header["Authorization"] = []string{"Token " + c.token}
|
header["Authorization"] = []string{"Token " + c.token}
|
||||||
}
|
}
|
||||||
conn, err := wsc.DialConfig(wsConf)
|
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseEventUrl, header)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("连接到反向Websocket API服务器 %v 时出现错误: %v", c.conf.ReverseApiUrl, err)
|
log.Warnf("连接到反向Websocket Event服务器 %v 时出现错误: %v", c.conf.ReverseEventUrl, err)
|
||||||
if c.conf.ReverseReconnectInterval != 0 {
|
if c.conf.ReverseReconnectInterval != 0 {
|
||||||
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
||||||
c.connectApi()
|
c.connectEvent()
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handshake := fmt.Sprintf(`{"meta_event_type":"lifecycle","post_type":"meta_event","self_id":%d,"sub_type":"connect","time":%d}`,
|
||||||
|
c.bot.Client.Uin, time.Now().Unix())
|
||||||
|
err = conn.WriteMessage(websocket.TextMessage, []byte(handshake))
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("反向Websocket 握手时出现错误: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
log.Infof("已连接到反向Websocket Event服务器 %v", c.conf.ReverseEventUrl)
|
log.Infof("已连接到反向Websocket Event服务器 %v", c.conf.ReverseEventUrl)
|
||||||
c.eventConn = conn
|
c.eventConn = &websocketConn{Conn: conn}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) connectUniversal() {
|
func (c *websocketClient) connectUniversal() {
|
||||||
log.Infof("开始尝试连接到反向Websocket Universal服务器: %v", c.conf.ReverseUrl)
|
log.Infof("开始尝试连接到反向Websocket Universal服务器: %v", c.conf.ReverseUrl)
|
||||||
wsConf, err := wsc.NewConfig(c.conf.ReverseUrl, c.conf.ReverseUrl)
|
header := http.Header{
|
||||||
if err != nil {
|
"X-Client-Role": []string{"Universal"},
|
||||||
log.Warnf("连接到反向Websocket Universal服务器 %v 时出现致命错误: %v", c.conf.ReverseUrl, err)
|
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
||||||
return
|
"User-Agent": []string{"CQHttp/4.15.0"},
|
||||||
}
|
}
|
||||||
wsConf.Header["X-Client-Role"] = []string{"Universal"}
|
|
||||||
wsConf.Header["X-Self-ID"] = []string{strconv.FormatInt(c.bot.Client.Uin, 10)}
|
|
||||||
wsConf.Header["User-Agent"] = []string{"CQHttp/4.15.0"}
|
|
||||||
if c.token != "" {
|
if c.token != "" {
|
||||||
wsConf.Header["Authorization"] = []string{"Token " + c.token}
|
header["Authorization"] = []string{"Token " + c.token}
|
||||||
}
|
}
|
||||||
conn, err := wsc.DialConfig(wsConf)
|
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseUrl, header)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("连接到反向Websocket Universal服务器 %v 时出现错误: %v", c.conf.ReverseUrl, err)
|
log.Warnf("连接到反向Websocket Universal服务器 %v 时出现错误: %v", c.conf.ReverseUrl, err)
|
||||||
if c.conf.ReverseReconnectInterval != 0 {
|
if c.conf.ReverseReconnectInterval != 0 {
|
||||||
@ -150,47 +155,48 @@ func (c *websocketClient) connectUniversal() {
|
|||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
go c.listenApi(conn, true)
|
|
||||||
c.universalConn = conn
|
handshake := fmt.Sprintf(`{"meta_event_type":"lifecycle","post_type":"meta_event","self_id":%d,"sub_type":"connect","time":%d}`,
|
||||||
|
c.bot.Client.Uin, time.Now().Unix())
|
||||||
|
err = conn.WriteMessage(websocket.TextMessage, []byte(handshake))
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("反向Websocket 握手时出现错误: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
wrappedConn := &websocketConn{Conn: conn}
|
||||||
|
go c.listenApi(wrappedConn, true)
|
||||||
|
c.universalConn = wrappedConn
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) listenApi(conn *wsc.Conn, u bool) {
|
func (c *websocketClient) listenApi(conn *websocketConn, u bool) {
|
||||||
defer conn.Close()
|
defer conn.Close()
|
||||||
for {
|
for {
|
||||||
var buf []byte
|
_, buf, err := conn.ReadMessage()
|
||||||
err := wsc.Message.Receive(conn, &buf)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
log.Warnf("监听反向WS API时出现错误: %v", err)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
j := gjson.ParseBytes(buf)
|
|
||||||
t := strings.ReplaceAll(j.Get("action").Str, "_async", "")
|
go conn.handleRequest(c.bot, buf)
|
||||||
log.Debugf("反向WS接收到API调用: %v 参数: %v", t, j.Get("params").Raw)
|
|
||||||
if f, ok := wsApi[t]; ok {
|
|
||||||
ret := f(c.bot, j.Get("params"))
|
|
||||||
if j.Get("echo").Exists() {
|
|
||||||
ret["echo"] = j.Get("echo").Value()
|
|
||||||
}
|
|
||||||
c.pushLock.Lock()
|
|
||||||
_, _ = conn.Write([]byte(ret.ToJson()))
|
|
||||||
c.pushLock.Unlock()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if c.conf.ReverseReconnectInterval != 0 {
|
if c.conf.ReverseReconnectInterval != 0 {
|
||||||
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
||||||
if u {
|
if !u {
|
||||||
c.connectUniversal()
|
go c.connectApi()
|
||||||
return
|
|
||||||
}
|
}
|
||||||
c.connectApi()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) onBotPushEvent(m coolq.MSG) {
|
func (c *websocketClient) onBotPushEvent(m coolq.MSG) {
|
||||||
c.pushLock.Lock()
|
|
||||||
defer c.pushLock.Unlock()
|
|
||||||
if c.eventConn != nil {
|
if c.eventConn != nil {
|
||||||
log.Debugf("向WS服务器 %v 推送Event: %v", c.eventConn.RemoteAddr().String(), m.ToJson())
|
log.Debugf("向WS服务器 %v 推送Event: %v", c.eventConn.RemoteAddr().String(), m.ToJson())
|
||||||
if _, err := c.eventConn.Write([]byte(m.ToJson())); err != nil {
|
conn := c.eventConn
|
||||||
|
conn.Lock()
|
||||||
|
defer conn.Unlock()
|
||||||
|
_ = c.eventConn.SetWriteDeadline(time.Now().Add(time.Second * 15))
|
||||||
|
if err := c.eventConn.WriteJSON(m); err != nil {
|
||||||
|
log.Warnf("向WS服务器 %v 推送Event时出现错误: %v", c.eventConn.RemoteAddr().String(), err)
|
||||||
_ = c.eventConn.Close()
|
_ = c.eventConn.Close()
|
||||||
if c.conf.ReverseReconnectInterval != 0 {
|
if c.conf.ReverseReconnectInterval != 0 {
|
||||||
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
||||||
@ -200,16 +206,29 @@ func (c *websocketClient) onBotPushEvent(m coolq.MSG) {
|
|||||||
}
|
}
|
||||||
if c.universalConn != nil {
|
if c.universalConn != nil {
|
||||||
log.Debugf("向WS服务器 %v 推送Event: %v", c.universalConn.RemoteAddr().String(), m.ToJson())
|
log.Debugf("向WS服务器 %v 推送Event: %v", c.universalConn.RemoteAddr().String(), m.ToJson())
|
||||||
_, _ = c.universalConn.Write([]byte(m.ToJson()))
|
conn := c.universalConn
|
||||||
|
conn.Lock()
|
||||||
|
defer conn.Unlock()
|
||||||
|
_ = c.universalConn.SetWriteDeadline(time.Now().Add(time.Second * 15))
|
||||||
|
if err := c.universalConn.WriteJSON(m); err != nil {
|
||||||
|
log.Warnf("向WS服务器 %v 推送Event时出现错误: %v", c.universalConn.RemoteAddr().String(), err)
|
||||||
|
_ = c.universalConn.Close()
|
||||||
|
if c.conf.ReverseReconnectInterval != 0 {
|
||||||
|
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
||||||
|
c.connectUniversal()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) event(w http.ResponseWriter, r *http.Request) {
|
func (s *websocketServer) event(w http.ResponseWriter, r *http.Request) {
|
||||||
if s.token != "" {
|
if s.token != "" {
|
||||||
if r.URL.Query().Get("access_token") != s.token && strings.SplitN(r.Header.Get("Authorization"), " ", 2)[1] != s.token {
|
if auth := r.URL.Query().Get("access_token"); auth != s.token {
|
||||||
log.Warnf("已拒绝 %v 的 Websocket 请求: Token错误", r.RemoteAddr)
|
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
||||||
w.WriteHeader(401)
|
log.Warnf("已拒绝 %v 的 Websocket 请求: Token鉴权失败", r.RemoteAddr)
|
||||||
return
|
w.WriteHeader(401)
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
c, err := upgrader.Upgrade(w, r, nil)
|
c, err := upgrader.Upgrade(w, r, nil)
|
||||||
@ -218,18 +237,29 @@ func (s *websocketServer) event(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
err = c.WriteMessage(websocket.TextMessage, []byte(s.handshake))
|
err = c.WriteMessage(websocket.TextMessage, []byte(s.handshake))
|
||||||
if err == nil {
|
if err != nil {
|
||||||
log.Infof("接受 Websocket 连接: %v (/event)", r.RemoteAddr)
|
log.Warnf("Websocket 握手时出现错误: %v", err)
|
||||||
s.eventConn = append(s.eventConn, c)
|
c.Close()
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.Infof("接受 Websocket 连接: %v (/event)", r.RemoteAddr)
|
||||||
|
|
||||||
|
conn := &websocketConn{Conn: c}
|
||||||
|
|
||||||
|
s.eventConnMutex.Lock()
|
||||||
|
s.eventConn = append(s.eventConn, conn)
|
||||||
|
s.eventConnMutex.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) api(w http.ResponseWriter, r *http.Request) {
|
func (s *websocketServer) api(w http.ResponseWriter, r *http.Request) {
|
||||||
if s.token != "" {
|
if s.token != "" {
|
||||||
if r.URL.Query().Get("access_token") != s.token && strings.SplitN(r.Header.Get("Authorization"), " ", 2)[1] != s.token {
|
if auth := r.URL.Query().Get("access_token"); auth != s.token {
|
||||||
log.Warnf("已拒绝 %v 的 Websocket 请求: Token错误", r.RemoteAddr)
|
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
||||||
w.WriteHeader(401)
|
log.Warnf("已拒绝 %v 的 Websocket 请求: Token鉴权失败", r.RemoteAddr)
|
||||||
return
|
w.WriteHeader(401)
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
c, err := upgrader.Upgrade(w, r, nil)
|
c, err := upgrader.Upgrade(w, r, nil)
|
||||||
@ -238,15 +268,18 @@ func (s *websocketServer) api(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
log.Infof("接受 Websocket 连接: %v (/api)", r.RemoteAddr)
|
log.Infof("接受 Websocket 连接: %v (/api)", r.RemoteAddr)
|
||||||
go s.listenApi(c)
|
conn := &websocketConn{Conn: c}
|
||||||
|
go s.listenApi(conn)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) any(w http.ResponseWriter, r *http.Request) {
|
func (s *websocketServer) any(w http.ResponseWriter, r *http.Request) {
|
||||||
if s.token != "" {
|
if s.token != "" {
|
||||||
if r.URL.Query().Get("access_token") != s.token && strings.SplitN(r.Header.Get("Authorization"), " ", 2)[1] != s.token {
|
if auth := r.URL.Query().Get("access_token"); auth != s.token {
|
||||||
log.Warnf("已拒绝 %v 的 Websocket 请求: Token错误", r.RemoteAddr)
|
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
||||||
w.WriteHeader(401)
|
log.Warnf("已拒绝 %v 的 Websocket 请求: Token鉴权失败", r.RemoteAddr)
|
||||||
return
|
w.WriteHeader(401)
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
c, err := upgrader.Upgrade(w, r, nil)
|
c, err := upgrader.Upgrade(w, r, nil)
|
||||||
@ -255,52 +288,74 @@ func (s *websocketServer) any(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
err = c.WriteMessage(websocket.TextMessage, []byte(s.handshake))
|
err = c.WriteMessage(websocket.TextMessage, []byte(s.handshake))
|
||||||
if err == nil {
|
if err != nil {
|
||||||
log.Infof("接受 Websocket 连接: %v (/)", r.RemoteAddr)
|
log.Warnf("Websocket 握手时出现错误: %v", err)
|
||||||
s.eventConn = append(s.eventConn, c)
|
c.Close()
|
||||||
s.listenApi(c)
|
return
|
||||||
}
|
}
|
||||||
|
log.Infof("接受 Websocket 连接: %v (/)", r.RemoteAddr)
|
||||||
|
conn := &websocketConn{Conn: c}
|
||||||
|
s.eventConn = append(s.eventConn, conn)
|
||||||
|
s.listenApi(conn)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) listenApi(c *websocket.Conn) {
|
func (s *websocketServer) listenApi(c *websocketConn) {
|
||||||
defer c.Close()
|
defer c.Close()
|
||||||
for {
|
for {
|
||||||
t, payload, err := c.ReadMessage()
|
t, payload, err := c.ReadMessage()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
if t == websocket.TextMessage {
|
if t == websocket.TextMessage {
|
||||||
j := gjson.ParseBytes(payload)
|
go c.handleRequest(s.bot, payload)
|
||||||
t := strings.ReplaceAll(j.Get("action").Str, "_async", "") //TODO: async support
|
|
||||||
log.Debugf("WS接收到API调用: %v 参数: %v", t, j.Get("params").Raw)
|
|
||||||
if f, ok := wsApi[t]; ok {
|
|
||||||
ret := f(s.bot, j.Get("params"))
|
|
||||||
if j.Get("echo").Exists() {
|
|
||||||
ret["echo"] = j.Get("echo").Value()
|
|
||||||
}
|
|
||||||
s.pushLock.Lock()
|
|
||||||
_ = c.WriteJSON(ret)
|
|
||||||
s.pushLock.Unlock()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) onBotPushEvent(m coolq.MSG) {
|
func (c *websocketConn) handleRequest(bot *coolq.CQBot, payload []byte) {
|
||||||
s.pushLock.Lock()
|
defer func() {
|
||||||
defer s.pushLock.Unlock()
|
if err := recover(); err != nil {
|
||||||
pos := 0
|
log.Printf("处置WS命令时发生无法恢复的异常:%v", err)
|
||||||
for _, conn := range s.eventConn {
|
c.Close()
|
||||||
log.Debugf("向WS客户端 %v 推送Event: %v", conn.RemoteAddr().String(), m.ToJson())
|
|
||||||
err := conn.WriteMessage(websocket.TextMessage, []byte(m.ToJson()))
|
|
||||||
if err != nil {
|
|
||||||
_ = conn.Close()
|
|
||||||
s.eventConn = append(s.eventConn[:pos], s.eventConn[pos+1:]...)
|
|
||||||
if pos > 0 {
|
|
||||||
pos++
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
pos++
|
}()
|
||||||
|
global.RateLimit(context.Background())
|
||||||
|
j := gjson.ParseBytes(payload)
|
||||||
|
t := strings.ReplaceAll(j.Get("action").Str, "_async", "")
|
||||||
|
log.Debugf("WS接收到API调用: %v 参数: %v", t, j.Get("params").Raw)
|
||||||
|
if f, ok := wsApi[t]; ok {
|
||||||
|
ret := f(bot, j.Get("params"))
|
||||||
|
if j.Get("echo").Exists() {
|
||||||
|
ret["echo"] = j.Get("echo").Value()
|
||||||
|
}
|
||||||
|
c.Lock()
|
||||||
|
defer c.Unlock()
|
||||||
|
_ = c.WriteJSON(ret)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *websocketServer) onBotPushEvent(m coolq.MSG) {
|
||||||
|
s.eventConnMutex.Lock()
|
||||||
|
defer s.eventConnMutex.Unlock()
|
||||||
|
for i, l := 0, len(s.eventConn); i < l; i++ {
|
||||||
|
conn := s.eventConn[i]
|
||||||
|
log.Debugf("向WS客户端 %v 推送Event: %v", conn.RemoteAddr().String(), m.ToJson())
|
||||||
|
conn.Lock()
|
||||||
|
if err := conn.WriteMessage(websocket.TextMessage, []byte(m.ToJson())); err != nil {
|
||||||
|
_ = conn.Close()
|
||||||
|
next := i + 1
|
||||||
|
if next >= l {
|
||||||
|
next = l - 1
|
||||||
|
}
|
||||||
|
s.eventConn[i], s.eventConn[next] = s.eventConn[next], s.eventConn[i]
|
||||||
|
s.eventConn = append(s.eventConn[:next], s.eventConn[next+1:]...)
|
||||||
|
i--
|
||||||
|
l--
|
||||||
|
conn = nil
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
conn.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -312,37 +367,43 @@ var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|||||||
return bot.CQGetFriendList()
|
return bot.CQGetFriendList()
|
||||||
},
|
},
|
||||||
"get_group_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_group_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetGroupList()
|
return bot.CQGetGroupList(p.Get("no_cache").Bool())
|
||||||
},
|
},
|
||||||
"get_group_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_group_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetGroupInfo(p.Get("group_id").Int())
|
return bot.CQGetGroupInfo(p.Get("group_id").Int())
|
||||||
},
|
},
|
||||||
"get_group_member_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_group_member_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetGroupMemberList(p.Get("group_id").Int())
|
return bot.CQGetGroupMemberList(p.Get("group_id").Int(), p.Get("no_cache").Bool())
|
||||||
},
|
},
|
||||||
"get_group_member_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_group_member_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetGroupMemberInfo(
|
return bot.CQGetGroupMemberInfo(
|
||||||
p.Get("group_id").Int(), p.Get("user_id").Int(),
|
p.Get("group_id").Int(), p.Get("user_id").Int(),
|
||||||
p.Get("no_cache").Bool(),
|
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
"send_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"send_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
autoEscape := global.EnsureBool(p.Get("auto_escape"), false)
|
||||||
|
if p.Get("message_type").Str == "private" {
|
||||||
|
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"), autoEscape)
|
||||||
|
}
|
||||||
|
if p.Get("message_type").Str == "group" {
|
||||||
|
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"), autoEscape)
|
||||||
|
}
|
||||||
if p.Get("group_id").Int() != 0 {
|
if p.Get("group_id").Int() != 0 {
|
||||||
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"))
|
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"), autoEscape)
|
||||||
}
|
}
|
||||||
if p.Get("user_id").Int() != 0 {
|
if p.Get("user_id").Int() != 0 {
|
||||||
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"))
|
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"), autoEscape)
|
||||||
}
|
}
|
||||||
return coolq.MSG{}
|
return coolq.MSG{}
|
||||||
},
|
},
|
||||||
"send_group_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"send_group_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"))
|
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"), global.EnsureBool(p.Get("auto_escape"), false))
|
||||||
},
|
},
|
||||||
"send_group_forward_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"send_group_forward_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSendGroupForwardMessage(p.Get("group_id").Int(), p.Get("messages"))
|
return bot.CQSendGroupForwardMessage(p.Get("group_id").Int(), p.Get("messages"))
|
||||||
},
|
},
|
||||||
"send_private_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"send_private_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"))
|
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"), global.EnsureBool(p.Get("auto_escape"), false))
|
||||||
},
|
},
|
||||||
"delete_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"delete_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQDeleteMessage(int32(p.Get("message_id").Int()))
|
return bot.CQDeleteMessage(int32(p.Get("message_id").Int()))
|
||||||
@ -363,7 +424,7 @@ var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|||||||
if p.Get("approve").Exists() {
|
if p.Get("approve").Exists() {
|
||||||
apr = p.Get("approve").Bool()
|
apr = p.Get("approve").Bool()
|
||||||
}
|
}
|
||||||
return bot.CQProcessGroupRequest(p.Get("flag").Str, subType, apr)
|
return bot.CQProcessGroupRequest(p.Get("flag").Str, subType, p.Get("reason").Str, apr)
|
||||||
},
|
},
|
||||||
"set_group_card": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"set_group_card": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSetGroupCard(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("card").Str)
|
return bot.CQSetGroupCard(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("card").Str)
|
||||||
@ -372,7 +433,7 @@ var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|||||||
return bot.CQSetGroupSpecialTitle(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("special_title").Str)
|
return bot.CQSetGroupSpecialTitle(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("special_title").Str)
|
||||||
},
|
},
|
||||||
"set_group_kick": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"set_group_kick": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSetGroupKick(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("message").Str)
|
return bot.CQSetGroupKick(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("message").Str, p.Get("reject_add_request").Bool())
|
||||||
},
|
},
|
||||||
"set_group_ban": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"set_group_ban": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSetGroupBan(p.Get("group_id").Int(), p.Get("user_id").Int(), func() uint32 {
|
return bot.CQSetGroupBan(p.Get("group_id").Int(), p.Get("user_id").Int(), func() uint32 {
|
||||||
@ -391,7 +452,18 @@ var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|||||||
}())
|
}())
|
||||||
},
|
},
|
||||||
"set_group_name": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"set_group_name": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSetGroupName(p.Get("group_id").Int(), p.Get("name").Str)
|
return bot.CQSetGroupName(p.Get("group_id").Int(), p.Get("group_name").Str)
|
||||||
|
},
|
||||||
|
"set_group_admin": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQSetGroupAdmin(p.Get("group_id").Int(), p.Get("user_id").Int(), func() bool {
|
||||||
|
if p.Get("enable").Exists() {
|
||||||
|
return p.Get("enable").Bool()
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}())
|
||||||
|
},
|
||||||
|
"_send_group_notice": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQSetGroupMemo(p.Get("group_id").Int(), p.Get("content").Str)
|
||||||
},
|
},
|
||||||
"set_group_leave": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"set_group_leave": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQSetGroupLeave(p.Get("group_id").Int())
|
return bot.CQSetGroupLeave(p.Get("group_id").Int())
|
||||||
@ -402,8 +474,24 @@ var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|||||||
"get_forward_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_forward_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetForwardMessage(p.Get("message_id").Str)
|
return bot.CQGetForwardMessage(p.Get("message_id").Str)
|
||||||
},
|
},
|
||||||
"get_group_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetGroupMessage(int32(p.Get("message_id").Int()))
|
return bot.CQGetMessage(int32(p.Get("message_id").Int()))
|
||||||
|
},
|
||||||
|
"get_group_honor_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetGroupHonorInfo(p.Get("group_id").Int(), p.Get("type").Str)
|
||||||
|
},
|
||||||
|
"set_restart": func(c *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
var delay int64 = 0
|
||||||
|
delay = p.Get("delay").Int()
|
||||||
|
if delay < 0 {
|
||||||
|
delay = 0
|
||||||
|
}
|
||||||
|
defer func(delay int64) {
|
||||||
|
time.Sleep(time.Duration(delay) * time.Millisecond)
|
||||||
|
Restart <- struct{}{}
|
||||||
|
}(delay)
|
||||||
|
return coolq.MSG{"data": nil, "retcode": 0, "status": "async"}
|
||||||
|
|
||||||
},
|
},
|
||||||
"can_send_image": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"can_send_image": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQCanSendImage()
|
return bot.CQCanSendImage()
|
||||||
@ -411,12 +499,45 @@ var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|||||||
"can_send_record": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"can_send_record": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQCanSendRecord()
|
return bot.CQCanSendRecord()
|
||||||
},
|
},
|
||||||
|
"get_stranger_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetStrangerInfo(p.Get("user_id").Int())
|
||||||
|
},
|
||||||
"get_status": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_status": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetStatus()
|
return bot.CQGetStatus()
|
||||||
},
|
},
|
||||||
"get_version_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
"get_version_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQGetVersionInfo()
|
return bot.CQGetVersionInfo()
|
||||||
},
|
},
|
||||||
|
"get_group_system_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetGroupSystemMessages()
|
||||||
|
},
|
||||||
|
"get_group_file_system_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetGroupFileSystemInfo(p.Get("group_id").Int())
|
||||||
|
},
|
||||||
|
"get_group_root_files": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetGroupRootFiles(p.Get("group_id").Int())
|
||||||
|
},
|
||||||
|
"get_group_files_by_folder": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetGroupFilesByFolderId(p.Get("group_id").Int(), p.Get("folder_id").Str)
|
||||||
|
},
|
||||||
|
"get_group_file_url": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetGroupFileUrl(p.Get("group_id").Int(), p.Get("file_id").Str, int32(p.Get("busid").Int()))
|
||||||
|
},
|
||||||
|
"_get_vip_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetVipInfo(p.Get("user_id").Int())
|
||||||
|
},
|
||||||
|
"reload_event_filter": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQReloadEventFilter()
|
||||||
|
},
|
||||||
|
".ocr_image": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQOcrImage(p.Get("image").Str)
|
||||||
|
},
|
||||||
|
".get_word_slices": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQGetWordSlices(p.Get("content").Str)
|
||||||
|
},
|
||||||
|
"set_group_portrait": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
|
return bot.CQSetGroupPortrait(p.Get("group_id").Int(), p.Get("file").String(), p.Get("cache").String())
|
||||||
|
},
|
||||||
".handle_quick_operation": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
".handle_quick_operation": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
||||||
return bot.CQHandleQuickOperation(p.Get("context"), p.Get("operation"))
|
return bot.CQHandleQuickOperation(p.Get("context"), p.Get("operation"))
|
||||||
},
|
},
|
||||||
|
Reference in New Issue
Block a user