|
@@ -11,6 +11,7 @@ import (
|
|
"io"
|
|
"io"
|
|
"os"
|
|
"os"
|
|
"os/exec"
|
|
"os/exec"
|
|
|
|
+ "path"
|
|
"path/filepath"
|
|
"path/filepath"
|
|
"strings"
|
|
"strings"
|
|
"sync"
|
|
"sync"
|
|
@@ -57,29 +58,48 @@ func init() {
|
|
}
|
|
}
|
|
|
|
|
|
type PublicKey struct {
|
|
type PublicKey struct {
|
|
- Id int64
|
|
|
|
- OwnerId int64 `xorm:"index"`
|
|
|
|
- Name string `xorm:"unique not null"`
|
|
|
|
- Content string `xorm:"text not null"`
|
|
|
|
- Created time.Time `xorm:"created"`
|
|
|
|
- Updated time.Time `xorm:"updated"`
|
|
|
|
|
|
+ Id int64
|
|
|
|
+ OwnerId int64 `xorm:"index"`
|
|
|
|
+ Name string `xorm:"unique not null"`
|
|
|
|
+ Fingerprint string
|
|
|
|
+ Content string `xorm:"text not null"`
|
|
|
|
+ Created time.Time `xorm:"created"`
|
|
|
|
+ Updated time.Time `xorm:"updated"`
|
|
}
|
|
}
|
|
|
|
|
|
func GenAuthorizedKey(keyId int64, key string) string {
|
|
func GenAuthorizedKey(keyId int64, key string) string {
|
|
return fmt.Sprintf(tmplPublicKey, appPath, keyId, key)
|
|
return fmt.Sprintf(tmplPublicKey, appPath, keyId, key)
|
|
}
|
|
}
|
|
|
|
|
|
-func AddPublicKey(key *PublicKey) error {
|
|
|
|
- _, err := orm.Insert(key)
|
|
|
|
|
|
+func AddPublicKey(key *PublicKey) (err error) {
|
|
|
|
+ // Calculate fingerprint.
|
|
|
|
+ tmpPath := filepath.Join(os.TempDir(), fmt.Sprintf("%d", time.Now().Nanosecond()),
|
|
|
|
+ "id_rsa.pub")
|
|
|
|
+ os.MkdirAll(path.Dir(tmpPath), os.ModePerm)
|
|
|
|
+ f, err := os.Create(tmpPath)
|
|
if err != nil {
|
|
if err != nil {
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ if _, err = f.WriteString(key.Content); err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
-
|
|
|
|
- err = SaveAuthorizedKeyFile(key)
|
|
|
|
|
|
+ f.Close()
|
|
|
|
+ stdout, _, err := com.ExecCmd("ssh-keygen", "-l", "-f", tmpPath)
|
|
if err != nil {
|
|
if err != nil {
|
|
- _, err2 := orm.Delete(key)
|
|
|
|
- if err2 != nil {
|
|
|
|
- // TODO: log the error
|
|
|
|
|
|
+ return err
|
|
|
|
+ } else if len(stdout) < 2 {
|
|
|
|
+ return errors.New("Not enough output for calculating fingerprint")
|
|
|
|
+ }
|
|
|
|
+ key.Fingerprint = strings.Split(stdout, " ")[1]
|
|
|
|
+
|
|
|
|
+ // Save SSH key.
|
|
|
|
+ if _, err = orm.Insert(key); err != nil {
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if err = SaveAuthorizedKeyFile(key); err != nil {
|
|
|
|
+ if _, err2 := orm.Delete(key); err2 != nil {
|
|
|
|
+ return err2
|
|
}
|
|
}
|
|
return err
|
|
return err
|
|
}
|
|
}
|