2023-09-29 13:40:35 -04:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
|
|
|
|
package syspolicy
|
|
|
|
|
|
|
|
import (
|
2024-10-08 10:50:14 -05:00
|
|
|
"tailscale.com/util/syspolicy/internal"
|
|
|
|
"tailscale.com/util/syspolicy/rsop"
|
|
|
|
"tailscale.com/util/syspolicy/setting"
|
|
|
|
"tailscale.com/util/syspolicy/source"
|
2023-09-29 13:40:35 -04:00
|
|
|
)
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// TODO(nickkhyl): delete this file once other repos are updated.
|
2023-09-29 13:40:35 -04:00
|
|
|
|
|
|
|
// Handler reads system policies from OS-specific storage.
|
2024-10-08 10:50:14 -05:00
|
|
|
//
|
|
|
|
// Deprecated: implementing a [source.Store] should be preferred.
|
2023-09-29 13:40:35 -04:00
|
|
|
type Handler interface {
|
2024-04-13 17:38:55 -07:00
|
|
|
// ReadString reads the policy setting's string value for the given key.
|
|
|
|
// It should return ErrNoSuchKey if the key does not have a value set.
|
2023-09-29 13:40:35 -04:00
|
|
|
ReadString(key string) (string, error)
|
2024-04-13 17:38:55 -07:00
|
|
|
// ReadUInt64 reads the policy setting's uint64 value for the given key.
|
|
|
|
// It should return ErrNoSuchKey if the key does not have a value set.
|
2023-09-29 13:40:35 -04:00
|
|
|
ReadUInt64(key string) (uint64, error)
|
2024-04-13 17:38:55 -07:00
|
|
|
// ReadBool reads the policy setting's boolean value for the given key.
|
|
|
|
// It should return ErrNoSuchKey if the key does not have a value set.
|
2023-09-29 21:27:04 -04:00
|
|
|
ReadBoolean(key string) (bool, error)
|
2024-04-23 22:23:48 -07:00
|
|
|
// ReadStringArray reads the policy setting's string array value for the given key.
|
|
|
|
// It should return ErrNoSuchKey if the key does not have a value set.
|
|
|
|
ReadStringArray(key string) ([]string, error)
|
2023-09-29 13:40:35 -04:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// RegisterHandler wraps and registers the specified handler as the device's
|
|
|
|
// policy [source.Store] for the program's lifetime.
|
|
|
|
//
|
|
|
|
// Deprecated: using [RegisterStore] should be preferred.
|
|
|
|
func RegisterHandler(h Handler) {
|
|
|
|
rsop.RegisterStore("DeviceHandler", setting.DeviceScope, WrapHandler(h))
|
|
|
|
}
|
2023-09-29 13:40:35 -04:00
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// TB is a subset of testing.TB that we use to set up test helpers.
|
|
|
|
// It's defined here to avoid pulling in the testing package.
|
|
|
|
type TB = internal.TB
|
2023-09-29 13:40:35 -04:00
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// SetHandlerForTest wraps and sets the specified handler as the device's policy
|
|
|
|
// [source.Store] for the duration of tb.
|
|
|
|
//
|
|
|
|
// Deprecated: using [MustRegisterStoreForTest] should be preferred.
|
|
|
|
func SetHandlerForTest(tb TB, h Handler) {
|
|
|
|
RegisterWellKnownSettingsForTest(tb)
|
|
|
|
MustRegisterStoreForTest(tb, "DeviceHandler-TestOnly", setting.DefaultScope(), WrapHandler(h))
|
2023-09-29 13:40:35 -04:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
var _ source.Store = (*handlerStore)(nil)
|
|
|
|
|
|
|
|
// handlerStore is a [source.Store] that calls the underlying [Handler].
|
|
|
|
//
|
|
|
|
// TODO(nickkhyl): remove it when the corp and android repos are updated.
|
|
|
|
type handlerStore struct {
|
|
|
|
h Handler
|
2023-09-29 13:40:35 -04:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// WrapHandler returns a [source.Store] that wraps the specified [Handler].
|
|
|
|
func WrapHandler(h Handler) source.Store {
|
|
|
|
return handlerStore{h}
|
2023-09-29 21:27:04 -04:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// Lock implements [source.Lockable].
|
|
|
|
func (s handlerStore) Lock() error {
|
|
|
|
if lockable, ok := s.h.(source.Lockable); ok {
|
|
|
|
return lockable.Lock()
|
|
|
|
}
|
|
|
|
return nil
|
2024-04-23 22:23:48 -07:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// Unlock implements [source.Lockable].
|
|
|
|
func (s handlerStore) Unlock() {
|
|
|
|
if lockable, ok := s.h.(source.Lockable); ok {
|
|
|
|
lockable.Unlock()
|
|
|
|
}
|
2023-09-29 13:40:35 -04:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// RegisterChangeCallback implements [source.Changeable].
|
|
|
|
func (s handlerStore) RegisterChangeCallback(callback func()) (unregister func(), err error) {
|
|
|
|
if changeable, ok := s.h.(source.Changeable); ok {
|
|
|
|
return changeable.RegisterChangeCallback(callback)
|
2023-09-29 13:40:35 -04:00
|
|
|
}
|
2024-10-08 10:50:14 -05:00
|
|
|
return func() {}, nil
|
2023-09-29 13:40:35 -04:00
|
|
|
}
|
2023-11-29 16:48:25 -05:00
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// ReadString implements [source.Store].
|
|
|
|
func (s handlerStore) ReadString(key setting.Key) (string, error) {
|
|
|
|
return s.h.ReadString(string(key))
|
2024-05-23 23:13:32 -07:00
|
|
|
}
|
|
|
|
|
2024-10-08 10:50:14 -05:00
|
|
|
// ReadUInt64 implements [source.Store].
|
|
|
|
func (s handlerStore) ReadUInt64(key setting.Key) (uint64, error) {
|
|
|
|
return s.h.ReadUInt64(string(key))
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadBoolean implements [source.Store].
|
|
|
|
func (s handlerStore) ReadBoolean(key setting.Key) (bool, error) {
|
|
|
|
return s.h.ReadBoolean(string(key))
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadStringArray implements [source.Store].
|
|
|
|
func (s handlerStore) ReadStringArray(key setting.Key) ([]string, error) {
|
|
|
|
return s.h.ReadStringArray(string(key))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Done implements [source.Expirable].
|
|
|
|
func (s handlerStore) Done() <-chan struct{} {
|
|
|
|
if expirable, ok := s.h.(source.Expirable); ok {
|
|
|
|
return expirable.Done()
|
|
|
|
}
|
|
|
|
return nil
|
2023-11-29 16:48:25 -05:00
|
|
|
}
|