mirror of https://github.com/tailscale/tailscale/
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
168 lines
4.1 KiB
Go
168 lines
4.1 KiB
Go
1 year ago
|
// Copyright (c) Tailscale Inc & AUTHORS
|
||
|
// SPDX-License-Identifier: BSD-3-Clause
|
||
|
|
||
|
package tka
|
||
|
|
||
|
import (
|
||
|
"crypto/hmac"
|
||
|
"crypto/sha256"
|
||
|
"encoding/binary"
|
||
|
"encoding/hex"
|
||
|
"fmt"
|
||
|
"net/url"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
DeeplinkTailscaleURLScheme = "tailscale"
|
||
|
DeeplinkCommandSign = "sign-device"
|
||
|
)
|
||
|
|
||
|
type DeeplinkValidationResult struct {
|
||
|
IsValid bool
|
||
|
Error string
|
||
|
Version uint8
|
||
|
NodeKey string
|
||
|
TLPub string
|
||
|
DeviceName string
|
||
|
OSName string
|
||
|
EmailAddress string
|
||
|
}
|
||
|
|
||
|
// GenerateHMAC computes a SHA-256 HMAC for the concatenation of components, using
|
||
|
// stateID as secret.
|
||
|
func generateHMAC(stateID uint64, components []string) []byte {
|
||
|
key := make([]byte, 8)
|
||
|
binary.LittleEndian.PutUint64(key, stateID)
|
||
|
mac := hmac.New(sha256.New, key)
|
||
|
for _, component := range components {
|
||
|
mac.Write([]byte(component))
|
||
|
}
|
||
|
return mac.Sum(nil)
|
||
|
}
|
||
|
|
||
|
// ValidateDeeplink validates a device signing deeplink using the authority's stateID.
|
||
|
// The input urlString follows this structure:
|
||
|
//
|
||
|
// tailscale://sign-device/v1/?nk=xxx&tp=xxx&dn=xxx&os=xxx&em=xxx&hm=xxx
|
||
|
//
|
||
|
// where:
|
||
|
// - "nk" is the nodekey of the node being signed
|
||
|
// - "tp" is the tailnet lock public key
|
||
|
// - "dn" is the name of the node
|
||
|
// - "os" is the operating system of the node
|
||
|
// - "em" is the email address associated with the node
|
||
|
// - "hm" is a SHA-256 HMAC computed over the concatenation of the above fields, encoded as a hex string
|
||
|
func (a *Authority) ValidateDeeplink(urlString string) DeeplinkValidationResult {
|
||
|
parsedUrl, err := url.Parse(urlString)
|
||
|
if err != nil {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: err.Error(),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if parsedUrl.Scheme != DeeplinkTailscaleURLScheme {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: fmt.Sprintf("unhandled scheme %s, expected %s", parsedUrl.Scheme, DeeplinkTailscaleURLScheme),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if parsedUrl.Host != DeeplinkCommandSign {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: fmt.Sprintf("unhandled host %s, expected %s", parsedUrl.Host, DeeplinkCommandSign),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
path := parsedUrl.EscapedPath()
|
||
|
pathComponents := strings.Split(path, "/")
|
||
|
if len(pathComponents) != 3 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "invalid path components number found",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if pathComponents[1] != "v1" {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: fmt.Sprintf("expected v1 deeplink version, found something else: %s", pathComponents[1]),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
nodeKey := parsedUrl.Query().Get("nk")
|
||
|
if len(nodeKey) == 0 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "missing nk (NodeKey) query parameter",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
tlPub := parsedUrl.Query().Get("tp")
|
||
|
if len(tlPub) == 0 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "missing tp (TLPub) query parameter",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
deviceName := parsedUrl.Query().Get("dn")
|
||
|
if len(deviceName) == 0 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "missing dn (DeviceName) query parameter",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
osName := parsedUrl.Query().Get("os")
|
||
|
if len(deviceName) == 0 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "missing os (OSName) query parameter",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
emailAddress := parsedUrl.Query().Get("em")
|
||
|
if len(emailAddress) == 0 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "missing em (EmailAddress) query parameter",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hmacString := parsedUrl.Query().Get("hm")
|
||
|
if len(hmacString) == 0 {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "missing hm (HMAC) query parameter",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
components := []string{nodeKey, tlPub, deviceName, osName, emailAddress}
|
||
|
stateID1, _ := a.StateIDs()
|
||
|
computedHMAC := generateHMAC(stateID1, components)
|
||
|
|
||
|
hmacHexBytes, err := hex.DecodeString(hmacString)
|
||
|
if err != nil {
|
||
|
return DeeplinkValidationResult{IsValid: false, Error: "could not hex-decode hmac"}
|
||
|
}
|
||
|
|
||
|
if !hmac.Equal(computedHMAC, hmacHexBytes) {
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: false,
|
||
|
Error: "hmac authentication failed",
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return DeeplinkValidationResult{
|
||
|
IsValid: true,
|
||
|
NodeKey: nodeKey,
|
||
|
TLPub: tlPub,
|
||
|
DeviceName: deviceName,
|
||
|
OSName: osName,
|
||
|
EmailAddress: emailAddress,
|
||
|
}
|
||
|
}
|