You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

173 lines
4.6 KiB

package postgres
import (
"context"
"database/sql"
"encoding/json"
"fmt"
"github.com/seaweedfs/seaweedfs/weed/credential"
"github.com/seaweedfs/seaweedfs/weed/pb/iam_pb"
)
func (store *PostgresStore) CreateServiceAccount(ctx context.Context, sa *iam_pb.ServiceAccount) error {
if sa == nil {
return fmt.Errorf("service account is nil")
}
if sa.Id == "" {
return fmt.Errorf("service account ID is required")
}
if !store.configured {
return fmt.Errorf("store not configured")
}
data, err := json.Marshal(sa)
if err != nil {
return fmt.Errorf("failed to marshal service account: %w", err)
}
accessKey := ""
if sa.Credential != nil {
accessKey = sa.Credential.AccessKey
}
_, err = store.db.ExecContext(ctx,
"INSERT INTO service_accounts (id, access_key, content) VALUES ($1, $2, $3)",
sa.Id, accessKey, data)
if err != nil {
return fmt.Errorf("failed to insert service account: %w", err)
}
return nil
}
func (store *PostgresStore) UpdateServiceAccount(ctx context.Context, id string, sa *iam_pb.ServiceAccount) error {
if sa == nil {
return fmt.Errorf("service account is nil")
}
if id == "" {
return fmt.Errorf("service account ID is required")
}
if sa.Id != id {
return fmt.Errorf("service account ID mismatch")
}
data, err := json.Marshal(sa)
if err != nil {
return fmt.Errorf("failed to marshal service account: %w", err)
}
accessKey := ""
if sa.Credential != nil {
accessKey = sa.Credential.AccessKey
}
result, err := store.db.ExecContext(ctx,
"UPDATE service_accounts SET access_key = $2, content = $3, updated_at = CURRENT_TIMESTAMP WHERE id = $1",
id, accessKey, data)
if err != nil {
return fmt.Errorf("failed to update service account: %w", err)
}
rows, err := result.RowsAffected()
if err != nil {
return err
}
if rows == 0 {
return credential.ErrServiceAccountNotFound
}
return nil
}
func (store *PostgresStore) DeleteServiceAccount(ctx context.Context, id string) error {
if !store.configured {
return fmt.Errorf("store not configured")
}
result, err := store.db.ExecContext(ctx, "DELETE FROM service_accounts WHERE id = $1", id)
if err != nil {
return fmt.Errorf("failed to delete service account: %w", err)
}
rows, err := result.RowsAffected()
if err != nil {
return err
}
if rows == 0 {
return credential.ErrServiceAccountNotFound
}
return nil
}
func (store *PostgresStore) GetServiceAccount(ctx context.Context, id string) (*iam_pb.ServiceAccount, error) {
if !store.configured {
return nil, fmt.Errorf("store not configured")
}
var content []byte
err := store.db.QueryRowContext(ctx, "SELECT content FROM service_accounts WHERE id = $1", id).Scan(&content)
if err != nil {
if err == sql.ErrNoRows {
return nil, credential.ErrServiceAccountNotFound
}
return nil, fmt.Errorf("failed to get service account: %w", err)
}
sa := &iam_pb.ServiceAccount{}
if err := json.Unmarshal(content, sa); err != nil {
return nil, fmt.Errorf("failed to unmarshal service account: %w", err)
}
return sa, nil
}
func (store *PostgresStore) ListServiceAccounts(ctx context.Context) ([]*iam_pb.ServiceAccount, error) {
if !store.configured {
return nil, fmt.Errorf("store not configured")
}
rows, err := store.db.QueryContext(ctx, "SELECT content FROM service_accounts")
if err != nil {
return nil, fmt.Errorf("failed to list service accounts: %w", err)
}
defer rows.Close()
var accounts []*iam_pb.ServiceAccount
for rows.Next() {
var content []byte
if err := rows.Scan(&content); err != nil {
return nil, fmt.Errorf("failed to scan service account: %w", err)
}
sa := &iam_pb.ServiceAccount{}
if err := json.Unmarshal(content, sa); err != nil {
return nil, fmt.Errorf("failed to unmarshal service account: %w", err)
}
accounts = append(accounts, sa)
}
if err := rows.Err(); err != nil {
return nil, fmt.Errorf("error iterating service accounts: %w", err)
}
return accounts, nil
}
func (store *PostgresStore) GetServiceAccountByAccessKey(ctx context.Context, accessKey string) (*iam_pb.ServiceAccount, error) {
if !store.configured {
return nil, fmt.Errorf("store not configured")
}
var content []byte
err := store.db.QueryRowContext(ctx, "SELECT content FROM service_accounts WHERE access_key = $1", accessKey).Scan(&content)
if err != nil {
if err == sql.ErrNoRows {
return nil, credential.ErrAccessKeyNotFound
}
return nil, fmt.Errorf("failed to query service account by access key: %w", err)
}
sa := &iam_pb.ServiceAccount{}
if err := json.Unmarshal(content, sa); err != nil {
return nil, fmt.Errorf("failed to unmarshal service account: %w", err)
}
return sa, nil
}