Skip to content

Commit 008d53e

Browse files
authored
Merge pull request #47 from jumpserver/dev
merge dev
2 parents f28d087 + 7988189 commit 008d53e

File tree

10 files changed

+326
-20
lines changed

10 files changed

+326
-20
lines changed

go-client/Makefile

+2-1
Original file line numberDiff line numberDiff line change
@@ -25,4 +25,5 @@ build-client:
2525
cp -R $(BASEPATH)/Scripts $(BUILDDIR)/darwin
2626
cp -R $(BUILDDIR)/* $(BASEPATH)/../interface/bin/
2727
cp $(BASEPATH)/config.json $(BASEPATH)/../interface/bin/
28-
cp $(BASEPATH)/putty.exe $(BASEPATH)/../interface/bin/windows/
28+
cp $(BASEPATH)/putty.exe $(BASEPATH)/../interface/bin/windows/
29+
cp $(BASEPATH)/pkg/autoit/*.dll $(BASEPATH)/../interface/bin/windows/

go-client/config.json

+35-3
Original file line numberDiff line numberDiff line change
@@ -227,18 +227,50 @@
227227
},
228228
{
229229
"name": "ssms17",
230-
"display_name": "SQL Server Management Studio 17",
230+
"display_name": "SQL Server Management Studio",
231231
"protocol": [
232232
"sqlserver"
233233
],
234234
"comment": {
235235
"zh": "SQL Server Management Studio (SSMS) 是一种集成环境,用于管理从 SQL Server 到 Azure SQL 数据库的任何 SQL 基础结构。",
236236
"en": "SQL Server Management Studio (SSMS) is an integrated environment for managing any SQL infrastructure, from SQL Server to Azure SQL Database."
237237
},
238-
"download_url": "https://download.microsoft.com/download/D/D/4/DD495084-ADA7-4827-ADD3-FC566EC05B90/SSMS-Setup-CHS.exe",
238+
"download_url": "https://download.microsoft.com/download/9/f/8/9f8197f4-0f71-42a3-8717-b2817c77b820/SSMS-Setup-CHS.exe",
239239
"type": "databases",
240240
"path": "",
241-
"arg_format": "-S {host},{port} -U {username} -P {value} -d {dbname}",
241+
"arg_format": "",
242+
"autoit": [
243+
{
244+
"cmd": "Wait",
245+
"type": "sleep",
246+
"element": "[REGEXPTITLE:连接到服务器|Connect]"
247+
},
248+
{
249+
"cmd": "ControlSend",
250+
"type": "{host},{port}",
251+
"element": "[CLASS:Edit; INSTANCE:1]"
252+
},
253+
{
254+
"cmd": "ControlSend",
255+
"type": "SQL Server",
256+
"element": "[NAME:comboBoxAuthentication]"
257+
},
258+
{
259+
"cmd": "ControlSend",
260+
"type": "{username}",
261+
"element": "[CLASS:Edit; INSTANCE:2]"
262+
},
263+
{
264+
"cmd": "ControlSend",
265+
"type": "{value}",
266+
"element": "[NAME:password]"
267+
},
268+
{
269+
"cmd": "ControlClick",
270+
"type": "connect",
271+
"element": "[NAME:connect]"
272+
}
273+
],
242274
"match_first": [],
243275
"is_internal": false,
244276
"is_default": false,

go-client/go.sum

+19
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,38 @@
1+
github.com/adrianriobo/goautoit v0.0.0-20210415141646-fc37def5a975 h1:3k1SNJREvY7Vb068kqfS4xXPJVIPhBXUBUHpyvV8W/w=
2+
github.com/adrianriobo/goautoit v0.0.0-20210415141646-fc37def5a975/go.mod h1:tKnCmOA6c/aHEc/rnnRjPMVG3YfAfCMMMuYbW8caej4=
13
github.com/benbjohnson/clock v1.1.0 h1:Q92kusRqC1XV2MjkWETPvjJVqKetz1OzxZB7mHJLju8=
24
github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA=
5+
github.com/cpuguy83/go-md2man/v2 v2.0.1 h1:r/myEWzV9lfsM1tFLgDyu0atFtJ1fXn261LKYj/3DxU=
36
github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
47
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
58
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
69
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
710
github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM=
811
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
12+
github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI=
913
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
14+
github.com/kr/pty v1.1.1 h1:VkoXIwSboBpnk99O/KFauAEILuNHv5DVFKZMBN/gUgw=
1015
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
16+
github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
1117
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
1218
github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I=
1319
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
1420
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
1521
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
22+
github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk=
1623
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
24+
github.com/shadow1163/goautoit v0.0.0-20190627080114-4aa9ce277b8b h1:y3S4bf/aB15R/hYmPzhPQ43T/ae8EUFmpq1HB1SRfuo=
25+
github.com/shadow1163/goautoit v0.0.0-20190627080114-4aa9ce277b8b/go.mod h1:W/p3CvwSrZFUdtEMhQn6+8Vj1LPQP3u43psKeB6zgYk=
1726
github.com/spf13/cobra v1.4.0 h1:y+wJpx64xcgO1V+RcnwW0LEHxTKRi2ZDPSBjWnrg88Q=
1827
github.com/spf13/cobra v1.4.0/go.mod h1:Wo4iy3BUC+X2Fybo0PDqwJIv3dNRiZLHQymsfxlB84g=
1928
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
2029
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
30+
github.com/stretchr/objx v0.1.0 h1:4G4v2dO3VZwixGIRoQ5Lfboy6nUhCyYzaqnIAPPhYs4=
2131
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
2232
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
2333
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
2434
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
35+
github.com/yuin/goldmark v1.3.5 h1:dPmz1Snjq0kmkz159iL7S6WzdahUTHnHB5M56WFVifs=
2536
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
2637
go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
2738
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
@@ -35,14 +46,18 @@ golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACk
3546
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
3647
golang.org/x/crypto v0.0.0-20220321153916-2c7772ba3064 h1:S25/rfnfsMVgORT4/J61MJ7rdyseOZOyvLIrZEZ7s6s=
3748
golang.org/x/crypto v0.0.0-20220321153916-2c7772ba3064/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
49+
golang.org/x/lint v0.0.0-20190930215403-16217165b5de h1:5hukYrvBGR8/eNkX5mdUezrA6JiaEZDtJb9Ei+1LlBs=
3850
golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
51+
golang.org/x/mod v0.4.2 h1:Gz96sIWK3OalVv/I/qNygP42zyoKp3xptRVCWRFEBvo=
3952
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
4053
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
4154
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
4255
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
4356
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM=
57+
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2 h1:CIJ76btIcR3eFI5EgSo6k1qKw9KJexJuRLI9G7Hp5wE=
4458
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
4559
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
60+
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ=
4661
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
4762
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
4863
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
@@ -56,15 +71,19 @@ golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1 h1:v+OssWQX+hTHEmOBgwxdZxK4
5671
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
5772
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
5873
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
74+
golang.org/x/text v0.3.6 h1:aRYxNxv6iGQlyVaZmk6ZgYEDa+Jg18DxebPSrd6bg1M=
5975
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
6076
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
6177
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
6278
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
79+
golang.org/x/tools v0.1.5 h1:ouewzE6p+/VEB31YYnTbEJdi8pFqKp4P4n85vwo3DHA=
6380
golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
6481
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
6582
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
83+
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE=
6684
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
6785
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
86+
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY=
6887
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
6988
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
7089
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=

go-client/pkg/autoit/AutoItX3.dll

710 KB
Binary file not shown.

go-client/pkg/autoit/AutoItX3_x64.dll

899 KB
Binary file not shown.

go-client/pkg/autoit/goautoit.go

+221
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,221 @@
1+
//go:build windows
2+
// +build windows
3+
4+
package autoit
5+
6+
import (
7+
"os"
8+
"path/filepath"
9+
"syscall"
10+
"unsafe"
11+
)
12+
13+
const (
14+
SWShowNormal = 1
15+
INTDEFAULT = -2147483647
16+
DefaultMouseButton = "left"
17+
)
18+
19+
// HWND -- window handle
20+
type HWND uintptr
21+
22+
// RECT -- http://msdn.microsoft.com/en-us/library/windows/desktop/dd162897.aspx
23+
type RECT struct {
24+
Left, Top, Right, Bottom int32
25+
}
26+
27+
// POINT --
28+
type POINT struct {
29+
X, Y int32
30+
}
31+
32+
var (
33+
dll64 *syscall.LazyDLL
34+
controlClick *syscall.LazyProc
35+
controlSend *syscall.LazyProc
36+
run *syscall.LazyProc
37+
winActivate *syscall.LazyProc
38+
winWait *syscall.LazyProc
39+
)
40+
41+
func LoadAuto() {
42+
autoItX3 := "AutoItX3_x64.dll"
43+
if (^uint(0) >> 63) == 0 {
44+
autoItX3 = "AutoItX3.dll"
45+
}
46+
dll64 := syscall.NewLazyDLL(filepath.Join(filepath.Dir(os.Args[0]), autoItX3))
47+
controlClick = dll64.NewProc("AU3_ControlClick")
48+
controlSend = dll64.NewProc("AU3_ControlSend")
49+
run = dll64.NewProc("AU3_Run")
50+
winActivate = dll64.NewProc("AU3_WinActivate")
51+
winWait = dll64.NewProc("AU3_WinWait")
52+
}
53+
54+
// Run -- Run a windows program
55+
// flag 3(max) 6(min) 9(normal) 0(hide)
56+
func Run(szProgram string, args ...interface{}) int {
57+
var szDir string
58+
var flag int
59+
var ok bool
60+
if len(args) == 0 {
61+
szDir = ""
62+
flag = SWShowNormal
63+
} else if len(args) == 1 {
64+
if szDir, ok = args[0].(string); !ok {
65+
panic("szDir must be a string")
66+
}
67+
flag = SWShowNormal
68+
} else if len(args) == 2 {
69+
if szDir, ok = args[0].(string); !ok {
70+
panic("szDir must be a string")
71+
}
72+
if flag, ok = args[1].(int); !ok {
73+
panic("flag must be a int")
74+
}
75+
} else {
76+
panic("Too more parameter")
77+
}
78+
pid, _, lastErr := run.Call(strPtr(szProgram), strPtr(szDir), intPtr(flag))
79+
if int(pid) == 0 {
80+
println(lastErr)
81+
}
82+
return int(pid)
83+
}
84+
85+
// WinWait -- wait window to active
86+
func WinWait(szTitle string, args ...interface{}) int {
87+
var szText string
88+
var nTimeout int
89+
var ok bool
90+
if len(args) == 0 {
91+
szText = ""
92+
nTimeout = 0
93+
} else if len(args) == 1 {
94+
if szText, ok = args[0].(string); !ok {
95+
panic("szText must be a string")
96+
}
97+
nTimeout = 0
98+
} else if len(args) == 2 {
99+
if szText, ok = args[0].(string); !ok {
100+
panic("szText must be a string")
101+
}
102+
if nTimeout, ok = args[1].(int); !ok {
103+
panic("nTimeout must be a int")
104+
}
105+
} else {
106+
panic("Too more parameter")
107+
}
108+
109+
handle, _, lastErr := winWait.Call(strPtr(szTitle), strPtr(szText), intPtr(nTimeout))
110+
if int(handle) == 0 {
111+
println(lastErr)
112+
}
113+
return int(handle)
114+
}
115+
116+
// ControlClick -- Sends a mouse click command to a given control.
117+
func ControlClick(title, text, control string, args ...interface{}) int {
118+
var button string
119+
var x, y, nClicks int
120+
var ok bool
121+
122+
if len(args) == 0 {
123+
button = DefaultMouseButton
124+
nClicks = 1
125+
x = INTDEFAULT
126+
y = INTDEFAULT
127+
} else if len(args) == 1 {
128+
if button, ok = args[0].(string); !ok {
129+
panic("button must be a string")
130+
}
131+
nClicks = 1
132+
x = INTDEFAULT
133+
y = INTDEFAULT
134+
} else if len(args) == 2 {
135+
if button, ok = args[0].(string); !ok {
136+
panic("button must be a string")
137+
}
138+
if nClicks, ok = args[1].(int); !ok {
139+
panic("nClicks must be a int")
140+
}
141+
x = INTDEFAULT
142+
y = INTDEFAULT
143+
} else if len(args) == 4 {
144+
if button, ok = args[0].(string); !ok {
145+
panic("button must be a string")
146+
}
147+
if nClicks, ok = args[1].(int); !ok {
148+
panic("nClicks must be a int")
149+
}
150+
if x, ok = args[2].(int); !ok {
151+
panic("x must be a int")
152+
}
153+
if y, ok = args[3].(int); !ok {
154+
panic("y must be a int")
155+
}
156+
} else {
157+
panic("Error parameters")
158+
}
159+
ret, _, lastErr := controlClick.Call(strPtr(title), strPtr(text), strPtr(control), strPtr(button), intPtr(nClicks), intPtr(x), intPtr(y))
160+
if int(ret) == 0 {
161+
println(lastErr)
162+
}
163+
return int(ret)
164+
}
165+
166+
// WinActivate ( "title" [, "text"]) int
167+
func WinActivate(title string, args ...interface{}) int {
168+
text := ""
169+
var ok bool
170+
argsLen := len(args)
171+
if argsLen > 1 {
172+
panic("argument count > 2")
173+
}
174+
if argsLen == 1 {
175+
if text, ok = args[0].(string); !ok {
176+
panic("text must be a string")
177+
}
178+
}
179+
ret, _, lastErr := winActivate.Call(strPtr(title), strPtr(text))
180+
if int(ret) == 0 {
181+
println(lastErr)
182+
}
183+
return int(ret)
184+
}
185+
186+
// ControlSend -- Sends a string of characters to a control.
187+
func ControlSend(title, text, control, sendText string, args ...interface{}) int {
188+
var nMode int
189+
var ok bool
190+
if len(args) == 0 {
191+
nMode = 0
192+
} else if len(args) == 1 {
193+
if nMode, ok = args[0].(int); !ok {
194+
panic("nMode must be a int")
195+
}
196+
} else {
197+
panic("Too more parameter")
198+
}
199+
ret, _, lastErr := controlSend.Call(strPtr(title), strPtr(text), strPtr(control), strPtr(sendText), intPtr(nMode))
200+
if int(ret) == 0 {
201+
println(lastErr)
202+
}
203+
return int(ret)
204+
}
205+
206+
func findTermChr(buff []uint16) int {
207+
for i, char := range buff {
208+
if char == 0x0 {
209+
return i
210+
}
211+
}
212+
panic("not supposed to happen")
213+
}
214+
215+
func intPtr(n int) uintptr {
216+
return uintptr(n)
217+
}
218+
219+
func strPtr(s string) uintptr {
220+
return uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(s)))
221+
}

0 commit comments

Comments
 (0)