2017-04-17 18:08:24 -04:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/docker/notary"
|
|
|
|
"github.com/docker/notary/client/changelist"
|
|
|
|
store "github.com/docker/notary/storage"
|
|
|
|
"github.com/docker/notary/tuf/data"
|
|
|
|
"github.com/docker/notary/tuf/utils"
|
2017-09-11 17:07:00 -04:00
|
|
|
"github.com/sirupsen/logrus"
|
2017-04-17 18:08:24 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// AddDelegation creates changelist entries to add provided delegation public keys and paths.
|
|
|
|
// This method composes AddDelegationRoleAndKeys and AddDelegationPaths (each creates one changelist if called).
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) AddDelegation(name data.RoleName, delegationKeys []data.PublicKey, paths []string) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
if len(delegationKeys) > 0 {
|
|
|
|
err := r.AddDelegationRoleAndKeys(name, delegationKeys)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(paths) > 0 {
|
|
|
|
err := r.AddDelegationPaths(name, paths)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddDelegationRoleAndKeys creates a changelist entry to add provided delegation public keys.
|
|
|
|
// This method is the simplest way to create a new delegation, because the delegation must have at least
|
|
|
|
// one key upon creation to be valid since we will reject the changelist while validating the threshold.
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) AddDelegationRoleAndKeys(name data.RoleName, delegationKeys []data.PublicKey) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
|
|
|
|
if !data.IsDelegation(name) {
|
|
|
|
return data.ErrInvalidRole{Role: name, Reason: "invalid delegation role name"}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf(`Adding delegation "%s" with threshold %d, and %d keys\n`,
|
|
|
|
name, notary.MinThreshold, len(delegationKeys))
|
|
|
|
|
|
|
|
// Defaulting to threshold of 1, since we don't allow for larger thresholds at the moment.
|
|
|
|
tdJSON, err := json.Marshal(&changelist.TUFDelegation{
|
|
|
|
NewThreshold: notary.MinThreshold,
|
|
|
|
AddKeys: data.KeyList(delegationKeys),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
template := newCreateDelegationChange(name, tdJSON)
|
2017-08-24 18:40:24 -04:00
|
|
|
return addChange(r.changelist, template, name)
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddDelegationPaths creates a changelist entry to add provided paths to an existing delegation.
|
|
|
|
// This method cannot create a new delegation itself because the role must meet the key threshold upon creation.
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) AddDelegationPaths(name data.RoleName, paths []string) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
|
|
|
|
if !data.IsDelegation(name) {
|
|
|
|
return data.ErrInvalidRole{Role: name, Reason: "invalid delegation role name"}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf(`Adding %s paths to delegation %s\n`, paths, name)
|
|
|
|
|
|
|
|
tdJSON, err := json.Marshal(&changelist.TUFDelegation{
|
|
|
|
AddPaths: paths,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
template := newCreateDelegationChange(name, tdJSON)
|
2017-08-24 18:40:24 -04:00
|
|
|
return addChange(r.changelist, template, name)
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveDelegationKeysAndPaths creates changelist entries to remove provided delegation key IDs and paths.
|
|
|
|
// This method composes RemoveDelegationPaths and RemoveDelegationKeys (each creates one changelist if called).
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) RemoveDelegationKeysAndPaths(name data.RoleName, keyIDs, paths []string) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
if len(paths) > 0 {
|
|
|
|
err := r.RemoveDelegationPaths(name, paths)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(keyIDs) > 0 {
|
|
|
|
err := r.RemoveDelegationKeys(name, keyIDs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveDelegationRole creates a changelist to remove all paths and keys from a role, and delete the role in its entirety.
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) RemoveDelegationRole(name data.RoleName) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
|
|
|
|
if !data.IsDelegation(name) {
|
|
|
|
return data.ErrInvalidRole{Role: name, Reason: "invalid delegation role name"}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf(`Removing delegation "%s"\n`, name)
|
|
|
|
|
|
|
|
template := newDeleteDelegationChange(name, nil)
|
2017-08-24 18:40:24 -04:00
|
|
|
return addChange(r.changelist, template, name)
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveDelegationPaths creates a changelist entry to remove provided paths from an existing delegation.
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) RemoveDelegationPaths(name data.RoleName, paths []string) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
|
|
|
|
if !data.IsDelegation(name) {
|
|
|
|
return data.ErrInvalidRole{Role: name, Reason: "invalid delegation role name"}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf(`Removing %s paths from delegation "%s"\n`, paths, name)
|
|
|
|
|
|
|
|
tdJSON, err := json.Marshal(&changelist.TUFDelegation{
|
|
|
|
RemovePaths: paths,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
template := newUpdateDelegationChange(name, tdJSON)
|
2017-08-24 18:40:24 -04:00
|
|
|
return addChange(r.changelist, template, name)
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveDelegationKeys creates a changelist entry to remove provided keys from an existing delegation.
|
|
|
|
// When this changelist is applied, if the specified keys are the only keys left in the role,
|
|
|
|
// the role itself will be deleted in its entirety.
|
|
|
|
// It can also delete a key from all delegations under a parent using a name
|
|
|
|
// with a wildcard at the end.
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) RemoveDelegationKeys(name data.RoleName, keyIDs []string) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
|
|
|
|
if !data.IsDelegation(name) && !data.IsWildDelegation(name) {
|
|
|
|
return data.ErrInvalidRole{Role: name, Reason: "invalid delegation role name"}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf(`Removing %s keys from delegation "%s"\n`, keyIDs, name)
|
|
|
|
|
|
|
|
tdJSON, err := json.Marshal(&changelist.TUFDelegation{
|
|
|
|
RemoveKeys: keyIDs,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
template := newUpdateDelegationChange(name, tdJSON)
|
2017-08-24 18:40:24 -04:00
|
|
|
return addChange(r.changelist, template, name)
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// ClearDelegationPaths creates a changelist entry to remove all paths from an existing delegation.
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) ClearDelegationPaths(name data.RoleName) error {
|
2017-04-17 18:08:24 -04:00
|
|
|
|
|
|
|
if !data.IsDelegation(name) {
|
|
|
|
return data.ErrInvalidRole{Role: name, Reason: "invalid delegation role name"}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf(`Removing all paths from delegation "%s"\n`, name)
|
|
|
|
|
|
|
|
tdJSON, err := json.Marshal(&changelist.TUFDelegation{
|
|
|
|
ClearAllPaths: true,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
template := newUpdateDelegationChange(name, tdJSON)
|
2017-08-24 18:40:24 -04:00
|
|
|
return addChange(r.changelist, template, name)
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
2017-08-24 18:40:24 -04:00
|
|
|
func newUpdateDelegationChange(name data.RoleName, content []byte) *changelist.TUFChange {
|
2017-04-17 18:08:24 -04:00
|
|
|
return changelist.NewTUFChange(
|
|
|
|
changelist.ActionUpdate,
|
|
|
|
name,
|
|
|
|
changelist.TypeTargetsDelegation,
|
|
|
|
"", // no path for delegations
|
|
|
|
content,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2017-08-24 18:40:24 -04:00
|
|
|
func newCreateDelegationChange(name data.RoleName, content []byte) *changelist.TUFChange {
|
2017-04-17 18:08:24 -04:00
|
|
|
return changelist.NewTUFChange(
|
|
|
|
changelist.ActionCreate,
|
|
|
|
name,
|
|
|
|
changelist.TypeTargetsDelegation,
|
|
|
|
"", // no path for delegations
|
|
|
|
content,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2017-08-24 18:40:24 -04:00
|
|
|
func newDeleteDelegationChange(name data.RoleName, content []byte) *changelist.TUFChange {
|
2017-04-17 18:08:24 -04:00
|
|
|
return changelist.NewTUFChange(
|
|
|
|
changelist.ActionDelete,
|
|
|
|
name,
|
|
|
|
changelist.TypeTargetsDelegation,
|
|
|
|
"", // no path for delegations
|
|
|
|
content,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDelegationRoles returns the keys and roles of the repository's delegations
|
|
|
|
// Also converts key IDs to canonical key IDs to keep consistent with signing prompts
|
2017-09-11 17:07:00 -04:00
|
|
|
func (r *repository) GetDelegationRoles() ([]data.Role, error) {
|
2017-04-17 18:08:24 -04:00
|
|
|
// Update state of the repo to latest
|
|
|
|
if err := r.Update(false); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// All top level delegations (ex: targets/level1) are stored exclusively in targets.json
|
|
|
|
_, ok := r.tufRepo.Targets[data.CanonicalTargetsRole]
|
|
|
|
if !ok {
|
2017-08-24 18:40:24 -04:00
|
|
|
return nil, store.ErrMetaNotFound{Resource: data.CanonicalTargetsRole.String()}
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// make a copy for traversing nested delegations
|
|
|
|
allDelegations := []data.Role{}
|
|
|
|
|
|
|
|
// Define a visitor function to populate the delegations list and translate their key IDs to canonical IDs
|
|
|
|
delegationCanonicalListVisitor := func(tgt *data.SignedTargets, validRole data.DelegationRole) interface{} {
|
|
|
|
// For the return list, update with a copy that includes canonicalKeyIDs
|
|
|
|
// These aren't validated by the validRole
|
|
|
|
canonicalDelegations, err := translateDelegationsToCanonicalIDs(tgt.Signed.Delegations)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
allDelegations = append(allDelegations, canonicalDelegations...)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
err := r.tufRepo.WalkTargets("", "", delegationCanonicalListVisitor)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return allDelegations, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func translateDelegationsToCanonicalIDs(delegationInfo data.Delegations) ([]data.Role, error) {
|
|
|
|
canonicalDelegations := make([]data.Role, len(delegationInfo.Roles))
|
|
|
|
// Do a copy by value to ensure local delegation metadata is untouched
|
|
|
|
for idx, origRole := range delegationInfo.Roles {
|
|
|
|
canonicalDelegations[idx] = *origRole
|
|
|
|
}
|
|
|
|
delegationKeys := delegationInfo.Keys
|
|
|
|
for i, delegation := range canonicalDelegations {
|
|
|
|
canonicalKeyIDs := []string{}
|
|
|
|
for _, keyID := range delegation.KeyIDs {
|
|
|
|
pubKey, ok := delegationKeys[keyID]
|
|
|
|
if !ok {
|
|
|
|
return []data.Role{}, fmt.Errorf("Could not translate canonical key IDs for %s", delegation.Name)
|
|
|
|
}
|
|
|
|
canonicalKeyID, err := utils.CanonicalKeyID(pubKey)
|
|
|
|
if err != nil {
|
|
|
|
return []data.Role{}, fmt.Errorf("Could not translate canonical key IDs for %s: %v", delegation.Name, err)
|
|
|
|
}
|
|
|
|
canonicalKeyIDs = append(canonicalKeyIDs, canonicalKeyID)
|
|
|
|
}
|
|
|
|
canonicalDelegations[i].KeyIDs = canonicalKeyIDs
|
|
|
|
}
|
|
|
|
return canonicalDelegations, nil
|
|
|
|
}
|