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
"errors"
"fmt"
2014-04-14 06:20:28 +04:00
"strings"
2014-03-13 09:16:14 +04:00
"time"
2014-03-22 14:20:00 +04:00
2014-04-14 06:20:28 +04:00
"github.com/gogits/git"
2014-05-03 09:37:49 +04:00
qlog "github.com/qiniu/log"
2014-04-14 05:00:12 +04:00
2014-03-23 14:27:01 +04:00
"github.com/gogits/gogs/modules/base"
2014-05-06 19:50:31 +04:00
"github.com/gogits/gogs/modules/hooks"
2014-03-22 14:20:00 +04:00
"github.com/gogits/gogs/modules/log"
2014-03-13 09:16:14 +04:00
)
// Operation types of user action.
const (
OP_CREATE_REPO = iota + 1
OP_DELETE_REPO
OP_STAR_REPO
OP_FOLLOW_REPO
OP_COMMIT_REPO
2014-03-25 22:04:57 +04:00
OP_CREATE_ISSUE
2014-03-13 09:16:14 +04:00
OP_PULL_REQUEST
2014-04-05 02:31:09 +04:00
OP_TRANSFER_REPO
2014-04-14 05:00:12 +04:00
OP_PUSH_TAG
2014-03-13 09:16:14 +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 {
2014-05-03 09:37:49 +04:00
Id int64
UserId int64 // Receiver user id.
OpType int // Operations: CREATE DELETE STAR ...
ActUserId int64 // Action user id.
ActUserName string // Action user name.
ActEmail string
RepoId int64
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 08:50:51 +04:00
}
func ( a Action ) GetOpType ( ) int {
return a . OpType
}
func ( a Action ) GetActUserName ( ) string {
return a . ActUserName
}
2014-03-29 15:50:25 +04:00
func ( a Action ) GetActEmail ( ) string {
return a . ActEmail
}
2014-03-15 08:50:51 +04:00
func ( a Action ) GetRepoName ( ) string {
return a . RepoName
2014-03-13 09:16:14 +04:00
}
2014-03-23 14:27:01 +04:00
func ( a Action ) GetBranch ( ) string {
return a . RefName
2014-03-16 19:30:35 +04:00
}
2014-03-23 14:27:01 +04:00
func ( a Action ) GetContent ( ) string {
return a . Content
2014-03-23 14:00:09 +04:00
}
2014-03-27 19:37:33 +04:00
// CommitRepoAction adds new action for committing repository.
2014-05-03 09:37:49 +04:00
func CommitRepoAction ( userId , repoUserId int64 , userName , actEmail string ,
2014-05-06 19:50:31 +04:00
repoId int64 , repoUserName , repoName string , refFullName string , commit * base . PushCommits ) error {
2014-04-14 05:00:12 +04:00
// log.Trace("action.CommitRepoAction(start): %d/%s", userId, repoName)
opType := OP_COMMIT_REPO
// Check it's tag push or branch.
2014-05-06 19:50:31 +04:00
if strings . HasPrefix ( refFullName , "refs/tags/" ) {
2014-04-14 06:20:28 +04:00
opType = OP_PUSH_TAG
commit = & base . PushCommits { }
}
2014-05-06 19:50:31 +04:00
refName := git . RefEndName ( refFullName )
2014-03-24 17:32:24 +04:00
2014-03-27 20:48:29 +04:00
bs , err := json . Marshal ( commit )
2014-03-16 19:02:59 +04:00
if err != nil {
2014-05-06 19:50:31 +04:00
return errors . New ( "action.CommitRepoAction(json): " + err . Error ( ) )
2014-03-16 19:02:59 +04:00
}
2014-03-20 07:39:00 +04:00
2014-03-27 19:37:33 +04:00
// Change repository bare status and update last updated time.
2014-05-03 09:37:49 +04:00
repo , err := GetRepositoryByName ( repoUserId , repoName )
2014-03-22 12:44:57 +04:00
if err != nil {
2014-05-06 19:50:31 +04:00
return errors . New ( "action.CommitRepoAction(GetRepositoryByName): " + err . Error ( ) )
2014-03-22 12:44:57 +04:00
}
2014-03-22 19:59:14 +04:00
repo . IsBare = false
2014-03-22 12:44:57 +04:00
if err = UpdateRepository ( repo ) ; err != nil {
2014-05-06 19:50:31 +04:00
return errors . New ( "action.CommitRepoAction(UpdateRepository): " + err . Error ( ) )
2014-03-22 12:44:57 +04:00
}
2014-03-25 16:53:11 +04:00
2014-05-01 17:00:30 +04:00
if err = NotifyWatchers ( & Action { ActUserId : userId , ActUserName : userName , ActEmail : actEmail ,
2014-05-03 09:37:49 +04:00
OpType : opType , Content : string ( bs ) , RepoId : repoId , RepoUserName : repoUserName ,
RepoName : repoName , RefName : refName ,
2014-05-02 04:29:51 +04:00
IsPrivate : repo . IsPrivate } ) ; err != nil {
2014-05-06 19:50:31 +04:00
return errors . New ( "action.CommitRepoAction(NotifyWatchers): " + err . Error ( ) )
2014-05-01 16:35:05 +04:00
2014-05-06 19:50:31 +04:00
}
2014-05-03 09:37:49 +04:00
qlog . Info ( "action.CommitRepoAction(end): %d/%s" , repoUserId , repoName )
2014-05-06 19:50:31 +04:00
// New push event hook.
ws , err := GetActiveWebhooksByRepoId ( repoId )
if err != nil {
return errors . New ( "action.CommitRepoAction(GetWebhooksByRepoId): " + err . Error ( ) )
} else if len ( ws ) == 0 {
return nil
}
commits := make ( [ ] * hooks . PayloadCommit , len ( commit . Commits ) )
for i , cmt := range commit . Commits {
commits [ i ] = & hooks . PayloadCommit {
Id : cmt . Sha1 ,
Message : cmt . Message ,
Url : fmt . Sprintf ( "%s%s/%s/commit/%s" , base . AppUrl , repoUserName , repoName , cmt . Sha1 ) ,
Author : & hooks . PayloadAuthor {
Name : cmt . AuthorName ,
Email : cmt . AuthorEmail ,
} ,
}
}
p := & hooks . Payload {
Ref : refFullName ,
Commits : commits ,
Pusher : & hooks . PayloadAuthor {
Name : userName ,
Email : actEmail ,
} ,
}
for _ , w := range ws {
w . GetEvent ( )
if ! w . HasPushEvent ( ) {
continue
}
p . Secret = w . Secret
hooks . AddHookTask ( & hooks . HookTask { hooks . HTT_WEBHOOK , w . Url , p , w . ContentType , w . IsSsl } )
}
2014-03-20 07:39:00 +04:00
return nil
2014-03-16 08:18:34 +04:00
}
2014-03-27 20:48:29 +04:00
// NewRepoAction adds new action for creating repository.
func NewRepoAction ( user * User , repo * Repository ) ( err error ) {
2014-03-29 15:50:25 +04:00
if err = NotifyWatchers ( & Action { ActUserId : user . Id , ActUserName : user . Name , ActEmail : user . Email ,
2014-05-02 04:29:51 +04:00
OpType : OP_CREATE_REPO , RepoId : repo . Id , RepoName : repo . Name , IsPrivate : repo . IsPrivate } ) ; err != nil {
2014-03-27 20:48:29 +04:00
log . Error ( "action.NewRepoAction(notify watchers): %d/%s" , user . Id , repo . Name )
return err
}
2014-03-22 14:20:00 +04:00
log . Trace ( "action.NewRepoAction: %s/%s" , user . LowerName , repo . LowerName )
2014-03-13 09:16:14 +04:00
return err
}
2014-04-05 02:31:09 +04:00
// TransferRepoAction adds new action for transfering repository.
func TransferRepoAction ( user , newUser * User , repo * Repository ) ( err error ) {
if err = NotifyWatchers ( & Action { ActUserId : user . Id , ActUserName : user . Name , ActEmail : user . Email ,
2014-05-02 04:29:51 +04:00
OpType : OP_TRANSFER_REPO , RepoId : repo . Id , RepoName : repo . Name , Content : newUser . Name ,
IsPrivate : repo . IsPrivate } ) ; err != nil {
2014-04-05 02:31:09 +04:00
log . Error ( "action.TransferRepoAction(notify watchers): %d/%s" , user . Id , repo . Name )
return err
}
log . Trace ( "action.TransferRepoAction: %s/%s" , user . LowerName , repo . LowerName )
return err
}
2014-03-15 13:30:59 +04:00
// GetFeeds returns action list of given user in given context.
2014-03-15 08:50:51 +04:00
func GetFeeds ( userid , offset int64 , isProfile bool ) ( [ ] Action , error ) {
2014-03-13 09:16:14 +04:00
actions := make ( [ ] Action , 0 , 20 )
2014-03-15 08:50:51 +04:00
sess := orm . Limit ( 20 , int ( offset ) ) . Desc ( "id" ) . Where ( "user_id=?" , userid )
if isProfile {
2014-05-01 17:00:30 +04:00
sess . Where ( "is_private=?" , false ) . And ( "act_user_id=?" , userid )
2014-03-15 08:50:51 +04:00
} else {
sess . And ( "act_user_id!=?" , userid )
}
err := sess . Find ( & actions )
2014-03-13 09:16:14 +04:00
return actions , err
}