feat(sftp-server): public key login (#7668)
This commit is contained in:
@ -12,7 +12,7 @@ var db *gorm.DB
|
||||
|
||||
func Init(d *gorm.DB) {
|
||||
db = d
|
||||
err := AutoMigrate(new(model.Storage), new(model.User), new(model.Meta), new(model.SettingItem), new(model.SearchNode), new(model.TaskItem))
|
||||
err := AutoMigrate(new(model.Storage), new(model.User), new(model.Meta), new(model.SettingItem), new(model.SearchNode), new(model.TaskItem), new(model.SSHPublicKey))
|
||||
if err != nil {
|
||||
log.Fatalf("failed migrate database: %s", err.Error())
|
||||
}
|
||||
|
57
internal/db/sshkey.go
Normal file
57
internal/db/sshkey.go
Normal file
@ -0,0 +1,57 @@
|
||||
package db
|
||||
|
||||
import (
|
||||
"github.com/alist-org/alist/v3/internal/model"
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
func GetSSHPublicKeyByUserId(userId uint, pageIndex, pageSize int) (keys []model.SSHPublicKey, count int64, err error) {
|
||||
keyDB := db.Model(&model.SSHPublicKey{})
|
||||
query := model.SSHPublicKey{UserId: userId}
|
||||
if err := keyDB.Where(query).Count(&count).Error; err != nil {
|
||||
return nil, 0, errors.Wrapf(err, "failed get user's keys count")
|
||||
}
|
||||
if err := keyDB.Where(query).Order(columnName("id")).Offset((pageIndex - 1) * pageSize).Limit(pageSize).Find(&keys).Error; err != nil {
|
||||
return nil, 0, errors.Wrapf(err, "failed get find user's keys")
|
||||
}
|
||||
return keys, count, nil
|
||||
}
|
||||
|
||||
func GetSSHPublicKeyById(id uint) (*model.SSHPublicKey, error) {
|
||||
var k model.SSHPublicKey
|
||||
if err := db.First(&k, id).Error; err != nil {
|
||||
return nil, errors.Wrapf(err, "failed get old key")
|
||||
}
|
||||
return &k, nil
|
||||
}
|
||||
|
||||
func GetSSHPublicKeyByUserTitle(userId uint, title string) (*model.SSHPublicKey, error) {
|
||||
key := model.SSHPublicKey{UserId: userId, Title: title}
|
||||
if err := db.Where(key).First(&key).Error; err != nil {
|
||||
return nil, errors.Wrapf(err, "failed find key with title of user")
|
||||
}
|
||||
return &key, nil
|
||||
}
|
||||
|
||||
func CreateSSHPublicKey(k *model.SSHPublicKey) error {
|
||||
return errors.WithStack(db.Create(k).Error)
|
||||
}
|
||||
|
||||
func UpdateSSHPublicKey(k *model.SSHPublicKey) error {
|
||||
return errors.WithStack(db.Save(k).Error)
|
||||
}
|
||||
|
||||
func GetSSHPublicKeys(pageIndex, pageSize int) (keys []model.SSHPublicKey, count int64, err error) {
|
||||
keyDB := db.Model(&model.SSHPublicKey{})
|
||||
if err := keyDB.Count(&count).Error; err != nil {
|
||||
return nil, 0, errors.Wrapf(err, "failed get keys count")
|
||||
}
|
||||
if err := keyDB.Order(columnName("id")).Offset((pageIndex - 1) * pageSize).Limit(pageSize).Find(&keys).Error; err != nil {
|
||||
return nil, 0, errors.Wrapf(err, "failed get find keys")
|
||||
}
|
||||
return keys, count, nil
|
||||
}
|
||||
|
||||
func DeleteSSHPublicKeyById(id uint) error {
|
||||
return errors.WithStack(db.Delete(&model.SSHPublicKey{}, id).Error)
|
||||
}
|
28
internal/model/sshkey.go
Normal file
28
internal/model/sshkey.go
Normal file
@ -0,0 +1,28 @@
|
||||
package model
|
||||
|
||||
import (
|
||||
"golang.org/x/crypto/ssh"
|
||||
"time"
|
||||
)
|
||||
|
||||
type SSHPublicKey struct {
|
||||
ID uint `json:"id" gorm:"primaryKey"`
|
||||
UserId uint `json:"-"`
|
||||
Title string `json:"title"`
|
||||
Fingerprint string `json:"fingerprint"`
|
||||
KeyStr string `gorm:"type:text" json:"-"`
|
||||
AddedTime time.Time `json:"added_time"`
|
||||
LastUsedTime time.Time `json:"last_used_time"`
|
||||
}
|
||||
|
||||
func (k *SSHPublicKey) GetKey() (ssh.PublicKey, error) {
|
||||
pubKey, _, _, _, err := ssh.ParseAuthorizedKey([]byte(k.KeyStr))
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return pubKey, nil
|
||||
}
|
||||
|
||||
func (k *SSHPublicKey) UpdateLastUsedTime() {
|
||||
k.LastUsedTime = time.Now()
|
||||
}
|
48
internal/op/sshkey.go
Normal file
48
internal/op/sshkey.go
Normal file
@ -0,0 +1,48 @@
|
||||
package op
|
||||
|
||||
import (
|
||||
"github.com/alist-org/alist/v3/internal/db"
|
||||
"github.com/alist-org/alist/v3/internal/model"
|
||||
"github.com/pkg/errors"
|
||||
"golang.org/x/crypto/ssh"
|
||||
"time"
|
||||
)
|
||||
|
||||
func CreateSSHPublicKey(k *model.SSHPublicKey) (error, bool) {
|
||||
_, err := db.GetSSHPublicKeyByUserTitle(k.UserId, k.Title)
|
||||
if err == nil {
|
||||
return errors.New("key with the same title already exists"), true
|
||||
}
|
||||
pubKey, _, _, _, err := ssh.ParseAuthorizedKey([]byte(k.KeyStr))
|
||||
if err != nil {
|
||||
return err, false
|
||||
}
|
||||
k.KeyStr = string(pubKey.Marshal())
|
||||
k.Fingerprint = ssh.FingerprintSHA256(pubKey)
|
||||
k.AddedTime = time.Now()
|
||||
k.LastUsedTime = k.AddedTime
|
||||
return db.CreateSSHPublicKey(k), true
|
||||
}
|
||||
|
||||
func GetSSHPublicKeyByUserId(userId uint, pageIndex, pageSize int) (keys []model.SSHPublicKey, count int64, err error) {
|
||||
return db.GetSSHPublicKeyByUserId(userId, pageIndex, pageSize)
|
||||
}
|
||||
|
||||
func GetSSHPublicKeyByIdAndUserId(id uint, userId uint) (*model.SSHPublicKey, error) {
|
||||
key, err := db.GetSSHPublicKeyById(id)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if key.UserId != userId {
|
||||
return nil, errors.Wrapf(err, "failed get old key")
|
||||
}
|
||||
return key, nil
|
||||
}
|
||||
|
||||
func UpdateSSHPublicKey(k *model.SSHPublicKey) error {
|
||||
return db.UpdateSSHPublicKey(k)
|
||||
}
|
||||
|
||||
func DeleteSSHPublicKeyById(keyId uint) error {
|
||||
return db.DeleteSSHPublicKeyById(keyId)
|
||||
}
|
Reference in New Issue
Block a user