leave_log.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. package service
  2. import (
  3. "dy-admin/internal/pcmserver/bus/cabinet_pkg"
  4. "dy-admin/internal/pcmserver/bus/model"
  5. "dy-admin/internal/pcmserver/bus/model/request"
  6. "dy-admin/internal/pcmserver/common"
  7. "dy-admin/internal/pcmserver/global"
  8. "dy-admin/internal/pcmserver/pkg/code"
  9. "dy-admin/internal/pcmserver/pkg/crontab"
  10. systemService "dy-admin/internal/pcmserver/sys/service"
  11. "dy-admin/internal/pcmserver/utils"
  12. "dy-admin/pkg/log"
  13. "fmt"
  14. "go.uber.org/zap"
  15. "gorm.io/gorm"
  16. "strconv"
  17. "time"
  18. "dy-admin/pkg/rescode"
  19. )
  20. type LeaveLogService struct {
  21. }
  22. func (lls *LeaveLogService) initLeaveLogCronTab() {
  23. var list []model.LeaveLog
  24. err := global.DB.Where("status in (?)", []string{model.StatusWaiting, model.StatusExecuting}).Find(&list).Error
  25. if err != nil {
  26. panic(err)
  27. }
  28. for _, leaveLog := range list {
  29. err = parseLeaveCronTab(global.DB, leaveLog)
  30. if err != nil {
  31. panic(err)
  32. }
  33. }
  34. }
  35. func (lls *LeaveLogService) CreateLeaveLog(leaveLog model.LeaveLog) (err error) {
  36. // 查询人员所在的柜子
  37. var staff model.Staff
  38. err = global.DB.Where("id = ?", leaveLog.StaffID).First(&staff).Error
  39. if err != nil {
  40. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  41. }
  42. // 查询柜子当前的状态
  43. memInfo := cabinet_pkg.GetCabinet(staff.CabinetID)
  44. if !memInfo.IsConnect {
  45. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrCabinetNotConnectCode])
  46. }
  47. // 时间转换
  48. startTime, err := utils.ParseMinuteTime(leaveLog.StartTimeStr)
  49. if err != nil {
  50. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveParseTimeCode])
  51. }
  52. endTime, err := utils.ParseMinuteTime(leaveLog.EndTimeStr)
  53. if err != nil {
  54. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveParseTimeCode])
  55. }
  56. if startTime.After(*endTime) {
  57. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveStartGTEndCode])
  58. }
  59. leaveLog.StartTime = startTime
  60. leaveLog.EndTime = endTime
  61. leaveLog.Status = model.StatusWaiting
  62. // 找到人员等待执行或者执行中的请假单
  63. var staffLeaveLogs []model.LeaveLog
  64. err = global.DB.Where("staff_id = ?", leaveLog.StaffID).Where("status in (?)", []string{model.StatusWaiting, model.StatusExecuting}).Find(&staffLeaveLogs).Error
  65. if err != nil {
  66. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  67. }
  68. for _, staffLeaveLog := range staffLeaveLogs {
  69. if staffLeaveLog.EndTime.After(*leaveLog.StartTime) {
  70. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveRepeatCode])
  71. }
  72. }
  73. tx := global.DB.Begin()
  74. defer func() {
  75. if err != nil {
  76. tx.Rollback()
  77. } else {
  78. tx.Commit()
  79. }
  80. }()
  81. err = tx.Create(&leaveLog).Error
  82. if err != nil {
  83. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  84. }
  85. return parseLeaveCronTab(tx, leaveLog)
  86. }
  87. func (lls *LeaveLogService) UpdateLeaveLog(leaveLog model.LeaveLog) (err error) {
  88. // 查询人员所在的柜子
  89. var staff model.Staff
  90. err = global.DB.Where("id = ?", leaveLog.StaffID).First(&staff).Error
  91. if err != nil {
  92. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  93. }
  94. // 查询柜子当前的状态
  95. memInfo := cabinet_pkg.GetCabinet(staff.CabinetID)
  96. if !memInfo.IsConnect {
  97. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrCabinetNotConnectCode])
  98. }
  99. // 时间转换
  100. startTime, err := utils.ParseMinuteTime(leaveLog.StartTimeStr)
  101. if err != nil {
  102. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveParseTimeCode])
  103. }
  104. endTime, err := utils.ParseMinuteTime(leaveLog.EndTimeStr)
  105. if err != nil {
  106. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveParseTimeCode])
  107. }
  108. if startTime.After(*endTime) {
  109. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveStartGTEndCode])
  110. }
  111. leaveLog.StartTime = startTime
  112. leaveLog.EndTime = endTime
  113. leaveLog.Status = model.StatusWaiting
  114. // 找到人员等待执行或者执行中的请假单
  115. //var staffLeaveLogs []model.LeaveLog
  116. //err = global.DB.Where("staff_id = ?", leaveLog.StaffID).Where("status in (?)", []string{model.StatusWaiting, model.StatusExecuting}).Find(&staffLeaveLogs).Error
  117. //if err != nil {
  118. // return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  119. //}
  120. //for _, staffLeaveLog := range staffLeaveLogs {
  121. // if staffLeaveLog.EndTime.After(*leaveLog.StartTime) {
  122. // return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveRepeatCode])
  123. // }
  124. //}
  125. tx := global.DB.Begin()
  126. defer func() {
  127. if err != nil {
  128. tx.Rollback()
  129. } else {
  130. tx.Commit()
  131. }
  132. }()
  133. err = tx.Updates(&leaveLog).Error
  134. if err != nil {
  135. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  136. }
  137. return parseLeaveCronTab(tx, leaveLog)
  138. }
  139. func (lls *LeaveLogService) updateLeaveLogStatus(leaveLogId int, status string) (err error) {
  140. err = global.DB.Model(&model.LeaveLog{}).Where("id = ?", leaveLogId).Update("status", status).Error
  141. if err != nil {
  142. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  143. }
  144. return
  145. }
  146. func (lls *LeaveLogService) DeleteLeaveLogByIds(ids common.IdsReq) (err error) {
  147. var count int64
  148. err = global.DB.Model(&model.LeaveLog{}).Where("status = ?", model.StatusExecuting).Where("id in (?)", ids.IDs).Count(&count).Error
  149. if err != nil {
  150. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  151. }
  152. if count != 0 {
  153. return rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveDeleteExecutingCode])
  154. }
  155. var levelLogs []model.LeaveLog
  156. err = global.DB.Where("status in (?)", []string{model.StatusWaiting, model.StatusFinish}).Where("id in (?)", ids.IDs).Find(&levelLogs).Error
  157. if err != nil {
  158. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  159. }
  160. for _, levelLog := range levelLogs {
  161. leaveJobName := strconv.Itoa(levelLog.ID) + "-" + strconv.Itoa(int(LeaveJob))
  162. resetLeaveJobName := strconv.Itoa(levelLog.ID) + "-" + strconv.Itoa(int(ResetLeaveJob))
  163. err = crontab.GetCron("leave").RemoveJob(leaveJobName)
  164. if err != nil {
  165. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveRemoveCornTabCode])
  166. }
  167. err = crontab.GetCron("leave").RemoveJob(resetLeaveJobName)
  168. if err != nil {
  169. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveRemoveCornTabCode])
  170. }
  171. }
  172. if err = global.DB.Delete(&[]model.LeaveLog{}, "id in (?)", ids.IDs).Error; err != nil {
  173. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  174. }
  175. return
  176. }
  177. func (lls *LeaveLogService) GetLeaveLog(id int) (*model.LeaveLog, error) {
  178. var record model.LeaveLog
  179. err := global.DB.Where("id = ?", id).First(&record).Error
  180. if err != nil {
  181. if err != gorm.ErrRecordNotFound {
  182. return nil, rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  183. } else {
  184. return nil, nil
  185. }
  186. }
  187. return &record, nil
  188. }
  189. func (lls *LeaveLogService) GetLeaveLogList(info request.LeaveLogRecordSearch, dataPermission *BusPermission) (levelLogs []model.LeaveLog, total int64, err error) {
  190. var limit, offset int
  191. if info.Page == 0 || info.PageSize == 0 {
  192. limit = -1
  193. offset = -1
  194. } else {
  195. limit = info.PageSize
  196. offset = info.PageSize * (info.Page - 1)
  197. }
  198. var (
  199. startTime, endTime *time.Time
  200. )
  201. if info.StartTimeStr != "" {
  202. startTime, err = utils.ParseMinuteTime(info.StartTimeStr)
  203. if err != nil {
  204. return levelLogs, total, rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveParseTimeCode])
  205. }
  206. }
  207. if info.EndTimeStr != "" {
  208. endTime, err = utils.ParseMinuteTime(info.EndTimeStr)
  209. if err != nil {
  210. return levelLogs, total, rescode.RegisterErrorWithCause(nil, code.ErrCodeMap[code.ErrLeaveParseTimeCode])
  211. }
  212. }
  213. var leaveModel model.LeaveLog
  214. db := global.DB.Model(&leaveModel).Scopes(Permission(leaveModel.TableName(), dataPermission))
  215. if info.Status != "" {
  216. db = db.Where("status = ?", info.Status)
  217. }
  218. if info.StaffName != "" {
  219. db = db.Where("staff_name LIKE ?", "%"+info.StaffName+"%")
  220. }
  221. if info.Type != "" {
  222. db = db.Where("type = ?", info.Type)
  223. }
  224. if info.DeptID != 0 {
  225. deptIds, err := systemService.ServicesGroupApp.DeptService.GetSonDeptIds(info.DeptID)
  226. if err != nil {
  227. log.Warn("get son deptIds err", zap.Error(err))
  228. } else {
  229. db = db.Where("department_id in (?)", deptIds)
  230. }
  231. }
  232. if startTime != nil && endTime == nil {
  233. db = db.Where("start_time > ?", startTime).Where("end_time < ?", time.Now())
  234. }
  235. if endTime != nil && startTime == nil {
  236. db = db.Where("start_time > ?", time.Now()).Where("end_time < ?", endTime)
  237. }
  238. if startTime != nil && endTime != nil {
  239. db = db.Where("start_time > ?", startTime).Where("end_time < ?", endTime)
  240. }
  241. err = db.Count(&total).Error
  242. if err != nil {
  243. return
  244. }
  245. if err = db.Order("id desc").Limit(limit).Offset(offset).Find(&levelLogs).Error; err != nil {
  246. return levelLogs, total, rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  247. }
  248. return
  249. }
  250. func (lls *LeaveLogService) getLeaveByStaffID(staffId int) (leaveLog *model.LeaveLog, err error) {
  251. var record model.LeaveLog
  252. err = global.DB.Where("staff_id = ?", staffId).Where("status = ?", model.StatusExecuting).First(&record).Error
  253. if err != nil {
  254. if err != gorm.ErrRecordNotFound {
  255. return nil, rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrDatabaseCode])
  256. } else {
  257. return nil, nil
  258. }
  259. }
  260. return &record, nil
  261. }
  262. func parseLeaveCronTab(db *gorm.DB, leaveLog model.LeaveLog) error {
  263. // 删除原来的定时任务
  264. leaveJobName := strconv.Itoa(leaveLog.ID) + "-" + strconv.Itoa(int(LeaveJob))
  265. resetLeaveJobName := strconv.Itoa(leaveLog.ID) + "-" + strconv.Itoa(int(ResetLeaveJob))
  266. err := crontab.GetCron("leave").RemoveJob(leaveJobName)
  267. if err != nil {
  268. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveRemoveCornTabCode])
  269. }
  270. err = crontab.GetCron("leave").RemoveJob(resetLeaveJobName)
  271. if err != nil {
  272. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveRemoveCornTabCode])
  273. }
  274. // 增加两个定时任务 更新人员请假状态。
  275. if leaveLog.StartTime.After(time.Now()) {
  276. if err := setLeaveCornTab(leaveLog.ID, leaveLog.StaffID, leaveLog.StartTime, LeaveJob); err != nil {
  277. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveStartCornTabCode])
  278. }
  279. } else {
  280. // 更行请假单状态
  281. leaveLog.Status = model.StatusExecuting
  282. err := db.Updates(leaveLog).Error
  283. if err != nil {
  284. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveStartCornTabCode])
  285. }
  286. // 更新用户状态
  287. err = ServicesGroupApp.StaffService.updateStaffLeaveStatus(leaveLog.StaffID, LeaveJob)
  288. if err != nil {
  289. log.Error("updateStaffLeaveStatus err", zap.Error(err))
  290. }
  291. }
  292. if leaveLog.EndTime.After(time.Now()) {
  293. // 启动定时任务
  294. if err := setLeaveCornTab(leaveLog.ID, leaveLog.StaffID, leaveLog.EndTime, ResetLeaveJob); err != nil {
  295. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveStartCornTabCode])
  296. }
  297. } else {
  298. leaveLog.Status = model.StatusFinish
  299. err := db.Updates(leaveLog).Error
  300. if err != nil {
  301. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveStartCornTabCode])
  302. }
  303. // 更新人员请假状态
  304. err = ServicesGroupApp.StaffService.updateStaffLeaveStatus(leaveLog.StaffID, ResetLeaveJob)
  305. if err != nil {
  306. log.Error("updateStaffLeaveStatus err", zap.Error(err))
  307. }
  308. }
  309. return nil
  310. }
  311. func setLeaveCornTab(leaveLogId, staffId int, cornTime *time.Time, action uint8) error {
  312. jobName := strconv.Itoa(leaveLogId) + "-" + strconv.Itoa(int(action))
  313. job := &LeaveCronJob{
  314. LeaveLogID: leaveLogId,
  315. JobName: jobName,
  316. StaffID: staffId,
  317. Action: action,
  318. }
  319. secondStr := strconv.Itoa(cornTime.Second())
  320. minuteStr := strconv.Itoa(cornTime.Minute())
  321. hourStr := strconv.Itoa(cornTime.Hour())
  322. month := cornTime.Format("01")
  323. day := strconv.Itoa(cornTime.Day())
  324. spec := fmt.Sprintf("%s %s %s %s %s %s", secondStr, minuteStr, hourStr, day, month, "*")
  325. err := crontab.GetCron("leave").AddJobWithParams(jobName, spec, job)
  326. if err != nil {
  327. return rescode.RegisterErrorWithCause(err, code.ErrCodeMap[code.ErrLeaveCornTabCode])
  328. }
  329. return nil
  330. }