ipn/ipnlocal: reenable profile tests on Windows
This fix does not seem ideal, but the test infrastructure using a local goos doesn't seem to avoid all of the associated challenges, but is somewhat deeply tied to the setup. The core issue this addresses for now is that when run on Windows there can be no code paths that attempt to use an invalid UID string, which on Windows is described in [1]. For the goos="linux" tests, we now explicitly skip the affected migration code if runtime.GOOS=="windows", and for the Windows test we explicitly use the running users uid, rather than just the string "user1". We also now make the case where a profile exists and has already been migrated a non-error condition toward the outer API. Updates #7876 [1] https://learn.microsoft.com/en-us/windows-server/identity/ad-ds/manage/understand-security-identifiers Signed-off-by: James Tucker <jftucker@gmail.com>
This commit is contained in:
parent
43819309e1
commit
095d3edd33
|
@ -22,6 +22,8 @@ import (
|
||||||
"tailscale.com/util/winutil"
|
"tailscale.com/util/winutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var errAlreadyMigrated = errors.New("profile migration already completed")
|
||||||
|
|
||||||
// profileManager is a wrapper around a StateStore that manages
|
// profileManager is a wrapper around a StateStore that manages
|
||||||
// multiple profiles and the current profile.
|
// multiple profiles and the current profile.
|
||||||
type profileManager struct {
|
type profileManager struct {
|
||||||
|
@ -66,7 +68,7 @@ func (pm *profileManager) SetCurrentUserID(uid ipn.WindowsUserID) error {
|
||||||
b, err := pm.store.ReadState(ipn.CurrentProfileKey(string(uid)))
|
b, err := pm.store.ReadState(ipn.CurrentProfileKey(string(uid)))
|
||||||
if err == ipn.ErrStateNotExist || len(b) == 0 {
|
if err == ipn.ErrStateNotExist || len(b) == 0 {
|
||||||
if runtime.GOOS == "windows" {
|
if runtime.GOOS == "windows" {
|
||||||
if err := pm.migrateFromLegacyPrefs(); err != nil {
|
if err := pm.migrateFromLegacyPrefs(); err != nil && !errors.Is(err, errAlreadyMigrated) {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -544,7 +546,14 @@ func newProfileManagerWithGOOS(store ipn.StateStore, logf logger.Logf, goos stri
|
||||||
if err := pm.setPrefsLocked(prefs); err != nil {
|
if err := pm.setPrefsLocked(prefs); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else if len(knownProfiles) == 0 && goos != "windows" {
|
// Most platform behavior is controlled by the goos parameter, however
|
||||||
|
// some behavior is implied by build tag and fails when run on Windows,
|
||||||
|
// so we explicitly avoid that behavior when running on Windows.
|
||||||
|
// Specifically this reaches down into legacy preference loading that is
|
||||||
|
// specialized by profiles_windows.go and fails in tests on an invalid
|
||||||
|
// uid passed in from the unix tests. The uid's used for Windows tests
|
||||||
|
// and runtime must be valid Windows security identifier structures.
|
||||||
|
} else if len(knownProfiles) == 0 && goos != "windows" && runtime.GOOS != "windows" {
|
||||||
// No known profiles, try a migration.
|
// No known profiles, try a migration.
|
||||||
if err := pm.migrateFromLegacyPrefs(); err != nil {
|
if err := pm.migrateFromLegacyPrefs(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -562,7 +571,7 @@ func (pm *profileManager) migrateFromLegacyPrefs() error {
|
||||||
sentinel, prefs, err := pm.loadLegacyPrefs()
|
sentinel, prefs, err := pm.loadLegacyPrefs()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
metricMigrationError.Add(1)
|
metricMigrationError.Add(1)
|
||||||
return err
|
return fmt.Errorf("load legacy prefs: %w", err)
|
||||||
}
|
}
|
||||||
if err := pm.SetPrefs(prefs); err != nil {
|
if err := pm.SetPrefs(prefs); err != nil {
|
||||||
metricMigrationError.Add(1)
|
metricMigrationError.Add(1)
|
||||||
|
|
|
@ -5,7 +5,7 @@ package ipnlocal
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"runtime"
|
"os/user"
|
||||||
"strconv"
|
"strconv"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -18,9 +18,6 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestProfileCurrentUserSwitch(t *testing.T) {
|
func TestProfileCurrentUserSwitch(t *testing.T) {
|
||||||
if runtime.GOOS == "windows" {
|
|
||||||
t.Skip("TODO(#7876): test regressed on windows while CI was broken")
|
|
||||||
}
|
|
||||||
store := new(mem.Store)
|
store := new(mem.Store)
|
||||||
|
|
||||||
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "linux")
|
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "linux")
|
||||||
|
@ -77,9 +74,6 @@ func TestProfileCurrentUserSwitch(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestProfileList(t *testing.T) {
|
func TestProfileList(t *testing.T) {
|
||||||
if runtime.GOOS == "windows" {
|
|
||||||
t.Skip("TODO(#7876): test regressed on windows while CI was broken")
|
|
||||||
}
|
|
||||||
store := new(mem.Store)
|
store := new(mem.Store)
|
||||||
|
|
||||||
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "linux")
|
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "linux")
|
||||||
|
@ -158,9 +152,6 @@ func TestProfileList(t *testing.T) {
|
||||||
|
|
||||||
// TestProfileManagement tests creating, loading, and switching profiles.
|
// TestProfileManagement tests creating, loading, and switching profiles.
|
||||||
func TestProfileManagement(t *testing.T) {
|
func TestProfileManagement(t *testing.T) {
|
||||||
if runtime.GOOS == "windows" {
|
|
||||||
t.Skip("TODO(#7876): test regressed on windows while CI was broken")
|
|
||||||
}
|
|
||||||
store := new(mem.Store)
|
store := new(mem.Store)
|
||||||
|
|
||||||
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "linux")
|
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "linux")
|
||||||
|
@ -312,10 +303,11 @@ func TestProfileManagement(t *testing.T) {
|
||||||
// TestProfileManagementWindows tests going into and out of Unattended mode on
|
// TestProfileManagementWindows tests going into and out of Unattended mode on
|
||||||
// Windows.
|
// Windows.
|
||||||
func TestProfileManagementWindows(t *testing.T) {
|
func TestProfileManagementWindows(t *testing.T) {
|
||||||
|
u, err := user.Current()
|
||||||
if runtime.GOOS == "windows" {
|
if err != nil {
|
||||||
t.Skip("TODO(#7876): test regressed on windows while CI was broken")
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
uid := ipn.WindowsUserID(u.Uid)
|
||||||
|
|
||||||
store := new(mem.Store)
|
store := new(mem.Store)
|
||||||
|
|
||||||
|
@ -365,8 +357,8 @@ func TestProfileManagementWindows(t *testing.T) {
|
||||||
|
|
||||||
{
|
{
|
||||||
t.Logf("Set user1 as logged in user")
|
t.Logf("Set user1 as logged in user")
|
||||||
if err := pm.SetCurrentUserID("user1"); err != nil {
|
if err := pm.SetCurrentUserID(uid); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatalf("can't set user id: %s", err)
|
||||||
}
|
}
|
||||||
checkProfiles(t)
|
checkProfiles(t)
|
||||||
t.Logf("Save prefs for user1")
|
t.Logf("Save prefs for user1")
|
||||||
|
@ -401,7 +393,7 @@ func TestProfileManagementWindows(t *testing.T) {
|
||||||
|
|
||||||
{
|
{
|
||||||
t.Logf("Set user1 as current user")
|
t.Logf("Set user1 as current user")
|
||||||
if err := pm.SetCurrentUserID("user1"); err != nil {
|
if err := pm.SetCurrentUserID(uid); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
wantCurProfile = "test"
|
wantCurProfile = "test"
|
||||||
|
@ -411,8 +403,8 @@ func TestProfileManagementWindows(t *testing.T) {
|
||||||
t.Logf("set unattended mode")
|
t.Logf("set unattended mode")
|
||||||
wantProfiles["test"] = setPrefs(t, "test", true)
|
wantProfiles["test"] = setPrefs(t, "test", true)
|
||||||
}
|
}
|
||||||
if pm.CurrentUserID() != "user1" {
|
if pm.CurrentUserID() != uid {
|
||||||
t.Fatalf("CurrentUserID = %q; want %q", pm.CurrentUserID(), "user1")
|
t.Fatalf("CurrentUserID = %q; want %q", pm.CurrentUserID(), uid)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Recreate the profile manager to ensure that it starts with test profile.
|
// Recreate the profile manager to ensure that it starts with test profile.
|
||||||
|
@ -421,7 +413,7 @@ func TestProfileManagementWindows(t *testing.T) {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
checkProfiles(t)
|
checkProfiles(t)
|
||||||
if pm.CurrentUserID() != "user1" {
|
if pm.CurrentUserID() != uid {
|
||||||
t.Fatalf("CurrentUserID = %q; want %q", pm.CurrentUserID(), "user1")
|
t.Fatalf("CurrentUserID = %q; want %q", pm.CurrentUserID(), uid)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,6 +6,7 @@ package ipnlocal
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io/fs"
|
||||||
"os"
|
"os"
|
||||||
"os/user"
|
"os/user"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
@ -21,8 +22,6 @@ const (
|
||||||
legacyPrefsExt = ".conf"
|
legacyPrefsExt = ".conf"
|
||||||
)
|
)
|
||||||
|
|
||||||
var errAlreadyMigrated = errors.New("profile migration already completed")
|
|
||||||
|
|
||||||
func legacyPrefsDir(uid ipn.WindowsUserID) (string, error) {
|
func legacyPrefsDir(uid ipn.WindowsUserID) (string, error) {
|
||||||
// TODO(aaron): Ideally we'd have the impersonation token for the pipe's
|
// TODO(aaron): Ideally we'd have the impersonation token for the pipe's
|
||||||
// client and use it to call SHGetKnownFolderPath, thus yielding the correct
|
// client and use it to call SHGetKnownFolderPath, thus yielding the correct
|
||||||
|
@ -56,6 +55,9 @@ func (pm *profileManager) loadLegacyPrefs() (string, ipn.PrefsView, error) {
|
||||||
|
|
||||||
prefsPath := filepath.Join(userLegacyPrefsDir, legacyPrefsFile+legacyPrefsExt)
|
prefsPath := filepath.Join(userLegacyPrefsDir, legacyPrefsFile+legacyPrefsExt)
|
||||||
prefs, err := ipn.LoadPrefs(prefsPath)
|
prefs, err := ipn.LoadPrefs(prefsPath)
|
||||||
|
if errors.Is(err, fs.ErrNotExist) {
|
||||||
|
return "", ipn.PrefsView{}, errAlreadyMigrated
|
||||||
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", ipn.PrefsView{}, err
|
return "", ipn.PrefsView{}, err
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue