mirror of
				https://gitcode.com/gitea/gitea.git
				synced 2025-10-26 05:04:27 +08:00 
			
		
		
		
	Status-API (#1332)
This commit is contained in:
		 Kim "BKC" Carlbäcker
					Kim "BKC" Carlbäcker
				
			
				
					committed by
					
						 Lunny Xiao
						Lunny Xiao
					
				
			
			
				
	
			
			
			 Lunny Xiao
						Lunny Xiao
					
				
			
						parent
						
							52627032bc
						
					
				
				
					commit
					4bea219128
				
			
							
								
								
									
										54
									
								
								models/fixtures/commit_status.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								models/fixtures/commit_status.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,54 @@ | ||||
| - | ||||
|   id: 1 | ||||
|   index: 1 | ||||
|   repo_id: 1 | ||||
|   state: "pending" | ||||
|   sha: "1234123412341234123412341234123412341234" | ||||
|   target_url: https://example.com/builds/ | ||||
|   description: My awesome CI-service | ||||
|   context: ci/awesomeness | ||||
|   creator_id: 2 | ||||
|  | ||||
| - | ||||
|   id: 2 | ||||
|   index: 2 | ||||
|   repo_id: 1 | ||||
|   state: "warning" | ||||
|   sha: "1234123412341234123412341234123412341234" | ||||
|   target_url: https://example.com/converage/ | ||||
|   description: My awesome Coverage service | ||||
|   context: cov/awesomeness | ||||
|   creator_id: 2 | ||||
|  | ||||
| - | ||||
|   id: 3 | ||||
|   index: 3 | ||||
|   repo_id: 1 | ||||
|   state: "success" | ||||
|   sha: "1234123412341234123412341234123412341234" | ||||
|   target_url: https://example.com/converage/ | ||||
|   description: My awesome Coverage service | ||||
|   context: cov/awesomeness | ||||
|   creator_id: 2 | ||||
|  | ||||
| - | ||||
|   id: 4 | ||||
|   index: 4 | ||||
|   repo_id: 1 | ||||
|   state: "failure" | ||||
|   sha: "1234123412341234123412341234123412341234" | ||||
|   target_url: https://example.com/builds/ | ||||
|   description: My awesome CI-service | ||||
|   context: ci/awesomeness | ||||
|   creator_id: 2 | ||||
|  | ||||
| - | ||||
|   id: 5 | ||||
|   index: 5 | ||||
|   repo_id: 1 | ||||
|   state: "error" | ||||
|   sha: "1234123412341234123412341234123412341234" | ||||
|   target_url: https://example.com/builds/ | ||||
|   description: My awesome deploy service | ||||
|   context: deploy/awesomeness | ||||
|   creator_id: 2 | ||||
| @ -106,6 +106,8 @@ var migrations = []Migration{ | ||||
| 	NewMigration("change mirror interval from hours to time.Duration", convertIntervalToDuration), | ||||
| 	// v28 -> v29 | ||||
| 	NewMigration("add field for repo size", addRepoSize), | ||||
| 	// v29 -> v30 | ||||
| 	NewMigration("add commit status table", addCommitStatus), | ||||
| } | ||||
|  | ||||
| // Migrate database to current version | ||||
|  | ||||
							
								
								
									
										34
									
								
								models/migrations/v29.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								models/migrations/v29.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | ||||
| // Copyright 2017 The Gogs Authors. All rights reserved. | ||||
| // Use of this source code is governed by a MIT-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package migrations | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"github.com/go-xorm/xorm" | ||||
| ) | ||||
|  | ||||
| // CommitStatus see models/status.go | ||||
| type CommitStatus struct { | ||||
| 	ID          int64  `xorm:"pk autoincr"` | ||||
| 	Index       int64  `xorm:"INDEX UNIQUE(repo_sha_index)"` | ||||
| 	RepoID      int64  `xorm:"INDEX UNIQUE(repo_sha_index)"` | ||||
| 	State       string `xorm:"VARCHAR(7) NOT NULL"` | ||||
| 	SHA         string `xorm:"VARCHAR(64) NOT NULL INDEX UNIQUE(repo_sha_index)"` | ||||
| 	TargetURL   string `xorm:"TEXT"` | ||||
| 	Description string `xorm:"TEXT"` | ||||
| 	Context     string `xorm:"TEXT"` | ||||
| 	CreatorID   int64  `xorm:"INDEX"` | ||||
|  | ||||
| 	CreatedUnix int64 `xorm:"INDEX"` | ||||
| 	UpdatedUnix int64 `xorm:"INDEX"` | ||||
| } | ||||
|  | ||||
| func addCommitStatus(x *xorm.Engine) error { | ||||
| 	if err := x.Sync2(new(CommitStatus)); err != nil { | ||||
| 		return fmt.Errorf("Sync2: %v", err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @ -118,6 +118,7 @@ func init() { | ||||
| 		new(ProtectedBranch), | ||||
| 		new(UserOpenID), | ||||
| 		new(IssueWatch), | ||||
| 		new(CommitStatus), | ||||
| 	) | ||||
|  | ||||
| 	gonicNames := []string{"SSL", "UID"} | ||||
|  | ||||
							
								
								
									
										265
									
								
								models/status.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										265
									
								
								models/status.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,265 @@ | ||||
| // Copyright 2017 Gitea. All rights reserved. | ||||
| // Use of this source code is governed by a MIT-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package models | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"code.gitea.io/git" | ||||
| 	"code.gitea.io/gitea/modules/log" | ||||
| 	"code.gitea.io/gitea/modules/setting" | ||||
| 	api "code.gitea.io/sdk/gitea" | ||||
|  | ||||
| 	"github.com/go-xorm/xorm" | ||||
| ) | ||||
|  | ||||
| // CommitStatusState holds the state of a Status | ||||
| // It can be "pending", "success", "error", "failure", and "warning" | ||||
| type CommitStatusState string | ||||
|  | ||||
| // IsWorseThan returns true if this State is worse than the given State | ||||
| func (css CommitStatusState) IsWorseThan(css2 CommitStatusState) bool { | ||||
| 	switch css { | ||||
| 	case CommitStatusError: | ||||
| 		return true | ||||
| 	case CommitStatusFailure: | ||||
| 		return css2 != CommitStatusError | ||||
| 	case CommitStatusWarning: | ||||
| 		return css2 != CommitStatusError && css2 != CommitStatusFailure | ||||
| 	case CommitStatusSuccess: | ||||
| 		return css2 != CommitStatusError && css2 != CommitStatusFailure && css2 != CommitStatusWarning | ||||
| 	default: | ||||
| 		return css2 != CommitStatusError && css2 != CommitStatusFailure && css2 != CommitStatusWarning && css2 != CommitStatusSuccess | ||||
| 	} | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	// CommitStatusPending is for when the Status is Pending | ||||
| 	CommitStatusPending CommitStatusState = "pending" | ||||
| 	// CommitStatusSuccess is for when the Status is Success | ||||
| 	CommitStatusSuccess CommitStatusState = "success" | ||||
| 	// CommitStatusError is for when the Status is Error | ||||
| 	CommitStatusError CommitStatusState = "error" | ||||
| 	// CommitStatusFailure is for when the Status is Failure | ||||
| 	CommitStatusFailure CommitStatusState = "failure" | ||||
| 	// CommitStatusWarning is for when the Status is Warning | ||||
| 	CommitStatusWarning CommitStatusState = "warning" | ||||
| ) | ||||
|  | ||||
| // CommitStatus holds a single Status of a single Commit | ||||
| type CommitStatus struct { | ||||
| 	ID          int64             `xorm:"pk autoincr"` | ||||
| 	Index       int64             `xorm:"INDEX UNIQUE(repo_sha_index)"` | ||||
| 	RepoID      int64             `xorm:"INDEX UNIQUE(repo_sha_index)"` | ||||
| 	Repo        *Repository       `xorm:"-"` | ||||
| 	State       CommitStatusState `xorm:"VARCHAR(7) NOT NULL"` | ||||
| 	SHA         string            `xorm:"VARCHAR(64) NOT NULL INDEX UNIQUE(repo_sha_index)"` | ||||
| 	TargetURL   string            `xorm:"TEXT"` | ||||
| 	Description string            `xorm:"TEXT"` | ||||
| 	Context     string            `xorm:"TEXT"` | ||||
| 	Creator     *User             `xorm:"-"` | ||||
| 	CreatorID   int64 | ||||
|  | ||||
| 	Created     time.Time `xorm:"-"` | ||||
| 	CreatedUnix int64     `xorm:"INDEX"` | ||||
| 	Updated     time.Time `xorm:"-"` | ||||
| 	UpdatedUnix int64     `xorm:"INDEX"` | ||||
| } | ||||
|  | ||||
| // BeforeInsert is invoked from XORM before inserting an object of this type. | ||||
| func (status *CommitStatus) BeforeInsert() { | ||||
| 	status.CreatedUnix = time.Now().Unix() | ||||
| 	status.UpdatedUnix = status.CreatedUnix | ||||
| } | ||||
|  | ||||
| // BeforeUpdate is invoked from XORM before updating this object. | ||||
| func (status *CommitStatus) BeforeUpdate() { | ||||
| 	status.UpdatedUnix = time.Now().Unix() | ||||
| } | ||||
|  | ||||
| // AfterSet is invoked from XORM after setting the value of a field of | ||||
| // this object. | ||||
| func (status *CommitStatus) AfterSet(colName string, _ xorm.Cell) { | ||||
| 	switch colName { | ||||
| 	case "created_unix": | ||||
| 		status.Created = time.Unix(status.CreatedUnix, 0).Local() | ||||
| 	case "updated_unix": | ||||
| 		status.Updated = time.Unix(status.UpdatedUnix, 0).Local() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (status *CommitStatus) loadRepo(e Engine) (err error) { | ||||
| 	if status.Repo == nil { | ||||
| 		status.Repo, err = getRepositoryByID(e, status.RepoID) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("getRepositoryByID [%d]: %v", status.RepoID, err) | ||||
| 		} | ||||
| 	} | ||||
| 	if status.Creator == nil && status.CreatorID > 0 { | ||||
| 		status.Creator, err = getUserByID(e, status.CreatorID) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("getUserByID [%d]: %v", status.CreatorID, err) | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // APIURL returns the absolute APIURL to this commit-status. | ||||
| func (status *CommitStatus) APIURL() string { | ||||
| 	status.loadRepo(x) | ||||
| 	return fmt.Sprintf("%sapi/v1/%s/statuses/%s", | ||||
| 		setting.AppURL, status.Repo.FullName(), status.SHA) | ||||
| } | ||||
|  | ||||
| // APIFormat assumes some fields assigned with values: | ||||
| // Required - Repo, Creator | ||||
| func (status *CommitStatus) APIFormat() *api.Status { | ||||
| 	status.loadRepo(x) | ||||
| 	apiStatus := &api.Status{ | ||||
| 		Created:     status.Created, | ||||
| 		Updated:     status.Created, | ||||
| 		State:       api.StatusState(status.State), | ||||
| 		TargetURL:   status.TargetURL, | ||||
| 		Description: status.Description, | ||||
| 		ID:          status.Index, | ||||
| 		URL:         status.APIURL(), | ||||
| 		Context:     status.Context, | ||||
| 	} | ||||
| 	if status.Creator != nil { | ||||
| 		apiStatus.Creator = status.Creator.APIFormat() | ||||
| 	} | ||||
|  | ||||
| 	return apiStatus | ||||
| } | ||||
|  | ||||
| // GetCommitStatuses returns all statuses for a given commit. | ||||
| func GetCommitStatuses(repo *Repository, sha string, page int) ([]*CommitStatus, error) { | ||||
| 	statuses := make([]*CommitStatus, 0, 10) | ||||
| 	sess := x.NewSession() | ||||
| 	defer sess.Close() | ||||
| 	return statuses, sess.Limit(10, page*10).Where("repo_id = ?", repo.ID).And("sha = ?", sha).Find(&statuses) | ||||
| } | ||||
|  | ||||
| // GetLatestCommitStatus returns all statuses with a unique context for a given commit. | ||||
| func GetLatestCommitStatus(repo *Repository, sha string, page int) ([]*CommitStatus, error) { | ||||
| 	statuses := make([]*CommitStatus, 0, 10) | ||||
| 	sess := x.NewSession() | ||||
| 	defer sess.Close() | ||||
|  | ||||
| 	return statuses, sess.Limit(10, page*10). | ||||
| 		Where("repo_id = ?", repo.ID).And("sha = ?", sha).Select("*"). | ||||
| 		GroupBy("context").Desc("created_unix").Find(&statuses) | ||||
| } | ||||
|  | ||||
| // GetCommitStatus populates a given status for a given commit. | ||||
| // NOTE: If ID or Index isn't given, and only Context, TargetURL and/or Description | ||||
| //       is given, the CommitStatus created _last_ will be returned. | ||||
| func GetCommitStatus(repo *Repository, sha string, status *CommitStatus) (*CommitStatus, error) { | ||||
| 	conds := &CommitStatus{ | ||||
| 		Context:     status.Context, | ||||
| 		State:       status.State, | ||||
| 		TargetURL:   status.TargetURL, | ||||
| 		Description: status.Description, | ||||
| 	} | ||||
| 	has, err := x.Where("repo_id = ?", repo.ID).And("sha = ?", sha).Desc("created_unix").Get(conds) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("GetCommitStatus[%s, %s]: %v", repo.RepoPath(), sha, err) | ||||
| 	} | ||||
| 	if !has { | ||||
| 		return nil, fmt.Errorf("GetCommitStatus[%s, %s]: not found", repo.RepoPath(), sha) | ||||
| 	} | ||||
|  | ||||
| 	return conds, nil | ||||
| } | ||||
|  | ||||
| // NewCommitStatusOptions holds options for creating a CommitStatus | ||||
| type NewCommitStatusOptions struct { | ||||
| 	Repo         *Repository | ||||
| 	Creator      *User | ||||
| 	SHA          string | ||||
| 	CommitStatus *CommitStatus | ||||
| } | ||||
|  | ||||
| func newCommitStatus(e Engine, opts NewCommitStatusOptions) error { | ||||
| 	opts.CommitStatus.Description = strings.TrimSpace(opts.CommitStatus.Description) | ||||
| 	opts.CommitStatus.Context = strings.TrimSpace(opts.CommitStatus.Context) | ||||
| 	opts.CommitStatus.TargetURL = strings.TrimSpace(opts.CommitStatus.TargetURL) | ||||
| 	opts.CommitStatus.SHA = opts.SHA | ||||
| 	opts.CommitStatus.CreatorID = opts.Creator.ID | ||||
|  | ||||
| 	if opts.Repo == nil { | ||||
| 		return fmt.Errorf("newCommitStatus[nil, %s]: no repository specified", opts.SHA) | ||||
| 	} | ||||
| 	opts.CommitStatus.RepoID = opts.Repo.ID | ||||
|  | ||||
| 	if opts.Creator == nil { | ||||
| 		return fmt.Errorf("newCommitStatus[%s, %s]: no user specified", opts.Repo.RepoPath(), opts.SHA) | ||||
| 	} | ||||
|  | ||||
| 	gitRepo, err := git.OpenRepository(opts.Repo.RepoPath()) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("OpenRepository[%s]: %v", opts.Repo.RepoPath(), err) | ||||
| 	} | ||||
| 	if _, err := gitRepo.GetCommit(opts.SHA); err != nil { | ||||
| 		return fmt.Errorf("GetCommit[%s]: %v", opts.SHA, err) | ||||
| 	} | ||||
|  | ||||
| 	sess := x.NewSession() | ||||
| 	defer sess.Close() | ||||
| 	if err = sess.Begin(); err != nil { | ||||
| 		return fmt.Errorf("newCommitStatus[%s, %s]: %v", opts.Repo.RepoPath(), opts.SHA, err) | ||||
| 	} | ||||
|  | ||||
| 	// Get the next Status Index | ||||
| 	var nextIndex int64 | ||||
| 	lastCommitStatus := &CommitStatus{ | ||||
| 		SHA:    opts.SHA, | ||||
| 		RepoID: opts.Repo.ID, | ||||
| 	} | ||||
| 	has, err := sess.Desc("index").Limit(1).Get(lastCommitStatus) | ||||
| 	if err != nil { | ||||
| 		sess.Rollback() | ||||
| 		return fmt.Errorf("newCommitStatus[%s, %s]: %v", opts.Repo.RepoPath(), opts.SHA, err) | ||||
| 	} | ||||
| 	if has { | ||||
| 		log.Debug("newCommitStatus[%s, %s]: found", opts.Repo.RepoPath(), opts.SHA) | ||||
| 		nextIndex = lastCommitStatus.Index | ||||
| 	} | ||||
| 	opts.CommitStatus.Index = nextIndex + 1 | ||||
| 	log.Debug("newCommitStatus[%s, %s]: %d", opts.Repo.RepoPath(), opts.SHA, opts.CommitStatus.Index) | ||||
|  | ||||
| 	// Insert new CommitStatus | ||||
| 	if _, err = sess.Insert(opts.CommitStatus); err != nil { | ||||
| 		sess.Rollback() | ||||
| 		return fmt.Errorf("newCommitStatus[%s, %s]: %v", opts.Repo.RepoPath(), opts.SHA, err) | ||||
| 	} | ||||
|  | ||||
| 	return sess.Commit() | ||||
| } | ||||
|  | ||||
| // NewCommitStatus creates a new CommitStatus given a bunch of parameters | ||||
| // NOTE: All text-values will be trimmed from whitespaces. | ||||
| // Requires: Repo, Creator, SHA | ||||
| func NewCommitStatus(repo *Repository, creator *User, sha string, status *CommitStatus) error { | ||||
| 	sess := x.NewSession() | ||||
| 	defer sess.Close() | ||||
|  | ||||
| 	if err := sess.Begin(); err != nil { | ||||
| 		return fmt.Errorf("NewCommitStatus[repo_id: %d, user_id: %d, sha: %s]: %v", repo.ID, creator.ID, sha, err) | ||||
| 	} | ||||
|  | ||||
| 	if err := newCommitStatus(sess, NewCommitStatusOptions{ | ||||
| 		Repo:         repo, | ||||
| 		Creator:      creator, | ||||
| 		SHA:          sha, | ||||
| 		CommitStatus: status, | ||||
| 	}); err != nil { | ||||
| 		return fmt.Errorf("NewCommitStatus[repo_id: %d, user_id: %d, sha: %s]: %v", repo.ID, creator.ID, sha, err) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										39
									
								
								models/status_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								models/status_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | ||||
| // Copyright 2017 Gitea. All rights reserved. | ||||
| // Use of this source code is governed by a MIT-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package models | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
|  | ||||
| func TestGetCommitStatuses(t *testing.T) { | ||||
| 	assert.NoError(t, PrepareTestDatabase()) | ||||
|  | ||||
| 	repo1 := AssertExistsAndLoadBean(t, &Repository{ID: 1}).(*Repository) | ||||
|  | ||||
| 	sha1 := "1234123412341234123412341234123412341234" | ||||
|  | ||||
| 	statuses, err := GetCommitStatuses(repo1, sha1, 0) | ||||
| 	assert.NoError(t, err) | ||||
| 	if assert.Equal(t, 5, len(statuses), "Expected to get 5 statuses") { | ||||
|  | ||||
| 		assert.Equal(t, statuses[0].Context, "ci/awesomeness") | ||||
| 		assert.Equal(t, statuses[0].State, CommitStatusPending) | ||||
|  | ||||
| 		assert.Equal(t, statuses[1].Context, "cov/awesomeness") | ||||
| 		assert.Equal(t, statuses[1].State, CommitStatusWarning) | ||||
|  | ||||
| 		assert.Equal(t, statuses[2].Context, "cov/awesomeness") | ||||
| 		assert.Equal(t, statuses[2].State, CommitStatusSuccess) | ||||
|  | ||||
| 		assert.Equal(t, statuses[3].Context, "ci/awesomeness") | ||||
| 		assert.Equal(t, statuses[3].State, CommitStatusFailure) | ||||
|  | ||||
| 		assert.Equal(t, statuses[4].Context, "deploy/awesomeness") | ||||
| 		assert.Equal(t, statuses[4].State, CommitStatusError) | ||||
| 	} | ||||
| } | ||||
| @ -412,6 +412,13 @@ func RegisterRoutes(m *macaron.Macaron) { | ||||
| 					}) | ||||
|  | ||||
| 				}, mustAllowPulls, context.ReferencesGitRepo()) | ||||
| 				m.Group("/statuses", func() { | ||||
| 					m.Combo("/:sha").Get(repo.GetCommitStatuses).Post(reqRepoWriter(), bind(api.CreateStatusOption{}), repo.NewCommitStatus) | ||||
| 				}) | ||||
| 				m.Group("/commits/:ref", func() { | ||||
| 					m.Get("/status", repo.GetCombinedCommitStatus) | ||||
| 					m.Get("/statuses", repo.GetCommitStatuses) | ||||
| 				}) | ||||
| 			}, repoAssignment()) | ||||
| 		}, reqToken()) | ||||
|  | ||||
|  | ||||
							
								
								
									
										127
									
								
								routers/api/v1/repo/status.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								routers/api/v1/repo/status.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,127 @@ | ||||
| // Copyright 2017 Gitea. All rights reserved. | ||||
| // Use of this source code is governed by a MIT-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package repo | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"code.gitea.io/gitea/models" | ||||
| 	"code.gitea.io/gitea/modules/context" | ||||
| 	api "code.gitea.io/sdk/gitea" | ||||
| ) | ||||
|  | ||||
| // NewCommitStatus creates a new CommitStatus | ||||
| func NewCommitStatus(ctx *context.APIContext, form api.CreateStatusOption) { | ||||
| 	sha := ctx.Params("sha") | ||||
| 	if len(sha) == 0 { | ||||
| 		sha = ctx.Params("ref") | ||||
| 	} | ||||
| 	if len(sha) == 0 { | ||||
| 		ctx.Error(400, "ref/sha not given", nil) | ||||
| 		return | ||||
| 	} | ||||
| 	status := &models.CommitStatus{ | ||||
| 		State:       models.CommitStatusState(form.State), | ||||
| 		TargetURL:   form.TargetURL, | ||||
| 		Description: form.Description, | ||||
| 		Context:     form.Context, | ||||
| 	} | ||||
| 	if err := models.NewCommitStatus(ctx.Repo.Repository, ctx.User, sha, status); err != nil { | ||||
| 		ctx.Error(500, "NewCommitStatus", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	newStatus, err := models.GetCommitStatus(ctx.Repo.Repository, sha, status) | ||||
| 	if err != nil { | ||||
| 		ctx.Error(500, "GetCommitStatus", err) | ||||
| 		return | ||||
| 	} | ||||
| 	ctx.JSON(201, newStatus.APIFormat()) | ||||
| } | ||||
|  | ||||
| // GetCommitStatuses returns all statuses for any given commit hash | ||||
| func GetCommitStatuses(ctx *context.APIContext) { | ||||
| 	sha := ctx.Params("sha") | ||||
| 	if len(sha) == 0 { | ||||
| 		sha = ctx.Params("ref") | ||||
| 	} | ||||
| 	if len(sha) == 0 { | ||||
| 		ctx.Error(400, "ref/sha not given", nil) | ||||
| 		return | ||||
| 	} | ||||
| 	repo := ctx.Repo.Repository | ||||
|  | ||||
| 	page := ctx.ParamsInt("page") | ||||
|  | ||||
| 	statuses, err := models.GetCommitStatuses(repo, sha, page) | ||||
| 	if err != nil { | ||||
| 		ctx.Error(500, "GetCommitStatuses", fmt.Errorf("GetCommitStatuses[%s, %s, %d]: %v", repo.FullName(), sha, page, err)) | ||||
| 	} | ||||
|  | ||||
| 	apiStatuses := make([]*api.Status, 0, len(statuses)) | ||||
| 	for _, status := range statuses { | ||||
| 		apiStatuses = append(apiStatuses, status.APIFormat()) | ||||
| 	} | ||||
|  | ||||
| 	ctx.JSON(200, apiStatuses) | ||||
| } | ||||
|  | ||||
| type combinedCommitStatus struct { | ||||
| 	State      models.CommitStatusState `json:"state"` | ||||
| 	SHA        string                   `json:"sha"` | ||||
| 	TotalCount int                      `json:"total_count"` | ||||
| 	Statuses   []*api.Status            `json:"statuses"` | ||||
| 	Repo       *api.Repository          `json:"repository"` | ||||
| 	CommitURL  string                   `json:"commit_url"` | ||||
| 	URL        string                   `json:"url"` | ||||
| } | ||||
|  | ||||
| // GetCombinedCommitStatus returns the combined status for any given commit hash | ||||
| func GetCombinedCommitStatus(ctx *context.APIContext) { | ||||
| 	sha := ctx.Params("sha") | ||||
| 	if len(sha) == 0 { | ||||
| 		sha = ctx.Params("ref") | ||||
| 	} | ||||
| 	if len(sha) == 0 { | ||||
| 		ctx.Error(400, "ref/sha not given", nil) | ||||
| 		return | ||||
| 	} | ||||
| 	repo := ctx.Repo.Repository | ||||
|  | ||||
| 	page := ctx.ParamsInt("page") | ||||
|  | ||||
| 	statuses, err := models.GetLatestCommitStatus(repo, sha, page) | ||||
| 	if err != nil { | ||||
| 		ctx.Error(500, "GetLatestCommitStatus", fmt.Errorf("GetLatestCommitStatus[%s, %s, %d]: %v", repo.FullName(), sha, page, err)) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if len(statuses) == 0 { | ||||
| 		ctx.Status(200) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	acl, err := models.AccessLevel(ctx.User.ID, repo) | ||||
| 	if err != nil { | ||||
| 		ctx.Error(500, "AccessLevel", fmt.Errorf("AccessLevel[%d, %s]: %v", ctx.User.ID, repo.FullName(), err)) | ||||
| 		return | ||||
| 	} | ||||
| 	retStatus := &combinedCommitStatus{ | ||||
| 		SHA:        sha, | ||||
| 		TotalCount: len(statuses), | ||||
| 		Repo:       repo.APIFormat(acl), | ||||
| 		URL:        "", | ||||
| 	} | ||||
|  | ||||
| 	retStatus.Statuses = make([]*api.Status, 0, len(statuses)) | ||||
| 	for _, status := range statuses { | ||||
| 		retStatus.Statuses = append(retStatus.Statuses, status.APIFormat()) | ||||
| 		if status.State.IsWorseThan(retStatus.State) { | ||||
| 			retStatus.State = status.State | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	ctx.JSON(200, retStatus) | ||||
| } | ||||
		Reference in New Issue
	
	Block a user