You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

issue_tracked_time.go 7.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. // Copyright 2017 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package models
  5. import (
  6. "time"
  7. "code.gitea.io/gitea/modules/setting"
  8. "xorm.io/builder"
  9. "xorm.io/xorm"
  10. )
  11. // TrackedTime represents a time that was spent for a specific issue.
  12. type TrackedTime struct {
  13. ID int64 `xorm:"pk autoincr"`
  14. IssueID int64 `xorm:"INDEX"`
  15. Issue *Issue `xorm:"-"`
  16. UserID int64 `xorm:"INDEX"`
  17. User *User `xorm:"-"`
  18. Created time.Time `xorm:"-"`
  19. CreatedUnix int64 `xorm:"created"`
  20. Time int64 `xorm:"NOT NULL"`
  21. Deleted bool `xorm:"NOT NULL DEFAULT false"`
  22. }
  23. // TrackedTimeList is a List of TrackedTime's
  24. type TrackedTimeList []*TrackedTime
  25. // AfterLoad is invoked from XORM after setting the values of all fields of this object.
  26. func (t *TrackedTime) AfterLoad() {
  27. t.Created = time.Unix(t.CreatedUnix, 0).In(setting.DefaultUILocation)
  28. }
  29. // LoadAttributes load Issue, User
  30. func (t *TrackedTime) LoadAttributes() (err error) {
  31. return t.loadAttributes(x)
  32. }
  33. func (t *TrackedTime) loadAttributes(e Engine) (err error) {
  34. if t.Issue == nil {
  35. t.Issue, err = getIssueByID(e, t.IssueID)
  36. if err != nil {
  37. return
  38. }
  39. err = t.Issue.loadRepo(e)
  40. if err != nil {
  41. return
  42. }
  43. }
  44. if t.User == nil {
  45. t.User, err = getUserByID(e, t.UserID)
  46. if err != nil {
  47. return
  48. }
  49. }
  50. return
  51. }
  52. // LoadAttributes load Issue, User
  53. func (tl TrackedTimeList) LoadAttributes() (err error) {
  54. for _, t := range tl {
  55. if err = t.LoadAttributes(); err != nil {
  56. return err
  57. }
  58. }
  59. return
  60. }
  61. // FindTrackedTimesOptions represent the filters for tracked times. If an ID is 0 it will be ignored.
  62. type FindTrackedTimesOptions struct {
  63. IssueID int64
  64. UserID int64
  65. RepositoryID int64
  66. MilestoneID int64
  67. CreatedAfterUnix int64
  68. CreatedBeforeUnix int64
  69. }
  70. // ToCond will convert each condition into a xorm-Cond
  71. func (opts *FindTrackedTimesOptions) ToCond() builder.Cond {
  72. cond := builder.NewCond().And(builder.Eq{"tracked_time.deleted": false})
  73. if opts.IssueID != 0 {
  74. cond = cond.And(builder.Eq{"issue_id": opts.IssueID})
  75. }
  76. if opts.UserID != 0 {
  77. cond = cond.And(builder.Eq{"user_id": opts.UserID})
  78. }
  79. if opts.RepositoryID != 0 {
  80. cond = cond.And(builder.Eq{"issue.repo_id": opts.RepositoryID})
  81. }
  82. if opts.MilestoneID != 0 {
  83. cond = cond.And(builder.Eq{"issue.milestone_id": opts.MilestoneID})
  84. }
  85. if opts.CreatedAfterUnix != 0 {
  86. cond = cond.And(builder.Gte{"tracked_time.created_unix": opts.CreatedAfterUnix})
  87. }
  88. if opts.CreatedBeforeUnix != 0 {
  89. cond = cond.And(builder.Lte{"tracked_time.created_unix": opts.CreatedBeforeUnix})
  90. }
  91. return cond
  92. }
  93. // ToSession will convert the given options to a xorm Session by using the conditions from ToCond and joining with issue table if required
  94. func (opts *FindTrackedTimesOptions) ToSession(e Engine) *xorm.Session {
  95. if opts.RepositoryID > 0 || opts.MilestoneID > 0 {
  96. return e.Join("INNER", "issue", "issue.id = tracked_time.issue_id").Where(opts.ToCond())
  97. }
  98. return e.Where(opts.ToCond())
  99. }
  100. func getTrackedTimes(e Engine, options FindTrackedTimesOptions) (trackedTimes TrackedTimeList, err error) {
  101. err = options.ToSession(e).Find(&trackedTimes)
  102. return
  103. }
  104. // GetTrackedTimes returns all tracked times that fit to the given options.
  105. func GetTrackedTimes(opts FindTrackedTimesOptions) (TrackedTimeList, error) {
  106. return getTrackedTimes(x, opts)
  107. }
  108. func getTrackedSeconds(e Engine, opts FindTrackedTimesOptions) (trackedSeconds int64, err error) {
  109. return opts.ToSession(e).SumInt(&TrackedTime{}, "time")
  110. }
  111. // GetTrackedSeconds return sum of seconds
  112. func GetTrackedSeconds(opts FindTrackedTimesOptions) (int64, error) {
  113. return getTrackedSeconds(x, opts)
  114. }
  115. // AddTime will add the given time (in seconds) to the issue
  116. func AddTime(user *User, issue *Issue, amount int64, created time.Time) (*TrackedTime, error) {
  117. sess := x.NewSession()
  118. defer sess.Close()
  119. if err := sess.Begin(); err != nil {
  120. return nil, err
  121. }
  122. t, err := addTime(sess, user, issue, amount, created)
  123. if err != nil {
  124. return nil, err
  125. }
  126. if err := issue.loadRepo(sess); err != nil {
  127. return nil, err
  128. }
  129. if _, err := createComment(sess, &CreateCommentOptions{
  130. Issue: issue,
  131. Repo: issue.Repo,
  132. Doer: user,
  133. Content: SecToTime(amount),
  134. Type: CommentTypeAddTimeManual,
  135. }); err != nil {
  136. return nil, err
  137. }
  138. return t, sess.Commit()
  139. }
  140. func addTime(e Engine, user *User, issue *Issue, amount int64, created time.Time) (*TrackedTime, error) {
  141. if created.IsZero() {
  142. created = time.Now()
  143. }
  144. tt := &TrackedTime{
  145. IssueID: issue.ID,
  146. UserID: user.ID,
  147. Time: amount,
  148. Created: created,
  149. }
  150. if _, err := e.Insert(tt); err != nil {
  151. return nil, err
  152. }
  153. return tt, nil
  154. }
  155. // TotalTimes returns the spent time for each user by an issue
  156. func TotalTimes(options FindTrackedTimesOptions) (map[*User]string, error) {
  157. trackedTimes, err := GetTrackedTimes(options)
  158. if err != nil {
  159. return nil, err
  160. }
  161. //Adding total time per user ID
  162. totalTimesByUser := make(map[int64]int64)
  163. for _, t := range trackedTimes {
  164. totalTimesByUser[t.UserID] += t.Time
  165. }
  166. totalTimes := make(map[*User]string)
  167. //Fetching User and making time human readable
  168. for userID, total := range totalTimesByUser {
  169. user, err := GetUserByID(userID)
  170. if err != nil {
  171. if IsErrUserNotExist(err) {
  172. continue
  173. }
  174. return nil, err
  175. }
  176. totalTimes[user] = SecToTime(total)
  177. }
  178. return totalTimes, nil
  179. }
  180. // DeleteIssueUserTimes deletes times for issue
  181. func DeleteIssueUserTimes(issue *Issue, user *User) error {
  182. sess := x.NewSession()
  183. defer sess.Close()
  184. if err := sess.Begin(); err != nil {
  185. return err
  186. }
  187. opts := FindTrackedTimesOptions{
  188. IssueID: issue.ID,
  189. UserID: user.ID,
  190. }
  191. removedTime, err := deleteTimes(sess, opts)
  192. if err != nil {
  193. return err
  194. }
  195. if removedTime == 0 {
  196. return ErrNotExist{}
  197. }
  198. if err := issue.loadRepo(sess); err != nil {
  199. return err
  200. }
  201. if _, err := createComment(sess, &CreateCommentOptions{
  202. Issue: issue,
  203. Repo: issue.Repo,
  204. Doer: user,
  205. Content: "- " + SecToTime(removedTime),
  206. Type: CommentTypeDeleteTimeManual,
  207. }); err != nil {
  208. return err
  209. }
  210. return sess.Commit()
  211. }
  212. // DeleteTime delete a specific Time
  213. func DeleteTime(t *TrackedTime) error {
  214. sess := x.NewSession()
  215. defer sess.Close()
  216. if err := sess.Begin(); err != nil {
  217. return err
  218. }
  219. if err := deleteTime(sess, t); err != nil {
  220. return err
  221. }
  222. if _, err := createComment(sess, &CreateCommentOptions{
  223. Issue: t.Issue,
  224. Repo: t.Issue.Repo,
  225. Doer: t.User,
  226. Content: "- " + SecToTime(t.Time),
  227. Type: CommentTypeDeleteTimeManual,
  228. }); err != nil {
  229. return err
  230. }
  231. return sess.Commit()
  232. }
  233. func deleteTimes(e Engine, opts FindTrackedTimesOptions) (removedTime int64, err error) {
  234. removedTime, err = getTrackedSeconds(e, opts)
  235. if err != nil || removedTime == 0 {
  236. return
  237. }
  238. _, err = opts.ToSession(e).Table("tracked_time").Cols("deleted").Update(&TrackedTime{Deleted: true})
  239. return
  240. }
  241. func deleteTime(e Engine, t *TrackedTime) error {
  242. if t.Deleted {
  243. return ErrNotExist{ID: t.ID}
  244. }
  245. t.Deleted = true
  246. _, err := e.ID(t.ID).Cols("deleted").Update(t)
  247. return err
  248. }
  249. // GetTrackedTimeByID returns raw TrackedTime without loading attributes by id
  250. func GetTrackedTimeByID(id int64) (*TrackedTime, error) {
  251. time := &TrackedTime{
  252. ID: id,
  253. }
  254. has, err := x.Get(time)
  255. if err != nil {
  256. return nil, err
  257. } else if !has {
  258. return nil, ErrNotExist{ID: id}
  259. }
  260. return time, nil
  261. }