2014-03-13 13:16:14 +08: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 23:02:59 +08:00
"encoding/json"
2014-05-06 11:50:31 -04:00
"errors"
"fmt"
2014-07-26 00:24:27 -04:00
"path"
2014-07-23 13:48:06 +02:00
"regexp"
2014-04-13 22:20:28 -04:00
"strings"
2014-03-13 13:16:14 +08:00
"time"
2014-07-24 10:15:05 +02:00
"unicode"
2014-03-22 06:20:00 -04:00
2015-11-13 00:01:51 +03:00
"github.com/Unknwon/com"
2015-11-13 10:05:50 -05:00
"github.com/go-xorm/xorm"
2015-08-20 00:12:43 +08:00
2015-08-28 23:36:13 +08:00
api "github.com/gogits/go-gogs-client"
2014-03-23 06:27:01 -04:00
"github.com/gogits/gogs/modules/base"
2014-07-26 00:24:27 -04:00
"github.com/gogits/gogs/modules/git"
2014-03-22 06:20:00 -04:00
"github.com/gogits/gogs/modules/log"
2014-05-25 20:11:25 -04:00
"github.com/gogits/gogs/modules/setting"
2014-03-13 13:16:14 +08:00
)
2014-07-26 00:24:27 -04:00
type ActionType int
2014-03-13 13:16:14 +08:00
const (
2015-09-02 16:18:09 -04:00
CREATE_REPO ActionType = iota + 1 // 1
RENAME_REPO // 2
STAR_REPO // 3
FOLLOW_REPO // 4
COMMIT_REPO // 5
CREATE_ISSUE // 6
CREATE_PULL_REQUEST // 7
TRANSFER_REPO // 8
PUSH_TAG // 9
COMMENT_ISSUE // 10
MERGE_PULL_REQUEST // 11
2014-03-13 13:16:14 +08:00
)
2014-07-23 13:48:06 +02:00
var (
ErrNotImplemented = errors . New ( "Not implemented yet" )
)
var (
// Same as Github. See https://help.github.com/articles/closing-issues-via-commit-messages
2015-02-06 20:47:21 -05: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 13:48:06 +02:00
)
2015-02-06 20:47:21 -05:00
func assembleKeywordsPattern ( words [ ] string ) string {
return fmt . Sprintf ( ` (?i)(?:%s) \S+ ` , strings . Join ( words , "|" ) )
}
2014-07-23 13:48:06 +02:00
func init ( ) {
2015-02-06 20:47:21 -05:00
IssueCloseKeywordsPat = regexp . MustCompile ( assembleKeywordsPattern ( IssueCloseKeywords ) )
IssueReopenKeywordsPat = regexp . MustCompile ( assembleKeywordsPattern ( IssueReopenKeywords ) )
2015-02-06 20:34:49 -05:00
IssueReferenceKeywordsPat = regexp . MustCompile ( ` (?i)(?:)(^| )\S+ ` )
2014-07-23 13:48:06 +02:00
}
2014-03-27 11: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 13:16:14 +08:00
type Action struct {
2015-03-17 21:51:39 -04:00
ID int64 ` xorm:"pk autoincr" `
UserID int64 // Receiver user id.
2014-07-26 00:24:27 -04:00
OpType ActionType
2015-03-17 21:51:39 -04:00
ActUserID int64 // Action user id.
2014-05-03 01:37:49 -04:00
ActUserName string // Action user name.
ActEmail string
2014-11-21 10:58:08 -05:00
ActAvatar string ` xorm:"-" `
2015-03-17 21:51:39 -04:00
RepoID int64
2014-05-03 01:37:49 -04:00
RepoUserName string
RepoName string
RefName string
IsPrivate bool ` xorm:"NOT NULL DEFAULT false" `
Content string ` xorm:"TEXT" `
Created time . Time ` xorm:"created" `
2014-03-15 00:50:51 -04:00
}
2015-08-20 00:12:43 +08:00
func ( a * Action ) AfterSet ( colName string , _ xorm . Cell ) {
switch colName {
case "created" :
2015-08-20 00:56:12 +08:00
a . Created = regulateTimeZone ( a . Created )
2015-08-20 00:12:43 +08:00
}
}
2014-03-15 00:50:51 -04:00
func ( a Action ) GetOpType ( ) int {
2014-07-26 00:24:27 -04:00
return int ( a . OpType )
2014-03-15 00:50:51 -04:00
}
func ( a Action ) GetActUserName ( ) string {
return a . ActUserName
}
2014-03-29 07:50:25 -04:00
func ( a Action ) GetActEmail ( ) string {
return a . ActEmail
}
2014-05-09 14:42:50 +08:00
func ( a Action ) GetRepoUserName ( ) string {
return a . RepoUserName
}
2014-03-15 00:50:51 -04:00
func ( a Action ) GetRepoName ( ) string {
return a . RepoName
2014-03-13 13:16:14 +08:00
}
2015-03-12 16:01:23 -04:00
func ( a Action ) GetRepoPath ( ) string {
return path . Join ( a . RepoUserName , a . RepoName )
}
2014-07-26 00:24:27 -04:00
func ( a Action ) GetRepoLink ( ) string {
2015-03-12 16:01:23 -04:00
if len ( setting . AppSubUrl ) > 0 {
return path . Join ( setting . AppSubUrl , a . GetRepoPath ( ) )
}
return "/" + a . GetRepoPath ( )
2014-07-26 00:24:27 -04:00
}
2014-03-23 06:27:01 -04:00
func ( a Action ) GetBranch ( ) string {
return a . RefName
2014-03-16 11:30:35 -04:00
}
2014-03-23 06:27:01 -04:00
func ( a Action ) GetContent ( ) string {
return a . Content
2014-03-23 18:00:09 +08:00
}
2014-07-26 00:24:27 -04:00
func ( a Action ) GetCreate ( ) time . Time {
return a . Created
}
func ( a Action ) GetIssueInfos ( ) [ ] string {
return strings . SplitN ( a . Content , "|" , 2 )
}
2015-11-12 23:09:48 +03:00
func ( a Action ) GetIssueTitle ( ) string {
2015-11-13 10:05:50 -05: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 10:05:50 -05:00
log . Error ( 4 , "GetIssueByIndex: %v" , err )
2015-11-13 12:11:45 -05:00
return "500 when get issue"
2015-11-13 00:16:51 +03:00
}
2015-11-12 23:09:48 +03:00
return issue . Name
}
2015-11-13 12:11:45 -05:00
func ( a Action ) GetIssueContent ( ) string {
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 09:29:52 -04:00
func newRepoAction ( e Engine , u * User , repo * Repository ) ( err error ) {
if err = notifyWatchers ( e , & Action {
ActUserID : u . Id ,
ActUserName : u . Name ,
ActEmail : u . Email ,
OpType : CREATE_REPO ,
RepoID : repo . ID ,
RepoUserName : repo . Owner . Name ,
RepoName : repo . Name ,
IsPrivate : repo . IsPrivate ,
} ) ; err != nil {
2015-11-08 14:31:49 -05:00
return fmt . Errorf ( "notify watchers '%d/%d': %v" , u . Id , repo . ID , err )
2015-09-01 09:29:52 -04: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 11:43:53 -04:00
func renameRepoAction ( e Engine , actUser * User , oldRepoName string , repo * Repository ) ( err error ) {
if err = notifyWatchers ( e , & Action {
ActUserID : actUser . Id ,
ActUserName : actUser . Name ,
ActEmail : actUser . Email ,
OpType : RENAME_REPO ,
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 04:34:08 -04:00
func issueIndexTrimRight ( c rune ) bool {
return ! unicode . IsDigit ( c )
}
2015-11-13 17:10:25 -05:00
type PushCommit struct {
Sha1 string
Message string
AuthorEmail string
AuthorName string
}
type PushCommits struct {
Len int
Commits [ ] * PushCommit
CompareUrl string
avatars map [ string ] string
}
func NewPushCommits ( ) * PushCommits {
return & PushCommits {
avatars : make ( map [ string ] string ) ,
}
}
// 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 {
push . avatars [ email ] = u . AvatarLink ( )
}
}
return push . avatars [ email ]
}
2015-08-27 13:26:38 +08:00
// updateIssuesCommit checks if issues are manipulated by commit message.
2015-11-13 17:10:25 -05:00
func updateIssuesCommit ( u * User , repo * Repository , repoUserName , repoName string , commits [ ] * PushCommit ) error {
2015-09-25 20:35:56 -04:00
// Commits are appended in the reverse order.
for i := len ( commits ) - 1 ; i >= 0 ; i -- {
c := commits [ i ]
2015-09-03 04:34:08 -04:00
refMarked := make ( map [ int64 ] bool )
2015-02-06 20:47:21 -05:00
for _ , ref := range IssueReferenceKeywordsPat . FindAllString ( c . Message , - 1 ) {
2015-09-03 04:34:08 -04:00
ref = ref [ strings . IndexByte ( ref , byte ( ' ' ) ) + 1 : ]
ref = strings . TrimRightFunc ( ref , issueIndexTrimRight )
2014-07-23 13:48:06 +02:00
if len ( ref ) == 0 {
continue
}
// Add repo name if missing
if ref [ 0 ] == '#' {
ref = fmt . Sprintf ( "%s/%s%s" , repoUserName , repoName , ref )
2015-09-03 04:34:08 -04:00
} else if ! strings . Contains ( ref , "/" ) {
2015-02-06 20:34:49 -05:00
// FIXME: We don't support User#ID syntax yet
2014-07-23 13:48:06 +02:00
// return ErrNotImplemented
continue
}
issue , err := GetIssueByRef ( ref )
if err != nil {
2015-09-25 21:06:31 -04:00
if IsErrIssueNotExist ( err ) {
continue
}
2014-07-23 13:48:06 +02:00
return err
}
2015-09-03 04:34:08 -04:00
if refMarked [ issue . ID ] {
continue
}
refMarked [ issue . ID ] = true
2014-09-19 20:11:34 -04:00
url := fmt . Sprintf ( "%s/%s/%s/commit/%s" , setting . AppSubUrl , repoUserName , repoName , c . Sha1 )
2014-07-24 10:15:05 +02:00
message := fmt . Sprintf ( ` <a href="%s">%s</a> ` , url , c . Message )
2015-09-10 07:53:40 -04:00
if err = CreateRefComment ( u , repo , issue , message , c . Sha1 ) ; err != nil {
2014-07-23 13:48:06 +02:00
return err
}
2015-02-02 16:34:07 +01:00
}
2015-09-03 04:34:08 -04:00
refMarked = make ( map [ int64 ] bool )
2015-09-03 04:44:20 -04:00
// FIXME: can merge this one and next one to a common function.
2015-02-06 20:47:21 -05:00
for _ , ref := range IssueCloseKeywordsPat . FindAllString ( c . Message , - 1 ) {
2015-09-03 04:34:08 -04:00
ref = ref [ strings . IndexByte ( ref , byte ( ' ' ) ) + 1 : ]
ref = strings . TrimRightFunc ( ref , issueIndexTrimRight )
2015-02-02 16:34:07 +01:00
if len ( ref ) == 0 {
continue
}
// Add repo name if missing
if ref [ 0 ] == '#' {
ref = fmt . Sprintf ( "%s/%s%s" , repoUserName , repoName , ref )
2015-09-03 04:44:20 -04:00
} else if ! strings . Contains ( ref , "/" ) {
2015-02-02 16:34:07 +01:00
// We don't support User#ID syntax yet
// return ErrNotImplemented
continue
}
issue , err := GetIssueByRef ( ref )
if err != nil {
2015-09-25 21:06:31 -04:00
if IsErrIssueNotExist ( err ) {
continue
}
2015-02-02 16:34:07 +01:00
return err
}
2014-07-23 13:48:06 +02:00
2015-09-03 04:34:08 -04:00
if refMarked [ issue . ID ] {
continue
}
refMarked [ issue . ID ] = true
if issue . RepoID != repo . ID || issue . IsClosed {
continue
}
if err = issue . ChangeStatus ( u , true ) ; err != nil {
return err
2014-07-23 13:48:06 +02:00
}
}
2015-02-06 20:34:49 -05:00
2015-09-03 04:34:08 -04:00
// It is conflict to have close and reopen at same time, so refsMarkd doesn't need to reinit here.
2015-02-06 20:47:21 -05:00
for _ , ref := range IssueReopenKeywordsPat . FindAllString ( c . Message , - 1 ) {
2015-09-03 04:34:08 -04:00
ref = ref [ strings . IndexByte ( ref , byte ( ' ' ) ) + 1 : ]
ref = strings . TrimRightFunc ( ref , issueIndexTrimRight )
2014-07-23 13:48:06 +02:00
2015-02-06 20:47:21 -05:00
if len ( ref ) == 0 {
continue
}
// Add repo name if missing
if ref [ 0 ] == '#' {
ref = fmt . Sprintf ( "%s/%s%s" , repoUserName , repoName , ref )
2015-09-03 04:44:20 -04:00
} else if ! strings . Contains ( ref , "/" ) {
2015-02-06 20:47:21 -05:00
// We don't support User#ID syntax yet
// return ErrNotImplemented
continue
}
issue , err := GetIssueByRef ( ref )
if err != nil {
2015-09-25 21:06:31 -04:00
if IsErrIssueNotExist ( err ) {
continue
}
2015-02-06 20:47:21 -05:00
return err
}
2015-09-03 04:34:08 -04:00
if refMarked [ issue . ID ] {
continue
}
refMarked [ issue . ID ] = true
if issue . RepoID != repo . ID || ! issue . IsClosed {
continue
}
if err = issue . ChangeStatus ( u , false ) ; err != nil {
return err
2015-02-06 20:47:21 -05:00
}
}
}
2014-07-23 13:48:06 +02:00
return nil
}
2014-03-27 11:37:33 -04:00
// CommitRepoAction adds new action for committing repository.
2015-08-28 23:36:13 +08:00
func CommitRepoAction (
userID , repoUserID int64 ,
userName , actEmail string ,
repoID int64 ,
repoUserName , repoName string ,
refFullName string ,
2015-11-13 17:10:25 -05:00
commit * PushCommits ,
2015-08-28 23:36:13 +08:00
oldCommitID string , newCommitID string ) error {
u , err := GetUserByID ( userID )
if err != nil {
return fmt . Errorf ( "GetUserByID: %v" , err )
}
repo , err := GetRepositoryByName ( repoUserID , repoName )
if err != nil {
return fmt . Errorf ( "GetRepositoryByName: %v" , err )
} else if err = repo . GetOwner ( ) ; err != nil {
return fmt . Errorf ( "GetOwner: %v" , err )
}
2014-04-13 21:00:12 -04:00
2015-10-31 18:59:07 -04:00
// Change repository bare status and update last updated time.
2015-11-05 19:18:59 -05:00
repo . IsBare = false
if err = UpdateRepository ( repo , false ) ; err != nil {
return fmt . Errorf ( "UpdateRepository: %v" , err )
2015-10-31 18:59:07 -04:00
}
2015-08-28 23:36:13 +08:00
isNewBranch := false
2014-07-26 00:24:27 -04:00
opType := COMMIT_REPO
2014-04-13 21:00:12 -04:00
// Check it's tag push or branch.
2014-05-06 11:50:31 -04:00
if strings . HasPrefix ( refFullName , "refs/tags/" ) {
2014-07-26 00:24:27 -04:00
opType = PUSH_TAG
2015-11-13 17:10:25 -05:00
commit = & PushCommits { }
2015-08-28 23:36:13 +08:00
} else {
// if not the first commit, set the compareUrl
if ! strings . HasPrefix ( oldCommitID , "0000000" ) {
commit . CompareUrl = fmt . Sprintf ( "%s/%s/compare/%s...%s" , repoUserName , repoName , oldCommitID , newCommitID )
} else {
isNewBranch = true
}
2014-04-13 22:20:28 -04:00
2015-11-15 17:07:44 -05:00
// NOTE: limit to detect latest 100 commits.
if len ( commit . Commits ) > 100 {
commit . Commits = commit . Commits [ len ( commit . Commits ) - 100 : ]
}
2015-08-28 23:36:13 +08:00
if err = updateIssuesCommit ( u , repo , repoUserName , repoName , commit . Commits ) ; err != nil {
2015-09-25 21:06:31 -04:00
log . Error ( 4 , "updateIssuesCommit: %v" , err )
2015-08-28 23:36:13 +08:00
}
2014-10-10 21:40:51 -04:00
}
2014-03-24 09:32:24 -04:00
2015-09-25 20:35:56 -04:00
if len ( commit . Commits ) > setting . FeedMaxCommitNum {
commit . Commits = commit . Commits [ : setting . FeedMaxCommitNum ]
}
2014-03-27 12:48:29 -04:00
bs , err := json . Marshal ( commit )
2014-03-16 23:02:59 +08:00
if err != nil {
2015-08-13 16:07:11 +08:00
return fmt . Errorf ( "Marshal: %v" , err )
2014-03-16 23:02:59 +08:00
}
2014-03-19 23:39:00 -04:00
2014-10-10 21:40:51 -04:00
refName := git . RefEndName ( refFullName )
2015-03-17 21:51:39 -04:00
if err = NotifyWatchers ( & Action {
2015-08-13 16:07:11 +08:00
ActUserID : u . Id ,
2015-03-17 21:51:39 -04:00
ActUserName : userName ,
ActEmail : actEmail ,
OpType : opType ,
Content : string ( bs ) ,
2015-08-13 16:07:11 +08:00
RepoID : repo . ID ,
2015-03-17 21:51:39 -04:00
RepoUserName : repoUserName ,
RepoName : repoName ,
RefName : refName ,
IsPrivate : repo . IsPrivate ,
} ) ; err != nil {
2015-08-28 23:36:13 +08:00
return fmt . Errorf ( "NotifyWatchers: %v" , err )
2014-05-08 08:18:03 -04:00
2014-09-04 07:17:00 -04:00
}
2015-08-28 23:36:13 +08:00
repoLink := fmt . Sprintf ( "%s%s/%s" , setting . AppUrl , repoUserName , repoName )
payloadRepo := & api . PayloadRepo {
ID : repo . ID ,
Name : repo . LowerName ,
URL : repoLink ,
Description : repo . Description ,
Website : repo . Website ,
Watchers : repo . NumWatches ,
Owner : & api . PayloadAuthor {
Name : repo . Owner . DisplayName ( ) ,
Email : repo . Owner . Email ,
UserName : repo . Owner . Name ,
} ,
Private : repo . IsPrivate ,
2014-05-06 11:50:31 -04:00
}
2014-09-17 09:11:51 -04:00
pusher_email , pusher_name := "" , ""
pusher , err := GetUserByName ( userName )
if err == nil {
pusher_email = pusher . Email
2015-08-27 13:26:38 +08:00
pusher_name = pusher . DisplayName ( )
2014-09-17 09:11:51 -04:00
}
2015-08-28 23:36:13 +08:00
payloadSender := & api . PayloadUser {
UserName : pusher . Name ,
ID : pusher . Id ,
AvatarUrl : setting . AppUrl + pusher . RelAvatarLink ( ) ,
}
2014-09-17 09:11:51 -04:00
2015-08-28 23:36:13 +08:00
switch opType {
case COMMIT_REPO : // Push
commits := make ( [ ] * api . PayloadCommit , len ( commit . Commits ) )
for i , cmt := range commit . Commits {
author_username := ""
author , err := GetUserByEmail ( cmt . AuthorEmail )
if err == nil {
author_username = author . Name
}
commits [ i ] = & api . PayloadCommit {
ID : cmt . Sha1 ,
Message : cmt . Message ,
URL : fmt . Sprintf ( "%s/commit/%s" , repoLink , cmt . Sha1 ) ,
Author : & api . PayloadAuthor {
Name : cmt . AuthorName ,
Email : cmt . AuthorEmail ,
UserName : author_username ,
} ,
}
2014-09-17 09:11:51 -04:00
}
2015-08-28 23:36:13 +08:00
p := & api . PushPayload {
Ref : refFullName ,
Before : oldCommitID ,
After : newCommitID ,
CompareUrl : setting . AppUrl + commit . CompareUrl ,
Commits : commits ,
Repo : payloadRepo ,
Pusher : & api . PayloadAuthor {
Name : pusher_name ,
Email : pusher_email ,
UserName : userName ,
2014-05-06 11:50:31 -04:00
} ,
2015-08-28 23:36:13 +08:00
Sender : payloadSender ,
2014-05-06 11:50:31 -04:00
}
2015-08-28 23:36:13 +08:00
if err = PrepareWebhooks ( repo , HOOK_EVENT_PUSH , p ) ; err != nil {
return fmt . Errorf ( "PrepareWebhooks: %v" , err )
2014-05-06 11:50:31 -04:00
}
2015-08-28 23:36:13 +08:00
if isNewBranch {
return PrepareWebhooks ( repo , HOOK_EVENT_CREATE , & api . CreatePayload {
Ref : refName ,
RefType : "branch" ,
Repo : payloadRepo ,
Sender : payloadSender ,
} )
2015-03-18 04:51:02 -04:00
}
2015-08-28 23:36:13 +08:00
case PUSH_TAG : // Create
return PrepareWebhooks ( repo , HOOK_EVENT_CREATE , & api . CreatePayload {
Ref : refName ,
RefType : "tag" ,
Repo : payloadRepo ,
Sender : payloadSender ,
} )
2014-05-06 11:50:31 -04:00
}
2014-09-10 08:53:16 -04:00
2014-03-19 23:39:00 -04:00
return nil
2014-03-16 12:18:34 +08:00
}
2015-02-23 02:15:53 -05:00
func transferRepoAction ( e Engine , actUser , oldOwner , newOwner * User , repo * Repository ) ( err error ) {
2015-09-01 09:29:52 -04:00
if err = notifyWatchers ( e , & Action {
2015-03-17 21:51:39 -04:00
ActUserID : actUser . Id ,
2015-02-23 02:15:53 -05:00
ActUserName : actUser . Name ,
ActEmail : actUser . Email ,
2014-09-25 22:36:07 -04:00
OpType : TRANSFER_REPO ,
2015-08-08 22:43:14 +08:00
RepoID : repo . ID ,
2015-02-23 02:15:53 -05:00
RepoUserName : newOwner . Name ,
2014-09-25 22:36:07 -04:00
RepoName : repo . Name ,
IsPrivate : repo . IsPrivate ,
2015-02-23 02:15:53 -05:00
Content : path . Join ( oldOwner . LowerName , repo . LowerName ) ,
2015-09-01 09:29:52 -04:00
} ) ; err != nil {
2015-11-08 14:31:49 -05:00
return fmt . Errorf ( "notify watchers '%d/%d': %v" , actUser . Id , repo . ID , err )
2014-04-04 18:31:09 -04:00
}
2014-09-25 22:42:31 -04:00
// Remove watch for organization.
if repo . Owner . IsOrganization ( ) {
2015-08-08 22:43:14 +08:00
if err = watchRepo ( e , repo . Owner . Id , repo . ID , false ) ; err != nil {
2015-02-13 00:58:46 -05:00
return fmt . Errorf ( "watch repository: %v" , err )
2014-09-25 22:42:31 -04:00
}
}
2015-09-01 09:29:52 -04:00
log . Trace ( "action.transferRepoAction: %s/%s" , actUser . Name , repo . Name )
2015-02-13 00:58:46 -05:00
return nil
}
// TransferRepoAction adds new action for transferring repository.
2015-09-01 09:29:52 -04:00
func TransferRepoAction ( actUser , oldOwner , newOwner * User , repo * Repository ) error {
2015-02-23 02:15:53 -05:00
return transferRepoAction ( x , actUser , oldOwner , newOwner , repo )
2014-04-04 18:31:09 -04:00
}
2015-09-02 16:18:09 -04:00
func mergePullRequestAction ( e Engine , actUser * User , repo * Repository , pull * Issue ) error {
return notifyWatchers ( e , & Action {
ActUserID : actUser . Id ,
ActUserName : actUser . Name ,
ActEmail : actUser . Email ,
OpType : MERGE_PULL_REQUEST ,
Content : fmt . Sprintf ( "%d|%s" , pull . Index , pull . Name ) ,
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 05:30:59 -04:00
// GetFeeds returns action list of given user in given context.
2014-07-26 23:53:16 -04:00
func GetFeeds ( uid , offset int64 , isProfile bool ) ( [ ] * Action , error ) {
2014-05-08 19:17:43 -04:00
actions := make ( [ ] * Action , 0 , 20 )
2014-07-26 23:53:16 -04:00
sess := x . Limit ( 20 , int ( offset ) ) . Desc ( "id" ) . Where ( "user_id=?" , uid )
2014-03-15 00:50:51 -04:00
if isProfile {
2014-09-29 18:58:04 -04:00
sess . And ( "is_private=?" , false ) . And ( "act_user_id=?" , uid )
2014-03-15 00:50:51 -04:00
}
err := sess . Find ( & actions )
2014-03-13 13:16:14 +08:00
return actions , err
}