aboutsummaryrefslogtreecommitdiff
path: root/internal/db/repos.go
diff options
context:
space:
mode:
Diffstat (limited to 'internal/db/repos.go')
-rw-r--r--internal/db/repos.go91
1 files changed, 90 insertions, 1 deletions
diff --git a/internal/db/repos.go b/internal/db/repos.go
index 10f292b9..7791a458 100644
--- a/internal/db/repos.go
+++ b/internal/db/repos.go
@@ -11,6 +11,7 @@ import (
"time"
api "github.com/gogs/go-gogs-client"
+ "github.com/pkg/errors"
"gorm.io/gorm"
"gogs.io/gogs/internal/errutil"
@@ -36,9 +37,14 @@ type ReposStore interface {
// Repositories that are owned directly by the given collaborator are not
// included.
GetByCollaboratorIDWithAccessMode(ctx context.Context, collaboratorID int64) (map[*Repository]AccessMode, error)
+ // GetByID returns the repository with given ID. It returns ErrRepoNotExist when
+ // not found.
+ GetByID(ctx context.Context, id int64) (*Repository, error)
// GetByName returns the repository with given owner and name. It returns
// ErrRepoNotExist when not found.
GetByName(ctx context.Context, ownerID int64, name string) (*Repository, error)
+ // Star marks the user to star the repository.
+ Star(ctx context.Context, userID, repoID int64) error
// Touch updates the updated time to the current time and removes the bare state
// of the given repository.
Touch(ctx context.Context, id int64) error
@@ -177,7 +183,18 @@ func (db *repos) Create(ctx context.Context, ownerID int64, opts CreateRepoOptio
IsFork: opts.Fork,
ForkID: opts.ForkID,
}
- return repo, db.WithContext(ctx).Create(repo).Error
+ return repo, db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
+ err = tx.Create(repo).Error
+ if err != nil {
+ return errors.Wrap(err, "create")
+ }
+
+ err = NewWatchesStore(tx).Watch(ctx, ownerID, repo.ID)
+ if err != nil {
+ return errors.Wrap(err, "watch")
+ }
+ return nil
+ })
}
func (db *repos) GetByCollaboratorID(ctx context.Context, collaboratorID int64, limit int, orderBy string) ([]*Repository, error) {
@@ -252,6 +269,18 @@ func (ErrRepoNotExist) NotFound() bool {
return true
}
+func (db *repos) GetByID(ctx context.Context, id int64) (*Repository, error) {
+ repo := new(Repository)
+ err := db.WithContext(ctx).Where("id = ?", id).First(repo).Error
+ if err != nil {
+ if err == gorm.ErrRecordNotFound {
+ return nil, ErrRepoNotExist{errutil.Args{"repoID": id}}
+ }
+ return nil, err
+ }
+ return repo, nil
+}
+
func (db *repos) GetByName(ctx context.Context, ownerID int64, name string) (*Repository, error) {
repo := new(Repository)
err := db.WithContext(ctx).
@@ -272,6 +301,66 @@ func (db *repos) GetByName(ctx context.Context, ownerID int64, name string) (*Re
return repo, nil
}
+func (db *repos) recountStars(tx *gorm.DB, userID, repoID int64) error {
+ /*
+ Equivalent SQL for PostgreSQL:
+
+ UPDATE repository
+ SET num_stars = (
+ SELECT COUNT(*) FROM star WHERE repo_id = @repoID
+ )
+ WHERE id = @repoID
+ */
+ err := tx.Model(&Repository{}).
+ Where("id = ?", repoID).
+ Update(
+ "num_stars",
+ tx.Model(&Star{}).Select("COUNT(*)").Where("repo_id = ?", repoID),
+ ).
+ Error
+ if err != nil {
+ return errors.Wrap(err, `update "repository.num_stars"`)
+ }
+
+ /*
+ Equivalent SQL for PostgreSQL:
+
+ UPDATE "user"
+ SET num_stars = (
+ SELECT COUNT(*) FROM star WHERE uid = @userID
+ )
+ WHERE id = @userID
+ */
+ err = tx.Model(&User{}).
+ Where("id = ?", userID).
+ Update(
+ "num_stars",
+ tx.Model(&Star{}).Select("COUNT(*)").Where("uid = ?", userID),
+ ).
+ Error
+ if err != nil {
+ return errors.Wrap(err, `update "user.num_stars"`)
+ }
+ return nil
+}
+
+func (db *repos) Star(ctx context.Context, userID, repoID int64) error {
+ return db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
+ s := &Star{
+ UserID: userID,
+ RepoID: repoID,
+ }
+ result := tx.FirstOrCreate(s, s)
+ if result.Error != nil {
+ return errors.Wrap(result.Error, "upsert")
+ } else if result.RowsAffected <= 0 {
+ return nil // Relation already exists
+ }
+
+ return db.recountStars(tx, userID, repoID)
+ })
+}
+
func (db *repos) Touch(ctx context.Context, id int64) error {
return db.WithContext(ctx).
Model(new(Repository)).