2023-03-28 19:39:22 +01:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
2023-08-11 09:12:13 -04:00
|
|
|
// SPDX-License-Identifier: BUSL-1.1
|
2023-03-28 19:39:22 +01:00
|
|
|
|
2020-08-17 19:30:25 -04:00
|
|
|
package token
|
|
|
|
|
|
|
|
import (
|
2022-11-10 10:26:01 -06:00
|
|
|
"os"
|
2020-08-17 19:30:25 -04:00
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/hashicorp/go-hclog"
|
|
|
|
"github.com/stretchr/testify/require"
|
2023-09-20 15:50:06 -06:00
|
|
|
|
|
|
|
"github.com/hashicorp/consul/sdk/testutil"
|
2020-08-17 19:30:25 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestStore_Load(t *testing.T) {
|
|
|
|
dataDir := testutil.TempDir(t, "datadir")
|
|
|
|
tokenFile := filepath.Join(dataDir, tokensPath)
|
|
|
|
logger := hclog.New(nil)
|
|
|
|
store := new(Store)
|
|
|
|
|
|
|
|
t.Run("with empty store", func(t *testing.T) {
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
DataDir: dataDir,
|
|
|
|
ACLAgentToken: "alfa",
|
|
|
|
ACLAgentRecoveryToken: "bravo",
|
|
|
|
ACLDefaultToken: "charlie",
|
|
|
|
ACLReplicationToken: "delta",
|
|
|
|
ACLConfigFileRegistrationToken: "echo",
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDNSToken: "foxtrot",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
|
|
|
require.Equal(t, "alfa", store.AgentToken())
|
2021-12-07 12:12:47 +00:00
|
|
|
require.Equal(t, "bravo", store.AgentRecoveryToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
require.Equal(t, "charlie", store.UserToken())
|
|
|
|
require.Equal(t, "delta", store.ReplicationToken())
|
2023-01-10 10:24:02 -06:00
|
|
|
require.Equal(t, "echo", store.ConfigFileRegistrationToken())
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "foxtrot", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("updated from Config", func(t *testing.T) {
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
DataDir: dataDir,
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDefaultToken: "sierra",
|
|
|
|
ACLAgentToken: "tango",
|
|
|
|
ACLAgentRecoveryToken: "uniform",
|
|
|
|
ACLReplicationToken: "victor",
|
|
|
|
ACLConfigFileRegistrationToken: "xray",
|
|
|
|
ACLDNSToken: "zulu",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
// ensures no error for missing persisted tokens file
|
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "sierra", store.UserToken())
|
|
|
|
require.Equal(t, "tango", store.AgentToken())
|
|
|
|
require.Equal(t, "uniform", store.AgentRecoveryToken())
|
|
|
|
require.Equal(t, "victor", store.ReplicationToken())
|
|
|
|
require.Equal(t, "xray", store.ConfigFileRegistrationToken())
|
|
|
|
require.Equal(t, "zulu", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with persisted tokens", func(t *testing.T) {
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
DataDir: dataDir,
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDefaultToken: "alpha",
|
|
|
|
ACLAgentToken: "bravo",
|
|
|
|
ACLAgentRecoveryToken: "charlie",
|
|
|
|
ACLReplicationToken: "delta",
|
|
|
|
ACLConfigFileRegistrationToken: "echo",
|
|
|
|
ACLDNSToken: "foxtrot",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
tokens := `{
|
2023-09-20 15:50:06 -06:00
|
|
|
"agent" : "golf",
|
|
|
|
"agent_recovery" : "hotel",
|
|
|
|
"default": "india",
|
|
|
|
"replication": "juliet",
|
|
|
|
"config_file_service_registration": "kilo",
|
|
|
|
"dns": "lima"
|
2020-08-17 19:30:25 -04:00
|
|
|
}`
|
|
|
|
|
2022-11-10 10:26:01 -06:00
|
|
|
require.NoError(t, os.WriteFile(tokenFile, []byte(tokens), 0600))
|
2020-08-17 19:30:25 -04:00
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
|
|
|
|
|
|
|
// no updates since token persistence is not enabled
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "alpha", store.UserToken())
|
|
|
|
require.Equal(t, "bravo", store.AgentToken())
|
|
|
|
require.Equal(t, "charlie", store.AgentRecoveryToken())
|
|
|
|
require.Equal(t, "delta", store.ReplicationToken())
|
|
|
|
require.Equal(t, "echo", store.ConfigFileRegistrationToken())
|
|
|
|
require.Equal(t, "foxtrot", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
|
|
|
|
cfg.EnablePersistence = true
|
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
|
|
|
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "golf", store.AgentToken())
|
|
|
|
require.Equal(t, "hotel", store.AgentRecoveryToken())
|
|
|
|
require.Equal(t, "india", store.UserToken())
|
|
|
|
require.Equal(t, "juliet", store.ReplicationToken())
|
|
|
|
require.Equal(t, "kilo", store.ConfigFileRegistrationToken())
|
|
|
|
require.Equal(t, "lima", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
|
|
|
|
// check store persistence was enabled
|
|
|
|
require.NotNil(t, store.persistence)
|
|
|
|
})
|
|
|
|
|
2021-12-07 12:12:47 +00:00
|
|
|
t.Run("persisted tokens include pre-1.11 agent_master naming", func(t *testing.T) {
|
|
|
|
cfg := Config{
|
|
|
|
EnablePersistence: true,
|
|
|
|
DataDir: dataDir,
|
|
|
|
ACLAgentRecoveryToken: "golf",
|
|
|
|
}
|
|
|
|
|
|
|
|
tokens := `{"agent_master": "juliett"}`
|
2022-11-10 10:26:01 -06:00
|
|
|
require.NoError(t, os.WriteFile(tokenFile, []byte(tokens), 0600))
|
2021-12-07 12:12:47 +00:00
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
|
|
|
|
|
|
|
require.Equal(t, "juliett", store.AgentRecoveryToken())
|
|
|
|
})
|
|
|
|
|
2020-08-17 19:30:25 -04:00
|
|
|
t.Run("with persisted tokens, persisted tokens override config", func(t *testing.T) {
|
|
|
|
tokens := `{
|
|
|
|
"agent" : "mike",
|
2021-12-07 12:12:47 +00:00
|
|
|
"agent_recovery" : "november",
|
2020-08-17 19:30:25 -04:00
|
|
|
"default": "oscar",
|
2023-01-10 10:24:02 -06:00
|
|
|
"replication" : "papa",
|
2023-09-20 15:50:06 -06:00
|
|
|
"config_file_service_registration" : "lima",
|
|
|
|
"dns": "kilo"
|
2020-08-17 19:30:25 -04:00
|
|
|
}`
|
|
|
|
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
EnablePersistence: true,
|
|
|
|
DataDir: dataDir,
|
|
|
|
ACLDefaultToken: "quebec",
|
|
|
|
ACLAgentToken: "romeo",
|
|
|
|
ACLAgentRecoveryToken: "sierra",
|
|
|
|
ACLReplicationToken: "tango",
|
|
|
|
ACLConfigFileRegistrationToken: "uniform",
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDNSToken: "victor",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
|
2022-11-10 10:26:01 -06:00
|
|
|
require.NoError(t, os.WriteFile(tokenFile, []byte(tokens), 0600))
|
2020-08-17 19:30:25 -04:00
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
|
|
|
|
|
|
|
require.Equal(t, "mike", store.AgentToken())
|
2021-12-07 12:12:47 +00:00
|
|
|
require.Equal(t, "november", store.AgentRecoveryToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
require.Equal(t, "oscar", store.UserToken())
|
|
|
|
require.Equal(t, "papa", store.ReplicationToken())
|
2023-01-10 10:24:02 -06:00
|
|
|
require.Equal(t, "lima", store.ConfigFileRegistrationToken())
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "kilo", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with some persisted tokens", func(t *testing.T) {
|
|
|
|
tokens := `{
|
2023-09-20 15:50:06 -06:00
|
|
|
"agent" : "xray",
|
|
|
|
"agent_recovery" : "zulu"
|
2020-08-17 19:30:25 -04:00
|
|
|
}`
|
|
|
|
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
EnablePersistence: true,
|
|
|
|
DataDir: dataDir,
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDefaultToken: "alpha",
|
|
|
|
ACLAgentToken: "bravo",
|
|
|
|
ACLAgentRecoveryToken: "charlie",
|
|
|
|
ACLReplicationToken: "delta",
|
|
|
|
ACLConfigFileRegistrationToken: "echo",
|
|
|
|
ACLDNSToken: "foxtrot",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
|
2022-11-10 10:26:01 -06:00
|
|
|
require.NoError(t, os.WriteFile(tokenFile, []byte(tokens), 0600))
|
2020-08-17 19:30:25 -04:00
|
|
|
require.NoError(t, store.Load(cfg, logger))
|
|
|
|
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "xray", store.AgentToken())
|
|
|
|
require.Equal(t, "zulu", store.AgentRecoveryToken())
|
|
|
|
|
|
|
|
require.Equal(t, "alpha", store.UserToken())
|
|
|
|
require.Equal(t, "delta", store.ReplicationToken())
|
|
|
|
require.Equal(t, "echo", store.ConfigFileRegistrationToken())
|
|
|
|
require.Equal(t, "foxtrot", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("persisted file contains invalid data", func(t *testing.T) {
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
EnablePersistence: true,
|
|
|
|
DataDir: dataDir,
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDefaultToken: "alpha",
|
|
|
|
ACLAgentToken: "bravo",
|
|
|
|
ACLAgentRecoveryToken: "charlie",
|
|
|
|
ACLReplicationToken: "delta",
|
|
|
|
ACLConfigFileRegistrationToken: "echo",
|
|
|
|
ACLDNSToken: "foxtrot",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
|
2022-11-10 10:26:01 -06:00
|
|
|
require.NoError(t, os.WriteFile(tokenFile, []byte{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}, 0600))
|
2020-08-17 19:30:25 -04:00
|
|
|
err := store.Load(cfg, logger)
|
|
|
|
require.Error(t, err)
|
|
|
|
require.Contains(t, err.Error(), "failed to decode tokens file")
|
|
|
|
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "alpha", store.UserToken())
|
|
|
|
require.Equal(t, "bravo", store.AgentToken())
|
|
|
|
require.Equal(t, "charlie", store.AgentRecoveryToken())
|
|
|
|
require.Equal(t, "delta", store.ReplicationToken())
|
|
|
|
require.Equal(t, "echo", store.ConfigFileRegistrationToken())
|
|
|
|
require.Equal(t, "foxtrot", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("persisted file contains invalid json", func(t *testing.T) {
|
|
|
|
cfg := Config{
|
2023-01-10 10:24:02 -06:00
|
|
|
EnablePersistence: true,
|
|
|
|
DataDir: dataDir,
|
|
|
|
ACLDefaultToken: "alfa",
|
|
|
|
ACLAgentToken: "bravo",
|
|
|
|
ACLAgentRecoveryToken: "charlie",
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLReplicationToken: "delta",
|
|
|
|
ACLConfigFileRegistrationToken: "echo",
|
|
|
|
ACLDNSToken: "foxtrot",
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
|
2022-11-10 10:26:01 -06:00
|
|
|
require.NoError(t, os.WriteFile(tokenFile, []byte("[1,2,3]"), 0600))
|
2020-08-17 19:30:25 -04:00
|
|
|
err := store.Load(cfg, logger)
|
|
|
|
require.Error(t, err)
|
|
|
|
require.Contains(t, err.Error(), "failed to decode tokens file")
|
|
|
|
|
|
|
|
require.Equal(t, "alfa", store.UserToken())
|
|
|
|
require.Equal(t, "bravo", store.AgentToken())
|
2021-12-07 12:12:47 +00:00
|
|
|
require.Equal(t, "charlie", store.AgentRecoveryToken())
|
2023-09-20 15:50:06 -06:00
|
|
|
require.Equal(t, "delta", store.ReplicationToken())
|
|
|
|
require.Equal(t, "echo", store.ConfigFileRegistrationToken())
|
|
|
|
require.Equal(t, "foxtrot", store.DNSToken())
|
2020-08-17 19:30:25 -04:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStore_WithPersistenceLock(t *testing.T) {
|
2023-09-20 15:50:06 -06:00
|
|
|
// ACLDefaultToken: alpha --> sierra
|
|
|
|
// ACLAgentToken: bravo --> tango
|
|
|
|
// ACLAgentRecoveryToken: charlie --> uniform
|
|
|
|
// ACLReplicationToken: delta --> victor
|
|
|
|
// ACLConfigFileRegistrationToken: echo --> xray
|
|
|
|
// ACLDNSToken: foxtrot --> zulu
|
2023-01-10 10:24:02 -06:00
|
|
|
setupStore := func() (string, *Store) {
|
|
|
|
dataDir := testutil.TempDir(t, "datadir")
|
|
|
|
store := new(Store)
|
|
|
|
cfg := Config{
|
|
|
|
EnablePersistence: true,
|
|
|
|
DataDir: dataDir,
|
2023-09-20 15:50:06 -06:00
|
|
|
ACLDefaultToken: "alpha",
|
|
|
|
ACLAgentToken: "bravo",
|
|
|
|
ACLAgentRecoveryToken: "charlie",
|
|
|
|
ACLReplicationToken: "delta",
|
|
|
|
ACLConfigFileRegistrationToken: "echo",
|
|
|
|
ACLDNSToken: "foxtrot",
|
2023-01-10 10:24:02 -06:00
|
|
|
}
|
|
|
|
err := store.Load(cfg, hclog.New(nil))
|
|
|
|
require.NoError(t, err)
|
2020-08-17 19:30:25 -04:00
|
|
|
|
2023-01-10 10:24:02 -06:00
|
|
|
return dataDir, store
|
|
|
|
}
|
2020-08-17 19:30:25 -04:00
|
|
|
|
2023-01-10 10:24:02 -06:00
|
|
|
requirePersistedTokens := func(t *testing.T, dataDir string, expected persistedTokens) {
|
|
|
|
t.Helper()
|
|
|
|
tokens, err := readPersistedFromFile(filepath.Join(dataDir, tokensPath))
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, expected, tokens)
|
2020-08-17 19:30:25 -04:00
|
|
|
}
|
|
|
|
|
2023-01-10 10:24:02 -06:00
|
|
|
t.Run("persist some tokens", func(t *testing.T) {
|
|
|
|
dataDir, store := setupStore()
|
|
|
|
err := store.WithPersistenceLock(func() error {
|
2023-09-20 15:50:06 -06:00
|
|
|
require.True(t, store.UpdateUserToken("sierra", TokenSourceAPI))
|
|
|
|
require.True(t, store.UpdateAgentRecoveryToken("tango", TokenSourceAPI))
|
2023-01-10 10:24:02 -06:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Only API-sourced tokens are persisted.
|
|
|
|
requirePersistedTokens(t, dataDir, persistedTokens{
|
2023-09-20 15:50:06 -06:00
|
|
|
Default: "sierra",
|
|
|
|
AgentRecovery: "tango",
|
2023-01-10 10:24:02 -06:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("persist all tokens", func(t *testing.T) {
|
|
|
|
dataDir, store := setupStore()
|
|
|
|
err := store.WithPersistenceLock(func() error {
|
2023-09-20 15:50:06 -06:00
|
|
|
require.True(t, store.UpdateUserToken("sierra", TokenSourceAPI))
|
|
|
|
require.True(t, store.UpdateAgentToken("tango", TokenSourceAPI))
|
|
|
|
require.True(t, store.UpdateAgentRecoveryToken("uniform", TokenSourceAPI))
|
|
|
|
require.True(t, store.UpdateReplicationToken("victor", TokenSourceAPI))
|
|
|
|
require.True(t, store.UpdateConfigFileRegistrationToken("xray", TokenSourceAPI))
|
|
|
|
require.True(t, store.UpdateDNSToken("zulu", TokenSourceAPI))
|
2023-01-10 10:24:02 -06:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
requirePersistedTokens(t, dataDir, persistedTokens{
|
2023-09-20 15:50:06 -06:00
|
|
|
Default: "sierra",
|
|
|
|
Agent: "tango",
|
|
|
|
AgentRecovery: "uniform",
|
|
|
|
Replication: "victor",
|
|
|
|
ConfigFileRegistration: "xray",
|
|
|
|
DNS: "zulu",
|
2023-01-10 10:24:02 -06:00
|
|
|
})
|
|
|
|
})
|
2020-08-17 19:30:25 -04:00
|
|
|
|
|
|
|
}
|