mirror of
https://github.com/Mrs4s/go-cqhttp.git
synced 2025-06-30 20:03:24 +00:00
Compare commits
254 Commits
v0.9.36-fi
...
v0.9.40-fi
Author | SHA1 | Date | |
---|---|---|---|
a90af94eda | |||
93da16897a | |||
604a05b625 | |||
af9e130273 | |||
3719ef1fe8 | |||
45102614b8 | |||
b385ed20b5 | |||
ca826f67b3 | |||
49a6cfd50a | |||
d2fb26da10 | |||
80517e25ad | |||
d4e2e2fb69 | |||
f86bb69536 | |||
a0d4dfd0eb | |||
8ef877777d | |||
b7991fc229 | |||
5b24492702 | |||
e88854a058 | |||
52a0061360 | |||
dd13fe6847 | |||
3bb3000e79 | |||
550e8d3531 | |||
42e5b0564c | |||
62024a4403 | |||
65112f8752 | |||
36c0aaae6b | |||
976f2233b4 | |||
f1a1b0ec30 | |||
3cc0453958 | |||
fff33e304c | |||
977ee0c859 | |||
4021f95c24 | |||
b1ca827187 | |||
c1a36caa99 | |||
e9b6e477f1 | |||
6d0a6c137d | |||
69703f560f | |||
580a5feabf | |||
5db56b99c6 | |||
e56a297437 | |||
f9d1cf93a2 | |||
916c79d2cf | |||
1dcfa0d025 | |||
d2be362f84 | |||
a0cdf2c410 | |||
8f8529af9e | |||
0afbe3d64a | |||
3851a60869 | |||
a9fb84eb47 | |||
56f648aaca | |||
945fbf6a2b | |||
185f11f6a0 | |||
8602490711 | |||
a1fb629798 | |||
396ce82baa | |||
2d90a16fbf | |||
afd0c8987d | |||
70a4d129c8 | |||
1ff9de190d | |||
569a718dc6 | |||
32771eb8b0 | |||
2ce0d145e8 | |||
02581ff8a9 | |||
ee0da0a14f | |||
71ce6f362f | |||
ce30d63907 | |||
19556cbab3 | |||
3bfdc4d8a2 | |||
fba4819adc | |||
e30e21e296 | |||
6e8ef3ef0f | |||
31cfb9283e | |||
0e71acaf38 | |||
879cbe4f19 | |||
775a210e3e | |||
f5bd4644fd | |||
e8129a7e57 | |||
adbe194e63 | |||
7e5b293d36 | |||
8dc2349a9b | |||
337b7fd134 | |||
967e79ecd1 | |||
cc14be66bd | |||
667ad8fbd6 | |||
1b3842fc98 | |||
203b68b740 | |||
eaf7067196 | |||
82bca7183d | |||
13975c02df | |||
d5c624d6f2 | |||
3bbcdc9171 | |||
c72bc3b1b0 | |||
22a02542af | |||
ff75b0a2a9 | |||
4a13f35003 | |||
8d0a0b7f4d | |||
4c570bdfe6 | |||
d4e8a3df4f | |||
307a09dd11 | |||
4445af6ff2 | |||
7b71cd9219 | |||
d291aa02fe | |||
dbd0f85025 | |||
0bacbde4d7 | |||
c92dd39191 | |||
7da1918c0c | |||
89d466d3e1 | |||
5b57aeb1d1 | |||
08567ec240 | |||
08dea6dcf1 | |||
252293afb9 | |||
ca0b931fa4 | |||
01dca0305c | |||
6053918ee3 | |||
e7547ed257 | |||
96a036201d | |||
ccee8ac77a | |||
ae806c53bb | |||
a9619b25bd | |||
69cda4e029 | |||
dc18944a98 | |||
717f816294 | |||
430e734ff6 | |||
21531fd70a | |||
2214570c4f | |||
18a839b699 | |||
23b90e288c | |||
22c9a6728b | |||
81ff71ee3b | |||
939307efc7 | |||
bf65b576f3 | |||
817bb0e493 | |||
d2529a04c9 | |||
02446f7791 | |||
cff7ac8e0b | |||
6f5e582213 | |||
a74c623cc1 | |||
614109e116 | |||
6d5b0e01a2 | |||
13df58331c | |||
61c9939f03 | |||
f6e620b412 | |||
a2a0b4c86f | |||
8532c4ecbe | |||
5aaade7835 | |||
3c309a8e06 | |||
7c794d113f | |||
c918182dec | |||
c0a813a093 | |||
61625f2e32 | |||
8a81b67511 | |||
164feca7c7 | |||
2ac94a7e95 | |||
219005d384 | |||
e19e797aac | |||
aa54ed5610 | |||
360f7188e1 | |||
0002185e45 | |||
5480deff4b | |||
9a81b02791 | |||
5280c68bfc | |||
0bd5afbf21 | |||
297284f957 | |||
eb618393e5 | |||
bdc2b03690 | |||
63d26dfdb5 | |||
7e49ed4869 | |||
19b6106c93 | |||
77a4f9a64f | |||
c69129f80b | |||
61f4e7826a | |||
ddd303375d | |||
46cae86442 | |||
d16fd6f34d | |||
1001cefb6f | |||
1d9f719d13 | |||
468befb0fd | |||
9f99b1571d | |||
77445fe96e | |||
bbd85eb219 | |||
2156b6083b | |||
b47c2aeac6 | |||
9beae584de | |||
59cd986050 | |||
7a9a021c54 | |||
de4de5052d | |||
b075280d2b | |||
4cb3d8f33e | |||
0cd06daf08 | |||
7ac31a8b76 | |||
c9c6e48b8c | |||
0c3ba714f7 | |||
1f7dcc4e91 | |||
ef713bc59c | |||
85624d1a27 | |||
b2d8a61a99 | |||
38a2c3a945 | |||
eaec0e545f | |||
50f0f572bc | |||
36532a8782 | |||
620ecdbdbb | |||
7737fc8457 | |||
e03a989083 | |||
56da9adeab | |||
249e528457 | |||
1eeae13b4a | |||
fe83ce716e | |||
a5a842a337 | |||
be47ff3437 | |||
392904f144 | |||
a4c3ef8923 | |||
816c142822 | |||
2e2002668b | |||
c58066e287 | |||
87ada9c57e | |||
b717e23626 | |||
f501b31152 | |||
2d020bc7f7 | |||
1843bd6a4e | |||
938ebf630f | |||
2bbf969cfa | |||
416a3460ab | |||
f98c334089 | |||
0e5276ea4d | |||
6113eb9200 | |||
1e56c4f986 | |||
9573ce1f16 | |||
a02219a64f | |||
773952d6d6 | |||
b8916d2570 | |||
6769421b7a | |||
4262430340 | |||
d20ff4eeb2 | |||
935faa1159 | |||
9c81b81820 | |||
808c275f3d | |||
d6fba106d3 | |||
e25536fbe5 | |||
1de12bdcd2 | |||
f8fc023e2f | |||
6c89921a87 | |||
cfc5d6d2ff | |||
a6a78dfcac | |||
5a8d918d9d | |||
ec3c2d6244 | |||
c1af182732 | |||
4c942964c7 | |||
0080011f40 | |||
589fbd21df | |||
97acd956f0 | |||
f7144523d9 | |||
8b1d950a29 | |||
8b14409aaf | |||
464122af4a |
30
.github/ISSUE_TEMPLATE/bug--.md
vendored
30
.github/ISSUE_TEMPLATE/bug--.md
vendored
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
name: Bug汇报
|
name: Bug汇报
|
||||||
about: Create a report to help us improve
|
about: 遇到了bug? 你可以在这里开始汇报
|
||||||
title: ''
|
title: ''
|
||||||
labels: ''
|
labels: ''
|
||||||
assignees: ''
|
assignees: ''
|
||||||
@ -8,22 +8,34 @@ assignees: ''
|
|||||||
---
|
---
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
在提出ISSUE前,希望您能花费一定的时间执行以下几个步骤以方便我们定位BUG.
|
在您发布此Issue前, 请您花一点时间查看下面几条指引🔽
|
||||||
1: 确定没有相同问题的ISSUE已被提出.
|
|
||||||
2: 准确填写环境信息.
|
1: ❗ | 确定没有相同问题的ISSUE已被提出. (教程: https://github.com/Mrs4s/go-cqhttp/issues/633)
|
||||||
3: 最好能打开DEBUG模式并复现相关问题.
|
2: 🌎| 请准确填写环境信息.
|
||||||
4: 如果涉及内存泄漏/CPU占用异常请打开DEBUG模式并下载pprof性能分析.
|
3: ❔ | 打开DEBUG模式复现,并提供出现问题前后至少 10 秒的完整日志内容。请自行删除日志内存在的个人信息及敏感内容。
|
||||||
|
4: ⚠ | 如果涉及内存泄漏/CPU占用异常请打开DEBUG模式并下载pprof性能分析.
|
||||||
|
|
||||||
|
注: 如果您不知道如何有效、精准地表述,我们建议您先阅读《提问的智慧》
|
||||||
|
(链接: https://github.com/ryanhanwu/How-To-Ask-Questions-The-Smart-Way/blob/main/README-zh_CN.md)
|
||||||
|
|
||||||
|
请确保您已经仔细阅读此教程,并勾选下方的确认框。(将 [ ] 修改为 [x])
|
||||||
|
--------
|
||||||
|
- [ ] 我已经仔细阅读上述教程和"提问前需知 [图+文]": https://github.com/Mrs4s/go-cqhttp/issues/633
|
||||||
|
- [ ] 我已知晓并同意,如果我不遵循以下格式提交 Issue,或者我使用的并非最新版本,或者我没有提供足够的环境信息,则我的 Issue 可能会被无条件自动关闭和锁定。
|
||||||
|
- [ ] 我已知晓并同意,我仅需要把选项前的 [ ] 替换为 [x]。如果我删除、修改这些复选框的其他部分,或是在 x 之前或之后留了空格,则我的 Issue 可能会被无条件自动关闭和锁定。
|
||||||
|
- [ ] 我已知晓并同意,此处仅用于汇报程序中存在的问题。若这个 Issue 是关于其他非程序本身问题或是新功能需求,则我的 Issue 可能会被无条件自动关闭和锁定。(这些问题应当在 Discussion 板块提出。)
|
||||||
|
--------
|
||||||
-->
|
-->
|
||||||
|
|
||||||
**环境信息**
|
**环境信息**
|
||||||
请根据实际使用环境修改以下信息
|
<!-- 请根据实际使用环境修改以下信息。请勿留空。 -->
|
||||||
go-cqhttp版本:
|
go-cqhttp版本:
|
||||||
运行环境:
|
运行环境:
|
||||||
连接方式:
|
连接方式:
|
||||||
使用协议:
|
使用协议:
|
||||||
|
|
||||||
**bug内容**
|
**bug内容**
|
||||||
请在这里详细描述bug的内容
|
<!-- 请在这里详细描述bug的内容 -->
|
||||||
|
|
||||||
**复现方法**
|
**复现方法**
|
||||||
请在这里分步骤的描述如何复现这个bug
|
<!-- 请在这里分步骤的描述如何复现这个bug -->
|
||||||
|
5
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
5
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
blank_issues_enabled: false
|
||||||
|
contact_links:
|
||||||
|
- name: 提出新建议/功能
|
||||||
|
url: https://github.com/Mrs4s/go-cqhttp/discussions/new
|
||||||
|
about: 请在这里提出你的建议或功能
|
64
.github/workflows/ci.yml
vendored
64
.github/workflows/ci.yml
vendored
@ -14,41 +14,47 @@ 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/amd64
|
# build and publish in parallel: linux/386, linux/amd64, windows/386, windows/amd64, darwin/amd64, darwin/arm64
|
||||||
goos: [linux, windows, darwin]
|
goos: [linux, windows, darwin]
|
||||||
goarch: ["386", amd64, arm]
|
goarch: ["386", amd64, arm, arm64]
|
||||||
exclude:
|
exclude:
|
||||||
- goos: darwin
|
- goos: darwin
|
||||||
goarch: arm
|
goarch: arm
|
||||||
- goos: darwin
|
- goos: darwin
|
||||||
goarch: "386"
|
goarch: "386"
|
||||||
|
- goos: windows
|
||||||
|
goarch: arm64
|
||||||
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.3
|
||||||
uses: actions/setup-go@v2.1.3
|
with:
|
||||||
with:
|
go-version: 1.16
|
||||||
go-version: 1.15
|
- name: Build binary file
|
||||||
|
env:
|
||||||
- name: Build binary file
|
GOOS: ${{ matrix.goos }}
|
||||||
env:
|
GOARCH: ${{ matrix.goarch }}
|
||||||
GOOS: ${{ matrix.goos }}
|
IS_PR: ${{ !!github.head_ref }}
|
||||||
GOARCH: ${{ matrix.goarch }}
|
run: |
|
||||||
IS_PR: ${{ !!github.head_ref }}
|
if [ $GOOS = "windows" ]; then export BINARY_SUFFIX="$BINARY_SUFFIX.exe"; fi
|
||||||
run: |
|
if $IS_PR ; then echo $PR_PROMPT; fi
|
||||||
if [ $GOOS = "windows" ]; then export BINARY_SUFFIX="$BINARY_SUFFIX.exe"; fi
|
export BINARY_NAME="$BINARY_PREFIX$GOOS_$GOARCH$BINARY_SUFFIX"
|
||||||
if $IS_PR ; then echo $PR_PROMPT; fi
|
export CGO_ENABLED=0
|
||||||
export BINARY_NAME="$BINARY_PREFIX$GOOS_$GOARCH$BINARY_SUFFIX"
|
go build -o "output/$BINARY_NAME" -trimpath -ldflags "$LD_FLAGS" .
|
||||||
export CGO_ENABLED=0
|
- name: Upload artifact
|
||||||
go build -o "output/$BINARY_NAME" -ldflags "$LD_FLAGS" .
|
uses: actions/upload-artifact@v2
|
||||||
|
if: ${{ !github.head_ref }}
|
||||||
- name: Upload artifact
|
with:
|
||||||
uses: actions/upload-artifact@v2
|
name: ${{ matrix.goos }}_${{ matrix.goarch }}
|
||||||
if: ${{ !github.head_ref }}
|
path: output/
|
||||||
with:
|
|
||||||
name: ${{ matrix.goos }}_${{ matrix.goarch }}
|
|
||||||
path: output/
|
|
||||||
|
|
||||||
|
|
||||||
|
golangci:
|
||||||
|
name: lint
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: golangci-lint
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: v1.29
|
||||||
|
12
.github/workflows/issuebot.yml
vendored
Normal file
12
.github/workflows/issuebot.yml
vendored
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
name: Issuebot
|
||||||
|
|
||||||
|
on:
|
||||||
|
issues:
|
||||||
|
types: [opened]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
new_issue:
|
||||||
|
name: Run Issuebot on new Issue
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: wfjsw/actions-gocqhttp-issue-prefilter@master
|
9
.github/workflows/release.yml
vendored
9
.github/workflows/release.yml
vendored
@ -8,14 +8,16 @@ 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/386, darwin/amd64, darwin/arm64
|
||||||
goos: [linux, windows, darwin]
|
goos: [linux, windows, darwin]
|
||||||
goarch: ["386", amd64, arm]
|
goarch: ["386", amd64, arm, arm64]
|
||||||
exclude:
|
exclude:
|
||||||
- goos: darwin
|
- goos: darwin
|
||||||
goarch: arm
|
goarch: arm
|
||||||
- goos: darwin
|
- goos: darwin
|
||||||
goarch: "386"
|
goarch: "386"
|
||||||
|
- goos: windows
|
||||||
|
goarch: arm64
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
@ -28,6 +30,7 @@ jobs:
|
|||||||
github_token: ${{ secrets.GITHUB_TOKEN }}
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
goos: ${{ matrix.goos }}
|
goos: ${{ matrix.goos }}
|
||||||
goarch: ${{ matrix.goarch }}
|
goarch: ${{ matrix.goarch }}
|
||||||
goversion: "https://golang.org/dl/go1.15.3.linux-amd64.tar.gz"
|
goversion: "https://golang.org/dl/go1.16.linux-amd64.tar.gz"
|
||||||
|
build_flags: -trimpath
|
||||||
ldflags: -w -s -X "github.com/Mrs4s/go-cqhttp/coolq.Version=${{ env.RELEASE_VERSION }}"
|
ldflags: -w -s -X "github.com/Mrs4s/go-cqhttp/coolq.Version=${{ env.RELEASE_VERSION }}"
|
||||||
|
|
5
.gitignore
vendored
5
.gitignore
vendored
@ -1,2 +1,7 @@
|
|||||||
vendor/
|
vendor/
|
||||||
.idea
|
.idea
|
||||||
|
config.hjson
|
||||||
|
device.json
|
||||||
|
codec/
|
||||||
|
data/
|
||||||
|
logs/
|
18
README.md
18
README.md
@ -6,7 +6,8 @@
|
|||||||
|
|
||||||
# go-cqhttp
|
# go-cqhttp
|
||||||
|
|
||||||
_✨ 基于 [Mirai](https://github.com/mamoe/mirai) 以及 [MiraiGo](https://github.com/Mrs4s/MiraiGo) 的 [cqhttp](https://github.com/howmanybots/onebot/blob/master/README.md) golang 原生实现 ✨_
|
_✨ 基于 [Mirai](https://github.com/mamoe/mirai) 以及 [MiraiGo](https://github.com/Mrs4s/MiraiGo) 的 [OneBot](https://github.com/howmanybots/onebot/blob/master/README.md) Golang 原生实现 ✨_
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -23,23 +24,21 @@ _✨ 基于 [Mirai](https://github.com/mamoe/mirai) 以及 [MiraiGo](https://git
|
|||||||
<a href="https://github.com/Mrs4s/go-cqhttp/actions">
|
<a href="https://github.com/Mrs4s/go-cqhttp/actions">
|
||||||
<img src="https://github.com/Mrs4s/go-cqhttp/workflows/CI/badge.svg" alt="action">
|
<img src="https://github.com/Mrs4s/go-cqhttp/workflows/CI/badge.svg" alt="action">
|
||||||
</a>
|
</a>
|
||||||
|
<a href="https://goreportcard.com/report/github.com/Mrs4s/go-cqhttp">
|
||||||
|
<img src="https://goreportcard.com/badge/github.com/Mrs4s/go-cqhttp" alt="GoReportCard">
|
||||||
|
</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<a href="http://ishkong.github.io/go-cqhttp-docs">文档</a>
|
<a href="https://docs.go-cqhttp.org/">文档</a>
|
||||||
·
|
·
|
||||||
<a href="https://github.com/Mrs4s/go-cqhttp/releases">下载</a>
|
<a href="https://github.com/Mrs4s/go-cqhttp/releases">下载</a>
|
||||||
·
|
·
|
||||||
<a href="https://ishkong.github.io/go-cqhttp-docs/guide/quick_start.html">开始使用</a>
|
<a href="https://docs.go-cqhttp.org/guide/quick_start.html">开始使用</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
go-cqhttp 在[原版 cqhttp](https://github.com/richardchien/coolq-http-api)的基础上做了部分修改和拓展.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 兼容性
|
## 兼容性
|
||||||
|
go-cqhttp兼容[OneBot-v11](https://github.com/howmanybots/onebot/tree/master/v11/specs)绝大多数内容,并在其基础上做了一些扩展,详情请看go-cqhttp的文档
|
||||||
|
|
||||||
### 接口
|
### 接口
|
||||||
|
|
||||||
@ -276,6 +275,7 @@ go-cqhttp 在[原版 cqhttp](https://github.com/richardchien/coolq-http-api)的
|
|||||||
- 重复提问
|
- 重复提问
|
||||||
|
|
||||||
> 请注意, 开发者并没有义务回复您的问题. 您应该具备基本的提问技巧。
|
> 请注意, 开发者并没有义务回复您的问题. 您应该具备基本的提问技巧。
|
||||||
|
> 有关如何提问,请阅读[《提问的智慧》](https://github.com/ryanhanwu/How-To-Ask-Questions-The-Smart-Way/blob/main/README-zh_CN.md)
|
||||||
|
|
||||||
## 性能
|
## 性能
|
||||||
|
|
||||||
|
722
coolq/api.go
722
coolq/api.go
File diff suppressed because it is too large
Load Diff
335
coolq/bot.go
335
coolq/bot.go
@ -3,26 +3,29 @@ package coolq
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/gob"
|
"encoding/gob"
|
||||||
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
"hash/crc32"
|
"hash/crc32"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"runtime/debug"
|
"runtime/debug"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"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/MiraiGo/utils"
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
|
||||||
jsoniter "github.com/json-iterator/go"
|
jsoniter "github.com/json-iterator/go"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
)
|
)
|
||||||
|
|
||||||
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
|
// CQBot CQBot结构体,存储Bot实例相关配置
|
||||||
type CQBot struct {
|
type CQBot struct {
|
||||||
Client *client.QQClient
|
Client *client.QQClient
|
||||||
|
|
||||||
@ -33,11 +36,14 @@ type CQBot struct {
|
|||||||
oneWayMsgCache sync.Map
|
oneWayMsgCache sync.Map
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MSG 消息Map
|
||||||
type MSG map[string]interface{}
|
type MSG map[string]interface{}
|
||||||
|
|
||||||
|
// ForceFragmented 是否启用强制分片
|
||||||
var ForceFragmented = false
|
var ForceFragmented = false
|
||||||
|
|
||||||
func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
// NewQQBot 初始化一个QQBot实例
|
||||||
|
func NewQQBot(cli *client.QQClient, conf *global.JSONConfig) *CQBot {
|
||||||
bot := &CQBot{
|
bot := &CQBot{
|
||||||
Client: cli,
|
Client: cli,
|
||||||
}
|
}
|
||||||
@ -55,6 +61,9 @@ func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
|||||||
}
|
}
|
||||||
bot.Client.OnPrivateMessage(bot.privateMessageEvent)
|
bot.Client.OnPrivateMessage(bot.privateMessageEvent)
|
||||||
bot.Client.OnGroupMessage(bot.groupMessageEvent)
|
bot.Client.OnGroupMessage(bot.groupMessageEvent)
|
||||||
|
if conf.EnableSelfMessage {
|
||||||
|
bot.Client.OnSelfGroupMessage(bot.groupMessageEvent)
|
||||||
|
}
|
||||||
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)
|
||||||
@ -72,6 +81,8 @@ func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
|||||||
bot.Client.OnNewFriendAdded(bot.friendAddedEvent)
|
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)
|
||||||
|
bot.Client.OnOtherClientStatusChanged(bot.otherClientStatusChangedEvent)
|
||||||
|
bot.Client.OnGroupDigest(bot.groupEssenceMsg)
|
||||||
go func() {
|
go func() {
|
||||||
i := conf.HeartbeatInterval
|
i := conf.HeartbeatInterval
|
||||||
if i < 0 {
|
if i < 0 {
|
||||||
@ -96,10 +107,12 @@ func NewQQBot(cli *client.QQClient, conf *global.JsonConfig) *CQBot {
|
|||||||
return bot
|
return bot
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// OnEventPush 注册事件上报函数
|
||||||
func (bot *CQBot) OnEventPush(f func(m MSG)) {
|
func (bot *CQBot) OnEventPush(f func(m MSG)) {
|
||||||
bot.events = append(bot.events, f)
|
bot.events = append(bot.events, f)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetMessage 获取给定消息id对应的消息
|
||||||
func (bot *CQBot) GetMessage(mid int32) MSG {
|
func (bot *CQBot) GetMessage(mid int32) MSG {
|
||||||
if bot.db != nil {
|
if bot.db != nil {
|
||||||
m := MSG{}
|
m := MSG{}
|
||||||
@ -117,29 +130,79 @@ func (bot *CQBot) GetMessage(mid int32) MSG {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) SendGroupMessage(groupId int64, m *message.SendingMessage) int32 {
|
// UploadLocalImageAsGroup 上传本地图片至群聊
|
||||||
|
func (bot *CQBot) UploadLocalImageAsGroup(groupCode int64, img *LocalImageElement) (*message.GroupImageElement, error) {
|
||||||
|
if img.Stream != nil {
|
||||||
|
return bot.Client.UploadGroupImage(groupCode, img.Stream)
|
||||||
|
}
|
||||||
|
return bot.Client.UploadGroupImageByFile(groupCode, img.File)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UploadLocalVideo 上传本地短视频至群聊
|
||||||
|
func (bot *CQBot) UploadLocalVideo(target int64, v *LocalVideoElement) (*message.ShortVideoElement, error) {
|
||||||
|
if v.File != "" {
|
||||||
|
video, err := os.Open(v.File)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer video.Close()
|
||||||
|
hash, _ := utils.ComputeMd5AndLength(io.MultiReader(video, v.thumb))
|
||||||
|
cacheFile := path.Join(global.CachePath, hex.EncodeToString(hash[:])+".cache")
|
||||||
|
_, _ = video.Seek(0, io.SeekStart)
|
||||||
|
_, _ = v.thumb.Seek(0, io.SeekStart)
|
||||||
|
return bot.Client.UploadGroupShortVideo(target, video, v.thumb, cacheFile)
|
||||||
|
}
|
||||||
|
return &v.ShortVideoElement, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// UploadLocalImageAsPrivate 上传本地图片至私聊
|
||||||
|
func (bot *CQBot) UploadLocalImageAsPrivate(userID int64, img *LocalImageElement) (*message.FriendImageElement, error) {
|
||||||
|
if img.Stream != nil {
|
||||||
|
return bot.Client.UploadPrivateImage(userID, img.Stream)
|
||||||
|
}
|
||||||
|
// need update.
|
||||||
|
f, err := os.Open(img.File)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
return bot.Client.UploadPrivateImage(userID, f)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendGroupMessage 发送群消息
|
||||||
|
func (bot *CQBot) SendGroupMessage(groupID int64, m *message.SendingMessage) int32 {
|
||||||
var newElem []message.IMessageElement
|
var newElem []message.IMessageElement
|
||||||
|
group := bot.Client.FindGroup(groupID)
|
||||||
for _, elem := range m.Elements {
|
for _, elem := range m.Elements {
|
||||||
if i, ok := elem.(*message.ImageElement); ok {
|
if i, ok := elem.(*LocalImageElement); ok {
|
||||||
gm, err := bot.Client.UploadGroupImage(groupId, i.Data)
|
gm, err := bot.UploadLocalImageAsGroup(groupID, i)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("警告: 群 %v 消息图片上传失败: %v", groupId, err)
|
log.Warnf("警告: 群 %v 消息图片上传失败: %v", groupID, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
newElem = append(newElem, gm)
|
newElem = append(newElem, gm)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*message.VoiceElement); ok {
|
if i, ok := elem.(*message.VoiceElement); ok {
|
||||||
gv, err := bot.Client.UploadGroupPtt(groupId, i.Data)
|
gv, err := bot.Client.UploadGroupPtt(groupID, bytes.NewReader(i.Data))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("警告: 群 %v 消息语音上传失败: %v", groupId, err)
|
log.Warnf("警告: 群 %v 消息语音上传失败: %v", groupID, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
newElem = append(newElem, gv)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if i, ok := elem.(*LocalVideoElement); ok {
|
||||||
|
gv, err := bot.UploadLocalVideo(groupID, i)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("警告: 群 %v 消息短视频上传失败: %v", groupID, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
newElem = append(newElem, gv)
|
newElem = append(newElem, gv)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*PokeElement); ok {
|
if i, ok := elem.(*PokeElement); ok {
|
||||||
if group := bot.Client.FindGroup(groupId); group != nil {
|
if group := bot.Client.FindGroup(groupID); group != nil {
|
||||||
if mem := group.FindMember(i.Target); mem != nil {
|
if mem := group.FindMember(i.Target); mem != nil {
|
||||||
mem.Poke()
|
mem.Poke()
|
||||||
return 0
|
return 0
|
||||||
@ -147,69 +210,20 @@ func (bot *CQBot) SendGroupMessage(groupId int64, m *message.SendingMessage) int
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*GiftElement); ok {
|
if i, ok := elem.(*GiftElement); ok {
|
||||||
bot.Client.SendGroupGift(uint64(groupId), uint64(i.Target), i.GiftId)
|
bot.Client.SendGroupGift(uint64(groupID), uint64(i.Target), i.GiftID)
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*QQMusicElement); ok {
|
if i, ok := elem.(*message.MusicShareElement); ok {
|
||||||
var msgStyle uint32 = 4
|
ret, err := bot.Client.SendGroupMusicShare(groupID, i)
|
||||||
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 {
|
if err != nil {
|
||||||
log.Warnf("警告: 群 %v 富文本消息发送失败: %v", groupId, err)
|
log.Warnf("警告: 群 %v 富文本消息发送失败: %v", groupID, err)
|
||||||
return -1
|
return -1
|
||||||
}
|
}
|
||||||
return bot.InsertGroupMessage(ret)
|
return bot.InsertGroupMessage(ret)
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*CloudMusicElement); ok {
|
if i, ok := elem.(*message.AtElement); ok && i.Target == 0 && group.SelfPermission() == client.Member {
|
||||||
ret, err := bot.Client.SendGroupRichMessage(groupId, 100495085, 1, 4, client.RichClientInfo{
|
newElem = append(newElem, message.NewText("@全体成员"))
|
||||||
Platform: 1,
|
continue
|
||||||
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)
|
||||||
}
|
}
|
||||||
@ -219,25 +233,20 @@ func (bot *CQBot) SendGroupMessage(groupId int64, m *message.SendingMessage) int
|
|||||||
}
|
}
|
||||||
m.Elements = newElem
|
m.Elements = newElem
|
||||||
bot.checkMedia(newElem)
|
bot.checkMedia(newElem)
|
||||||
ret := bot.Client.SendGroupMessage(groupId, m, ForceFragmented)
|
ret := bot.Client.SendGroupMessage(groupID, m, ForceFragmented)
|
||||||
if ret == nil || ret.Id == -1 {
|
if ret == nil || ret.Id == -1 {
|
||||||
log.Warnf("群消息发送失败: 账号可能被风控.")
|
log.Warnf("群消息发送失败: 账号可能被风控.")
|
||||||
if !ForceFragmented {
|
return -1
|
||||||
log.Warnf("将尝试分片发送...")
|
|
||||||
ret = bot.Client.SendGroupMessage(groupId, m, true)
|
|
||||||
}
|
|
||||||
if ret == nil || ret.Id == -1 {
|
|
||||||
return -1
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return bot.InsertGroupMessage(ret)
|
return bot.InsertGroupMessage(ret)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SendPrivateMessage 发送私聊消息
|
||||||
func (bot *CQBot) SendPrivateMessage(target int64, m *message.SendingMessage) int32 {
|
func (bot *CQBot) SendPrivateMessage(target int64, m *message.SendingMessage) int32 {
|
||||||
var newElem []message.IMessageElement
|
var newElem []message.IMessageElement
|
||||||
for _, elem := range m.Elements {
|
for _, elem := range m.Elements {
|
||||||
if i, ok := elem.(*message.ImageElement); ok {
|
if i, ok := elem.(*LocalImageElement); ok {
|
||||||
fm, err := bot.Client.UploadPrivateImage(target, i.Data)
|
fm, err := bot.UploadLocalImageAsPrivate(target, i)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("警告: 私聊 %v 消息图片上传失败.", target)
|
log.Warnf("警告: 私聊 %v 消息图片上传失败.", target)
|
||||||
continue
|
continue
|
||||||
@ -252,59 +261,23 @@ func (bot *CQBot) SendPrivateMessage(target int64, m *message.SendingMessage) in
|
|||||||
if i, ok := elem.(*message.VoiceElement); ok {
|
if i, ok := elem.(*message.VoiceElement); ok {
|
||||||
fv, err := bot.Client.UploadPrivatePtt(target, i.Data)
|
fv, err := bot.Client.UploadPrivatePtt(target, i.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("警告: 好友 %v 消息语音上传失败: %v", target, err)
|
log.Warnf("警告: 私聊 %v 消息语音上传失败: %v", target, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
newElem = append(newElem, fv)
|
newElem = append(newElem, fv)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if i, ok := elem.(*QQMusicElement); ok {
|
if i, ok := elem.(*LocalVideoElement); ok {
|
||||||
var msgStyle uint32 = 4
|
gv, err := bot.UploadLocalVideo(target, i)
|
||||||
if i.MusicUrl == "" {
|
if err != nil {
|
||||||
msgStyle = 0 // fix vip song
|
log.Warnf("警告: 私聊 %v 消息短视频上传失败: %v", target, err)
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
bot.Client.SendFriendRichMessage(target, 100497308, 1, msgStyle, client.RichClientInfo{
|
newElem = append(newElem, gv)
|
||||||
Platform: 1,
|
continue
|
||||||
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 {
|
if i, ok := elem.(*message.MusicShareElement); ok {
|
||||||
bot.Client.SendFriendRichMessage(target, 100495085, 1, 4, client.RichClientInfo{
|
bot.Client.SendFriendMusicShare(target, i)
|
||||||
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
|
return 0
|
||||||
}
|
}
|
||||||
newElem = append(newElem, elem)
|
newElem = append(newElem, elem)
|
||||||
@ -324,7 +297,7 @@ func (bot *CQBot) SendPrivateMessage(target int64, m *message.SendingMessage) in
|
|||||||
} else if code, ok := bot.tempMsgCache.Load(target); ok { // 临时会话
|
} else if code, ok := bot.tempMsgCache.Load(target); ok { // 临时会话
|
||||||
msg := bot.Client.SendTempMessage(code.(int64), target, m)
|
msg := bot.Client.SendTempMessage(code.(int64), target, m)
|
||||||
if msg != nil {
|
if msg != nil {
|
||||||
id = msg.Id
|
id = bot.InsertTempMessage(target, msg)
|
||||||
}
|
}
|
||||||
} else if _, ok := bot.oneWayMsgCache.Load(target); ok { // 单向好友
|
} else if _, ok := bot.oneWayMsgCache.Load(target); ok { // 单向好友
|
||||||
msg := bot.Client.SendPrivateMessage(target, m)
|
msg := bot.Client.SendPrivateMessage(target, m)
|
||||||
@ -338,6 +311,7 @@ func (bot *CQBot) SendPrivateMessage(target int64, m *message.SendingMessage) in
|
|||||||
return id
|
return id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InsertGroupMessage 群聊消息入数据库
|
||||||
func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
||||||
val := MSG{
|
val := MSG{
|
||||||
"message-id": m.Id,
|
"message-id": m.Id,
|
||||||
@ -348,7 +322,7 @@ func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
|||||||
"time": m.Time,
|
"time": m.Time,
|
||||||
"message": ToStringMessage(m.Elements, m.GroupCode, true),
|
"message": ToStringMessage(m.Elements, m.GroupCode, true),
|
||||||
}
|
}
|
||||||
id := ToGlobalId(m.GroupCode, m.Id)
|
id := toGlobalID(m.GroupCode, m.Id)
|
||||||
if bot.db != nil {
|
if bot.db != nil {
|
||||||
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 {
|
||||||
@ -363,6 +337,7 @@ func (bot *CQBot) InsertGroupMessage(m *message.GroupMessage) int32 {
|
|||||||
return id
|
return id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InsertPrivateMessage 私聊消息入数据库
|
||||||
func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 {
|
func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 {
|
||||||
val := MSG{
|
val := MSG{
|
||||||
"message-id": m.Id,
|
"message-id": m.Id,
|
||||||
@ -372,7 +347,7 @@ func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 {
|
|||||||
"time": m.Time,
|
"time": m.Time,
|
||||||
"message": ToStringMessage(m.Elements, m.Sender.Uin, true),
|
"message": ToStringMessage(m.Elements, m.Sender.Uin, true),
|
||||||
}
|
}
|
||||||
id := ToGlobalId(m.Sender.Uin, m.Id)
|
id := toGlobalID(m.Sender.Uin, m.Id)
|
||||||
if bot.db != nil {
|
if bot.db != nil {
|
||||||
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 {
|
||||||
@ -387,10 +362,39 @@ func (bot *CQBot) InsertPrivateMessage(m *message.PrivateMessage) int32 {
|
|||||||
return id
|
return id
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToGlobalId(code int64, msgId int32) int32 {
|
// InsertTempMessage 临时消息入数据库
|
||||||
return int32(crc32.ChecksumIEEE([]byte(fmt.Sprintf("%d-%d", code, msgId))))
|
func (bot *CQBot) InsertTempMessage(target int64, m *message.TempMessage) int32 {
|
||||||
|
val := MSG{
|
||||||
|
"message-id": m.Id,
|
||||||
|
// FIXME(InsertTempMessage) InternalId missing
|
||||||
|
"group": m.GroupCode,
|
||||||
|
"group-name": m.GroupName,
|
||||||
|
"target": target,
|
||||||
|
"sender": m.Sender,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
"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)
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// toGlobalID 构建`code`-`msgID`的字符串并返回其CRC32 Checksum的值
|
||||||
|
func toGlobalID(code int64, msgID int32) int32 {
|
||||||
|
return int32(crc32.ChecksumIEEE([]byte(fmt.Sprintf("%d-%d", code, msgID))))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Release 释放Bot实例
|
||||||
func (bot *CQBot) Release() {
|
func (bot *CQBot) Release() {
|
||||||
if bot.db != nil {
|
if bot.db != nil {
|
||||||
_ = bot.db.Close()
|
_ = bot.db.Close()
|
||||||
@ -398,7 +402,7 @@ func (bot *CQBot) Release() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) dispatchEventMessage(m MSG) {
|
func (bot *CQBot) dispatchEventMessage(m MSG) {
|
||||||
if global.EventFilter != nil && global.EventFilter.Eval(global.MSG(m)) == false {
|
if global.EventFilter != nil && !global.EventFilter.Eval(global.MSG(m)) {
|
||||||
log.Debug("Event filtered!")
|
log.Debug("Event filtered!")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -419,6 +423,76 @@ func (bot *CQBot) dispatchEventMessage(m MSG) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) formatGroupMessage(m *message.GroupMessage) MSG {
|
||||||
|
cqm := ToStringMessage(m.Elements, m.GroupCode, true)
|
||||||
|
gm := MSG{
|
||||||
|
"anonymous": nil,
|
||||||
|
"font": 0,
|
||||||
|
"group_id": m.GroupCode,
|
||||||
|
"message": ToFormattedMessage(m.Elements, m.GroupCode, false),
|
||||||
|
"message_type": "group",
|
||||||
|
"message_seq": m.Id,
|
||||||
|
"post_type": func() string {
|
||||||
|
if m.Sender.Uin == bot.Client.Uin {
|
||||||
|
return "message_sent"
|
||||||
|
}
|
||||||
|
return "message"
|
||||||
|
}(),
|
||||||
|
"raw_message": cqm,
|
||||||
|
"self_id": bot.Client.Uin,
|
||||||
|
"sender": MSG{
|
||||||
|
"age": 0,
|
||||||
|
"area": "",
|
||||||
|
"level": "",
|
||||||
|
"sex": "unknown",
|
||||||
|
"user_id": m.Sender.Uin,
|
||||||
|
},
|
||||||
|
"sub_type": "normal",
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
"user_id": m.Sender.Uin,
|
||||||
|
}
|
||||||
|
if m.Sender.IsAnonymous() {
|
||||||
|
gm["anonymous"] = MSG{
|
||||||
|
"flag": m.Sender.AnonymousInfo.AnonymousId + "|" + m.Sender.AnonymousInfo.AnonymousNick,
|
||||||
|
"id": m.Sender.Uin,
|
||||||
|
"name": m.Sender.AnonymousInfo.AnonymousNick,
|
||||||
|
}
|
||||||
|
gm["sender"].(MSG)["nickname"] = "匿名消息"
|
||||||
|
gm["sub_type"] = "anonymous"
|
||||||
|
} else {
|
||||||
|
group := bot.Client.FindGroup(m.GroupCode)
|
||||||
|
mem := group.FindMember(m.Sender.Uin)
|
||||||
|
if mem == nil {
|
||||||
|
log.Warnf("获取 %v 成员信息失败,尝试刷新成员列表", m.Sender.Uin)
|
||||||
|
t, err := bot.Client.GetGroupMembers(group)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("刷新群 %v 成员列表失败: %v", group.Uin, err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
group.Members = t
|
||||||
|
mem = group.FindMember(m.Sender.Uin)
|
||||||
|
if mem == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ms := gm["sender"].(MSG)
|
||||||
|
ms["role"] = func() string {
|
||||||
|
switch mem.Permission {
|
||||||
|
case client.Owner:
|
||||||
|
return "owner"
|
||||||
|
case client.Administrator:
|
||||||
|
return "admin"
|
||||||
|
default:
|
||||||
|
return "member"
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
ms["nickname"] = mem.Nickname
|
||||||
|
ms["card"] = mem.CardName
|
||||||
|
ms["title"] = mem.SpecialTitle
|
||||||
|
}
|
||||||
|
return gm
|
||||||
|
}
|
||||||
|
|
||||||
func formatGroupName(group *client.GroupInfo) string {
|
func formatGroupName(group *client.GroupInfo) string {
|
||||||
return fmt.Sprintf("%s(%d)", group.Name, group.Code)
|
return fmt.Sprintf("%s(%d)", group.Name, group.Code)
|
||||||
}
|
}
|
||||||
@ -430,7 +504,8 @@ func formatMemberName(mem *client.GroupMemberInfo) string {
|
|||||||
return fmt.Sprintf("%s(%d)", mem.DisplayName(), mem.Uin)
|
return fmt.Sprintf("%s(%d)", mem.DisplayName(), mem.Uin)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m MSG) ToJson() string {
|
// ToJSON 生成JSON字符串
|
||||||
|
func (m MSG) ToJSON() string {
|
||||||
b, _ := json.Marshal(m)
|
b, _ := json.Marshal(m)
|
||||||
return string(b)
|
return string(b)
|
||||||
}
|
}
|
||||||
|
696
coolq/cqcode.go
696
coolq/cqcode.go
File diff suppressed because it is too large
Load Diff
25
coolq/cqcode_test.go
Normal file
25
coolq/cqcode_test.go
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
package coolq
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
)
|
||||||
|
|
||||||
|
var bot = NewQQBot(client.NewClient(1, ""), global.DefaultConfig())
|
||||||
|
|
||||||
|
func TestCQBot_ConvertStringMessage(t *testing.T) {
|
||||||
|
for _, v := range bot.ConvertStringMessage(`[CQ:face,id=115,text=111][CQ:face,id=217]] [CQ:text,text=123] [`, false) {
|
||||||
|
fmt.Println(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var bench = `asdfqwerqwerqwer[CQ:face,id=115,text=111]asdfasdfasdfasdfasdfasdfasd[CQ:face,id=217]] [CQ:text,text=123] [`
|
||||||
|
|
||||||
|
func BenchmarkCQBot_ConvertStringMessage(b *testing.B) {
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
bot.ConvertStringMessage(bench, false)
|
||||||
|
}
|
||||||
|
}
|
2
coolq/doc.go
Normal file
2
coolq/doc.go
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
// Package coolq 包含CQBot实例,CQ码处理,消息发送,消息处理等的相关函数与结构体
|
||||||
|
package coolq
|
162
coolq/event.go
162
coolq/event.go
@ -17,15 +17,17 @@ import (
|
|||||||
|
|
||||||
var format = "string"
|
var format = "string"
|
||||||
|
|
||||||
|
// SetMessageFormat 设置消息上报格式,默认为string
|
||||||
func SetMessageFormat(f string) {
|
func SetMessageFormat(f string) {
|
||||||
format = f
|
format = f
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToFormattedMessage(e []message.IMessageElement, code int64, raw ...bool) (r interface{}) {
|
// ToFormattedMessage 将给定[]message.IMessageElement转换为通过coolq.SetMessageFormat所定义的消息上报格式
|
||||||
|
func ToFormattedMessage(e []message.IMessageElement, id int64, isRaw ...bool) (r interface{}) {
|
||||||
if format == "string" {
|
if format == "string" {
|
||||||
r = ToStringMessage(e, code, raw...)
|
r = ToStringMessage(e, id, isRaw...)
|
||||||
} else if format == "array" {
|
} else if format == "array" {
|
||||||
r = ToArrayMessage(e, code, raw...)
|
r = ToArrayMessage(e, id, isRaw...)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -91,67 +93,30 @@ func (bot *CQBot) groupMessageEvent(c *client.QQClient, m *message.GroupMessage)
|
|||||||
id = bot.InsertGroupMessage(m)
|
id = bot.InsertGroupMessage(m)
|
||||||
}
|
}
|
||||||
log.Infof("收到群 %v(%v) 内 %v(%v) 的消息: %v (%v)", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, cqm, id)
|
log.Infof("收到群 %v(%v) 内 %v(%v) 的消息: %v (%v)", m.GroupName, m.GroupCode, m.Sender.DisplayName(), m.Sender.Uin, cqm, id)
|
||||||
gm := MSG{
|
gm := bot.formatGroupMessage(m)
|
||||||
"anonymous": nil,
|
if gm == nil {
|
||||||
"font": 0,
|
return
|
||||||
"group_id": m.GroupCode,
|
|
||||||
"message": ToFormattedMessage(m.Elements, m.GroupCode, false),
|
|
||||||
"message_id": id,
|
|
||||||
"message_type": "group",
|
|
||||||
"post_type": "message",
|
|
||||||
"raw_message": cqm,
|
|
||||||
"self_id": c.Uin,
|
|
||||||
"sender": MSG{
|
|
||||||
"age": 0,
|
|
||||||
"area": "",
|
|
||||||
"level": "",
|
|
||||||
"sex": "unknown",
|
|
||||||
"user_id": m.Sender.Uin,
|
|
||||||
},
|
|
||||||
"sub_type": "normal",
|
|
||||||
"time": time.Now().Unix(),
|
|
||||||
"user_id": m.Sender.Uin,
|
|
||||||
}
|
|
||||||
if m.Sender.IsAnonymous() {
|
|
||||||
gm["anonymous"] = MSG{
|
|
||||||
"flag": m.Sender.AnonymousInfo.AnonymousId + "|" + m.Sender.AnonymousInfo.AnonymousNick,
|
|
||||||
"id": m.Sender.Uin,
|
|
||||||
"name": m.Sender.AnonymousInfo.AnonymousNick,
|
|
||||||
}
|
|
||||||
gm["sender"].(MSG)["nickname"] = "匿名消息"
|
|
||||||
gm["sub_type"] = "anonymous"
|
|
||||||
} else {
|
|
||||||
mem := c.FindGroup(m.GroupCode).FindMember(m.Sender.Uin)
|
|
||||||
ms := gm["sender"].(MSG)
|
|
||||||
ms["role"] = func() string {
|
|
||||||
switch mem.Permission {
|
|
||||||
case client.Owner:
|
|
||||||
return "owner"
|
|
||||||
case client.Administrator:
|
|
||||||
return "admin"
|
|
||||||
default:
|
|
||||||
return "member"
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
ms["nickname"] = mem.Nickname
|
|
||||||
ms["card"] = mem.CardName
|
|
||||||
ms["title"] = mem.SpecialTitle
|
|
||||||
}
|
}
|
||||||
|
gm["message_id"] = id
|
||||||
bot.dispatchEventMessage(gm)
|
bot.dispatchEventMessage(gm)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bot *CQBot) tempMessageEvent(c *client.QQClient, m *message.TempMessage) {
|
func (bot *CQBot) tempMessageEvent(c *client.QQClient, m *message.TempMessage) {
|
||||||
bot.checkMedia(m.Elements)
|
bot.checkMedia(m.Elements)
|
||||||
cqm := ToStringMessage(m.Elements, 0, true)
|
cqm := ToStringMessage(m.Elements, m.Sender.Uin, true)
|
||||||
bot.tempMsgCache.Store(m.Sender.Uin, m.GroupCode)
|
bot.tempMsgCache.Store(m.Sender.Uin, m.GroupCode)
|
||||||
|
id := m.Id
|
||||||
|
if bot.db != nil {
|
||||||
|
id = bot.InsertTempMessage(m.Sender.Uin, m)
|
||||||
|
}
|
||||||
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",
|
||||||
"message_type": "private",
|
"message_type": "private",
|
||||||
"sub_type": "group",
|
"sub_type": "group",
|
||||||
"message_id": m.Id,
|
"message_id": id,
|
||||||
"user_id": m.Sender.Uin,
|
"user_id": m.Sender.Uin,
|
||||||
"message": ToFormattedMessage(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,
|
||||||
@ -206,7 +171,7 @@ func (bot *CQBot) groupMutedEvent(c *client.QQClient, e *client.GroupMuteEvent)
|
|||||||
|
|
||||||
func (bot *CQBot) groupRecallEvent(c *client.QQClient, e *client.GroupMessageRecalledEvent) {
|
func (bot *CQBot) groupRecallEvent(c *client.QQClient, e *client.GroupMessageRecalledEvent) {
|
||||||
g := c.FindGroup(e.GroupCode)
|
g := c.FindGroup(e.GroupCode)
|
||||||
gid := ToGlobalId(e.GroupCode, e.MessageId)
|
gid := toGlobalID(e.GroupCode, e.MessageId)
|
||||||
log.Infof("群 %v 内 %v 撤回了 %v 的消息: %v.",
|
log.Infof("群 %v 内 %v 撤回了 %v 的消息: %v.",
|
||||||
formatGroupName(g), formatMemberName(g.FindMember(e.OperatorUin)), formatMemberName(g.FindMember(e.AuthorUin)), gid)
|
formatGroupName(g), formatMemberName(g.FindMember(e.OperatorUin)), formatMemberName(g.FindMember(e.AuthorUin)), gid)
|
||||||
bot.dispatchEventMessage(MSG{
|
bot.dispatchEventMessage(MSG{
|
||||||
@ -300,7 +265,7 @@ func (bot *CQBot) friendNotifyEvent(c *client.QQClient, e client.INotifyEvent) {
|
|||||||
|
|
||||||
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)
|
||||||
if f != nil {
|
if f != nil {
|
||||||
log.Infof("好友 %v(%v) 撤回了消息: %v", f.Nickname, f.Uin, gid)
|
log.Infof("好友 %v(%v) 撤回了消息: %v", f.Nickname, f.Uin, gid)
|
||||||
} else {
|
} else {
|
||||||
@ -455,6 +420,67 @@ func (bot *CQBot) groupJoinReqEvent(c *client.QQClient, e *client.UserJoinGroupR
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) otherClientStatusChangedEvent(c *client.QQClient, e *client.OtherClientStatusChangedEvent) {
|
||||||
|
if e.Online {
|
||||||
|
log.Infof("Bot 账号在客户端 %v (%v) 登录.", e.Client.DeviceName, e.Client.DeviceKind)
|
||||||
|
} else {
|
||||||
|
log.Infof("Bot 账号在客户端 %v (%v) 登出.", e.Client.DeviceName, e.Client.DeviceKind)
|
||||||
|
}
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"notice_type": "client_status",
|
||||||
|
"online": e.Online,
|
||||||
|
"client": MSG{
|
||||||
|
"app_id": e.Client.AppId,
|
||||||
|
"device_name": e.Client.DeviceName,
|
||||||
|
"device_kind": e.Client.DeviceKind,
|
||||||
|
},
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bot *CQBot) groupEssenceMsg(c *client.QQClient, e *client.GroupDigestEvent) {
|
||||||
|
g := c.FindGroup(e.GroupCode)
|
||||||
|
gid := toGlobalID(e.GroupCode, e.MessageID)
|
||||||
|
if e.OperationType == 1 {
|
||||||
|
log.Infof(
|
||||||
|
"群 %v 内 %v 将 %v 的消息(%v)设为了精华消息.",
|
||||||
|
formatGroupName(g),
|
||||||
|
formatMemberName(g.FindMember(e.OperatorUin)),
|
||||||
|
formatMemberName(g.FindMember(e.SenderUin)),
|
||||||
|
gid,
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
log.Infof(
|
||||||
|
"群 %v 内 %v 将 %v 的消息(%v)移出了精华消息.",
|
||||||
|
formatGroupName(g),
|
||||||
|
formatMemberName(g.FindMember(e.OperatorUin)),
|
||||||
|
formatMemberName(g.FindMember(e.SenderUin)),
|
||||||
|
gid,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
if e.OperatorUin == bot.Client.Uin {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bot.dispatchEventMessage(MSG{
|
||||||
|
"post_type": "notice",
|
||||||
|
"group_id": e.GroupCode,
|
||||||
|
"notice_type": "essence",
|
||||||
|
"sub_type": func() string {
|
||||||
|
if e.OperationType == 1 {
|
||||||
|
return "add"
|
||||||
|
}
|
||||||
|
return "delete"
|
||||||
|
}(),
|
||||||
|
"self_id": c.Uin,
|
||||||
|
"sender_id": e.SenderUin,
|
||||||
|
"operator_id": e.OperatorUin,
|
||||||
|
"time": time.Now().Unix(),
|
||||||
|
"message_id": gid,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func (bot *CQBot) groupIncrease(groupCode, operatorUin, userUin int64) MSG {
|
func (bot *CQBot) groupIncrease(groupCode, operatorUin, userUin int64) MSG {
|
||||||
return MSG{
|
return MSG{
|
||||||
"post_type": "notice",
|
"post_type": "notice",
|
||||||
@ -499,8 +525,8 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
|||||||
switch i := elem.(type) {
|
switch i := elem.(type) {
|
||||||
case *message.ImageElement:
|
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.ImagePath, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
w.Write(i.Md5)
|
w.Write(i.Md5)
|
||||||
w.WriteUInt32(uint32(i.Size))
|
w.WriteUInt32(uint32(i.Size))
|
||||||
w.WriteString(i.Filename)
|
w.WriteString(i.Filename)
|
||||||
@ -510,8 +536,8 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
|||||||
i.Filename = filename
|
i.Filename = filename
|
||||||
case *message.GroupImageElement:
|
case *message.GroupImageElement:
|
||||||
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.ImagePath, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
w.Write(i.Md5)
|
w.Write(i.Md5)
|
||||||
w.WriteUInt32(uint32(i.Size))
|
w.WriteUInt32(uint32(i.Size))
|
||||||
w.WriteString(filename)
|
w.WriteString(filename)
|
||||||
@ -520,8 +546,8 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
|||||||
}
|
}
|
||||||
case *message.FriendImageElement:
|
case *message.FriendImageElement:
|
||||||
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.ImagePath, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
w.Write(i.Md5)
|
w.Write(i.Md5)
|
||||||
w.WriteUInt32(uint32(0)) // 发送时会调用url, 大概没事
|
w.WriteUInt32(uint32(0)) // 发送时会调用url, 大概没事
|
||||||
w.WriteString(filename)
|
w.WriteString(filename)
|
||||||
@ -530,8 +556,8 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
|||||||
}
|
}
|
||||||
case *message.GroupFlashImgElement:
|
case *message.GroupFlashImgElement:
|
||||||
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.ImagePath, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
w.Write(i.Md5)
|
w.Write(i.Md5)
|
||||||
w.WriteUInt32(uint32(i.Size))
|
w.WriteUInt32(uint32(i.Size))
|
||||||
w.WriteString(i.Filename)
|
w.WriteString(i.Filename)
|
||||||
@ -541,8 +567,8 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
|||||||
i.Filename = filename
|
i.Filename = filename
|
||||||
case *message.FriendFlashImgElement:
|
case *message.FriendFlashImgElement:
|
||||||
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.ImagePath, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.IMAGE_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.ImagePath, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
w.Write(i.Md5)
|
w.Write(i.Md5)
|
||||||
w.WriteUInt32(uint32(i.Size))
|
w.WriteUInt32(uint32(i.Size))
|
||||||
w.WriteString(i.Filename)
|
w.WriteString(i.Filename)
|
||||||
@ -553,20 +579,22 @@ func (bot *CQBot) checkMedia(e []message.IMessageElement) {
|
|||||||
case *message.VoiceElement:
|
case *message.VoiceElement:
|
||||||
i.Name = strings.ReplaceAll(i.Name, "{", "")
|
i.Name = strings.ReplaceAll(i.Name, "{", "")
|
||||||
i.Name = strings.ReplaceAll(i.Name, "}", "")
|
i.Name = strings.ReplaceAll(i.Name, "}", "")
|
||||||
if !global.PathExists(path.Join(global.VOICE_PATH, i.Name)) {
|
if !global.PathExists(path.Join(global.VoicePath, i.Name)) {
|
||||||
b, err := global.GetBytes(i.Url)
|
b, err := global.GetBytes(i.Url)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("语音文件 %v 下载失败: %v", i.Name, err)
|
log.Warnf("语音文件 %v 下载失败: %v", i.Name, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
_ = ioutil.WriteFile(path.Join(global.VOICE_PATH, i.Name), b, 0644)
|
_ = ioutil.WriteFile(path.Join(global.VoicePath, i.Name), b, 0644)
|
||||||
}
|
}
|
||||||
case *message.ShortVideoElement:
|
case *message.ShortVideoElement:
|
||||||
filename := hex.EncodeToString(i.Md5) + ".video"
|
filename := hex.EncodeToString(i.Md5) + ".video"
|
||||||
if !global.PathExists(path.Join(global.VIDEO_PATH, filename)) {
|
if !global.PathExists(path.Join(global.VideoPath, filename)) {
|
||||||
_ = ioutil.WriteFile(path.Join(global.VIDEO_PATH, filename), binary.NewWriterF(func(w *binary.Writer) {
|
_ = ioutil.WriteFile(path.Join(global.VideoPath, filename), binary.NewWriterF(func(w *binary.Writer) {
|
||||||
w.Write(i.Md5)
|
w.Write(i.Md5)
|
||||||
|
w.Write(i.ThumbMd5)
|
||||||
w.WriteUInt32(uint32(i.Size))
|
w.WriteUInt32(uint32(i.Size))
|
||||||
|
w.WriteUInt32(uint32(i.ThumbSize))
|
||||||
w.WriteString(i.Name)
|
w.WriteString(i.Name)
|
||||||
w.Write(i.Uuid)
|
w.Write(i.Uuid)
|
||||||
}), 0644)
|
}), 0644)
|
||||||
|
@ -16,48 +16,51 @@ go-cqhttp 支持导入CQHTTP的配置文件, 具体步骤为:
|
|||||||
|
|
||||||
````json
|
````json
|
||||||
{
|
{
|
||||||
"uin": 0,
|
"uin": 0,
|
||||||
"password": "",
|
"password": "",
|
||||||
"encrypt_password": false,
|
"encrypt_password": false,
|
||||||
"password_encrypted": "",
|
"password_encrypted": "",
|
||||||
"enable_db": true,
|
"enable_db": true,
|
||||||
"access_token": "",
|
"enable_self_message": false,
|
||||||
"relogin": {
|
"access_token": "",
|
||||||
"enabled": true,
|
"relogin": {
|
||||||
"relogin_delay": 3,
|
"enabled": true,
|
||||||
"max_relogin_times": 0
|
"relogin_delay": 3,
|
||||||
},
|
"max_relogin_times": 0
|
||||||
"_rate_limit": {
|
},
|
||||||
"enabled": false,
|
"_rate_limit": {
|
||||||
"frequency": 1,
|
"enabled": false,
|
||||||
"bucket_size": 1
|
"frequency": 1,
|
||||||
},
|
"bucket_size": 1
|
||||||
"post_message_format": "string",
|
},
|
||||||
"ignore_invalid_cqcode": false,
|
"post_message_format": "string",
|
||||||
"force_fragmented": true,
|
"ignore_invalid_cqcode": false,
|
||||||
"heartbeat_interval": 5,
|
"force_fragmented": true,
|
||||||
"use_sso_address": false,
|
"heartbeat_interval": 5,
|
||||||
"http_config": {
|
"use_sso_address": false,
|
||||||
"enabled": true,
|
"http_config": {
|
||||||
"host": "0.0.0.0",
|
"enabled": true,
|
||||||
"port": 5700,
|
"host": "0.0.0.0",
|
||||||
"timeout": 5,
|
"port": 5700,
|
||||||
"post_urls": {"url:port": "secret"}
|
"timeout": 5,
|
||||||
},
|
"post_urls": {
|
||||||
"ws_config": {
|
"url:port": "secret"
|
||||||
"enabled": true,
|
}
|
||||||
"host": "0.0.0.0",
|
},
|
||||||
"port": 6700
|
"ws_config": {
|
||||||
},
|
"enabled": true,
|
||||||
"ws_reverse_servers": [
|
"host": "0.0.0.0",
|
||||||
{
|
"port": 6700
|
||||||
"enabled": false,
|
},
|
||||||
"reverse_url": "ws://you_websocket_universal.server",
|
"ws_reverse_servers": [
|
||||||
"reverse_api_url": "ws://you_websocket_api.server",
|
{
|
||||||
"reverse_event_url": "ws://you_websocket_event.server",
|
"enabled": false,
|
||||||
"reverse_reconnect_interval": 3000
|
"reverse_url": "ws://you_websocket_universal.server",
|
||||||
}
|
"reverse_api_url": "ws://you_websocket_api.server",
|
||||||
]
|
"reverse_event_url": "ws://you_websocket_event.server",
|
||||||
|
"reverse_reconnect_interval": 3000
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
````
|
````
|
||||||
|
|
||||||
@ -68,6 +71,7 @@ go-cqhttp 支持导入CQHTTP的配置文件, 具体步骤为:
|
|||||||
| encrypt_password | bool | 是否对密码进行加密. |
|
| encrypt_password | bool | 是否对密码进行加密. |
|
||||||
| password_encrypted | string | 加密后的密码(请勿修改) |
|
| password_encrypted | string | 加密后的密码(请勿修改) |
|
||||||
| enable_db | bool | 是否开启内置数据库, 关闭后将无法使用 **回复/撤回** 等上下文相关接口 |
|
| enable_db | bool | 是否开启内置数据库, 关闭后将无法使用 **回复/撤回** 等上下文相关接口 |
|
||||||
|
| enable_self_message | bool | 是否启用 `message_sent` 事件 |
|
||||||
| access_token | string | 同CQHTTP的 `access_token` 用于身份验证 |
|
| access_token | string | 同CQHTTP的 `access_token` 用于身份验证 |
|
||||||
| relogin | bool | 是否自动重新登录 |
|
| relogin | bool | 是否自动重新登录 |
|
||||||
| relogin_delay | int | 重登录延时(秒) |
|
| relogin_delay | int | 重登录延时(秒) |
|
||||||
@ -127,6 +131,7 @@ go-cqhttp 支持导入CQHTTP的配置文件, 具体步骤为:
|
|||||||
将文件 `address.txt` 创建到 `go-cqhttp` 工作目录, 并键入 `IP:PORT` 以换行符为分割即可.
|
将文件 `address.txt` 创建到 `go-cqhttp` 工作目录, 并键入 `IP:PORT` 以换行符为分割即可.
|
||||||
|
|
||||||
示例:
|
示例:
|
||||||
|
|
||||||
````
|
````
|
||||||
1.1.1.1:53
|
1.1.1.1:53
|
||||||
1.1.2.2:8899
|
1.1.2.2:8899
|
||||||
|
416
docs/cqhttp.md
416
docs/cqhttp.md
@ -26,9 +26,9 @@
|
|||||||
- [获取图片信息](#获取图片信息)
|
- [获取图片信息](#获取图片信息)
|
||||||
- [获取消息](#获取消息)
|
- [获取消息](#获取消息)
|
||||||
- [获取合并转发内容](#获取合并转发内容)
|
- [获取合并转发内容](#获取合并转发内容)
|
||||||
- [发送合并转发(群)](#发送合并转发(群))
|
- [发送合并转发(群)](#发送合并转发群)
|
||||||
- [获取中文分词](#获取中文分词)
|
- [获取中文分词](#获取中文分词)
|
||||||
- [图片OCR](#图片OCR)
|
- [图片OCR](#图片ocr)
|
||||||
- [获取中文分词](#获取中文分词)
|
- [获取中文分词](#获取中文分词)
|
||||||
- [获取群系统消息](#获取群文件系统信息)
|
- [获取群系统消息](#获取群文件系统信息)
|
||||||
- [获取群文件系统信息](#获取群文件系统信息)
|
- [获取群文件系统信息](#获取群文件系统信息)
|
||||||
@ -36,7 +36,16 @@
|
|||||||
- [获取群子目录文件列表](#获取群子目录文件列表)
|
- [获取群子目录文件列表](#获取群子目录文件列表)
|
||||||
- [获取群文件资源链接](#获取群文件资源链接)
|
- [获取群文件资源链接](#获取群文件资源链接)
|
||||||
- [获取状态](#获取状态)
|
- [获取状态](#获取状态)
|
||||||
- [获取群子目录文件列表](#设置群名)
|
- [获取群@全体成员剩余次数](#获取群全体成员剩余次数)
|
||||||
|
- [下载文件到缓存目录](#下载文件到缓存目录)
|
||||||
|
- [获取群消息历史记录](#获取群消息历史记录)
|
||||||
|
- [设置群名](#设置群名)
|
||||||
|
- [获取用户VIP信息](#获取用户vip信息)
|
||||||
|
- [发送群公告](#发送群公告)
|
||||||
|
- [设置精华消息](#设置精华消息)
|
||||||
|
- [移出精华消息](#移出精华消息)
|
||||||
|
- [获取精华消息列表](#获取精华消息列表)
|
||||||
|
- [重载事件过滤器](#重载事件过滤器)
|
||||||
|
|
||||||
##### 事件
|
##### 事件
|
||||||
- [群消息撤回](#群消息撤回)
|
- [群消息撤回](#群消息撤回)
|
||||||
@ -47,6 +56,7 @@
|
|||||||
- [群成员荣誉变更提示](#群成员荣誉变更提示)
|
- [群成员荣誉变更提示](#群成员荣誉变更提示)
|
||||||
- [群成员名片更新](#群成员名片更新)
|
- [群成员名片更新](#群成员名片更新)
|
||||||
- [接收到离线文件](#接收到离线文件)
|
- [接收到离线文件](#接收到离线文件)
|
||||||
|
- [群精华消息](#精华消息)
|
||||||
|
|
||||||
</p>
|
</p>
|
||||||
</details>
|
</details>
|
||||||
@ -61,13 +71,14 @@ Type : `image`
|
|||||||
|
|
||||||
参数:
|
参数:
|
||||||
|
|
||||||
| 参数名 | 可能的值 | 说明 |
|
| 参数名 | 可能的值 | 说明 |
|
||||||
| ------- | --------------- | --------------------------------------------------------------- |
|
| ------- | --------------- | ---------------------------------------------------------------------- |
|
||||||
| `file` | - | 图片文件名 |
|
| `file` | - | 图片文件名 |
|
||||||
| `type` | `flash`,`show` | 图片类型,`flash` 表示闪照,`show` 表示秀图,默认普通图片 |
|
| `type` | `flash`,`show` | 图片类型,`flash` 表示闪照,`show` 表示秀图,默认普通图片 |
|
||||||
| `url` | - | 图片 URL |
|
| `url` | - | 图片 URL |
|
||||||
| `cache` | `0` `1` | 只在通过网络 URL 发送时有效,表示是否使用已缓存的文件,默认 `1` |
|
| `cache` | `0` `1` | 只在通过网络 URL 发送时有效,表示是否使用已缓存的文件,默认 `1` |
|
||||||
| `id` | - | 发送秀图时的特效id,默认为40000 |
|
| `id` | - | 发送秀图时的特效id,默认为40000 |
|
||||||
|
| `c` | `2` `3` | 通过网络下载图片时的线程数, 默认单线程. (在资源不支持并发时会自动处理) |
|
||||||
|
|
||||||
可用的特效ID:
|
可用的特效ID:
|
||||||
|
|
||||||
@ -90,13 +101,72 @@ Type : `reply`
|
|||||||
|
|
||||||
范围: **发送/接收**
|
范围: **发送/接收**
|
||||||
|
|
||||||
|
> 注意: 如果id存在则优先处理id
|
||||||
|
|
||||||
参数:
|
参数:
|
||||||
|
|
||||||
| 参数名 | 类型 | 说明 |
|
| 参数名 | 类型 | 说明 |
|
||||||
| ------ | ---- | ------------------------------------- |
|
| ------ | ------ | --------------------------------------------------- |
|
||||||
| `id` | int | 回复时所引用的消息id, 必须为本群消息. |
|
| `id` | int | 回复时所引用的消息id, 必须为本群消息. |
|
||||||
|
| `text` | string | 自定义回复的信息 |
|
||||||
|
| `qq` | int64 | 自定义回复时的自定义QQ, 如果使用自定义信息必须指定. |
|
||||||
|
| `time` | int64 | 可选. 自定义回复时的时间, 格式为Unix时间 |
|
||||||
|
| `seq` | int64 | 起始消息序号, 可通过 `get_msg` 获得 |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
示例: `[CQ:reply,id=123456]`
|
示例: `[CQ:reply,id=123456]`
|
||||||
|
\
|
||||||
|
自定义回复示例: `[CQ:reply,text=Hello World,qq=10086,time=3376656000,seq=5123]`
|
||||||
|
|
||||||
|
### 音乐分享 <Badge text="发"/>
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "music",
|
||||||
|
"data": {
|
||||||
|
"type": "163",
|
||||||
|
"id": "28949129"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
[CQ:music,type=163,id=28949129]
|
||||||
|
```
|
||||||
|
|
||||||
|
| 参数名 | 收 | 发 | 可能的值 | 说明 |
|
||||||
|
| ------ | --- | --- | ---------- | -------------------------------- |
|
||||||
|
| `type` | | ✓ | `qq` `163` | 分别表示使用 QQ 音乐、网易云音乐 |
|
||||||
|
| `id` | | ✓ | - | 歌曲 ID |
|
||||||
|
|
||||||
|
### 音乐自定义分享 <Badge text="发"/>
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "music",
|
||||||
|
"data": {
|
||||||
|
"type": "custom",
|
||||||
|
"url": "http://baidu.com",
|
||||||
|
"audio": "http://baidu.com/1.mp3",
|
||||||
|
"title": "音乐标题"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
[CQ:music,type=custom,url=http://baidu.com,audio=http://baidu.com/1.mp3,title=音乐标题]
|
||||||
|
```
|
||||||
|
|
||||||
|
| 参数名 | 收 | 发 | 可能的值 | 说明 |
|
||||||
|
| --------- | --- | --- | ------------------------ | ----------------------------------------------------- |
|
||||||
|
| `type` | | ✓ | `custom` | 表示音乐自定义分享 |
|
||||||
|
| `subtype` | | ✓ | `qq,163,migu,kugou,kuwo` | 表示分享类型,不填写发送为xml卡片,推荐填写提高稳定性 |
|
||||||
|
| `url` | | ✓ | - | 点击后跳转目标 URL |
|
||||||
|
| `audio` | | ✓ | - | 音乐 URL |
|
||||||
|
| `title` | | ✓ | - | 标题 |
|
||||||
|
| `content` | | ✓ | - | 内容描述 |
|
||||||
|
| `image` | | ✓ | - | 图片 URL |
|
||||||
|
|
||||||
### 红包
|
### 红包
|
||||||
|
|
||||||
@ -174,8 +244,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]`
|
||||||
@ -188,14 +258,15 @@ Type: `node`
|
|||||||
|
|
||||||
参数:
|
参数:
|
||||||
|
|
||||||
| 参数名 | 类型 | 说明 | 特殊说明 |
|
| 参数名 | 类型 | 说明 | 特殊说明 |
|
||||||
| --------- | ------- | -------------- | ------------------------------------------------------------ |
|
| --------- | ------- | -------------- | -------------------------------------------------------------------------------------- |
|
||||||
| `id` | int32 | 转发消息id | 直接引用他人的消息合并转发, 实际查看顺序为原消息发送顺序 **与下面的自定义消息二选一** |
|
| `id` | int32 | 转发消息id | 直接引用他人的消息合并转发, 实际查看顺序为原消息发送顺序 **与下面的自定义消息二选一** |
|
||||||
| `name` | string | 发送者显示名字 | 用于自定义消息 (自定义消息并合并转发,实际查看顺序为自定义消息段顺序) |
|
| `name` | string | 发送者显示名字 | 用于自定义消息 (自定义消息并合并转发,实际查看顺序为自定义消息段顺序) |
|
||||||
| `uin` | int64 | 发送者QQ号 | 用于自定义消息 |
|
| `uin` | int64 | 发送者QQ号 | 用于自定义消息 |
|
||||||
| `content` | message | 具体消息 | 用于自定义消息 **不支持引用回复** |
|
| `content` | message | 具体消息 | 用于自定义消息 |
|
||||||
|
| `seq` | 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://git.io/JtxtN) 文档说明, `data` 应全为字符串, 但由于需要接收`message` 类型的消息, 所以 *仅限此Type的content字段* 支持Array套娃**
|
||||||
|
|
||||||
示例:
|
示例:
|
||||||
|
|
||||||
@ -255,7 +326,9 @@ Type: `node`
|
|||||||
"data": {
|
"data": {
|
||||||
"name": "自定义发送者",
|
"name": "自定义发送者",
|
||||||
"uin": "10086",
|
"uin": "10086",
|
||||||
"content": "我是自定义消息"
|
"content": "我是自定义消息",
|
||||||
|
"seq": "5123",
|
||||||
|
"time": "3376656000"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -266,6 +339,20 @@ Type: `node`
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
````
|
````
|
||||||
|
### 短视频消息
|
||||||
|
|
||||||
|
Type: `video`
|
||||||
|
|
||||||
|
范围: **发送/接收**
|
||||||
|
|
||||||
|
参数:
|
||||||
|
|
||||||
|
| 参数名 | 类型 | 说明 |
|
||||||
|
| ------- | ------- | ---------------------------------------------------------------------- |
|
||||||
|
| `file` | string | 支持http和file发送 |
|
||||||
|
| `cover` | string | 视频封面,支持http,file和base64发送,格式必须为jpg |
|
||||||
|
| `c` | `2` `3` | 通过网络下载视频时的线程数, 默认单线程. (在资源不支持并发时会自动处理) |
|
||||||
|
示例: `[CQ:image,file=file:///C:\\Users\Richard\Pictures\1.mp4]`
|
||||||
|
|
||||||
### XML 消息
|
### XML 消息
|
||||||
|
|
||||||
@ -483,7 +570,7 @@ Type: `tts`
|
|||||||
|
|
||||||
响应示例
|
响应示例
|
||||||
|
|
||||||
````json
|
````json5
|
||||||
{
|
{
|
||||||
"data": {
|
"data": {
|
||||||
"messages": [
|
"messages": [
|
||||||
@ -501,7 +588,7 @@ Type: `tts`
|
|||||||
"nickname": "发送者B",
|
"nickname": "发送者B",
|
||||||
"user_id": 10087
|
"user_id": 10087
|
||||||
},
|
},
|
||||||
"time": 1595694393
|
"time": 1595694393 // 可选
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@ -519,7 +606,13 @@ Type: `tts`
|
|||||||
| 字段 | 类型 | 说明 |
|
| 字段 | 类型 | 说明 |
|
||||||
| ---------- | -------------- | ---------------------------- |
|
| ---------- | -------------- | ---------------------------- |
|
||||||
| `group_id` | int64 | 群号 |
|
| `group_id` | int64 | 群号 |
|
||||||
| `messages` | forward node[] | 自定义转发消息, 具体看CQCode |
|
| `messages` | forward node[] | 自定义转发消息, 具体看 [CQCode](https://github.com/Mrs4s/go-cqhttp/blob/master/docs/cqhttp.md#%E5%90%88%E5%B9%B6%E8%BD%AC%E5%8F%91%E6%B6%88%E6%81%AF%E8%8A%82%E7%82%B9) |
|
||||||
|
|
||||||
|
响应数据
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------ | ------ | ------ |
|
||||||
|
| `message_id` | string | 消息id |
|
||||||
|
|
||||||
### 获取中文分词
|
### 获取中文分词
|
||||||
|
|
||||||
@ -537,11 +630,63 @@ Type: `tts`
|
|||||||
| -------- | -------- | ---- |
|
| -------- | -------- | ---- |
|
||||||
| `slices` | string[] | 词组 |
|
| `slices` | string[] | 词组 |
|
||||||
|
|
||||||
|
### 设置精华消息
|
||||||
|
|
||||||
|
终结点: `/set_essence_msg`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------ | ----- | ------ |
|
||||||
|
| `message_id` | int32 | 消息ID |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
无
|
||||||
|
|
||||||
|
### 移出精华消息
|
||||||
|
|
||||||
|
终结点: `/delete_essence_msg`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------ | ----- | ------ |
|
||||||
|
| `message_id` | int32 | 消息ID |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
无
|
||||||
|
|
||||||
|
### 获取精华消息列表
|
||||||
|
|
||||||
|
终结点: `/get_essence_msg_list`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ----- | ---- |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
响应内容为 JSON 数组,每个元素如下:
|
||||||
|
|
||||||
|
| 字段名 | 数据类型 | 说明 |
|
||||||
|
| --------------- | -------- | ------------ |
|
||||||
|
| `sender_id` | int64 | 发送者QQ 号 |
|
||||||
|
| `sender_nick` | string | 发送者昵称 |
|
||||||
|
| `sender_time` | int64 | 消息发送时间 |
|
||||||
|
| `operator_id` | int64 | 操作者QQ 号 |
|
||||||
|
| `operator_nick` | string | 操作者昵称 |
|
||||||
|
| `operator_time` | int64 | 精华设置时间 |
|
||||||
|
| `message_id` | int32 | 消息ID |
|
||||||
|
|
||||||
### 图片OCR
|
### 图片OCR
|
||||||
|
|
||||||
> 注意: 目前图片OCR接口仅支持接受的图片
|
> 注意: 目前图片OCR接口仅支持接受的图片
|
||||||
|
|
||||||
终结点: `/.ocr_image`
|
终结点: `/ocr_image`
|
||||||
|
|
||||||
**参数**
|
**参数**
|
||||||
|
|
||||||
@ -707,6 +852,22 @@ Type: `tts`
|
|||||||
| `creator_name` | string | 创建者名字 |
|
| `creator_name` | string | 创建者名字 |
|
||||||
| `total_file_count` | int32 | 子文件数量 |
|
| `total_file_count` | int32 | 子文件数量 |
|
||||||
|
|
||||||
|
### 上传群文件
|
||||||
|
|
||||||
|
终结点: `/upload_group_file`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ------ | ------------------------- |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
| `file` | string | 本地文件路径 |
|
||||||
|
| `name` | string | 储存名称 |
|
||||||
|
| `folder` | string | 父目录ID |
|
||||||
|
|
||||||
|
> 在不提供 `folder` 参数的情况下默认上传到根目录
|
||||||
|
> 只能上传本地文件, 需要上传 `http` 文件的话请先调用 `download_file` API下载
|
||||||
|
|
||||||
### 获取状态
|
### 获取状态
|
||||||
|
|
||||||
终结点: `/get_status`
|
终结点: `/get_status`
|
||||||
@ -720,7 +881,7 @@ Type: `tts`
|
|||||||
| `plugins_good` | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
| `plugins_good` | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
||||||
| `app_good` | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
| `app_good` | bool | 原 `CQHTTP` 字段, 恒定为 `true` |
|
||||||
| `online` | bool | 表示BOT是否在线 |
|
| `online` | bool | 表示BOT是否在线 |
|
||||||
| `goold` | bool | 同 `online` |
|
| `good` | bool | 同 `online` |
|
||||||
| `stat` | Statistics | 运行统计 |
|
| `stat` | Statistics | 运行统计 |
|
||||||
|
|
||||||
**Statistics**
|
**Statistics**
|
||||||
@ -744,17 +905,158 @@ Type: `tts`
|
|||||||
|
|
||||||
**参数**
|
**参数**
|
||||||
|
|
||||||
| 字段 | 类型 | 说明 |
|
| 字段 | 类型 | 说明 |
|
||||||
| ---------- | ------ | ------------------------- |
|
| ---------- | ----- | ---- |
|
||||||
| `group_id` | int64 | 群号 |
|
| `group_id` | int64 | 群号 |
|
||||||
|
|
||||||
**响应数据**
|
**响应数据**
|
||||||
|
|
||||||
| 字段 | 类型 | 说明 |
|
| 字段 | 类型 | 说明 |
|
||||||
| ------------------------------- | ---------- | ------------------------------- |
|
| ------------------------------- | ----- | --------------------------------- |
|
||||||
| `can_at_all` | bool | 是否可以@全体成员 |
|
| `can_at_all` | bool | 是否可以@全体成员 |
|
||||||
| `remain_at_all_count_for_group` | int16 | 群内所有管理当天剩余@全体成员次数 |
|
| `remain_at_all_count_for_group` | int16 | 群内所有管理当天剩余@全体成员次数 |
|
||||||
| `remain_at_all_count_for_uin` | int16 | BOT当天剩余@全体成员次数 |
|
| `remain_at_all_count_for_uin` | int16 | BOT当天剩余@全体成员次数 |
|
||||||
|
|
||||||
|
### 下载文件到缓存目录
|
||||||
|
|
||||||
|
终结点: `/download_file`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| -------------- | --------------- | ------------ |
|
||||||
|
| `url` | string | 链接地址 |
|
||||||
|
| `thread_count` | int32 | 下载线程数 |
|
||||||
|
| `headers` | string or array | 自定义请求头 |
|
||||||
|
|
||||||
|
**`headers`格式:**
|
||||||
|
|
||||||
|
字符串:
|
||||||
|
|
||||||
|
```
|
||||||
|
User-Agent=YOUR_UA[\r\n]Referer=https://www.baidu.com
|
||||||
|
```
|
||||||
|
|
||||||
|
> `[\r\n]` 为换行符, 使用http请求时请注意编码
|
||||||
|
|
||||||
|
JSON数组:
|
||||||
|
|
||||||
|
```
|
||||||
|
[
|
||||||
|
"User-Agent=YOUR_UA",
|
||||||
|
"Referer=https://www.baidu.com",
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------ | ------ | -------------------- |
|
||||||
|
| `file` | string | 下载文件的*绝对路径* |
|
||||||
|
|
||||||
|
> 通过这个API下载的文件能直接放入CQ码作为图片或语音发送
|
||||||
|
> 调用后会阻塞直到下载完成后才会返回数据,请注意下载大文件时的超时
|
||||||
|
|
||||||
|
### 获取群消息历史记录
|
||||||
|
|
||||||
|
终结点:`/get_group_msg_history`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------- | ----- | ----------------------------------- |
|
||||||
|
| `message_seq` | int64 | 起始消息序号, 可通过 `get_msg` 获得 |
|
||||||
|
| `group_id` | int64 | 群号 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | --------- | -------------------------- |
|
||||||
|
| `messages` | []Message | 从起始序号开始的前19条消息 |
|
||||||
|
|
||||||
|
> 不提供起始序号将默认获取最新的消息
|
||||||
|
|
||||||
|
### 获取当前账号在线客户端列表
|
||||||
|
|
||||||
|
终结点:`/get_online_clients`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ---- | ------------ |
|
||||||
|
| `no_cache` | bool | 是否无视缓存 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| --------- | -------- | -------------- |
|
||||||
|
| `clients` | []Device | 在线客户端列表 |
|
||||||
|
|
||||||
|
**Device**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------- | ------ | -------- |
|
||||||
|
| `app_id` | int64 | 客户端ID |
|
||||||
|
| `device_name` | string | 设备名称 |
|
||||||
|
| `device_kind` | string | 设备类型 |
|
||||||
|
|
||||||
|
### 检查链接安全性
|
||||||
|
|
||||||
|
终结点:`/check_url_safely`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ------ | ------------------------- |
|
||||||
|
| `url` | string | 需要检查的链接 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ---------- | ---------- | ------------ |
|
||||||
|
| `level` | int | 安全等级, 1: 安全 2: 未知 3: 危险 |
|
||||||
|
|
||||||
|
### 获取用户VIP信息
|
||||||
|
|
||||||
|
终结点:`/_get_vip_info`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段名 | 数据类型 | 默认值 | 说明 |
|
||||||
|
| --------- | -------- | ------ | ----- |
|
||||||
|
| `user_id` | int64 | | QQ 号 |
|
||||||
|
|
||||||
|
**响应数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 说明 |
|
||||||
|
| ------------------ | ------- | ------------ |
|
||||||
|
| `user_id` | int64 | QQ 号 |
|
||||||
|
| `nickname` | string | 用户昵称 |
|
||||||
|
| `level` | int64 | QQ 等级 |
|
||||||
|
| `level_speed` | float64 | 等级加速度 |
|
||||||
|
| `vip_level` | string | 会员等级 |
|
||||||
|
| `vip_growth_speed` | int64 | 会员成长速度 |
|
||||||
|
| `vip_growth_total` | int64 | 会员成长总值 |
|
||||||
|
|
||||||
|
### 发送群公告
|
||||||
|
|
||||||
|
终结点: `/_send_group_notice`
|
||||||
|
|
||||||
|
**参数**
|
||||||
|
|
||||||
|
| 字段名 | 数据类型 | 默认值 | 说明 |
|
||||||
|
| ---------- | -------- | ------ | -------- |
|
||||||
|
| `group_id` | int64 | | 群号 |
|
||||||
|
| `content` | string | | 公告内容 |
|
||||||
|
|
||||||
|
`该 API 没有响应数据`
|
||||||
|
|
||||||
|
### 重载事件过滤器
|
||||||
|
|
||||||
|
终结点:`/reload_event_filter`
|
||||||
|
|
||||||
|
`该 API 无需参数也没有响应数据`
|
||||||
|
|
||||||
|
|
||||||
## 事件
|
## 事件
|
||||||
|
|
||||||
@ -786,16 +1088,16 @@ Type: `tts`
|
|||||||
|
|
||||||
**事件数据**
|
**事件数据**
|
||||||
|
|
||||||
| 字段名 | 数据类型 | 可能的值 | 说明 |
|
| 字段名 | 数据类型 | 可能的值 | 说明 |
|
||||||
| ------------- | ------ | -------- | --- |
|
| ------------- | -------- | -------- | ------------ |
|
||||||
| `post_type` | string | `notice` | 上报类型 |
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
| `notice_type` | string | `notify` | 消息类型 |
|
| `notice_type` | string | `notify` | 消息类型 |
|
||||||
| `sub_type` | string | `poke` | 提示类型 |
|
| `sub_type` | string | `poke` | 提示类型 |
|
||||||
| `self_id` | int64 | | BOT QQ 号 |
|
| `self_id` | int64 | | BOT QQ 号 |
|
||||||
| `sender_id` | int64 | | 发送者 QQ 号 |
|
| `sender_id` | int64 | | 发送者 QQ 号 |
|
||||||
| `user_id` | int64 | | 发送者 QQ 号 |
|
| `user_id` | int64 | | 发送者 QQ 号 |
|
||||||
| `target_id` | int64 | | 被戳者 QQ 号 |
|
| `target_id` | int64 | | 被戳者 QQ 号 |
|
||||||
| `time` | int64 | | 时间 |
|
| `time` | int64 | | 时间 |
|
||||||
|
|
||||||
### 群内戳一戳
|
### 群内戳一戳
|
||||||
|
|
||||||
@ -877,3 +1179,27 @@ Type: `tts`
|
|||||||
| `name` | string | | 文件名 |
|
| `name` | string | | 文件名 |
|
||||||
| `size` | int64 | | 文件大小 |
|
| `size` | int64 | | 文件大小 |
|
||||||
| `url` | string | | 下载链接 |
|
| `url` | string | | 下载链接 |
|
||||||
|
|
||||||
|
### 其他客户端在线状态变更
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | --------------- | ------------ |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `client_status` | 消息类型 |
|
||||||
|
| `client` | Device | | 客户端信息 |
|
||||||
|
| `online` | bool | | 当前是否在线 |
|
||||||
|
|
||||||
|
### 精华消息
|
||||||
|
|
||||||
|
**上报数据**
|
||||||
|
|
||||||
|
| 字段 | 类型 | 可能的值 | 说明 |
|
||||||
|
| ------------- | ------ | -------------- | -------------------------- |
|
||||||
|
| `post_type` | string | `notice` | 上报类型 |
|
||||||
|
| `notice_type` | string | `essence` | 消息类型 |
|
||||||
|
| `sub_type` | string | `add`,`delete` | 添加为`add`,移出为`delete` |
|
||||||
|
| `sender_id` | int64 | | 消息发送者ID |
|
||||||
|
| `operator_id` | int64 | | 操作者ID |
|
||||||
|
| `message_id` | int32 | | 消息ID |
|
||||||
|
@ -107,6 +107,14 @@ enable ws?(Y/n)
|
|||||||
|
|
||||||
# 进阶指南
|
# 进阶指南
|
||||||
|
|
||||||
|
## 跳过启动的五秒延时
|
||||||
|
|
||||||
|
使用命令行参数 `faststart`即可跳过启动的五秒钟延时,例如
|
||||||
|
|
||||||
|
```
|
||||||
|
.\go-cqhttp.exe faststart
|
||||||
|
```
|
||||||
|
|
||||||
## 如何自己构建
|
## 如何自己构建
|
||||||
|
|
||||||
1. [下载源码](https://github.com/Mrs4s/go-cqhttp/archive/master.zip)并解压 || 使用`git clone https://github.com/Mrs4s/go-cqhttp.git`来拉取
|
1. [下载源码](https://github.com/Mrs4s/go-cqhttp/archive/master.zip)并解压 || 使用`git clone https://github.com/Mrs4s/go-cqhttp.git`来拉取
|
||||||
|
@ -52,7 +52,7 @@
|
|||||||
|
|
||||||
此方案需要您有一台可以操作的 `Windows` 电脑.
|
此方案需要您有一台可以操作的 `Windows` 电脑.
|
||||||
|
|
||||||
首先下载工具: [蓝奏云](https://wws.lanzous.com/iw1Ejjr4rtg) [Google Drive](https://drive.google.com/file/d/1M6c3-kuD7ziHwl4hq75LcI119yB6HGdI/view?usp=sharing)
|
首先下载工具: [蓝奏云](https://wws.lanzous.com/i2vn0jrofte) [Google Drive](https://drive.google.com/file/d/1peMDHqgP8AgWBVp5vP-cfhcGrb2ksSrE/view?usp=sharing)
|
||||||
|
|
||||||
解压并打开工具:
|
解压并打开工具:
|
||||||
|
|
||||||
|
@ -2,44 +2,46 @@ package global
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"os/exec"
|
||||||
"path"
|
"path"
|
||||||
"sync"
|
|
||||||
|
|
||||||
log "github.com/sirupsen/logrus"
|
"github.com/Mrs4s/go-cqhttp/global/codec"
|
||||||
"github.com/wdvxdr1123/go-silk/silk"
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
var codec silk.Encoder
|
// EncoderSilk 将音频编码为Silk
|
||||||
var useCodec = true
|
func EncoderSilk(data []byte) ([]byte, error) {
|
||||||
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 := md5.New()
|
||||||
h.Write(data)
|
_, err := h.Write(data)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "calc md5 failed")
|
||||||
|
}
|
||||||
tempName := fmt.Sprintf("%x", h.Sum(nil))
|
tempName := fmt.Sprintf("%x", h.Sum(nil))
|
||||||
if silkPath := path.Join("data/cache", tempName+".silk"); PathExists(silkPath) {
|
if silkPath := path.Join("data/cache", tempName+".silk"); PathExists(silkPath) {
|
||||||
return ioutil.ReadFile(silkPath)
|
return ioutil.ReadFile(silkPath)
|
||||||
}
|
}
|
||||||
slk, err := codec.EncodeToSilk(data, tempName, true)
|
slk, err := codec.EncodeToSilk(data, tempName, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, errors.Wrap(err, "encode silk failed")
|
||||||
}
|
}
|
||||||
return slk, nil
|
return slk, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EncodeMP4 将给定视频文件编码为MP4
|
||||||
|
func EncodeMP4(src string, dst string) error { // -y 覆盖文件
|
||||||
|
cmd1 := exec.Command("ffmpeg", "-i", src, "-y", "-c", "copy", "-map", "0", dst)
|
||||||
|
err := cmd1.Run()
|
||||||
|
if err != nil {
|
||||||
|
cmd2 := exec.Command("ffmpeg", "-i", src, "-y", "-c:v", "h264", "-c:a", "mp3", dst)
|
||||||
|
return errors.Wrap(cmd2.Run(), "convert mp4 failed")
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// ExtractCover 获取给定视频文件的Cover
|
||||||
|
func ExtractCover(src string, target string) error {
|
||||||
|
cmd := exec.Command("ffmpeg", "-i", src, "-y", "-r", "1", "-f", "image2", target)
|
||||||
|
return errors.Wrap(cmd.Run(), "extract video cover failed")
|
||||||
|
}
|
||||||
|
53
global/codec/codec.go
Normal file
53
global/codec/codec.go
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
// +build linux windows,!arm darwin,!arm64
|
||||||
|
// +build 386 amd64 arm arm64
|
||||||
|
|
||||||
|
// Package codec Slik编码核心模块
|
||||||
|
package codec
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
"github.com/wdvxdr1123/go-silk"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
silkCachePath = "data/cache"
|
||||||
|
)
|
||||||
|
|
||||||
|
//EncodeToSilk 将音频编码为Silk
|
||||||
|
func EncodeToSilk(record []byte, tempName string, useCache bool) (silkWav []byte, err error) {
|
||||||
|
// 1. 写入缓存文件
|
||||||
|
rawPath := path.Join(silkCachePath, tempName+".wav")
|
||||||
|
err = ioutil.WriteFile(rawPath, record, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "write temp file error")
|
||||||
|
}
|
||||||
|
defer os.Remove(rawPath)
|
||||||
|
|
||||||
|
// 2.转换pcm
|
||||||
|
pcmPath := path.Join(silkCachePath, tempName+".pcm")
|
||||||
|
cmd := exec.Command("ffmpeg", "-i", rawPath, "-f", "s16le", "-ar", "24000", "-ac", "1", pcmPath)
|
||||||
|
if err = cmd.Run(); err != nil {
|
||||||
|
return nil, errors.Wrap(err, "convert pcm file error")
|
||||||
|
}
|
||||||
|
defer os.Remove(pcmPath)
|
||||||
|
|
||||||
|
// 3. 转silk
|
||||||
|
pcm, err := ioutil.ReadFile(pcmPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "read pcm file err")
|
||||||
|
}
|
||||||
|
silkWav, err = silk.EncodePcmBuffToSilk(pcm, 24000, 24000, true)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "silk encode error")
|
||||||
|
}
|
||||||
|
if useCache {
|
||||||
|
silkPath := path.Join(silkCachePath, tempName+".silk")
|
||||||
|
err = ioutil.WriteFile(silkPath, silkWav, 0666)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
8
global/codec/codec_darwin_arm64.go
Normal file
8
global/codec/codec_darwin_arm64.go
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
package codec
|
||||||
|
|
||||||
|
import "github.com/pkg/errors"
|
||||||
|
|
||||||
|
//EncodeToSilk 将音频编码为Silk
|
||||||
|
func EncodeToSilk(record []byte, tempName string, useCache bool) ([]byte, error) {
|
||||||
|
return nil, errors.New("not supported now")
|
||||||
|
}
|
10
global/codec/codec_unsupportedarch.go
Normal file
10
global/codec/codec_unsupportedarch.go
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
// +build !arm,!arm64,!amd64,!386
|
||||||
|
|
||||||
|
package codec
|
||||||
|
|
||||||
|
import "errors"
|
||||||
|
|
||||||
|
//EncodeToSilk 将音频编码为Silk
|
||||||
|
func EncodeToSilk(record []byte, tempName string, useCache bool) ([]byte, error) {
|
||||||
|
return nil, errors.New("not supported now")
|
||||||
|
}
|
10
global/codec/codec_unsupportedos.go
Normal file
10
global/codec/codec_unsupportedos.go
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
// +build !windows,!linux,!darwin
|
||||||
|
|
||||||
|
package codec
|
||||||
|
|
||||||
|
import "errors"
|
||||||
|
|
||||||
|
//EncodeToSilk 将音频编码为Silk
|
||||||
|
func EncodeToSilk(record []byte, tempName string, useCache bool) ([]byte, error) {
|
||||||
|
return nil, errors.New("not supported now")
|
||||||
|
}
|
8
global/codec/codec_windows_arm.go
Normal file
8
global/codec/codec_windows_arm.go
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
package codec
|
||||||
|
|
||||||
|
import "errors"
|
||||||
|
|
||||||
|
//EncodeToSilk 将音频编码为Silk
|
||||||
|
func EncodeToSilk(record []byte, tempName string, useCache bool) ([]byte, error) {
|
||||||
|
return nil, errors.New("not supported now")
|
||||||
|
}
|
@ -12,6 +12,7 @@ import (
|
|||||||
|
|
||||||
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
|
// DefaultConfigWithComments 为go-cqhttp的默认配置文件
|
||||||
var DefaultConfigWithComments = `
|
var DefaultConfigWithComments = `
|
||||||
/*
|
/*
|
||||||
go-cqhttp 默认配置文件
|
go-cqhttp 默认配置文件
|
||||||
@ -118,8 +119,8 @@ var DefaultConfigWithComments = `
|
|||||||
use_sso_address: false
|
use_sso_address: false
|
||||||
// 是否启用 DEBUG
|
// 是否启用 DEBUG
|
||||||
debug: false
|
debug: false
|
||||||
// 日志等级
|
// 日志等级 trace,debug,info,warn,error
|
||||||
log_level: ""
|
log_level: "info"
|
||||||
// WebUi 设置
|
// WebUi 设置
|
||||||
web_ui: {
|
web_ui: {
|
||||||
// 是否启用 WebUi
|
// 是否启用 WebUi
|
||||||
@ -134,12 +135,17 @@ var DefaultConfigWithComments = `
|
|||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
type JsonConfig struct {
|
// PasswordHash 存储QQ密码哈希供登录使用
|
||||||
|
var PasswordHash [16]byte
|
||||||
|
|
||||||
|
// JSONConfig Config对应的结构体
|
||||||
|
type JSONConfig struct {
|
||||||
Uin int64 `json:"uin"`
|
Uin int64 `json:"uin"`
|
||||||
Password string `json:"password"`
|
Password string `json:"password"`
|
||||||
EncryptPassword bool `json:"encrypt_password"`
|
EncryptPassword bool `json:"encrypt_password"`
|
||||||
PasswordEncrypted string `json:"password_encrypted"`
|
PasswordEncrypted string `json:"password_encrypted"`
|
||||||
EnableDB bool `json:"enable_db"`
|
EnableDB bool `json:"enable_db"`
|
||||||
|
EnableSelfMessage bool `json:"enable_self_message"`
|
||||||
AccessToken string `json:"access_token"`
|
AccessToken string `json:"access_token"`
|
||||||
ReLogin struct {
|
ReLogin struct {
|
||||||
Enabled bool `json:"enabled"`
|
Enabled bool `json:"enabled"`
|
||||||
@ -153,39 +159,41 @@ type JsonConfig struct {
|
|||||||
} `json:"_rate_limit"`
|
} `json:"_rate_limit"`
|
||||||
IgnoreInvalidCQCode bool `json:"ignore_invalid_cqcode"`
|
IgnoreInvalidCQCode bool `json:"ignore_invalid_cqcode"`
|
||||||
ForceFragmented bool `json:"force_fragmented"`
|
ForceFragmented bool `json:"force_fragmented"`
|
||||||
FixUrl bool `json:"fix_url"`
|
FixURL bool `json:"fix_url"`
|
||||||
ProxyRewrite string `json:"proxy_rewrite"`
|
ProxyRewrite string `json:"proxy_rewrite"`
|
||||||
HeartbeatInterval time.Duration `json:"heartbeat_interval"`
|
HeartbeatInterval time.Duration `json:"heartbeat_interval"`
|
||||||
HttpConfig *GoCQHttpConfig `json:"http_config"`
|
HTTPConfig *GoCQHTTPConfig `json:"http_config"`
|
||||||
WSConfig *GoCQWebsocketConfig `json:"ws_config"`
|
WSConfig *GoCQWebSocketConfig `json:"ws_config"`
|
||||||
ReverseServers []*GoCQReverseWebsocketConfig `json:"ws_reverse_servers"`
|
ReverseServers []*GoCQReverseWebSocketConfig `json:"ws_reverse_servers"`
|
||||||
PostMessageFormat string `json:"post_message_format"`
|
PostMessageFormat string `json:"post_message_format"`
|
||||||
UseSSOAddress bool `json:"use_sso_address"`
|
UseSSOAddress bool `json:"use_sso_address"`
|
||||||
Debug bool `json:"debug"`
|
Debug bool `json:"debug"`
|
||||||
LogLevel string `json:"log_level"`
|
LogLevel string `json:"log_level"`
|
||||||
WebUi *GoCqWebUi `json:"web_ui"`
|
WebUI *GoCQWebUI `json:"web_ui"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type CQHttpApiConfig struct {
|
// CQHTTPAPIConfig HTTPAPI对应的Config结构体
|
||||||
|
type CQHTTPAPIConfig struct {
|
||||||
Host string `json:"host"`
|
Host string `json:"host"`
|
||||||
Port uint16 `json:"port"`
|
Port uint16 `json:"port"`
|
||||||
UseHttp bool `json:"use_http"`
|
UseHTTP bool `json:"use_http"`
|
||||||
WSHost string `json:"ws_host"`
|
WSHost string `json:"ws_host"`
|
||||||
WSPort uint16 `json:"ws_port"`
|
WSPort uint16 `json:"ws_port"`
|
||||||
UseWS bool `json:"use_ws"`
|
UseWS bool `json:"use_ws"`
|
||||||
WSReverseUrl string `json:"ws_reverse_url"`
|
WSReverseURL string `json:"ws_reverse_url"`
|
||||||
WSReverseApiUrl string `json:"ws_reverse_api_url"`
|
WSReverseAPIURL string `json:"ws_reverse_api_url"`
|
||||||
WSReverseEventUrl string `json:"ws_reverse_event_url"`
|
WSReverseEventURL string `json:"ws_reverse_event_url"`
|
||||||
WSReverseReconnectInterval uint16 `json:"ws_reverse_reconnect_interval"`
|
WSReverseReconnectInterval uint16 `json:"ws_reverse_reconnect_interval"`
|
||||||
WSReverseReconnectOnCode1000 bool `json:"ws_reverse_reconnect_on_code_1000"`
|
WSReverseReconnectOnCode1000 bool `json:"ws_reverse_reconnect_on_code_1000"`
|
||||||
UseWsReverse bool `json:"use_ws_reverse"`
|
UseWsReverse bool `json:"use_ws_reverse"`
|
||||||
PostUrl string `json:"post_url"`
|
PostURL string `json:"post_url"`
|
||||||
AccessToken string `json:"access_token"`
|
AccessToken string `json:"access_token"`
|
||||||
Secret string `json:"secret"`
|
Secret string `json:"secret"`
|
||||||
PostMessageFormat string `json:"post_message_format"`
|
PostMessageFormat string `json:"post_message_format"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type GoCQHttpConfig struct {
|
// GoCQHTTPConfig 正向HTTP对应config结构体
|
||||||
|
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"`
|
||||||
@ -193,29 +201,33 @@ type GoCQHttpConfig struct {
|
|||||||
PostUrls map[string]string `json:"post_urls"`
|
PostUrls map[string]string `json:"post_urls"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type GoCQWebsocketConfig struct {
|
// GoCQWebSocketConfig 正向WebSocket对应Config结构体
|
||||||
|
type GoCQWebSocketConfig 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"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type GoCQReverseWebsocketConfig struct {
|
// GoCQReverseWebSocketConfig 反向WebSocket对应Config结构体
|
||||||
|
type GoCQReverseWebSocketConfig struct {
|
||||||
Enabled bool `json:"enabled"`
|
Enabled bool `json:"enabled"`
|
||||||
ReverseUrl string `json:"reverse_url"`
|
ReverseURL string `json:"reverse_url"`
|
||||||
ReverseApiUrl string `json:"reverse_api_url"`
|
ReverseAPIURL string `json:"reverse_api_url"`
|
||||||
ReverseEventUrl string `json:"reverse_event_url"`
|
ReverseEventURL string `json:"reverse_event_url"`
|
||||||
ReverseReconnectInterval uint16 `json:"reverse_reconnect_interval"`
|
ReverseReconnectInterval uint16 `json:"reverse_reconnect_interval"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type GoCqWebUi struct {
|
// GoCQWebUI WebUI对应Config结构体
|
||||||
|
type GoCQWebUI struct {
|
||||||
Enabled bool `json:"enabled"`
|
Enabled bool `json:"enabled"`
|
||||||
Host string `json:"host"`
|
Host string `json:"host"`
|
||||||
WebUiPort uint64 `json:"web_ui_port"`
|
WebUIPort uint64 `json:"web_ui_port"`
|
||||||
WebInput bool `json:"web_input"`
|
WebInput bool `json:"web_input"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func DefaultConfig() *JsonConfig {
|
// DefaultConfig 返回一份默认配置对应结构体
|
||||||
return &JsonConfig{
|
func DefaultConfig() *JSONConfig {
|
||||||
|
return &JSONConfig{
|
||||||
EnableDB: true,
|
EnableDB: true,
|
||||||
ReLogin: struct {
|
ReLogin: struct {
|
||||||
Enabled bool `json:"enabled"`
|
Enabled bool `json:"enabled"`
|
||||||
@ -237,42 +249,43 @@ func DefaultConfig() *JsonConfig {
|
|||||||
},
|
},
|
||||||
PostMessageFormat: "string",
|
PostMessageFormat: "string",
|
||||||
ForceFragmented: false,
|
ForceFragmented: false,
|
||||||
HttpConfig: &GoCQHttpConfig{
|
HTTPConfig: &GoCQHTTPConfig{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
Host: "0.0.0.0",
|
Host: "0.0.0.0",
|
||||||
Port: 5700,
|
Port: 5700,
|
||||||
PostUrls: map[string]string{},
|
PostUrls: map[string]string{},
|
||||||
},
|
},
|
||||||
WSConfig: &GoCQWebsocketConfig{
|
WSConfig: &GoCQWebSocketConfig{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
Host: "0.0.0.0",
|
Host: "0.0.0.0",
|
||||||
Port: 6700,
|
Port: 6700,
|
||||||
},
|
},
|
||||||
ReverseServers: []*GoCQReverseWebsocketConfig{
|
ReverseServers: []*GoCQReverseWebSocketConfig{
|
||||||
{
|
{
|
||||||
Enabled: false,
|
Enabled: false,
|
||||||
ReverseUrl: "ws://you_websocket_universal.server",
|
ReverseURL: "ws://you_websocket_universal.server",
|
||||||
ReverseApiUrl: "ws://you_websocket_api.server",
|
ReverseAPIURL: "ws://you_websocket_api.server",
|
||||||
ReverseEventUrl: "ws://you_websocket_event.server",
|
ReverseEventURL: "ws://you_websocket_event.server",
|
||||||
ReverseReconnectInterval: 3000,
|
ReverseReconnectInterval: 3000,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
WebUi: &GoCqWebUi{
|
WebUI: &GoCQWebUI{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
Host: "127.0.0.1",
|
Host: "127.0.0.1",
|
||||||
WebInput: false,
|
WebInput: false,
|
||||||
WebUiPort: 9999,
|
WebUIPort: 9999,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func Load(p string) *JsonConfig {
|
// LoadConfig 加载配置文件
|
||||||
|
func LoadConfig(p string) *JSONConfig {
|
||||||
if !PathExists(p) {
|
if !PathExists(p) {
|
||||||
log.Warnf("尝试加载配置文件 %v 失败: 文件不存在", p)
|
log.Warnf("尝试加载配置文件 %v 失败: 文件不存在", p)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
var dat map[string]interface{}
|
var dat map[string]interface{}
|
||||||
var c = JsonConfig{}
|
var c = JSONConfig{}
|
||||||
err := hjson.Unmarshal([]byte(ReadAllText(p)), &dat)
|
err := hjson.Unmarshal([]byte(ReadAllText(p)), &dat)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
b, _ := json.Marshal(dat)
|
b, _ := json.Marshal(dat)
|
||||||
@ -287,7 +300,8 @@ func Load(p string) *JsonConfig {
|
|||||||
return &c
|
return &c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *JsonConfig) Save(p string) error {
|
// Save 写入配置文件至path
|
||||||
|
func (c *JSONConfig) Save(path string) error {
|
||||||
data, err := hjson.MarshalWithOptions(c, hjson.EncoderOptions{
|
data, err := hjson.MarshalWithOptions(c, hjson.EncoderOptions{
|
||||||
Eol: "\n",
|
Eol: "\n",
|
||||||
BracesSameLine: true,
|
BracesSameLine: true,
|
||||||
@ -296,5 +310,5 @@ func (c *JsonConfig) Save(p string) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return WriteAllText(p, string(data))
|
return WriteAllText(path, string(data))
|
||||||
}
|
}
|
||||||
|
2
global/doc.go
Normal file
2
global/doc.go
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
// Package global 包含文件下载,视频音频编码,本地文件缓存处理,消息过滤器,调用速率限制,gocq主配置等的相关函数与结构体
|
||||||
|
package global
|
@ -10,8 +10,14 @@ import (
|
|||||||
"github.com/tidwall/gjson"
|
"github.com/tidwall/gjson"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// MSG 消息Map
|
||||||
type MSG map[string]interface{}
|
type MSG map[string]interface{}
|
||||||
|
|
||||||
|
// Get 尝试从消息Map中取出key为s的值,若不存在则返回MSG{}
|
||||||
|
//
|
||||||
|
// 若所给key对应的值的类型是global.MSG,则返回此值
|
||||||
|
//
|
||||||
|
// 若所给key对应值的类型不是global.MSG,则返回MSG{"__str__": Val}
|
||||||
func (m MSG) Get(s string) MSG {
|
func (m MSG) Get(s string) MSG {
|
||||||
if v, ok := m[s]; ok {
|
if v, ok := m[s]; ok {
|
||||||
if msg, ok := v.(MSG); ok {
|
if msg, ok := v.(MSG); ok {
|
||||||
@ -19,17 +25,25 @@ func (m MSG) Get(s string) MSG {
|
|||||||
}
|
}
|
||||||
return MSG{"__str__": v} // 用这个名字应该没问题吧
|
return MSG{"__str__": v} // 用这个名字应该没问题吧
|
||||||
}
|
}
|
||||||
return MSG{}
|
return nil // 不存在为空
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// String 将消息Map转化为String。若Map存在key "__str__",则返回此key对应的值,否则将输出整张消息Map对应的JSON字符串
|
||||||
func (m MSG) String() string {
|
func (m MSG) String() string {
|
||||||
|
if m == nil {
|
||||||
|
return "" // 空 JSON
|
||||||
|
}
|
||||||
if str, ok := m["__str__"]; ok {
|
if str, ok := m["__str__"]; ok {
|
||||||
|
if str == nil {
|
||||||
|
return "" // 空 JSON
|
||||||
|
}
|
||||||
return fmt.Sprint(str)
|
return fmt.Sprint(str)
|
||||||
}
|
}
|
||||||
str, _ := json.MarshalToString(m)
|
str, _ := json.MarshalToString(m)
|
||||||
return str
|
return str
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Filter 定义了一个消息上报过滤接口
|
||||||
type Filter interface {
|
type Filter interface {
|
||||||
Eval(payload MSG) bool
|
Eval(payload MSG) bool
|
||||||
}
|
}
|
||||||
@ -39,6 +53,7 @@ type operationNode struct {
|
|||||||
filter Filter
|
filter Filter
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NotOperator 定义了过滤器中Not操作符
|
||||||
type NotOperator struct {
|
type NotOperator struct {
|
||||||
operand Filter
|
operand Filter
|
||||||
}
|
}
|
||||||
@ -52,10 +67,12 @@ func notOperatorConstruct(argument gjson.Result) *NotOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行Not过滤
|
||||||
func (op *NotOperator) Eval(payload MSG) bool {
|
func (op *NotOperator) Eval(payload MSG) bool {
|
||||||
return !op.operand.Eval(payload)
|
return !op.operand.Eval(payload)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AndOperator 定义了过滤器中And操作符
|
||||||
type AndOperator struct {
|
type AndOperator struct {
|
||||||
operands []operationNode
|
operands []operationNode
|
||||||
}
|
}
|
||||||
@ -91,9 +108,10 @@ func andOperatorConstruct(argument gjson.Result) *AndOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
func (andOperator *AndOperator) Eval(payload MSG) bool {
|
// Eval 对payload执行And过滤
|
||||||
|
func (op *AndOperator) Eval(payload MSG) bool {
|
||||||
res := true
|
res := true
|
||||||
for _, operand := range andOperator.operands {
|
for _, operand := range op.operands {
|
||||||
|
|
||||||
if len(operand.key) == 0 {
|
if len(operand.key) == 0 {
|
||||||
// is an operator
|
// is an operator
|
||||||
@ -104,13 +122,14 @@ func (andOperator *AndOperator) Eval(payload MSG) bool {
|
|||||||
res = res && operand.filter.Eval(val)
|
res = res && operand.filter.Eval(val)
|
||||||
}
|
}
|
||||||
|
|
||||||
if res == false {
|
if !res {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// OrOperator 定义了过滤器中Or操作符
|
||||||
type OrOperator struct {
|
type OrOperator struct {
|
||||||
operands []Filter
|
operands []Filter
|
||||||
}
|
}
|
||||||
@ -127,17 +146,19 @@ func orOperatorConstruct(argument gjson.Result) *OrOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行Or过滤
|
||||||
func (op *OrOperator) Eval(payload MSG) bool {
|
func (op *OrOperator) Eval(payload MSG) bool {
|
||||||
res := false
|
res := false
|
||||||
for _, operand := range op.operands {
|
for _, operand := range op.operands {
|
||||||
res = res || operand.Eval(payload)
|
res = res || operand.Eval(payload)
|
||||||
if res == true {
|
if res {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EqualOperator 定义了过滤器中Equal操作符
|
||||||
type EqualOperator struct {
|
type EqualOperator struct {
|
||||||
operand string
|
operand string
|
||||||
}
|
}
|
||||||
@ -148,10 +169,12 @@ func equalOperatorConstruct(argument gjson.Result) *EqualOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行Equal过滤
|
||||||
func (op *EqualOperator) Eval(payload MSG) bool {
|
func (op *EqualOperator) Eval(payload MSG) bool {
|
||||||
return payload.String() == op.operand
|
return payload.String() == op.operand
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NotEqualOperator 定义了过滤器中NotEqual操作符
|
||||||
type NotEqualOperator struct {
|
type NotEqualOperator struct {
|
||||||
operand string
|
operand string
|
||||||
}
|
}
|
||||||
@ -162,10 +185,12 @@ func notEqualOperatorConstruct(argument gjson.Result) *NotEqualOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行NotEqual过滤
|
||||||
func (op *NotEqualOperator) Eval(payload MSG) bool {
|
func (op *NotEqualOperator) Eval(payload MSG) bool {
|
||||||
return !(payload.String() == op.operand)
|
return !(payload.String() == op.operand)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InOperator 定义了过滤器中In操作符
|
||||||
type InOperator struct {
|
type InOperator struct {
|
||||||
operandString string
|
operandString string
|
||||||
operandArray []string
|
operandArray []string
|
||||||
@ -188,6 +213,7 @@ func inOperatorConstruct(argument gjson.Result) *InOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行In过滤
|
||||||
func (op *InOperator) Eval(payload MSG) bool {
|
func (op *InOperator) Eval(payload MSG) bool {
|
||||||
payloadStr := payload.String()
|
payloadStr := payload.String()
|
||||||
if op.operandArray != nil {
|
if op.operandArray != nil {
|
||||||
@ -201,6 +227,7 @@ func (op *InOperator) Eval(payload MSG) bool {
|
|||||||
return strings.Contains(op.operandString, payloadStr)
|
return strings.Contains(op.operandString, payloadStr)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ContainsOperator 定义了过滤器中Contains操作符
|
||||||
type ContainsOperator struct {
|
type ContainsOperator struct {
|
||||||
operand string
|
operand string
|
||||||
}
|
}
|
||||||
@ -214,10 +241,12 @@ func containsOperatorConstruct(argument gjson.Result) *ContainsOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行Contains过滤
|
||||||
func (op *ContainsOperator) Eval(payload MSG) bool {
|
func (op *ContainsOperator) Eval(payload MSG) bool {
|
||||||
return strings.Contains(payload.String(), op.operand)
|
return strings.Contains(payload.String(), op.operand)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RegexOperator 定义了过滤器中Regex操作符
|
||||||
type RegexOperator struct {
|
type RegexOperator struct {
|
||||||
regex *regexp.Regexp
|
regex *regexp.Regexp
|
||||||
}
|
}
|
||||||
@ -231,11 +260,13 @@ func regexOperatorConstruct(argument gjson.Result) *RegexOperator {
|
|||||||
return op
|
return op
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Eval 对payload执行RegexO过滤
|
||||||
func (op *RegexOperator) Eval(payload MSG) bool {
|
func (op *RegexOperator) Eval(payload MSG) bool {
|
||||||
matched := op.regex.MatchString(payload.String())
|
matched := op.regex.MatchString(payload.String())
|
||||||
return matched
|
return matched
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Generate 根据给定操作符名opName及操作符参数argument创建一个过滤器实例
|
||||||
func Generate(opName string, argument gjson.Result) Filter {
|
func Generate(opName string, argument gjson.Result) Filter {
|
||||||
switch opName {
|
switch opName {
|
||||||
case "not":
|
case "not":
|
||||||
@ -259,8 +290,10 @@ func Generate(opName string, argument gjson.Result) Filter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var EventFilter Filter = nil
|
// EventFilter 初始化一个nil过滤器
|
||||||
|
var EventFilter Filter
|
||||||
|
|
||||||
|
// BootFilter 启动事件过滤器
|
||||||
func BootFilter() {
|
func BootFilter() {
|
||||||
defer func() {
|
defer func() {
|
||||||
if e := recover(); e != nil {
|
if e := recover(); e != nil {
|
||||||
|
159
global/fs.go
159
global/fs.go
@ -1,90 +1,114 @@
|
|||||||
package global
|
package global
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"compress/bzip2"
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net"
|
"net"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/kardianos/osext"
|
||||||
|
|
||||||
"github.com/dustin/go-humanize"
|
"github.com/dustin/go-humanize"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
const (
|
||||||
IMAGE_PATH = path.Join("data", "images")
|
// ImagePath go-cqhttp使用的图片缓存目录
|
||||||
IMAGE_PATH_OLD = path.Join("data", "image")
|
ImagePath = "data/images"
|
||||||
VOICE_PATH = path.Join("data", "voices")
|
// ImagePathOld 兼容旧版go-cqhtto使用的图片缓存目录
|
||||||
VOICE_PATH_OLD = path.Join("data", "record")
|
ImagePathOld = "data/image"
|
||||||
VIDEO_PATH = path.Join("data", "videos")
|
// VoicePath go-cqhttp使用的语音缓存目录
|
||||||
CACHE_PATH = path.Join("data", "cache")
|
VoicePath = "data/voices"
|
||||||
|
// VoicePathOld 兼容旧版go-cqhtto使用的语音缓存目录
|
||||||
HEADER_AMR = []byte("#!AMR")
|
VoicePathOld = "data/record"
|
||||||
HEADER_SILK = []byte("\x02#!SILK_V3")
|
// VideoPath go-cqhttp使用的视频缓存目录
|
||||||
|
VideoPath = "data/videos"
|
||||||
ErrSyntax = errors.New("syntax error")
|
// CachePath go-cqhttp使用的缓存目录
|
||||||
|
CachePath = "data/cache"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// ErrSyntax Path语法错误时返回的错误
|
||||||
|
ErrSyntax = errors.New("syntax error")
|
||||||
|
// HeaderAmr AMR文件头
|
||||||
|
HeaderAmr = []byte("#!AMR")
|
||||||
|
// HeaderSilk Silkv3文件头
|
||||||
|
HeaderSilk = []byte("\x02#!SILK_V3")
|
||||||
|
)
|
||||||
|
|
||||||
|
// PathExists 判断给定path是否存在
|
||||||
func PathExists(path string) bool {
|
func PathExists(path string) bool {
|
||||||
_, err := os.Stat(path)
|
_, err := os.Stat(path)
|
||||||
return err == nil || os.IsExist(err)
|
return err == nil || os.IsExist(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadAllText 读取给定path对应文件,无法读取时返回空值
|
||||||
func ReadAllText(path string) string {
|
func ReadAllText(path string) string {
|
||||||
b, err := ioutil.ReadFile(path)
|
b, err := ioutil.ReadFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
log.Error(err)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return string(b)
|
return string(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WriteAllText 将给定text写入给定path
|
||||||
func WriteAllText(path, text string) error {
|
func WriteAllText(path, text string) error {
|
||||||
return ioutil.WriteFile(path, []byte(text), 0644)
|
return ioutil.WriteFile(path, []byte(text), 0644)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check 检测err是否为nil
|
||||||
func Check(err error) {
|
func Check(err error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("遇到错误: %v", err)
|
log.Fatalf("遇到错误: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsAMRorSILK 判断给定文件是否为Amr或Silk格式
|
||||||
func IsAMRorSILK(b []byte) bool {
|
func IsAMRorSILK(b []byte) bool {
|
||||||
return bytes.HasPrefix(b, HEADER_AMR) || bytes.HasPrefix(b, HEADER_SILK)
|
return bytes.HasPrefix(b, HeaderAmr) || bytes.HasPrefix(b, HeaderSilk)
|
||||||
}
|
}
|
||||||
|
|
||||||
func FindFile(f, cache, PATH string) (data []byte, err error) {
|
// FindFile 从给定的File寻找文件,并返回文件byte数组。File是一个合法的URL。Path为文件寻找位置。
|
||||||
|
// 对于HTTP/HTTPS形式的URL,Cache为"1"或空时表示启用缓存
|
||||||
|
func FindFile(file, cache, PATH string) (data []byte, err error) {
|
||||||
data, err = nil, ErrSyntax
|
data, err = nil, ErrSyntax
|
||||||
if strings.HasPrefix(f, "http") || strings.HasPrefix(f, "https") {
|
if strings.HasPrefix(file, "http") || strings.HasPrefix(file, "https") {
|
||||||
if cache == "" {
|
if cache == "" {
|
||||||
cache = "1"
|
cache = "1"
|
||||||
}
|
}
|
||||||
hash := md5.Sum([]byte(f))
|
hash := md5.Sum([]byte(file))
|
||||||
cacheFile := path.Join(CACHE_PATH, hex.EncodeToString(hash[:])+".cache")
|
cacheFile := path.Join(CachePath, hex.EncodeToString(hash[:])+".cache")
|
||||||
if PathExists(cacheFile) && cache == "1" {
|
if PathExists(cacheFile) && cache == "1" {
|
||||||
return ioutil.ReadFile(cacheFile)
|
return ioutil.ReadFile(cacheFile)
|
||||||
}
|
}
|
||||||
data, err = GetBytes(f)
|
data, err = GetBytes(file)
|
||||||
_ = ioutil.WriteFile(cacheFile, data, 0644)
|
_ = ioutil.WriteFile(cacheFile, data, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else if strings.HasPrefix(f, "base64") {
|
} else if strings.HasPrefix(file, "base64") {
|
||||||
data, err = base64.StdEncoding.DecodeString(strings.ReplaceAll(f, "base64://", ""))
|
data, err = base64.StdEncoding.DecodeString(strings.ReplaceAll(file, "base64://", ""))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else if strings.HasPrefix(f, "file") {
|
} else if strings.HasPrefix(file, "file") {
|
||||||
var fu *url.URL
|
var fu *url.URL
|
||||||
fu, err = url.Parse(f)
|
fu, err = url.Parse(file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -95,8 +119,8 @@ func FindFile(f, cache, PATH string) (data []byte, err error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else if PathExists(path.Join(PATH, f)) {
|
} else if PathExists(path.Join(PATH, file)) {
|
||||||
data, err = ioutil.ReadFile(path.Join(PATH, f))
|
data, err = ioutil.ReadFile(path.Join(PATH, file))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -104,19 +128,20 @@ func FindFile(f, cache, PATH string) (data []byte, err error) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DelFile 删除一个给定path,并返回删除结果
|
||||||
func DelFile(path string) bool {
|
func DelFile(path string) bool {
|
||||||
err := os.Remove(path)
|
err := os.Remove(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 删除失败
|
// 删除失败
|
||||||
log.Error(err)
|
log.Error(err)
|
||||||
return false
|
return false
|
||||||
} else {
|
|
||||||
// 删除成功
|
|
||||||
log.Info(path + "删除成功")
|
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
|
// 删除成功
|
||||||
|
log.Info(path + "删除成功")
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadAddrFile 从给定path中读取合法的IP地址与端口,每个IP地址以换行符"\n"作为分隔
|
||||||
func ReadAddrFile(path string) []*net.TCPAddr {
|
func ReadAddrFile(path string) []*net.TCPAddr {
|
||||||
d, err := ioutil.ReadFile(path)
|
d, err := ioutil.ReadFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -135,10 +160,12 @@ func ReadAddrFile(path string) []*net.TCPAddr {
|
|||||||
return ret
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WriteCounter 写入量计算实例
|
||||||
type WriteCounter struct {
|
type WriteCounter struct {
|
||||||
Total uint64
|
Total uint64
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Write 方法将写入的byte长度追加至写入的总长度Total中
|
||||||
func (wc *WriteCounter) Write(p []byte) (int, error) {
|
func (wc *WriteCounter) Write(p []byte) (int, error) {
|
||||||
n := len(p)
|
n := len(p)
|
||||||
wc.Total += uint64(n)
|
wc.Total += uint64(n)
|
||||||
@ -146,7 +173,81 @@ func (wc *WriteCounter) Write(p []byte) (int, error) {
|
|||||||
return n, nil
|
return n, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (wc WriteCounter) PrintProgress() {
|
// PrintProgress 方法将打印当前的总写入量
|
||||||
|
func (wc *WriteCounter) PrintProgress() {
|
||||||
fmt.Printf("\r%s", strings.Repeat(" ", 35))
|
fmt.Printf("\r%s", strings.Repeat(" ", 35))
|
||||||
fmt.Printf("\rDownloading... %s complete", humanize.Bytes(wc.Total))
|
fmt.Printf("\rDownloading... %s complete", humanize.Bytes(wc.Total))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UpdateFromStream copy form getlantern/go-update
|
||||||
|
func UpdateFromStream(updateWith io.Reader) (err error, errRecover error) {
|
||||||
|
updatePath, err := osext.Executable()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
var newBytes []byte
|
||||||
|
// no patch to apply, go on through
|
||||||
|
var fileHeader []byte
|
||||||
|
bufBytes := bufio.NewReader(updateWith)
|
||||||
|
fileHeader, err = bufBytes.Peek(2)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// The content is always bzip2 compressed except when running test, in
|
||||||
|
// which case is not prefixed with the magic byte sequence for sure.
|
||||||
|
if bytes.Equal([]byte{0x42, 0x5a}, fileHeader) {
|
||||||
|
// Identifying bzip2 files.
|
||||||
|
updateWith = bzip2.NewReader(bufBytes)
|
||||||
|
} else {
|
||||||
|
updateWith = io.Reader(bufBytes)
|
||||||
|
}
|
||||||
|
newBytes, err = ioutil.ReadAll(updateWith)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// get the directory the executable exists in
|
||||||
|
updateDir := filepath.Dir(updatePath)
|
||||||
|
filename := filepath.Base(updatePath)
|
||||||
|
// Copy the contents of of newbinary to a the new executable file
|
||||||
|
newPath := filepath.Join(updateDir, fmt.Sprintf(".%s.new", filename))
|
||||||
|
fp, err := os.OpenFile(newPath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0755)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// We won't log this error, because it's always going to happen.
|
||||||
|
defer func() { _ = fp.Close() }()
|
||||||
|
if _, err = io.Copy(fp, bytes.NewReader(newBytes)); err != nil {
|
||||||
|
log.Errorf("Unable to copy data: %v\n", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// if we don't call fp.Close(), windows won't let us move the new executable
|
||||||
|
// because the file will still be "in use"
|
||||||
|
if err := fp.Close(); err != nil {
|
||||||
|
log.Errorf("Unable to close file: %v\n", err)
|
||||||
|
}
|
||||||
|
// this is where we'll move the executable to so that we can swap in the updated replacement
|
||||||
|
oldPath := filepath.Join(updateDir, fmt.Sprintf(".%s.old", filename))
|
||||||
|
|
||||||
|
// delete any existing old exec file - this is necessary on Windows for two reasons:
|
||||||
|
// 1. after a successful update, Windows can't remove the .old file because the process is still running
|
||||||
|
// 2. windows rename operations fail if the destination file already exists
|
||||||
|
_ = os.Remove(oldPath)
|
||||||
|
|
||||||
|
// move the existing executable to a new file in the same directory
|
||||||
|
err = os.Rename(updatePath, oldPath)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// move the new executable in to become the new program
|
||||||
|
err = os.Rename(newPath, updatePath)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
// copy unsuccessful
|
||||||
|
errRecover = os.Rename(oldPath, updatePath)
|
||||||
|
} else {
|
||||||
|
// copy successful, remove the old binary
|
||||||
|
_ = os.Remove(oldPath)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
165
global/log_hook.go
Normal file
165
global/log_hook.go
Normal file
@ -0,0 +1,165 @@
|
|||||||
|
package global
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"reflect"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LocalHook logrus本地钩子
|
||||||
|
type LocalHook struct {
|
||||||
|
lock *sync.Mutex
|
||||||
|
levels []logrus.Level // hook级别
|
||||||
|
formatter logrus.Formatter // 格式
|
||||||
|
path string // 写入path
|
||||||
|
writer io.Writer // io
|
||||||
|
}
|
||||||
|
|
||||||
|
// Levels ref: logrus/hooks.go impl Hook interface
|
||||||
|
func (hook *LocalHook) Levels() []logrus.Level {
|
||||||
|
if len(hook.levels) == 0 {
|
||||||
|
return logrus.AllLevels
|
||||||
|
}
|
||||||
|
return hook.levels
|
||||||
|
}
|
||||||
|
|
||||||
|
func (hook *LocalHook) ioWrite(entry *logrus.Entry) error {
|
||||||
|
log, err := hook.formatter.Format(entry)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = hook.writer.Write(log)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (hook *LocalHook) pathWrite(entry *logrus.Entry) error {
|
||||||
|
dir := filepath.Dir(hook.path)
|
||||||
|
if err := os.MkdirAll(dir, os.ModePerm); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
fd, err := os.OpenFile(hook.path, os.O_WRONLY|os.O_APPEND|os.O_CREATE, 0666)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer fd.Close()
|
||||||
|
|
||||||
|
log, err := hook.formatter.Format(entry)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = fd.Write(log)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fire ref: logrus/hooks.go impl Hook interface
|
||||||
|
func (hook *LocalHook) Fire(entry *logrus.Entry) error {
|
||||||
|
hook.lock.Lock()
|
||||||
|
defer hook.lock.Unlock()
|
||||||
|
|
||||||
|
if hook.writer != nil {
|
||||||
|
return hook.ioWrite(entry)
|
||||||
|
}
|
||||||
|
|
||||||
|
if hook.path != "" {
|
||||||
|
return hook.pathWrite(entry)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetFormatter 设置日志格式
|
||||||
|
func (hook *LocalHook) SetFormatter(formatter logrus.Formatter) {
|
||||||
|
hook.lock.Lock()
|
||||||
|
defer hook.lock.Unlock()
|
||||||
|
|
||||||
|
if formatter == nil {
|
||||||
|
// 用默认的
|
||||||
|
formatter = &logrus.TextFormatter{DisableColors: true}
|
||||||
|
} else {
|
||||||
|
switch f := formatter.(type) {
|
||||||
|
case *logrus.TextFormatter:
|
||||||
|
textFormatter := f
|
||||||
|
textFormatter.DisableColors = true
|
||||||
|
default:
|
||||||
|
// todo
|
||||||
|
}
|
||||||
|
}
|
||||||
|
logrus.SetFormatter(formatter)
|
||||||
|
hook.formatter = formatter
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetWriter 设置Writer
|
||||||
|
func (hook *LocalHook) SetWriter(writer io.Writer) {
|
||||||
|
hook.lock.Lock()
|
||||||
|
defer hook.lock.Unlock()
|
||||||
|
hook.writer = writer
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetPath 设置日志写入路径
|
||||||
|
func (hook *LocalHook) SetPath(path string) {
|
||||||
|
hook.lock.Lock()
|
||||||
|
defer hook.lock.Unlock()
|
||||||
|
hook.path = path
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewLocalHook 初始化本地日志钩子实现
|
||||||
|
func NewLocalHook(args interface{}, formatter logrus.Formatter, levels ...logrus.Level) *LocalHook {
|
||||||
|
hook := &LocalHook{
|
||||||
|
lock: new(sync.Mutex),
|
||||||
|
}
|
||||||
|
hook.SetFormatter(formatter)
|
||||||
|
hook.levels = append(hook.levels, levels...)
|
||||||
|
|
||||||
|
switch arg := args.(type) {
|
||||||
|
case string:
|
||||||
|
hook.SetPath(arg)
|
||||||
|
case io.Writer:
|
||||||
|
hook.SetWriter(arg)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("unsupported type: %v", reflect.TypeOf(args)))
|
||||||
|
}
|
||||||
|
|
||||||
|
return hook
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetLogLevel 获取日志等级
|
||||||
|
//
|
||||||
|
// 可能的值有
|
||||||
|
//
|
||||||
|
// "trace","debug","info","warn","warn","error"
|
||||||
|
func GetLogLevel(level string) []logrus.Level {
|
||||||
|
switch level {
|
||||||
|
case "trace":
|
||||||
|
return []logrus.Level{logrus.TraceLevel, logrus.DebugLevel,
|
||||||
|
logrus.InfoLevel, logrus.WarnLevel, logrus.ErrorLevel,
|
||||||
|
logrus.FatalLevel, logrus.PanicLevel}
|
||||||
|
case "debug":
|
||||||
|
return []logrus.Level{logrus.DebugLevel, logrus.InfoLevel,
|
||||||
|
logrus.WarnLevel, logrus.ErrorLevel,
|
||||||
|
logrus.FatalLevel, logrus.PanicLevel}
|
||||||
|
case "info":
|
||||||
|
return []logrus.Level{logrus.InfoLevel, logrus.WarnLevel,
|
||||||
|
logrus.ErrorLevel, logrus.FatalLevel, logrus.PanicLevel}
|
||||||
|
case "warn":
|
||||||
|
return []logrus.Level{logrus.WarnLevel, logrus.ErrorLevel,
|
||||||
|
logrus.FatalLevel, logrus.PanicLevel}
|
||||||
|
case "error":
|
||||||
|
return []logrus.Level{logrus.ErrorLevel, logrus.FatalLevel,
|
||||||
|
logrus.PanicLevel}
|
||||||
|
default:
|
||||||
|
return []logrus.Level{logrus.InfoLevel, logrus.WarnLevel,
|
||||||
|
logrus.ErrorLevel, logrus.FatalLevel, logrus.PanicLevel}
|
||||||
|
}
|
||||||
|
}
|
252
global/net.go
252
global/net.go
@ -1,50 +1,59 @@
|
|||||||
package global
|
package global
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"bytes"
|
"bytes"
|
||||||
"compress/gzip"
|
"compress/gzip"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/guonaihong/gout"
|
"io"
|
||||||
"github.com/pkg/errors"
|
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"os"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/guonaihong/gout"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
"github.com/tidwall/gjson"
|
"github.com/tidwall/gjson"
|
||||||
)
|
)
|
||||||
|
|
||||||
var client = &http.Client{
|
var (
|
||||||
Timeout: time.Second * 15,
|
client = &http.Client{
|
||||||
Transport: &http.Transport{
|
Transport: &http.Transport{
|
||||||
Proxy: func(request *http.Request) (u *url.URL, e error) {
|
Proxy: func(request *http.Request) (u *url.URL, e error) {
|
||||||
if Proxy == "" {
|
if Proxy == "" {
|
||||||
return http.ProxyFromEnvironment(request)
|
return http.ProxyFromEnvironment(request)
|
||||||
}
|
}
|
||||||
return url.Parse(Proxy)
|
return url.Parse(Proxy)
|
||||||
|
},
|
||||||
|
ForceAttemptHTTP2: true,
|
||||||
|
MaxConnsPerHost: 0,
|
||||||
|
MaxIdleConns: 0,
|
||||||
|
MaxIdleConnsPerHost: 999,
|
||||||
},
|
},
|
||||||
DialContext: (&net.Dialer{
|
}
|
||||||
Timeout: 30 * time.Second,
|
|
||||||
KeepAlive: 30 * time.Second,
|
|
||||||
}).DialContext,
|
|
||||||
ForceAttemptHTTP2: true,
|
|
||||||
MaxIdleConns: 100,
|
|
||||||
IdleConnTimeout: 90 * time.Second,
|
|
||||||
TLSHandshakeTimeout: 10 * time.Second,
|
|
||||||
ExpectContinueTimeout: 1 * time.Second,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
var Proxy string
|
// Proxy 存储Config.proxy_rewrite,用于设置代理
|
||||||
|
Proxy string
|
||||||
|
|
||||||
|
// ErrOverSize 响应主体过大时返回此错误
|
||||||
|
ErrOverSize = errors.New("oversize")
|
||||||
|
|
||||||
|
// UserAgent HTTP请求时使用的UA
|
||||||
|
UserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36 Edg/87.0.664.66"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetBytes 对给定URL发送Get请求,返回响应主体
|
||||||
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{UserAgent}
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -64,6 +73,199 @@ func GetBytes(url string) ([]byte, error) {
|
|||||||
return body, nil
|
return body, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DownloadFile 将给定URL对应的文件下载至给定Path
|
||||||
|
func DownloadFile(url, path string, limit int64, headers map[string]string) error {
|
||||||
|
file, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE, 0666)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for k, v := range headers {
|
||||||
|
req.Header.Set(k, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, ok := headers["User-Agent"]; !ok {
|
||||||
|
req.Header["User-Agent"] = []string{UserAgent}
|
||||||
|
}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
if limit > 0 && resp.ContentLength > limit {
|
||||||
|
return ErrOverSize
|
||||||
|
}
|
||||||
|
_, err = io.Copy(file, resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DownloadFileMultiThreading 使用threadCount个线程将给定URL对应的文件下载至给定Path
|
||||||
|
func DownloadFileMultiThreading(url, path string, limit int64, threadCount int, headers map[string]string) error {
|
||||||
|
if threadCount < 2 {
|
||||||
|
return DownloadFile(url, path, limit, headers)
|
||||||
|
}
|
||||||
|
type BlockMetaData struct {
|
||||||
|
BeginOffset int64
|
||||||
|
EndOffset int64
|
||||||
|
DownloadedSize int64
|
||||||
|
}
|
||||||
|
var blocks []*BlockMetaData
|
||||||
|
var contentLength int64
|
||||||
|
errUnsupportedMultiThreading := errors.New("unsupported multi-threading")
|
||||||
|
// 初始化分块或直接下载
|
||||||
|
initOrDownload := func() error {
|
||||||
|
copyStream := func(s io.ReadCloser) error {
|
||||||
|
file, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE, 0666)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
if _, err = io.Copy(file, s); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return errUnsupportedMultiThreading
|
||||||
|
}
|
||||||
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for k, v := range headers {
|
||||||
|
req.Header.Set(k, v)
|
||||||
|
|
||||||
|
}
|
||||||
|
if _, ok := headers["User-Agent"]; !ok {
|
||||||
|
req.Header["User-Agent"] = []string{UserAgent}
|
||||||
|
}
|
||||||
|
req.Header.Set("range", "bytes=0-")
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if resp.StatusCode < 200 || resp.StatusCode >= 300 {
|
||||||
|
return errors.New("response status unsuccessful: " + strconv.FormatInt(int64(resp.StatusCode), 10))
|
||||||
|
}
|
||||||
|
if resp.StatusCode == 200 {
|
||||||
|
if limit > 0 && resp.ContentLength > limit {
|
||||||
|
return ErrOverSize
|
||||||
|
}
|
||||||
|
return copyStream(resp.Body)
|
||||||
|
}
|
||||||
|
if resp.StatusCode == 206 {
|
||||||
|
contentLength = resp.ContentLength
|
||||||
|
if limit > 0 && resp.ContentLength > limit {
|
||||||
|
return ErrOverSize
|
||||||
|
}
|
||||||
|
blockSize := func() int64 {
|
||||||
|
if contentLength > 1024*1024 {
|
||||||
|
return (contentLength / int64(threadCount)) - 10
|
||||||
|
}
|
||||||
|
return contentLength
|
||||||
|
|
||||||
|
}()
|
||||||
|
if blockSize == contentLength {
|
||||||
|
return copyStream(resp.Body)
|
||||||
|
}
|
||||||
|
var tmp int64
|
||||||
|
for tmp+blockSize < contentLength {
|
||||||
|
blocks = append(blocks, &BlockMetaData{
|
||||||
|
BeginOffset: tmp,
|
||||||
|
EndOffset: tmp + blockSize - 1,
|
||||||
|
})
|
||||||
|
tmp += blockSize
|
||||||
|
}
|
||||||
|
blocks = append(blocks, &BlockMetaData{
|
||||||
|
BeginOffset: tmp,
|
||||||
|
EndOffset: contentLength - 1,
|
||||||
|
})
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return errors.New("unknown status code")
|
||||||
|
}
|
||||||
|
// 下载分块
|
||||||
|
downloadBlock := func(block *BlockMetaData) error {
|
||||||
|
req, _ := http.NewRequest("GET", url, nil)
|
||||||
|
file, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE, 0666)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
_, _ = file.Seek(block.BeginOffset, io.SeekStart)
|
||||||
|
writer := bufio.NewWriter(file)
|
||||||
|
defer writer.Flush()
|
||||||
|
|
||||||
|
for k, v := range headers {
|
||||||
|
req.Header.Set(k, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, ok := headers["User-Agent"]; ok {
|
||||||
|
req.Header["User-Agent"] = []string{UserAgent}
|
||||||
|
}
|
||||||
|
req.Header.Set("range", "bytes="+strconv.FormatInt(block.BeginOffset, 10)+"-"+strconv.FormatInt(block.EndOffset, 10))
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
if resp.StatusCode < 200 || resp.StatusCode >= 300 {
|
||||||
|
return errors.New("response status unsuccessful: " + strconv.FormatInt(int64(resp.StatusCode), 10))
|
||||||
|
}
|
||||||
|
var buffer = make([]byte, 1024)
|
||||||
|
i, err := resp.Body.Read(buffer)
|
||||||
|
for {
|
||||||
|
if err != nil && err != io.EOF {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
i64 := int64(len(buffer[:i]))
|
||||||
|
needSize := block.EndOffset + 1 - block.BeginOffset
|
||||||
|
if i64 > needSize {
|
||||||
|
i64 = needSize
|
||||||
|
err = io.EOF
|
||||||
|
}
|
||||||
|
_, e := writer.Write(buffer[:i64])
|
||||||
|
if e != nil {
|
||||||
|
return e
|
||||||
|
}
|
||||||
|
block.BeginOffset += i64
|
||||||
|
block.DownloadedSize += i64
|
||||||
|
if err == io.EOF || block.BeginOffset > block.EndOffset {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
i, err = resp.Body.Read(buffer)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := initOrDownload(); err != nil {
|
||||||
|
if err == errUnsupportedMultiThreading {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
wg := sync.WaitGroup{}
|
||||||
|
wg.Add(len(blocks))
|
||||||
|
var lastErr error
|
||||||
|
for i := range blocks {
|
||||||
|
go func(b *BlockMetaData) {
|
||||||
|
defer wg.Done()
|
||||||
|
if err := downloadBlock(b); err != nil {
|
||||||
|
lastErr = err
|
||||||
|
}
|
||||||
|
}(blocks[i])
|
||||||
|
}
|
||||||
|
wg.Wait()
|
||||||
|
return lastErr
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetSliderTicket 通过给定的验证链接raw和id,获取验证结果Ticket
|
||||||
func GetSliderTicket(raw, id string) (string, error) {
|
func GetSliderTicket(raw, id string) (string, error) {
|
||||||
var rsp string
|
var rsp string
|
||||||
if err := gout.POST("https://api.shkong.com/gocqhttpapi/task").SetJSON(gout.H{
|
if err := gout.POST("https://api.shkong.com/gocqhttpapi/task").SetJSON(gout.H{
|
||||||
@ -79,6 +281,7 @@ func GetSliderTicket(raw, id string) (string, error) {
|
|||||||
return g.Get("ticket").Str, nil
|
return g.Get("ticket").Str, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// QQMusicSongInfo 通过给定id在QQ音乐上查找曲目信息
|
||||||
func QQMusicSongInfo(id string) (gjson.Result, error) {
|
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}}`)
|
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 {
|
if err != nil {
|
||||||
@ -87,6 +290,7 @@ func QQMusicSongInfo(id string) (gjson.Result, error) {
|
|||||||
return gjson.ParseBytes(d).Get("songinfo.data"), nil
|
return gjson.ParseBytes(d).Get("songinfo.data"), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NeteaseMusicSongInfo 通过给定id在wdd音乐上查找曲目信息
|
||||||
func NeteaseMusicSongInfo(id string) (gjson.Result, error) {
|
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))
|
d, err := GetBytes(fmt.Sprintf("http://music.163.com/api/song/detail/?id=%s&ids=%%5B%s%%5D", id, id))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -21,6 +21,15 @@ var falseSet = map[string]struct{}{
|
|||||||
"0": {},
|
"0": {},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EnsureBool 判断给定的p是否可表示为合法Bool类型,否则返回defaultVal
|
||||||
|
//
|
||||||
|
// 支持的合法类型有
|
||||||
|
//
|
||||||
|
// type bool
|
||||||
|
//
|
||||||
|
// type gjson.True or gjson.False
|
||||||
|
//
|
||||||
|
// type string "true","yes","1" or "false","no","0" (case insensitive)
|
||||||
func EnsureBool(p interface{}, defaultVal bool) bool {
|
func EnsureBool(p interface{}, defaultVal bool) bool {
|
||||||
var str string
|
var str string
|
||||||
if b, ok := p.(bool); ok {
|
if b, ok := p.(bool); ok {
|
||||||
@ -54,9 +63,13 @@ func EnsureBool(p interface{}, defaultVal bool) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// VersionNameCompare 检查版本名是否需要更新, 仅适用于 go-cqhttp 的版本命名规则
|
// VersionNameCompare 检查版本名是否需要更新, 仅适用于 go-cqhttp 的版本命名规则
|
||||||
|
//
|
||||||
// 例: v0.9.29-fix2 == v0.9.29-fix2 -> false
|
// 例: v0.9.29-fix2 == v0.9.29-fix2 -> false
|
||||||
|
//
|
||||||
// v0.9.29-fix1 < v0.9.29-fix2 -> true
|
// v0.9.29-fix1 < v0.9.29-fix2 -> true
|
||||||
|
//
|
||||||
// v0.9.29-fix2 > v0.9.29-fix1 -> false
|
// v0.9.29-fix2 > v0.9.29-fix1 -> false
|
||||||
|
//
|
||||||
// v0.9.29-fix2 < v0.9.30 -> true
|
// v0.9.29-fix2 < v0.9.30 -> true
|
||||||
func VersionNameCompare(current, remote string) bool {
|
func VersionNameCompare(current, remote string) bool {
|
||||||
sp := regexp.MustCompile(`[0-9]\d*`)
|
sp := regexp.MustCompile(`[0-9]\d*`)
|
||||||
@ -72,8 +85,9 @@ func VersionNameCompare(current, remote string) bool {
|
|||||||
return len(cur) < len(re)
|
return len(cur) < len(re)
|
||||||
}
|
}
|
||||||
|
|
||||||
func SplitUrl(s string) []string {
|
// SplitURL 将给定URL字符串分割为两部分,用于URL预处理防止风控
|
||||||
reg := regexp.MustCompile(`[a-zA-Z0-9][-a-zA-Z0-9]{0,62}(\.[a-zA-Z0-9][-a-zA-Z0-9]{0,62})+\.?`)
|
func SplitURL(s string) []string {
|
||||||
|
reg := regexp.MustCompile(`(?i)[a-z\d][-a-z\d]{0,62}(\.[a-z\d][-a-z\d]{0,62})+\.?`)
|
||||||
idx := reg.FindAllStringIndex(s, -1)
|
idx := reg.FindAllStringIndex(s, -1)
|
||||||
if len(idx) == 0 {
|
if len(idx) == 0 {
|
||||||
return []string{s}
|
return []string{s}
|
||||||
|
@ -9,13 +9,15 @@ import (
|
|||||||
var limiter *rate.Limiter
|
var limiter *rate.Limiter
|
||||||
var limitEnable = false
|
var limitEnable = false
|
||||||
|
|
||||||
|
// RateLimit 执行API调用速率限制
|
||||||
func RateLimit(ctx context.Context) {
|
func RateLimit(ctx context.Context) {
|
||||||
if limitEnable {
|
if limitEnable {
|
||||||
_ = limiter.Wait(ctx)
|
_ = limiter.Wait(ctx)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func InitLimiter(r float64, b int) {
|
// InitLimiter 初始化速率限制器
|
||||||
|
func InitLimiter(frequency float64, bucketSize int) {
|
||||||
limitEnable = true
|
limitEnable = true
|
||||||
limiter = rate.NewLimiter(rate.Limit(r), b)
|
limiter = rate.NewLimiter(rate.Limit(frequency), bucketSize)
|
||||||
}
|
}
|
||||||
|
7
global/terminal/double_click.go
Normal file
7
global/terminal/double_click.go
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package terminal
|
||||||
|
|
||||||
|
func RunningByDoubleClick() bool {
|
||||||
|
return false
|
||||||
|
}
|
23
global/terminal/double_click_windows.go
Normal file
23
global/terminal/double_click_windows.go
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
// +build windows
|
||||||
|
|
||||||
|
package terminal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// RunningByDoubleClick 检查是否通过双击直接运行
|
||||||
|
func RunningByDoubleClick() bool {
|
||||||
|
kernel32 := syscall.NewLazyDLL("kernel32.dll")
|
||||||
|
lp := kernel32.NewProc("GetConsoleProcessList")
|
||||||
|
if lp != nil {
|
||||||
|
var ids [2]uint32
|
||||||
|
var maxCount uint32 = 2
|
||||||
|
ret, _, _ := lp.Call(uintptr(unsafe.Pointer(&ids)), uintptr(maxCount))
|
||||||
|
if ret > 1 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
20
go.mod
20
go.mod
@ -1,30 +1,28 @@
|
|||||||
module github.com/Mrs4s/go-cqhttp
|
module github.com/Mrs4s/go-cqhttp
|
||||||
|
|
||||||
go 1.15
|
go 1.16
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20201225164755-c519c532d0d5
|
github.com/Mrs4s/MiraiGo v0.0.0-20210302163721-39a33e4c4965
|
||||||
github.com/dustin/go-humanize v1.0.0
|
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-contrib/pprof v1.3.0
|
github.com/gin-contrib/pprof v1.3.0
|
||||||
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.4
|
github.com/guonaihong/gout v0.1.4
|
||||||
github.com/hjson/hjson-go v3.1.0+incompatible
|
github.com/hjson/hjson-go v3.1.0+incompatible
|
||||||
github.com/json-iterator/go v1.1.10
|
github.com/json-iterator/go v1.1.10
|
||||||
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0 // indirect
|
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0
|
||||||
github.com/kr/binarydist v0.1.0 // indirect
|
|
||||||
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible
|
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible
|
||||||
github.com/lestrrat-go/strftime v1.0.3 // indirect
|
github.com/lestrrat-go/strftime v1.0.4 // indirect
|
||||||
github.com/pkg/errors v0.9.1
|
github.com/pkg/errors v0.9.1
|
||||||
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5
|
|
||||||
github.com/sirupsen/logrus v1.7.0
|
github.com/sirupsen/logrus v1.7.0
|
||||||
github.com/syndtr/goleveldb v1.0.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.7
|
github.com/tidwall/gjson v1.6.8
|
||||||
github.com/wdvxdr1123/go-silk v0.0.0-20201210140933-bcdbcb2f1093
|
github.com/wdvxdr1123/go-silk v0.0.0-20210207032612-169bbdf8861d
|
||||||
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/crypto v0.0.0-20201221181555-eec23a3978ad
|
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9
|
||||||
|
golang.org/x/net v0.0.0-20201021035429-f5854403a974 // indirect
|
||||||
|
golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf
|
||||||
golang.org/x/time v0.0.0-20201208040808-7e3f01d25324
|
golang.org/x/time v0.0.0-20201208040808-7e3f01d25324
|
||||||
)
|
)
|
||||||
|
74
go.sum
74
go.sum
@ -1,8 +1,7 @@
|
|||||||
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-20201225164755-c519c532d0d5 h1:kvaZXnhjdOtRip6xlXu9hEyJhPzKs8ylrdE2eb+N4NQ=
|
github.com/Mrs4s/MiraiGo v0.0.0-20210302163721-39a33e4c4965 h1:zrm7q2Wnj82pqdbaF72es7SYm2aV4dlrwfvVcNoR4hc=
|
||||||
github.com/Mrs4s/MiraiGo v0.0.0-20201225164755-c519c532d0d5/go.mod h1:7brUNAmygY22+PDCUiVT4MLeyvGHDBjW9f+67DKeHTw=
|
github.com/Mrs4s/MiraiGo v0.0.0-20210302163721-39a33e4c4965/go.mod h1:yhqA0NyKxUf7I/0HR/1OMchveFggX8wde04gqdGrNfU=
|
||||||
github.com/a8m/syncmap v0.0.0-20200818084611-4bbbd178de97/go.mod h1:f3iF7/3t9i9hsYF8DPgT0XeIVyNzevhMCKf2445Q6pE=
|
|
||||||
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=
|
||||||
@ -12,20 +11,6 @@ github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25Kn
|
|||||||
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/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/pprof v1.3.0 h1:G9eK6HnbkSqDZBYbzG4wrjCsA4e+cvYAHUZw6W+W9K0=
|
github.com/gin-contrib/pprof v1.3.0 h1:G9eK6HnbkSqDZBYbzG4wrjCsA4e+cvYAHUZw6W+W9K0=
|
||||||
github.com/gin-contrib/pprof v1.3.0/go.mod h1:waMjT1H9b179t3CxuG1cV3DHpga6ybizwfBaM5OXaB0=
|
github.com/gin-contrib/pprof v1.3.0/go.mod h1:waMjT1H9b179t3CxuG1cV3DHpga6ybizwfBaM5OXaB0=
|
||||||
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
||||||
@ -41,8 +26,6 @@ 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=
|
||||||
@ -78,15 +61,13 @@ github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/
|
|||||||
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0 h1:iQTw/8FWTuc7uiaSepXwyf3o52HaUYcV+Tu66S3F5GA=
|
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/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/kr/binarydist v0.1.0 h1:6kAoLA9FMMnNGSehX0s1PdjbEaACznAv/W219j2uvyo=
|
|
||||||
github.com/kr/binarydist v0.1.0/go.mod h1:DY7S//GCoz1BCd0B0EVrinCKAZN3pXe+MDaIZbXQVgM=
|
|
||||||
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.4.0+incompatible h1:Y6sqxHMyB1D2YSzWkLibYKgg+SwmyFU9dF2hn6MdTj4=
|
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible h1:Y6sqxHMyB1D2YSzWkLibYKgg+SwmyFU9dF2hn6MdTj4=
|
||||||
github.com/lestrrat-go/file-rotatelogs v2.4.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.3 h1:qqOPU7y+TM8Y803I8fG9c/DyKG3xH/xkng6keC1015Q=
|
github.com/lestrrat-go/strftime v1.0.4 h1:T1Rb9EPkAhgxKqbcMIPguPq8glqXTA1koF8n9BHElA8=
|
||||||
github.com/lestrrat-go/strftime v1.0.3/go.mod h1:E1nN3pCbtMSu1yjSVeyuRFVm/U0xoR76fd03sz+Qz4g=
|
github.com/lestrrat-go/strftime v1.0.4/go.mod h1:E1nN3pCbtMSu1yjSVeyuRFVm/U0xoR76fd03sz+Qz4g=
|
||||||
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 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc=
|
||||||
@ -98,15 +79,13 @@ github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S
|
|||||||
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
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/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/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/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 h1:OdAsTTz6OkFY5QxjkYwrChwuRruF69c169dPK26NUlk=
|
||||||
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5/go.mod h1:GEXHk5HgEKCvEIIrSpFI3ozzG5xOKA2DVlEX/gGnewM=
|
github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
|
||||||
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.7.0 h1:ShrD1U9pZB12TX0cVy0DtePoCH97K8EtX+mg7ZARUtM=
|
github.com/sirupsen/logrus v1.7.0 h1:ShrD1U9pZB12TX0cVy0DtePoCH97K8EtX+mg7ZARUtM=
|
||||||
github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
|
github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
|
||||||
@ -115,14 +94,13 @@ github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+
|
|||||||
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/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE=
|
github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE=
|
||||||
github.com/syndtr/goleveldb v1.0.0/go.mod h1:ZVVdQEZoIme9iO1Ch2Jdy24qqXrMMOU6lpPAyBWyWuQ=
|
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.3/go.mod h1:BaHyNc5bjzYkPqgLq7mdVzeiRtULKULXLgZFKsxEHI0=
|
github.com/tidwall/gjson v1.6.8 h1:CTmXMClGYPAmln7652e69B7OLXfTi5ABcPPwjIWUv7w=
|
||||||
github.com/tidwall/gjson v1.6.7 h1:Mb1M9HZCRWEcXQ8ieJo7auYyyiSux6w9XN3AdTpxJrE=
|
github.com/tidwall/gjson v1.6.8/go.mod h1:zeFuBCIqD4sN/gmqBzZ4j7Jd6UcA2Fc56x7QFsv+8fI=
|
||||||
github.com/tidwall/gjson v1.6.7/go.mod h1:zeFuBCIqD4sN/gmqBzZ4j7Jd6UcA2Fc56x7QFsv+8fI=
|
|
||||||
github.com/tidwall/match v1.0.1/go.mod h1:LujAq0jyVjBy028G1WhWfIzbpQfMO8bBZ6Tyb0+pL9E=
|
|
||||||
github.com/tidwall/match v1.0.3 h1:FQUVvBImDutD8wJLN6c5eMzWtjgONK9MwIBCOrUJKeE=
|
github.com/tidwall/match v1.0.3 h1:FQUVvBImDutD8wJLN6c5eMzWtjgONK9MwIBCOrUJKeE=
|
||||||
github.com/tidwall/match v1.0.3/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM=
|
github.com/tidwall/match v1.0.3/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM=
|
||||||
github.com/tidwall/pretty v1.0.2 h1:Z7S3cePv9Jwm1KwS0513MRaoUe3S01WPbLNV40pwWZU=
|
github.com/tidwall/pretty v1.0.2 h1:Z7S3cePv9Jwm1KwS0513MRaoUe3S01WPbLNV40pwWZU=
|
||||||
@ -131,28 +109,23 @@ 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/wdvxdr1123/go-silk v0.0.0-20201210140933-bcdbcb2f1093 h1:t38EBwI2hFJz1sQJ402VqzdA3eMidkY1c3w/8z0SftA=
|
github.com/wdvxdr1123/go-silk v0.0.0-20210207032612-169bbdf8861d h1:gJTKbjZtlMt/almOeFi/UpVtT3RHqRWscgEuDtnF5TU=
|
||||||
github.com/wdvxdr1123/go-silk v0.0.0-20201210140933-bcdbcb2f1093/go.mod h1:5q9LFlBr+yX/J8Jd/9wHdXwkkjFkNyQIS7kX2Lgx/Zs=
|
github.com/wdvxdr1123/go-silk v0.0.0-20210207032612-169bbdf8861d/go.mod h1:twOxzexmM2Il1ReUu1fB5tnUotOq/dp56xjk/ZHwb1I=
|
||||||
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=
|
||||||
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
|
||||||
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=
|
||||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9 h1:psW17arqaxU48Z5kZ0CQnkZWQJsqcURM6tKiBApRjXI=
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
||||||
golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad h1:DN0cp81fZ3njFcrLCytUHRSUkqBjfTo4Tx9RJTWs0EY=
|
|
||||||
golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I=
|
|
||||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||||
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
|
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
|
||||||
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/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
|
||||||
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-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-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||||
golang.org/x/net v0.0.0-20201021035429-f5854403a974 h1:IX6qOQeG5uLjB/hjjwjedwfjND0hgjPMMyO1RoIXQNI=
|
golang.org/x/net v0.0.0-20201021035429-f5854403a974 h1:IX6qOQeG5uLjB/hjjwjedwfjND0hgjPMMyO1RoIXQNI=
|
||||||
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
|
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
|
||||||
@ -160,7 +133,6 @@ golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAG
|
|||||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/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-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/sync v0.0.0-20201020160332-67f06af15bc9/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-20180909124046-d0be0721c37e/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=
|
||||||
@ -168,10 +140,12 @@ golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7w
|
|||||||
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-20191026070338-33540a1f6037/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/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f h1:+Nyd8tzPX9R7BWHguqsrbFdRx3WQ/1ib8I44HXV5yTA=
|
|
||||||
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/term v0.0.0-20201117132131-f5c789dd3221 h1:/ZHdbVpdR/jk3g30/d4yUL0JU9kksj8+F/bnQUVLGDM=
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw=
|
golang.org/x/sys v0.0.0-20201126233918-771906719818 h1:f1CIuDlJhwANEC2MM87MBEVMr3jl5bifgsfj90XAF9c=
|
||||||
|
golang.org/x/sys v0.0.0-20201126233918-771906719818/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf h1:MZ2shdL+ZM/XzY3ZGOnh4Nlpnxz5GSOhOmtHo3iPU6M=
|
||||||
|
golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
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/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
@ -181,14 +155,8 @@ golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGm
|
|||||||
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=
|
||||||
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||||
golang.org/x/tools v0.0.0-20190501045030-23463209683d/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
|
||||||
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.0.0-20201218024724-ae774e9781d2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
|
||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
||||||
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
||||||
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
|
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
|
||||||
@ -214,5 +182,13 @@ 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=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
|
modernc.org/libc v1.7.6 h1:P0qDJAlSR6hSAuE8mQgz9eH/GzigfEd3IIn7HmTQgT0=
|
||||||
|
modernc.org/libc v1.7.6/go.mod h1:U1eq8YWr/Kc1RWCMFUWEdkTg8OTcfLw2kY8EDwl039w=
|
||||||
|
modernc.org/mathutil v1.1.1/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E=
|
||||||
|
modernc.org/mathutil v1.2.2 h1:+yFk8hBprV+4c0U9GjFtL+dV3N8hOJ8JCituQcMShFY=
|
||||||
|
modernc.org/mathutil v1.2.2/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E=
|
||||||
|
modernc.org/memory v1.0.4 h1:utMBrFcpnQDdNsmM6asmyH/FM9TqLPS7XF7otpJmrwM=
|
||||||
|
modernc.org/memory v1.0.4/go.mod h1:nV2OApxradM3/OVbs2/0OsP6nPfakXpi50C7dcoHXlc=
|
||||||
|
345
main.go
345
main.go
@ -2,9 +2,15 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
|
"crypto/aes"
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
|
"crypto/sha1"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global/terminal"
|
||||||
|
rotatelogs "github.com/lestrrat-go/file-rotatelogs"
|
||||||
|
easy "github.com/t-tomalak/logrus-easy-formatter"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -16,78 +22,72 @@ import (
|
|||||||
"runtime"
|
"runtime"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/Mrs4s/go-cqhttp/server"
|
"github.com/Mrs4s/go-cqhttp/server"
|
||||||
"github.com/guonaihong/gout"
|
"github.com/guonaihong/gout"
|
||||||
"github.com/tidwall/gjson"
|
"github.com/tidwall/gjson"
|
||||||
"golang.org/x/crypto/ssh/terminal"
|
"golang.org/x/crypto/pbkdf2"
|
||||||
|
"golang.org/x/term"
|
||||||
|
|
||||||
"github.com/Mrs4s/MiraiGo/binary"
|
"github.com/Mrs4s/MiraiGo/binary"
|
||||||
"github.com/Mrs4s/MiraiGo/client"
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
"github.com/Mrs4s/go-cqhttp/coolq"
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
"github.com/getlantern/go-update"
|
|
||||||
jsoniter "github.com/json-iterator/go"
|
jsoniter "github.com/json-iterator/go"
|
||||||
rotatelogs "github.com/lestrrat-go/file-rotatelogs"
|
|
||||||
"github.com/rifflock/lfshook"
|
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
easy "github.com/t-tomalak/logrus-easy-formatter"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
var conf *global.JSONConfig
|
||||||
|
var isFastStart = false
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
log.SetFormatter(&easy.Formatter{
|
logFormatter := &easy.Formatter{
|
||||||
TimestampFormat: "2006-01-02 15:04:05",
|
TimestampFormat: "2006-01-02 15:04:05",
|
||||||
LogFormat: "[%time%] [%lvl%]: %msg% \n",
|
LogFormat: "[%time%] [%lvl%]: %msg% \n",
|
||||||
})
|
}
|
||||||
w, err := rotatelogs.New(path.Join("logs", "%Y-%m-%d.log"), rotatelogs.WithRotationTime(time.Hour*24))
|
w, err := rotatelogs.New(path.Join("logs", "%Y-%m-%d.log"), rotatelogs.WithRotationTime(time.Hour*24))
|
||||||
if err == nil {
|
if err != nil {
|
||||||
log.SetOutput(io.MultiWriter(os.Stderr, w))
|
log.Errorf("rotatelogs init err: %v", err)
|
||||||
|
panic(err)
|
||||||
}
|
}
|
||||||
if !global.PathExists(global.IMAGE_PATH) {
|
|
||||||
if err := os.MkdirAll(global.IMAGE_PATH, 0755); err != nil {
|
conf = getConfig()
|
||||||
log.Fatalf("创建图片缓存文件夹失败: %v", err)
|
if conf == nil {
|
||||||
}
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
if !global.PathExists(global.VOICE_PATH) {
|
|
||||||
if err := os.MkdirAll(global.VOICE_PATH, 0755); err != nil {
|
// 在debug模式下,将在标准输出中打印当前执行行数
|
||||||
log.Fatalf("创建语音缓存文件夹失败: %v", err)
|
if conf.Debug {
|
||||||
}
|
log.SetReportCaller(true)
|
||||||
}
|
|
||||||
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)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.AddHook(global.NewLocalHook(w, logFormatter, global.GetLogLevel(conf.LogLevel)...))
|
||||||
|
|
||||||
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.hjson 文件.")
|
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 {
|
||||||
log.Fatalf("读取文件 cqhttp.json 失败: %v", err)
|
log.Fatalf("读取文件 cqhttp.json 失败: %v", err)
|
||||||
}
|
}
|
||||||
goConf := global.DefaultConfig()
|
goConf := global.DefaultConfig()
|
||||||
goConf.AccessToken = conf.AccessToken
|
goConf.AccessToken = conf.AccessToken
|
||||||
goConf.HttpConfig.Host = conf.Host
|
goConf.HTTPConfig.Host = conf.Host
|
||||||
goConf.HttpConfig.Port = conf.Port
|
goConf.HTTPConfig.Port = conf.Port
|
||||||
goConf.WSConfig.Host = conf.WSHost
|
goConf.WSConfig.Host = conf.WSHost
|
||||||
goConf.WSConfig.Port = conf.WSPort
|
goConf.WSConfig.Port = conf.WSPort
|
||||||
if conf.PostUrl != "" {
|
if conf.PostURL != "" {
|
||||||
goConf.HttpConfig.PostUrls[conf.PostUrl] = conf.Secret
|
goConf.HTTPConfig.PostUrls[conf.PostURL] = conf.Secret
|
||||||
}
|
}
|
||||||
if conf.UseWsReverse {
|
if conf.UseWsReverse {
|
||||||
goConf.ReverseServers[0].Enabled = true
|
goConf.ReverseServers[0].Enabled = true
|
||||||
goConf.ReverseServers[0].ReverseUrl = conf.WSReverseUrl
|
goConf.ReverseServers[0].ReverseURL = conf.WSReverseURL
|
||||||
goConf.ReverseServers[0].ReverseApiUrl = conf.WSReverseApiUrl
|
goConf.ReverseServers[0].ReverseAPIURL = conf.WSReverseAPIURL
|
||||||
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.hjson"); err != nil {
|
if err := goConf.Save("config.hjson"); err != nil {
|
||||||
@ -95,11 +95,36 @@ func init() {
|
|||||||
}
|
}
|
||||||
_ = os.Remove("cqhttp.json")
|
_ = os.Remove("cqhttp.json")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if !global.PathExists(global.ImagePath) {
|
||||||
|
if err := os.MkdirAll(global.ImagePath, 0755); err != nil {
|
||||||
|
log.Fatalf("创建图片缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !global.PathExists(global.VoicePath) {
|
||||||
|
if err := os.MkdirAll(global.VoicePath, 0755); err != nil {
|
||||||
|
log.Fatalf("创建语音缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !global.PathExists(global.VideoPath) {
|
||||||
|
if err := os.MkdirAll(global.VideoPath, 0755); err != nil {
|
||||||
|
log.Fatalf("创建视频缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !global.PathExists(global.CachePath) {
|
||||||
|
if err := os.MkdirAll(global.CachePath, 0755); err != nil {
|
||||||
|
log.Fatalf("创建发送图片缓存文件夹失败: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
if terminal.RunningByDoubleClick() {
|
||||||
|
log.Warning("警告: 强烈不推荐通过双击直接运行本程序, 这将导致一些非预料的后果.")
|
||||||
|
log.Warning("将等待10s后启动")
|
||||||
|
time.Sleep(time.Second * 10)
|
||||||
|
}
|
||||||
var byteKey []byte
|
var byteKey []byte
|
||||||
var isFastStart = false
|
|
||||||
arg := os.Args
|
arg := os.Args
|
||||||
if len(arg) > 1 {
|
if len(arg) > 1 {
|
||||||
for i := range arg {
|
for i := range arg {
|
||||||
@ -121,91 +146,12 @@ func main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var conf *global.JsonConfig
|
|
||||||
if global.PathExists("config.json") {
|
|
||||||
conf = global.Load("config.json")
|
|
||||||
_ = conf.Save("config.hjson")
|
|
||||||
_ = os.Remove("config.json")
|
|
||||||
} else if os.Getenv("UIN") != "" {
|
|
||||||
log.Infof("将从环境变量加载配置.")
|
|
||||||
uin, _ := strconv.ParseInt(os.Getenv("UIN"), 10, 64)
|
|
||||||
pwd := os.Getenv("PASS")
|
|
||||||
post := os.Getenv("HTTP_POST")
|
|
||||||
conf = &global.JsonConfig{
|
|
||||||
Uin: uin,
|
|
||||||
Password: pwd,
|
|
||||||
HttpConfig: &global.GoCQHttpConfig{
|
|
||||||
Enabled: true,
|
|
||||||
Host: "0.0.0.0",
|
|
||||||
Port: 5700,
|
|
||||||
PostUrls: map[string]string{},
|
|
||||||
},
|
|
||||||
WSConfig: &global.GoCQWebsocketConfig{
|
|
||||||
Enabled: true,
|
|
||||||
Host: "0.0.0.0",
|
|
||||||
Port: 6700,
|
|
||||||
},
|
|
||||||
PostMessageFormat: "string",
|
|
||||||
Debug: os.Getenv("DEBUG") == "true",
|
|
||||||
}
|
|
||||||
if post != "" {
|
|
||||||
conf.HttpConfig.PostUrls[post] = os.Getenv("HTTP_SECRET")
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
conf = global.Load("config.hjson")
|
|
||||||
}
|
|
||||||
if conf == nil {
|
|
||||||
err := global.WriteAllText("config.hjson", global.DefaultConfigWithComments)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("创建默认配置文件时出现错误: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
log.Infof("默认配置文件已生成, 请编辑 config.hjson 后重启程序.")
|
|
||||||
time.Sleep(time.Second * 5)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if conf.Uin == 0 || (conf.Password == "" && conf.PasswordEncrypted == "") {
|
if conf.Uin == 0 || (conf.Password == "" && conf.PasswordEncrypted == "") {
|
||||||
log.Warnf("请修改 config.hjson 以添加账号密码.")
|
log.Warnf("请修改 config.hjson 以添加账号密码.")
|
||||||
time.Sleep(time.Second * 5)
|
if !isFastStart {
|
||||||
return
|
time.Sleep(time.Second * 5)
|
||||||
}
|
|
||||||
|
|
||||||
// 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"),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
return
|
||||||
log.AddHook(lfshook.NewHook(
|
|
||||||
logPathMap,
|
|
||||||
&easy.Formatter{
|
|
||||||
TimestampFormat: "2006-01-02 15:04:05",
|
|
||||||
LogFormat: "[%time%] [%lvl%]: %msg% \n",
|
|
||||||
},
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Info("当前版本:", coolq.Version)
|
log.Info("当前版本:", coolq.Version)
|
||||||
@ -214,7 +160,7 @@ func main() {
|
|||||||
log.Warnf("已开启Debug模式.")
|
log.Warnf("已开启Debug模式.")
|
||||||
log.Debugf("开发交流群: 192548878")
|
log.Debugf("开发交流群: 192548878")
|
||||||
server.Debug = true
|
server.Debug = true
|
||||||
if conf.WebUi == nil || !conf.WebUi.Enabled {
|
if conf.WebUI == nil || !conf.WebUI.Enabled {
|
||||||
log.Warnf("警告: 在Debug模式下未启用WebUi服务, 将无法进行性能分析.")
|
log.Warnf("警告: 在Debug模式下未启用WebUi服务, 将无法进行性能分析.")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -232,15 +178,11 @@ func main() {
|
|||||||
}
|
}
|
||||||
if conf.EncryptPassword && conf.PasswordEncrypted == "" {
|
if conf.EncryptPassword && conf.PasswordEncrypted == "" {
|
||||||
log.Infof("密码加密已启用, 请输入Key对密码进行加密: (Enter 提交)")
|
log.Infof("密码加密已启用, 请输入Key对密码进行加密: (Enter 提交)")
|
||||||
byteKey, _ := terminal.ReadPassword(int(os.Stdin.Fd()))
|
byteKey, _ = term.ReadPassword(int(os.Stdin.Fd()))
|
||||||
key := md5.Sum(byteKey)
|
global.PasswordHash = md5.Sum([]byte(conf.Password))
|
||||||
if encrypted := EncryptPwd(conf.Password, key[:]); encrypted != "" {
|
conf.Password = ""
|
||||||
conf.Password = ""
|
conf.PasswordEncrypted = "AES:" + PasswordHashEncrypt(global.PasswordHash[:], byteKey)
|
||||||
conf.PasswordEncrypted = encrypted
|
_ = conf.Save("config.hjson")
|
||||||
_ = conf.Save("config.hjson")
|
|
||||||
} else {
|
|
||||||
log.Warnf("加密时出现问题.")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if conf.PasswordEncrypted != "" {
|
if conf.PasswordEncrypted != "" {
|
||||||
if len(byteKey) == 0 {
|
if len(byteKey) == 0 {
|
||||||
@ -256,13 +198,30 @@ func main() {
|
|||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
byteKey, _ = terminal.ReadPassword(int(os.Stdin.Fd()))
|
byteKey, _ = term.ReadPassword(int(os.Stdin.Fd()))
|
||||||
cancel <- struct{}{}
|
cancel <- struct{}{}
|
||||||
} else {
|
} else {
|
||||||
log.Infof("密码加密已启用, 使用运行时传递的参数进行解密,按 Ctrl+C 取消.")
|
log.Infof("密码加密已启用, 使用运行时传递的参数进行解密,按 Ctrl+C 取消.")
|
||||||
}
|
}
|
||||||
key := md5.Sum(byteKey)
|
|
||||||
conf.Password = DecryptPwd(conf.PasswordEncrypted, key[:])
|
// 升级客户端密码加密方案,MD5+TEA 加密密码 -> PBKDF2+AES 加密 MD5
|
||||||
|
// 升级后的 PasswordEncrypted 字符串以"AES:"开始,其后为 Hex 编码的16字节加密 MD5
|
||||||
|
if !strings.HasPrefix(conf.PasswordEncrypted, "AES:") {
|
||||||
|
password := OldPasswordDecrypt(conf.PasswordEncrypted, byteKey)
|
||||||
|
passwordHash := md5.Sum([]byte(password))
|
||||||
|
newPasswordHash := PasswordHashEncrypt(passwordHash[:], byteKey)
|
||||||
|
conf.PasswordEncrypted = "AES:" + newPasswordHash
|
||||||
|
_ = conf.Save("config.hjson")
|
||||||
|
log.Debug("密码加密方案升级完成")
|
||||||
|
}
|
||||||
|
|
||||||
|
ph, err := PasswordHashDecrypt(conf.PasswordEncrypted[4:], byteKey)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("加密存储的密码损坏,请尝试重新配置密码")
|
||||||
|
}
|
||||||
|
copy(global.PasswordHash[:], ph)
|
||||||
|
} else {
|
||||||
|
global.PasswordHash = md5.Sum([]byte(conf.Password))
|
||||||
}
|
}
|
||||||
if !isFastStart {
|
if !isFastStart {
|
||||||
log.Info("Bot将在5秒后登录并开始信息处理, 按 Ctrl+C 取消.")
|
log.Info("Bot将在5秒后登录并开始信息处理, 按 Ctrl+C 取消.")
|
||||||
@ -282,7 +241,7 @@ func main() {
|
|||||||
}
|
}
|
||||||
return "未知"
|
return "未知"
|
||||||
}())
|
}())
|
||||||
cli := client.NewClient(conf.Uin, conf.Password)
|
cli := client.NewClientMd5(conf.Uin, global.PasswordHash)
|
||||||
cli.OnLog(func(c *client.QQClient, e *client.LogEvent) {
|
cli.OnLog(func(c *client.QQClient, e *client.LogEvent) {
|
||||||
switch e.Type {
|
switch e.Type {
|
||||||
case "INFO":
|
case "INFO":
|
||||||
@ -309,26 +268,26 @@ func main() {
|
|||||||
log.Infof("收到服务器地址更新通知, 将在下一次重连时应用. ")
|
log.Infof("收到服务器地址更新通知, 将在下一次重连时应用. ")
|
||||||
return true
|
return true
|
||||||
})
|
})
|
||||||
if conf.WebUi == nil {
|
if conf.WebUI == nil {
|
||||||
conf.WebUi = &global.GoCqWebUi{
|
conf.WebUI = &global.GoCQWebUI{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
WebInput: false,
|
WebInput: false,
|
||||||
Host: "0.0.0.0",
|
Host: "0.0.0.0",
|
||||||
WebUiPort: 9999,
|
WebUIPort: 9999,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if conf.WebUi.WebUiPort <= 0 {
|
if conf.WebUI.WebUIPort <= 0 {
|
||||||
conf.WebUi.WebUiPort = 9999
|
conf.WebUI.WebUIPort = 9999
|
||||||
}
|
}
|
||||||
if conf.WebUi.Host == "" {
|
if conf.WebUI.Host == "" {
|
||||||
conf.WebUi.Host = "127.0.0.1"
|
conf.WebUI.Host = "127.0.0.1"
|
||||||
}
|
}
|
||||||
global.Proxy = conf.ProxyRewrite
|
global.Proxy = conf.ProxyRewrite
|
||||||
b := server.WebServer.Run(fmt.Sprintf("%s:%d", conf.WebUi.Host, conf.WebUi.WebUiPort), cli)
|
b := server.WebServer.Run(fmt.Sprintf("%s:%d", conf.WebUI.Host, conf.WebUI.WebUIPort), cli)
|
||||||
c := server.Console
|
c := server.Console
|
||||||
r := server.Restart
|
r := server.Restart
|
||||||
go checkUpdate()
|
go checkUpdate()
|
||||||
signal.Notify(c, os.Interrupt, os.Kill)
|
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
|
||||||
select {
|
select {
|
||||||
case <-c:
|
case <-c:
|
||||||
b.Release()
|
b.Release()
|
||||||
@ -339,25 +298,50 @@ func main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func EncryptPwd(pwd string, key []byte) string {
|
// PasswordHashEncrypt 使用key加密给定passwordHash
|
||||||
tea := binary.NewTeaCipher(key)
|
func PasswordHashEncrypt(passwordHash []byte, key []byte) string {
|
||||||
if tea == nil {
|
if len(passwordHash) != 16 {
|
||||||
return ""
|
panic("密码加密参数错误")
|
||||||
}
|
}
|
||||||
return base64.StdEncoding.EncodeToString(tea.Encrypt([]byte(pwd)))
|
|
||||||
|
key = pbkdf2.Key(key, key, 114514, 32, sha1.New)
|
||||||
|
|
||||||
|
cipher, _ := aes.NewCipher(key)
|
||||||
|
result := make([]byte, 16)
|
||||||
|
cipher.Encrypt(result, passwordHash)
|
||||||
|
|
||||||
|
return hex.EncodeToString(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
func DecryptPwd(ePwd string, key []byte) string {
|
// PasswordHashDecrypt 使用key解密给定passwordHash
|
||||||
|
func PasswordHashDecrypt(encryptedPasswordHash string, key []byte) ([]byte, error) {
|
||||||
|
ciphertext, err := hex.DecodeString(encryptedPasswordHash)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
key = pbkdf2.Key(key, key, 114514, 32, sha1.New)
|
||||||
|
|
||||||
|
cipher, _ := aes.NewCipher(key)
|
||||||
|
result := make([]byte, 16)
|
||||||
|
cipher.Decrypt(result, ciphertext)
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// OldPasswordDecrypt 使用key解密老password,仅供兼容使用
|
||||||
|
func OldPasswordDecrypt(encryptedPassword string, key []byte) string {
|
||||||
defer func() {
|
defer func() {
|
||||||
if pan := recover(); pan != nil {
|
if pan := recover(); pan != nil {
|
||||||
log.Fatalf("密码解密失败: %v", pan)
|
log.Fatalf("密码解密失败: %v", pan)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
encrypted, err := base64.StdEncoding.DecodeString(ePwd)
|
encKey := md5.Sum(key)
|
||||||
|
encrypted, err := base64.StdEncoding.DecodeString(encryptedPassword)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
tea := binary.NewTeaCipher(key)
|
tea := binary.NewTeaCipher(encKey[:])
|
||||||
if tea == nil {
|
if tea == nil {
|
||||||
panic("密钥错误")
|
panic("密钥错误")
|
||||||
}
|
}
|
||||||
@ -388,7 +372,7 @@ func checkUpdate() {
|
|||||||
log.Infof("检查更新完成. 当前已运行最新版本.")
|
log.Infof("检查更新完成. 当前已运行最新版本.")
|
||||||
}
|
}
|
||||||
|
|
||||||
func selfUpdate(imageUrl string) {
|
func selfUpdate(imageURL string) {
|
||||||
console := bufio.NewReader(os.Stdin)
|
console := bufio.NewReader(os.Stdin)
|
||||||
readLine := func() (str string) {
|
readLine := func() (str string) {
|
||||||
str, _ = console.ReadString('\n')
|
str, _ = console.ReadString('\n')
|
||||||
@ -416,8 +400,8 @@ func selfUpdate(imageUrl string) {
|
|||||||
url := fmt.Sprintf(
|
url := fmt.Sprintf(
|
||||||
"%v/Mrs4s/go-cqhttp/releases/download/%v/go-cqhttp-%v-%v-%v",
|
"%v/Mrs4s/go-cqhttp/releases/download/%v/go-cqhttp-%v-%v-%v",
|
||||||
func() string {
|
func() string {
|
||||||
if imageUrl != "" {
|
if imageURL != "" {
|
||||||
return imageUrl
|
return imageURL
|
||||||
}
|
}
|
||||||
return "https://github.com"
|
return "https://github.com"
|
||||||
}(),
|
}(),
|
||||||
@ -436,7 +420,7 @@ func selfUpdate(imageUrl string) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
wc := global.WriteCounter{}
|
wc := global.WriteCounter{}
|
||||||
err, _ = update.New().FromStream(io.TeeReader(resp.Body, &wc))
|
err, _ = global.UpdateFromStream(io.TeeReader(resp.Body, &wc))
|
||||||
fmt.Println()
|
fmt.Println()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("更新失败!")
|
log.Error("更新失败!")
|
||||||
@ -450,6 +434,8 @@ func selfUpdate(imageUrl string) {
|
|||||||
} else {
|
} else {
|
||||||
log.Warn("已取消更新!")
|
log.Warn("已取消更新!")
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
log.Info("当前版本已经是最新版本!")
|
||||||
}
|
}
|
||||||
log.Info("按 Enter 继续....")
|
log.Info("按 Enter 继续....")
|
||||||
readLine()
|
readLine()
|
||||||
@ -457,7 +443,7 @@ func selfUpdate(imageUrl string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func restart(Args []string) {
|
func restart(Args []string) {
|
||||||
cmd := &exec.Cmd{}
|
var cmd *exec.Cmd
|
||||||
if runtime.GOOS == "windows" {
|
if runtime.GOOS == "windows" {
|
||||||
file, err := exec.LookPath(Args[0])
|
file, err := exec.LookPath(Args[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -484,5 +470,54 @@ func restart(Args []string) {
|
|||||||
Stdout: os.Stdout,
|
Stdout: os.Stdout,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
cmd.Start()
|
_ = cmd.Start()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getConfig() *global.JSONConfig {
|
||||||
|
var conf *global.JSONConfig
|
||||||
|
if global.PathExists("config.json") {
|
||||||
|
conf = global.LoadConfig("config.json")
|
||||||
|
_ = conf.Save("config.hjson")
|
||||||
|
_ = os.Remove("config.json")
|
||||||
|
} else if os.Getenv("UIN") != "" {
|
||||||
|
log.Infof("将从环境变量加载配置.")
|
||||||
|
uin, _ := strconv.ParseInt(os.Getenv("UIN"), 10, 64)
|
||||||
|
pwd := os.Getenv("PASS")
|
||||||
|
post := os.Getenv("HTTP_POST")
|
||||||
|
conf = &global.JSONConfig{
|
||||||
|
Uin: uin,
|
||||||
|
Password: pwd,
|
||||||
|
HTTPConfig: &global.GoCQHTTPConfig{
|
||||||
|
Enabled: true,
|
||||||
|
Host: "0.0.0.0",
|
||||||
|
Port: 5700,
|
||||||
|
PostUrls: map[string]string{},
|
||||||
|
},
|
||||||
|
WSConfig: &global.GoCQWebSocketConfig{
|
||||||
|
Enabled: true,
|
||||||
|
Host: "0.0.0.0",
|
||||||
|
Port: 6700,
|
||||||
|
},
|
||||||
|
PostMessageFormat: "string",
|
||||||
|
Debug: os.Getenv("DEBUG") == "true",
|
||||||
|
}
|
||||||
|
if post != "" {
|
||||||
|
conf.HTTPConfig.PostUrls[post] = os.Getenv("HTTP_SECRET")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
conf = global.LoadConfig("config.hjson")
|
||||||
|
}
|
||||||
|
if conf == nil {
|
||||||
|
err := global.WriteAllText("config.hjson", global.DefaultConfigWithComments)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("创建默认配置文件时出现错误: %v", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
log.Infof("默认配置文件已生成, 请编辑 config.hjson 后重启程序.")
|
||||||
|
if !isFastStart {
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return conf
|
||||||
}
|
}
|
||||||
|
382
server/api.go
Normal file
382
server/api.go
Normal file
@ -0,0 +1,382 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
|
)
|
||||||
|
|
||||||
|
type resultGetter interface {
|
||||||
|
Get(string) gjson.Result
|
||||||
|
}
|
||||||
|
|
||||||
|
type apiCaller struct {
|
||||||
|
bot *coolq.CQBot
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLoginInfo(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetLoginInfo()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getFriendList(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetFriendList()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupList(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupList(p.Get("no_cache").Bool())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupInfo(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupInfo(p.Get("group_id").Int(), p.Get("no_cache").Bool())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupMemberList(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupMemberList(p.Get("group_id").Int(), p.Get("no_cache").Bool())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupMemberInfo(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupMemberInfo(
|
||||||
|
p.Get("group_id").Int(), p.Get("user_id").Int(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
func sendMSG(bot *coolq.CQBot, p resultGetter) 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 {
|
||||||
|
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"), autoEscape)
|
||||||
|
}
|
||||||
|
if p.Get("user_id").Int() != 0 {
|
||||||
|
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"), autoEscape)
|
||||||
|
}
|
||||||
|
return coolq.MSG{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func sendGroupMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"),
|
||||||
|
global.EnsureBool(p.Get("auto_escape"), false))
|
||||||
|
}
|
||||||
|
|
||||||
|
func sendGroupForwardMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSendGroupForwardMessage(p.Get("group_id").Int(), p.Get("messages"))
|
||||||
|
}
|
||||||
|
|
||||||
|
func sendPrivateMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"),
|
||||||
|
global.EnsureBool(p.Get("auto_escape"), false))
|
||||||
|
}
|
||||||
|
|
||||||
|
func deleteMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQDeleteMessage(int32(p.Get("message_id").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func setFriendAddRequest(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
apr := true
|
||||||
|
if p.Get("approve").Exists() {
|
||||||
|
apr = p.Get("approve").Bool()
|
||||||
|
}
|
||||||
|
return bot.CQProcessFriendRequest(p.Get("flag").Str, apr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupAddRequest(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
subType := p.Get("sub_type").Str
|
||||||
|
apr := true
|
||||||
|
if subType == "" {
|
||||||
|
subType = p.Get("type").Str
|
||||||
|
}
|
||||||
|
if p.Get("approve").Exists() {
|
||||||
|
apr = p.Get("approve").Bool()
|
||||||
|
}
|
||||||
|
return bot.CQProcessGroupRequest(p.Get("flag").Str, subType, p.Get("reason").Str, apr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupCard(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupCard(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("card").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupSpecialTitle(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupSpecialTitle(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("special_title").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupKick(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupKick(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("message").Str, p.Get("reject_add_request").Bool())
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupBan(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupBan(p.Get("group_id").Int(), p.Get("user_id").Int(), func() uint32 {
|
||||||
|
if p.Get("duration").Exists() {
|
||||||
|
return uint32(p.Get("duration").Int())
|
||||||
|
}
|
||||||
|
return 1800
|
||||||
|
}())
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupWholeBan(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupWholeBan(p.Get("group_id").Int(), func() bool {
|
||||||
|
if p.Get("enable").Exists() {
|
||||||
|
return p.Get("enable").Bool()
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}())
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupName(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupName(p.Get("group_id").Int(), p.Get("group_name").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupAdmin(bot *coolq.CQBot, p resultGetter) 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
|
||||||
|
}())
|
||||||
|
}
|
||||||
|
|
||||||
|
func sendGroupNotice(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupMemo(p.Get("group_id").Int(), p.Get("content").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupLeave(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupLeave(p.Get("group_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getImage(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetImage(p.Get("file").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getForwardMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
id := p.Get("message_id").Str
|
||||||
|
if id == "" {
|
||||||
|
id = p.Get("id").Str
|
||||||
|
}
|
||||||
|
return bot.CQGetForwardMessage(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetMessage(int32(p.Get("message_id").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func downloadFile(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
headers := map[string]string{}
|
||||||
|
headersToken := p.Get("headers")
|
||||||
|
if headersToken.IsArray() {
|
||||||
|
for _, sub := range headersToken.Array() {
|
||||||
|
str := strings.SplitN(sub.String(), "=", 2)
|
||||||
|
if len(str) == 2 {
|
||||||
|
headers[str[0]] = str[1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if headersToken.Type == gjson.String {
|
||||||
|
lines := strings.Split(headersToken.String(), "\r\n")
|
||||||
|
for _, sub := range lines {
|
||||||
|
str := strings.SplitN(sub, "=", 2)
|
||||||
|
if len(str) == 2 {
|
||||||
|
headers[str[0]] = str[1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return bot.CQDownloadFile(p.Get("url").Str, headers, int(p.Get("thread_count").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupHonorInfo(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupHonorInfo(p.Get("group_id").Int(), p.Get("type").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setRestart(_ *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
var delay int64
|
||||||
|
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"}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func canSendImage(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQCanSendImage()
|
||||||
|
}
|
||||||
|
|
||||||
|
func canSendRecord(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQCanSendRecord()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getStrangerInfo(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetStrangerInfo(p.Get("user_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getStatus(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetStatus()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getVersionInfo(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetVersionInfo()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupSystemMSG(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupSystemMessages()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupFileSystemInfo(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupFileSystemInfo(p.Get("group_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupRootFiles(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupRootFiles(p.Get("group_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupFilesByFolder(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupFilesByFolderID(p.Get("group_id").Int(), p.Get("folder_id").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupFileURL(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupFileURL(p.Get("group_id").Int(), p.Get("file_id").Str, int32(p.Get("busid").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func uploadGroupFile(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQUploadGroupFile(p.Get("group_id").Int(), p.Get("file").Str, p.Get("name").Str, p.Get("folder").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupMsgHistory(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetGroupMessageHistory(p.Get("group_id").Int(), p.Get("message_seq").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getVipInfo(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetVipInfo(p.Get("user_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func reloadEventFilter(bot *coolq.CQBot, _ resultGetter) coolq.MSG {
|
||||||
|
return bot.CQReloadEventFilter()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGroupAtAllRemain(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetAtAllRemain(p.Get("group_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func ocrImage(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQOcrImage(p.Get("image").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getOnlineClients(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetOnlineClients(p.Get("no_cache").Bool())
|
||||||
|
}
|
||||||
|
|
||||||
|
func getWordSlices(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetWordSlices(p.Get("content").Str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupPortrait(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetGroupPortrait(p.Get("group_id").Int(), p.Get("file").String(), p.Get("cache").String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func setEssenceMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQSetEssenceMessage(int32(p.Get("message_id").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func deleteEssenceMSG(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQDeleteEssenceMessage(int32(p.Get("message_id").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func getEssenceMsgList(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQGetEssenceMessageList(p.Get("group_id").Int())
|
||||||
|
}
|
||||||
|
|
||||||
|
func checkUrlSafely(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQCheckURLSafely(p.Get("url").String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func setGroupAnonymousBan(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
obj := p.Get("anonymous")
|
||||||
|
flag := p.Get("anonymous_flag")
|
||||||
|
if !flag.Exists() {
|
||||||
|
flag = p.Get("flag")
|
||||||
|
}
|
||||||
|
if !flag.Exists() && !obj.Exists() {
|
||||||
|
return coolq.Failed(100, "FLAG_NOT_FOUND", "flag未找到")
|
||||||
|
}
|
||||||
|
if !flag.Exists() {
|
||||||
|
flag = obj.Get("flag")
|
||||||
|
}
|
||||||
|
return bot.CQSetGroupAnonymousBan(p.Get("group_id").Int(), flag.String(), int32(p.Get("duration").Int()))
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleQuickOperation(bot *coolq.CQBot, p resultGetter) coolq.MSG {
|
||||||
|
return bot.CQHandleQuickOperation(p.Get("context"), p.Get("operation"))
|
||||||
|
}
|
||||||
|
|
||||||
|
var API = map[string]func(*coolq.CQBot, resultGetter) coolq.MSG{
|
||||||
|
"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,
|
||||||
|
"send_msg": sendMSG,
|
||||||
|
"send_group_msg": sendGroupMSG,
|
||||||
|
"send_group_forward_msg": sendGroupForwardMSG,
|
||||||
|
"send_private_msg": sendPrivateMSG,
|
||||||
|
"delete_msg": deleteMSG,
|
||||||
|
"set_friend_add_request": setFriendAddRequest,
|
||||||
|
"set_group_add_request": setGroupAddRequest,
|
||||||
|
"set_group_card": setGroupCard,
|
||||||
|
"set_group_special_title": setGroupSpecialTitle,
|
||||||
|
"set_group_kick": setGroupKick,
|
||||||
|
"set_group_ban": setGroupBan,
|
||||||
|
"set_group_whole_ban": setGroupWholeBan,
|
||||||
|
"set_group_name": setGroupName,
|
||||||
|
"set_group_admin": setGroupAdmin,
|
||||||
|
"_send_group_notice": sendGroupNotice,
|
||||||
|
"set_group_leave": setGroupLeave,
|
||||||
|
"get_image": getImage,
|
||||||
|
"get_forward_msg": getForwardMSG,
|
||||||
|
"get_msg": getMSG,
|
||||||
|
"download_file": downloadFile,
|
||||||
|
"get_group_honor_info": getGroupHonorInfo,
|
||||||
|
"set_restart": setRestart,
|
||||||
|
"can_send_image": canSendImage,
|
||||||
|
"can_send_record": canSendRecord,
|
||||||
|
"get_stranger_info": getStrangerInfo,
|
||||||
|
"get_status": getStatus,
|
||||||
|
"get_version_info": getVersionInfo,
|
||||||
|
"get_group_system_msg": getGroupSystemMSG,
|
||||||
|
"get_group_file_system_info": getGroupFileSystemInfo,
|
||||||
|
"get_group_root_files": getGroupRootFiles,
|
||||||
|
"get_group_files_by_folder": getGroupFilesByFolder,
|
||||||
|
"get_group_file_url": getGroupFileURL,
|
||||||
|
"upload_group_file": uploadGroupFile,
|
||||||
|
"get_group_msg_history": getGroupMsgHistory,
|
||||||
|
"_get_vip_info": getVipInfo,
|
||||||
|
"reload_event_filter": reloadEventFilter,
|
||||||
|
".ocr_image": ocrImage,
|
||||||
|
"ocr_image": ocrImage,
|
||||||
|
"get_group_at_all_remain": getGroupAtAllRemain,
|
||||||
|
"get_online_clients": getOnlineClients,
|
||||||
|
".get_word_slices": getWordSlices,
|
||||||
|
"set_group_portrait": setGroupPortrait,
|
||||||
|
"set_essence_msg": setEssenceMSG,
|
||||||
|
"delete_essence_msg": deleteEssenceMSG,
|
||||||
|
"get_essence_msg_list": getEssenceMsgList,
|
||||||
|
"check_url_safely": checkUrlSafely,
|
||||||
|
"set_group_anonymous_ban": setGroupAnonymousBan,
|
||||||
|
".handle_quick_operation": handleQuickOperation,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (api *apiCaller) callAPI(action string, p resultGetter) coolq.MSG {
|
||||||
|
if f, ok := API[action]; ok {
|
||||||
|
return f(api.bot, p)
|
||||||
|
} else {
|
||||||
|
return coolq.Failed(404, "API_NOT_FOUND", "API不存在")
|
||||||
|
}
|
||||||
|
}
|
@ -5,8 +5,6 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/Mrs4s/MiraiGo/utils"
|
|
||||||
"github.com/gin-contrib/pprof"
|
|
||||||
"image"
|
"image"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -14,8 +12,12 @@ import (
|
|||||||
"os/signal"
|
"os/signal"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/Mrs4s/MiraiGo/utils"
|
||||||
|
"github.com/gin-contrib/pprof"
|
||||||
|
|
||||||
"github.com/Mrs4s/MiraiGo/client"
|
"github.com/Mrs4s/MiraiGo/client"
|
||||||
"github.com/Mrs4s/go-cqhttp/coolq"
|
"github.com/Mrs4s/go-cqhttp/coolq"
|
||||||
"github.com/Mrs4s/go-cqhttp/global"
|
"github.com/Mrs4s/go-cqhttp/global"
|
||||||
@ -28,39 +30,45 @@ import (
|
|||||||
|
|
||||||
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
var json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
|
|
||||||
|
// WebInput 网页输入channel
|
||||||
var WebInput = make(chan string, 1) //长度1,用于阻塞
|
var WebInput = make(chan string, 1) //长度1,用于阻塞
|
||||||
|
|
||||||
|
// Console 控制台channel
|
||||||
var Console = make(chan os.Signal, 1)
|
var Console = make(chan os.Signal, 1)
|
||||||
|
|
||||||
|
// Restart 重启信号监听channel
|
||||||
var Restart = make(chan struct{}, 1)
|
var Restart = make(chan struct{}, 1)
|
||||||
|
|
||||||
var JsonConfig *global.JsonConfig
|
// JSONConfig go-cqhttp配置
|
||||||
|
var JSONConfig *global.JSONConfig
|
||||||
|
|
||||||
type webServer struct {
|
type webServer struct {
|
||||||
engine *gin.Engine
|
engine *gin.Engine
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
Cli *client.QQClient
|
Cli *client.QQClient
|
||||||
Conf *global.JsonConfig //old config
|
Conf *global.JSONConfig //old config
|
||||||
Console *bufio.Reader
|
Console *bufio.Reader
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WebServer Admin子站的Server
|
||||||
var WebServer = &webServer{}
|
var WebServer = &webServer{}
|
||||||
|
|
||||||
// admin 子站的 路由映射
|
// APIAdminRoutingTable Admin子站的路由映射
|
||||||
var HttpuriAdmin = map[string]func(s *webServer, c *gin.Context){
|
var APIAdminRoutingTable = map[string]func(s *webServer, c *gin.Context){
|
||||||
"do_restart": AdminDoRestart, //热重启
|
"do_restart": AdminDoRestart, //热重启
|
||||||
"do_process_restart": AdminProcessRestart, //进程重启
|
"do_process_restart": AdminProcessRestart, //进程重启
|
||||||
"get_web_write": AdminWebWrite, //获取是否验证码输入
|
"get_web_write": AdminWebWrite, //获取是否验证码输入
|
||||||
"do_web_write": AdminDoWebWrite, //web上进行输入操作
|
"do_web_write": AdminDoWebWrite, //web上进行输入操作
|
||||||
"do_restart_docker": AdminDoRestartDocker, //直接停止(依赖supervisord/docker)重新拉起
|
"do_restart_docker": AdminDoRestartDocker, //直接停止(依赖supervisord/docker)重新拉起
|
||||||
"do_config_base": AdminDoConfigBase, //修改config.json中的基础部分
|
"do_config_base": AdminDoConfigBase, //修改config.json中的基础部分
|
||||||
"do_config_http": AdminDoConfigHttp, //修改config.json的http部分
|
"do_config_http": AdminDoConfigHTTP, //修改config.json的http部分
|
||||||
"do_config_ws": AdminDoConfigWs, //修改config.json的正向ws部分
|
"do_config_ws": AdminDoConfigWS, //修改config.json的正向ws部分
|
||||||
"do_config_reverse": AdminDoConfigReverse, //修改config.json 中的反向ws部分
|
"do_config_reverse": AdminDoConfigReverseWS, //修改config.json 中的反向ws部分
|
||||||
"do_config_json": AdminDoConfigJson, //直接修改 config.json配置
|
"do_config_json": AdminDoConfigJSON, //直接修改 config.json配置
|
||||||
"get_config_json": AdminGetConfigJson, //拉取 当前的config.json配置
|
"get_config_json": AdminGetConfigJSON, //拉取 当前的config.json配置
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Failed 构建失败返回MSG
|
||||||
func Failed(code int, msg string) coolq.MSG {
|
func Failed(code int, msg string) coolq.MSG {
|
||||||
return coolq.MSG{"data": nil, "retcode": code, "status": "failed", "msg": msg}
|
return coolq.MSG{"data": nil, "retcode": code, "status": "failed", "msg": msg}
|
||||||
}
|
}
|
||||||
@ -68,18 +76,18 @@ func Failed(code int, msg string) coolq.MSG {
|
|||||||
func (s *webServer) Run(addr string, cli *client.QQClient) *coolq.CQBot {
|
func (s *webServer) Run(addr string, cli *client.QQClient) *coolq.CQBot {
|
||||||
s.Cli = cli
|
s.Cli = cli
|
||||||
s.Conf = GetConf()
|
s.Conf = GetConf()
|
||||||
JsonConfig = s.Conf
|
JSONConfig = s.Conf
|
||||||
gin.SetMode(gin.ReleaseMode)
|
gin.SetMode(gin.ReleaseMode)
|
||||||
s.engine = gin.New()
|
s.engine = gin.New()
|
||||||
|
|
||||||
s.engine.Use(AuthMiddleWare())
|
s.engine.Use(AuthMiddleWare())
|
||||||
|
|
||||||
//通用路由
|
// 通用路由
|
||||||
s.engine.Any("/admin/:action", s.admin)
|
s.engine.Any("/admin/:action", s.admin)
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
//开启端口监听
|
// 开启端口监听
|
||||||
if s.Conf.WebUi != nil && s.Conf.WebUi.Enabled {
|
if s.Conf.WebUI != nil && s.Conf.WebUI.Enabled {
|
||||||
if Debug {
|
if Debug {
|
||||||
pprof.Register(s.engine)
|
pprof.Register(s.engine)
|
||||||
log.Debugf("pprof 性能分析服务已启动在 http://%v/debug/pprof, 如果有任何性能问题请下载报告并提交给开发者", addr)
|
log.Debugf("pprof 性能分析服务已启动在 http://%v/debug/pprof, 如果有任何性能问题请下载报告并提交给开发者", addr)
|
||||||
@ -91,246 +99,280 @@ func (s *webServer) Run(addr string, cli *client.QQClient) *coolq.CQBot {
|
|||||||
log.Error(err)
|
log.Error(err)
|
||||||
log.Infof("请检查端口是否被占用.")
|
log.Infof("请检查端口是否被占用.")
|
||||||
c := make(chan os.Signal, 1)
|
c := make(chan os.Signal, 1)
|
||||||
signal.Notify(c, os.Interrupt, os.Kill)
|
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
|
||||||
<-c
|
<-c
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
//关闭端口监听
|
// 关闭端口监听
|
||||||
c := make(chan os.Signal, 1)
|
c := make(chan os.Signal, 1)
|
||||||
signal.Notify(c, os.Interrupt, os.Kill)
|
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
|
||||||
<-c
|
<-c
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
s.Dologin()
|
s.Dologin()
|
||||||
s.UpServer()
|
s.UpServer()
|
||||||
b := s.bot //外部引入 bot对象,用于操作bot
|
b := s.bot // 外部引入 bot对象,用于操作bot
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *webServer) Dologin() {
|
// logincore 登录核心实现
|
||||||
|
func (s *webServer) logincore(relogin bool) {
|
||||||
|
|
||||||
s.Console = bufio.NewReader(os.Stdin)
|
s.Console = bufio.NewReader(os.Stdin)
|
||||||
readLine := func() (str string) {
|
readLine := func() (str string) {
|
||||||
str, _ = s.Console.ReadString('\n')
|
str, _ = s.Console.ReadString('\n')
|
||||||
str = strings.TrimSpace(str)
|
str = strings.TrimSpace(str)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
conf := GetConf()
|
|
||||||
cli := s.Cli
|
if s.Cli.Online {
|
||||||
cli.AllowSlider = true
|
log.Warn("Bot已登录")
|
||||||
rsp, err := cli.Login()
|
return
|
||||||
count := 0
|
}
|
||||||
for {
|
|
||||||
global.Check(err)
|
var times uint = 1 // 重试次数
|
||||||
|
for res, err := s.Cli.Login(); ; res, err = s.Cli.Login() {
|
||||||
|
|
||||||
var text string
|
var text string
|
||||||
if !rsp.Success {
|
count := 0
|
||||||
switch rsp.Error {
|
|
||||||
case client.SliderNeededError:
|
if res == nil {
|
||||||
log.Warnf("登录需要滑条验证码, 请选择解决方案: ")
|
goto Relogin
|
||||||
log.Warnf("1. 自行抓包. (推荐)")
|
}
|
||||||
log.Warnf("2. 使用Cef自动处理.")
|
|
||||||
log.Warnf("3. 不提交滑块并继续.(可能会导致上网环境异常错误)")
|
Again: // 不执行 s.Cli.Login() 的循环,适用输入验证码等更新 res 的操作
|
||||||
log.Warnf("详细信息请参考文档 -> https://github.com/Mrs4s/go-cqhttp/blob/master/docs/slider.md <-")
|
if err == nil && res.Success { // 登录成功
|
||||||
log.Warn("请输入(1 - 3): ")
|
break
|
||||||
|
} else if err == client.ErrAlreadyOnline {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
switch res.Error {
|
||||||
|
case client.SliderNeededError:
|
||||||
|
log.Warnf("登录需要滑条验证码, 请选择解决方案: ")
|
||||||
|
log.Warnf("1. 自行抓包. (推荐)")
|
||||||
|
log.Warnf("2. 使用Cef自动处理.")
|
||||||
|
log.Warnf("3. 不提交滑块并继续.(可能会导致上网环境异常错误)")
|
||||||
|
log.Warnf("详细信息请参考文档 -> https://github.com/Mrs4s/go-cqhttp/blob/master/docs/slider.md <-")
|
||||||
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
|
log.Warnf("请输入(1 - 3): (http://%s:%d/admin/do_web_write 输入)", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
|
text = <-WebInput
|
||||||
|
} else {
|
||||||
|
log.Warn("请输入(1 - 3):")
|
||||||
text = readLine()
|
text = readLine()
|
||||||
if strings.Contains(text, "1") {
|
}
|
||||||
log.Warnf("请用浏览器打开 -> %v <- 并获取Ticket.", rsp.VerifyUrl)
|
if strings.Contains(text, "1") {
|
||||||
log.Warn("请输入Ticket: (Enter 提交)")
|
log.Warnf("请用浏览器打开 -> %v <- 并获取Ticket.", res.VerifyUrl)
|
||||||
text = readLine()
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
rsp, err = cli.SubmitTicket(strings.TrimSpace(text))
|
log.Warnf("请输入Ticket: (http://%s:%d/admin/do_web_write 输入)", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
return
|
|
||||||
}
|
|
||||||
if strings.Contains(text, "3") {
|
|
||||||
cli.AllowSlider = false
|
|
||||||
cli.Disconnect()
|
|
||||||
rsp, err = cli.Login()
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
id := utils.RandomStringRange(6, "0123456789")
|
|
||||||
log.Warnf("滑块ID为 %v 请在30S内处理.", id)
|
|
||||||
ticket, err := global.GetSliderTicket(rsp.VerifyUrl, id)
|
|
||||||
if err != nil {
|
|
||||||
log.Warnf("错误: " + err.Error())
|
|
||||||
os.Exit(0)
|
|
||||||
}
|
|
||||||
rsp, err = cli.SubmitTicket(ticket)
|
|
||||||
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
|
text = <-WebInput
|
||||||
} else {
|
} else {
|
||||||
log.Warn("请输入验证码 (captcha.jpg): (Enter 提交)")
|
log.Warn("请输入Ticket: (Enter 提交)")
|
||||||
text = readLine()
|
text = readLine()
|
||||||
}
|
}
|
||||||
rsp, err = cli.SubmitCaptcha(strings.ReplaceAll(text, "\n", ""), rsp.CaptchaSign)
|
res, err = s.Cli.SubmitTicket(strings.TrimSpace(text))
|
||||||
global.DelFile("captcha.jpg")
|
goto Again
|
||||||
|
}
|
||||||
|
if strings.Contains(text, "3") {
|
||||||
|
s.Cli.AllowSlider = false
|
||||||
|
s.Cli.Disconnect()
|
||||||
continue
|
continue
|
||||||
case client.SMSNeededError:
|
}
|
||||||
log.Warnf("账号已开启设备锁, 按下 Enter 向手机 %v 发送短信验证码.", rsp.SMSPhone)
|
id := utils.RandomStringRange(6, "0123456789")
|
||||||
|
log.Warnf("滑块ID为 %v 请在30S内处理.", id)
|
||||||
|
ticket, err := global.GetSliderTicket(res.VerifyUrl, id)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("错误: " + err.Error())
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
res, err = s.Cli.SubmitTicket(ticket)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("错误: " + err.Error())
|
||||||
|
continue // 尝试重新登录
|
||||||
|
}
|
||||||
|
goto Again
|
||||||
|
case client.NeedCaptcha:
|
||||||
|
_ = ioutil.WriteFile("captcha.jpg", res.CaptchaImage, 0644)
|
||||||
|
img, _, _ := image.Decode(bytes.NewReader(res.CaptchaImage))
|
||||||
|
fmt.Println(asciiart.New("image", img).Art)
|
||||||
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
|
log.Warnf("请输入验证码 (captcha.jpg): (http://%s:%d/admin/do_web_write 输入)", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
|
text = <-WebInput
|
||||||
|
} else {
|
||||||
|
log.Warn("请输入验证码 (captcha.jpg): (Enter 提交)")
|
||||||
|
text = readLine()
|
||||||
|
}
|
||||||
|
global.DelFile("captcha.jpg")
|
||||||
|
res, err = s.Cli.SubmitCaptcha(strings.ReplaceAll(text, "\n", ""), res.CaptchaSign)
|
||||||
|
goto Again
|
||||||
|
case client.SMSNeededError:
|
||||||
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
|
log.Warnf("账号已开启设备锁, 已向手机 %v 发送短信验证码.", res.SMSPhone)
|
||||||
|
} else {
|
||||||
|
log.Warnf("账号已开启设备锁, 按下 Enter 向手机 %v 发送短信验证码.", res.SMSPhone)
|
||||||
readLine()
|
readLine()
|
||||||
if !cli.RequestSMS() {
|
}
|
||||||
|
if !s.Cli.RequestSMS() {
|
||||||
|
log.Warnf("发送验证码失败,可能是请求过于频繁.")
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
|
log.Warnf("请输入短信验证码: (http://%s:%d/admin/do_web_write 输入)", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
|
text = <-WebInput
|
||||||
|
} else {
|
||||||
|
log.Warn("请输入短信验证码: (Enter 提交)")
|
||||||
|
text = readLine()
|
||||||
|
}
|
||||||
|
res, err = s.Cli.SubmitSMS(strings.ReplaceAll(strings.ReplaceAll(text, "\n", ""), "\r", ""))
|
||||||
|
goto Again
|
||||||
|
case client.SMSOrVerifyNeededError:
|
||||||
|
log.Warnf("账号已开启设备锁,请选择验证方式:")
|
||||||
|
log.Warnf("1. 向手机 %v 发送短信验证码", res.SMSPhone)
|
||||||
|
log.Warnf("2. 使用手机QQ扫码验证.")
|
||||||
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
|
log.Warnf("请输入(1 - 2): (http://%s:%d/admin/do_web_write 输入)", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
|
text = <-WebInput
|
||||||
|
} else {
|
||||||
|
log.Warn("请输入(1 - 2):")
|
||||||
|
text = readLine()
|
||||||
|
}
|
||||||
|
if strings.Contains(text, "1") {
|
||||||
|
if !s.Cli.RequestSMS() {
|
||||||
log.Warnf("发送验证码失败,可能是请求过于频繁.")
|
log.Warnf("发送验证码失败,可能是请求过于频繁.")
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
}
|
}
|
||||||
log.Warn("请输入短信验证码: (Enter 提交)")
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
text = readLine()
|
log.Warnf("请输入短信验证码: (http://%s:%d/admin/do_web_write 输入)....", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
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
|
text = <-WebInput
|
||||||
} else {
|
} else {
|
||||||
log.Infof("按 Enter 继续....")
|
log.Warn("请输入短信验证码: (Enter 提交)")
|
||||||
readLine()
|
text = readLine()
|
||||||
}
|
}
|
||||||
log.Info(text)
|
res, err = s.Cli.SubmitSMS(strings.ReplaceAll(strings.ReplaceAll(text, "\n", ""), "\r", ""))
|
||||||
os.Exit(0)
|
goto Again
|
||||||
return
|
}
|
||||||
case client.OtherLoginError, client.UnknownLoginError:
|
log.Warnf("请前往 -> %v <- 验证.", res.VerifyUrl)
|
||||||
msg := rsp.ErrorMessage
|
log.Infof("按 Enter 继续....")
|
||||||
if strings.Contains(msg, "版本") {
|
readLine()
|
||||||
msg = "密码错误或账号被冻结"
|
continue
|
||||||
}
|
case client.UnsafeDeviceError:
|
||||||
if strings.Contains(msg, "上网环境") && count < 5 {
|
log.Warnf("账号已开启设备锁,请前往 -> %v <- 验证.", res.VerifyUrl)
|
||||||
cli.Disconnect()
|
if s.Conf.WebUI != nil && s.Conf.WebUI.WebInput {
|
||||||
rsp, err = cli.Login()
|
log.Infof(" (http://%s:%d/admin/do_web_write 确认后继续)....", s.Conf.WebUI.Host, s.Conf.WebUI.WebUIPort)
|
||||||
count++
|
text = <-WebInput
|
||||||
log.Warnf("错误: 当前上网环境异常. 将更换服务器并重试.")
|
} else {
|
||||||
time.Sleep(time.Second)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
log.Warnf("登录失败: %v", msg)
|
|
||||||
log.Infof("按 Enter 继续....")
|
log.Infof("按 Enter 继续....")
|
||||||
readLine()
|
readLine()
|
||||||
os.Exit(0)
|
}
|
||||||
|
log.Info(text)
|
||||||
|
continue
|
||||||
|
case client.OtherLoginError, client.UnknownLoginError:
|
||||||
|
msg := res.ErrorMessage
|
||||||
|
if strings.Contains(msg, "版本") {
|
||||||
|
msg = "密码错误或账号被冻结"
|
||||||
|
}
|
||||||
|
if strings.Contains(msg, "上网环境") && count < 5 {
|
||||||
|
s.Cli.Disconnect()
|
||||||
|
log.Warnf("错误: 当前上网环境异常. 将更换服务器并重试.")
|
||||||
|
count++
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if strings.Contains(msg, "冻结") {
|
||||||
|
log.Fatalf("账号被冻结, 放弃重连")
|
||||||
|
}
|
||||||
|
log.Warnf("登录失败: %v", msg)
|
||||||
|
log.Infof("按 Enter 继续....")
|
||||||
|
readLine()
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
Relogin:
|
||||||
|
if relogin {
|
||||||
|
if times > s.Conf.ReLogin.MaxReloginTimes && s.Conf.ReLogin.MaxReloginTimes != 0 {
|
||||||
|
log.Fatal("重连失败: 重连次数达到设置的上限值")
|
||||||
|
s.bot.Release()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
log.Warnf("将在 %v 秒后尝试重连. 重连次数:%v", s.Conf.ReLogin.ReLoginDelay, times)
|
||||||
|
times++
|
||||||
|
time.Sleep(time.Second * time.Duration(s.Conf.ReLogin.ReLoginDelay))
|
||||||
|
s.Cli.Disconnect()
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
break
|
|
||||||
}
|
}
|
||||||
log.Infof("登录成功 欢迎使用: %v", cli.Nickname)
|
if relogin {
|
||||||
time.Sleep(time.Second)
|
log.Info("重连成功")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dologin 主程序登录
|
||||||
|
func (s *webServer) Dologin() {
|
||||||
|
|
||||||
|
s.Cli.AllowSlider = true
|
||||||
|
s.logincore(false)
|
||||||
|
log.Infof("登录成功 欢迎使用: %v", s.Cli.Nickname)
|
||||||
log.Info("开始加载好友列表...")
|
log.Info("开始加载好友列表...")
|
||||||
global.Check(cli.ReloadFriendList())
|
global.Check(s.Cli.ReloadFriendList())
|
||||||
log.Infof("共加载 %v 个好友.", len(cli.FriendList))
|
log.Infof("共加载 %v 个好友.", len(s.Cli.FriendList))
|
||||||
log.Infof("开始加载群列表...")
|
log.Infof("开始加载群列表...")
|
||||||
global.Check(cli.ReloadGroupList())
|
global.Check(s.Cli.ReloadGroupList())
|
||||||
log.Infof("共加载 %v 个群.", len(cli.GroupList))
|
log.Infof("共加载 %v 个群.", len(s.Cli.GroupList))
|
||||||
s.bot = coolq.NewQQBot(cli, conf)
|
s.bot = coolq.NewQQBot(s.Cli, s.Conf)
|
||||||
if conf.PostMessageFormat != "string" && conf.PostMessageFormat != "array" {
|
if s.Conf.PostMessageFormat != "string" && s.Conf.PostMessageFormat != "array" {
|
||||||
log.Warnf("post_message_format 配置错误, 将自动使用 string")
|
log.Warnf("post_message_format 配置错误, 将自动使用 string")
|
||||||
coolq.SetMessageFormat("string")
|
coolq.SetMessageFormat("string")
|
||||||
} else {
|
} else {
|
||||||
coolq.SetMessageFormat(conf.PostMessageFormat)
|
coolq.SetMessageFormat(s.Conf.PostMessageFormat)
|
||||||
}
|
}
|
||||||
if conf.RateLimit.Enabled {
|
if s.Conf.RateLimit.Enabled {
|
||||||
global.InitLimiter(conf.RateLimit.Frequency, conf.RateLimit.BucketSize)
|
global.InitLimiter(s.Conf.RateLimit.Frequency, s.Conf.RateLimit.BucketSize)
|
||||||
}
|
}
|
||||||
log.Info("正在加载事件过滤器.")
|
log.Info("正在加载事件过滤器.")
|
||||||
global.BootFilter()
|
global.BootFilter()
|
||||||
global.InitCodec()
|
coolq.IgnoreInvalidCQCode = s.Conf.IgnoreInvalidCQCode
|
||||||
coolq.IgnoreInvalidCQCode = conf.IgnoreInvalidCQCode
|
coolq.SplitURL = s.Conf.FixURL
|
||||||
coolq.SplitUrl = conf.FixUrl
|
coolq.ForceFragmented = s.Conf.ForceFragmented
|
||||||
coolq.ForceFragmented = conf.ForceFragmented
|
|
||||||
log.Info("资源初始化完成, 开始处理信息.")
|
log.Info("资源初始化完成, 开始处理信息.")
|
||||||
log.Info("アトリは、高性能ですから!")
|
log.Info("アトリは、高性能ですから!")
|
||||||
cli.OnDisconnected(func(bot *client.QQClient, e *client.ClientDisconnectedEvent) {
|
|
||||||
if conf.ReLogin.Enabled {
|
s.Cli.OnDisconnected(func(q *client.QQClient, e *client.ClientDisconnectedEvent) {
|
||||||
var times uint = 1
|
if !s.Conf.ReLogin.Enabled {
|
||||||
for {
|
return
|
||||||
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.Warnf("Bot已离线 (%v),尝试重连", e.Message)
|
||||||
log.Fatalf("Bot已离线:%v", e.Message)
|
s.logincore(true)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *webServer) admin(c *gin.Context) {
|
func (s *webServer) admin(c *gin.Context) {
|
||||||
action := c.Param("action")
|
action := c.Param("action")
|
||||||
log.Debugf("WebServer接收到cgi调用: %v", action)
|
log.Debugf("WebServer接收到cgi调用: %v", action)
|
||||||
if f, ok := HttpuriAdmin[action]; ok {
|
if f, ok := APIAdminRoutingTable[action]; ok {
|
||||||
f(s, c)
|
f(s, c)
|
||||||
} else {
|
} else {
|
||||||
c.JSON(200, coolq.Failed(404))
|
c.JSON(200, coolq.Failed(404))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 获取当前配置文件信息
|
// GetConf 获取当前配置文件信息
|
||||||
func GetConf() *global.JsonConfig {
|
func GetConf() *global.JSONConfig {
|
||||||
if JsonConfig != nil {
|
if JSONConfig != nil {
|
||||||
return JsonConfig
|
return JSONConfig
|
||||||
}
|
}
|
||||||
conf := global.Load("config.hjson")
|
conf := global.LoadConfig("config.hjson")
|
||||||
return conf
|
return conf
|
||||||
}
|
}
|
||||||
|
|
||||||
// admin 控制器 登录验证
|
// AuthMiddleWare Admin控制器登录验证
|
||||||
func AuthMiddleWare() gin.HandlerFunc {
|
func AuthMiddleWare() gin.HandlerFunc {
|
||||||
return func(c *gin.Context) {
|
return func(c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
//处理跨域问题
|
// 处理跨域问题
|
||||||
c.Header("Access-Control-Allow-Origin", "*")
|
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-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-Allow-Methods", "POST, GET, OPTIONS, PUT, PATCH, DELETE")
|
||||||
@ -380,7 +422,7 @@ func AuthMiddleWare() gin.HandlerFunc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *webServer) DoReLogin() { // TODO: 协议层的 ReLogin
|
func (s *webServer) DoReLogin() { // TODO: 协议层的 ReLogin
|
||||||
JsonConfig = nil
|
JSONConfig = nil
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
OldConf := s.Conf
|
OldConf := s.Conf
|
||||||
cli := client.NewClient(conf.Uin, conf.Password)
|
cli := client.NewClient(conf.Uin, conf.Password)
|
||||||
@ -418,72 +460,71 @@ func (s *webServer) DoReLogin() { // TODO: 协议层的 ReLogin
|
|||||||
})
|
})
|
||||||
s.Cli = cli
|
s.Cli = cli
|
||||||
s.Dologin()
|
s.Dologin()
|
||||||
//关闭之前的 server
|
// 关闭之前的 server
|
||||||
if OldConf.HttpConfig != nil && OldConf.HttpConfig.Enabled {
|
if OldConf.HTTPConfig != nil && OldConf.HTTPConfig.Enabled {
|
||||||
HttpServer.ShutDown()
|
cqHTTPServer.ShutDown()
|
||||||
}
|
}
|
||||||
//if OldConf.WSConfig != nil && OldConf.WSConfig.Enabled {
|
// if OldConf.WSConfig != nil && OldConf.WSConfig.Enabled {
|
||||||
// server.WsShutdown()
|
// server.WsShutdown()
|
||||||
//}
|
// }
|
||||||
//s.UpServer()
|
// s.UpServer()
|
||||||
|
|
||||||
s.ReloadServer()
|
s.ReloadServer()
|
||||||
s.Conf = conf
|
s.Conf = conf
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *webServer) UpServer() {
|
func (s *webServer) UpServer() {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
if conf.HttpConfig != nil && conf.HttpConfig.Enabled {
|
if conf.HTTPConfig != nil && conf.HTTPConfig.Enabled {
|
||||||
go HttpServer.Run(fmt.Sprintf("%s:%d", conf.HttpConfig.Host, conf.HttpConfig.Port), conf.AccessToken, s.bot)
|
go cqHTTPServer.Run(fmt.Sprintf("%s:%d", conf.HTTPConfig.Host, conf.HTTPConfig.Port), conf.AccessToken, s.bot)
|
||||||
for k, v := range conf.HttpConfig.PostUrls {
|
for k, v := range conf.HTTPConfig.PostUrls {
|
||||||
NewHttpClient().Run(k, v, conf.HttpConfig.Timeout, s.bot)
|
newHTTPClient().Run(k, v, conf.HTTPConfig.Timeout, s.bot)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if conf.WSConfig != nil && conf.WSConfig.Enabled {
|
if conf.WSConfig != nil && conf.WSConfig.Enabled {
|
||||||
go WebsocketServer.Run(fmt.Sprintf("%s:%d", conf.WSConfig.Host, conf.WSConfig.Port), conf.AccessToken, s.bot)
|
go WebSocketServer.Run(fmt.Sprintf("%s:%d", conf.WSConfig.Host, conf.WSConfig.Port), conf.AccessToken, s.bot)
|
||||||
}
|
}
|
||||||
for _, rc := range conf.ReverseServers {
|
for _, rc := range conf.ReverseServers {
|
||||||
go NewWebsocketClient(rc, conf.AccessToken, s.bot).Run()
|
go NewWebSocketClient(rc, conf.AccessToken, s.bot).Run()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 暂不支持ws服务的重启
|
// 暂不支持ws服务的重启
|
||||||
func (s *webServer) ReloadServer() {
|
func (s *webServer) ReloadServer() {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
if conf.HttpConfig != nil && conf.HttpConfig.Enabled {
|
if conf.HTTPConfig != nil && conf.HTTPConfig.Enabled {
|
||||||
go HttpServer.Run(fmt.Sprintf("%s:%d", conf.HttpConfig.Host, conf.HttpConfig.Port), conf.AccessToken, s.bot)
|
go cqHTTPServer.Run(fmt.Sprintf("%s:%d", conf.HTTPConfig.Host, conf.HTTPConfig.Port), conf.AccessToken, s.bot)
|
||||||
for k, v := range conf.HttpConfig.PostUrls {
|
for k, v := range conf.HTTPConfig.PostUrls {
|
||||||
NewHttpClient().Run(k, v, conf.HttpConfig.Timeout, s.bot)
|
newHTTPClient().Run(k, v, conf.HTTPConfig.Timeout, s.bot)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for _, rc := range conf.ReverseServers {
|
for _, rc := range conf.ReverseServers {
|
||||||
go NewWebsocketClient(rc, conf.AccessToken, s.bot).Run()
|
go NewWebSocketClient(rc, conf.AccessToken, s.bot).Run()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 热重启
|
// AdminDoRestart 热重启
|
||||||
func AdminDoRestart(s *webServer, c *gin.Context) {
|
func AdminDoRestart(s *webServer, c *gin.Context) {
|
||||||
s.bot.Release()
|
s.bot.Release()
|
||||||
s.bot = nil
|
s.bot = nil
|
||||||
s.Cli = nil
|
s.Cli = nil
|
||||||
s.DoReLogin()
|
s.DoReLogin()
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 进程重启
|
// AdminProcessRestart 进程重启
|
||||||
func AdminProcessRestart(s *webServer, c *gin.Context) {
|
func AdminProcessRestart(s *webServer, c *gin.Context) {
|
||||||
Restart <- struct{}{}
|
Restart <- struct{}{}
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// 冷重启
|
// AdminDoRestartDocker 冷重启
|
||||||
func AdminDoRestartDocker(s *webServer, c *gin.Context) {
|
func AdminDoRestartDocker(s *webServer, c *gin.Context) {
|
||||||
Console <- os.Kill
|
Console <- os.Kill
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// web输入 html 页面
|
// AdminWebWrite web输入html页面
|
||||||
func AdminWebWrite(s *webServer, c *gin.Context) {
|
func AdminWebWrite(s *webServer, c *gin.Context) {
|
||||||
pic := global.ReadAllText("captcha.jpg")
|
pic := global.ReadAllText("captcha.jpg")
|
||||||
var picbase64 string
|
var picbase64 string
|
||||||
@ -500,14 +541,14 @@ func AdminWebWrite(s *webServer, c *gin.Context) {
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// web输入 处理
|
// AdminDoWebWrite web输入处理
|
||||||
func AdminDoWebWrite(s *webServer, c *gin.Context) {
|
func AdminDoWebWrite(s *webServer, c *gin.Context) {
|
||||||
input := c.PostForm("input")
|
input := c.PostForm("input")
|
||||||
WebInput <- input
|
WebInput <- input
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// 普通配置修改
|
// AdminDoConfigBase 普通配置修改
|
||||||
func AdminDoConfigBase(s *webServer, c *gin.Context) {
|
func AdminDoConfigBase(s *webServer, c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
conf.Uin, _ = strconv.ParseInt(c.PostForm("uin"), 10, 64)
|
conf.Uin, _ = strconv.ParseInt(c.PostForm("uin"), 10, 64)
|
||||||
@ -522,38 +563,38 @@ func AdminDoConfigBase(s *webServer, c *gin.Context) {
|
|||||||
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
} else {
|
} else {
|
||||||
JsonConfig = nil
|
JSONConfig = nil
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// http配置修改
|
// AdminDoConfigHTTP HTTP配置修改
|
||||||
func AdminDoConfigHttp(s *webServer, c *gin.Context) {
|
func AdminDoConfigHTTP(s *webServer, c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
p, _ := strconv.ParseUint(c.PostForm("port"), 10, 16)
|
p, _ := strconv.ParseUint(c.PostForm("port"), 10, 16)
|
||||||
conf.HttpConfig.Port = uint16(p)
|
conf.HTTPConfig.Port = uint16(p)
|
||||||
conf.HttpConfig.Host = c.PostForm("host")
|
conf.HTTPConfig.Host = c.PostForm("host")
|
||||||
if c.PostForm("enable") == "true" {
|
if c.PostForm("enable") == "true" {
|
||||||
conf.HttpConfig.Enabled = true
|
conf.HTTPConfig.Enabled = true
|
||||||
} else {
|
} else {
|
||||||
conf.HttpConfig.Enabled = false
|
conf.HTTPConfig.Enabled = false
|
||||||
}
|
}
|
||||||
t, _ := strconv.ParseInt(c.PostForm("timeout"), 10, 32)
|
t, _ := strconv.ParseInt(c.PostForm("timeout"), 10, 32)
|
||||||
conf.HttpConfig.Timeout = int32(t)
|
conf.HTTPConfig.Timeout = int32(t)
|
||||||
if c.PostForm("post_url") != "" {
|
if c.PostForm("post_url") != "" {
|
||||||
conf.HttpConfig.PostUrls[c.PostForm("post_url")] = c.PostForm("post_secret")
|
conf.HTTPConfig.PostUrls[c.PostForm("post_url")] = c.PostForm("post_secret")
|
||||||
}
|
}
|
||||||
if err := conf.Save("config.hjson"); err != nil {
|
if err := conf.Save("config.hjson"); err != nil {
|
||||||
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
} else {
|
} else {
|
||||||
JsonConfig = nil
|
JSONConfig = nil
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ws配置修改
|
// AdminDoConfigWS ws配置修改
|
||||||
func AdminDoConfigWs(s *webServer, c *gin.Context) {
|
func AdminDoConfigWS(s *webServer, c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
p, _ := strconv.ParseUint(c.PostForm("port"), 10, 16)
|
p, _ := strconv.ParseUint(c.PostForm("port"), 10, 16)
|
||||||
conf.WSConfig.Port = uint16(p)
|
conf.WSConfig.Port = uint16(p)
|
||||||
@ -567,17 +608,17 @@ func AdminDoConfigWs(s *webServer, c *gin.Context) {
|
|||||||
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
} else {
|
} else {
|
||||||
JsonConfig = nil
|
JSONConfig = nil
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 反向ws配置修改
|
// AdminDoConfigReverseWS 反向ws配置修改
|
||||||
func AdminDoConfigReverse(s *webServer, c *gin.Context) {
|
func AdminDoConfigReverseWS(s *webServer, c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
conf.ReverseServers[0].ReverseApiUrl = c.PostForm("reverse_api_url")
|
conf.ReverseServers[0].ReverseAPIURL = c.PostForm("reverse_api_url")
|
||||||
conf.ReverseServers[0].ReverseUrl = c.PostForm("reverse_url")
|
conf.ReverseServers[0].ReverseURL = c.PostForm("reverse_url")
|
||||||
conf.ReverseServers[0].ReverseEventUrl = c.PostForm("reverse_event_url")
|
conf.ReverseServers[0].ReverseEventURL = c.PostForm("reverse_event_url")
|
||||||
t, _ := strconv.ParseUint(c.PostForm("reverse_reconnect_interval"), 10, 16)
|
t, _ := strconv.ParseUint(c.PostForm("reverse_reconnect_interval"), 10, 16)
|
||||||
conf.ReverseServers[0].ReverseReconnectInterval = uint16(t)
|
conf.ReverseServers[0].ReverseReconnectInterval = uint16(t)
|
||||||
if c.PostForm("enable") == "true" {
|
if c.PostForm("enable") == "true" {
|
||||||
@ -589,16 +630,16 @@ func AdminDoConfigReverse(s *webServer, c *gin.Context) {
|
|||||||
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
} else {
|
} else {
|
||||||
JsonConfig = nil
|
JSONConfig = nil
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// config.json配置修改
|
// AdminDoConfigJSON config.hjson配置修改
|
||||||
func AdminDoConfigJson(s *webServer, c *gin.Context) {
|
func AdminDoConfigJSON(s *webServer, c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
Json := c.PostForm("json")
|
JSON := c.PostForm("json")
|
||||||
err := json.Unmarshal([]byte(Json), &conf)
|
err := json.Unmarshal([]byte(JSON), &conf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("尝试加载配置文件 %v 时出现错误: %v", "config.hjson", err)
|
log.Warnf("尝试加载配置文件 %v 时出现错误: %v", "config.hjson", err)
|
||||||
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
@ -608,13 +649,13 @@ func AdminDoConfigJson(s *webServer, c *gin.Context) {
|
|||||||
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
log.Fatalf("保存 config.hjson 时出现错误: %v", err)
|
||||||
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
c.JSON(200, Failed(502, "保存 config.hjson 时出现错误:"+fmt.Sprintf("%v", err)))
|
||||||
} else {
|
} else {
|
||||||
JsonConfig = nil
|
JSONConfig = nil
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{}))
|
c.JSON(200, coolq.OK(coolq.MSG{}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 拉取config.json配置
|
// AdminGetConfigJSON 拉取config.hjson配置
|
||||||
func AdminGetConfigJson(s *webServer, c *gin.Context) {
|
func AdminGetConfigJSON(s *webServer, c *gin.Context) {
|
||||||
conf := GetConf()
|
conf := GetConf()
|
||||||
c.JSON(200, coolq.OK(coolq.MSG{"config": conf}))
|
c.JSON(200, coolq.OK(coolq.MSG{"config": conf}))
|
||||||
|
|
||||||
|
2
server/doc.go
Normal file
2
server/doc.go
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
// Package server 包含Admin子站,HTTP,WebSocket,反向WebSocket请求处理的相关函数与结构体
|
||||||
|
package server
|
426
server/http.go
426
server/http.go
@ -7,16 +7,14 @@ import (
|
|||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"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/Mrs4s/go-cqhttp/global"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/guonaihong/gout"
|
"github.com/guonaihong/gout"
|
||||||
|
"github.com/guonaihong/gout/dataflow"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"github.com/tidwall/gjson"
|
"github.com/tidwall/gjson"
|
||||||
)
|
)
|
||||||
@ -24,7 +22,8 @@ import (
|
|||||||
type httpServer struct {
|
type httpServer struct {
|
||||||
engine *gin.Engine
|
engine *gin.Engine
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
Http *http.Server
|
HTTP *http.Server
|
||||||
|
api apiCaller
|
||||||
}
|
}
|
||||||
|
|
||||||
type httpClient struct {
|
type httpClient struct {
|
||||||
@ -34,13 +33,20 @@ type httpClient struct {
|
|||||||
timeout int32
|
timeout int32
|
||||||
}
|
}
|
||||||
|
|
||||||
var HttpServer = &httpServer{}
|
type httpContext struct {
|
||||||
|
ctx *gin.Context
|
||||||
|
}
|
||||||
|
|
||||||
|
var cqHTTPServer = &httpServer{}
|
||||||
|
|
||||||
|
// Debug 是否启用Debug模式
|
||||||
var Debug = false
|
var Debug = false
|
||||||
|
|
||||||
func (s *httpServer) Run(addr, authToken string, bot *coolq.CQBot) {
|
func (s *httpServer) Run(addr, authToken string, bot *coolq.CQBot) {
|
||||||
gin.SetMode(gin.ReleaseMode)
|
gin.SetMode(gin.ReleaseMode)
|
||||||
s.engine = gin.New()
|
s.engine = gin.New()
|
||||||
s.bot = bot
|
s.bot = bot
|
||||||
|
s.api = apiCaller{s.bot}
|
||||||
s.engine.Use(func(c *gin.Context) {
|
s.engine.Use(func(c *gin.Context) {
|
||||||
if c.Request.Method != "GET" && c.Request.Method != "POST" {
|
if c.Request.Method != "GET" && c.Request.Method != "POST" {
|
||||||
log.Warnf("已拒绝客户端 %v 的请求: 方法错误", c.Request.RemoteAddr)
|
log.Warnf("已拒绝客户端 %v 的请求: 方法错误", c.Request.RemoteAddr)
|
||||||
@ -84,20 +90,21 @@ func (s *httpServer) Run(addr, authToken string, bot *coolq.CQBot) {
|
|||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
log.Infof("CQ HTTP 服务器已启动: %v", addr)
|
log.Infof("CQ HTTP 服务器已启动: %v", addr)
|
||||||
s.Http = &http.Server{
|
s.HTTP = &http.Server{
|
||||||
Addr: addr,
|
Addr: addr,
|
||||||
Handler: s.engine,
|
Handler: s.engine,
|
||||||
}
|
}
|
||||||
if err := s.Http.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
if err := s.HTTP.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
||||||
log.Error(err)
|
log.Error(err)
|
||||||
log.Infof("请检查端口是否被占用.")
|
log.Infof("HTTP 服务启动失败, 请检查端口是否被占用.")
|
||||||
|
log.Warnf("将在五秒后退出.")
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewHttpClient() *httpClient {
|
func newHTTPClient() *httpClient {
|
||||||
return &httpClient{}
|
return &httpClient{}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,7 +129,11 @@ func (c *httpClient) onBotPushEvent(m coolq.MSG) {
|
|||||||
}
|
}
|
||||||
if c.secret != "" {
|
if c.secret != "" {
|
||||||
mac := hmac.New(sha1.New, []byte(c.secret))
|
mac := hmac.New(sha1.New, []byte(c.secret))
|
||||||
mac.Write([]byte(m.ToJson()))
|
_, err := mac.Write([]byte(m.ToJSON()))
|
||||||
|
if err != nil {
|
||||||
|
log.Error(err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
h["X-Signature"] = "sha1=" + hex.EncodeToString(mac.Sum(nil))
|
h["X-Signature"] = "sha1=" + hex.EncodeToString(mac.Sum(nil))
|
||||||
}
|
}
|
||||||
return h
|
return h
|
||||||
@ -136,12 +147,12 @@ func (c *httpClient) onBotPushEvent(m coolq.MSG) {
|
|||||||
return nil
|
return nil
|
||||||
}).Do()
|
}).Do()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("上报Event数据 %v 到 %v 失败: %v", m.ToJson(), 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)
|
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))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -149,407 +160,38 @@ func (s *httpServer) HandleActions(c *gin.Context) {
|
|||||||
global.RateLimit(context.Background())
|
global.RateLimit(context.Background())
|
||||||
action := strings.ReplaceAll(c.Param("action"), "_async", "")
|
action := strings.ReplaceAll(c.Param("action"), "_async", "")
|
||||||
log.Debugf("HTTPServer接收到API调用: %v", action)
|
log.Debugf("HTTPServer接收到API调用: %v", action)
|
||||||
if f, ok := httpApi[action]; ok {
|
c.JSON(200, s.api.callAPI(action, httpContext{ctx: c}))
|
||||||
f(s, c)
|
|
||||||
} else {
|
|
||||||
c.JSON(200, coolq.Failed(404))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func GetLoginInfo(s *httpServer, c *gin.Context) {
|
func (h httpContext) Get(k string) gjson.Result {
|
||||||
c.JSON(200, s.bot.CQGetLoginInfo())
|
c := h.ctx
|
||||||
}
|
|
||||||
|
|
||||||
func GetFriendList(s *httpServer, c *gin.Context) {
|
|
||||||
c.JSON(200, s.bot.CQGetFriendList())
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetGroupList(s *httpServer, c *gin.Context) {
|
|
||||||
nc := getParamOrDefault(c, "no_cache", "false")
|
|
||||||
c.JSON(200, s.bot.CQGetGroupList(nc == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetGroupInfo(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
nc := getParamOrDefault(c, "no_cache", "false")
|
|
||||||
c.JSON(200, s.bot.CQGetGroupInfo(gid, nc == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetGroupMemberList(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
nc := getParamOrDefault(c, "no_cache", "false")
|
|
||||||
c.JSON(200, s.bot.CQGetGroupMemberList(gid, nc == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetGroupMemberInfo(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.CQGetGroupMemberInfo(gid, uid))
|
|
||||||
}
|
|
||||||
|
|
||||||
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") != "" {
|
|
||||||
SendGroupMessage(s, c)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if getParam(c, "user_id") != "" {
|
|
||||||
SendPrivateMessage(s, c)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func SendPrivateMessage(s *httpServer, c *gin.Context) {
|
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
|
||||||
msg, t := getParamWithType(c, "message")
|
|
||||||
autoEscape := global.EnsureBool(getParam(c, "auto_escape"), false)
|
|
||||||
if t == gjson.JSON {
|
|
||||||
c.JSON(200, s.bot.CQSendPrivateMessage(uid, gjson.Parse(msg), autoEscape))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
c.JSON(200, s.bot.CQSendPrivateMessage(uid, msg, autoEscape))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SendGroupMessage(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
msg, t := getParamWithType(c, "message")
|
|
||||||
autoEscape := global.EnsureBool(getParam(c, "auto_escape"), false)
|
|
||||||
if t == gjson.JSON {
|
|
||||||
c.JSON(200, s.bot.CQSendGroupMessage(gid, gjson.Parse(msg), autoEscape))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
c.JSON(200, s.bot.CQSendGroupMessage(gid, msg, autoEscape))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SendGroupForwardMessage(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
msg := getParam(c, "messages")
|
|
||||||
c.JSON(200, s.bot.CQSendGroupForwardMessage(gid, gjson.Parse(msg)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetImage(s *httpServer, c *gin.Context) {
|
|
||||||
file := getParam(c, "file")
|
|
||||||
c.JSON(200, s.bot.CQGetImage(file))
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetMessage(s *httpServer, c *gin.Context) {
|
|
||||||
mid, _ := strconv.ParseInt(getParam(c, "message_id"), 10, 32)
|
|
||||||
c.JSON(200, s.bot.CQGetMessage(int32(mid)))
|
|
||||||
}
|
|
||||||
|
|
||||||
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")
|
|
||||||
approve := getParamOrDefault(c, "approve", "true")
|
|
||||||
c.JSON(200, s.bot.CQProcessFriendRequest(flag, approve == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func ProcessGroupRequest(s *httpServer, c *gin.Context) {
|
|
||||||
flag := getParam(c, "flag")
|
|
||||||
subType := getParam(c, "sub_type")
|
|
||||||
if subType == "" {
|
|
||||||
subType = getParam(c, "type")
|
|
||||||
}
|
|
||||||
approve := getParamOrDefault(c, "approve", "true")
|
|
||||||
c.JSON(200, s.bot.CQProcessGroupRequest(flag, subType, getParam(c, "reason"), approve == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetGroupCard(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.CQSetGroupCard(gid, uid, getParam(c, "card")))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetSpecialTitle(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.CQSetGroupSpecialTitle(gid, uid, getParam(c, "special_title")))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetGroupKick(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
|
||||||
msg := getParam(c, "message")
|
|
||||||
block := getParamOrDefault(c, "reject_add_request", "false")
|
|
||||||
c.JSON(200, s.bot.CQSetGroupKick(gid, uid, msg, block == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetGroupBan(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
uid, _ := strconv.ParseInt(getParam(c, "user_id"), 10, 64)
|
|
||||||
i, _ := strconv.ParseInt(getParamOrDefault(c, "duration", "1800"), 10, 64)
|
|
||||||
c.JSON(200, s.bot.CQSetGroupBan(gid, uid, uint32(i)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetWholeBan(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
c.JSON(200, s.bot.CQSetGroupWholeBan(gid, getParamOrDefault(c, "enable", "true") == "true"))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetGroupName(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
c.JSON(200, s.bot.CQSetGroupName(gid, getParam(c, "group_name")))
|
|
||||||
}
|
|
||||||
|
|
||||||
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)
|
|
||||||
c.JSON(200, s.bot.CQSetGroupLeave(gid))
|
|
||||||
}
|
|
||||||
|
|
||||||
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")
|
|
||||||
if resId == "" {
|
|
||||||
resId = getParam(c, "id")
|
|
||||||
}
|
|
||||||
c.JSON(200, s.bot.CQGetForwardMessage(resId))
|
|
||||||
}
|
|
||||||
|
|
||||||
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)
|
|
||||||
c.JSON(200, s.bot.CQDeleteMessage(int32(mid)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func CanSendImage(s *httpServer, c *gin.Context) {
|
|
||||||
c.JSON(200, s.bot.CQCanSendImage())
|
|
||||||
}
|
|
||||||
|
|
||||||
func CanSendRecord(s *httpServer, c *gin.Context) {
|
|
||||||
c.JSON(200, s.bot.CQCanSendRecord())
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetStatus(s *httpServer, c *gin.Context) {
|
|
||||||
c.JSON(200, s.bot.CQGetStatus())
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetVersionInfo(s *httpServer, c *gin.Context) {
|
|
||||||
c.JSON(200, s.bot.CQGetVersionInfo())
|
|
||||||
}
|
|
||||||
|
|
||||||
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 GetGroupAtAllRemain(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
c.JSON(200, s.bot.CQGetAtAllRemain(gid))
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetGroupAnonymousBan(s *httpServer, c *gin.Context) {
|
|
||||||
gid, _ := strconv.ParseInt(getParam(c, "group_id"), 10, 64)
|
|
||||||
d, _ := strconv.ParseInt(getParam(c, "duration"), 10, 64)
|
|
||||||
flag := getParam(c, "flag")
|
|
||||||
if flag == "" {
|
|
||||||
flag = getParam(c, "anonymous_flag")
|
|
||||||
}
|
|
||||||
if flag == "" {
|
|
||||||
o := gjson.Parse(getParam(c, "anonymous"))
|
|
||||||
flag = o.Get("flag").String()
|
|
||||||
}
|
|
||||||
c.JSON(200, s.bot.CQSetGroupAnonymousBan(gid, flag, int32(d)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func HandleQuickOperation(s *httpServer, c *gin.Context) {
|
|
||||||
if c.Request.Method != "POST" {
|
|
||||||
c.AbortWithStatus(404)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if i, ok := c.Get("json_body"); ok {
|
|
||||||
body := i.(gjson.Result)
|
|
||||||
c.JSON(200, s.bot.CQHandleQuickOperation(body.Get("context"), body.Get("operation")))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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 {
|
|
||||||
r := getParam(c, k)
|
|
||||||
if r != "" {
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
return def
|
|
||||||
}
|
|
||||||
|
|
||||||
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, gjson.Null
|
return gjson.Result{Type: gjson.String, Str: q}
|
||||||
}
|
}
|
||||||
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 strings.Contains(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, gjson.Null
|
return gjson.Result{Type: gjson.String, Str: p}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if strings.Contains(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)
|
return obj.(gjson.Result).Get(k)
|
||||||
if res.Exists() {
|
|
||||||
switch res.Type {
|
|
||||||
case gjson.JSON:
|
|
||||||
return res.Raw, gjson.JSON
|
|
||||||
case gjson.String:
|
|
||||||
return res.Str, gjson.String
|
|
||||||
case gjson.Number:
|
|
||||||
return strconv.FormatInt(res.Int(), 10), gjson.Number // 似乎没有需要接受 float 类型的api
|
|
||||||
case gjson.True:
|
|
||||||
return "true", gjson.True
|
|
||||||
case gjson.False:
|
|
||||||
return "false", gjson.False
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return "", gjson.Null
|
return gjson.Result{Type: gjson.Null, Str: ""}
|
||||||
}
|
|
||||||
|
|
||||||
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,
|
|
||||||
"set_group_anonymous_ban": SetGroupAnonymousBan,
|
|
||||||
".handle_quick_operation": HandleQuickOperation,
|
|
||||||
".ocr_image": OcrImage,
|
|
||||||
"ocr_image": OcrImage,
|
|
||||||
"get_group_at_all_remain": GetGroupAtAllRemain,
|
|
||||||
".get_word_slices": GetWordSlices,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *httpServer) ShutDown() {
|
func (s *httpServer) ShutDown() {
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
if err := s.Http.Shutdown(ctx); err != nil {
|
if err := s.HTTP.Shutdown(ctx); err != nil {
|
||||||
log.Fatal("http Server Shutdown:", err)
|
log.Fatal("http Server Shutdown:", err)
|
||||||
}
|
}
|
||||||
select {
|
<-ctx.Done()
|
||||||
case <-ctx.Done():
|
log.Println("timeout of 5 seconds.")
|
||||||
log.Println("timeout of 5 seconds.")
|
|
||||||
}
|
|
||||||
log.Println("http Server exiting")
|
log.Println("http Server exiting")
|
||||||
}
|
}
|
||||||
|
@ -17,36 +17,39 @@ import (
|
|||||||
"github.com/tidwall/gjson"
|
"github.com/tidwall/gjson"
|
||||||
)
|
)
|
||||||
|
|
||||||
type websocketServer struct {
|
type webSocketServer struct {
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
token string
|
token string
|
||||||
eventConn []*websocketConn
|
eventConn []*webSocketConn
|
||||||
eventConnMutex sync.Mutex
|
eventConnMutex sync.Mutex
|
||||||
handshake string
|
handshake string
|
||||||
}
|
}
|
||||||
|
|
||||||
type websocketClient struct {
|
// WebSocketClient WebSocket客户端实例
|
||||||
conf *global.GoCQReverseWebsocketConfig
|
type WebSocketClient struct {
|
||||||
|
conf *global.GoCQReverseWebSocketConfig
|
||||||
token string
|
token string
|
||||||
bot *coolq.CQBot
|
bot *coolq.CQBot
|
||||||
|
|
||||||
universalConn *websocketConn
|
universalConn *webSocketConn
|
||||||
eventConn *websocketConn
|
eventConn *webSocketConn
|
||||||
}
|
}
|
||||||
|
|
||||||
type websocketConn struct {
|
type webSocketConn struct {
|
||||||
*websocket.Conn
|
*websocket.Conn
|
||||||
sync.Mutex
|
sync.Mutex
|
||||||
|
apiCaller apiCaller
|
||||||
}
|
}
|
||||||
|
|
||||||
var WebsocketServer = &websocketServer{}
|
// WebSocketServer 初始化一个WebSocketServer实例
|
||||||
|
var WebSocketServer = &webSocketServer{}
|
||||||
var upgrader = websocket.Upgrader{
|
var upgrader = websocket.Upgrader{
|
||||||
CheckOrigin: func(r *http.Request) bool {
|
CheckOrigin: func(r *http.Request) bool {
|
||||||
return true
|
return true
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
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.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}`,
|
||||||
@ -56,34 +59,36 @@ func (s *websocketServer) Run(addr, authToken string, b *coolq.CQBot) {
|
|||||||
http.HandleFunc("/api", s.api)
|
http.HandleFunc("/api", s.api)
|
||||||
http.HandleFunc("/", s.any)
|
http.HandleFunc("/", s.any)
|
||||||
go func() {
|
go func() {
|
||||||
log.Infof("CQ Websocket 服务器已启动: %v", addr)
|
log.Infof("CQ WebSocket 服务器已启动: %v", addr)
|
||||||
log.Fatal(http.ListenAndServe(addr, nil))
|
log.Fatal(http.ListenAndServe(addr, nil))
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewWebsocketClient(conf *global.GoCQReverseWebsocketConfig, authToken string, b *coolq.CQBot) *websocketClient {
|
// NewWebSocketClient 初始化一个NWebSocket客户端
|
||||||
return &websocketClient{conf: conf, token: authToken, bot: b}
|
func NewWebSocketClient(conf *global.GoCQReverseWebSocketConfig, authToken string, b *coolq.CQBot) *WebSocketClient {
|
||||||
|
return &WebSocketClient{conf: conf, token: authToken, bot: b}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) Run() {
|
// Run 运行实例
|
||||||
|
func (c *WebSocketClient) Run() {
|
||||||
if !c.conf.Enabled {
|
if !c.conf.Enabled {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if c.conf.ReverseUrl != "" {
|
if c.conf.ReverseURL != "" {
|
||||||
c.connectUniversal()
|
c.connectUniversal()
|
||||||
} else {
|
} else {
|
||||||
if c.conf.ReverseApiUrl != "" {
|
if c.conf.ReverseAPIURL != "" {
|
||||||
c.connectApi()
|
c.connectAPI()
|
||||||
}
|
}
|
||||||
if c.conf.ReverseEventUrl != "" {
|
if c.conf.ReverseEventURL != "" {
|
||||||
c.connectEvent()
|
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)
|
||||||
header := http.Header{
|
header := http.Header{
|
||||||
"X-Client-Role": []string{"API"},
|
"X-Client-Role": []string{"API"},
|
||||||
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
||||||
@ -92,22 +97,22 @@ func (c *websocketClient) connectApi() {
|
|||||||
if c.token != "" {
|
if c.token != "" {
|
||||||
header["Authorization"] = []string{"Token " + c.token}
|
header["Authorization"] = []string{"Token " + c.token}
|
||||||
}
|
}
|
||||||
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseApiUrl, header)
|
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 {
|
||||||
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
||||||
c.connectApi()
|
c.connectAPI()
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
log.Infof("已连接到反向Websocket API服务器 %v", c.conf.ReverseApiUrl)
|
log.Infof("已连接到反向WebSocket API服务器 %v", c.conf.ReverseAPIURL)
|
||||||
wrappedConn := &websocketConn{Conn: conn}
|
wrappedConn := &webSocketConn{Conn: conn, apiCaller: apiCaller{c.bot}}
|
||||||
go c.listenApi(wrappedConn, false)
|
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)
|
||||||
header := http.Header{
|
header := http.Header{
|
||||||
"X-Client-Role": []string{"Event"},
|
"X-Client-Role": []string{"Event"},
|
||||||
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
||||||
@ -116,9 +121,9 @@ func (c *websocketClient) connectEvent() {
|
|||||||
if c.token != "" {
|
if c.token != "" {
|
||||||
header["Authorization"] = []string{"Token " + c.token}
|
header["Authorization"] = []string{"Token " + c.token}
|
||||||
}
|
}
|
||||||
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseEventUrl, header)
|
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseEventURL, header)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("连接到反向Websocket Event服务器 %v 时出现错误: %v", c.conf.ReverseEventUrl, 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.connectEvent()
|
c.connectEvent()
|
||||||
@ -130,15 +135,15 @@ func (c *websocketClient) connectEvent() {
|
|||||||
c.bot.Client.Uin, time.Now().Unix())
|
c.bot.Client.Uin, time.Now().Unix())
|
||||||
err = conn.WriteMessage(websocket.TextMessage, []byte(handshake))
|
err = conn.WriteMessage(websocket.TextMessage, []byte(handshake))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("反向Websocket 握手时出现错误: %v", err)
|
log.Warnf("反向WebSocket 握手时出现错误: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Infof("已连接到反向Websocket Event服务器 %v", c.conf.ReverseEventUrl)
|
log.Infof("已连接到反向WebSocket Event服务器 %v", c.conf.ReverseEventURL)
|
||||||
c.eventConn = &websocketConn{Conn: conn}
|
c.eventConn = &webSocketConn{Conn: conn, apiCaller: apiCaller{c.bot}}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) connectUniversal() {
|
func (c *WebSocketClient) connectUniversal() {
|
||||||
log.Infof("开始尝试连接到反向Websocket Universal服务器: %v", c.conf.ReverseUrl)
|
log.Infof("开始尝试连接到反向WebSocket Universal服务器: %v", c.conf.ReverseURL)
|
||||||
header := http.Header{
|
header := http.Header{
|
||||||
"X-Client-Role": []string{"Universal"},
|
"X-Client-Role": []string{"Universal"},
|
||||||
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
"X-Self-ID": []string{strconv.FormatInt(c.bot.Client.Uin, 10)},
|
||||||
@ -147,9 +152,9 @@ func (c *websocketClient) connectUniversal() {
|
|||||||
if c.token != "" {
|
if c.token != "" {
|
||||||
header["Authorization"] = []string{"Token " + c.token}
|
header["Authorization"] = []string{"Token " + c.token}
|
||||||
}
|
}
|
||||||
conn, _, err := websocket.DefaultDialer.Dial(c.conf.ReverseUrl, header)
|
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 {
|
||||||
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
time.Sleep(time.Millisecond * time.Duration(c.conf.ReverseReconnectInterval))
|
||||||
c.connectUniversal()
|
c.connectUniversal()
|
||||||
@ -161,15 +166,15 @@ func (c *websocketClient) connectUniversal() {
|
|||||||
c.bot.Client.Uin, time.Now().Unix())
|
c.bot.Client.Uin, time.Now().Unix())
|
||||||
err = conn.WriteMessage(websocket.TextMessage, []byte(handshake))
|
err = conn.WriteMessage(websocket.TextMessage, []byte(handshake))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("反向Websocket 握手时出现错误: %v", err)
|
log.Warnf("反向WebSocket 握手时出现错误: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
wrappedConn := &websocketConn{Conn: conn}
|
wrappedConn := &webSocketConn{Conn: conn, apiCaller: apiCaller{c.bot}}
|
||||||
go c.listenApi(wrappedConn, true)
|
go c.listenAPI(wrappedConn, true)
|
||||||
c.universalConn = wrappedConn
|
c.universalConn = wrappedConn
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) listenApi(conn *websocketConn, u bool) {
|
func (c *WebSocketClient) listenAPI(conn *webSocketConn, u bool) {
|
||||||
defer conn.Close()
|
defer conn.Close()
|
||||||
for {
|
for {
|
||||||
_, buf, err := conn.ReadMessage()
|
_, buf, err := conn.ReadMessage()
|
||||||
@ -184,14 +189,14 @@ func (c *websocketClient) listenApi(conn *websocketConn, u bool) {
|
|||||||
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 {
|
||||||
go c.connectApi()
|
go c.connectAPI()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketClient) onBotPushEvent(m coolq.MSG) {
|
func (c *WebSocketClient) onBotPushEvent(m coolq.MSG) {
|
||||||
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())
|
||||||
conn := c.eventConn
|
conn := c.eventConn
|
||||||
conn.Lock()
|
conn.Lock()
|
||||||
defer conn.Unlock()
|
defer conn.Unlock()
|
||||||
@ -206,7 +211,7 @@ 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())
|
||||||
conn := c.universalConn
|
conn := c.universalConn
|
||||||
conn.Lock()
|
conn.Lock()
|
||||||
defer conn.Unlock()
|
defer conn.Unlock()
|
||||||
@ -222,11 +227,11 @@ func (c *websocketClient) onBotPushEvent(m coolq.MSG) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
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 auth := r.URL.Query().Get("access_token"); auth != s.token {
|
if auth := r.URL.Query().Get("access_token"); auth != s.token {
|
||||||
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
||||||
log.Warnf("已拒绝 %v 的 Websocket 请求: Token鉴权失败", r.RemoteAddr)
|
log.Warnf("已拒绝 %v 的 WebSocket 请求: Token鉴权失败", r.RemoteAddr)
|
||||||
w.WriteHeader(401)
|
w.WriteHeader(401)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -234,30 +239,30 @@ func (s *websocketServer) event(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
c, err := upgrader.Upgrade(w, r, nil)
|
c, err := upgrader.Upgrade(w, r, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("处理 Websocket 请求时出现错误: %v", err)
|
log.Warnf("处理 WebSocket 请求时出现错误: %v", err)
|
||||||
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.Warnf("Websocket 握手时出现错误: %v", err)
|
log.Warnf("WebSocket 握手时出现错误: %v", err)
|
||||||
c.Close()
|
c.Close()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Infof("接受 Websocket 连接: %v (/event)", r.RemoteAddr)
|
log.Infof("接受 WebSocket 连接: %v (/event)", r.RemoteAddr)
|
||||||
|
|
||||||
conn := &websocketConn{Conn: c}
|
conn := &webSocketConn{Conn: c, apiCaller: apiCaller{s.bot}}
|
||||||
|
|
||||||
s.eventConnMutex.Lock()
|
s.eventConnMutex.Lock()
|
||||||
s.eventConn = append(s.eventConn, conn)
|
s.eventConn = append(s.eventConn, conn)
|
||||||
s.eventConnMutex.Unlock()
|
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 auth := r.URL.Query().Get("access_token"); auth != s.token {
|
if auth := r.URL.Query().Get("access_token"); auth != s.token {
|
||||||
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
||||||
log.Warnf("已拒绝 %v 的 Websocket 请求: Token鉴权失败", r.RemoteAddr)
|
log.Warnf("已拒绝 %v 的 WebSocket 请求: Token鉴权失败", r.RemoteAddr)
|
||||||
w.WriteHeader(401)
|
w.WriteHeader(401)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -265,19 +270,19 @@ func (s *websocketServer) api(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
c, err := upgrader.Upgrade(w, r, nil)
|
c, err := upgrader.Upgrade(w, r, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("处理 Websocket 请求时出现错误: %v", err)
|
log.Warnf("处理 WebSocket 请求时出现错误: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
log.Infof("接受 Websocket 连接: %v (/api)", r.RemoteAddr)
|
log.Infof("接受 WebSocket 连接: %v (/api)", r.RemoteAddr)
|
||||||
conn := &websocketConn{Conn: c}
|
conn := &webSocketConn{Conn: c, apiCaller: apiCaller{s.bot}}
|
||||||
go s.listenApi(conn)
|
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 auth := r.URL.Query().Get("access_token"); auth != s.token {
|
if auth := r.URL.Query().Get("access_token"); auth != s.token {
|
||||||
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
if auth := strings.SplitN(r.Header.Get("Authorization"), " ", 2); len(auth) != 2 || auth[1] != s.token {
|
||||||
log.Warnf("已拒绝 %v 的 Websocket 请求: Token鉴权失败", r.RemoteAddr)
|
log.Warnf("已拒绝 %v 的 WebSocket 请求: Token鉴权失败", r.RemoteAddr)
|
||||||
w.WriteHeader(401)
|
w.WriteHeader(401)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -285,22 +290,22 @@ func (s *websocketServer) any(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
c, err := upgrader.Upgrade(w, r, nil)
|
c, err := upgrader.Upgrade(w, r, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("处理 Websocket 请求时出现错误: %v", err)
|
log.Warnf("处理 WebSocket 请求时出现错误: %v", err)
|
||||||
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.Warnf("Websocket 握手时出现错误: %v", err)
|
log.Warnf("WebSocket 握手时出现错误: %v", err)
|
||||||
c.Close()
|
c.Close()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
log.Infof("接受 Websocket 连接: %v (/)", r.RemoteAddr)
|
log.Infof("接受 WebSocket 连接: %v (/)", r.RemoteAddr)
|
||||||
conn := &websocketConn{Conn: c}
|
conn := &webSocketConn{Conn: c, apiCaller: apiCaller{s.bot}}
|
||||||
s.eventConn = append(s.eventConn, conn)
|
s.eventConn = append(s.eventConn, conn)
|
||||||
s.listenApi(conn)
|
s.listenAPI(conn)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) listenApi(c *websocketConn) {
|
func (s *webSocketServer) listenAPI(c *webSocketConn) {
|
||||||
defer c.Close()
|
defer c.Close()
|
||||||
for {
|
for {
|
||||||
t, payload, err := c.ReadMessage()
|
t, payload, err := c.ReadMessage()
|
||||||
@ -314,7 +319,7 @@ func (s *websocketServer) listenApi(c *websocketConn) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *websocketConn) handleRequest(bot *coolq.CQBot, payload []byte) {
|
func (c *webSocketConn) handleRequest(_ *coolq.CQBot, payload []byte) {
|
||||||
defer func() {
|
defer func() {
|
||||||
if err := recover(); err != nil {
|
if err := recover(); err != nil {
|
||||||
log.Printf("处置WS命令时发生无法恢复的异常:%v\n%s", err, debug.Stack())
|
log.Printf("处置WS命令时发生无法恢复的异常:%v\n%s", err, debug.Stack())
|
||||||
@ -325,33 +330,23 @@ func (c *websocketConn) handleRequest(bot *coolq.CQBot, payload []byte) {
|
|||||||
j := gjson.ParseBytes(payload)
|
j := gjson.ParseBytes(payload)
|
||||||
t := strings.ReplaceAll(j.Get("action").Str, "_async", "")
|
t := strings.ReplaceAll(j.Get("action").Str, "_async", "")
|
||||||
log.Debugf("WS接收到API调用: %v 参数: %v", t, j.Get("params").Raw)
|
log.Debugf("WS接收到API调用: %v 参数: %v", t, j.Get("params").Raw)
|
||||||
if f, ok := wsApi[t]; ok {
|
ret := c.apiCaller.callAPI(t, j.Get("params"))
|
||||||
ret := f(bot, j.Get("params"))
|
if j.Get("echo").Exists() {
|
||||||
if j.Get("echo").Exists() {
|
ret["echo"] = j.Get("echo").Value()
|
||||||
ret["echo"] = j.Get("echo").Value()
|
|
||||||
}
|
|
||||||
c.Lock()
|
|
||||||
defer c.Unlock()
|
|
||||||
_ = c.WriteJSON(ret)
|
|
||||||
} else {
|
|
||||||
ret := coolq.Failed(1404, "API_NOT_FOUND", "API不存在")
|
|
||||||
if j.Get("echo").Exists() {
|
|
||||||
ret["echo"] = j.Get("echo").Value()
|
|
||||||
}
|
|
||||||
c.Lock()
|
|
||||||
defer c.Unlock()
|
|
||||||
_ = c.WriteJSON(ret)
|
|
||||||
}
|
}
|
||||||
|
c.Lock()
|
||||||
|
defer c.Unlock()
|
||||||
|
_ = c.WriteJSON(ret)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *websocketServer) onBotPushEvent(m coolq.MSG) {
|
func (s *webSocketServer) onBotPushEvent(m coolq.MSG) {
|
||||||
s.eventConnMutex.Lock()
|
s.eventConnMutex.Lock()
|
||||||
defer s.eventConnMutex.Unlock()
|
defer s.eventConnMutex.Unlock()
|
||||||
for i, l := 0, len(s.eventConn); i < l; i++ {
|
for i, l := 0, len(s.eventConn); i < l; i++ {
|
||||||
conn := s.eventConn[i]
|
conn := s.eventConn[i]
|
||||||
log.Debugf("向WS客户端 %v 推送Event: %v", conn.RemoteAddr().String(), m.ToJson())
|
log.Debugf("向WS客户端 %v 推送Event: %v", conn.RemoteAddr().String(), m.ToJSON())
|
||||||
conn.Lock()
|
conn.Lock()
|
||||||
if err := conn.WriteMessage(websocket.TextMessage, []byte(m.ToJson())); err != nil {
|
if err := conn.WriteMessage(websocket.TextMessage, []byte(m.ToJSON())); err != nil {
|
||||||
_ = conn.Close()
|
_ = conn.Close()
|
||||||
next := i + 1
|
next := i + 1
|
||||||
if next >= l {
|
if next >= l {
|
||||||
@ -367,211 +362,3 @@ func (s *websocketServer) onBotPushEvent(m coolq.MSG) {
|
|||||||
conn.Unlock()
|
conn.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var wsApi = map[string]func(*coolq.CQBot, gjson.Result) coolq.MSG{
|
|
||||||
"get_login_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQGetLoginInfo()
|
|
||||||
},
|
|
||||||
"get_friend_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQGetFriendList()
|
|
||||||
},
|
|
||||||
"get_group_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQGetGroupList(p.Get("no_cache").Bool())
|
|
||||||
},
|
|
||||||
"get_group_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQGetGroupInfo(p.Get("group_id").Int(), p.Get("no_cache").Bool())
|
|
||||||
},
|
|
||||||
"get_group_member_list": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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 {
|
|
||||||
return bot.CQGetGroupMemberInfo(
|
|
||||||
p.Get("group_id").Int(), p.Get("user_id").Int(),
|
|
||||||
)
|
|
||||||
},
|
|
||||||
"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 {
|
|
||||||
return bot.CQSendGroupMessage(p.Get("group_id").Int(), p.Get("message"), autoEscape)
|
|
||||||
}
|
|
||||||
if p.Get("user_id").Int() != 0 {
|
|
||||||
return bot.CQSendPrivateMessage(p.Get("user_id").Int(), p.Get("message"), autoEscape)
|
|
||||||
}
|
|
||||||
return 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"), global.EnsureBool(p.Get("auto_escape"), false))
|
|
||||||
},
|
|
||||||
"send_group_forward_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQSendGroupForwardMessage(p.Get("group_id").Int(), p.Get("messages"))
|
|
||||||
},
|
|
||||||
"send_private_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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 {
|
|
||||||
return bot.CQDeleteMessage(int32(p.Get("message_id").Int()))
|
|
||||||
},
|
|
||||||
"set_friend_add_request": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
apr := true
|
|
||||||
if p.Get("approve").Exists() {
|
|
||||||
apr = p.Get("approve").Bool()
|
|
||||||
}
|
|
||||||
return bot.CQProcessFriendRequest(p.Get("flag").Str, apr)
|
|
||||||
},
|
|
||||||
"set_group_add_request": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
subType := p.Get("sub_type").Str
|
|
||||||
apr := true
|
|
||||||
if subType == "" {
|
|
||||||
subType = p.Get("type").Str
|
|
||||||
}
|
|
||||||
if p.Get("approve").Exists() {
|
|
||||||
apr = p.Get("approve").Bool()
|
|
||||||
}
|
|
||||||
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 {
|
|
||||||
return bot.CQSetGroupCard(p.Get("group_id").Int(), p.Get("user_id").Int(), p.Get("card").Str)
|
|
||||||
},
|
|
||||||
"set_group_special_title": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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 {
|
|
||||||
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 {
|
|
||||||
return bot.CQSetGroupBan(p.Get("group_id").Int(), p.Get("user_id").Int(), func() uint32 {
|
|
||||||
if p.Get("duration").Exists() {
|
|
||||||
return uint32(p.Get("duration").Int())
|
|
||||||
}
|
|
||||||
return 1800
|
|
||||||
}())
|
|
||||||
},
|
|
||||||
"set_group_whole_ban": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQSetGroupWholeBan(p.Get("group_id").Int(), func() bool {
|
|
||||||
if p.Get("enable").Exists() {
|
|
||||||
return p.Get("enable").Bool()
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}())
|
|
||||||
},
|
|
||||||
"set_group_name": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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 {
|
|
||||||
return bot.CQSetGroupLeave(p.Get("group_id").Int())
|
|
||||||
},
|
|
||||||
"get_image": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQGetImage(p.Get("file").Str)
|
|
||||||
},
|
|
||||||
"get_forward_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
id := p.Get("message_id").Str
|
|
||||||
if id == "" {
|
|
||||||
id = p.Get("id").Str
|
|
||||||
}
|
|
||||||
return bot.CQGetForwardMessage(id)
|
|
||||||
},
|
|
||||||
"get_msg": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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 {
|
|
||||||
return bot.CQCanSendImage()
|
|
||||||
},
|
|
||||||
"can_send_record": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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 {
|
|
||||||
return bot.CQGetStatus()
|
|
||||||
},
|
|
||||||
"get_version_info": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
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)
|
|
||||||
},
|
|
||||||
"ocr_image": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQOcrImage(p.Get("image").Str)
|
|
||||||
},
|
|
||||||
"get_group_at_all_remain": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQGetAtAllRemain(p.Get("group_id").Int())
|
|
||||||
},
|
|
||||||
".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())
|
|
||||||
},
|
|
||||||
"set_group_anonymous_ban": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
obj := p.Get("anonymous")
|
|
||||||
flag := p.Get("anonymous_flag")
|
|
||||||
if !flag.Exists() {
|
|
||||||
flag = p.Get("flag")
|
|
||||||
}
|
|
||||||
if !flag.Exists() && !obj.Exists() {
|
|
||||||
return coolq.Failed(100, "FLAG_NOT_FOUND", "flag未找到")
|
|
||||||
}
|
|
||||||
if !flag.Exists() {
|
|
||||||
flag = obj.Get("flag")
|
|
||||||
}
|
|
||||||
return bot.CQSetGroupAnonymousBan(p.Get("group_id").Int(), flag.String(), int32(p.Get("duration").Int()))
|
|
||||||
},
|
|
||||||
".handle_quick_operation": func(bot *coolq.CQBot, p gjson.Result) coolq.MSG {
|
|
||||||
return bot.CQHandleQuickOperation(p.Get("context"), p.Get("operation"))
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
Reference in New Issue
Block a user