diff --git a/models/error.go b/models/error.go index 251c90a68d3b2..85d250c84a72a 100644 --- a/models/error.go +++ b/models/error.go @@ -213,8 +213,9 @@ func (err ErrKeyNotExist) Error() string { // ErrKeyAlreadyExist represents a "KeyAlreadyExist" kind of error. type ErrKeyAlreadyExist struct { - OwnerID int64 - Content string + OwnerID int64 + Fingerprint string + Content string } // IsErrKeyAlreadyExist checks if an error is a ErrKeyAlreadyExist. @@ -224,7 +225,8 @@ func IsErrKeyAlreadyExist(err error) bool { } func (err ErrKeyAlreadyExist) Error() string { - return fmt.Sprintf("public key already exists [owner_id: %d, content: %s]", err.OwnerID, err.Content) + return fmt.Sprintf("public key already exists [owner_id: %d, finter_print: %s, content: %s]", + err.OwnerID, err.Fingerprint, err.Content) } // ErrKeyNameAlreadyUsed represents a "KeyNameAlreadyUsed" kind of error. diff --git a/models/ssh_key.go b/models/ssh_key.go index 9ca45fa6e8670..e82fd3aad3e63 100644 --- a/models/ssh_key.go +++ b/models/ssh_key.go @@ -354,41 +354,50 @@ func appendAuthorizedKeysToFile(keys ...*PublicKey) error { return nil } -// checkKeyContent only checks if key content has been used as public key, +// checkKeyFingerprint only checks if key fingerprint has been used as public key, // it is OK to use same key as deploy key for multiple repositories/users. -func checkKeyContent(content string) error { - has, err := x.Get(&PublicKey{ - Content: content, - Type: KeyTypeUser, +func checkKeyFingerprint(e Engine, fingerprint string) error { + has, err := e.Get(&PublicKey{ + Fingerprint: fingerprint, + Type: KeyTypeUser, }) if err != nil { return err } else if has { - return ErrKeyAlreadyExist{0, content} + return ErrKeyAlreadyExist{0, fingerprint, ""} } return nil } -func addKey(e Engine, key *PublicKey) (err error) { +func calcFingerprint(publicKeyContent string) (string, error) { // Calculate fingerprint. tmpPath := strings.Replace(path.Join(os.TempDir(), fmt.Sprintf("%d", time.Now().Nanosecond()), "id_rsa.pub"), "\\", "/", -1) dir := path.Dir(tmpPath) if err := os.MkdirAll(dir, os.ModePerm); err != nil { - return fmt.Errorf("Failed to create dir %s: %v", dir, err) + return "", fmt.Errorf("Failed to create dir %s: %v", dir, err) } - if err = ioutil.WriteFile(tmpPath, []byte(key.Content), 0644); err != nil { - return err + if err := ioutil.WriteFile(tmpPath, []byte(publicKeyContent), 0644); err != nil { + return "", err } stdout, stderr, err := process.GetManager().Exec("AddPublicKey", "ssh-keygen", "-lf", tmpPath) if err != nil { - return fmt.Errorf("'ssh-keygen -lf %s' failed with error '%s': %s", tmpPath, err, stderr) + return "", fmt.Errorf("'ssh-keygen -lf %s' failed with error '%s': %s", tmpPath, err, stderr) } else if len(stdout) < 2 { - return errors.New("not enough output for calculating fingerprint: " + stdout) + return "", errors.New("not enough output for calculating fingerprint: " + stdout) + } + return strings.Split(stdout, " ")[1], nil +} + +func addKey(e Engine, key *PublicKey) (err error) { + if len(key.Fingerprint) <= 0 { + key.Fingerprint, err = calcFingerprint(key.Content) + if err != nil { + return err + } } - key.Fingerprint = strings.Split(stdout, " ")[1] // Save SSH key. if _, err = e.Insert(key); err != nil { @@ -405,7 +414,13 @@ func addKey(e Engine, key *PublicKey) (err error) { // AddPublicKey adds new public key to database and authorized_keys file. func AddPublicKey(ownerID int64, name, content string) (*PublicKey, error) { log.Trace(content) - if err := checkKeyContent(content); err != nil { + + fingerprint, err := calcFingerprint(content) + if err != nil { + return nil, err + } + + if err := checkKeyFingerprint(x, fingerprint); err != nil { return nil, err } @@ -426,11 +441,12 @@ func AddPublicKey(ownerID int64, name, content string) (*PublicKey, error) { } key := &PublicKey{ - OwnerID: ownerID, - Name: name, - Content: content, - Mode: AccessModeWrite, - Type: KeyTypeUser, + OwnerID: ownerID, + Name: name, + Fingerprint: fingerprint, + Content: content, + Mode: AccessModeWrite, + Type: KeyTypeUser, } if err = addKey(sess, key); err != nil { return nil, fmt.Errorf("addKey: %v", err) @@ -665,14 +681,15 @@ func HasDeployKey(keyID, repoID int64) bool { // AddDeployKey add new deploy key to database and authorized_keys file. func AddDeployKey(repoID int64, name, content string) (*DeployKey, error) { - if err := checkKeyContent(content); err != nil { + fingerprint, err := calcFingerprint(content) + if err != nil { return nil, err } pkey := &PublicKey{ - Content: content, - Mode: AccessModeRead, - Type: KeyTypeDeploy, + Fingerprint: fingerprint, + Mode: AccessModeRead, + Type: KeyTypeDeploy, } has, err := x.Get(pkey) if err != nil { @@ -687,6 +704,8 @@ func AddDeployKey(repoID int64, name, content string) (*DeployKey, error) { // First time use this deploy key. if !has { + pkey.Content = content + pkey.Name = name if err = addKey(sess, pkey); err != nil { return nil, fmt.Errorf("addKey: %v", err) }