Skip to content

Commit

Permalink
feat: add user email support
Browse files Browse the repository at this point in the history
  • Loading branch information
aymanbagabas committed Dec 11, 2023
1 parent caca389 commit 8fbd41a
Show file tree
Hide file tree
Showing 12 changed files with 345 additions and 23 deletions.
122 changes: 121 additions & 1 deletion pkg/backend/user.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ func (d *Backend) User(ctx context.Context, username string) (proto.User, error)
var m models.User
var pks []ssh.PublicKey
var hl models.Handle
var ems []proto.UserEmail
if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
var err error
m, err = d.store.FindUserByUsername(ctx, tx, username)
Expand All @@ -38,6 +39,15 @@ func (d *Backend) User(ctx context.Context, username string) (proto.User, error)
return err
}

Check warning on line 40 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L39-L40

Added lines #L39 - L40 were not covered by tests

emails, err := d.store.ListUserEmails(ctx, tx, m.ID)
if err != nil {
return err
}

Check warning on line 45 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L44-L45

Added lines #L44 - L45 were not covered by tests

for _, e := range emails {
ems = append(ems, &userEmail{e})
}

hl, err = d.store.GetHandleByUserID(ctx, tx, m.ID)
return err
}); err != nil {
Expand All @@ -53,6 +63,7 @@ func (d *Backend) User(ctx context.Context, username string) (proto.User, error)
user: m,
publicKeys: pks,
handle: hl,
emails: ems,
}, nil
}

Expand All @@ -61,6 +72,7 @@ func (d *Backend) UserByID(ctx context.Context, id int64) (proto.User, error) {
var m models.User
var pks []ssh.PublicKey
var hl models.Handle
var ems []proto.UserEmail
if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
var err error
m, err = d.store.GetUserByID(ctx, tx, id)
Expand All @@ -73,6 +85,15 @@ func (d *Backend) UserByID(ctx context.Context, id int64) (proto.User, error) {
return err
}

Check warning on line 86 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L85-L86

Added lines #L85 - L86 were not covered by tests

emails, err := d.store.ListUserEmails(ctx, tx, m.ID)
if err != nil {
return err
}

Check warning on line 91 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L90-L91

Added lines #L90 - L91 were not covered by tests

for _, e := range emails {
ems = append(ems, &userEmail{e})
}

Check warning on line 95 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L94-L95

Added lines #L94 - L95 were not covered by tests

hl, err = d.store.GetHandleByUserID(ctx, tx, m.ID)
return err
}); err != nil {
Expand All @@ -88,6 +109,7 @@ func (d *Backend) UserByID(ctx context.Context, id int64) (proto.User, error) {
user: m,
publicKeys: pks,
handle: hl,
emails: ems,
}, nil
}

Expand All @@ -98,6 +120,7 @@ func (d *Backend) UserByPublicKey(ctx context.Context, pk ssh.PublicKey) (proto.
var m models.User
var pks []ssh.PublicKey
var hl models.Handle
var ems []proto.UserEmail
if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
var err error
m, err = d.store.FindUserByPublicKey(ctx, tx, pk)
Expand All @@ -110,6 +133,15 @@ func (d *Backend) UserByPublicKey(ctx context.Context, pk ssh.PublicKey) (proto.
return err
}

Check warning on line 134 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L133-L134

Added lines #L133 - L134 were not covered by tests

emails, err := d.store.ListUserEmails(ctx, tx, m.ID)
if err != nil {
return err
}

Check warning on line 139 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L138-L139

Added lines #L138 - L139 were not covered by tests

for _, e := range emails {
ems = append(ems, &userEmail{e})
}

Check warning on line 143 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L142-L143

Added lines #L142 - L143 were not covered by tests

hl, err = d.store.GetHandleByUserID(ctx, tx, m.ID)
return err
}); err != nil {
Expand All @@ -125,6 +157,7 @@ func (d *Backend) UserByPublicKey(ctx context.Context, pk ssh.PublicKey) (proto.
user: m,
publicKeys: pks,
handle: hl,
emails: ems,
}, nil
}

Expand All @@ -134,6 +167,7 @@ func (d *Backend) UserByAccessToken(ctx context.Context, token string) (proto.Us
var m models.User
var pks []ssh.PublicKey
var hl models.Handle
var ems []proto.UserEmail
token = HashToken(token)

if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
Expand All @@ -156,6 +190,15 @@ func (d *Backend) UserByAccessToken(ctx context.Context, token string) (proto.Us
return err
}

Check warning on line 191 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L190-L191

Added lines #L190 - L191 were not covered by tests

emails, err := d.store.ListUserEmails(ctx, tx, m.ID)
if err != nil {
return err
}

Check warning on line 196 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L195-L196

Added lines #L195 - L196 were not covered by tests

for _, e := range emails {
ems = append(ems, &userEmail{e})
}

Check warning on line 200 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L199-L200

Added lines #L199 - L200 were not covered by tests

hl, err = d.store.GetHandleByUserID(ctx, tx, m.ID)
return err
}); err != nil {
Expand All @@ -171,6 +214,7 @@ func (d *Backend) UserByAccessToken(ctx context.Context, token string) (proto.Us
user: m,
publicKeys: pks,
handle: hl,
emails: ems,
}, nil
}

Expand Down Expand Up @@ -228,7 +272,7 @@ func (d *Backend) AddPublicKey(ctx context.Context, username string, pk ssh.Publ
// It implements backend.Backend.
func (d *Backend) CreateUser(ctx context.Context, username string, opts proto.UserOptions) (proto.User, error) {
if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
return d.store.CreateUser(ctx, tx, username, opts.Admin, opts.PublicKeys)
return d.store.CreateUser(ctx, tx, username, opts.Admin, opts.PublicKeys, opts.Emails)
}); err != nil {
return nil, db.WrapError(err)
}
Expand Down Expand Up @@ -335,10 +379,60 @@ func (d *Backend) SetPassword(ctx context.Context, username string, rawPassword
)
}

// AddUserEmail adds an email to a user.
func (d *Backend) AddUserEmail(ctx context.Context, user proto.User, email string) error {
return db.WrapError(
d.db.TransactionContext(ctx, func(tx *db.Tx) error {
return d.store.AddUserEmail(ctx, tx, user.ID(), email, false)
}),
)
}

// ListUserEmails lists the emails of a user.
func (d *Backend) ListUserEmails(ctx context.Context, user proto.User) ([]proto.UserEmail, error) {
var ems []proto.UserEmail
if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
emails, err := d.store.ListUserEmails(ctx, tx, user.ID())
if err != nil {
return err
}

Check warning on line 398 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L392-L398

Added lines #L392 - L398 were not covered by tests

for _, e := range emails {
ems = append(ems, &userEmail{e})
}

Check warning on line 402 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L400-L402

Added lines #L400 - L402 were not covered by tests

return nil
}); err != nil {
return nil, db.WrapError(err)
}

Check warning on line 407 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L404-L407

Added lines #L404 - L407 were not covered by tests

return ems, nil

Check warning on line 409 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L409

Added line #L409 was not covered by tests
}

// RemoveUserEmail deletes an email for a user.
// The deleted email must not be the primary email.
func (d *Backend) RemoveUserEmail(ctx context.Context, user proto.User, email string) error {
return db.WrapError(
d.db.TransactionContext(ctx, func(tx *db.Tx) error {
return d.store.RemoveUserEmail(ctx, tx, user.ID(), email)
}),
)
}

// SetUserPrimaryEmail sets the primary email of a user.
func (d *Backend) SetUserPrimaryEmail(ctx context.Context, user proto.User, email string) error {
return db.WrapError(
d.db.TransactionContext(ctx, func(tx *db.Tx) error {
return d.store.SetUserPrimaryEmail(ctx, tx, user.ID(), email)
}),
)
}

type user struct {
user models.User
publicKeys []ssh.PublicKey
handle models.Handle
emails []proto.UserEmail
}

var _ proto.User = (*user)(nil)
Expand Down Expand Up @@ -371,3 +465,29 @@ func (u *user) Password() string {

return ""
}

// Emails implements proto.User.
func (u *user) Emails() []proto.UserEmail {
return u.emails
}

type userEmail struct {
email models.UserEmail
}

var _ proto.UserEmail = (*userEmail)(nil)

// Email implements proto.UserEmail.
func (e *userEmail) Email() string {
return e.email.Email
}

// ID implements proto.UserEmail.
func (e *userEmail) ID() int64 {
return e.email.ID

Check warning on line 487 in pkg/backend/user.go

View check run for this annotation

Codecov / codecov/patch

pkg/backend/user.go#L486-L487

Added lines #L486 - L487 were not covered by tests
}

// IsPrimary implements proto.UserEmail.
func (e *userEmail) IsPrimary() bool {
return e.email.IsPrimary
}
9 changes: 6 additions & 3 deletions pkg/db/migrate/0004_create_orgs_teams_postgres.up.sql
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ CREATE TABLE IF NOT EXISTS user_emails (
id SERIAL PRIMARY KEY,
user_id INTEGER NOT NULL,
email TEXT NOT NULL UNIQUE,
is_primary BOOLEAN NOT NULL,
is_primary BOOLEAN NOT NULL DEFAULT false,
created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
updated_at TIMESTAMP NOT NULL,
CONSTRAINT user_id_fk
Expand All @@ -78,6 +78,9 @@ CREATE TABLE IF NOT EXISTS user_emails (
ON UPDATE CASCADE
);

-- Create unique index for primary email
CREATE UNIQUE INDEX user_emails_user_id_is_primary_idx ON user_emails (user_id) WHERE is_primary;

-- Add name to users table
ALTER TABLE users ADD COLUMN name TEXT;

Expand Down Expand Up @@ -112,7 +115,7 @@ ALTER TABLE repos ADD CONSTRAINT org_id_fk
ALTER TABLE repos ALTER COLUMN user_id DROP NOT NULL;

-- Check that both user_id and org_id can't be null
ALTER TABLE repos ADD CONSTRAINT user_id_org_id_not_null CHECK (user_id IS NULL <> org_id IS NULL);
ALTER TABLE repos ADD CONSTRAINT user_id_org_id_not_null CHECK ((user_id IS NULL) <> (org_id IS NULL));

-- Add team_id to collabs table
ALTER TABLE collabs ADD COLUMN team_id INTEGER;
Expand All @@ -125,7 +128,7 @@ ALTER TABLE collabs ADD CONSTRAINT team_id_fk
ALTER TABLE collabs ALTER COLUMN user_id DROP NOT NULL;

-- Check that both user_id and team_id can't be null
ALTER TABLE collabs ADD CONSTRAINT user_id_team_id_not_null CHECK (user_id IS NULL <> team_id IS NULL);
ALTER TABLE collabs ADD CONSTRAINT user_id_team_id_not_null CHECK ((user_id IS NULL) <> (team_id IS NULL));

-- Alter unique constraint on collabs table
ALTER TABLE collabs DROP CONSTRAINT collabs_user_id_repo_id_key;
Expand Down
5 changes: 4 additions & 1 deletion pkg/db/migrate/0004_create_orgs_teams_sqlite.up.sql
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,7 @@ CREATE TABLE IF NOT EXISTS user_emails (
id INTEGER PRIMARY KEY AUTOINCREMENT,
user_id INTEGER NOT NULL,
email TEXT NOT NULL UNIQUE,
is_primary BOOLEAN NOT NULL,
is_primary BOOLEAN NOT NULL DEFAULT false,
created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
updated_at TIMESTAMP NOT NULL,
CONSTRAINT user_id_fk
Expand All @@ -80,6 +80,9 @@ CREATE TABLE IF NOT EXISTS user_emails (
ON UPDATE CASCADE
);

-- Create unique index for primary email
CREATE UNIQUE INDEX user_emails_user_id_is_primary_idx ON user_emails (user_id) WHERE is_primary;

ALTER TABLE users RENAME TO _users_old;

CREATE TABLE IF NOT EXISTS users (
Expand Down
17 changes: 17 additions & 0 deletions pkg/proto/user.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,8 @@ type User interface {
PublicKeys() []ssh.PublicKey
// Password returns the user's password hash.
Password() string
// Emails returns the user's emails.
Emails() []UserEmail
}

// UserOptions are options for creating a user.
Expand All @@ -22,4 +24,19 @@ type UserOptions struct {
Admin bool
// PublicKeys are the user's public keys.
PublicKeys []ssh.PublicKey
// Emails are the user's emails.
// The first email in the slice will be set as the user's primary email.
Emails []string
}

// UserEmail represents a user's email address.
type UserEmail interface {
// ID returns the email's ID.
ID() int64

// Email returns the email address.
Email() string

// IsPrimary returns whether the email is the user's primary email.
IsPrimary() bool
}
2 changes: 1 addition & 1 deletion pkg/ssh/cmd/org.go
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ func OrgCommand() *cobra.Command {
Use: "list",
Short: "List organizations",
Args: cobra.NoArgs,
RunE: func(cmd *cobra.Command, args []string) error {
RunE: func(cmd *cobra.Command, _ []string) error {
ctx := cmd.Context()
be := backend.FromContext(ctx)
user := proto.UserFromContext(ctx)
Expand Down
Loading

0 comments on commit 8fbd41a

Please sign in to comment.