seal.go 35 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018
  1. package seal
  2. import (
  3. "errors"
  4. "fmt"
  5. "github.com/beego/beego/v2/client/orm"
  6. "hongze/hz_crm_api/models/company"
  7. "hongze/hz_crm_api/models/contract"
  8. "hongze/hz_crm_api/models/seal"
  9. "hongze/hz_crm_api/models/seal/request"
  10. "hongze/hz_crm_api/models/seal/response"
  11. "hongze/hz_crm_api/models/system"
  12. "hongze/hz_crm_api/services"
  13. "hongze/hz_crm_api/utils"
  14. "mime/multipart"
  15. "reflect"
  16. "strconv"
  17. "strings"
  18. "time"
  19. )
  20. // UploadCheckBackFile 上传签回用印附件(实际操作)
  21. func UploadCheckBackFile(sealId int, fileUrl string, opUser *system.Admin) (sealInfo *seal.Seal, err error) {
  22. //查询当前用印信息
  23. sealInfo, err = seal.GetSealInfoByContractId(sealId)
  24. if err != nil {
  25. return
  26. }
  27. //用印状态判断
  28. if sealInfo.Status != "已审批" && sealInfo.Status != "已签回" {
  29. err = errors.New("用印状态异常,不允许上传签回用印附件,当前用印状态:" + sealInfo.Status)
  30. return
  31. }
  32. sealInfo.CheckBackFileUrl = fileUrl
  33. sealInfo.Status = "已签回"
  34. sealInfo.ModifyTime = time.Now()
  35. sealInfo.CheckBackFileTime = time.Now()
  36. err = sealInfo.Update([]string{"CheckBackFileUrl", "Status", "ModifyTime", "CheckBackFileTime"})
  37. if err != nil {
  38. return
  39. }
  40. // 给销售发送模版消息
  41. sellerInfo, tErr := system.GetSysAdminById(sealInfo.UserId)
  42. if tErr == nil {
  43. if sellerInfo.Mobile != "" {
  44. go services.SendSealFinishedWxTemplateMsg(sellerInfo.Mobile, sealInfo.CompanyName, sealInfo.SealId)
  45. }
  46. }
  47. return
  48. }
  49. // UploadCheckBackFileFromSeal 用印上传签回用印附件
  50. func UploadCheckBackFileFromSeal(sealId int, ext string, fileMulti multipart.File, opUser *system.Admin) (sealInfo *seal.Seal, err error) {
  51. //查询当前用印信息
  52. sealInfo, err = seal.GetSealInfoById(sealId)
  53. if err != nil {
  54. return
  55. }
  56. //用印状态判断
  57. if sealInfo.Status != "已审批" && sealInfo.Status != "已签回" {
  58. err = errors.New("用印状态异常,不允许上传签回用印附件,当前用印状态:" + sealInfo.Status)
  59. return
  60. }
  61. fileName := ``
  62. //保存的文件名
  63. fileName = sealInfo.CompanyName + "_" + sealInfo.Code + "(签回)"
  64. if sealInfo.ContractId > 0 {
  65. //获取合同信息
  66. contractInfo, tmpErr := contract.GetContractById(sealInfo.ContractId)
  67. if tmpErr != nil {
  68. err = tmpErr
  69. return
  70. }
  71. //保存的文件名
  72. fileName = contractInfo.CompanyName + "_" + contractInfo.ContractCode + "(签回)"
  73. }
  74. //非正式环境下,文件名上面还是加上随机数
  75. if utils.RunMode != "release" {
  76. fileName += "_" + utils.GetRandStringNoSpecialChar(10)
  77. }
  78. fileName += ext
  79. //上传到阿里云
  80. resourceUrl, err := services.UploadToOssAndFileName(fileMulti, fileName)
  81. if err != nil {
  82. err = errors.New("文件保存失败,Err:" + err.Error())
  83. return
  84. }
  85. sealInfo.CheckBackFileUrl = resourceUrl
  86. sealInfo.Status = "已签回"
  87. sealInfo.ModifyTime = time.Now()
  88. sealInfo.CheckBackFileTime = time.Now()
  89. err = sealInfo.Update([]string{"CheckBackFileUrl", "Status", "ModifyTime", "CheckBackFileTime"})
  90. if err != nil {
  91. return
  92. }
  93. // 给销售发送模版消息
  94. sellerInfo, tErr := system.GetSysAdminById(sealInfo.UserId)
  95. if tErr == nil {
  96. if sellerInfo.Mobile != "" {
  97. go services.SendSealFinishedWxTemplateMsg(sellerInfo.Mobile, sealInfo.CompanyName, sealInfo.SealId)
  98. }
  99. }
  100. return
  101. }
  102. // ApplySeal 申请用印
  103. func ApplySeal(sysUser *system.Admin, req request.SealApprovalApplyReq) (err error) {
  104. o := orm.NewOrm()
  105. to, err := o.Begin()
  106. if err != nil {
  107. return
  108. }
  109. defer func() {
  110. if err != nil {
  111. _ = to.Rollback()
  112. } else {
  113. _ = to.Commit()
  114. }
  115. }()
  116. // 添加用印
  117. sealInfo, err := addSeal(sysUser.AdminId, req.ContractId, req.FileNum, sysUser.RealName, req.Use, req.CompanyName, req.UseCompanyName, req.CreditCode, req.ServiceType, req.SealType, req.Remark, req.FileUrls, req.AffiliatedCompany)
  118. if err != nil {
  119. return
  120. }
  121. // 添加操作日志
  122. err = seal.AddSealOperationRecord(sealInfo.SealId, sysUser.AdminId, 0, "apply", sysUser.RealName, "提交审批", "")
  123. if err != nil {
  124. return
  125. }
  126. // 用印审批
  127. err = SubmitApproval(sealInfo)
  128. return
  129. }
  130. // addSeal 新增用印
  131. func addSeal(userId, contractId, fileNum int, userName, use, companyName, useCompanyName, creditCode, serviceType, sealType, remark string, fileUrls []string, affiliatedCompany string) (sealInfo *seal.Seal, err error) {
  132. if !strings.Contains(strings.Join(seal.EnumUse, ","), use) {
  133. err = errors.New("用印用途异常")
  134. return
  135. }
  136. if !strings.Contains(strings.Join(seal.EnumServiceType, ","), serviceType) {
  137. err = errors.New("业务类型异常")
  138. return
  139. }
  140. sealCode, err := seal.GetSealCode()
  141. if err != nil {
  142. return
  143. }
  144. fileUrl := ""
  145. now := time.Now()
  146. attachments := make([]*seal.Attachment, 0)
  147. if len(fileUrls) == 1 {
  148. fileUrl = fileUrls[0]
  149. }
  150. sealInfo = &seal.Seal{
  151. Code: sealCode,
  152. UserId: userId,
  153. UserName: userName,
  154. Use: use,
  155. CompanyName: companyName,
  156. UseCompanyName: useCompanyName,
  157. CreditCode: creditCode,
  158. ServiceType: serviceType,
  159. SealType: sealType,
  160. Status: "待提交",
  161. Remark: remark,
  162. FileUrl: fileUrl,
  163. FileNum: fileNum,
  164. ContractId: contractId,
  165. ModifyTime: now,
  166. CreateTime: now,
  167. AffiliatedCompany: affiliatedCompany,
  168. }
  169. err = seal.AddSeal(sealInfo)
  170. if err != nil {
  171. return
  172. }
  173. if len(fileUrls) > 1 {
  174. for _, v := range fileUrls {
  175. tmp := &seal.Attachment{
  176. SealId: sealInfo.SealId,
  177. FileUrl: v,
  178. ModifyTime: now,
  179. CreateTime: now,
  180. }
  181. attachments = append(attachments, tmp)
  182. }
  183. err = seal.AddAttachments(attachments)
  184. }
  185. return
  186. }
  187. // EditApply 编辑用印审批
  188. func EditApply(sysUser *system.Admin, req request.SealApprovalEditReq) (err error) {
  189. o := orm.NewOrm()
  190. to, err := o.Begin()
  191. if err != nil {
  192. return
  193. }
  194. defer func() {
  195. if err != nil {
  196. _ = to.Rollback()
  197. } else {
  198. _ = to.Commit()
  199. }
  200. }()
  201. // 编辑用印
  202. sealInfo, err := editSeal(req.SealId, sysUser.AdminId, req.ContractId, req.FileNum, req.Use, req.CompanyName, req.UseCompanyName, req.CreditCode, req.ServiceType, req.SealType, req.Remark, req.FileUrls, req.AffiliatedCompany)
  203. if err != nil {
  204. return
  205. }
  206. // 添加操作日志
  207. err = seal.AddSealOperationRecord(sealInfo.SealId, sysUser.AdminId, 0, "edit", sysUser.RealName, "重提审批", "")
  208. if err != nil {
  209. return
  210. }
  211. // 用印审批
  212. err = SubmitApproval(sealInfo)
  213. return
  214. }
  215. // editSeal 编辑用印申请
  216. func editSeal(sealId, userId, contractId, fileNum int, use, companyName, userCompanyName, creditCode, serviceType, sealType, remark string, fileUrls []string, affiliatedCompany string) (sealInfo *seal.Seal, err error) {
  217. if !strings.Contains(strings.Join(seal.EnumUse, ","), use) {
  218. err = errors.New("用印用途异常")
  219. return
  220. }
  221. if !strings.Contains(strings.Join(seal.EnumServiceType, ","), serviceType) {
  222. err = errors.New("业务类型异常")
  223. return
  224. }
  225. sealInfo, err = seal.GetSealInfoById(sealId)
  226. if err != nil {
  227. if err.Error() == utils.ErrNoRow() {
  228. err = errors.New("获取用印申请失败")
  229. }
  230. return
  231. }
  232. if sealInfo.UserId != userId {
  233. err = errors.New("当前账号无操作权限")
  234. return
  235. }
  236. ignoreStatus := []string{"待提交", "已撤回", "已驳回"}
  237. if !strings.Contains(strings.Join(ignoreStatus, ","), sealInfo.Status) {
  238. err = errors.New("用印状态异常,不允许修改,当前用印状态:" + sealInfo.Status)
  239. return
  240. }
  241. fileUrl := ""
  242. now := time.Now()
  243. attachments := make([]*seal.Attachment, 0)
  244. if len(fileUrls) == 1 {
  245. fileUrl = fileUrls[0]
  246. }
  247. sealInfo.Use = use
  248. sealInfo.CompanyName = companyName
  249. sealInfo.UseCompanyName = userCompanyName
  250. sealInfo.CreditCode = creditCode
  251. sealInfo.ServiceType = serviceType
  252. sealInfo.SealType = sealType
  253. sealInfo.Remark = remark
  254. sealInfo.FileUrl = fileUrl
  255. sealInfo.FileNum = fileNum
  256. sealInfo.ContractId = contractId
  257. sealInfo.ModifyTime = now
  258. sealInfo.CreateTime = now // 更新提交时间
  259. sealInfo.Status = "待提交"
  260. sealInfo.AffiliatedCompany = affiliatedCompany
  261. err = sealInfo.Update([]string{"Use", "CompanyName", "UseCompanyName", "CreditCode", "ServiceType", "SealType", "Remark", "FileUrl", "FileNum", "ContractId", "ModifyTime", "CreateTime", "Status", "AffiliatedCompany"})
  262. if err != nil {
  263. return
  264. }
  265. //删除原有的附件,新增最新的附件
  266. err = seal.DelAttachmentBySealId(sealId)
  267. if err != nil {
  268. return
  269. }
  270. if len(fileUrls) > 1 {
  271. for _, v := range fileUrls {
  272. tmp := &seal.Attachment{
  273. SealId: sealInfo.SealId,
  274. FileUrl: v,
  275. ModifyTime: now,
  276. CreateTime: now,
  277. }
  278. attachments = append(attachments, tmp)
  279. }
  280. err = seal.AddAttachments(attachments)
  281. }
  282. return
  283. }
  284. // VerifierEditApply 合规编辑用印申请
  285. func VerifierEditApply(sysUser *system.Admin, req request.SealApprovalEditReq) (err error) {
  286. // 审批权限校验
  287. sealInfo, approvalInfo, approvalRecord, err := CheckApprovalAuth(req.SealId, sysUser)
  288. if err != nil {
  289. return
  290. }
  291. // 数据状态校验
  292. if sysUser.RoleTypeCode != utils.ROLE_TYPE_CODE_COMPLIANCE {
  293. err = errors.New("当前账号无修改权限")
  294. return
  295. }
  296. if sealInfo.Status != "待审批" {
  297. err = errors.New("用印状态异常,不允许修改,当前用印状态:" + sealInfo.Status)
  298. return
  299. }
  300. if !strings.Contains(strings.Join(seal.EnumUse, ","), req.Use) {
  301. err = errors.New("用印用途异常")
  302. return
  303. }
  304. if !strings.Contains(strings.Join(seal.EnumServiceType, ","), req.ServiceType) {
  305. err = errors.New("业务类型异常")
  306. return
  307. }
  308. // 变更字段判断
  309. updateCol := make([]string, 0)
  310. updateContent := make([]string, 0)
  311. if sealInfo.Use != req.Use {
  312. updateCol = append(updateCol, "Use")
  313. updateContent = append(updateContent, "用印用途")
  314. }
  315. if sealInfo.SealType != req.SealType {
  316. updateCol = append(updateCol, "SealType")
  317. updateContent = append(updateContent, "加盖何种印章")
  318. }
  319. if sealInfo.Remark != req.Remark {
  320. updateCol = append(updateCol, "Remark")
  321. updateContent = append(updateContent, "备注")
  322. }
  323. if sealInfo.FileNum != req.FileNum {
  324. updateCol = append(updateCol, "FileNum")
  325. updateContent = append(updateContent, "文件份数")
  326. }
  327. //判断附件地址是否相等:
  328. sealAttachements, err := seal.GetAttachmentBySealId(sealInfo.SealId)
  329. if err != nil {
  330. err = errors.New(fmt.Sprint("获取用印附件失败,Err:"+err.Error(), err))
  331. return
  332. }
  333. oldFileUrls := make([]string, 0)
  334. if sealInfo.FileUrl != "" {
  335. oldFileUrls = append(oldFileUrls, sealInfo.FileUrl)
  336. }
  337. if len(sealAttachements) > 0 {
  338. for _, v := range sealAttachements {
  339. oldFileUrls = append(oldFileUrls, v.FileUrl)
  340. }
  341. }
  342. attachmentFlag := false
  343. if !reflect.DeepEqual(oldFileUrls, req.FileUrls) {
  344. updateCol = append(updateCol, "FileUrl")
  345. updateContent = append(updateContent, "文件附件")
  346. attachmentFlag = true
  347. }
  348. if len(updateCol) <= 0 {
  349. err = errors.New("没有变更信息")
  350. return
  351. }
  352. content := "用印修改:" + sysUser.RealName + "修改了" + strings.Join(updateContent, "、")
  353. // 通过盖章类型判断是否需要变更审批流程
  354. originFlowType := GetFlowTypeBySealType(sealInfo.SealType)
  355. reqFlowType := GetFlowTypeBySealType(req.SealType)
  356. //如果变更了盖章类型(涉及到审批流程变更),那么需要变更审批流
  357. //变更审批流:先驳回用印申请,再修改用印,再发消息给申请人,再替申请人重新发起申请,
  358. //然后遍历审批流程,当前操作人节点之前的全部审核通过,最后通过当前操作人的节点(改成如果一级审批人如果是自己,那么直接通过这一级审批,否则不通过,让审批单重走流程)
  359. newFileUrl := ""
  360. if len(req.FileUrls) == 1 {
  361. newFileUrl = req.FileUrls[0]
  362. }
  363. if originFlowType != reqFlowType {
  364. // 先驳回
  365. err = reject(sealInfo, approvalInfo, approvalRecord, sysUser, "")
  366. if err != nil {
  367. return
  368. }
  369. //修改用印申请,给申请人发消息
  370. sealInfo.Use = req.Use
  371. sealInfo.SealType = req.SealType
  372. sealInfo.Remark = req.Remark
  373. sealInfo.FileNum = req.FileNum
  374. sealInfo.FileUrl = newFileUrl
  375. sealInfo.ModifyTime = time.Now()
  376. verifierEdit(sealInfo, updateCol, approvalRecord.ContractApprovalRecordId, content, sysUser, attachmentFlag, req.FileUrls)
  377. // 重新获取最新的用印单
  378. sealInfo, tmpErr := seal.GetSealInfoById(sealInfo.SealId)
  379. if tmpErr != nil {
  380. err = tmpErr
  381. return
  382. }
  383. err = reapply(sealInfo, sysUser)
  384. if err != nil {
  385. return
  386. }
  387. // 重新发起申请
  388. } else {
  389. // 修改用印申请,给申请人发消息
  390. sealInfo.Use = req.Use
  391. sealInfo.SealType = req.SealType
  392. sealInfo.Remark = req.Remark
  393. sealInfo.FileNum = req.FileNum
  394. sealInfo.FileUrl = newFileUrl
  395. sealInfo.ModifyTime = time.Now()
  396. verifierEdit(sealInfo, updateCol, approvalRecord.ContractApprovalRecordId, content, sysUser, attachmentFlag, req.FileUrls)
  397. // 审核通过
  398. err = ApprovedApproval(sealInfo.SealId, sysUser, "")
  399. }
  400. // 操作日志
  401. err = seal.AddSealOperationRecord(sealInfo.SealId, sysUser.AdminId, approvalRecord.ContractApprovalRecordId, "edit", sysUser.RealName, "审批人修改信息", "")
  402. if err != nil {
  403. return
  404. }
  405. return
  406. }
  407. // verifierEdit 审批人修改用印信息
  408. func verifierEdit(sealInfo *seal.Seal, updateCol []string, approvalRecordId int, content string, opUser *system.Admin, attachmentFlag bool, fileUrls []string) {
  409. if attachmentFlag {
  410. // 同时更新用印申请和附件
  411. _ = updateSealAndAttachment(sealInfo, updateCol, fileUrls)
  412. } else {
  413. _ = sealInfo.Update(updateCol)
  414. }
  415. //给用印人,发送修改消息
  416. sysUserMobile := ""
  417. sysUser, _ := system.GetSysAdminById(opUser.AdminId)
  418. if sysUser != nil {
  419. sysUserMobile = sysUser.Mobile
  420. }
  421. go services.AddCompanyApprovalMessage(opUser.AdminId, sealInfo.UserId, 0, approvalRecordId, 3, sourceType, 2, sealInfo.CompanyName, content, content, sysUserMobile)
  422. return
  423. }
  424. // reapply 重新提交审批单
  425. func reapply(sealInfo *seal.Seal, opUser *system.Admin) (err error) {
  426. // 发起审批
  427. approvalInfo, contractApprovalRecordList, err := addApproval(sealInfo)
  428. if err != nil {
  429. return
  430. }
  431. // 校验一级审批人是否与当前提交人一致,如果一致的话,那么审批通过且不发送消息给一级审批人
  432. var contractApprovalRecord *contract.ContractApprovalRecord
  433. for _, tmpContractApprovalRecord := range contractApprovalRecordList {
  434. if tmpContractApprovalRecord.NodeId == approvalInfo.CurrNodeId && tmpContractApprovalRecord.ApproveUserId == opUser.AdminId {
  435. contractApprovalRecord = tmpContractApprovalRecord
  436. }
  437. }
  438. // 如果一级审批人和当前提交人一致,那么审批通过,且不发送消息给一级审批人
  439. if contractApprovalRecord != nil {
  440. err = ApprovedApproval(sealInfo.SealId, opUser, "")
  441. } else {
  442. //如果一级审批人和当前提交人不一致,那么发送消息给一级审批人,重走审批流程
  443. //待办通知
  444. {
  445. //发送消息下级审批人
  446. go MessageToNodeUser(contractApprovalRecordList[0].NodeId, approvalInfo.ApplyUserId, approvalInfo.ContractApprovalId, 1, 1, sealInfo.CompanyName, sealInfo.Use)
  447. }
  448. }
  449. return
  450. }
  451. // GetSealApprovalPageList 获取用印审批分页列表
  452. func GetSealApprovalPageList(condition, joinCondition string, pars []interface{}, startSize, pageSize int, sysUser *system.Admin) (newList []*response.SealApprovalList, total int, err error) {
  453. // 1.列表数据
  454. list, total, err := seal.GetSealApprovalListByWhere(condition, joinCondition, pars, startSize, pageSize)
  455. if err != nil {
  456. err = errors.New(fmt.Sprint("获取用印审批列表失败,Err:"+err.Error(), err))
  457. return
  458. }
  459. // 2.操作权限
  460. if len(list) > 0 {
  461. // 取出所有列表最新的审批IDs
  462. approvalIdSlice := make([]string, 0)
  463. // 取出所有列表的关联合同id
  464. contractIdSlice := make([]string, 0)
  465. for i := 0; i < len(list); i++ {
  466. approvalIdSlice = append(approvalIdSlice, strconv.Itoa(list[i].ContractApprovalId))
  467. contractIdSlice = append(contractIdSlice, strconv.Itoa(list[i].ContractId))
  468. }
  469. approvalIdStr := strings.Join(approvalIdSlice, ",")
  470. // 通过审批IDs获取所有的审批流
  471. approvalRecordList, tempErr := contract.GetContractApprovalRecordListByContractApprovalIds(approvalIdStr)
  472. //approvalRecordList, tempErr := contract.GetContractApprovalRecordList(approvalIdStr, sysUser.AdminId)
  473. if tempErr != nil {
  474. err = errors.New(fmt.Sprint("获取审批流失败,Err:"+tempErr.Error(), err))
  475. return
  476. }
  477. // 获取自己的审批流列表数据
  478. selfContractApprovalRecordList, tempErr := contract.GetSelfContractApprovalRecordList(approvalIdStr, sysUser.AdminId)
  479. if tempErr != nil {
  480. err = errors.New(fmt.Sprint("获取审批列表失败,Err:"+tempErr.Error(), err))
  481. return
  482. }
  483. selfContractApprovalRecordMap := make(map[int]*contract.ContractApprovalRecord)
  484. for i := 0; i < len(selfContractApprovalRecordList); i++ {
  485. selfContractApprovalRecordMap[selfContractApprovalRecordList[i].ContractApprovalId] = selfContractApprovalRecordList[i]
  486. }
  487. // 获取所有关联的合同列表
  488. selfContractMap := make(map[int]*contract.ContractList)
  489. {
  490. if len(contractIdSlice) > 0 {
  491. contractIdStr := strings.Join(contractIdSlice, ",")
  492. contractList, tempErr := contract.GetContractListByContractIds(contractIdStr)
  493. if tempErr != nil {
  494. err = errors.New(fmt.Sprint("获取合同失败,Err:"+tempErr.Error(), err))
  495. return
  496. }
  497. for i := 0; i < len(contractList); i++ {
  498. selfContractMap[contractList[i].ContractId] = contractList[i]
  499. }
  500. }
  501. }
  502. for i := 0; i < len(list); i++ {
  503. item := list[i]
  504. list[i].CreateTimeStr = item.CreateTime.Format(utils.FormatDateTime)
  505. list[i].ModifyTimeStr = item.ModifyTime.Format(utils.FormatDateTime)
  506. list[i].ApproveTimeStr = item.ApproveTime.Format(utils.FormatDateTime)
  507. list[i].InvalidTimeStr = item.InvalidTime.Format(utils.FormatDateTime)
  508. list[i].CheckBackFileTimeStr = item.CheckBackFileTime.Format(utils.FormatDateTime)
  509. if item.ApproveStatus == "已审批" || item.ApproveStatus == "已驳回" {
  510. list[i].ApproveTime = item.ModifyTime
  511. list[i].ApproveTimeStr = item.ModifyTime.Format(utils.FormatDateTime)
  512. }
  513. // 如果当前状态是待审批的话,那么校验自己的审批流数据,然后去返回对应的状态
  514. if item.Status == "待审批" {
  515. // 如果是自己发起的审批单,同时已经经过了一轮审批,那么标记为处理中
  516. if item.ApplyUserId == sysUser.AdminId && item.CurrNodeId > item.StartNodeId {
  517. list[i].Status = "处理中"
  518. } else if sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_ADMIN && item.CurrNodeId > item.StartNodeId {
  519. // 如果是超管角色,同时已经经过了一轮审批,那么标记为处理中
  520. list[i].Status = "处理中"
  521. } else {
  522. // 此处可能出现同一审批人同时有权限处理一级二级的情况
  523. maxNodeId := 0
  524. tempStatus := "待审批"
  525. for _, selfRecord := range selfContractApprovalRecordList {
  526. if item.ContractApprovalId == selfRecord.ContractApprovalId {
  527. if selfRecord.NodeId > maxNodeId {
  528. maxNodeId = selfRecord.NodeId
  529. }
  530. if maxNodeId < item.CurrNodeId && selfRecord.NodeType != "cc" {
  531. tempStatus = "处理中"
  532. } else if maxNodeId == item.CurrNodeId && selfRecord.NodeType != "cc" {
  533. tempStatus = "待审批"
  534. }
  535. }
  536. }
  537. list[i].Status = tempStatus
  538. /*if contractApprovalRecord, has := selfContractApprovalRecordMap[item.ContractApprovalId]; has {
  539. if contractApprovalRecord.NodeId < item.CurrNodeId && contractApprovalRecord.NodeType != "cc" {
  540. list[i].Status = "处理中"
  541. }
  542. }*/
  543. }
  544. }
  545. // 合同编码
  546. if selfContract, has := selfContractMap[item.ContractId]; has {
  547. list[i].ContractCode = selfContract.ContractCode
  548. }
  549. // 取出item对应approval_id的审批流
  550. var tempApprovalRecordMap []*contract.ContractApprovalRecord
  551. for _, recordItem := range approvalRecordList {
  552. if item.ContractApprovalId == recordItem.ContractApprovalId {
  553. tempApprovalRecordMap = append(tempApprovalRecordMap, recordItem)
  554. }
  555. }
  556. // 获取对该条数据的操作权限
  557. opButton, tempErr := handleListOpButton(item, tempApprovalRecordMap, sysUser)
  558. if tempErr != nil {
  559. err = errors.New(fmt.Sprint("获取操作权限失败,Err:"+tempErr.Error(), err))
  560. return
  561. }
  562. tempNewList := &response.SealApprovalList{
  563. SealApprovalItem: item,
  564. OpButton: opButton,
  565. }
  566. newList = append(newList, tempNewList)
  567. }
  568. }
  569. return
  570. }
  571. // handleOpButton 列表数据操作权限
  572. func handleListOpButton(approvalItem *seal.SealApprovalItem, recordList []*contract.ContractApprovalRecord, sysUser *system.Admin) (opButton response.SealApprovalOpButton, err error) {
  573. // 待审批状态下,如果是自己发起的审批单,同时已经经过了一轮审批,那么标记为处理中
  574. if approvalItem.Status == "待审批" {
  575. if approvalItem.ApplyUserId == sysUser.AdminId && approvalItem.CurrNodeId > approvalItem.StartNodeId {
  576. approvalItem.Status = "处理中"
  577. }
  578. }
  579. flowNodeMap := make(map[int][]contract.ContractApprovalRecord, 0)
  580. keySort := make([]int, 0)
  581. for _, approvalRecord := range recordList {
  582. // 如果当前节点正好是该节点,同时审批单状态是待审批状态,然后当前账号又有权限,该账号也正是审批人,那么允许审批操作
  583. if approvalItem.CurrNodeId == approvalRecord.NodeId && approvalItem.Status == "待审批" {
  584. if sysUser.AdminId == approvalRecord.ApproveUserId && approvalRecord.NodeType == "check" {
  585. opButton.Approval = true
  586. }
  587. // 合规在审批过程中具有 部分修改权限
  588. if sysUser.AdminId == approvalRecord.ApproveUserId && sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_COMPLIANCE && approvalRecord.NodeType == "check" {
  589. opButton.CheckEdit = true
  590. }
  591. }
  592. // 如果当前节点正好是该节点,同时审批单状态又是第一层节点
  593. if approvalItem.CurrNodeId == approvalRecord.NodeId && approvalRecord.PrevNodeId == 0 && approvalItem.Status == "待审批" {
  594. // 发起人具有撤回审批单权限
  595. if sysUser.AdminId == approvalItem.ApplyUserId {
  596. opButton.Cancel = true
  597. }
  598. }
  599. if tmpFlowNodeList, ok := flowNodeMap[approvalRecord.NodeId]; ok {
  600. flowNodeMap[approvalRecord.NodeId] = append(tmpFlowNodeList, *approvalRecord)
  601. } else {
  602. tmpFlowNodeList := make([]contract.ContractApprovalRecord, 1)
  603. tmpFlowNodeList[0] = *approvalRecord
  604. flowNodeMap[approvalRecord.NodeId] = tmpFlowNodeList
  605. keySort = append(keySort, approvalRecord.NodeId)
  606. }
  607. // 待审批状态下,如果当前审批节点就是操作人,审批节点超过当前审批节点,且节点类型是抄送人,,那么标记为处理中
  608. if approvalItem.Status == "待审批" {
  609. if approvalRecord.ApproveUserId == sysUser.AdminId && approvalRecord.NodeId < approvalItem.CurrNodeId && approvalRecord.NodeType == "cc" {
  610. approvalItem.Status = "处理中"
  611. }
  612. }
  613. }
  614. // 作废权限
  615. if sysUser.AdminId == approvalItem.ApplyUserId && approvalItem.Status == "已审批" {
  616. opButton.Invalid = true
  617. }
  618. // 是否具有签回合同用印权限
  619. uploadStatus := []string{"已审批", "已签回"}
  620. if sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_COMPLIANCE && strings.Contains(strings.Join(uploadStatus, ","), approvalItem.Status) {
  621. opButton.UploadFile = true
  622. }
  623. // 编辑权限
  624. editStatus := []string{"已驳回", "待提交", "已撤回"}
  625. if sysUser.AdminId == approvalItem.ApplyUserId && strings.Contains(strings.Join(editStatus, ","), approvalItem.Status) {
  626. opButton.Edit = true
  627. }
  628. return
  629. }
  630. // GetSealApprovalDetail 获取审批详情
  631. func GetSealApprovalDetail(sealId int, sysUser *system.Admin) (sealMore *response.SealMoreResp, flowNodeListResp [][]contract.ContractApprovalRecord, opButton response.SealApprovalOpButton, err error) {
  632. sealInfo, err := seal.GetSealInfoById(sealId)
  633. if err != nil {
  634. err = errors.New(fmt.Sprint("获取用印详情失败,Err:"+err.Error(), err))
  635. return
  636. }
  637. sealAttachements, err := seal.GetAttachmentBySealId(sealId)
  638. if err != nil {
  639. err = errors.New(fmt.Sprint("获取用印附件失败,Err:"+err.Error(), err))
  640. return
  641. }
  642. fileUrls := make([]string, 0)
  643. if sealInfo.FileUrl != "" {
  644. fileUrls = append(fileUrls, sealInfo.FileUrl)
  645. }
  646. for _, v := range sealAttachements {
  647. fileUrls = append(fileUrls, v.FileUrl)
  648. }
  649. sealMore = &response.SealMoreResp{
  650. Seal: sealInfo,
  651. FileUrls: fileUrls,
  652. }
  653. // 查询最近一次审批单信息
  654. lastApprovalInfo, err := contract.GetLastContractApprovalByContractId(sealInfo.SealId, "seal")
  655. if err != nil {
  656. if err.Error() != utils.ErrNoRow() {
  657. err = errors.New("获取最近一次审批单信息失败,ERR:" + err.Error())
  658. }
  659. err = nil
  660. return
  661. }
  662. if lastApprovalInfo != nil {
  663. // 获取审批操作和审批流数据
  664. tmpSealOpButton, tmpFlowNodeListResp, tmpErr := GetSealOpButton(sealInfo, lastApprovalInfo, sysUser)
  665. if tmpErr != nil {
  666. err = errors.New("获取用印按钮权限、审批流程失败,Err:" + tmpErr.Error())
  667. return
  668. }
  669. opButton = tmpSealOpButton
  670. flowNodeListResp = tmpFlowNodeListResp
  671. }
  672. return
  673. }
  674. // GetSealOpButton 获取用印审批操作权限
  675. func GetSealOpButton(sealInfo *seal.Seal, contractApprovalInfo *contract.ContractApproval, opUser *system.Admin) (sealOpButton response.SealApprovalOpButton, flowNodeListResp [][]contract.ContractApprovalRecord, err error) {
  676. // 审批流
  677. approvalRecordList, err := contract.GetApprovalRecordListByApprovalId(contractApprovalInfo.ContractApprovalId)
  678. if err != nil {
  679. err = errors.New("获取审批失败,Err:" + err.Error())
  680. return
  681. }
  682. // 待审批状态下,如果是自己发起的审批单,同时已经经过了一轮审批,那么标记为处理中
  683. if contractApprovalInfo.Status == "待审批" {
  684. if contractApprovalInfo.ApplyUserId == opUser.AdminId && contractApprovalInfo.CurrNodeId > contractApprovalInfo.StartNodeId {
  685. sealInfo.Status = "处理中"
  686. }
  687. }
  688. flowNodeMap := make(map[int][]contract.ContractApprovalRecord, 0)
  689. keySort := make([]int, 0)
  690. for _, approvalRecord := range approvalRecordList {
  691. // 如果当前节点正好是该节点,同时审批单状态是待审批状态,然后当前账号又有权限,该账号也正是审批人,那么允许审批操作
  692. if contractApprovalInfo.CurrNodeId == approvalRecord.NodeId && contractApprovalInfo.Status == "待审批" {
  693. if opUser.AdminId == approvalRecord.ApproveUserId && approvalRecord.NodeType == "check" {
  694. sealOpButton.Approval = true
  695. }
  696. // 合规在审批过程中具有 部分修改权限
  697. if opUser.AdminId == approvalRecord.ApproveUserId && opUser.RoleTypeCode == utils.ROLE_TYPE_CODE_COMPLIANCE && approvalRecord.NodeType == "check" {
  698. sealOpButton.CheckEdit = true
  699. }
  700. }
  701. // 如果当前节点正好是该节点,同时审批单状态又是第一层节点
  702. if contractApprovalInfo.CurrNodeId == approvalRecord.NodeId && approvalRecord.PrevNodeId == 0 && contractApprovalInfo.Status == "待审批" {
  703. // 发起人具有撤回审批单权限
  704. if opUser.AdminId == sealInfo.UserId {
  705. sealOpButton.Cancel = true
  706. }
  707. }
  708. if tmpFlowNodeList, ok := flowNodeMap[approvalRecord.NodeId]; ok {
  709. flowNodeMap[approvalRecord.NodeId] = append(tmpFlowNodeList, *approvalRecord)
  710. } else {
  711. tmpFlowNodeList := make([]contract.ContractApprovalRecord, 1)
  712. tmpFlowNodeList[0] = *approvalRecord
  713. flowNodeMap[approvalRecord.NodeId] = tmpFlowNodeList
  714. keySort = append(keySort, approvalRecord.NodeId)
  715. }
  716. // 待审批状态下,如果当前审批节点就是操作人,审批节点超过当前审批节点,且节点类型是抄送人,,那么标记为处理中
  717. if contractApprovalInfo.Status == "待审批" {
  718. if approvalRecord.ApproveUserId == opUser.AdminId && approvalRecord.NodeId < contractApprovalInfo.CurrNodeId && approvalRecord.NodeType == "cc" {
  719. sealInfo.Status = "处理中"
  720. }
  721. }
  722. }
  723. for _, key := range keySort {
  724. flowNodeListResp = append(flowNodeListResp, flowNodeMap[key])
  725. }
  726. // 作废权限
  727. if opUser.AdminId == sealInfo.UserId && sealInfo.Status == "已审批" {
  728. sealOpButton.Invalid = true
  729. }
  730. // 是否具有签回合同用印权限
  731. uploadStatus := []string{"已审批", "已签回"}
  732. if opUser.RoleTypeCode == utils.ROLE_TYPE_CODE_COMPLIANCE && strings.Contains(strings.Join(uploadStatus, ","), sealInfo.Status) {
  733. sealOpButton.UploadFile = true
  734. }
  735. // 编辑权限
  736. editStatus := []string{"已驳回", "待提交", "已撤回"}
  737. if opUser.AdminId == sealInfo.UserId && strings.Contains(strings.Join(editStatus, ","), sealInfo.Status) {
  738. sealOpButton.Edit = true
  739. }
  740. return
  741. }
  742. // CancelSealApproval 撤回用印审批
  743. func CancelSealApproval(sealId int, sysUser *system.Admin) (err error) {
  744. // 审批信息
  745. sealInfo, err := seal.GetSealInfoById(sealId)
  746. if err != nil {
  747. if err.Error() == utils.ErrNoRow() {
  748. err = errors.New("用印审批信息有误,Err:" + err.Error())
  749. }
  750. return
  751. }
  752. // 审批状态判断
  753. if sealInfo.Status != "待审批" {
  754. err = errors.New("用印状态异常,不允许撤回申请,当前用印状态:" + sealInfo.Status)
  755. return
  756. }
  757. // 操作人与申请人是否一致
  758. if sealInfo.UserId != sysUser.AdminId {
  759. err = errors.New("不允许撤回非本人提交的申请:" + strconv.Itoa(sealInfo.UserId))
  760. return
  761. }
  762. // 获取最近一次待审批的审批单信息
  763. approvalInfo, err := contract.GetLastContractApprovalByContractId(sealInfo.SealId, "seal")
  764. if err != nil {
  765. return
  766. }
  767. if approvalInfo.Status != "待审批" {
  768. err = errors.New("审批单状态异常,审批单状态:" + approvalInfo.Status)
  769. return
  770. }
  771. //获取当前审批单的审批流信息
  772. approvalRecordList, err := contract.GetContractApprovalRecordListByContractApprovalIdAndNodeId(approvalInfo.ContractApprovalId, approvalInfo.CurrNodeId)
  773. if err != nil {
  774. err = errors.New("获取审批流失败,Err:" + err.Error())
  775. return
  776. }
  777. if len(approvalRecordList) <= 0 {
  778. err = errors.New("找不到对应的审批流")
  779. return
  780. }
  781. for _, tmpApprovalRecord := range approvalRecordList {
  782. // 判断当前节点是否存在上级节点,如果存在,那么说明
  783. if tmpApprovalRecord.PrevNodeId > 0 {
  784. err = errors.New("合同已存在审批操作,不允许撤回申请")
  785. return
  786. }
  787. }
  788. // 获取当前审批单中所有待审批的流程流
  789. contractApprovalRecordList, err := contract.GetPendingContractApprovalRecordListByContractId(approvalInfo.ContractApprovalId, approvalInfo.CurrNodeId)
  790. if err != nil {
  791. return
  792. }
  793. // 撤回审批流
  794. err = approvalInfo.Cancel(approvalInfo, contractApprovalRecordList)
  795. if err != nil {
  796. return
  797. }
  798. // 添加操作日志
  799. err = seal.AddSealOperationRecord(sealInfo.SealId, sysUser.AdminId, 0, "cancel", sysUser.RealName, "撤回申请", "")
  800. if err != nil {
  801. return
  802. }
  803. // 作废原有消息
  804. for _, contractApprovalRecord := range contractApprovalRecordList {
  805. go company.CancelCompanyApprovalMessage(contractApprovalRecord.ContractApprovalRecordId, 3)
  806. }
  807. return
  808. }
  809. // DelSealApproval 删除用印审批
  810. func DelSealApproval(sealId int, sysUser *system.Admin) (err error) {
  811. // 审批信息
  812. sealInfo, err := seal.GetSealInfoById(sealId)
  813. if err != nil {
  814. if err.Error() == utils.ErrNoRow() {
  815. err = errors.New("用印审批信息有误,Err:" + err.Error())
  816. }
  817. return
  818. }
  819. // 审批状态判断
  820. if sealInfo.Status != "已撤回" {
  821. err = errors.New("用印状态异常,不允许删除,当前用印状态:" + sealInfo.Status)
  822. return
  823. }
  824. // 操作人与申请人是否一致
  825. if sealInfo.UserId != sysUser.AdminId {
  826. err = errors.New("不允许删除非本人提交的申请:" + strconv.Itoa(sealInfo.UserId))
  827. return
  828. }
  829. // 删除用印审批
  830. err = seal.DeleteSeal(sealInfo)
  831. if err != nil {
  832. err = errors.New("删除用印申请失败:" + strconv.Itoa(sealInfo.UserId))
  833. return
  834. }
  835. // 添加操作日志
  836. err = seal.AddSealOperationRecord(sealInfo.SealId, sysUser.AdminId, 0, "del", sysUser.RealName, "删除申请", "")
  837. if err != nil {
  838. return
  839. }
  840. return
  841. }
  842. // InvalidSealApproval 作废用印审批
  843. func InvalidSealApproval(sealId int, opUser *system.Admin) (err error) {
  844. //查询当前用印信息
  845. sealInfo, err := seal.GetSealInfoById(sealId)
  846. if err != nil {
  847. if err.Error() == utils.ErrNoRow() {
  848. err = errors.New(fmt.Sprint("根据用印编号:", sealId, " 找不到初始用印"))
  849. }
  850. return
  851. }
  852. if sealInfo.UserId != opUser.AdminId && opUser.RoleTypeCode != utils.ROLE_TYPE_CODE_COMPLIANCE {
  853. err = errors.New("当前账号无操作权限")
  854. return
  855. }
  856. if sealInfo.Status != "已审批" {
  857. err = errors.New("用印状态异常,不允许作废,当前用印状态:" + sealInfo.Status)
  858. return
  859. }
  860. //如果删除状态 >0,那么代表已经被删除了
  861. if sealInfo.IsDelete > 0 {
  862. err = errors.New("该用印已删除")
  863. return
  864. }
  865. //用印作废
  866. err = seal.Invalid(sealInfo)
  867. if err != nil {
  868. return
  869. }
  870. // 操作日志
  871. err = seal.AddSealOperationRecord(sealInfo.SealId, opUser.AdminId, 0, "invalid", opUser.RealName, "作废审批", "")
  872. if err != nil {
  873. return
  874. }
  875. //获取最近一次提交的审批单信息
  876. contractApproval, err := contract.GetLastContractApprovalByContractId(sealInfo.SealId, "seal")
  877. if err != nil {
  878. err = nil
  879. return
  880. }
  881. //如果有提测信息,那么需要给对应的审批人发送消息
  882. if contractApproval != nil {
  883. //作废后需要给审批者发送消息
  884. contractApprovalRecordList, tmpErr := contract.GetContractApprovalRecordListByContractApprovalId(contractApproval.ContractApprovalId)
  885. if tmpErr != nil {
  886. return
  887. }
  888. content := fmt.Sprint("作废", sealInfo.CompanyName, "用印申请")
  889. for _, contractApprovalRecord := range contractApprovalRecordList {
  890. if contractApprovalRecord.NodeType == "check" { //如果是审批人,那么需要发送消息给对方
  891. //操作人信息
  892. sysUserMobile := ""
  893. sysUser, _ := system.GetSysAdminById(contractApprovalRecord.ApproveUserId)
  894. if sysUser != nil {
  895. sysUserMobile = sysUser.Mobile
  896. }
  897. go services.AddCompanyApprovalMessage(opUser.AdminId, contractApprovalRecord.ApproveUserId, 0, contractApprovalRecord.ContractApprovalRecordId, 3, sourceType, 2, sealInfo.CompanyName, content, content, sysUserMobile)
  898. }
  899. }
  900. }
  901. return
  902. }
  903. // GetSealOperationRecordList 获取用印操作日志列表
  904. func GetSealOperationRecordList(sealId int) (list []*seal.SealOperationRecordList, err error) {
  905. list, err = seal.GetSealOperationListBySealId(sealId)
  906. if len(list) > 0 {
  907. for k, item := range list {
  908. list[k].CreateTimeStr = item.CreateTime.Format(utils.FormatDateTime)
  909. }
  910. }
  911. return
  912. }
  913. // updateSealAndAttachment 更新用印申请,并更新附件
  914. func updateSealAndAttachment(sealInfo *seal.Seal, updateCol []string, fileUrls []string) (err error) {
  915. o := orm.NewOrm()
  916. to, err := o.Begin()
  917. if err != nil {
  918. return
  919. }
  920. defer func() {
  921. if err != nil {
  922. _ = to.Rollback()
  923. } else {
  924. _ = to.Commit()
  925. }
  926. }()
  927. err = sealInfo.Update(updateCol)
  928. if err != nil {
  929. return
  930. }
  931. now := time.Now()
  932. //删除原有的附件,新增最新的附件
  933. err = seal.DelAttachmentBySealId(sealInfo.SealId)
  934. if err != nil {
  935. return
  936. }
  937. attachments := make([]*seal.Attachment, 0)
  938. if len(fileUrls) > 1 {
  939. for _, v := range fileUrls {
  940. tmp := &seal.Attachment{
  941. SealId: sealInfo.SealId,
  942. FileUrl: v,
  943. ModifyTime: now,
  944. CreateTime: now,
  945. }
  946. attachments = append(attachments, tmp)
  947. }
  948. err = seal.AddAttachments(attachments)
  949. }
  950. return
  951. }