2017-09-12 09:48:13 +03:00
// Copyright 2017 The Gitea 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 (
"time"
2017-12-14 02:45:31 +03:00
"code.gitea.io/gitea/modules/setting"
2017-11-01 05:25:14 +03:00
api "code.gitea.io/sdk/gitea"
2017-09-12 09:48:13 +03:00
"github.com/go-xorm/builder"
)
// TrackedTime represents a time that was spent for a specific issue.
type TrackedTime struct {
ID int64 ` xorm:"pk autoincr" json:"id" `
IssueID int64 ` xorm:"INDEX" json:"issue_id" `
UserID int64 ` xorm:"INDEX" json:"user_id" `
Created time . Time ` xorm:"-" json:"created" `
2017-10-01 19:52:35 +03:00
CreatedUnix int64 ` xorm:"created" json:"-" `
2017-09-12 09:48:13 +03:00
Time int64 ` json:"time" `
}
2017-10-01 19:52:35 +03:00
// AfterLoad is invoked from XORM after setting the values of all fields of this object.
func ( t * TrackedTime ) AfterLoad ( ) {
2017-12-14 02:45:31 +03:00
t . Created = time . Unix ( t . CreatedUnix , 0 ) . In ( setting . UILocation )
2017-09-12 09:48:13 +03:00
}
2017-11-01 05:25:14 +03:00
// APIFormat converts TrackedTime to API format
func ( t * TrackedTime ) APIFormat ( ) * api . TrackedTime {
return & api . TrackedTime {
ID : t . ID ,
IssueID : t . IssueID ,
UserID : t . UserID ,
Time : t . Time ,
Created : t . Created ,
}
}
2017-09-12 09:48:13 +03:00
// FindTrackedTimesOptions represent the filters for tracked times. If an ID is 0 it will be ignored.
type FindTrackedTimesOptions struct {
IssueID int64
UserID int64
RepositoryID int64
}
// ToCond will convert each condition into a xorm-Cond
func ( opts * FindTrackedTimesOptions ) ToCond ( ) builder . Cond {
cond := builder . NewCond ( )
if opts . IssueID != 0 {
cond = cond . And ( builder . Eq { "issue_id" : opts . IssueID } )
}
if opts . UserID != 0 {
cond = cond . And ( builder . Eq { "user_id" : opts . UserID } )
}
if opts . RepositoryID != 0 {
cond = cond . And ( builder . Eq { "issue.repo_id" : opts . RepositoryID } )
}
return cond
}
// GetTrackedTimes returns all tracked times that fit to the given options.
func GetTrackedTimes ( options FindTrackedTimesOptions ) ( trackedTimes [ ] * TrackedTime , err error ) {
if options . RepositoryID > 0 {
err = x . Join ( "INNER" , "issue" , "issue.id = tracked_time.issue_id" ) . Where ( options . ToCond ( ) ) . Find ( & trackedTimes )
return
}
err = x . Where ( options . ToCond ( ) ) . Find ( & trackedTimes )
return
}
// AddTime will add the given time (in seconds) to the issue
func AddTime ( user * User , issue * Issue , time int64 ) ( * TrackedTime , error ) {
tt := & TrackedTime {
IssueID : issue . ID ,
UserID : user . ID ,
Time : time ,
}
if _ , err := x . Insert ( tt ) ; err != nil {
return nil , err
}
if _ , err := CreateComment ( & CreateCommentOptions {
Issue : issue ,
Repo : issue . Repo ,
Doer : user ,
Content : secToTime ( time ) ,
Type : CommentTypeAddTimeManual ,
} ) ; err != nil {
return nil , err
}
return tt , nil
}
// TotalTimes returns the spent time for each user by an issue
func TotalTimes ( options FindTrackedTimesOptions ) ( map [ * User ] string , error ) {
trackedTimes , err := GetTrackedTimes ( options )
if err != nil {
return nil , err
}
//Adding total time per user ID
totalTimesByUser := make ( map [ int64 ] int64 )
for _ , t := range trackedTimes {
totalTimesByUser [ t . UserID ] += t . Time
}
totalTimes := make ( map [ * User ] string )
//Fetching User and making time human readable
for userID , total := range totalTimesByUser {
user , err := GetUserByID ( userID )
if err != nil {
if IsErrUserNotExist ( err ) {
continue
}
return nil , err
}
totalTimes [ user ] = secToTime ( total )
}
return totalTimes , nil
}