-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathssh_pool.go
More file actions
100 lines (83 loc) · 1.81 KB
/
ssh_pool.go
File metadata and controls
100 lines (83 loc) · 1.81 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
package shaw
import (
"time"
"github.com/emptyinterface/extio"
"golang.org/x/crypto/ssh"
)
type (
SSHPool struct {
clients []*SSHClient
}
PoolCommandSession struct {
cmd *Command
sessions []*CommandSession
reports []*CommandReport
start, end time.Time
}
CommandReport struct {
session *CommandSession
cmd *Command
err error
start, end time.Time
}
)
func NewSSHPool(clients ...*SSHClient) *SSHPool {
return &SSHPool{
clients: clients,
}
}
func (p *SSHPool) AddClient(client *SSHClient) {
p.clients = append(p.clients, client)
}
func (p *SSHPool) NewCommandSession(cmd *Command) *PoolCommandSession {
ps := &PoolCommandSession{
cmd: cmd,
}
var bc *extio.Broadcaster
if cmd.Stdin != nil {
bc = extio.NewBroadcaster(cmd.Stdin)
}
for _, client := range p.clients {
// make copy
// all vars but stdin can be copied
c := *cmd
if bc != nil {
c.Stdin = bc.NewReader()
}
ps.sessions = append(ps.sessions, client.NewCommandSession(&c))
}
return ps
}
func (ps *PoolCommandSession) Start() {
ps.start = time.Now()
for _, s := range ps.sessions {
ps.reports = append(ps.reports, &CommandReport{
session: s,
cmd: s.cmd,
start: time.Now(),
err: s.Start(),
})
}
}
func (ps *PoolCommandSession) Wait() []*CommandReport {
for i, s := range ps.sessions {
if err := s.Wait(); err != nil && ps.reports[i].err == nil {
ps.reports[i].err = err
}
ps.reports[i].end = s.end
}
ps.end = time.Now()
return ps.reports
}
func (ps *PoolCommandSession) Run() []*CommandReport {
ps.Start()
return ps.Wait()
}
func (ps *PoolCommandSession) Signal(sig ssh.Signal) []*CommandReport {
for i, s := range ps.sessions {
if err := s.Signal(sig); err != nil && ps.reports[i].err == nil {
ps.reports[i].err = err
}
}
return ps.Wait()
}