| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293 |
- // Copyright 2024 Gitea. All rights reserved.
- // SPDX-License-Identifier: MIT
-
- package git
-
- import (
- "context"
-
- "code.gitea.io/gitea/models/db"
- "code.gitea.io/gitea/modules/commitstatus"
- "code.gitea.io/gitea/modules/setting"
-
- "xorm.io/builder"
- )
-
- // CommitStatusSummary holds the latest commit Status of a single Commit
- type CommitStatusSummary struct {
- ID int64 `xorm:"pk autoincr"`
- RepoID int64 `xorm:"INDEX UNIQUE(repo_id_sha)"`
- SHA string `xorm:"VARCHAR(64) NOT NULL INDEX UNIQUE(repo_id_sha)"`
- State commitstatus.CommitStatusState `xorm:"VARCHAR(7) NOT NULL"`
- TargetURL string `xorm:"TEXT"`
- }
-
- func init() {
- db.RegisterModel(new(CommitStatusSummary))
- }
-
- type RepoSHA struct {
- RepoID int64
- SHA string
- }
-
- func GetLatestCommitStatusForRepoAndSHAs(ctx context.Context, repoSHAs []RepoSHA) ([]*CommitStatus, error) {
- cond := builder.NewCond()
- for _, rs := range repoSHAs {
- cond = cond.Or(builder.Eq{"repo_id": rs.RepoID, "sha": rs.SHA})
- }
-
- var summaries []CommitStatusSummary
- if err := db.GetEngine(ctx).Where(cond).Find(&summaries); err != nil {
- return nil, err
- }
-
- commitStatuses := make([]*CommitStatus, 0, len(repoSHAs))
- for _, summary := range summaries {
- commitStatuses = append(commitStatuses, &CommitStatus{
- RepoID: summary.RepoID,
- SHA: summary.SHA,
- State: summary.State,
- TargetURL: summary.TargetURL,
- })
- }
- return commitStatuses, nil
- }
-
- func UpdateCommitStatusSummary(ctx context.Context, repoID int64, sha string) error {
- commitStatuses, err := GetLatestCommitStatus(ctx, repoID, sha, db.ListOptionsAll)
- if err != nil {
- return err
- }
- // it guarantees that commitStatuses is not empty because this function is always called after a commit status is created
- if len(commitStatuses) == 0 {
- setting.PanicInDevOrTesting("no commit statuses found for repo %d and sha %s", repoID, sha)
- }
- state := CalcCommitStatus(commitStatuses) // non-empty commitStatuses is guaranteed
- // mysql will return 0 when update a record which state hasn't been changed which behaviour is different from other database,
- // so we need to use insert in on duplicate
- if setting.Database.Type.IsMySQL() {
- _, err := db.GetEngine(ctx).Exec("INSERT INTO commit_status_summary (repo_id,sha,state,target_url) VALUES (?,?,?,?) ON DUPLICATE KEY UPDATE state=?",
- repoID, sha, state.State, state.TargetURL, state.State)
- return err
- }
-
- if cnt, err := db.GetEngine(ctx).Where("repo_id=? AND sha=?", repoID, sha).
- Cols("state, target_url").
- Update(&CommitStatusSummary{
- State: state.State,
- TargetURL: state.TargetURL,
- }); err != nil {
- return err
- } else if cnt == 0 {
- _, err = db.GetEngine(ctx).Insert(&CommitStatusSummary{
- RepoID: repoID,
- SHA: sha,
- State: state.State,
- TargetURL: state.TargetURL,
- })
- return err
- }
- return nil
- }
|