blob: 7700d18f1ab68a7ade7f58c87116582761442437 [file] [log] [blame]
Adam Langleyfa50e742014-04-09 13:57:52 -07001// Copyright 2012 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5package agent
6
7import (
Peter Moodye84a34b2016-04-19 09:49:29 -07008 "crypto"
9 "crypto/rand"
10 "fmt"
Gh0u1L52faea142017-07-22 22:00:11 -070011 pseudorand "math/rand"
12 "reflect"
13 "strings"
Adam Langleyfa50e742014-04-09 13:57:52 -070014 "testing"
15
Andrew Gerranda73c6bb2014-11-10 08:50:25 +110016 "golang.org/x/crypto/ssh"
Adam Langleyfa50e742014-04-09 13:57:52 -070017)
18
19func TestServer(t *testing.T) {
20 c1, c2, err := netPipe()
21 if err != nil {
22 t.Fatalf("netPipe: %v", err)
23 }
24 defer c1.Close()
25 defer c2.Close()
26 client := NewClient(c1)
27
28 go ServeAgent(NewKeyring(), c2)
29
Peter Moody7d5b0be2015-07-08 15:14:43 -070030 testAgentInterface(t, client, testPrivateKeys["rsa"], nil, 0)
Adam Langleyfa50e742014-04-09 13:57:52 -070031}
32
33func TestLockServer(t *testing.T) {
34 testLockAgent(NewKeyring(), t)
35}
36
37func TestSetupForwardAgent(t *testing.T) {
38 a, b, err := netPipe()
39 if err != nil {
40 t.Fatalf("netPipe: %v", err)
41 }
42
43 defer a.Close()
44 defer b.Close()
45
Gh0u1L5b176d7d2017-07-22 21:49:35 -070046 _, socket, cleanup := startOpenSSHAgent(t)
Adam Langleyfa50e742014-04-09 13:57:52 -070047 defer cleanup()
48
49 serverConf := ssh.ServerConfig{
50 NoClientAuth: true,
51 }
52 serverConf.AddHostKey(testSigners["rsa"])
53 incoming := make(chan *ssh.ServerConn, 1)
54 go func() {
55 conn, _, _, err := ssh.NewServerConn(a, &serverConf)
Adam Langleyfa50e742014-04-09 13:57:52 -070056 incoming <- conn
Nicola Murinoedc325d2023-07-29 03:33:00 +000057 if err != nil {
58 t.Errorf("NewServerConn error: %v", err)
59 return
60 }
Adam Langleyfa50e742014-04-09 13:57:52 -070061 }()
62
Han-Wen Nienhuyse4e27992017-03-29 19:21:25 +020063 conf := ssh.ClientConfig{
64 HostKeyCallback: ssh.InsecureIgnoreHostKey(),
65 }
Adam Langleyfa50e742014-04-09 13:57:52 -070066 conn, chans, reqs, err := ssh.NewClientConn(b, "", &conf)
67 if err != nil {
68 t.Fatalf("NewClientConn: %v", err)
69 }
70 client := ssh.NewClient(conn, chans, reqs)
71
72 if err := ForwardToRemote(client, socket); err != nil {
73 t.Fatalf("SetupForwardAgent: %v", err)
74 }
Adam Langleyfa50e742014-04-09 13:57:52 -070075 server := <-incoming
Nicola Murinoedc325d2023-07-29 03:33:00 +000076 if server == nil {
77 t.Fatal("Unable to get server")
78 }
Adam Langleyfa50e742014-04-09 13:57:52 -070079 ch, reqs, err := server.OpenChannel(channelType, nil)
80 if err != nil {
81 t.Fatalf("OpenChannel(%q): %v", channelType, err)
82 }
83 go ssh.DiscardRequests(reqs)
84
85 agentClient := NewClient(ch)
Peter Moody7d5b0be2015-07-08 15:14:43 -070086 testAgentInterface(t, agentClient, testPrivateKeys["rsa"], nil, 0)
Adam Langleyfa50e742014-04-09 13:57:52 -070087 conn.Close()
88}
Mark Severson33051862016-04-02 19:02:50 -060089
90func TestV1ProtocolMessages(t *testing.T) {
91 c1, c2, err := netPipe()
92 if err != nil {
93 t.Fatalf("netPipe: %v", err)
94 }
95 defer c1.Close()
96 defer c2.Close()
97 c := NewClient(c1)
98
99 go ServeAgent(NewKeyring(), c2)
100
101 testV1ProtocolMessages(t, c.(*client))
102}
103
104func testV1ProtocolMessages(t *testing.T, c *client) {
105 reply, err := c.call([]byte{agentRequestV1Identities})
106 if err != nil {
107 t.Fatalf("v1 request all failed: %v", err)
108 }
109 if msg, ok := reply.(*agentV1IdentityMsg); !ok || msg.Numkeys != 0 {
110 t.Fatalf("invalid request all response: %#v", reply)
111 }
112
113 reply, err = c.call([]byte{agentRemoveAllV1Identities})
114 if err != nil {
115 t.Fatalf("v1 remove all failed: %v", err)
116 }
117 if _, ok := reply.(*successAgentMsg); !ok {
118 t.Fatalf("invalid remove all response: %#v", reply)
119 }
120}
Peter Moodye84a34b2016-04-19 09:49:29 -0700121
122func verifyKey(sshAgent Agent) error {
123 keys, err := sshAgent.List()
124 if err != nil {
125 return fmt.Errorf("listing keys: %v", err)
126 }
127
128 if len(keys) != 1 {
129 return fmt.Errorf("bad number of keys found. expected 1, got %d", len(keys))
130 }
131
132 buf := make([]byte, 128)
133 if _, err := rand.Read(buf); err != nil {
134 return fmt.Errorf("rand: %v", err)
135 }
136
137 sig, err := sshAgent.Sign(keys[0], buf)
138 if err != nil {
139 return fmt.Errorf("sign: %v", err)
140 }
141
142 if err := keys[0].Verify(buf, sig); err != nil {
143 return fmt.Errorf("verify: %v", err)
144 }
145 return nil
146}
147
148func addKeyToAgent(key crypto.PrivateKey) error {
149 sshAgent := NewKeyring()
150 if err := sshAgent.Add(AddedKey{PrivateKey: key}); err != nil {
151 return fmt.Errorf("add: %v", err)
152 }
153 return verifyKey(sshAgent)
154}
155
156func TestKeyTypes(t *testing.T) {
157 for k, v := range testPrivateKeys {
158 if err := addKeyToAgent(v); err != nil {
159 t.Errorf("error adding key type %s, %v", k, err)
160 }
Peter Moody9a6bac92016-07-07 15:51:40 -0700161 if err := addCertToAgentSock(v, nil); err != nil {
162 t.Errorf("error adding key type %s, %v", k, err)
163 }
Peter Moodye84a34b2016-04-19 09:49:29 -0700164 }
165}
166
Peter Moody811831d2016-06-20 18:10:02 -0700167func addCertToAgentSock(key crypto.PrivateKey, cert *ssh.Certificate) error {
168 a, b, err := netPipe()
169 if err != nil {
170 return err
171 }
172 agentServer := NewKeyring()
173 go ServeAgent(agentServer, a)
174
175 agentClient := NewClient(b)
176 if err := agentClient.Add(AddedKey{PrivateKey: key, Certificate: cert}); err != nil {
177 return fmt.Errorf("add: %v", err)
178 }
179 return verifyKey(agentClient)
180}
181
Peter Moodye84a34b2016-04-19 09:49:29 -0700182func addCertToAgent(key crypto.PrivateKey, cert *ssh.Certificate) error {
183 sshAgent := NewKeyring()
184 if err := sshAgent.Add(AddedKey{PrivateKey: key, Certificate: cert}); err != nil {
185 return fmt.Errorf("add: %v", err)
186 }
187 return verifyKey(sshAgent)
188}
189
190func TestCertTypes(t *testing.T) {
191 for keyType, key := range testPublicKeys {
192 cert := &ssh.Certificate{
193 ValidPrincipals: []string{"gopher1"},
194 ValidAfter: 0,
195 ValidBefore: ssh.CertTimeInfinity,
196 Key: key,
197 Serial: 1,
198 CertType: ssh.UserCert,
199 SignatureKey: testPublicKeys["rsa"],
200 Permissions: ssh.Permissions{
201 CriticalOptions: map[string]string{},
202 Extensions: map[string]string{},
203 },
204 }
205 if err := cert.SignCert(rand.Reader, testSigners["rsa"]); err != nil {
206 t.Fatalf("signcert: %v", err)
207 }
208 if err := addCertToAgent(testPrivateKeys[keyType], cert); err != nil {
209 t.Fatalf("%v", err)
210 }
Peter Moody811831d2016-06-20 18:10:02 -0700211 if err := addCertToAgentSock(testPrivateKeys[keyType], cert); err != nil {
212 t.Fatalf("%v", err)
213 }
Peter Moodye84a34b2016-04-19 09:49:29 -0700214 }
215}
Gh0u1L52faea142017-07-22 22:00:11 -0700216
217func TestParseConstraints(t *testing.T) {
218 // Test LifetimeSecs
219 var msg = constrainLifetimeAgentMsg{pseudorand.Uint32()}
220 lifetimeSecs, _, _, err := parseConstraints(ssh.Marshal(msg))
221 if err != nil {
222 t.Fatalf("parseConstraints: %v", err)
223 }
224 if lifetimeSecs != msg.LifetimeSecs {
225 t.Errorf("got lifetime %v, want %v", lifetimeSecs, msg.LifetimeSecs)
226 }
227
228 // Test ConfirmBeforeUse
229 _, confirmBeforeUse, _, err := parseConstraints([]byte{agentConstrainConfirm})
230 if err != nil {
231 t.Fatalf("%v", err)
232 }
233 if !confirmBeforeUse {
234 t.Error("got comfirmBeforeUse == false")
235 }
236
237 // Test ConstraintExtensions
238 var data []byte
239 var expect []ConstraintExtension
240 for i := 0; i < 10; i++ {
241 var ext = ConstraintExtension{
242 ExtensionName: fmt.Sprintf("name%d", i),
243 ExtensionDetails: []byte(fmt.Sprintf("details: %d", i)),
244 }
245 expect = append(expect, ext)
Nicola Murino2aeefc32023-09-03 18:26:04 +0200246 if i%2 == 0 {
247 data = append(data, agentConstrainExtension)
248 } else {
249 data = append(data, agentConstrainExtensionV00)
250 }
Gh0u1L52faea142017-07-22 22:00:11 -0700251 data = append(data, ssh.Marshal(ext)...)
252 }
253 _, _, extensions, err := parseConstraints(data)
254 if err != nil {
255 t.Fatalf("%v", err)
256 }
257 if !reflect.DeepEqual(expect, extensions) {
258 t.Errorf("got extension %v, want %v", extensions, expect)
259 }
260
261 // Test Unknown Constraint
262 _, _, _, err = parseConstraints([]byte{128})
263 if err == nil || !strings.Contains(err.Error(), "unknown constraint") {
264 t.Errorf("unexpected error: %v", err)
265 }
266}