-
Notifications
You must be signed in to change notification settings - Fork 66
/
Copy pathcygwin.go
138 lines (130 loc) · 2.91 KB
/
cygwin.go
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
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
package app
import (
"bytes"
"context"
"crypto/rand"
"encoding/binary"
"fmt"
"github.com/buptczq/WinCryptSSHAgent/utils"
"io"
"net"
"os"
"path/filepath"
"sync"
"syscall"
"time"
)
type Cygwin struct {
running bool
sockfile string
}
func createCygwinSocket(filename string, port int) ([]byte, error) {
os.Remove(filename)
file, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0666)
if err != nil {
return nil, err
}
var uuid [16]byte
_, err = rand.Read(uuid[:])
if err != nil {
return nil, err
}
file.WriteString(fmt.Sprintf("!<socket >%d s %s", port, utils.UUIDToString(uuid)))
file.Close()
if err := utils.SetFileAttributes(filename, syscall.FILE_ATTRIBUTE_SYSTEM|syscall.FILE_ATTRIBUTE_READONLY); err != nil {
return nil, err
}
return uuid[:], nil
}
func cygwinHandshake(conn net.Conn, uuid []byte) error {
var cuuid [16]byte
_, err := conn.Read(cuuid[:])
if err != nil {
return err
}
if !bytes.Equal(uuid[:], cuuid[:]) {
return fmt.Errorf("invalid uuid")
}
conn.Write(uuid[:])
pidsUids := make([]byte, 12)
_, err = conn.Read(pidsUids[:])
if err != nil {
return err
}
pid := os.Getpid()
gid := pid // for cygwin's AF_UNIX -> AF_INET, pid = gid
binary.LittleEndian.PutUint32(pidsUids, uint32(pid))
binary.LittleEndian.PutUint32(pidsUids[8:], uint32(gid))
if _, err = conn.Write(pidsUids); err != nil {
return err
}
return nil
}
func (s *Cygwin) Run(ctx context.Context, handler func(conn io.ReadWriteCloser)) error {
home, err := os.UserHomeDir()
if err != nil {
return err
}
sockfile := filepath.Join(home, CYGWIN_SOCK)
s.sockfile = sockfile
// listen tcp socket
l, err := net.Listen("tcp", "localhost:0")
if err != nil {
return err
}
defer func() {
defer l.Close()
os.Remove(sockfile)
}()
// cygwin socket uuid
port := l.Addr().(*net.TCPAddr).Port
uuid, err := createCygwinSocket(sockfile, port)
if err != nil {
return err
}
s.running = true
// loop
wg := new(sync.WaitGroup)
for {
select {
case <-ctx.Done():
wg.Wait()
return nil
default:
}
utils.SetListenerDeadline(l, time.Now().Add(time.Second))
conn, err := l.Accept()
if opErr, ok := err.(*net.OpError); ok && opErr.Timeout() {
continue
}
if err != nil {
return err
}
err = cygwinHandshake(conn, uuid)
if err != nil {
conn.Close()
continue
}
wg.Add(1)
go func() {
handler(conn)
wg.Done()
}()
}
}
func (*Cygwin) AppId() AppId {
return APP_CYGWIN
}
func (s *Cygwin) Menu(register func(id AppId, name string, handler func())) {
register(s.AppId(), "Show "+s.AppId().String()+" Settings", s.onClick)
}
func (s *Cygwin) onClick() {
if s.running {
help := fmt.Sprintf(`export SSH_AUTH_SOCK="%s"`, s.sockfile)
if utils.MessageBox(s.AppId().FullName()+" (OK to copy):", help, utils.MB_OKCANCEL) == utils.IDOK {
utils.SetClipBoard(help)
}
} else {
utils.MessageBox("Error:", s.AppId().String()+" agent doesn't work!", utils.MB_ICONWARNING)
}
}