Updates #12614 Change-Id: I9d08a1330b9c55e1a23e7979a707e11d8e090d79 Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>main
parent
038cdb4640
commit
ba76578447
@ -0,0 +1,13 @@ |
||||
// Copyright (c) Tailscale Inc & AUTHORS
|
||||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
// Code generated by gen.go; DO NOT EDIT.
|
||||
|
||||
//go:build ts_omit_posture
|
||||
|
||||
package buildfeatures |
||||
|
||||
// HasPosture is whether the binary was built with support for modular feature "Device posture checking support".
|
||||
// Specifically, it's whether the binary was NOT built with the "ts_omit_posture" build tag.
|
||||
// It's a const so it can be used for dead code elimination.
|
||||
const HasPosture = false |
||||
@ -0,0 +1,13 @@ |
||||
// Copyright (c) Tailscale Inc & AUTHORS
|
||||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
// Code generated by gen.go; DO NOT EDIT.
|
||||
|
||||
//go:build !ts_omit_posture
|
||||
|
||||
package buildfeatures |
||||
|
||||
// HasPosture is whether the binary was built with support for modular feature "Device posture checking support".
|
||||
// Specifically, it's whether the binary was NOT built with the "ts_omit_posture" build tag.
|
||||
// It's a const so it can be used for dead code elimination.
|
||||
const HasPosture = true |
||||
@ -0,0 +1,8 @@ |
||||
// Copyright (c) Tailscale Inc & AUTHORS
|
||||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
//go:build !ts_omit_posture
|
||||
|
||||
package condregister |
||||
|
||||
import _ "tailscale.com/feature/posture" |
||||
@ -0,0 +1,114 @@ |
||||
// Copyright (c) Tailscale Inc & AUTHORS
|
||||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
// Package posture registers support for device posture checking,
|
||||
// reporting machine-specific information to the control plane
|
||||
// when enabled by the user and tailnet.
|
||||
package posture |
||||
|
||||
import ( |
||||
"encoding/json" |
||||
"net/http" |
||||
|
||||
"tailscale.com/ipn/ipnext" |
||||
"tailscale.com/ipn/ipnlocal" |
||||
"tailscale.com/posture" |
||||
"tailscale.com/syncs" |
||||
"tailscale.com/tailcfg" |
||||
"tailscale.com/types/logger" |
||||
"tailscale.com/util/syspolicy/pkey" |
||||
"tailscale.com/util/syspolicy/ptype" |
||||
) |
||||
|
||||
func init() { |
||||
ipnext.RegisterExtension("posture", newExtension) |
||||
ipnlocal.RegisterC2N("GET /posture/identity", handleC2NPostureIdentityGet) |
||||
} |
||||
|
||||
func newExtension(logf logger.Logf, b ipnext.SafeBackend) (ipnext.Extension, error) { |
||||
e := &extension{ |
||||
logf: logger.WithPrefix(logf, "posture: "), |
||||
} |
||||
return e, nil |
||||
} |
||||
|
||||
type extension struct { |
||||
logf logger.Logf |
||||
|
||||
// lastKnownHardwareAddrs is a list of the previous known hardware addrs.
|
||||
// Previously known hwaddrs are kept to work around an issue on Windows
|
||||
// where all addresses might disappear.
|
||||
// http://go/corp/25168
|
||||
lastKnownHardwareAddrs syncs.AtomicValue[[]string] |
||||
} |
||||
|
||||
func (e *extension) Name() string { return "posture" } |
||||
func (e *extension) Init(h ipnext.Host) error { return nil } |
||||
func (e *extension) Shutdown() error { return nil } |
||||
|
||||
func handleC2NPostureIdentityGet(b *ipnlocal.LocalBackend, w http.ResponseWriter, r *http.Request) { |
||||
e, ok := ipnlocal.GetExt[*extension](b) |
||||
if !ok { |
||||
http.Error(w, "posture extension not available", http.StatusInternalServerError) |
||||
return |
||||
} |
||||
e.logf("c2n: GET /posture/identity received") |
||||
|
||||
res := tailcfg.C2NPostureIdentityResponse{} |
||||
|
||||
// Only collect posture identity if enabled on the client,
|
||||
// this will first check syspolicy, MDM settings like Registry
|
||||
// on Windows or defaults on macOS. If they are not set, it falls
|
||||
// back to the cli-flag, `--posture-checking`.
|
||||
choice, err := b.PolicyClient().GetPreferenceOption(pkey.PostureChecking, ptype.ShowChoiceByPolicy) |
||||
if err != nil { |
||||
e.logf( |
||||
"c2n: failed to read PostureChecking from syspolicy, returning default from CLI: %s; got error: %s", |
||||
b.Prefs().PostureChecking(), |
||||
err, |
||||
) |
||||
} |
||||
|
||||
if choice.ShouldEnable(b.Prefs().PostureChecking()) { |
||||
res.SerialNumbers, err = posture.GetSerialNumbers(b.PolicyClient(), e.logf) |
||||
if err != nil { |
||||
e.logf("c2n: GetSerialNumbers returned error: %v", err) |
||||
} |
||||
|
||||
// TODO(tailscale/corp#21371, 2024-07-10): once this has landed in a stable release
|
||||
// and looks good in client metrics, remove this parameter and always report MAC
|
||||
// addresses.
|
||||
if r.FormValue("hwaddrs") == "true" { |
||||
res.IfaceHardwareAddrs, err = e.getHardwareAddrs() |
||||
if err != nil { |
||||
e.logf("c2n: GetHardwareAddrs returned error: %v", err) |
||||
} |
||||
} |
||||
} else { |
||||
res.PostureDisabled = true |
||||
} |
||||
|
||||
e.logf("c2n: posture identity disabled=%v reported %d serials %d hwaddrs", res.PostureDisabled, len(res.SerialNumbers), len(res.IfaceHardwareAddrs)) |
||||
|
||||
w.Header().Set("Content-Type", "application/json") |
||||
json.NewEncoder(w).Encode(res) |
||||
} |
||||
|
||||
// getHardwareAddrs returns the hardware addresses for the machine. If the list
|
||||
// of hardware addresses is empty, it will return the previously known hardware
|
||||
// addresses. Both the current, and previously known hardware addresses might be
|
||||
// empty.
|
||||
func (e *extension) getHardwareAddrs() ([]string, error) { |
||||
addrs, err := posture.GetHardwareAddrs() |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
|
||||
if len(addrs) == 0 { |
||||
e.logf("getHardwareAddrs: got empty list of hwaddrs, returning previous list") |
||||
return e.lastKnownHardwareAddrs.Load(), nil |
||||
} |
||||
|
||||
e.lastKnownHardwareAddrs.Store(addrs) |
||||
return addrs, nil |
||||
} |
||||
Loading…
Reference in new issue