Skip to content

Commit fffcb34

Browse files
committed
feat: supported lark custom bot webhook
1 parent 3c77fd5 commit fffcb34

File tree

8 files changed

+465
-0
lines changed

8 files changed

+465
-0
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919

2020
*.out
2121
.idea
22+
.vscode
2223

2324
report.json
2425
coverage.txt

docs/services/lark.md

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
# Lark
2+
3+
## URL Format
4+
5+
!!! info ""
6+
lark://__`host`__/__`token`__?[secret=__`secret`__]
7+
8+
--8<-- "docs/services/lark/config.md"
9+
10+
## Create Custom Bot in Lark
11+
12+
Official Documents: [Link](https://open.larksuite.com/document/client-docs/bot-v3/add-custom-bot)
13+
14+
1. Invite custom bot join group.
15+
16+
a. Enter the target group, click the `More` button in the upper right corner of the group, and then click `Settings`.
17+
18+
b. On the right-side `Settings`, click on `Group Bot`.
19+
20+
c. Click `Add a Bot` on the `Group Bot`.
21+
22+
b. In `Add Bot` dialog box, find the `Custom Bot` and add it.
23+
24+
e. Set the name and description of the custom robot, and click `Add`.
25+
26+
2. Get the webhook address of the custom robot and click `Finish`.
27+
28+
## Get Host and Token of Custom Bot
29+
30+
If you are using `Lark`, then the `Host` is `open.larksuite.com`.
31+
32+
If you are using `Feishu` or `飞书`, then the `Host` is `open.feishu.cn`.
33+
34+
`Token` is the last part of the webhook address. For example, if the webhook address is `https://open.larksuite.com/open-apis/bot/v2/hook/xxxxxxxxxxxxxxxxx`, then the token corresponds to `xxxxxxxxxxxxxxxxx`.
35+
36+
## Get Secret of Custom Bot
37+
38+
1. In the group settings, open the bot list, find the custom bot and click on it to enter the configuration page.
39+
40+
2. In the `Security Settings`, select `Signature Verification`.
41+
42+
3. Click `Copy` to copy the secret.
43+
44+
4. Click `Save` to make the configuration take effect.

docs/services/overview.md

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ Click on the service for a more thorough explanation. <!-- @formatter:off -->
2222
| [Teams](./teams.md) | *teams://__`group`__@__`tenant`__/__`altId`__/__`groupOwner`__?host=__`organization`__.webhook.office.com* |
2323
| [Telegram](./telegram.md) | *telegram://__`token`__@telegram?chats=__`@channel-1`__[,__`chat-id-1`__,...]* |
2424
| [Zulip Chat](./zulip.md) | *zulip://__`bot-mail`__:__`bot-key`__@__`zulip-domain`__/?stream=__`name-or-id`__&topic=__`name`__* |
25+
| [Lark](./lark.md) | *lark://__`host`__/__`token`__?secret=__`secret`__* |
2526

2627
## Specialized services
2728

pkg/router/servicemap.go

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import (
88
"github.com/containrrr/shoutrrr/pkg/services/gotify"
99
"github.com/containrrr/shoutrrr/pkg/services/ifttt"
1010
"github.com/containrrr/shoutrrr/pkg/services/join"
11+
"github.com/containrrr/shoutrrr/pkg/services/lark"
1112
"github.com/containrrr/shoutrrr/pkg/services/logger"
1213
"github.com/containrrr/shoutrrr/pkg/services/matrix"
1314
"github.com/containrrr/shoutrrr/pkg/services/mattermost"
@@ -46,4 +47,5 @@ var serviceMap = map[string]func() t.Service{
4647
"teams": func() t.Service { return &teams.Service{} },
4748
"telegram": func() t.Service { return &telegram.Service{} },
4849
"zulip": func() t.Service { return &zulip.Service{} },
50+
"lark": func() t.Service { return &lark.Service{} },
4951
}

pkg/services/lark/lark.go

+145
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,145 @@
1+
package lark
2+
3+
import (
4+
"bytes"
5+
"crypto/hmac"
6+
"crypto/sha256"
7+
"encoding/base64"
8+
"encoding/json"
9+
"errors"
10+
"fmt"
11+
"io"
12+
"net/http"
13+
"net/url"
14+
"strconv"
15+
"time"
16+
17+
"github.com/containrrr/shoutrrr/pkg/format"
18+
"github.com/containrrr/shoutrrr/pkg/services/standard"
19+
"github.com/containrrr/shoutrrr/pkg/types"
20+
)
21+
22+
const (
23+
apiFormat = "https://%s/open-apis/bot/v2/hook/%s"
24+
maxLength = 4096
25+
defaultTime = 30 * time.Second
26+
)
27+
28+
type Service struct {
29+
standard.Standard
30+
config *Config
31+
pkr format.PropKeyResolver
32+
}
33+
34+
var (
35+
ErrInvalidHost = errors.New("invalid host, use 'open.larksuite.com' or 'open.feishu.cn'")
36+
ErrNoPath = errors.New("no path, path like 'xxx' in 'https://open.larksuite.com/open-apis/bot/v2/hook/xxx'")
37+
ErrLargeMessage = errors.New("message exceeds the max length")
38+
39+
httpClient = &http.Client{Timeout: defaultTime}
40+
)
41+
42+
const (
43+
larkHost = "open.larksuite.com"
44+
feishuHost = "open.feishu.com"
45+
)
46+
47+
// Send notification to Lark
48+
func (service *Service) Send(message string, params *types.Params) error {
49+
if len(message) > maxLength {
50+
return ErrLargeMessage
51+
}
52+
53+
config := *service.config
54+
if err := service.pkr.UpdateConfigFromParams(&config, params); err != nil {
55+
return err
56+
}
57+
58+
if config.Host != larkHost && config.Host != feishuHost {
59+
return ErrInvalidHost
60+
}
61+
62+
if config.Path == "" {
63+
return ErrNoPath
64+
}
65+
66+
return service.sendMessage(message, config)
67+
}
68+
69+
// Initialize loads ServiceConfig from configURL and sets logger for this Service
70+
func (service *Service) Initialize(configURL *url.URL, logger types.StdLogger) error {
71+
service.Logger.SetLogger(logger)
72+
service.config = &Config{}
73+
service.pkr = format.NewPropKeyResolver(service.config)
74+
if err := service.config.setURL(&service.pkr, configURL); err != nil {
75+
return err
76+
}
77+
return nil
78+
}
79+
80+
func (service *Service) genSign(secret string, timestamp int64) string {
81+
//timestamp + key calculate sha256, then base64 encode
82+
stringToSign := fmt.Sprintf("%v", timestamp) + "\n" + secret
83+
84+
var data []byte
85+
h := hmac.New(sha256.New, []byte(stringToSign))
86+
h.Write(data)
87+
signature := base64.StdEncoding.EncodeToString(h.Sum(nil))
88+
return signature
89+
}
90+
91+
func (service *Service) sendMessage(message string, cfg Config) error {
92+
url := fmt.Sprintf(apiFormat, cfg.Host, cfg.Path)
93+
body := service.getRequestBody(message, cfg.Title, cfg.Secret)
94+
data, err := json.Marshal(body)
95+
if err != nil {
96+
return err
97+
}
98+
service.Logf("Lark Request Body: %s", string(data))
99+
req, err := http.NewRequest(http.MethodPost, url, bytes.NewReader(data))
100+
if err != nil {
101+
return err
102+
}
103+
req.Header.Set("Content-Type", "application/json")
104+
resp, err := httpClient.Do(req)
105+
if err != nil {
106+
return err
107+
}
108+
defer resp.Body.Close()
109+
data, err = io.ReadAll(resp.Body)
110+
if err != nil {
111+
return err
112+
}
113+
response := Response{}
114+
if err := json.Unmarshal(data, &response); err != nil {
115+
return err
116+
}
117+
if response.Code == 0 {
118+
return nil
119+
}
120+
return fmt.Errorf("code: %d, msg: %s", response.Code, response.Msg)
121+
}
122+
123+
func (service *Service) getRequestBody(message, title, secret string) *RequestBody {
124+
body := &RequestBody{}
125+
if secret != "" {
126+
ts := time.Now().Unix()
127+
body.Timestamp = strconv.FormatInt(ts, 10)
128+
body.Sign = service.genSign(secret, ts)
129+
}
130+
if title == "" {
131+
body.MsgType = MsgTypeText
132+
body.Content.Text = message
133+
return body
134+
}
135+
body.MsgType = MsgTypePost
136+
body.Content.Post = &Post{
137+
En: &Message{
138+
Title: title,
139+
Content: [][]Item{{
140+
{Tag: TagValueText, Text: message},
141+
}},
142+
},
143+
}
144+
return body
145+
}

pkg/services/lark/lark_config.go

+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
package lark
2+
3+
import (
4+
"net/url"
5+
"strings"
6+
7+
"github.com/containrrr/shoutrrr/pkg/format"
8+
"github.com/containrrr/shoutrrr/pkg/types"
9+
)
10+
11+
const Scheme = "lark"
12+
13+
type Config struct {
14+
Host string `desc:"Custom bot URL Host" default:"open.larksuite.com" url:"Host"`
15+
Secret string `desc:"Custom bot secret" default:"" key:"secret"`
16+
Path string `desc:"Custom bot token" url:"Path"`
17+
Title string `desc:"Message Title" default:"" key:"title"`
18+
}
19+
20+
func (config *Config) Enums() map[string]types.EnumFormatter {
21+
return map[string]types.EnumFormatter{}
22+
}
23+
24+
func (config *Config) GetURL() *url.URL {
25+
resolver := format.NewPropKeyResolver(config)
26+
return config.getURL(&resolver)
27+
}
28+
29+
func (config *Config) getURL(resolver types.ConfigQueryResolver) *url.URL {
30+
31+
return &url.URL{
32+
Host: config.Host,
33+
Path: "/" + config.Path,
34+
Scheme: Scheme,
35+
ForceQuery: true,
36+
RawQuery: format.BuildQuery(resolver),
37+
}
38+
39+
}
40+
41+
func (config *Config) SetURL(url *url.URL) error {
42+
resolver := format.NewPropKeyResolver(config)
43+
return config.setURL(&resolver, url)
44+
}
45+
46+
func (config *Config) setURL(resolver types.ConfigQueryResolver, url *url.URL) error {
47+
48+
config.Host = url.Host
49+
config.Path = strings.Trim(url.Path, "/")
50+
// config.Password = url.Hostname()
51+
52+
for key, vals := range url.Query() {
53+
if err := resolver.Set(key, vals[0]); err != nil {
54+
return err
55+
}
56+
}
57+
58+
return nil
59+
}

pkg/services/lark/lark_message.go

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
package lark
2+
3+
type RequestBody struct {
4+
MsgType MsgType `json:"msg_type"`
5+
Content Content `json:"content"`
6+
Timestamp string `json:"timestamp,omitempty"`
7+
Sign string `json:"sign,omitempty"`
8+
}
9+
10+
type MsgType string
11+
12+
const (
13+
MsgTypeText MsgType = "text"
14+
MsgTypePost MsgType = "post"
15+
)
16+
17+
type Content struct {
18+
Text string `json:"text,omitempty"`
19+
Post *Post `json:"post,omitempty"`
20+
}
21+
22+
type Post struct {
23+
Zh *Message `json:"zh_cn,omitempty"`
24+
En *Message `json:"en_us,omitempty"`
25+
}
26+
27+
type Message struct {
28+
Title string `json:"title"`
29+
Content [][]Item `json:"content"`
30+
}
31+
32+
type Item struct {
33+
Tag TagValue `json:"tag"`
34+
Text string `json:"text,omitempty"`
35+
Link string `json:"href,omitempty"`
36+
}
37+
38+
type TagValue string
39+
40+
const (
41+
TagValueText TagValue = "text"
42+
TagValueLink TagValue = "a"
43+
)
44+
45+
type Response struct {
46+
Code int `json:"code"`
47+
Msg string `json:"msg"`
48+
Data any `json:"data"`
49+
}

0 commit comments

Comments
 (0)