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 (
2021-11-19 16:39:57 +03:00
"context"
2017-09-12 09:48:13 +03:00
"fmt"
"time"
2017-12-11 07:37:04 +03:00
2021-09-19 14:49:59 +03:00
"code.gitea.io/gitea/models/db"
2021-11-24 12:49:20 +03:00
user_model "code.gitea.io/gitea/models/user"
2019-08-15 17:46:21 +03:00
"code.gitea.io/gitea/modules/timeutil"
2022-02-15 19:50:10 +03:00
"code.gitea.io/gitea/modules/util"
2017-09-12 09:48:13 +03:00
)
2021-11-21 12:11:48 +03:00
// ErrIssueStopwatchNotExist represents an error that stopwatch is not exist
type ErrIssueStopwatchNotExist struct {
UserID int64
IssueID int64
}
func ( err ErrIssueStopwatchNotExist ) Error ( ) string {
return fmt . Sprintf ( "issue stopwatch doesn't exist[uid: %d, issue_id: %d" , err . UserID , err . IssueID )
}
// ErrIssueStopwatchAlreadyExist represents an error that stopwatch is already exist
type ErrIssueStopwatchAlreadyExist struct {
UserID int64
IssueID int64
}
func ( err ErrIssueStopwatchAlreadyExist ) Error ( ) string {
return fmt . Sprintf ( "issue stopwatch already exists[uid: %d, issue_id: %d" , err . UserID , err . IssueID )
}
2017-09-12 09:48:13 +03:00
// Stopwatch represents a stopwatch for time tracking.
type Stopwatch struct {
2019-08-15 17:46:21 +03:00
ID int64 ` xorm:"pk autoincr" `
IssueID int64 ` xorm:"INDEX" `
UserID int64 ` xorm:"INDEX" `
CreatedUnix timeutil . TimeStamp ` xorm:"created" `
2017-09-12 09:48:13 +03:00
}
2021-09-19 14:49:59 +03:00
func init ( ) {
db . RegisterModel ( new ( Stopwatch ) )
}
2021-01-21 17:51:52 +03:00
// Seconds returns the amount of time passed since creation, based on local server time
func ( s Stopwatch ) Seconds ( ) int64 {
return int64 ( timeutil . TimeStampNow ( ) - s . CreatedUnix )
}
// Duration returns a human-readable duration string based on local server time
func ( s Stopwatch ) Duration ( ) string {
2022-02-15 19:50:10 +03:00
return util . SecToTime ( s . Seconds ( ) )
2021-01-21 17:51:52 +03:00
}
2021-11-21 12:11:48 +03:00
func getStopwatch ( ctx context . Context , userID , issueID int64 ) ( sw * Stopwatch , exists bool , err error ) {
2017-09-12 09:48:13 +03:00
sw = new ( Stopwatch )
2021-11-21 12:11:48 +03:00
exists , err = db . GetEngine ( ctx ) .
2017-09-12 09:48:13 +03:00
Where ( "user_id = ?" , userID ) .
And ( "issue_id = ?" , issueID ) .
Get ( sw )
return
}
2022-04-25 23:45:22 +03:00
// UserIDCount is a simple coalition of UserID and Count
type UserStopwatch struct {
UserID int64
StopWatches [ ] * Stopwatch
}
// GetUIDsAndNotificationCounts between the two provided times
func GetUIDsAndStopwatch ( ) ( [ ] * UserStopwatch , error ) {
sws := [ ] * Stopwatch { }
2022-06-02 05:36:46 +03:00
if err := db . GetEngine ( db . DefaultContext ) . Where ( "issue_id != 0" ) . Find ( & sws ) ; err != nil {
2022-04-25 23:45:22 +03:00
return nil , err
}
if len ( sws ) == 0 {
return [ ] * UserStopwatch { } , nil
}
lastUserID := int64 ( - 1 )
res := [ ] * UserStopwatch { }
for _ , sw := range sws {
if lastUserID == sw . UserID {
lastUserStopwatch := res [ len ( res ) - 1 ]
lastUserStopwatch . StopWatches = append ( lastUserStopwatch . StopWatches , sw )
} else {
res = append ( res , & UserStopwatch {
UserID : sw . UserID ,
StopWatches : [ ] * Stopwatch { sw } ,
} )
}
}
return res , nil
}
2019-12-12 07:23:05 +03:00
// GetUserStopwatches return list of all stopwatches of a user
2021-09-24 14:32:56 +03:00
func GetUserStopwatches ( userID int64 , listOptions db . ListOptions ) ( [ ] * Stopwatch , error ) {
2020-09-18 15:09:26 +03:00
sws := make ( [ ] * Stopwatch , 0 , 8 )
2021-09-23 18:45:36 +03:00
sess := db . GetEngine ( db . DefaultContext ) . Where ( "stopwatch.user_id = ?" , userID )
2020-01-24 22:00:29 +03:00
if listOptions . Page != 0 {
2021-09-24 14:32:56 +03:00
sess = db . SetSessionPagination ( sess , & listOptions )
2020-01-24 22:00:29 +03:00
}
2020-09-18 15:09:26 +03:00
err := sess . Find ( & sws )
2019-12-12 07:23:05 +03:00
if err != nil {
return nil , err
}
return sws , nil
}
2021-08-12 15:43:08 +03:00
// CountUserStopwatches return count of all stopwatches of a user
func CountUserStopwatches ( userID int64 ) ( int64 , error ) {
2021-09-23 18:45:36 +03:00
return db . GetEngine ( db . DefaultContext ) . Where ( "user_id = ?" , userID ) . Count ( & Stopwatch { } )
2021-08-12 15:43:08 +03:00
}
2017-09-12 09:48:13 +03:00
// StopwatchExists returns true if the stopwatch exists
2021-03-14 21:52:12 +03:00
func StopwatchExists ( userID , issueID int64 ) bool {
2021-11-21 12:11:48 +03:00
_ , exists , _ := getStopwatch ( db . DefaultContext , userID , issueID )
2017-09-12 09:48:13 +03:00
return exists
}
// HasUserStopwatch returns true if the user has a stopwatch
2022-05-20 17:08:52 +03:00
func HasUserStopwatch ( ctx context . Context , userID int64 ) ( exists bool , sw * Stopwatch , err error ) {
2017-09-12 09:48:13 +03:00
sw = new ( Stopwatch )
2022-05-20 17:08:52 +03:00
exists , err = db . GetEngine ( ctx ) .
2017-09-12 09:48:13 +03:00
Where ( "user_id = ?" , userID ) .
Get ( sw )
return
}
2021-11-21 12:11:48 +03:00
// FinishIssueStopwatchIfPossible if stopwatch exist then finish it otherwise ignore
2021-11-24 12:49:20 +03:00
func FinishIssueStopwatchIfPossible ( ctx context . Context , user * user_model . User , issue * Issue ) error {
2021-11-21 12:11:48 +03:00
_ , exists , err := getStopwatch ( ctx , user . ID , issue . ID )
2021-11-19 16:39:57 +03:00
if err != nil {
2021-07-26 23:46:06 +03:00
return err
}
2021-11-21 12:11:48 +03:00
if ! exists {
return nil
}
return FinishIssueStopwatch ( ctx , user , issue )
}
// CreateOrStopIssueStopwatch create an issue stopwatch if it's not exist, otherwise finish it
2021-11-24 12:49:20 +03:00
func CreateOrStopIssueStopwatch ( user * user_model . User , issue * Issue ) error {
2021-11-21 12:11:48 +03:00
_ , exists , err := getStopwatch ( db . DefaultContext , user . ID , issue . ID )
if err != nil {
2021-07-26 23:46:06 +03:00
return err
}
2021-11-21 12:11:48 +03:00
if exists {
return FinishIssueStopwatch ( db . DefaultContext , user , issue )
}
return CreateIssueStopwatch ( db . DefaultContext , user , issue )
2021-07-26 23:46:06 +03:00
}
2021-11-21 12:11:48 +03:00
// FinishIssueStopwatch if stopwatch exist then finish it otherwise return an error
2021-11-24 12:49:20 +03:00
func FinishIssueStopwatch ( ctx context . Context , user * user_model . User , issue * Issue ) error {
2021-11-21 12:11:48 +03:00
sw , exists , err := getStopwatch ( ctx , user . ID , issue . ID )
2017-09-12 09:48:13 +03:00
if err != nil {
return err
}
2021-11-21 12:11:48 +03:00
if ! exists {
return ErrIssueStopwatchNotExist {
UserID : user . ID ,
IssueID : issue . ID ,
}
}
// Create tracked time out of the time difference between start date and actual date
timediff := time . Now ( ) . Unix ( ) - int64 ( sw . CreatedUnix )
// Create TrackedTime
tt := & TrackedTime {
Created : time . Now ( ) ,
IssueID : issue . ID ,
UserID : user . ID ,
Time : timediff ,
}
if err := db . Insert ( ctx , tt ) ; err != nil {
2018-12-13 18:55:43 +03:00
return err
}
2022-04-08 12:11:15 +03:00
if err := issue . LoadRepo ( ctx ) ; err != nil {
2021-11-21 12:11:48 +03:00
return err
}
2017-09-12 09:48:13 +03:00
2022-04-08 12:11:15 +03:00
if _ , err := CreateCommentCtx ( ctx , & CreateCommentOptions {
2021-11-21 12:11:48 +03:00
Doer : user ,
Issue : issue ,
Repo : issue . Repo ,
2022-02-15 19:50:10 +03:00
Content : util . SecToTime ( timediff ) ,
2021-11-21 12:11:48 +03:00
Type : CommentTypeStopTracking ,
TimeID : tt . ID ,
} ) ; err != nil {
return err
}
2022-05-20 17:08:52 +03:00
_ , err = db . DeleteByBean ( ctx , sw )
2021-11-21 12:11:48 +03:00
return err
}
2017-09-12 09:48:13 +03:00
2021-11-21 12:11:48 +03:00
// CreateIssueStopwatch creates a stopwatch if not exist, otherwise return an error
2021-11-24 12:49:20 +03:00
func CreateIssueStopwatch ( ctx context . Context , user * user_model . User , issue * Issue ) error {
2022-04-08 12:11:15 +03:00
if err := issue . LoadRepo ( ctx ) ; err != nil {
2021-11-21 12:11:48 +03:00
return err
}
2017-09-12 09:48:13 +03:00
2021-11-21 12:11:48 +03:00
// if another stopwatch is running: stop it
2022-05-20 17:08:52 +03:00
exists , sw , err := HasUserStopwatch ( ctx , user . ID )
2021-11-21 12:11:48 +03:00
if err != nil {
return err
}
if exists {
2022-05-20 17:08:52 +03:00
issue , err := getIssueByID ( ctx , sw . IssueID )
2020-07-12 13:01:20 +03:00
if err != nil {
return err
}
2021-11-21 12:11:48 +03:00
if err := FinishIssueStopwatch ( ctx , user , issue ) ; err != nil {
2017-09-12 09:48:13 +03:00
return err
}
2021-11-21 12:11:48 +03:00
}
2017-09-12 09:48:13 +03:00
2021-11-21 12:11:48 +03:00
// Create stopwatch
sw = & Stopwatch {
UserID : user . ID ,
IssueID : issue . ID ,
}
if err := db . Insert ( ctx , sw ) ; err != nil {
return err
2017-09-12 09:48:13 +03:00
}
2021-11-21 12:11:48 +03:00
2022-04-08 12:11:15 +03:00
if err := issue . LoadRepo ( ctx ) ; err != nil {
2021-11-21 12:11:48 +03:00
return err
}
2022-04-08 12:11:15 +03:00
if _ , err := CreateCommentCtx ( ctx , & CreateCommentOptions {
2021-11-21 12:11:48 +03:00
Doer : user ,
Issue : issue ,
Repo : issue . Repo ,
Type : CommentTypeStartTracking ,
} ) ; err != nil {
return err
}
2017-09-12 09:48:13 +03:00
return nil
}
// CancelStopwatch removes the given stopwatch and logs it into issue's timeline.
2021-11-24 12:49:20 +03:00
func CancelStopwatch ( user * user_model . User , issue * Issue ) error {
2021-11-19 16:39:57 +03:00
ctx , committer , err := db . TxContext ( )
if err != nil {
2021-07-26 23:46:06 +03:00
return err
}
2021-11-19 16:39:57 +03:00
defer committer . Close ( )
if err := cancelStopwatch ( ctx , user , issue ) ; err != nil {
2021-07-26 23:46:06 +03:00
return err
}
2021-11-19 16:39:57 +03:00
return committer . Commit ( )
2021-07-26 23:46:06 +03:00
}
2021-11-24 12:49:20 +03:00
func cancelStopwatch ( ctx context . Context , user * user_model . User , issue * Issue ) error {
2021-11-19 16:39:57 +03:00
e := db . GetEngine ( ctx )
2021-11-21 12:11:48 +03:00
sw , exists , err := getStopwatch ( ctx , user . ID , issue . ID )
2017-09-12 09:48:13 +03:00
if err != nil {
return err
}
if exists {
2021-07-26 23:46:06 +03:00
if _ , err := e . Delete ( sw ) ; err != nil {
2017-09-12 09:48:13 +03:00
return err
}
2022-04-08 12:11:15 +03:00
if err := issue . LoadRepo ( ctx ) ; err != nil {
2021-11-21 12:11:48 +03:00
return err
}
2022-04-08 12:11:15 +03:00
if _ , err := CreateCommentCtx ( ctx , & CreateCommentOptions {
2017-09-12 09:48:13 +03:00
Doer : user ,
Issue : issue ,
Repo : issue . Repo ,
Type : CommentTypeCancelTracking ,
} ) ; err != nil {
return err
}
}
return nil
}