2022-11-09 10:58:10 +05:00
|
|
|
// Copyright (c) 2022 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package ipnlocal
|
|
|
|
|
|
|
|
import (
|
2022-11-17 19:05:02 +05:00
|
|
|
"fmt"
|
2022-11-09 10:58:10 +05:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"tailscale.com/ipn"
|
|
|
|
"tailscale.com/ipn/store/mem"
|
2022-11-16 16:17:36 +05:00
|
|
|
"tailscale.com/tailcfg"
|
|
|
|
"tailscale.com/types/key"
|
2022-11-09 10:58:10 +05:00
|
|
|
"tailscale.com/types/logger"
|
|
|
|
"tailscale.com/types/persist"
|
|
|
|
)
|
|
|
|
|
|
|
|
// TestProfileManagement tests creating, loading, and switching profiles.
|
|
|
|
func TestProfileManagement(t *testing.T) {
|
|
|
|
store := new(mem.Store)
|
|
|
|
|
|
|
|
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "", "linux")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
wantCurProfile := ""
|
|
|
|
wantProfiles := map[string]ipn.PrefsView{
|
|
|
|
"": emptyPrefs,
|
|
|
|
}
|
|
|
|
checkProfiles := func(t *testing.T) {
|
|
|
|
t.Helper()
|
|
|
|
prof := pm.CurrentProfile()
|
|
|
|
t.Logf("\tCurrentProfile = %q", prof)
|
|
|
|
if prof.Name != wantCurProfile {
|
|
|
|
t.Fatalf("CurrentProfile = %q; want %q", prof, wantCurProfile)
|
|
|
|
}
|
|
|
|
profiles := pm.Profiles()
|
|
|
|
wantLen := len(wantProfiles)
|
|
|
|
if _, ok := wantProfiles[""]; ok {
|
|
|
|
wantLen--
|
|
|
|
}
|
|
|
|
if len(profiles) != wantLen {
|
|
|
|
t.Fatalf("Profiles = %v; want %v", profiles, wantProfiles)
|
|
|
|
}
|
|
|
|
p := pm.CurrentPrefs()
|
|
|
|
if !p.Valid() {
|
|
|
|
t.Fatalf("CurrentPrefs = %v; want valid", p)
|
|
|
|
}
|
|
|
|
if !p.Equals(wantProfiles[wantCurProfile]) {
|
|
|
|
t.Fatalf("CurrentPrefs = %v; want %v", p.Pretty(), wantProfiles[wantCurProfile].Pretty())
|
|
|
|
}
|
|
|
|
for _, p := range profiles {
|
|
|
|
got, err := pm.loadSavedPrefs(p.Key)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Use Hostname as a proxy for all prefs.
|
2022-11-16 16:17:36 +05:00
|
|
|
if !got.Equals(wantProfiles[p.Name]) {
|
|
|
|
t.Fatalf("Prefs for profile %q =\n got=%+v\nwant=%v", p, got.Pretty(), wantProfiles[p.Name].Pretty())
|
2022-11-09 10:58:10 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-11-16 16:17:36 +05:00
|
|
|
logins := make(map[string]tailcfg.UserID)
|
2022-11-17 19:05:02 +05:00
|
|
|
nodeIDs := make(map[string]tailcfg.StableNodeID)
|
2022-11-09 10:58:10 +05:00
|
|
|
setPrefs := func(t *testing.T, loginName string) ipn.PrefsView {
|
2022-11-16 16:17:36 +05:00
|
|
|
t.Helper()
|
2022-11-09 10:58:10 +05:00
|
|
|
p := pm.CurrentPrefs().AsStruct()
|
2022-11-17 19:05:02 +05:00
|
|
|
uid := logins[loginName]
|
|
|
|
if uid.IsZero() {
|
|
|
|
uid = tailcfg.UserID(len(logins) + 1)
|
|
|
|
logins[loginName] = uid
|
|
|
|
}
|
|
|
|
nid := nodeIDs[loginName]
|
|
|
|
if nid.IsZero() {
|
|
|
|
nid = tailcfg.StableNodeID(fmt.Sprint(len(nodeIDs) + 1))
|
|
|
|
nodeIDs[loginName] = nid
|
2022-11-16 16:17:36 +05:00
|
|
|
}
|
2022-11-09 10:58:10 +05:00
|
|
|
p.Persist = &persist.Persist{
|
2022-11-16 16:17:36 +05:00
|
|
|
LoginName: loginName,
|
|
|
|
PrivateNodeKey: key.NewNode(),
|
|
|
|
UserProfile: tailcfg.UserProfile{
|
2022-11-17 19:05:02 +05:00
|
|
|
ID: uid,
|
2022-11-16 16:17:36 +05:00
|
|
|
LoginName: loginName,
|
|
|
|
},
|
2022-11-17 19:05:02 +05:00
|
|
|
NodeID: nid,
|
2022-11-09 10:58:10 +05:00
|
|
|
}
|
|
|
|
if err := pm.SetPrefs(p.View()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return p.View()
|
|
|
|
}
|
|
|
|
t.Logf("Check initial state from empty store")
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
{
|
|
|
|
t.Logf("Set prefs for default profile")
|
|
|
|
wantProfiles["user@1.example.com"] = setPrefs(t, "user@1.example.com")
|
|
|
|
wantCurProfile = "user@1.example.com"
|
|
|
|
delete(wantProfiles, "")
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Create new profile")
|
|
|
|
pm.NewProfile()
|
|
|
|
wantCurProfile = ""
|
|
|
|
wantProfiles[""] = emptyPrefs
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
{
|
|
|
|
t.Logf("Set prefs for test profile")
|
|
|
|
wantProfiles["user@2.example.com"] = setPrefs(t, "user@2.example.com")
|
|
|
|
wantCurProfile = "user@2.example.com"
|
|
|
|
delete(wantProfiles, "")
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Recreate profile manager from store")
|
|
|
|
// Recreate the profile manager to ensure that it can load the profiles
|
|
|
|
// from the store at startup.
|
|
|
|
pm, err = newProfileManagerWithGOOS(store, logger.Discard, "", "linux")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Delete default profile")
|
|
|
|
if err := pm.DeleteProfile(pm.findProfileByName("user@1.example.com").ID); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
delete(wantProfiles, "user@1.example.com")
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Recreate profile manager from store after deleting default profile")
|
|
|
|
// Recreate the profile manager to ensure that it can load the profiles
|
|
|
|
// from the store at startup.
|
|
|
|
pm, err = newProfileManagerWithGOOS(store, logger.Discard, "", "linux")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
2022-11-16 16:17:36 +05:00
|
|
|
|
2022-11-17 19:05:02 +05:00
|
|
|
t.Logf("Create new profile - 2")
|
2022-11-16 16:17:36 +05:00
|
|
|
pm.NewProfile()
|
|
|
|
wantCurProfile = ""
|
|
|
|
wantProfiles[""] = emptyPrefs
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Login with the existing profile")
|
|
|
|
wantProfiles["user@2.example.com"] = setPrefs(t, "user@2.example.com")
|
|
|
|
delete(wantProfiles, "")
|
|
|
|
wantCurProfile = "user@2.example.com"
|
|
|
|
checkProfiles(t)
|
2022-11-17 19:05:02 +05:00
|
|
|
|
|
|
|
t.Logf("Tag the current the profile")
|
|
|
|
nodeIDs["tagged-node.2.ts.net"] = nodeIDs["user@2.example.com"]
|
|
|
|
wantProfiles["tagged-node.2.ts.net"] = setPrefs(t, "tagged-node.2.ts.net")
|
|
|
|
delete(wantProfiles, "user@2.example.com")
|
|
|
|
wantCurProfile = "tagged-node.2.ts.net"
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Relogin")
|
|
|
|
wantProfiles["user@2.example.com"] = setPrefs(t, "user@2.example.com")
|
|
|
|
delete(wantProfiles, "tagged-node.2.ts.net")
|
|
|
|
wantCurProfile = "user@2.example.com"
|
|
|
|
checkProfiles(t)
|
2022-11-09 10:58:10 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
// TestProfileManagementWindows tests going into and out of Unattended mode on
|
|
|
|
// Windows.
|
|
|
|
func TestProfileManagementWindows(t *testing.T) {
|
|
|
|
store := new(mem.Store)
|
|
|
|
|
|
|
|
pm, err := newProfileManagerWithGOOS(store, logger.Discard, "", "windows")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
wantCurProfile := ""
|
|
|
|
wantProfiles := map[string]ipn.PrefsView{
|
|
|
|
"": emptyPrefs,
|
|
|
|
}
|
|
|
|
checkProfiles := func(t *testing.T) {
|
|
|
|
t.Helper()
|
|
|
|
prof := pm.CurrentProfile()
|
|
|
|
t.Logf("\tCurrentProfile = %q", prof)
|
|
|
|
if prof.Name != wantCurProfile {
|
|
|
|
t.Fatalf("CurrentProfile = %q; want %q", prof, wantCurProfile)
|
|
|
|
}
|
|
|
|
if p := pm.CurrentPrefs(); !p.Equals(wantProfiles[wantCurProfile]) {
|
|
|
|
t.Fatalf("CurrentPrefs = %+v; want %+v", p.Pretty(), wantProfiles[wantCurProfile].Pretty())
|
|
|
|
}
|
|
|
|
}
|
2022-11-16 16:17:36 +05:00
|
|
|
logins := make(map[string]tailcfg.UserID)
|
2022-11-09 10:58:10 +05:00
|
|
|
setPrefs := func(t *testing.T, loginName string, forceDaemon bool) ipn.PrefsView {
|
2022-11-16 16:17:36 +05:00
|
|
|
id := logins[loginName]
|
|
|
|
if id.IsZero() {
|
|
|
|
id = tailcfg.UserID(len(logins) + 1)
|
|
|
|
logins[loginName] = id
|
|
|
|
}
|
2022-11-09 10:58:10 +05:00
|
|
|
p := pm.CurrentPrefs().AsStruct()
|
|
|
|
p.ForceDaemon = forceDaemon
|
|
|
|
p.Persist = &persist.Persist{
|
|
|
|
LoginName: loginName,
|
2022-11-16 16:17:36 +05:00
|
|
|
UserProfile: tailcfg.UserProfile{
|
|
|
|
ID: id,
|
|
|
|
LoginName: loginName,
|
|
|
|
},
|
2022-11-09 10:58:10 +05:00
|
|
|
}
|
|
|
|
if err := pm.SetPrefs(p.View()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return p.View()
|
|
|
|
}
|
|
|
|
t.Logf("Check initial state from empty store")
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
{
|
|
|
|
t.Logf("Set user1 as logged in user")
|
|
|
|
if err := pm.SetCurrentUser("user1"); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
t.Logf("Save prefs for user1")
|
|
|
|
wantProfiles["default"] = setPrefs(t, "default", false)
|
|
|
|
wantCurProfile = "default"
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
{
|
|
|
|
t.Logf("Create new profile")
|
|
|
|
pm.NewProfile()
|
|
|
|
wantCurProfile = ""
|
|
|
|
wantProfiles[""] = emptyPrefs
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
t.Logf("Save as test profile")
|
|
|
|
wantProfiles["test"] = setPrefs(t, "test", false)
|
|
|
|
wantCurProfile = "test"
|
|
|
|
checkProfiles(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Logf("Recreate profile manager from store, should reset prefs")
|
|
|
|
// Recreate the profile manager to ensure that it can load the profiles
|
|
|
|
// from the store at startup.
|
|
|
|
pm, err = newProfileManagerWithGOOS(store, logger.Discard, "", "windows")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
wantCurProfile = ""
|
|
|
|
wantProfiles[""] = emptyPrefs
|
|
|
|
checkProfiles(t)
|
|
|
|
|
|
|
|
{
|
|
|
|
t.Logf("Set user1 as current user")
|
|
|
|
if err := pm.SetCurrentUser("user1"); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
wantCurProfile = "test"
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
{
|
|
|
|
t.Logf("set unattended mode")
|
|
|
|
wantProfiles["test"] = setPrefs(t, "test", true)
|
|
|
|
}
|
|
|
|
if pm.CurrentUser() != "user1" {
|
|
|
|
t.Fatalf("CurrentUserID = %q; want %q", pm.CurrentUser(), "user1")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recreate the profile manager to ensure that it starts with test profile.
|
|
|
|
pm, err = newProfileManagerWithGOOS(store, logger.Discard, "", "windows")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
checkProfiles(t)
|
|
|
|
if pm.CurrentUser() != "user1" {
|
|
|
|
t.Fatalf("CurrentUserID = %q; want %q", pm.CurrentUser(), "user1")
|
|
|
|
}
|
|
|
|
}
|