Files

415 lines
12 KiB
Go
Raw Permalink Normal View History

2014-03-16 05:24:13 -04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
// Copyright 2019 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
2014-03-16 05:24:13 -04:00
2021-12-10 16:14:24 +08:00
package asymkey
2014-02-17 23:57:23 +08:00
import (
2021-12-10 16:14:24 +08:00
"context"
2014-02-17 23:57:23 +08:00
"fmt"
2014-03-16 05:24:13 -04:00
"strings"
2014-02-17 23:57:23 +08:00
"time"
2014-03-02 15:25:09 -05:00
2022-01-02 21:12:35 +08:00
"code.gitea.io/gitea/models/auth"
"code.gitea.io/gitea/models/db"
2021-11-28 19:58:28 +08:00
"code.gitea.io/gitea/models/perm"
user_model "code.gitea.io/gitea/models/user"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/timeutil"
"code.gitea.io/gitea/modules/util"
2014-02-17 23:57:23 +08:00
"golang.org/x/crypto/ssh"
2021-07-24 11:16:34 +01:00
"xorm.io/builder"
)
2016-11-26 01:36:03 +01:00
// KeyType specifies the key type
2015-08-06 22:48:11 +08:00
type KeyType int
const (
2016-11-26 01:36:03 +01:00
// KeyTypeUser specifies the user key
2016-11-07 17:53:22 +01:00
KeyTypeUser = iota + 1
2016-11-26 01:36:03 +01:00
// KeyTypeDeploy specifies the deploy key
2016-11-07 17:53:22 +01:00
KeyTypeDeploy
2020-10-11 02:38:09 +02:00
// KeyTypePrincipal specifies the authorized principal key
KeyTypePrincipal
2015-08-06 22:48:11 +08:00
)
2016-07-26 10:47:25 +08:00
// PublicKey represents a user or deploy SSH public key.
2014-02-17 23:57:23 +08:00
type PublicKey struct {
2021-11-28 19:58:28 +08:00
ID int64 `xorm:"pk autoincr"`
OwnerID int64 `xorm:"INDEX NOT NULL"`
Name string `xorm:"NOT NULL"`
Fingerprint string `xorm:"INDEX NOT NULL"`
Content string `xorm:"MEDIUMTEXT NOT NULL"`
2021-11-28 19:58:28 +08:00
Mode perm.AccessMode `xorm:"NOT NULL DEFAULT 2"`
Type KeyType `xorm:"NOT NULL DEFAULT 1"`
LoginSourceID int64 `xorm:"NOT NULL DEFAULT 0"`
CreatedUnix timeutil.TimeStamp `xorm:"created"`
UpdatedUnix timeutil.TimeStamp `xorm:"updated"`
HasRecentActivity bool `xorm:"-"`
HasUsed bool `xorm:"-"`
2021-12-19 06:37:18 +01:00
Verified bool `xorm:"NOT NULL DEFAULT false"`
2015-08-06 22:48:11 +08:00
}
func init() {
db.RegisterModel(new(PublicKey))
}
// AfterLoad is invoked from XORM after setting the values of all fields of this object.
func (key *PublicKey) AfterLoad() {
key.HasUsed = key.UpdatedUnix > key.CreatedUnix
key.HasRecentActivity = key.UpdatedUnix.AddDuration(7*24*time.Hour) > timeutil.TimeStampNow()
2014-02-17 23:57:23 +08:00
}
2016-07-26 10:47:25 +08:00
// OmitEmail returns content of public key without email address.
2016-11-26 01:36:03 +01:00
func (key *PublicKey) OmitEmail() string {
fields := strings.Split(key.Content, " ") // format: ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQC... comment
if len(fields) < 2 {
setting.PanicInDevOrTesting("invalid public key %d content: %s", key.ID, key.Content)
return "" // not a valid public key, it shouldn't really happen, the value is managed internally
}
return strings.Join(fields[:2], " ")
2014-11-23 02:33:47 -05:00
}
func addKey(ctx context.Context, key *PublicKey) (err error) {
2018-10-20 23:25:14 +02:00
if len(key.Fingerprint) == 0 {
2022-06-05 03:18:50 +08:00
key.Fingerprint, err = CalcFingerprint(key.Content)
if err != nil {
return err
}
2014-03-16 06:16:03 -04:00
}
// Save SSH key.
if err = db.Insert(ctx, key); err != nil {
2014-03-16 06:16:03 -04:00
return err
2015-08-06 22:48:11 +08:00
}
2016-07-26 10:47:25 +08:00
return appendAuthorizedKeysToFile(key)
2015-08-06 22:48:11 +08:00
}
// AddPublicKey adds new public key to database and authorized_keys file.
func AddPublicKey(ctx context.Context, ownerID int64, name, content string, authSourceID int64, verified bool) (*PublicKey, error) {
log.Trace(content)
2022-06-05 03:18:50 +08:00
fingerprint, err := CalcFingerprint(content)
if err != nil {
return nil, err
}
return db.WithTx2(ctx, func(ctx context.Context) (*PublicKey, error) {
if err := checkKeyFingerprint(ctx, fingerprint); err != nil {
return nil, err
}
2014-02-17 23:57:23 +08:00
// Key name of same user cannot be duplicated.
has, err := db.GetEngine(ctx).
Where("owner_id = ? AND name = ?", ownerID, name).
Get(new(PublicKey))
if err != nil {
return nil, err
} else if has {
return nil, ErrKeyNameAlreadyUsed{ownerID, name}
}
2015-08-06 22:48:11 +08:00
key := &PublicKey{
OwnerID: ownerID,
Name: name,
Fingerprint: fingerprint,
Content: content,
Mode: perm.AccessModeWrite,
Type: KeyTypeUser,
LoginSourceID: authSourceID,
Verified: verified,
}
if err = addKey(ctx, key); err != nil {
return nil, fmt.Errorf("addKey: %w", err)
}
2015-08-06 22:48:11 +08:00
return key, nil
})
2014-02-17 23:57:23 +08:00
}
2015-08-06 22:48:11 +08:00
// GetPublicKeyByID returns public key by given ID.
func GetPublicKeyByID(ctx context.Context, keyID int64) (*PublicKey, error) {
2014-08-09 15:40:10 -07:00
key := new(PublicKey)
has, err := db.GetEngine(ctx).
2020-03-22 23:12:55 +08:00
ID(keyID).
2016-11-10 16:16:32 +01:00
Get(key)
2014-08-09 15:40:10 -07:00
if err != nil {
return nil, err
} else if !has {
2015-08-06 22:48:11 +08:00
return nil, ErrKeyNotExist{keyID}
2014-08-09 15:40:10 -07:00
}
return key, nil
}
// SearchPublicKeyByContent searches content as prefix (leak e-mail part)
// and returns public key found.
func SearchPublicKeyByContent(ctx context.Context, content string) (*PublicKey, error) {
key := new(PublicKey)
has, err := db.GetEngine(ctx).
2016-11-10 16:16:32 +01:00
Where("content like ?", content+"%").
Get(key)
if err != nil {
return nil, err
} else if !has {
return nil, ErrKeyNotExist{}
}
return key, nil
}
// SearchPublicKeyByContentExact searches content
// and returns public key found.
func SearchPublicKeyByContentExact(ctx context.Context, content string) (*PublicKey, error) {
2020-10-11 02:38:09 +02:00
key := new(PublicKey)
has, err := db.GetEngine(ctx).
2020-10-11 02:38:09 +02:00
Where("content = ?", content).
Get(key)
if err != nil {
return nil, err
} else if !has {
return nil, ErrKeyNotExist{}
}
return key, nil
}
type FindPublicKeyOptions struct {
db.ListOptions
OwnerID int64
Fingerprint string
KeyTypes []KeyType
NotKeytype KeyType
LoginSourceID int64
}
func (opts FindPublicKeyOptions) ToConds() builder.Cond {
2018-11-01 03:40:49 +00:00
cond := builder.NewCond()
if opts.OwnerID > 0 {
cond = cond.And(builder.Eq{"owner_id": opts.OwnerID})
2018-11-01 03:40:49 +00:00
}
if opts.Fingerprint != "" {
cond = cond.And(builder.Eq{"fingerprint": opts.Fingerprint})
2018-11-01 03:40:49 +00:00
}
if len(opts.KeyTypes) > 0 {
2024-01-15 10:19:25 +08:00
cond = cond.And(builder.In("`type`", opts.KeyTypes))
2020-01-24 19:00:29 +00:00
}
if opts.NotKeytype > 0 {
2024-01-15 10:19:25 +08:00
cond = cond.And(builder.Neq{"`type`": opts.NotKeytype})
}
if opts.LoginSourceID > 0 {
cond = cond.And(builder.Eq{"login_source_id": opts.LoginSourceID})
}
return cond
}
2017-04-07 17:40:38 -07:00
// UpdatePublicKeyUpdated updates public key use time.
func UpdatePublicKeyUpdated(ctx context.Context, id int64) error {
// Check if key exists before update as affected rows count is unreliable
// and will return 0 affected rows if two updates are made at the same time
if cnt, err := db.GetEngine(ctx).ID(id).Count(&PublicKey{}); err != nil {
return err
} else if cnt != 1 {
return ErrKeyNotExist{id}
}
_, err := db.GetEngine(ctx).ID(id).Cols("updated_unix").Update(&PublicKey{
UpdatedUnix: timeutil.TimeStampNow(),
2017-04-07 17:40:38 -07:00
})
if err != nil {
return err
}
return nil
}
// PublicKeysAreExternallyManaged returns whether the provided KeyID represents an externally managed Key
func PublicKeysAreExternallyManaged(ctx context.Context, keys []*PublicKey) ([]bool, error) {
sourceCache := make(map[int64]*auth.Source, len(keys))
externals := make([]bool, len(keys))
for i, key := range keys {
if key.LoginSourceID == 0 {
externals[i] = false
continue
}
source, ok := sourceCache[key.LoginSourceID]
if !ok {
var err error
source, err = auth.GetSourceByID(ctx, key.LoginSourceID)
if err != nil {
2022-01-02 21:12:35 +08:00
if auth.IsErrSourceNotExist(err) {
externals[i] = false
sourceCache[key.LoginSourceID] = &auth.Source{
ID: key.LoginSourceID,
}
continue
}
return nil, err
}
}
2022-01-02 21:12:35 +08:00
if sshKeyProvider, ok := source.Cfg.(auth.SSHKeyProvider); ok && sshKeyProvider.ProvidesSSHKeys() {
// Disable setting SSH keys for this user
externals[i] = true
}
}
return externals, nil
}
// PublicKeyIsExternallyManaged returns whether the provided KeyID represents an externally managed Key
func PublicKeyIsExternallyManaged(ctx context.Context, id int64) (bool, error) {
key, err := GetPublicKeyByID(ctx, id)
if err != nil {
return false, err
}
if key.LoginSourceID == 0 {
return false, nil
}
source, err := auth.GetSourceByID(ctx, key.LoginSourceID)
if err != nil {
2022-01-02 21:12:35 +08:00
if auth.IsErrSourceNotExist(err) {
return false, nil
}
return false, err
}
2022-01-02 21:12:35 +08:00
if sshKeyProvider, ok := source.Cfg.(auth.SSHKeyProvider); ok && sshKeyProvider.ProvidesSSHKeys() {
// Disable setting SSH keys for this user
return true, nil
}
return false, nil
}
2021-07-24 11:16:34 +01:00
// deleteKeysMarkedForDeletion returns true if ssh keys needs update
func deleteKeysMarkedForDeletion(ctx context.Context, keys []string) (bool, error) {
return db.WithTx2(ctx, func(ctx context.Context) (bool, error) {
// Delete keys marked for deletion
var sshKeysNeedUpdate bool
for _, KeyToDelete := range keys {
key, err := SearchPublicKeyByContent(ctx, KeyToDelete)
if err != nil {
log.Error("SearchPublicKeyByContent: %v", err)
continue
}
if _, err = db.DeleteByID[PublicKey](ctx, key.ID); err != nil {
log.Error("DeleteByID[PublicKey]: %v", err)
continue
}
sshKeysNeedUpdate = true
}
2015-08-06 22:48:11 +08:00
return sshKeysNeedUpdate, nil
})
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
// AddPublicKeysBySource add a users public keys. Returns true if there are changes.
func AddPublicKeysBySource(ctx context.Context, usr *user_model.User, s *auth.Source, sshPublicKeys []string, verified bool) bool {
2021-07-24 11:16:34 +01:00
var sshKeysNeedUpdate bool
for _, sshKey := range sshPublicKeys {
var err error
found := false
keys := []byte(sshKey)
loop:
for len(keys) > 0 && err == nil {
var out ssh.PublicKey
// We ignore options as they are not relevant to Gitea
out, _, _, keys, err = ssh.ParseAuthorizedKey(keys)
2020-10-11 02:38:09 +02:00
if err != nil {
2021-07-24 11:16:34 +01:00
break loop
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
found = true
marshalled := string(ssh.MarshalAuthorizedKey(out))
marshalled = marshalled[:len(marshalled)-1]
sshKeyName := fmt.Sprintf("%s-%s", s.Name, ssh.FingerprintSHA256(out))
if _, err := AddPublicKey(ctx, usr.ID, sshKeyName, marshalled, s.ID, verified); err != nil {
2021-07-24 11:16:34 +01:00
if IsErrKeyAlreadyExist(err) {
log.Trace("AddPublicKeysBySource[%s]: Public SSH Key %s already exists for user", sshKeyName, usr.Name)
} else {
log.Error("AddPublicKeysBySource[%s]: Error adding Public SSH Key for user %s: %v", sshKeyName, usr.Name, err)
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
} else {
log.Trace("AddPublicKeysBySource[%s]: Added Public SSH Key for user %s", sshKeyName, usr.Name)
sshKeysNeedUpdate = true
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
}
if !found && err != nil {
log.Warn("AddPublicKeysBySource[%s]: Skipping invalid Public SSH Key for user %s: %v", s.Name, usr.Name, sshKey)
2020-10-11 02:38:09 +02:00
}
}
2021-07-24 11:16:34 +01:00
return sshKeysNeedUpdate
2020-10-11 02:38:09 +02:00
}
// SynchronizePublicKeys updates a user's public keys. Returns true if there are changes.
func SynchronizePublicKeys(ctx context.Context, usr *user_model.User, s *auth.Source, sshPublicKeys []string, verified bool) bool {
2021-07-24 11:16:34 +01:00
var sshKeysNeedUpdate bool
2020-10-11 02:38:09 +02:00
2021-07-24 11:16:34 +01:00
log.Trace("synchronizePublicKeys[%s]: Handling Public SSH Key synchronization for user %s", s.Name, usr.Name)
2020-10-11 02:38:09 +02:00
// Get Public Keys from DB with the current auth source
2021-07-24 11:16:34 +01:00
var giteaKeys []string
keys, err := db.Find[PublicKey](ctx, FindPublicKeyOptions{
OwnerID: usr.ID,
LoginSourceID: s.ID,
})
2021-07-24 11:16:34 +01:00
if err != nil {
log.Error("synchronizePublicKeys[%s]: Error listing Public SSH Keys for user %s: %v", s.Name, usr.Name, err)
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
for _, v := range keys {
giteaKeys = append(giteaKeys, v.OmitEmail())
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
// Process the provided keys to remove duplicates and name part
var providedKeys []string
for _, v := range sshPublicKeys {
sshKeySplit := strings.Split(v, " ")
if len(sshKeySplit) > 1 {
key := strings.Join(sshKeySplit[:2], " ")
2023-01-11 13:31:16 +08:00
if !util.SliceContainsString(providedKeys, key) {
2021-07-24 11:16:34 +01:00
providedKeys = append(providedKeys, key)
}
}
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
// Check if Public Key sync is needed
2023-01-11 13:31:16 +08:00
if util.SliceSortedEqual(giteaKeys, providedKeys) {
2021-07-24 11:16:34 +01:00
log.Trace("synchronizePublicKeys[%s]: Public Keys are already in sync for %s (Source:%v/DB:%v)", s.Name, usr.Name, len(providedKeys), len(giteaKeys))
return false
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
log.Trace("synchronizePublicKeys[%s]: Public Key needs update for user %s (Source:%v/DB:%v)", s.Name, usr.Name, len(providedKeys), len(giteaKeys))
2020-10-11 02:38:09 +02:00
2021-07-24 11:16:34 +01:00
// Add new Public SSH Keys that doesn't already exist in DB
var newKeys []string
for _, key := range providedKeys {
2023-01-11 13:31:16 +08:00
if !util.SliceContainsString(giteaKeys, key) {
2021-07-24 11:16:34 +01:00
newKeys = append(newKeys, key)
}
}
if AddPublicKeysBySource(ctx, usr, s, newKeys, verified) {
2021-07-24 11:16:34 +01:00
sshKeysNeedUpdate = true
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
// Mark keys from DB that no longer exist in the source for deletion
var giteaKeysToDelete []string
for _, giteaKey := range giteaKeys {
2023-01-11 13:31:16 +08:00
if !util.SliceContainsString(providedKeys, giteaKey) {
2021-07-24 11:16:34 +01:00
log.Trace("synchronizePublicKeys[%s]: Marking Public SSH Key for deletion for user %s: %v", s.Name, usr.Name, giteaKey)
giteaKeysToDelete = append(giteaKeysToDelete, giteaKey)
}
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
// Delete keys from DB that no longer exist in the source
needUpd, err := deleteKeysMarkedForDeletion(ctx, giteaKeysToDelete)
if err != nil {
2021-07-24 11:16:34 +01:00
log.Error("synchronizePublicKeys[%s]: Error deleting Public Keys marked for deletion for user %s: %v", s.Name, usr.Name, err)
}
2021-07-24 11:16:34 +01:00
if needUpd {
sshKeysNeedUpdate = true
2020-10-11 02:38:09 +02:00
}
2021-07-24 11:16:34 +01:00
return sshKeysNeedUpdate
2020-10-11 02:38:09 +02:00
}