Skip to content

Commit

Permalink
Add multi-account migration
Browse files Browse the repository at this point in the history
  • Loading branch information
williammartin committed Oct 18, 2023
1 parent c622ecb commit 3393abc
Show file tree
Hide file tree
Showing 2 changed files with 223 additions and 0 deletions.
124 changes: 124 additions & 0 deletions pkg/config/migration/multi_account.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,124 @@
package migration

import (
"fmt"

"github.com/cli/go-gh/v2/pkg/config"
)

type CowardlyRefusalError struct {
Reason string
}

func (e CowardlyRefusalError) Error() string {
// Consider whether we should add a call to action here like "open an issue with the contents of your redacted hosts.yml"
return fmt.Sprintf("cowardly refusing to continue with multi account migration: %s", e.Reason)
}

var hostsKey = []string{"hosts"}

// This migration exists to take a hosts section of the following structure:
//
// github.com:
// user: williammartin
// git_protocol: https
// editor: vim
// github.localhost:
// user: monalisa
// git_protocol: https
//
// We want this to migrate to something like:
//
// github.com:
// active_user: williammartin
// users:
// williammartin:
// git_protocol: https
// editor: vim
//
// github.localhost:
// active_user: monalisa
// users:
// monalisa:
// git_protocol: https
//
// The reason for this is that we can then add new users under a host.

type MultiAccount struct{}

func (m MultiAccount) Do(c *config.Config) (bool, error) {
hostnames, err := c.Keys(hostsKey)
// [github.com, github.localhost]
if err != nil {
return false, CowardlyRefusalError{"couldn't get hosts configuration"}
}

// If there are no hosts then it doesn't matter whether we migrate or not,
// so lets avoid any confusion and say there's no migration required.
if len(hostnames) == 0 {
return false, nil
}

// If there is an active user set for the first host then we must have already
// migrated, so no migration is required. Note that this is a little janky but
// without introducing a version into the hosts (which would require special handling elsewhere),
// or a version into the config (which would require config backup to move in sync with hosts backup)
// it seems like the best option for now.
if _, err := c.Get(append(hostsKey, hostnames[0], "active_user")); err == nil {
return false, nil
}

// Otherwise let's get to the business of migrating!
for _, hostname := range hostnames {
configEntryKeys, err := c.Keys(append(hostsKey, hostname))
// e.g. [user, git_protocol, editor]
if err != nil {
return false, CowardlyRefusalError{fmt.Sprintf("couldn't get host configuration despite %q existing", hostname)}
}

// Get the user so that we can nest under it in future
username, err := c.Get(append(hostsKey, hostname, "user"))
if err != nil {
panic(err)
}

for _, configEntryKey := range configEntryKeys {
// We would expect that these keys map directly to values
// e.g. [williammartin, https, vim] but it's possible that a manually
// edited config file might nest further but we don't support that.
//
// We could consider throwing away the nested values, but I suppose
// I'd rather make the user take a destructive action even if we have a backup.
// If they have configuration here, it's probably for a reason.
keys, err := c.Keys(append(hostsKey, hostname, configEntryKey))
if err == nil && len(keys) > 0 {
return false, CowardlyRefusalError{"hosts file has entries that are surprisingly deeply nested"}
}

configEntryValue, err := c.Get(append(hostsKey, hostname, configEntryKey))
if err != nil {
return false, CowardlyRefusalError{fmt.Sprintf("couldn't get configuration entry value despite %q / %q existing", hostname, configEntryKey)}
}

// Remove all these entries, because we are going to move
if err := c.Remove(append(hostsKey, hostname, configEntryKey)); err != nil {
return false, CowardlyRefusalError{fmt.Sprintf("couldn't remove configuration entry %q despite %q / %q existing", configEntryKey, hostname, configEntryKey)}
}

// If this is the user key, we don't need to do anything with it because it's
// now part of the final structure.
if configEntryKey == "user" {
continue
}

// Set these entries in their new location under the user
c.Set(append(hostsKey, hostname, "users", username, configEntryKey), configEntryValue)
}

// And after migrating all the existing values, we'll add one new "active" key to indicate the user
// that is active for this host after the migration.
c.Set(append(hostsKey, hostname, "active_user"), username)
}

return true, nil
}
99 changes: 99 additions & 0 deletions pkg/config/migration/multi_account_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
package migration_test

import (
"testing"

"github.com/cli/go-gh/v2/pkg/config"
"github.com/cli/go-gh/v2/pkg/config/migration"
"github.com/stretchr/testify/require"
)

func TestMigration(t *testing.T) {
cfg := config.ReadFromString(`
hosts:
github.com:
user: user1
oauth_token: xxxxxxxxxxxxxxxxxxxx
git_protocol: ssh
enterprise.com:
user: user2
oauth_token: yyyyyyyyyyyyyyyyyyyy
git_protocol: https
`)

var m migration.MultiAccount
required, err := m.Do(cfg)

require.NoError(t, err)
require.True(t, required, "migration should be required")

// Do some simple checks here for depth and multiple migrations
// but I don't really want to write a full tree traversal matcher.
requireKeyWithValue(t, cfg, []string{"hosts", "github.com", "active_user"}, "user1")
requireKeyWithValue(t, cfg, []string{"hosts", "github.com", "users", "user1", "git_protocol"}, "ssh")

requireKeyWithValue(t, cfg, []string{"hosts", "enterprise.com", "active_user"}, "user2")
requireKeyWithValue(t, cfg, []string{"hosts", "enterprise.com", "users", "user2", "git_protocol"}, "https")
}

func TestMigrationErrorsWithDeeplyNestedEntries(t *testing.T) {
cfg := config.ReadFromString(`
hosts:
github.com:
user: user1
nested:
too: deep
`)

var m migration.MultiAccount
_, err := m.Do(cfg)

require.ErrorContains(t, err, "hosts file has entries that are surprisingly deeply nested")
}

func TestMigrationReturnsNotRequredWhenNoHosts(t *testing.T) {
cfg := config.ReadFromString(`
hosts:
`)

var m migration.MultiAccount
required, err := m.Do(cfg)

require.NoError(t, err)
require.False(t, required, "migration should not be required when already migrated")
}

func TestMigrationReturnsNotRequiredWhenAlreadyMigrated(t *testing.T) {
cfg := config.ReadFromString(`
hosts:
github.com:
active_user: user1
users:
user1:
oauth_token: xxxxxxxxxxxxxxxxxxxx
git_protocol: ssh
`)

var m migration.MultiAccount
required, err := m.Do(cfg)

require.NoError(t, err)
require.False(t, required, "migration should not be required when already migrated")
}

func requireKeyWithValue(t *testing.T, cfg *config.Config, keys []string, value string) {
t.Helper()

actual, err := cfg.Get(keys)
require.NoError(t, err)

require.Equal(t, value, actual)
}

func requireNoKey(t *testing.T, cfg *config.Config, keys []string) {
t.Helper()

_, err := cfg.Get(keys)
var keyNotFoundError *config.KeyNotFoundError
require.ErrorAs(t, err, &keyNotFoundError)
}

0 comments on commit 3393abc

Please sign in to comment.