Skip to content

Commit

Permalink
feat: initial orgs & teams support
Browse files Browse the repository at this point in the history
Squashed commit of the following:

commit 26d3df790c79a0803033ba71b7c394a27c401f1e
Author: Ayman Bagabas <[email protected]>
Date:   Thu Nov 30 10:33:32 2023 -0800

    feat: add team collaborators (#19)

    Repository collaborators can be either individual users, or organization
    teams. When it's an org team, the repo must belong to the same
    organization that the team belongs to. This validation happens in the
    app logic.

commit 8a598a1898228432114d81d158eb25578b112353
Merge: 9399cd6a60f9 5ae66bc932bd
Author: Carlos Alexandro Becker <[email protected]>
Date:   Tue Nov 14 09:37:42 2023 -0300

    Merge pull request #15 from charmbracelet/orgs

    feat: initial organization support

commit 5ae66bc932bde406b994877c7c32def5115d9afc
Author: Carlos Alexandro Becker <[email protected]>
Date:   Tue Nov 14 10:06:20 2023 +0000

    fix: code review

    Signed-off-by: Carlos Alexandro Becker <[email protected]>

commit 4d16aecedec8b1a46ee1ea73c2d1cc4b51f0ebd1
Author: Carlos Alexandro Becker <[email protected]>
Date:   Mon Nov 13 18:55:04 2023 +0000

    wip team

commit 3374c8cd24f3c03188cf1f9e27b9242fd9df00a1
Author: Carlos Alexandro Becker <[email protected]>
Date:   Mon Nov 13 17:15:54 2023 +0000

    wip team

commit 890975daab842cab7207d68535c7ed5e2dec98c7
Author: Carlos Alexandro Becker <[email protected]>
Date:   Mon Nov 13 15:59:22 2023 +0000

    wip teams

commit d67773949c1ca93ca392d156241f9f166e471723
Author: Carlos Alexandro Becker <[email protected]>
Date:   Mon Nov 13 15:44:33 2023 +0000

    fix: team signatures

commit b3881958409bd98bc8861f4048826e9efaca7b6d
Author: Carlos Alexandro Becker <[email protected]>
Date:   Mon Nov 13 15:44:26 2023 +0000

    fix: rename method

commit 509585f7c6444e0a0106d1100204bee0ee65e70f
Author: Carlos Alexandro Becker <[email protected]>
Date:   Mon Nov 13 12:32:03 2023 +0000

    wip

commit ab0124e25c12cf2d6c8d75f793ac80a236bbbe82
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 20:47:15 2023 +0000

    fix: cr

commit 3e6de9b54dbb4d0cefe3891f424976c760c39ee3
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 20:44:46 2023 +0000

    fixes

commit 452815ea7231b3d362b95ea7ab2149492862974a
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 20:37:44 2023 +0000

    wip

commit aa25e4fc40104c9c390b2d69eb4fc33128938074
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 20:35:39 2023 +0000

    wip

commit 28241cc64e57b2b3a30ed6897b1156a88965e99b
Merge: 12a70b3e5e0c 9399cd6a60f9
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 20:33:30 2023 +0000

    Merge remote-tracking branch 'origin/orgs-teams' into orgs

commit 9399cd6a60f9d3a6e6bf95aa4bb454161944c540
Author: Ayman Bagabas <[email protected]>
Date:   Thu Nov 9 12:22:02 2023 -0500

    feat: add email user relations and models

commit a0715c42d628528292501591c6b5764a5ee80ff8
Author: Ayman Bagabas <[email protected]>
Date:   Thu Nov 9 10:08:23 2023 -0500

    fix: carlos comments

commit 12a70b3e5e0cc625dd48511d035e632b551fd05b
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 13:40:37 2023 +0000

    fix: admin

commit 637c8bccd6877e3447a9e1f6f13e1a1105a3616f
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 12:57:01 2023 +0000

    wip

commit 4ec7653f4ba6fed4faee21f8416d2dbbe46ef3d3
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 12:32:46 2023 +0000

    fix: merge issues

commit d8b8e22f98f5700d84d599e9a0483ab1d0572497
Merge: c2bf2721d2d0 777e451128b1
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 12:27:26 2023 +0000

    Merge remote-tracking branch 'origin/orgs-teams' into orgs

commit c2bf2721d2d0b17e4cfa2cc1a45d9a21cf4197fd
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 02:25:19 2023 +0000

    wip

commit 92b5f57ec09a013b051dd6828dd1beb4ed95d641
Author: Carlos Alexandro Becker <[email protected]>
Date:   Thu Nov 9 02:10:37 2023 +0000

    wip

commit 83f6cf906a5779168706277ec37e3bb84578b55b
Author: Carlos Alexandro Becker <[email protected]>
Date:   Wed Nov 8 19:42:02 2023 +0000

    wip

commit 777e451128b141304341e6497d6a96a6f4185f3a
Author: Ayman Bagabas <[email protected]>
Date:   Wed Nov 8 12:56:25 2023 -0500

    fix: lint

commit 50f2b054550489bf8df81a96f553764096734c89
Author: Ayman Bagabas <[email protected]>
Date:   Wed Nov 8 12:53:24 2023 -0500

    feat: add models and missing columns

commit 84cb588
Author: Ayman Bagabas <[email protected]>
Date:   Wed Nov 8 12:39:44 2023 -0500

    fix(backend): update backend to use handles table

commit af16adab439e5905704be986e32be48ef85d2846
Author: Carlos Alexandro Becker <[email protected]>
Date:   Wed Nov 8 17:34:18 2023 +0000

    wip: adding orgs

commit a222f24
Author: Ayman Bagabas <[email protected]>
Date:   Wed Nov 8 07:33:06 2023 -0800

    Add organizations and teams migration (#9)

    * feat(db): pre/post migration

    * feat(db): add create orgs/teams migration

commit f7f521e
Author: Ayman Bagabas <[email protected]>
Date:   Tue Nov 7 16:44:09 2023 -0500

    wip
  • Loading branch information
aymanbagabas committed Dec 11, 2023
1 parent c3915b2 commit caca389
Show file tree
Hide file tree
Showing 46 changed files with 1,750 additions and 241 deletions.
95 changes: 95 additions & 0 deletions pkg/backend/access.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
package backend

import (
"context"

"github.com/charmbracelet/soft-serve/pkg/access"
"github.com/charmbracelet/soft-serve/pkg/proto"
"github.com/charmbracelet/soft-serve/pkg/sshutils"
"golang.org/x/crypto/ssh"
)

// AccessLevel returns the access level of a user for a repository.
//
// It implements backend.Backend.
func (d *Backend) AccessLevel(ctx context.Context, repo string, username string) access.AccessLevel {
user, _ := d.User(ctx, username)
return d.AccessLevelForUser(ctx, repo, user)
}

// AccessLevelByPublicKey returns the access level of a user's public key for a repository.
//
// It implements backend.Backend.
func (d *Backend) AccessLevelByPublicKey(ctx context.Context, repo string, pk ssh.PublicKey) access.AccessLevel {
for _, k := range d.cfg.AdminKeys() {
if sshutils.KeysEqual(pk, k) {
return access.AdminAccess
}
}

user, _ := d.UserByPublicKey(ctx, pk)
if user != nil {
return d.AccessLevel(ctx, repo, user.Username())
}

return d.AccessLevel(ctx, repo, "")
}

// AccessLevelForUser returns the access level of a user for a repository.
// TODO: user repository ownership

Check failure on line 39 in pkg/backend/access.go

View workflow job for this annotation

GitHub Actions / lint-soft

Comment should end in a period (godot)
func (d *Backend) AccessLevelForUser(ctx context.Context, repo string, user proto.User) access.AccessLevel {
var username string
anon := d.AnonAccess(ctx)
if user != nil {
username = user.Username()
}

// If the user is an admin, they have admin access.
if user != nil && user.IsAdmin() {
return access.AdminAccess
}

// If the repository exists, check if the user is a collaborator.
r := proto.RepositoryFromContext(ctx)
if r == nil {
r, _ = d.Repository(ctx, repo)
}

if r != nil {
if user != nil {
// If the user is the owner, they have admin access.
if r.UserID() == user.ID() {
return access.AdminAccess
}
}

// If the user is a collaborator, they have return their access level.
collabAccess, isCollab, _ := d.IsCollaborator(ctx, repo, username)
if isCollab {
if anon > collabAccess {
return anon
}
return collabAccess
}

// If the repository is private, the user has no access.
if r.IsPrivate() {
return access.NoAccess
}

// Otherwise, the user has read-only access.
return access.ReadOnlyAccess
}

if user != nil {
// If the repository doesn't exist, the user has read/write access.
if anon > access.ReadWriteAccess {
return anon
}

return access.ReadWriteAccess
}

// If the user doesn't exist, give them the anonymous access level.
return anon
}
28 changes: 21 additions & 7 deletions pkg/backend/collab.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ import (
// It implements backend.Backend.
func (d *Backend) AddCollaborator(ctx context.Context, repo string, username string, level access.AccessLevel) error {
username = strings.ToLower(username)
if err := utils.ValidateUsername(username); err != nil {
if err := utils.ValidateHandle(username); err != nil {
return err
}

Expand Down Expand Up @@ -50,19 +50,33 @@ func (d *Backend) AddCollaborator(ctx context.Context, repo string, username str
func (d *Backend) Collaborators(ctx context.Context, repo string) ([]string, error) {
repo = utils.SanitizeRepo(repo)
var users []models.User
var usernames []string
if err := d.db.TransactionContext(ctx, func(tx *db.Tx) error {
var err error
users, err = d.store.ListCollabsByRepoAsUsers(ctx, tx, repo)
return err
if err != nil {
return err
}

ids := make([]int64, len(users))
for i, u := range users {
ids[i] = u.ID
}

handles, err := d.store.ListHandlesForIDs(ctx, tx, ids)
if err != nil {
return err
}

for _, h := range handles {
usernames = append(usernames, h.Handle)
}

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

var usernames []string
for _, u := range users {
usernames = append(usernames, u.Username)
}

return usernames, nil
}

Expand Down
66 changes: 66 additions & 0 deletions pkg/backend/org.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
package backend

import (
"context"

"github.com/charmbracelet/soft-serve/pkg/db/models"
"github.com/charmbracelet/soft-serve/pkg/proto"
)

// CreateOrg creates a new organization.
func (d *Backend) CreateOrg(ctx context.Context, owner proto.User, name, email string) (proto.Org, error) {
o, err := d.store.CreateOrg(ctx, d.db, owner.ID(), name, email)
if err != nil {
return org{}, err
}
return org{o}, err
}

// ListOrgs lists all organizations for a user.
func (d *Backend) ListOrgs(ctx context.Context, user proto.User) ([]proto.Org, error) {
orgs, err := d.store.ListOrgs(ctx, d.db, user.ID())
var r []proto.Org
for _, o := range orgs {
r = append(r, org{o})
}
return r, err
}

// FindOrganization finds an organization belonging to a user by name.
func (d *Backend) FindOrganization(ctx context.Context, user proto.User, name string) (proto.Org, error) {
o, err := d.store.FindOrgByHandle(ctx, d.db, user.ID(), name)
return org{o}, err
}

// DeleteOrganization deletes an organization for a user.
func (d *Backend) DeleteOrganization(ctx context.Context, user proto.User, name string) error {
o, err := d.store.FindOrgByHandle(ctx, d.db, user.ID(), name)
if err != nil {
return err
}
return d.store.DeleteOrgByID(ctx, d.db, user.ID(), o.ID)
}

type org struct {
o models.Organization
}

var _ proto.Org = org{}

// DisplayName implements proto.Org.
func (o org) DisplayName() string {
if o.o.Name.Valid {
return o.o.Name.String
}
return ""
}

// ID implements proto.Org.
func (o org) ID() int64 {
return o.o.ID
}

// Name implements proto.Org.
func (o org) Name() string {
return o.o.Handle.Handle
}
72 changes: 72 additions & 0 deletions pkg/backend/team.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
package backend

import (
"context"

"github.com/charmbracelet/soft-serve/pkg/db/models"
"github.com/charmbracelet/soft-serve/pkg/proto"
)

// CreateTeam creates a new team for an organization.
func (d *Backend) CreateTeam(ctx context.Context, org proto.Org, owner proto.User, name string) (proto.Team, error) {
m, err := d.store.CreateTeam(ctx, d.db, owner.ID(), org.ID(), name)
if err != nil {
return team{}, err
}
return team{m}, err
}

// ListTeams lists all teams for a user.
func (d *Backend) ListTeams(ctx context.Context, user proto.User) ([]proto.Team, error) {
teams, err := d.store.ListTeams(ctx, d.db, user.ID())
var r []proto.Team
for _, m := range teams {
r = append(r, team{m})
}
return r, err
}

// GetTeam gets a team by organization id and team name.
func (d *Backend) GetTeam(ctx context.Context, user proto.User, org proto.Org, name string) (proto.Team, error) {
m, err := d.store.FindTeamByOrgName(ctx, d.db, user.ID(), org.ID(), name)
if err != nil {
return team{}, err
}
return team{m}, err
}

// FindTeam finds a team by name.
func (d *Backend) FindTeam(ctx context.Context, user proto.User, name string) ([]proto.Team, error) {
m, err := d.store.FindTeamByName(ctx, d.db, user.ID(), name)
var r []proto.Team
for _, m := range m {
r = append(r, team{m})
}
return r, err
}

// DeleteTeam deletes a team.
func (d *Backend) DeleteTeam(ctx context.Context, _ proto.User, team proto.Team) error {
return d.store.DeleteTeamByID(ctx, d.db, team.ID())
}

type team struct {
t models.Team
}

var _ proto.Team = team{}

// ID implements proto.Team.
func (t team) ID() int64 {
return t.t.ID
}

// Name implements proto.Team.
func (t team) Name() string {
return t.t.Name
}

// Org implements proto.Team.
func (t team) Org() int64 {
return t.t.OrganizationID
}
Loading

0 comments on commit caca389

Please sign in to comment.