mirror of https://github.com/docker/docs.git
1397 lines
47 KiB
Go
1397 lines
47 KiB
Go
// Actually start up a notary server and run through basic TUF and key
|
|
// interactions via the client.
|
|
|
|
// Note - if using Yubikey, retrieving pins/touch doesn't seem to work right
|
|
// when running in the midst of all tests.
|
|
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/rand"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
ctxu "github.com/docker/distribution/context"
|
|
"github.com/docker/notary"
|
|
"github.com/docker/notary/cryptoservice"
|
|
"github.com/docker/notary/passphrase"
|
|
"github.com/docker/notary/server"
|
|
"github.com/docker/notary/server/storage"
|
|
"github.com/docker/notary/trustmanager"
|
|
"github.com/docker/notary/tuf/data"
|
|
"github.com/docker/notary/tuf/utils"
|
|
"github.com/spf13/cobra"
|
|
"github.com/stretchr/testify/assert"
|
|
"golang.org/x/net/context"
|
|
)
|
|
|
|
var testPassphrase = "passphrase"
|
|
var NewNotaryCommand func() *cobra.Command
|
|
|
|
// run a command and return the output as a string
|
|
func runCommand(t *testing.T, tempDir string, args ...string) (string, error) {
|
|
b := new(bytes.Buffer)
|
|
|
|
// Create an empty config file so we don't load the default on ~/.notary/config.json
|
|
configFile := filepath.Join(tempDir, "config.json")
|
|
|
|
cmd := NewNotaryCommand()
|
|
cmd.SetArgs(append([]string{"-c", configFile, "-d", tempDir}, args...))
|
|
cmd.SetOutput(b)
|
|
retErr := cmd.Execute()
|
|
output, err := ioutil.ReadAll(b)
|
|
assert.NoError(t, err)
|
|
|
|
// Clean up state to mimic running a fresh command next time
|
|
for _, command := range cmd.Commands() {
|
|
command.ResetFlags()
|
|
}
|
|
|
|
return string(output), retErr
|
|
}
|
|
|
|
func setupServerHandler(metaStore storage.MetaStore) http.Handler {
|
|
ctx := context.WithValue(context.Background(), "metaStore", metaStore)
|
|
|
|
ctx = context.WithValue(ctx, "keyAlgorithm", data.ECDSAKey)
|
|
|
|
// Eat the logs instead of spewing them out
|
|
var b bytes.Buffer
|
|
l := logrus.New()
|
|
l.Out = &b
|
|
ctx = ctxu.WithLogger(ctx, logrus.NewEntry(l))
|
|
|
|
cryptoService := cryptoservice.NewCryptoService(trustmanager.NewKeyMemoryStore(passphrase.ConstantRetriever("pass")))
|
|
return server.RootHandler(nil, ctx, cryptoService, nil, nil)
|
|
}
|
|
|
|
// makes a testing notary-server
|
|
func setupServer() *httptest.Server {
|
|
return httptest.NewServer(setupServerHandler(storage.NewMemStorage()))
|
|
}
|
|
|
|
// Initializes a repo, adds a target, publishes the target, lists the target,
|
|
// verifies the target, and then removes the target.
|
|
func TestClientTufInteraction(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
tempFile, err := ioutil.TempFile("", "targetfile")
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
defer os.Remove(tempFile.Name())
|
|
|
|
var (
|
|
output string
|
|
target = "sdgkadga"
|
|
)
|
|
// -- tests --
|
|
|
|
// init repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// add a target
|
|
_, err = runCommand(t, tempDir, "add", "gun", target, tempFile.Name())
|
|
assert.NoError(t, err)
|
|
|
|
// check status - see target
|
|
output, err = runCommand(t, tempDir, "status", "gun")
|
|
assert.NoError(t, err)
|
|
assert.True(t, strings.Contains(output, target))
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// check status - no targets
|
|
output, err = runCommand(t, tempDir, "status", "gun")
|
|
assert.NoError(t, err)
|
|
assert.False(t, strings.Contains(string(output), target))
|
|
|
|
// list repo - see target
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.True(t, strings.Contains(string(output), target))
|
|
|
|
// lookup target and repo - see target
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "lookup", "gun", target)
|
|
assert.NoError(t, err)
|
|
assert.True(t, strings.Contains(string(output), target))
|
|
|
|
// verify repo - empty file
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "verify", "gun", target)
|
|
assert.NoError(t, err)
|
|
|
|
// remove target
|
|
_, err = runCommand(t, tempDir, "remove", "gun", target)
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list repo - don't see target
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.False(t, strings.Contains(string(output), target))
|
|
}
|
|
|
|
// Initialize repo and test delegations commands by adding, listing, and removing delegations
|
|
func TestClientDelegationsInteraction(t *testing.T) {
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
// Setup certificate
|
|
tempFile, err := ioutil.TempFile("", "pemfile")
|
|
assert.NoError(t, err)
|
|
|
|
privKey, err := trustmanager.GenerateECDSAKey(rand.Reader)
|
|
startTime := time.Now()
|
|
endTime := startTime.AddDate(10, 0, 0)
|
|
cert, err := cryptoservice.GenerateCertificate(privKey, "gun", startTime, endTime)
|
|
assert.NoError(t, err)
|
|
|
|
_, err = tempFile.Write(trustmanager.CertToPEM(cert))
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
defer os.Remove(tempFile.Name())
|
|
|
|
rawPubBytes, _ := ioutil.ReadFile(tempFile.Name())
|
|
parsedPubKey, _ := trustmanager.ParsePEMPublicKey(rawPubBytes)
|
|
keyID, err := utils.CanonicalKeyID(parsedPubKey)
|
|
assert.NoError(t, err)
|
|
|
|
var output string
|
|
|
|
// -- tests --
|
|
|
|
// init repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - none yet
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No delegations present in this repository.")
|
|
|
|
// add new valid delegation with single new cert, and no path
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", tempFile.Name())
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Addition of delegation role")
|
|
assert.Contains(t, output, keyID)
|
|
assert.NotContains(t, output, "path")
|
|
|
|
// check status - see delegation
|
|
output, err = runCommand(t, tempDir, "status", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Unpublished changes for gun")
|
|
|
|
// list delegations - none yet because still unpublished
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No delegations present in this repository.")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// check status - no changelist
|
|
output, err = runCommand(t, tempDir, "status", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No unpublished changes for gun")
|
|
|
|
// list delegations - we should see our added delegation, with no paths
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "targets/delegation")
|
|
assert.Contains(t, output, keyID)
|
|
assert.NotContains(t, output, "\"\"")
|
|
|
|
// add all paths to this delegation
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--all-paths")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Addition of delegation role")
|
|
assert.Contains(t, output, "\"\"")
|
|
assert.Contains(t, output, "<all paths>")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see our added delegation, with no paths
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "targets/delegation")
|
|
assert.Contains(t, output, "\"\"")
|
|
assert.Contains(t, output, "<all paths>")
|
|
|
|
// Setup another certificate
|
|
tempFile2, err := ioutil.TempFile("", "pemfile2")
|
|
assert.NoError(t, err)
|
|
|
|
privKey, err = trustmanager.GenerateECDSAKey(rand.Reader)
|
|
startTime = time.Now()
|
|
endTime = startTime.AddDate(10, 0, 0)
|
|
cert, err = cryptoservice.GenerateCertificate(privKey, "gun", startTime, endTime)
|
|
assert.NoError(t, err)
|
|
|
|
_, err = tempFile2.Write(trustmanager.CertToPEM(cert))
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, err)
|
|
tempFile2.Close()
|
|
defer os.Remove(tempFile2.Name())
|
|
|
|
rawPubBytes2, _ := ioutil.ReadFile(tempFile2.Name())
|
|
parsedPubKey2, _ := trustmanager.ParsePEMPublicKey(rawPubBytes2)
|
|
keyID2, err := utils.CanonicalKeyID(parsedPubKey2)
|
|
assert.NoError(t, err)
|
|
|
|
// add to the delegation by specifying the same role, this time add a scoped path
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", tempFile2.Name(), "--paths", "path")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Addition of delegation role")
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see two keys
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, ",")
|
|
assert.Contains(t, output, "path")
|
|
assert.Contains(t, output, keyID)
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// remove the delegation's first key
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", keyID)
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Removal of delegation role")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see the delegation but with only the second key
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.NotContains(t, output, keyID)
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// remove the delegation's second key
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", keyID2)
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Removal of delegation role")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see no delegations
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No delegations present in this repository.")
|
|
|
|
// add delegation with multiple certs and multiple paths
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", tempFile.Name(), tempFile2.Name(), "--paths", "path1,path2")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Addition of delegation role")
|
|
assert.Contains(t, output, keyID)
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see two keys
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, ",")
|
|
assert.Contains(t, output, "path1,path2")
|
|
assert.Contains(t, output, keyID)
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// add delegation with multiple certs and multiple paths
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--paths", "path3")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Addition of delegation role")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see two keys
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, ",")
|
|
assert.Contains(t, output, "path1,path2,path3")
|
|
|
|
// just remove two paths from this delegation
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", "--paths", "path2,path3")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Removal of delegation role")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see the same two keys, and only path1
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, ",")
|
|
assert.Contains(t, output, "path1")
|
|
assert.NotContains(t, output, "path2")
|
|
assert.NotContains(t, output, "path3")
|
|
assert.Contains(t, output, keyID)
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// remove the remaining path, should not remove the delegation entirely
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", "--paths", "path1")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Removal of delegation role")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see the same two keys, and no paths
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, ",")
|
|
assert.NotContains(t, output, "path1")
|
|
assert.NotContains(t, output, "path2")
|
|
assert.NotContains(t, output, "path3")
|
|
assert.Contains(t, output, keyID)
|
|
assert.Contains(t, output, keyID2)
|
|
|
|
// Add a bunch of individual paths so we can test a delegation remove --all-paths
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--paths", "abcdef,123456")
|
|
assert.NoError(t, err)
|
|
|
|
// Add more individual paths so we can test a delegation remove --all-paths
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--paths", "banana/split,apple/crumble/pie,orange.peel,kiwi")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see all of our paths
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "abcdef")
|
|
assert.Contains(t, output, "123456")
|
|
assert.Contains(t, output, "banana/split")
|
|
assert.Contains(t, output, "apple/crumble/pie")
|
|
assert.Contains(t, output, "orange.peel")
|
|
assert.Contains(t, output, "kiwi")
|
|
|
|
// Try adding "", and check that adding it with other paths clears out the others
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--paths", "\"\",grapefruit,pomegranate")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see all of our old paths, and ""
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "abcdef")
|
|
assert.Contains(t, output, "123456")
|
|
assert.Contains(t, output, "banana/split")
|
|
assert.Contains(t, output, "apple/crumble/pie")
|
|
assert.Contains(t, output, "orange.peel")
|
|
assert.Contains(t, output, "kiwi")
|
|
assert.Contains(t, output, "\"\"")
|
|
assert.NotContains(t, output, "grapefruit")
|
|
assert.NotContains(t, output, "pomegranate")
|
|
|
|
// Try removing just ""
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", "--paths", "\"\"")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see all of our old paths without ""
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "abcdef")
|
|
assert.Contains(t, output, "123456")
|
|
assert.Contains(t, output, "banana/split")
|
|
assert.Contains(t, output, "apple/crumble/pie")
|
|
assert.Contains(t, output, "orange.peel")
|
|
assert.Contains(t, output, "kiwi")
|
|
assert.NotContains(t, output, "\"\"")
|
|
|
|
// Remove --all-paths to clear out all paths from this delegation
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", "--all-paths")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see all of our paths
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.NotContains(t, output, "abcdef")
|
|
assert.NotContains(t, output, "123456")
|
|
assert.NotContains(t, output, "banana/split")
|
|
assert.NotContains(t, output, "apple/crumble/pie")
|
|
assert.NotContains(t, output, "orange.peel")
|
|
assert.NotContains(t, output, "kiwi")
|
|
|
|
// Check that we ignore other --paths if we pass in --all-paths on an add
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--all-paths", "--paths", "grapefruit,pomegranate")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should only see "", and not the other paths specified
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "\"\"")
|
|
assert.NotContains(t, output, "grapefruit")
|
|
assert.NotContains(t, output, "pomegranate")
|
|
|
|
// Add those extra paths we ignored to set up the next test
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/delegation", "--paths", "grapefruit,pomegranate")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// Check that we ignore other --paths if we pass in --all-paths on a remove
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", "--all-paths", "--paths", "pomegranate")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see no paths
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.NotContains(t, output, "\"\"")
|
|
assert.NotContains(t, output, "grapefruit")
|
|
assert.NotContains(t, output, "pomegranate")
|
|
|
|
// remove by force to delete the delegation entirely
|
|
output, err = runCommand(t, tempDir, "delegation", "remove", "gun", "targets/delegation", "-y")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Forced removal (including all keys and paths) of delegation role")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see no delegations
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No delegations present in this repository.")
|
|
}
|
|
|
|
// Initialize repo and test publishing targets with delegation roles
|
|
func TestClientDelegationsPublishing(t *testing.T) {
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
// Setup certificate for delegation role
|
|
tempFile, err := ioutil.TempFile("", "pemfile")
|
|
assert.NoError(t, err)
|
|
|
|
privKey, err := trustmanager.GenerateRSAKey(rand.Reader, 2048)
|
|
assert.NoError(t, err)
|
|
privKeyBytesNoRole, err := trustmanager.KeyToPEM(privKey, "")
|
|
assert.NoError(t, err)
|
|
privKeyBytesWithRole, err := trustmanager.KeyToPEM(privKey, "user")
|
|
assert.NoError(t, err)
|
|
startTime := time.Now()
|
|
endTime := startTime.AddDate(10, 0, 0)
|
|
cert, err := cryptoservice.GenerateCertificate(privKey, "gun", startTime, endTime)
|
|
assert.NoError(t, err)
|
|
|
|
_, err = tempFile.Write(trustmanager.CertToPEM(cert))
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
defer os.Remove(tempFile.Name())
|
|
|
|
rawPubBytes, _ := ioutil.ReadFile(tempFile.Name())
|
|
parsedPubKey, _ := trustmanager.ParsePEMPublicKey(rawPubBytes)
|
|
canonicalKeyID, err := utils.CanonicalKeyID(parsedPubKey)
|
|
assert.NoError(t, err)
|
|
|
|
// Set up targets for publishing
|
|
tempTargetFile, err := ioutil.TempFile("", "targetfile")
|
|
assert.NoError(t, err)
|
|
tempTargetFile.Close()
|
|
defer os.Remove(tempTargetFile.Name())
|
|
|
|
var target = "sdgkadga"
|
|
|
|
var output string
|
|
|
|
// init repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - none yet
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No delegations present in this repository.")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// validate that we have all keys, including snapshot
|
|
assertNumKeys(t, tempDir, 1, 2, true)
|
|
|
|
// rotate the snapshot key to server
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "key", "rotate", "gun", "snapshot", "-r")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// validate that we lost the snapshot signing key
|
|
_, signingKeyIDs := assertNumKeys(t, tempDir, 1, 1, true)
|
|
targetKeyID := signingKeyIDs[0]
|
|
|
|
// add new valid delegation with single new cert
|
|
output, err = runCommand(t, tempDir, "delegation", "add", "gun", "targets/releases", tempFile.Name(), "--paths", "\"\"")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "Addition of delegation role")
|
|
assert.Contains(t, output, canonicalKeyID)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list delegations - we should see our one delegation
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "delegation", "list", "gun")
|
|
assert.NoError(t, err)
|
|
assert.NotContains(t, output, "No delegations present in this repository.")
|
|
|
|
// remove the targets key to demonstrate that delegates don't need this key
|
|
keyDir := filepath.Join(tempDir, "private", "tuf_keys")
|
|
assert.NoError(t, os.Remove(filepath.Join(keyDir, "gun", targetKeyID+".key")))
|
|
|
|
// Note that we need to use the canonical key ID, followed by the base of the role here
|
|
err = ioutil.WriteFile(filepath.Join(keyDir, canonicalKeyID+"_releases.key"), privKeyBytesNoRole, 0700)
|
|
assert.NoError(t, err)
|
|
|
|
// add a target using the delegation -- will only add to targets/releases
|
|
_, err = runCommand(t, tempDir, "add", "gun", target, tempTargetFile.Name(), "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see no targets until we publish
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No targets")
|
|
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "status", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see our target!
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "targets/releases")
|
|
|
|
// remove the target for this role only
|
|
_, err = runCommand(t, tempDir, "remove", "gun", target, "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see no targets
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No targets present")
|
|
|
|
// Try adding a target with a different key style - private/tuf_keys/canonicalKeyID.key with "user" set as the "role" PEM header
|
|
// First remove the old key and add the new style
|
|
assert.NoError(t, os.Remove(filepath.Join(keyDir, canonicalKeyID+"_releases.key")))
|
|
err = ioutil.WriteFile(filepath.Join(keyDir, canonicalKeyID+".key"), privKeyBytesWithRole, 0700)
|
|
assert.NoError(t, err)
|
|
|
|
// add a target using the delegation -- will only add to targets/releases
|
|
_, err = runCommand(t, tempDir, "add", "gun", target, tempTargetFile.Name(), "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see no targets until we publish
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No targets")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see our target!
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "targets/releases")
|
|
|
|
// remove the target for this role only
|
|
_, err = runCommand(t, tempDir, "remove", "gun", target, "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// Now remove this key, and make a new file to import the delegation's key from
|
|
assert.NoError(t, os.Remove(filepath.Join(keyDir, canonicalKeyID+".key")))
|
|
tempPrivFile, err := ioutil.TempFile("/tmp", "privfile")
|
|
assert.NoError(t, err)
|
|
defer os.Remove(tempPrivFile.Name())
|
|
|
|
// Write the private key to a file so we can import it
|
|
_, err = tempPrivFile.Write(privKeyBytesNoRole)
|
|
assert.NoError(t, err)
|
|
tempPrivFile.Close()
|
|
|
|
// Import the private key, associating it with our delegation role
|
|
_, err = runCommand(t, tempDir, "key", "import", tempPrivFile.Name(), "--role", "targets/releases")
|
|
assert.NoError(t, err)
|
|
|
|
// add a target using the delegation -- will only add to targets/releases
|
|
_, err = runCommand(t, tempDir, "add", "gun", target, tempTargetFile.Name(), "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see no targets until we publish
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "No targets")
|
|
|
|
// publish repo
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "publish", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
// list targets for targets/releases - we should see our target!
|
|
output, err = runCommand(t, tempDir, "-s", server.URL, "list", "gun", "--roles", "targets/releases")
|
|
assert.NoError(t, err)
|
|
assert.Contains(t, output, "targets/releases")
|
|
}
|
|
|
|
// Splits a string into lines, and returns any lines that are not empty (
|
|
// striped of whitespace)
|
|
func splitLines(chunk string) []string {
|
|
splitted := strings.Split(strings.TrimSpace(chunk), "\n")
|
|
var results []string
|
|
|
|
for _, line := range splitted {
|
|
line := strings.TrimSpace(line)
|
|
if line != "" {
|
|
results = append(results, line)
|
|
}
|
|
}
|
|
|
|
return results
|
|
}
|
|
|
|
// List keys, parses the output, and returns the unique key IDs as an array
|
|
// of root key IDs and an array of signing key IDs. Output expected looks like:
|
|
// ROLE GUN KEY ID LOCATION
|
|
// ----------------------------------------------------------------
|
|
// root 8bd63a896398b558ac... file (.../private)
|
|
// snapshot repo e9e9425cd9a85fc7a5... file (.../private)
|
|
// targets repo f5b84e2d92708c5acb... file (.../private)
|
|
func getUniqueKeys(t *testing.T, tempDir string) ([]string, []string) {
|
|
output, err := runCommand(t, tempDir, "key", "list")
|
|
assert.NoError(t, err)
|
|
lines := splitLines(output)
|
|
if len(lines) == 1 && lines[0] == "No signing keys found." {
|
|
return []string{}, []string{}
|
|
}
|
|
if len(lines) < 3 { // 2 lines of header, at least 1 line with keys
|
|
t.Logf("This output is not what is expected by the test:\n%s", output)
|
|
}
|
|
|
|
var (
|
|
rootMap = make(map[string]bool)
|
|
nonrootMap = make(map[string]bool)
|
|
root []string
|
|
nonroot []string
|
|
)
|
|
// first two lines are header
|
|
for _, line := range lines[2:] {
|
|
parts := strings.Fields(line)
|
|
var (
|
|
placeToGo map[string]bool
|
|
keyID string
|
|
)
|
|
if strings.TrimSpace(parts[0]) == "root" {
|
|
// no gun, so there are only 3 fields
|
|
placeToGo, keyID = rootMap, parts[1]
|
|
} else {
|
|
// gun comes between role and key ID
|
|
placeToGo, keyID = nonrootMap, parts[2]
|
|
}
|
|
// keys are 32-chars long (32 byte shasum, hex-encoded)
|
|
assert.Len(t, keyID, 64)
|
|
placeToGo[keyID] = true
|
|
}
|
|
for k := range rootMap {
|
|
root = append(root, k)
|
|
}
|
|
for k := range nonrootMap {
|
|
nonroot = append(nonroot, k)
|
|
}
|
|
|
|
return root, nonroot
|
|
}
|
|
|
|
// List keys, parses the output, and asserts something about the number of root
|
|
// keys and number of signing keys, as well as returning them.
|
|
func assertNumKeys(t *testing.T, tempDir string, numRoot, numSigning int,
|
|
rootOnDisk bool) ([]string, []string) {
|
|
|
|
root, signing := getUniqueKeys(t, tempDir)
|
|
assert.Len(t, root, numRoot)
|
|
assert.Len(t, signing, numSigning)
|
|
for _, rootKeyID := range root {
|
|
_, err := os.Stat(filepath.Join(
|
|
tempDir, "private", "root_keys", rootKeyID+".key"))
|
|
// os.IsExist checks to see if the error is because a file already
|
|
// exists, and hence it isn't actually the right function to use here
|
|
assert.Equal(t, rootOnDisk, !os.IsNotExist(err))
|
|
|
|
// this function is declared is in the build-tagged setup files
|
|
verifyRootKeyOnHardware(t, rootKeyID)
|
|
}
|
|
return root, signing
|
|
}
|
|
|
|
// Adds the given target to the gun, publishes it, and lists it to ensure that
|
|
// it appears. Returns the listing output.
|
|
func assertSuccessfullyPublish(
|
|
t *testing.T, tempDir, url, gun, target, fname string) string {
|
|
|
|
_, err := runCommand(t, tempDir, "add", gun, target, fname)
|
|
assert.NoError(t, err)
|
|
|
|
_, err = runCommand(t, tempDir, "-s", url, "publish", gun)
|
|
assert.NoError(t, err)
|
|
|
|
output, err := runCommand(t, tempDir, "-s", url, "list", gun)
|
|
assert.NoError(t, err)
|
|
assert.True(t, strings.Contains(string(output), target))
|
|
|
|
return output
|
|
}
|
|
|
|
// Tests root key generation and key rotation
|
|
func TestClientKeyGenerationRotation(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
tempfiles := make([]string, 2)
|
|
for i := 0; i < 2; i++ {
|
|
tempFile, err := ioutil.TempFile("", "targetfile")
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
tempfiles[i] = tempFile.Name()
|
|
defer os.Remove(tempFile.Name())
|
|
}
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
var target = "sdgkadga"
|
|
|
|
// -- tests --
|
|
|
|
// starts out with no keys
|
|
assertNumKeys(t, tempDir, 0, 0, true)
|
|
|
|
// generate root key produces a single root key and no other keys
|
|
_, err := runCommand(t, tempDir, "key", "generate", data.ECDSAKey)
|
|
assert.NoError(t, err)
|
|
assertNumKeys(t, tempDir, 1, 0, true)
|
|
|
|
// initialize a repo, should have signing keys and no new root key
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun")
|
|
assert.NoError(t, err)
|
|
origRoot, origSign := assertNumKeys(t, tempDir, 1, 2, true)
|
|
|
|
// publish using the original keys
|
|
assertSuccessfullyPublish(t, tempDir, server.URL, "gun", target, tempfiles[0])
|
|
|
|
// rotate the signing keys
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "key", "rotate", "gun", data.CanonicalSnapshotRole)
|
|
assert.NoError(t, err)
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "key", "rotate", "gun", data.CanonicalTargetsRole)
|
|
assert.NoError(t, err)
|
|
root, sign := assertNumKeys(t, tempDir, 1, 2, true)
|
|
assert.Equal(t, origRoot[0], root[0])
|
|
|
|
// just do a cursory rotation check that the keys aren't equal anymore
|
|
for _, origKey := range origSign {
|
|
for _, key := range sign {
|
|
assert.NotEqual(
|
|
t, key, origKey, "One of the signing keys was not removed")
|
|
}
|
|
}
|
|
|
|
// publish using the new keys
|
|
output := assertSuccessfullyPublish(
|
|
t, tempDir, server.URL, "gun", target+"2", tempfiles[1])
|
|
// assert that the previous target is sitll there
|
|
assert.True(t, strings.Contains(string(output), target))
|
|
}
|
|
|
|
// Tests backup/restore root+signing keys - repo with restored keys should be
|
|
// able to publish successfully
|
|
func TestClientKeyBackupAndRestore(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
dirs := make([]string, 3)
|
|
for i := 0; i < 3; i++ {
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
dirs[i] = tempDir
|
|
}
|
|
|
|
tempfiles := make([]string, 2)
|
|
for i := 0; i < 2; i++ {
|
|
tempFile, err := ioutil.TempFile("", "tempfile")
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
tempfiles[i] = tempFile.Name()
|
|
defer os.Remove(tempFile.Name())
|
|
}
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
var (
|
|
target = "sdgkadga"
|
|
err error
|
|
)
|
|
|
|
// create two repos and publish a target
|
|
for _, gun := range []string{"gun1", "gun2"} {
|
|
_, err = runCommand(t, dirs[0], "-s", server.URL, "init", gun)
|
|
assert.NoError(t, err)
|
|
|
|
assertSuccessfullyPublish(
|
|
t, dirs[0], server.URL, gun, target, tempfiles[0])
|
|
}
|
|
assertNumKeys(t, dirs[0], 1, 4, true)
|
|
|
|
// -- tests --
|
|
zipfile := tempfiles[0] + ".zip"
|
|
defer os.Remove(zipfile)
|
|
|
|
// backup then restore all keys
|
|
_, err = runCommand(t, dirs[0], "key", "backup", zipfile)
|
|
assert.NoError(t, err)
|
|
|
|
_, err = runCommand(t, dirs[1], "key", "restore", zipfile)
|
|
assert.NoError(t, err)
|
|
// all keys should be there, including root because the root key was backed up to disk,
|
|
// and export just backs up all the keys on disk
|
|
assertNumKeys(t, dirs[1], 1, 4, !rootOnHardware())
|
|
|
|
// can list and publish to both repos using restored keys
|
|
for _, gun := range []string{"gun1", "gun2"} {
|
|
output, err := runCommand(t, dirs[1], "-s", server.URL, "list", gun)
|
|
assert.NoError(t, err)
|
|
assert.True(t, strings.Contains(string(output), target))
|
|
|
|
assertSuccessfullyPublish(
|
|
t, dirs[1], server.URL, gun, target+"2", tempfiles[1])
|
|
}
|
|
|
|
// backup and restore keys for one gun
|
|
_, err = runCommand(t, dirs[0], "key", "backup", zipfile, "-g", "gun1")
|
|
assert.NoError(t, err)
|
|
|
|
_, err = runCommand(t, dirs[2], "key", "restore", zipfile)
|
|
assert.NoError(t, err)
|
|
|
|
// this function is declared is in the build-tagged setup files
|
|
if rootOnHardware() {
|
|
// hardware root is still present, and the key will ONLY be on hardware
|
|
// and not on disk
|
|
assertNumKeys(t, dirs[2], 1, 2, false)
|
|
} else {
|
|
// only 2 signing keys should be there, and no root key
|
|
assertNumKeys(t, dirs[2], 0, 2, true)
|
|
}
|
|
}
|
|
|
|
// Generate a root key and export the root key only. Return the key ID
|
|
// exported.
|
|
func exportRoot(t *testing.T, exportTo string) string {
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
// generate root key produces a single root key and no other keys
|
|
_, err := runCommand(t, tempDir, "key", "generate", data.ECDSAKey)
|
|
assert.NoError(t, err)
|
|
oldRoot, _ := assertNumKeys(t, tempDir, 1, 0, true)
|
|
|
|
// export does not require a password
|
|
oldNewCommand := NewNotaryCommand
|
|
NewNotaryCommand = func() *cobra.Command {
|
|
commander := ¬aryCommander{getRetriever: func() passphrase.Retriever { return nil }}
|
|
return commander.GetCommand()
|
|
}
|
|
defer func() { // but import will, later
|
|
NewNotaryCommand = oldNewCommand
|
|
}()
|
|
|
|
_, err = runCommand(
|
|
t, tempDir, "key", "export", oldRoot[0], exportTo)
|
|
assert.NoError(t, err)
|
|
|
|
return oldRoot[0]
|
|
}
|
|
|
|
// Tests import/export root key only
|
|
func TestClientKeyImportExportRootOnly(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
var (
|
|
target = "sdgkadga"
|
|
rootKeyID string
|
|
)
|
|
|
|
tempFile, err := ioutil.TempFile("", "pemfile")
|
|
assert.NoError(t, err)
|
|
// close later, because we might need to write to it
|
|
defer os.Remove(tempFile.Name())
|
|
|
|
// -- tests --
|
|
|
|
if rootOnHardware() {
|
|
t.Log("Cannot export a key from hardware. Will generate one to import.")
|
|
|
|
privKey, err := trustmanager.GenerateECDSAKey(rand.Reader)
|
|
assert.NoError(t, err)
|
|
|
|
pemBytes, err := trustmanager.EncryptPrivateKey(privKey, "root", testPassphrase)
|
|
assert.NoError(t, err)
|
|
|
|
nBytes, err := tempFile.Write(pemBytes)
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
assert.Equal(t, len(pemBytes), nBytes)
|
|
rootKeyID = privKey.ID()
|
|
} else {
|
|
tempFile.Close()
|
|
rootKeyID = exportRoot(t, tempFile.Name())
|
|
}
|
|
|
|
// import the key
|
|
_, err = runCommand(t, tempDir, "key", "import", tempFile.Name())
|
|
assert.NoError(t, err)
|
|
|
|
// if there is hardware available, root will only be on hardware, and not
|
|
// on disk
|
|
newRoot, _ := assertNumKeys(t, tempDir, 1, 0, !rootOnHardware())
|
|
assert.Equal(t, rootKeyID, newRoot[0])
|
|
|
|
// Just to make sure, init a repo and publish
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun")
|
|
assert.NoError(t, err)
|
|
assertNumKeys(t, tempDir, 1, 2, !rootOnHardware())
|
|
assertSuccessfullyPublish(
|
|
t, tempDir, server.URL, "gun", target, tempFile.Name())
|
|
|
|
// Now assert that bad root keys give an error
|
|
// Try importing an unencrypted root key:
|
|
privKey, err := trustmanager.GenerateECDSAKey(rand.Reader)
|
|
assert.NoError(t, err)
|
|
decryptedPEMBytes, err := trustmanager.KeyToPEM(privKey, data.CanonicalRootRole)
|
|
decryptedKeyFile, err := ioutil.TempFile("", "decryptedPem")
|
|
assert.NoError(t, err)
|
|
// close later, because we might need to write to it
|
|
defer os.Remove(decryptedKeyFile.Name())
|
|
|
|
nBytes, err := decryptedKeyFile.Write(decryptedPEMBytes)
|
|
assert.NoError(t, err)
|
|
decryptedKeyFile.Close()
|
|
assert.Equal(t, len(decryptedPEMBytes), nBytes)
|
|
// import the key
|
|
_, err = runCommand(t, tempDir, "key", "import", decryptedKeyFile.Name())
|
|
assert.Error(t, err)
|
|
|
|
// Now try importing an invalid PEM as a root key
|
|
invalidPEMBytes := []byte("this is not PEM")
|
|
invalidPEMFile, err := ioutil.TempFile("", "invalidPem")
|
|
assert.NoError(t, err)
|
|
// close later, because we might need to write to it
|
|
defer os.Remove(invalidPEMFile.Name())
|
|
|
|
nBytes, err = invalidPEMFile.Write(invalidPEMBytes)
|
|
assert.NoError(t, err)
|
|
invalidPEMFile.Close()
|
|
assert.Equal(t, len(invalidPEMBytes), nBytes)
|
|
// import the key
|
|
_, err = runCommand(t, tempDir, "key", "import", invalidPEMFile.Name())
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
// Helper method to get the subdirectory for TUF keys
|
|
func getKeySubdir(role, gun string) string {
|
|
subdir := notary.PrivDir
|
|
switch role {
|
|
case data.CanonicalRootRole:
|
|
return filepath.Join(subdir, notary.RootKeysSubdir)
|
|
case data.CanonicalTargetsRole:
|
|
return filepath.Join(subdir, notary.NonRootKeysSubdir, gun)
|
|
case data.CanonicalSnapshotRole:
|
|
return filepath.Join(subdir, notary.NonRootKeysSubdir, gun)
|
|
default:
|
|
return filepath.Join(subdir, notary.NonRootKeysSubdir)
|
|
}
|
|
}
|
|
|
|
// Tests importing and exporting keys for all different roles and GUNs
|
|
func TestClientKeyImportExportAllRoles(t *testing.T) {
|
|
if rootOnHardware() {
|
|
t.Log("Cannot import or export a non-root key from hardware. Will skip test.")
|
|
return
|
|
}
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
// -- tests --
|
|
_, err := runCommand(t, tempDir, "-s", server.URL, "init", "gun")
|
|
assert.NoError(t, err)
|
|
|
|
testRoles := append(data.BaseRoles, "targets/releases")
|
|
// Test importing and exporting keys to all base roles and delegation role
|
|
for _, role := range testRoles {
|
|
// Do this while importing keys that have the PEM header role set or have --role set on import
|
|
for _, setKeyRole := range []bool{true, false} {
|
|
// Make a new key for this role
|
|
privKey, err := trustmanager.GenerateECDSAKey(rand.Reader)
|
|
assert.NoError(t, err)
|
|
|
|
// Make a tempfile for importing
|
|
tempFile, err := ioutil.TempFile("", "pemfile")
|
|
assert.NoError(t, err)
|
|
|
|
// Specify the role in the PEM header
|
|
pemBytes, err := trustmanager.EncryptPrivateKey(privKey, role, testPassphrase)
|
|
assert.NoError(t, err)
|
|
ioutil.WriteFile(tempFile.Name(), pemBytes, 0644)
|
|
|
|
// If we need to set the key role with the --role flag, do so on import
|
|
if setKeyRole {
|
|
// If it's targets/snapshot we must specify the GUN
|
|
if role == data.CanonicalTargetsRole || role == data.CanonicalSnapshotRole {
|
|
_, err = runCommand(t, tempDir, "key", "import", tempFile.Name(), "--gun", "gun", "--role", role)
|
|
} else {
|
|
_, err = runCommand(t, tempDir, "key", "import", tempFile.Name(), "--role", role)
|
|
}
|
|
} else {
|
|
// If it's targets/snapshot we must specify the GUN
|
|
if role == data.CanonicalTargetsRole || role == data.CanonicalSnapshotRole {
|
|
_, err = runCommand(t, tempDir, "key", "import", tempFile.Name(), "--gun", "gun")
|
|
} else {
|
|
_, err = runCommand(t, tempDir, "key", "import", tempFile.Name())
|
|
}
|
|
}
|
|
assert.NoError(t, err)
|
|
|
|
// Test that we imported correctly
|
|
keySubdir := getKeySubdir(role, "gun")
|
|
_, err = os.Stat(filepath.Join(tempDir, keySubdir, privKey.ID()+".key"))
|
|
assert.Nil(t, err)
|
|
|
|
// Remove the input file so we can test exporting
|
|
assert.NoError(t, os.Remove(tempFile.Name()))
|
|
|
|
// Make a tempfile for exporting to
|
|
tempFile, err = ioutil.TempFile("", "pemfile")
|
|
assert.NoError(t, err)
|
|
|
|
// Ensure exporting this key by ID gets the same key
|
|
_, err = runCommand(t, tempDir, "key", "export", privKey.ID(), tempFile.Name())
|
|
assert.NoError(t, err)
|
|
// Compare the bytes of the exported file and the root key file in the repo
|
|
exportedBytes, err := ioutil.ReadFile(tempFile.Name())
|
|
assert.NoError(t, err)
|
|
repoBytes, err := ioutil.ReadFile(filepath.Join(tempDir, keySubdir, privKey.ID()+".key"))
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, repoBytes, exportedBytes)
|
|
|
|
// Ensure exporting this key and changing the passphrase works
|
|
_, err = runCommand(t, tempDir, "key", "export", privKey.ID(), tempFile.Name(), "-p")
|
|
assert.NoError(t, err)
|
|
|
|
// Remove the export file for cleanup
|
|
assert.NoError(t, os.Remove(tempFile.Name()))
|
|
}
|
|
}
|
|
}
|
|
|
|
func assertNumCerts(t *testing.T, tempDir string, expectedNum int) []string {
|
|
output, err := runCommand(t, tempDir, "cert", "list")
|
|
assert.NoError(t, err)
|
|
lines := splitLines(strings.TrimSpace(output))
|
|
|
|
if expectedNum == 0 {
|
|
assert.Len(t, lines, 1)
|
|
assert.Equal(t, "No trusted root certificates present.", lines[0])
|
|
return []string{}
|
|
}
|
|
|
|
assert.Len(t, lines, expectedNum+2)
|
|
return lines[2:]
|
|
}
|
|
|
|
// TestClientCertInteraction
|
|
func TestClientCertInteraction(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
// -- tests --
|
|
_, err := runCommand(t, tempDir, "-s", server.URL, "init", "gun1")
|
|
assert.NoError(t, err)
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun2")
|
|
assert.NoError(t, err)
|
|
certs := assertNumCerts(t, tempDir, 2)
|
|
// root is always on disk, because even if there's a yubikey a backup is created
|
|
assertNumKeys(t, tempDir, 1, 4, true)
|
|
|
|
// remove certs for one gun
|
|
_, err = runCommand(t, tempDir, "cert", "remove", "-g", "gun1", "-y")
|
|
assert.NoError(t, err)
|
|
certs = assertNumCerts(t, tempDir, 1)
|
|
// assert that when we remove cert by gun, we do not remove repo signing keys
|
|
// (root is always on disk, because even if there's a yubikey a backup is created)
|
|
assertNumKeys(t, tempDir, 1, 4, true)
|
|
// assert that when we remove cert by gun, we also remove TUF metadata
|
|
_, err = os.Stat(filepath.Join(tempDir, "tuf", "gun1"))
|
|
assert.Error(t, err)
|
|
|
|
// remove a single cert
|
|
certID := strings.Fields(certs[0])[1]
|
|
// passing an empty gun here because the string for the previous gun has
|
|
// has already been stored (a drawback of running these commands without)
|
|
// shelling out
|
|
_, err = runCommand(t, tempDir, "cert", "remove", certID, "-y", "-g", "")
|
|
assert.NoError(t, err)
|
|
assertNumCerts(t, tempDir, 0)
|
|
// assert that when we remove the last cert ID for a gun, we also remove TUF metadata
|
|
_, err = os.Stat(filepath.Join(tempDir, "tuf", "gun2"))
|
|
assert.Error(t, err)
|
|
|
|
// Setup certificate with nonexistent repo GUN
|
|
// Check that we can only remove one certificate when specifying one ID
|
|
startTime := time.Now()
|
|
privKey, err := trustmanager.GenerateECDSAKey(rand.Reader)
|
|
assert.NoError(t, err)
|
|
noGunCert, err := cryptoservice.GenerateCertificate(
|
|
privKey, "nonexistent", startTime, startTime.AddDate(10, 0, 0))
|
|
assert.NoError(t, err)
|
|
certStore, err := trustmanager.NewX509FileStore(filepath.Join(tempDir, "trusted_certificates"))
|
|
assert.NoError(t, err)
|
|
err = certStore.AddCert(noGunCert)
|
|
assert.NoError(t, err)
|
|
|
|
certs = assertNumCerts(t, tempDir, 1)
|
|
certID = strings.Fields(certs[0])[1]
|
|
|
|
privKey, err = trustmanager.GenerateECDSAKey(rand.Reader)
|
|
assert.NoError(t, err)
|
|
noGunCert2, err := cryptoservice.GenerateCertificate(
|
|
privKey, "nonexistent", startTime, startTime.AddDate(10, 0, 0))
|
|
assert.NoError(t, err)
|
|
err = certStore.AddCert(noGunCert2)
|
|
assert.NoError(t, err)
|
|
|
|
certs = assertNumCerts(t, tempDir, 2)
|
|
|
|
// passing an empty gun to overwrite previously stored gun
|
|
_, err = runCommand(t, tempDir, "cert", "remove", certID, "-y", "-g", "")
|
|
assert.NoError(t, err)
|
|
|
|
// Since another cert with the same GUN exists, we didn't remove everything
|
|
assertNumCerts(t, tempDir, 1)
|
|
}
|
|
|
|
// Tests default root key generation
|
|
func TestDefaultRootKeyGeneration(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
// -- tests --
|
|
|
|
// starts out with no keys
|
|
assertNumKeys(t, tempDir, 0, 0, true)
|
|
|
|
// generate root key with no algorithm produces a single ECDSA root key and no other keys
|
|
_, err := runCommand(t, tempDir, "key", "generate")
|
|
assert.NoError(t, err)
|
|
assertNumKeys(t, tempDir, 1, 0, true)
|
|
}
|
|
|
|
// Tests the interaction with the verbose and log-level flags
|
|
func TestLogLevelFlags(t *testing.T) {
|
|
// Test default to fatal
|
|
n := notaryCommander{}
|
|
n.setVerbosityLevel()
|
|
assert.Equal(t, "fatal", logrus.GetLevel().String())
|
|
|
|
// Test that verbose (-v) sets to error
|
|
n.verbose = true
|
|
n.setVerbosityLevel()
|
|
assert.Equal(t, "error", logrus.GetLevel().String())
|
|
|
|
// Test that debug (-D) sets to debug
|
|
n.debug = true
|
|
n.setVerbosityLevel()
|
|
assert.Equal(t, "debug", logrus.GetLevel().String())
|
|
|
|
// Test that unsetting verboseError still uses verboseDebug
|
|
n.verbose = false
|
|
n.setVerbosityLevel()
|
|
assert.Equal(t, "debug", logrus.GetLevel().String())
|
|
}
|
|
|
|
func TestClientKeyPassphraseChange(t *testing.T) {
|
|
// -- setup --
|
|
setUp(t)
|
|
|
|
tempDir := tempDirWithConfig(t, "{}")
|
|
defer os.RemoveAll(tempDir)
|
|
|
|
server := setupServer()
|
|
defer server.Close()
|
|
|
|
target := "sdgkadga"
|
|
tempFile, err := ioutil.TempFile("/tmp", "targetfile")
|
|
assert.NoError(t, err)
|
|
tempFile.Close()
|
|
defer os.Remove(tempFile.Name())
|
|
|
|
// -- tests --
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun1")
|
|
assert.NoError(t, err)
|
|
|
|
// we should have three keys stored locally in total: root, targets, snapshot
|
|
rootIDs, signingIDs := assertNumKeys(t, tempDir, 1, 2, true)
|
|
for _, keyID := range signingIDs {
|
|
// try changing the private key passphrase
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "key", "passwd", keyID)
|
|
assert.NoError(t, err)
|
|
|
|
// assert that the signing keys (number and IDs) didn't change
|
|
_, signingIDs = assertNumKeys(t, tempDir, 1, 2, true)
|
|
assert.Contains(t, signingIDs, keyID)
|
|
|
|
// make sure we can still publish with this signing key
|
|
assertSuccessfullyPublish(t, tempDir, server.URL, "gun1", target, tempFile.Name())
|
|
}
|
|
|
|
// only one rootID, try changing the private key passphrase
|
|
rootID := rootIDs[0]
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "key", "passwd", rootID)
|
|
assert.NoError(t, err)
|
|
|
|
// make sure we can init a new repo with this key
|
|
_, err = runCommand(t, tempDir, "-s", server.URL, "init", "gun2")
|
|
assert.NoError(t, err)
|
|
|
|
// assert that the root key ID didn't change
|
|
rootIDs, _ = assertNumKeys(t, tempDir, 1, 4, true)
|
|
assert.Equal(t, rootID, rootIDs[0])
|
|
}
|
|
|
|
func tempDirWithConfig(t *testing.T, config string) string {
|
|
tempDir, err := ioutil.TempDir("", "repo")
|
|
assert.NoError(t, err)
|
|
err = ioutil.WriteFile(filepath.Join(tempDir, "config.json"), []byte(config), 0644)
|
|
assert.NoError(t, err)
|
|
return tempDir
|
|
}
|
|
|
|
func TestMain(m *testing.M) {
|
|
if testing.Short() {
|
|
// skip
|
|
os.Exit(0)
|
|
}
|
|
os.Exit(m.Run())
|
|
}
|