2014-03-13 09:16:14 +04:00
// Copyright 2014 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 models
import (
2014-03-16 19:02:59 +04:00
"encoding/json"
2014-05-06 19:50:31 +04:00
"fmt"
2014-07-26 08:24:27 +04:00
"path"
2014-07-23 15:48:06 +04:00
"regexp"
2014-04-14 06:20:28 +04:00
"strings"
2014-03-13 09:16:14 +04:00
"time"
2014-07-24 12:15:05 +04:00
"unicode"
2014-03-22 14:20:00 +04:00
2015-11-13 00:01:51 +03:00
"github.com/Unknwon/com"
2015-11-13 18:05:50 +03:00
"github.com/go-xorm/xorm"
2015-08-19 19:12:43 +03:00
2016-11-10 19:24:48 +03:00
"code.gitea.io/git"
2016-11-11 12:39:44 +03:00
api "code.gitea.io/sdk/gitea"
2015-08-28 18:36:13 +03:00
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
2014-03-13 09:16:14 +04:00
)
2014-07-26 08:24:27 +04:00
type ActionType int
2014-03-13 09:16:14 +04:00
const (
2016-11-10 18:16:32 +03:00
ActionCreateRepo ActionType = iota + 1 // 1
ActionRenameRepo // 2
ActionStarRepo // 3
ActionWatchRepo // 4
ActionCommitRepo // 5
ActionCreateIssue // 6
2016-11-07 18:37:32 +03:00
ActionCreatePullRequest // 7
2016-11-10 18:16:32 +03:00
ActionTransferRepo // 8
ActionPushTag // 9
ActionCommentIssue // 10
2016-11-07 18:37:32 +03:00
ActionMergePullRequest // 11
2016-11-10 18:16:32 +03:00
ActionCloseIssue // 12
ActionReopenIssue // 13
2016-11-07 18:37:32 +03:00
ActionClosePullRequest // 14
ActionReopenPullRequest // 15
2014-03-13 09:16:14 +04:00
)
2014-07-23 15:48:06 +04:00
var (
// Same as Github. See https://help.github.com/articles/closing-issues-via-commit-messages
2015-02-07 04:47:21 +03:00
IssueCloseKeywords = [ ] string { "close" , "closes" , "closed" , "fix" , "fixes" , "fixed" , "resolve" , "resolves" , "resolved" }
IssueReopenKeywords = [ ] string { "reopen" , "reopens" , "reopened" }
IssueCloseKeywordsPat , IssueReopenKeywordsPat * regexp . Regexp
IssueReferenceKeywordsPat * regexp . Regexp
2014-07-23 15:48:06 +04:00
)
2015-02-07 04:47:21 +03:00
func assembleKeywordsPattern ( words [ ] string ) string {
return fmt . Sprintf ( ` (?i)(?:%s) \S+ ` , strings . Join ( words , "|" ) )
}
2014-07-23 15:48:06 +04:00
func init ( ) {
2015-02-07 04:47:21 +03:00
IssueCloseKeywordsPat = regexp . MustCompile ( assembleKeywordsPattern ( IssueCloseKeywords ) )
IssueReopenKeywordsPat = regexp . MustCompile ( assembleKeywordsPattern ( IssueReopenKeywords ) )
2015-02-07 04:34:49 +03:00
IssueReferenceKeywordsPat = regexp . MustCompile ( ` (?i)(?:)(^| )\S+ ` )
2014-07-23 15:48:06 +04:00
}
2014-03-27 19:37:33 +04:00
// Action represents user operation type and other information to repository.,
// it implemented interface base.Actioner so that can be used in template render.
2014-03-13 09:16:14 +04:00
type Action struct {
2015-03-18 04:51:39 +03:00
ID int64 ` xorm:"pk autoincr" `
UserID int64 // Receiver user id.
2014-07-26 08:24:27 +04:00
OpType ActionType
2015-03-18 04:51:39 +03:00
ActUserID int64 // Action user id.
2014-05-03 09:37:49 +04:00
ActUserName string // Action user name.
2014-11-21 18:58:08 +03:00
ActAvatar string ` xorm:"-" `
2015-03-18 04:51:39 +03:00
RepoID int64
2014-05-03 09:37:49 +04:00
RepoUserName string
RepoName string
RefName string
IsPrivate bool ` xorm:"NOT NULL DEFAULT false" `
Content string ` xorm:"TEXT" `
2016-03-10 03:53:30 +03:00
Created time . Time ` xorm:"-" `
CreatedUnix int64
}
func ( a * Action ) BeforeInsert ( ) {
2016-07-23 15:24:44 +03:00
a . CreatedUnix = time . Now ( ) . Unix ( )
2014-03-15 08:50:51 +04:00
}
2015-08-19 19:12:43 +03:00
func ( a * Action ) AfterSet ( colName string , _ xorm . Cell ) {
switch colName {
2016-03-10 03:53:30 +03:00
case "created_unix" :
a . Created = time . Unix ( a . CreatedUnix , 0 ) . Local ( )
2015-08-19 19:12:43 +03:00
}
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetOpType ( ) int {
2014-07-26 08:24:27 +04:00
return int ( a . OpType )
2014-03-15 08:50:51 +04:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetActUserName ( ) string {
2014-03-15 08:50:51 +04:00
return a . ActUserName
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) ShortActUserName ( ) string {
return base . EllipsisString ( a . ActUserName , 20 )
}
func ( a * Action ) GetRepoUserName ( ) string {
2014-05-09 10:42:50 +04:00
return a . RepoUserName
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) ShortRepoUserName ( ) string {
return base . EllipsisString ( a . RepoUserName , 20 )
}
func ( a * Action ) GetRepoName ( ) string {
2014-03-15 08:50:51 +04:00
return a . RepoName
2014-03-13 09:16:14 +04:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) ShortRepoName ( ) string {
return base . EllipsisString ( a . RepoName , 33 )
}
func ( a * Action ) GetRepoPath ( ) string {
2016-01-15 13:00:39 +03:00
return path . Join ( a . RepoUserName , a . RepoName )
}
func ( a * Action ) ShortRepoPath ( ) string {
2016-01-11 15:41:43 +03:00
return path . Join ( a . ShortRepoUserName ( ) , a . ShortRepoName ( ) )
2015-03-12 23:01:23 +03:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetRepoLink ( ) string {
2015-03-12 23:01:23 +03:00
if len ( setting . AppSubUrl ) > 0 {
return path . Join ( setting . AppSubUrl , a . GetRepoPath ( ) )
}
return "/" + a . GetRepoPath ( )
2014-07-26 08:24:27 +04:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetBranch ( ) string {
2014-03-23 14:27:01 +04:00
return a . RefName
2014-03-16 19:30:35 +04:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetContent ( ) string {
2014-03-23 14:27:01 +04:00
return a . Content
2014-03-23 14:00:09 +04:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetCreate ( ) time . Time {
2014-07-26 08:24:27 +04:00
return a . Created
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetIssueInfos ( ) [ ] string {
2014-07-26 08:24:27 +04:00
return strings . SplitN ( a . Content , "|" , 2 )
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetIssueTitle ( ) string {
2015-11-13 18:05:50 +03:00
index := com . StrTo ( a . GetIssueInfos ( ) [ 0 ] ) . MustInt64 ( )
issue , err := GetIssueByIndex ( a . RepoID , index )
2015-11-13 00:16:51 +03:00
if err != nil {
2015-11-13 18:05:50 +03:00
log . Error ( 4 , "GetIssueByIndex: %v" , err )
2015-11-13 20:11:45 +03:00
return "500 when get issue"
2015-11-13 00:16:51 +03:00
}
2016-08-14 13:32:24 +03:00
return issue . Title
2015-11-12 23:09:48 +03:00
}
2016-01-11 15:41:43 +03:00
func ( a * Action ) GetIssueContent ( ) string {
2015-11-13 20:11:45 +03:00
index := com . StrTo ( a . GetIssueInfos ( ) [ 0 ] ) . MustInt64 ( )
issue , err := GetIssueByIndex ( a . RepoID , index )
if err != nil {
log . Error ( 4 , "GetIssueByIndex: %v" , err )
return "500 when get issue"
}
return issue . Content
}
2015-09-01 16:29:52 +03:00
func newRepoAction ( e Engine , u * User , repo * Repository ) ( err error ) {
if err = notifyWatchers ( e , & Action {
2016-07-23 20:08:22 +03:00
ActUserID : u . ID ,
2015-09-01 16:29:52 +03:00
ActUserName : u . Name ,
2016-11-07 18:37:32 +03:00
OpType : ActionCreateRepo ,
2015-09-01 16:29:52 +03:00
RepoID : repo . ID ,
RepoUserName : repo . Owner . Name ,
RepoName : repo . Name ,
IsPrivate : repo . IsPrivate ,
} ) ; err != nil {
2016-07-23 20:08:22 +03:00
return fmt . Errorf ( "notify watchers '%d/%d': %v" , u . ID , repo . ID , err )
2015-09-01 16:29:52 +03:00
}
log . Trace ( "action.newRepoAction: %s/%s" , u . Name , repo . Name )
return err
}
// NewRepoAction adds new action for creating repository.
func NewRepoAction ( u * User , repo * Repository ) ( err error ) {
return newRepoAction ( x , u , repo )
}
2015-09-01 18:43:53 +03:00
func renameRepoAction ( e Engine , actUser * User , oldRepoName string , repo * Repository ) ( err error ) {
if err = notifyWatchers ( e , & Action {
2016-07-23 20:08:22 +03:00
ActUserID : actUser . ID ,
2015-09-01 18:43:53 +03:00
ActUserName : actUser . Name ,
2016-11-07 18:37:32 +03:00
OpType : ActionRenameRepo ,
2015-09-01 18:43:53 +03:00
RepoID : repo . ID ,
RepoUserName : repo . Owner . Name ,
RepoName : repo . Name ,
IsPrivate : repo . IsPrivate ,
Content : oldRepoName ,
} ) ; err != nil {
return fmt . Errorf ( "notify watchers: %v" , err )
}
log . Trace ( "action.renameRepoAction: %s/%s" , actUser . Name , repo . Name )
return nil
}
// RenameRepoAction adds new action for renaming a repository.
func RenameRepoAction ( actUser * User , oldRepoName string , repo * Repository ) error {
return renameRepoAction ( x , actUser , oldRepoName , repo )
}
2015-09-03 11:34:08 +03:00
func issueIndexTrimRight ( c rune ) bool {
return ! unicode . IsDigit ( c )
}
2015-11-14 01:10:25 +03:00
type PushCommit struct {
2016-08-10 08:01:57 +03:00
Sha1 string
Message string
AuthorEmail string
AuthorName string
CommitterEmail string
CommitterName string
Timestamp time . Time
2015-11-14 01:10:25 +03:00
}
type PushCommits struct {
Len int
Commits [ ] * PushCommit
2016-08-14 14:17:26 +03:00
CompareURL string
2015-11-14 01:10:25 +03:00
avatars map [ string ] string
}
func NewPushCommits ( ) * PushCommits {
return & PushCommits {
avatars : make ( map [ string ] string ) ,
}
}
2015-12-10 04:46:05 +03:00
func ( pc * PushCommits ) ToApiPayloadCommits ( repoLink string ) [ ] * api . PayloadCommit {
commits := make ( [ ] * api . PayloadCommit , len ( pc . Commits ) )
2016-08-10 04:28:06 +03:00
for i , commit := range pc . Commits {
authorUsername := ""
author , err := GetUserByEmail ( commit . AuthorEmail )
2015-12-10 04:46:05 +03:00
if err == nil {
2016-08-10 04:28:06 +03:00
authorUsername = author . Name
2015-12-10 04:46:05 +03:00
}
2016-08-10 08:01:57 +03:00
committerUsername := ""
committer , err := GetUserByEmail ( commit . CommitterEmail )
if err == nil {
// TODO: check errors other than email not found.
committerUsername = committer . Name
}
2015-12-10 04:46:05 +03:00
commits [ i ] = & api . PayloadCommit {
2016-08-10 04:28:06 +03:00
ID : commit . Sha1 ,
Message : commit . Message ,
URL : fmt . Sprintf ( "%s/commit/%s" , repoLink , commit . Sha1 ) ,
2016-08-14 14:17:26 +03:00
Author : & api . PayloadUser {
2016-08-10 04:28:06 +03:00
Name : commit . AuthorName ,
Email : commit . AuthorEmail ,
UserName : authorUsername ,
2015-12-10 04:46:05 +03:00
} ,
2016-08-14 14:17:26 +03:00
Committer : & api . PayloadUser {
2016-08-10 08:01:57 +03:00
Name : commit . CommitterName ,
Email : commit . CommitterEmail ,
UserName : committerUsername ,
} ,
2016-08-10 04:28:06 +03:00
Timestamp : commit . Timestamp ,
2015-12-10 04:46:05 +03:00
}
}
return commits
}
2015-11-14 01:10:25 +03:00
// AvatarLink tries to match user in database with e-mail
// in order to show custom avatar, and falls back to general avatar link.
func ( push * PushCommits ) AvatarLink ( email string ) string {
_ , ok := push . avatars [ email ]
if ! ok {
u , err := GetUserByEmail ( email )
if err != nil {
push . avatars [ email ] = base . AvatarLink ( email )
if ! IsErrUserNotExist ( err ) {
log . Error ( 4 , "GetUserByEmail: %v" , err )
}
} else {
2016-08-05 22:12:54 +03:00
push . avatars [ email ] = u . RelAvatarLink ( )
2015-11-14 01:10:25 +03:00
}
}
return push . avatars [ email ]
}
2016-08-17 09:06:38 +03:00
// UpdateIssuesCommit checks if issues are manipulated by commit message.
func UpdateIssuesCommit ( doer * User , repo * Repository , commits [ ] * PushCommit ) error {
2015-09-26 03:35:56 +03:00
// Commits are appended in the reverse order.
for i := len ( commits ) - 1 ; i >= 0 ; i -- {
c := commits [ i ]
2015-09-03 11:34:08 +03:00
refMarked := make ( map [ int64 ] bool )
2015-02-07 04:47:21 +03:00
for _ , ref := range IssueReferenceKeywordsPat . FindAllString ( c . Message , - 1 ) {
2015-09-03 11:34:08 +03:00
ref = ref [ strings . IndexByte ( ref , byte ( ' ' ) ) + 1 : ]
ref = strings . TrimRightFunc ( ref , issueIndexTrimRight )
2014-07-23 15:48:06 +04:00
if len ( ref ) == 0 {
continue
}
// Add repo name if missing
if ref [ 0 ] == '#' {
2016-08-17 09:06:38 +03:00
ref = fmt . Sprintf ( "%s%s" , repo . FullName ( ) , ref )
2015-09-03 11:34:08 +03:00
} else if ! strings . Contains ( ref , "/" ) {
2015-02-07 04:34:49 +03:00
// FIXME: We don't support User#ID syntax yet
2014-07-23 15:48:06 +04:00
// return ErrNotImplemented
continue
}
issue , err := GetIssueByRef ( ref )
if err != nil {
2015-09-26 04:06:31 +03:00
if IsErrIssueNotExist ( err ) {
continue
}
2014-07-23 15:48:06 +04:00
return err
}
2015-09-03 11:34:08 +03:00
if refMarked [ issue . ID ] {
continue
}
refMarked [ issue . ID ] = true
2016-08-17 09:06:38 +03:00
message := fmt . Sprintf ( ` <a href="%s/commit/%s">%s</a> ` , repo . Link ( ) , c . Sha1 , c . Message )
if err = CreateRefComment ( doer , repo , issue , message , c . Sha1 ) ; err != nil {
2014-07-23 15:48:06 +04:00
return err
}
2015-02-02 18:34:07 +03:00
}
2015-09-03 11:34:08 +03:00
refMarked = make ( map [ int64 ] bool )
2015-09-03 11:44:20 +03:00
// FIXME: can merge this one and next one to a common function.
2015-02-07 04:47:21 +03:00
for _ , ref := range IssueCloseKeywordsPat . FindAllString ( c . Message , - 1 ) {
2015-09-03 11:34:08 +03:00
ref = ref [ strings . IndexByte ( ref , byte ( ' ' ) ) + 1 : ]
ref = strings . TrimRightFunc ( ref , issueIndexTrimRight )
2015-02-02 18:34:07 +03:00
if len ( ref ) == 0 {
continue
}
// Add repo name if missing
if ref [ 0 ] == '#' {
2016-08-17 09:06:38 +03:00
ref = fmt . Sprintf ( "%s%s" , repo . FullName ( ) , ref )
2015-09-03 11:44:20 +03:00
} else if ! strings . Contains ( ref , "/" ) {
2015-02-02 18:34:07 +03:00
// We don't support User#ID syntax yet
// return ErrNotImplemented
continue
}
issue , err := GetIssueByRef ( ref )
if err != nil {
2015-09-26 04:06:31 +03:00
if IsErrIssueNotExist ( err ) {
continue
}
2015-02-02 18:34:07 +03:00
return err
}
2014-07-23 15:48:06 +04:00
2015-09-03 11:34:08 +03:00
if refMarked [ issue . ID ] {
continue
}
refMarked [ issue . ID ] = true
if issue . RepoID != repo . ID || issue . IsClosed {
continue
}
2016-08-17 09:06:38 +03:00
if err = issue . ChangeStatus ( doer , repo , true ) ; err != nil {
2015-09-03 11:34:08 +03:00
return err
2014-07-23 15:48:06 +04:00
}
}
2015-02-07 04:34:49 +03:00
2015-09-03 11:34:08 +03:00
// It is conflict to have close and reopen at same time, so refsMarkd doesn't need to reinit here.
2015-02-07 04:47:21 +03:00
for _ , ref := range IssueReopenKeywordsPat . FindAllString ( c . Message , - 1 ) {
2015-09-03 11:34:08 +03:00
ref = ref [ strings . IndexByte ( ref , byte ( ' ' ) ) + 1 : ]
ref = strings . TrimRightFunc ( ref , issueIndexTrimRight )
2014-07-23 15:48:06 +04:00
2015-02-07 04:47:21 +03:00
if len ( ref ) == 0 {
continue
}
// Add repo name if missing
if ref [ 0 ] == '#' {
2016-08-17 09:06:38 +03:00
ref = fmt . Sprintf ( "%s%s" , repo . FullName ( ) , ref )
2015-09-03 11:44:20 +03:00
} else if ! strings . Contains ( ref , "/" ) {
2015-02-07 04:47:21 +03:00
// We don't support User#ID syntax yet
// return ErrNotImplemented
continue
}
issue , err := GetIssueByRef ( ref )
if err != nil {
2015-09-26 04:06:31 +03:00
if IsErrIssueNotExist ( err ) {
continue
}
2015-02-07 04:47:21 +03:00
return err
}
2015-09-03 11:34:08 +03:00
if refMarked [ issue . ID ] {
continue
}
refMarked [ issue . ID ] = true
if issue . RepoID != repo . ID || ! issue . IsClosed {
continue
}
2016-08-17 09:06:38 +03:00
if err = issue . ChangeStatus ( doer , repo , false ) ; err != nil {
2015-09-03 11:34:08 +03:00
return err
2015-02-07 04:47:21 +03:00
}
}
}
2014-07-23 15:48:06 +04:00
return nil
}
2016-08-17 09:06:38 +03:00
type CommitRepoActionOptions struct {
PusherName string
RepoOwnerID int64
RepoName string
RefFullName string
OldCommitID string
NewCommitID string
Commits * PushCommits
}
2015-08-28 18:36:13 +03:00
2016-08-17 09:06:38 +03:00
// CommitRepoAction adds new commit actio to the repository, and prepare corresponding webhooks.
func CommitRepoAction ( opts CommitRepoActionOptions ) error {
pusher , err := GetUserByName ( opts . PusherName )
2015-08-28 18:36:13 +03:00
if err != nil {
2016-08-17 09:06:38 +03:00
return fmt . Errorf ( "GetUserByName [%s]: %v" , opts . PusherName , err )
2015-08-28 18:36:13 +03:00
}
2016-08-17 09:06:38 +03:00
repo , err := GetRepositoryByName ( opts . RepoOwnerID , opts . RepoName )
2015-08-28 18:36:13 +03:00
if err != nil {
2016-08-17 09:06:38 +03:00
return fmt . Errorf ( "GetRepositoryByName [owner_id: %d, name: %s]: %v" , opts . RepoOwnerID , opts . RepoName , err )
2015-08-28 18:36:13 +03:00
}
2014-04-14 05:00:12 +04:00
2015-11-01 01:59:07 +03:00
// Change repository bare status and update last updated time.
2015-11-06 03:18:59 +03:00
repo . IsBare = false
if err = UpdateRepository ( repo , false ) ; err != nil {
return fmt . Errorf ( "UpdateRepository: %v" , err )
2015-11-01 01:59:07 +03:00
}
2015-08-28 18:36:13 +03:00
isNewBranch := false
2016-11-07 18:37:32 +03:00
opType := ActionCommitRepo
2014-04-14 05:00:12 +04:00
// Check it's tag push or branch.
2016-08-17 09:06:38 +03:00
if strings . HasPrefix ( opts . RefFullName , git . TAG_PREFIX ) {
2016-11-07 18:37:32 +03:00
opType = ActionPushTag
2016-08-17 09:06:38 +03:00
opts . Commits = & PushCommits { }
2015-08-28 18:36:13 +03:00
} else {
2016-08-17 09:06:38 +03:00
// if not the first commit, set the compare URL.
if opts . OldCommitID == git . EMPTY_SHA {
2015-08-28 18:36:13 +03:00
isNewBranch = true
2016-08-17 09:06:38 +03:00
} else {
opts . Commits . CompareURL = repo . ComposeCompareURL ( opts . OldCommitID , opts . NewCommitID )
2015-08-28 18:36:13 +03:00
}
2014-04-14 06:20:28 +04:00
2016-08-17 09:06:38 +03:00
if err = UpdateIssuesCommit ( pusher , repo , opts . Commits . Commits ) ; err != nil {
2015-09-26 04:06:31 +03:00
log . Error ( 4 , "updateIssuesCommit: %v" , err )
2015-08-28 18:36:13 +03:00
}
2014-10-11 05:40:51 +04:00
}
2014-03-24 17:32:24 +04:00
2016-08-17 09:06:38 +03:00
if len ( opts . Commits . Commits ) > setting . UI . FeedMaxCommitNum {
opts . Commits . Commits = opts . Commits . Commits [ : setting . UI . FeedMaxCommitNum ]
2015-09-26 03:35:56 +03:00
}
2016-08-17 09:06:38 +03:00
data , err := json . Marshal ( opts . Commits )
2014-03-16 19:02:59 +04:00
if err != nil {
2015-08-13 11:07:11 +03:00
return fmt . Errorf ( "Marshal: %v" , err )
2014-03-16 19:02:59 +04:00
}
2014-03-20 07:39:00 +04:00
2016-08-17 09:06:38 +03:00
refName := git . RefEndName ( opts . RefFullName )
2015-03-18 04:51:39 +03:00
if err = NotifyWatchers ( & Action {
2016-08-17 09:06:38 +03:00
ActUserID : pusher . ID ,
ActUserName : pusher . Name ,
2015-03-18 04:51:39 +03:00
OpType : opType ,
2016-08-17 09:06:38 +03:00
Content : string ( data ) ,
2015-08-13 11:07:11 +03:00
RepoID : repo . ID ,
2016-08-17 09:06:38 +03:00
RepoUserName : repo . MustOwner ( ) . Name ,
2016-06-27 11:10:12 +03:00
RepoName : repo . Name ,
2015-03-18 04:51:39 +03:00
RefName : refName ,
IsPrivate : repo . IsPrivate ,
} ) ; err != nil {
2015-08-28 18:36:13 +03:00
return fmt . Errorf ( "NotifyWatchers: %v" , err )
2014-09-04 15:17:00 +04:00
}
2016-08-17 09:06:38 +03:00
defer func ( ) {
go HookQueue . Add ( repo . ID )
} ( )
2014-09-17 17:11:51 +04:00
2016-08-17 09:06:38 +03:00
apiPusher := pusher . APIFormat ( )
2016-08-14 14:17:26 +03:00
apiRepo := repo . APIFormat ( nil )
2015-08-28 18:36:13 +03:00
switch opType {
2016-11-07 18:37:32 +03:00
case ActionCommitRepo : // Push
if err = PrepareWebhooks ( repo , HookEventPush , & api . PushPayload {
2016-08-17 09:06:38 +03:00
Ref : opts . RefFullName ,
Before : opts . OldCommitID ,
After : opts . NewCommitID ,
CompareURL : setting . AppUrl + opts . Commits . CompareURL ,
Commits : opts . Commits . ToApiPayloadCommits ( repo . HTMLURL ( ) ) ,
2016-08-14 14:17:26 +03:00
Repo : apiRepo ,
Pusher : apiPusher ,
Sender : apiPusher ,
2016-08-14 13:32:24 +03:00
} ) ; err != nil {
2015-08-28 18:36:13 +03:00
return fmt . Errorf ( "PrepareWebhooks: %v" , err )
2014-05-06 19:50:31 +04:00
}
2015-08-28 18:36:13 +03:00
if isNewBranch {
2016-11-07 18:37:32 +03:00
return PrepareWebhooks ( repo , HookEventCreate , & api . CreatePayload {
2015-08-28 18:36:13 +03:00
Ref : refName ,
RefType : "branch" ,
2016-08-14 14:17:26 +03:00
Repo : apiRepo ,
Sender : apiPusher ,
2015-08-28 18:36:13 +03:00
} )
2015-03-18 11:51:02 +03:00
}
2016-11-07 18:37:32 +03:00
case ActionPushTag : // Create
return PrepareWebhooks ( repo , HookEventCreate , & api . CreatePayload {
2015-08-28 18:36:13 +03:00
Ref : refName ,
RefType : "tag" ,
2016-08-14 14:17:26 +03:00
Repo : apiRepo ,
Sender : apiPusher ,
2015-08-28 18:36:13 +03:00
} )
2014-05-06 19:50:31 +04:00
}
2014-09-10 16:53:16 +04:00
2014-03-20 07:39:00 +04:00
return nil
2014-03-16 08:18:34 +04:00
}
2016-08-17 09:06:38 +03:00
func transferRepoAction ( e Engine , doer , oldOwner * User , repo * Repository ) ( err error ) {
2015-09-01 16:29:52 +03:00
if err = notifyWatchers ( e , & Action {
2016-08-17 09:06:38 +03:00
ActUserID : doer . ID ,
ActUserName : doer . Name ,
2016-11-07 18:37:32 +03:00
OpType : ActionTransferRepo ,
2015-08-08 17:43:14 +03:00
RepoID : repo . ID ,
2016-08-17 09:06:38 +03:00
RepoUserName : repo . Owner . Name ,
2014-09-26 06:36:07 +04:00
RepoName : repo . Name ,
IsPrivate : repo . IsPrivate ,
2016-07-09 18:37:32 +03:00
Content : path . Join ( oldOwner . Name , repo . Name ) ,
2015-09-01 16:29:52 +03:00
} ) ; err != nil {
2016-08-17 09:06:38 +03:00
return fmt . Errorf ( "notifyWatchers: %v" , err )
2014-04-05 02:31:09 +04:00
}
2014-09-26 06:42:31 +04:00
// Remove watch for organization.
2016-08-17 09:06:38 +03:00
if oldOwner . IsOrganization ( ) {
if err = watchRepo ( e , oldOwner . ID , repo . ID , false ) ; err != nil {
return fmt . Errorf ( "watchRepo [false]: %v" , err )
2014-09-26 06:42:31 +04:00
}
}
2015-02-13 08:58:46 +03:00
return nil
}
2016-08-17 09:06:38 +03:00
// TransferRepoAction adds new action for transferring repository,
// the Owner field of repository is assumed to be new owner.
func TransferRepoAction ( doer , oldOwner * User , repo * Repository ) error {
return transferRepoAction ( x , doer , oldOwner , repo )
2014-04-05 02:31:09 +04:00
}
2016-08-17 09:06:38 +03:00
func mergePullRequestAction ( e Engine , doer * User , repo * Repository , issue * Issue ) error {
2015-09-02 23:18:09 +03:00
return notifyWatchers ( e , & Action {
2016-08-17 09:06:38 +03:00
ActUserID : doer . ID ,
ActUserName : doer . Name ,
2016-11-07 18:37:32 +03:00
OpType : ActionMergePullRequest ,
2016-08-17 09:06:38 +03:00
Content : fmt . Sprintf ( "%d|%s" , issue . Index , issue . Title ) ,
2015-09-02 23:18:09 +03:00
RepoID : repo . ID ,
RepoUserName : repo . Owner . Name ,
RepoName : repo . Name ,
IsPrivate : repo . IsPrivate ,
} )
}
// MergePullRequestAction adds new action for merging pull request.
func MergePullRequestAction ( actUser * User , repo * Repository , pull * Issue ) error {
return mergePullRequestAction ( x , actUser , repo , pull )
}
2014-03-15 13:30:59 +04:00
// GetFeeds returns action list of given user in given context.
2016-07-24 09:32:46 +03:00
// actorID is the user who's requesting, ctxUserID is the user/org that is requested.
// actorID can be -1 when isProfile is true or to skip the permission check.
func GetFeeds ( ctxUser * User , actorID , offset int64 , isProfile bool ) ( [ ] * Action , error ) {
2014-05-09 03:17:43 +04:00
actions := make ( [ ] * Action , 0 , 20 )
2016-11-10 18:16:32 +03:00
sess := x .
Limit ( 20 , int ( offset ) ) .
Desc ( "id" ) .
Where ( "user_id = ?" , ctxUser . ID )
2014-03-15 08:50:51 +04:00
if isProfile {
2016-11-10 18:16:32 +03:00
sess .
And ( "is_private = ?" , false ) .
And ( "act_user_id = ?" , ctxUser . ID )
2016-07-24 09:32:46 +03:00
} else if actorID != - 1 && ctxUser . IsOrganization ( ) {
// FIXME: only need to get IDs here, not all fields of repository.
repos , _ , err := ctxUser . GetUserRepositories ( actorID , 1 , ctxUser . NumRepos )
if err != nil {
return nil , fmt . Errorf ( "GetUserRepositories: %v" , err )
2016-02-06 10:52:21 +03:00
}
var repoIDs [ ] int64
2016-07-24 09:32:46 +03:00
for _ , repo := range repos {
2016-02-06 10:52:21 +03:00
repoIDs = append ( repoIDs , repo . ID )
}
if len ( repoIDs ) > 0 {
sess . In ( "repo_id" , repoIDs )
}
2014-03-15 08:50:51 +04:00
}
2016-02-06 10:52:21 +03:00
2014-03-15 08:50:51 +04:00
err := sess . Find ( & actions )
2014-03-13 09:16:14 +04:00
return actions , err
}