/
webauthn_credential_persister.go
100 lines (82 loc) · 2.86 KB
/
webauthn_credential_persister.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
package persistence
import (
"database/sql"
"errors"
"fmt"
"github.com/gobuffalo/pop/v6"
"github.com/gofrs/uuid"
"github.com/teamhanko/hanko/backend/persistence/models"
)
type WebauthnCredentialPersister interface {
Get(string) (*models.WebauthnCredential, error)
Create(models.WebauthnCredential) error
Update(models.WebauthnCredential) error
Delete(models.WebauthnCredential) error
GetFromUser(uuid.UUID) ([]models.WebauthnCredential, error)
}
type webauthnCredentialPersister struct {
db *pop.Connection
}
func NewWebauthnCredentialPersister(db *pop.Connection) WebauthnCredentialPersister {
return &webauthnCredentialPersister{db: db}
}
func (p *webauthnCredentialPersister) Get(id string) (*models.WebauthnCredential, error) {
credential := models.WebauthnCredential{}
err := p.db.Find(&credential, id)
if err != nil && errors.Is(err, sql.ErrNoRows) {
return nil, nil
}
if err != nil {
return nil, fmt.Errorf("failed to get credential: %w", err)
}
return &credential, nil
}
// Create stores a new `WebauthnCredential`. Please run inside a transaction, since `Transports` associated with the
// credential are stored separately in another table.
func (p *webauthnCredentialPersister) Create(credential models.WebauthnCredential) error {
vErr, err := p.db.ValidateAndCreate(&credential)
if err != nil {
return fmt.Errorf("failed to store credential: %w", err)
}
if vErr != nil && vErr.HasAny() {
return fmt.Errorf("credential object validation failed: %w", vErr)
}
// Eager creation seems to be broken, so we need to store the transports separately.
// See: https://github.com/gobuffalo/pop/issues/608
vErr, err = p.db.ValidateAndCreate(&credential.Transports)
if err != nil {
return fmt.Errorf("failed to store credential transport: %w", err)
}
if vErr != nil && vErr.HasAny() {
return fmt.Errorf("credential transport validation failed: %w", vErr)
}
return nil
}
func (p *webauthnCredentialPersister) Update(credential models.WebauthnCredential) error {
vErr, err := p.db.ValidateAndUpdate(&credential)
if err != nil {
return fmt.Errorf("failed to update credential: %w", err)
}
if vErr != nil && vErr.HasAny() {
return fmt.Errorf("credential object validation failed: %w", vErr)
}
return nil
}
func (p *webauthnCredentialPersister) Delete(credential models.WebauthnCredential) error {
err := p.db.Destroy(&credential)
if err != nil {
return fmt.Errorf("failed to delete credential: %w", err)
}
return nil
}
func (p *webauthnCredentialPersister) GetFromUser(userId uuid.UUID) ([]models.WebauthnCredential, error) {
var credentials []models.WebauthnCredential
err := p.db.Eager().Where("user_id = ?", &userId).Order("created_at asc").All(&credentials)
if err != nil && errors.Is(err, sql.ErrNoRows) {
return credentials, nil
}
if err != nil {
return nil, fmt.Errorf("failed to get credentials: %w", err)
}
return credentials, nil
}