forked from marshallbrekka/go-u2fhost
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathauthenticate.go
122 lines (102 loc) · 3.29 KB
/
authenticate.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
package u2fhost
import (
"encoding/base64"
"encoding/json"
"fmt"
butil "github.com/convox/go-u2fhost/bytes"
)
// Authenticates with the device using the AuthenticateRequest,
// returning an AuthenticateResponse.
func (dev *HidDevice) Authenticate(req *AuthenticateRequest) (*AuthenticateResponse, error) {
clientData, request, err := authenticateRequest(req)
if err != nil {
return nil, err
}
authModifier := u2fAuthEnforce
if req.CheckOnly {
authModifier = u2fAuthCheckOnly
}
status, response, err := dev.hidDevice.SendAPDU(u2fCommandAuthenticate, authModifier, 0, request)
if err != nil {
return nil, err
}
if status == u2fStatusNoError {
response := authenticateResponse(status, response, clientData, req)
// Clear out the authenticator data if the original request was not webauthn.
if !req.WebAuthn {
response.AuthenticatorData = ""
}
return response, nil
}
// If we are in webauthn mode, try a backwards compatible mode for u2f
// but don't turn off the webauthn flag
if req.WebAuthn && status == u2fStatusWrongData {
u2fReq := *req
u2fReq.AppId = fmt.Sprintf("https://%s", req.AppId)
clientData, request, err = authenticateRequest(&u2fReq)
if err != nil {
return nil, err
}
status, response, err = dev.hidDevice.SendAPDU(u2fCommandAuthenticate, authModifier, 0, request)
if err != nil {
return nil, err
}
if status == u2fStatusNoError {
return authenticateResponse(status, response, clientData, &u2fReq), nil
}
}
return nil, u2ferror(status)
}
func authenticateResponse(status uint16, response, clientData []byte, req *AuthenticateRequest) *AuthenticateResponse {
authenticatorData := append(sha256([]byte(req.AppId)), response[0:5]...)
if req.WebAuthn {
return &AuthenticateResponse{
KeyHandle: req.KeyHandle,
ClientData: websafeEncode(clientData),
SignatureData: websafeEncode(response[5:]),
AuthenticatorData: websafeEncode(authenticatorData),
}
} else {
return &AuthenticateResponse{
KeyHandle: req.KeyHandle,
ClientData: websafeEncode(clientData),
SignatureData: websafeEncode(response),
AuthenticatorData: base64.StdEncoding.EncodeToString(authenticatorData),
}
}
}
func authenticateRequest(req *AuthenticateRequest) ([]byte, []byte, error) {
// Get the channel id public key, if any
cid, err := channelIdPublicKey(req.ChannelIdPublicKey, req.ChannelIdUnused)
if err != nil {
return nil, nil, err
}
// Construct the client json
keyHandle, err := websafeDecode(req.KeyHandle)
if err != nil {
return []byte{}, []byte{}, fmt.Errorf("base64 key handle: %s", err)
}
client := clientData{
Challenge: req.Challenge,
Origin: req.Facet,
ChannelIdPublicKey: cid,
}
if req.WebAuthn {
client.Type = "webauthn.get"
} else {
client.Type = "navigator.id.getAssertion"
}
clientJson, err := json.Marshal(client)
if err != nil {
return nil, nil, fmt.Errorf("Error marshaling clientData to json: %s", err)
}
// Pack into byte array
// https://fidoalliance.org/specs/fido-u2f-v1.0-nfc-bt-amendment-20150514/fido-u2f-raw-message-formats.html#authentication-request-message---u2f_authenticate
request := butil.Concat(
sha256(clientJson),
sha256([]byte(req.AppId)),
[]byte{byte(len(keyHandle))},
keyHandle,
)
return []byte(clientJson), request, nil
}