warhouse_service.go 51KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469
  1. package service
  2. import (
  3. "XT_New/models"
  4. "XT_New/models/service"
  5. "XT_New/utils"
  6. "errors"
  7. "fmt"
  8. _ "fmt"
  9. "github.com/jinzhu/gorm"
  10. "math"
  11. "strconv"
  12. "strings"
  13. "time"
  14. )
  15. // 药品出库
  16. func DrugsDelivery(orgID int64, advice *models.DoctorAdvice) (err error) {
  17. // 1.判断药品是否来自专用字典的药品库
  18. // 2.判断当天当前机构有没有创建出库单,没有则创建
  19. // 3.创建出库流程
  20. // 3.1 实现先进先出逻辑 由于药品执行后,不可以修改和删除,所以不考虑出库后的退库和修改出库数量等
  21. isHasWay := false
  22. record_time := int64(0)
  23. if advice.Way == 1 {
  24. isHasWay = true
  25. record_time = advice.RecordDate
  26. }
  27. fmt.Println("isHasway99999999999999999999999999999999", isHasWay)
  28. if isHasWay {
  29. //判断当天当前机构有没有创建出库单,没有则创建
  30. out, err := FindDrugStockOutByIsSys(orgID, 1, record_time)
  31. if err == gorm.ErrRecordNotFound {
  32. timeStr := time.Now().Format("2006-01-02")
  33. timeArr := strings.Split(timeStr, "-")
  34. total, _ := FindAllDrugWarehouseOut(orgID)
  35. total = total + 1
  36. warehousing_out_order := strconv.FormatInt(orgID, 10) + timeArr[0] + timeArr[1] + timeArr[2] + "000"
  37. number, _ := strconv.ParseInt(warehousing_out_order, 10, 64)
  38. number = number + total
  39. warehousing_out_order = "YPCKD" + strconv.FormatInt(number, 10)
  40. // creater := adminUserInfo.AdminUser.Id
  41. warehouseOut := models.DrugWarehouseOut{
  42. WarehouseOutOrderNumber: warehousing_out_order,
  43. OperationTime: time.Now().Unix(),
  44. OrgId: orgID,
  45. Creater: 0,
  46. Ctime: time.Now().Unix(),
  47. Status: 1,
  48. WarehouseOutTime: record_time,
  49. Dealer: 0,
  50. Manufacturer: 0,
  51. Type: 1,
  52. IsSys: 1,
  53. }
  54. err := AddSigleDrugWarehouseOut(&warehouseOut)
  55. if err != nil {
  56. utils.TraceLog("创建出库单失败 err = %v", err)
  57. return err
  58. } else {
  59. out = warehouseOut
  60. }
  61. }
  62. // 出库流程
  63. // 1.查询改药品在药品库的规格信息,并将处方里的规格进行换算(尽量将拆零单位转换成包装单位)
  64. drup, _ := FindBaseDrugLibRecord(orgID, advice.DrugId)
  65. if drup.ID > 0 {
  66. prescribingNumber := advice.PrescribingNumber
  67. DrugDeliverInfo(orgID, prescribingNumber, &out, &drup, advice)
  68. } else {
  69. return errors.New("药品信息不存在")
  70. }
  71. }
  72. return
  73. }
  74. // 药品出库 递归方式
  75. func DrugDeliverInfo(orgID int64, prescribingNumber float64, warehouseout *models.DrugWarehouseOut, drup *models.BaseDrugLib, advice *models.DoctorAdvice) (err error) {
  76. // 判断处方里药品单位是拆零单位还是包装单位, 如果是包装单位,则根据规格,将包装数量转为拆零数量
  77. var deliver_number int64 = 0
  78. var stock_number int64 = 0
  79. prescribingNumber_temp := strconv.FormatFloat(math.Abs(prescribingNumber), 'f', 0, 64)
  80. count, _ := strconv.ParseInt(prescribingNumber_temp, 10, 64)
  81. fmt.Println("单位1", advice.PrescribingNumber)
  82. fmt.Println("单位2", drup.MaxUnit)
  83. if advice.PrescribingNumberUnit == drup.MaxUnit {
  84. deliver_number = count * drup.MinNumber
  85. } else {
  86. deliver_number = count
  87. }
  88. // 根据先进先出原则,查询最先入库的批次,进行出库
  89. // 如果没有对应的库存,则报错
  90. //开启事物
  91. warehouse, err := FindLastDrugWarehousingInfoByID(advice.DrugId)
  92. if err != nil {
  93. return err
  94. }
  95. // 将该批次的剩余库存数量转换为拆零数量
  96. stock_number = warehouse.StockMaxNumber*drup.MinNumber + warehouse.StockMinNumber
  97. fmt.Println("stock_number999999999999", stock_number)
  98. fmt.Println("deliver_number88888888888", deliver_number)
  99. // 当库存数量大于或等于出库数量的话,则正常出库该批次
  100. if stock_number >= deliver_number {
  101. warehouseOutInfo := &models.DrugWarehouseOutInfo{
  102. WarehouseOutOrderNumber: warehouseout.WarehouseOutOrderNumber,
  103. WarehouseOutId: warehouseout.ID,
  104. Status: 1,
  105. Ctime: time.Now().Unix(),
  106. Remark: "",
  107. OrgId: orgID,
  108. Type: 1,
  109. Manufacturer: 0,
  110. Dealer: 0,
  111. IsSys: 1,
  112. SysRecordTime: advice.RecordDate,
  113. DrugId: advice.DrugId,
  114. }
  115. warehouseOutInfo.Count = deliver_number
  116. warehouseOutInfo.CountUnit = drup.MinUnit
  117. errOne := AddSigleDrugWarehouseOutInfo(warehouseOutInfo)
  118. if errOne != nil {
  119. return errOne
  120. } else {
  121. details := &models.DrugAutomaticReduceDetail{
  122. WarehouseOutId: warehouseOutInfo.ID,
  123. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  124. PatientId: advice.PatientId,
  125. Ctime: time.Now().Unix(),
  126. Mtime: time.Now().Unix(),
  127. Status: 1,
  128. RecordTime: advice.RecordDate,
  129. OrgId: orgID,
  130. DrugId: advice.DrugId,
  131. Count: deliver_number,
  132. CountUnit: drup.MinUnit,
  133. }
  134. errTwo := AddSigleDrugAutoReduceRecordInfo(details)
  135. if errTwo != nil {
  136. return errTwo
  137. }
  138. }
  139. // 出库完成后,要减去对应批次的库存数量
  140. // 判断处方里药品单位是拆零单位还是包装单位, 如果是拆零单位,则根据规格,将拆零数量转为包装数量
  141. var maxNumber int64 = 0
  142. var minNumber int64 = 0
  143. if advice.PrescribingNumberUnit == drup.MinUnit {
  144. maxNumber = count / drup.MinNumber
  145. minNumber = count % drup.MinNumber
  146. } else {
  147. maxNumber = count
  148. }
  149. if warehouse.StockMaxNumber < maxNumber {
  150. return errors.New("库存数量不足")
  151. }
  152. warehouse.StockMaxNumber = warehouse.StockMaxNumber - maxNumber
  153. warehouse.Mtime = time.Now().Unix()
  154. if warehouse.StockMinNumber < minNumber {
  155. warehouse.StockMaxNumber = warehouse.StockMaxNumber - 1
  156. warehouse.StockMinNumber = warehouse.StockMinNumber + drup.MinNumber - minNumber
  157. } else {
  158. warehouse.StockMinNumber = warehouse.StockMinNumber - minNumber
  159. }
  160. if warehouse.StockMaxNumber < 0 {
  161. return errors.New("库存数量不足")
  162. }
  163. errThree := UpDateDrugWarehouseInfoByStock(&warehouse)
  164. if errThree != nil {
  165. return errThree
  166. }
  167. return nil
  168. } else {
  169. // 当该批次的库存数量小于出库数量的话,则先把该批次出库完后,再进行递归出库
  170. warehouseOutInfo := &models.DrugWarehouseOutInfo{
  171. WarehouseOutOrderNumber: warehouseout.WarehouseOutOrderNumber,
  172. WarehouseOutId: warehouseout.ID,
  173. Status: 1,
  174. Ctime: time.Now().Unix(),
  175. Remark: "",
  176. OrgId: orgID,
  177. Type: 1,
  178. Manufacturer: 0,
  179. Dealer: 0,
  180. IsSys: 1,
  181. SysRecordTime: advice.RecordDate,
  182. DrugId: advice.DrugId,
  183. }
  184. warehouseOutInfo.Count = stock_number
  185. warehouseOutInfo.CountUnit = drup.MinUnit
  186. errOne := AddSigleDrugWarehouseOutInfo(warehouseOutInfo)
  187. if errOne != nil {
  188. return errOne
  189. } else {
  190. details := &models.DrugAutomaticReduceDetail{
  191. WarehouseOutId: warehouseOutInfo.ID,
  192. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  193. PatientId: advice.PatientId,
  194. Ctime: time.Now().Unix(),
  195. Mtime: time.Now().Unix(),
  196. Status: 1,
  197. RecordTime: advice.RecordDate,
  198. OrgId: orgID,
  199. DrugId: advice.DrugId,
  200. Count: stock_number,
  201. CountUnit: drup.MinUnit,
  202. }
  203. errTwo := AddSigleDrugAutoReduceRecordInfo(details)
  204. if errTwo != nil {
  205. return errTwo
  206. }
  207. }
  208. // 出库完成后,要将该批次库存清零
  209. warehouse.StockMaxNumber = 0
  210. warehouse.StockMinNumber = 0
  211. warehouse.Mtime = time.Now().Unix()
  212. errThree := UpDateDrugWarehouseInfoByStock(&warehouse)
  213. if errThree != nil {
  214. return errThree
  215. }
  216. // 清零完该库存后,还有剩余出库未出完,进行对应的递归操作
  217. prescribingNumber_two_temp := deliver_number - stock_number
  218. overPlusNumber := float64(prescribingNumber_two_temp)
  219. advice.PrescribingNumber = float64(prescribingNumber_two_temp)
  220. advice.PrescribingNumberUnit = drup.MinUnit
  221. DrugDeliverInfo(orgID, overPlusNumber, warehouseout, drup, advice)
  222. }
  223. return
  224. }
  225. // 药品出库
  226. func HisDrugsDelivery(orgID int64, advice *models.HisDoctorAdviceInfo) (err error) {
  227. // 1.判断药品是否来自专用字典的药品库
  228. // 2.判断当天当前机构有没有创建出库单,没有则创建
  229. // 3.创建出库流程
  230. // 3.1 实现先进先出逻辑 由于药品执行后,不可以修改和删除,所以不考虑出库后的退库和修改出库数量等
  231. isHasWay := false
  232. record_time := int64(0)
  233. isHasWay = true
  234. record_time = advice.RecordDate
  235. if isHasWay {
  236. //判断当天当前机构有没有创建出库单,没有则创建
  237. out, err := FindDrugStockOutByIsSys(orgID, 1, record_time)
  238. if err == gorm.ErrRecordNotFound {
  239. timeStr := time.Now().Format("2006-01-02")
  240. timeArr := strings.Split(timeStr, "-")
  241. total, _ := FindAllDrugWarehouseOut(orgID)
  242. total = total + 1
  243. warehousing_out_order := strconv.FormatInt(orgID, 10) + timeArr[0] + timeArr[1] + timeArr[2] + "000"
  244. number, _ := strconv.ParseInt(warehousing_out_order, 10, 64)
  245. number = number + total
  246. warehousing_out_order = "YPCKD" + strconv.FormatInt(number, 10)
  247. // creater := adminUserInfo.AdminUser.Id
  248. warehouseOut := models.DrugWarehouseOut{
  249. WarehouseOutOrderNumber: warehousing_out_order,
  250. OperationTime: time.Now().Unix(),
  251. OrgId: orgID,
  252. Creater: 0,
  253. Ctime: time.Now().Unix(),
  254. Status: 1,
  255. WarehouseOutTime: record_time,
  256. Dealer: 0,
  257. Manufacturer: 0,
  258. Type: 1,
  259. IsSys: 1,
  260. }
  261. err := AddSigleDrugWarehouseOut(&warehouseOut)
  262. if err != nil {
  263. utils.TraceLog("创建出库单失败 err = %v", err)
  264. return err
  265. } else {
  266. out = warehouseOut
  267. }
  268. }
  269. // 出库流程
  270. // 1.查询改药品在药品库的规格信息,并将处方里的规格进行换算(尽量将拆零单位转换成包装单位)
  271. drup, _ := FindBaseDrugLibRecord(orgID, advice.DrugId)
  272. if drup.ID > 0 {
  273. prescribingNumber := advice.PrescribingNumber
  274. HisDrugDeliverInfo(orgID, prescribingNumber, &out, &drup, advice)
  275. } else {
  276. return errors.New("药品信息不存在")
  277. }
  278. }
  279. return
  280. }
  281. // 药品出库 递归方式
  282. func HisDrugDeliverInfo(orgID int64, prescribingNumber float64, warehouseout *models.DrugWarehouseOut, drup *models.BaseDrugLib, advice *models.HisDoctorAdviceInfo) (err error) {
  283. // 判断处方里药品单位是拆零单位还是包装单位, 如果是包装单位,则根据规格,将包装数量转为拆零数量
  284. var deliver_number int64 = 0
  285. var stock_number int64 = 0
  286. prescribingNumber_temp := strconv.FormatFloat(math.Abs(prescribingNumber), 'f', 0, 64)
  287. count, _ := strconv.ParseInt(prescribingNumber_temp, 10, 64)
  288. fmt.Println("单位99999999999999999999999999", advice.PrescribingNumberUnit)
  289. fmt.Println("drup777777777777766666666", drup.MaxUnit)
  290. if advice.PrescribingNumberUnit == drup.MaxUnit {
  291. deliver_number = count * drup.MinNumber
  292. } else {
  293. deliver_number = count
  294. }
  295. // 根据先进先出原则,查询最先入库的批次,进行出库
  296. // 如果没有对应的库存,则报错
  297. //开启事物
  298. warehouse, err := FindLastDrugWarehousingInfoByID(advice.DrugId)
  299. if err != nil {
  300. return err
  301. }
  302. // 将该批次的剩余库存数量转换为拆零数量
  303. stock_number = warehouse.StockMaxNumber*drup.MinNumber + warehouse.StockMinNumber
  304. // 当库存数量大于或等于出库数量的话,则正常出库该批次
  305. if stock_number >= deliver_number {
  306. warehouseOutInfo := &models.DrugWarehouseOutInfo{
  307. WarehouseOutOrderNumber: warehouseout.WarehouseOutOrderNumber,
  308. WarehouseOutId: warehouseout.ID,
  309. Status: 1,
  310. Ctime: time.Now().Unix(),
  311. Remark: "",
  312. OrgId: orgID,
  313. Type: 1,
  314. Manufacturer: 0,
  315. Dealer: 0,
  316. IsSys: 1,
  317. SysRecordTime: advice.RecordDate,
  318. DrugId: advice.DrugId,
  319. }
  320. warehouseOutInfo.Count = deliver_number
  321. warehouseOutInfo.CountUnit = drup.MinUnit
  322. errOne := AddSigleDrugWarehouseOutInfo(warehouseOutInfo)
  323. if errOne != nil {
  324. return errOne
  325. } else {
  326. details := &models.DrugAutomaticReduceDetail{
  327. WarehouseOutId: warehouseOutInfo.ID,
  328. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  329. PatientId: advice.PatientId,
  330. Ctime: time.Now().Unix(),
  331. Mtime: time.Now().Unix(),
  332. Status: 1,
  333. RecordTime: advice.RecordDate,
  334. OrgId: orgID,
  335. DrugId: advice.DrugId,
  336. Count: deliver_number,
  337. CountUnit: drup.MinUnit,
  338. }
  339. errTwo := AddSigleDrugAutoReduceRecordInfo(details)
  340. if errTwo != nil {
  341. return errTwo
  342. }
  343. }
  344. // 出库完成后,要减去对应批次的库存数量
  345. // 判断处方里药品单位是拆零单位还是包装单位, 如果是拆零单位,则根据规格,将拆零数量转为包装数量
  346. var maxNumber int64 = 0
  347. var minNumber int64 = 0
  348. if advice.PrescribingNumberUnit == drup.MinUnit {
  349. maxNumber = count / drup.MinNumber
  350. minNumber = count % drup.MinNumber
  351. } else {
  352. maxNumber = count
  353. }
  354. if warehouse.StockMaxNumber < maxNumber {
  355. return errors.New("库存数量不足")
  356. }
  357. warehouse.StockMaxNumber = warehouse.StockMaxNumber - maxNumber
  358. warehouse.Mtime = time.Now().Unix()
  359. if warehouse.StockMinNumber < minNumber {
  360. warehouse.StockMaxNumber = warehouse.StockMaxNumber - 1
  361. warehouse.StockMinNumber = warehouse.StockMinNumber + drup.MinNumber - minNumber
  362. } else {
  363. warehouse.StockMinNumber = warehouse.StockMinNumber - minNumber
  364. }
  365. if warehouse.StockMaxNumber < 0 {
  366. return errors.New("库存数量不足")
  367. }
  368. errThree := UpDateDrugWarehouseInfoByStock(&warehouse)
  369. if errThree != nil {
  370. return errThree
  371. }
  372. return nil
  373. } else {
  374. // 当该批次的库存数量小于出库数量的话,则先把该批次出库完后,再进行递归出库
  375. warehouseOutInfo := &models.DrugWarehouseOutInfo{
  376. WarehouseOutOrderNumber: warehouseout.WarehouseOutOrderNumber,
  377. WarehouseOutId: warehouseout.ID,
  378. Status: 1,
  379. Ctime: time.Now().Unix(),
  380. Remark: "",
  381. OrgId: orgID,
  382. Type: 1,
  383. Manufacturer: 0,
  384. Dealer: 0,
  385. IsSys: 1,
  386. SysRecordTime: advice.RecordDate,
  387. DrugId: advice.DrugId,
  388. }
  389. warehouseOutInfo.Count = stock_number
  390. warehouseOutInfo.CountUnit = drup.MinUnit
  391. errOne := AddSigleDrugWarehouseOutInfo(warehouseOutInfo)
  392. if errOne != nil {
  393. return errOne
  394. } else {
  395. details := &models.DrugAutomaticReduceDetail{
  396. WarehouseOutId: warehouseOutInfo.ID,
  397. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  398. PatientId: advice.PatientId,
  399. Ctime: time.Now().Unix(),
  400. Mtime: time.Now().Unix(),
  401. Status: 1,
  402. RecordTime: advice.RecordDate,
  403. OrgId: orgID,
  404. DrugId: advice.DrugId,
  405. Count: stock_number,
  406. CountUnit: drup.MinUnit,
  407. }
  408. errTwo := AddSigleDrugAutoReduceRecordInfo(details)
  409. if errTwo != nil {
  410. return errTwo
  411. }
  412. }
  413. // 出库完成后,要将该批次库存清零
  414. warehouse.StockMaxNumber = 0
  415. warehouse.StockMinNumber = 0
  416. warehouse.Mtime = time.Now().Unix()
  417. errThree := UpDateDrugWarehouseInfoByStock(&warehouse)
  418. if errThree != nil {
  419. return errThree
  420. }
  421. // 清零完该库存后,还有剩余出库未出完,进行对应的递归操作
  422. prescribingNumber_two_temp := deliver_number - stock_number
  423. overPlusNumber := float64(prescribingNumber_two_temp)
  424. advice.PrescribingNumber = float64(prescribingNumber_two_temp)
  425. advice.PrescribingNumberUnit = drup.MinUnit
  426. HisDrugDeliverInfo(orgID, overPlusNumber, warehouseout, drup, advice)
  427. }
  428. return
  429. }
  430. // 药品出库 递归方式
  431. func AutoDrugDeliverInfo(orgID int64, prescribingNumber int64, warehouseout *models.DrugWarehouseOut, drup *models.BaseDrugLib, advice *models.DrugWarehouseOutInfo) (err error) {
  432. // 判断处方里药品单位是拆零单位还是包装单位, 如果是包装单位,则根据规格,将包装数量转为拆零数量
  433. var deliver_number int64 = 0
  434. var stock_number int64 = 0
  435. fmt.Println("单位1", advice.CountUnit)
  436. fmt.Println("单位2", drup.MaxUnit)
  437. if advice.CountUnit == drup.MaxUnit {
  438. deliver_number = prescribingNumber * drup.MinNumber
  439. } else {
  440. deliver_number = prescribingNumber
  441. }
  442. // 根据先进先出原则,查询最先入库的批次,进行出库
  443. // 如果没有对应的库存,则报错
  444. //开启事物
  445. warehouse, err := FindLastDrugWarehousingInfoByID(advice.DrugId)
  446. if err != nil {
  447. return err
  448. }
  449. // 将该批次的剩余库存数量转换为拆零数量
  450. stock_number = warehouse.StockMaxNumber*drup.MinNumber + warehouse.StockMinNumber
  451. fmt.Println("88i8uu7777777777777777777777777777", stock_number)
  452. fmt.Println("77777544656666666666666666", deliver_number)
  453. // 当库存数量大于或等于出库数量的话,则正常出库该批次
  454. if stock_number >= deliver_number {
  455. warehouseOutInfo := &models.DrugWarehouseOutInfo{
  456. WarehouseOutOrderNumber: warehouseout.WarehouseOutOrderNumber,
  457. WarehouseOutId: warehouseout.ID,
  458. Status: 1,
  459. Ctime: time.Now().Unix(),
  460. Remark: "",
  461. OrgId: orgID,
  462. Type: 1,
  463. Manufacturer: 0,
  464. Dealer: 0,
  465. IsSys: 0,
  466. SysRecordTime: advice.Ctime,
  467. DrugId: advice.DrugId,
  468. }
  469. warehouseOutInfo.Count = deliver_number
  470. warehouseOutInfo.CountUnit = advice.CountUnit
  471. errOne := AddSigleDrugWarehouseOutInfo(warehouseOutInfo)
  472. if errOne != nil {
  473. return errOne
  474. }
  475. // 出库完成后,要减去对应批次的库存数量
  476. // 判断处方里药品单位是拆零单位还是包装单位, 如果是拆零单位,则根据规格,将拆零数量转为包装数量
  477. var maxNumber int64 = 0
  478. var minNumber int64 = 0
  479. if advice.CountUnit == drup.MinUnit {
  480. maxNumber = prescribingNumber / drup.MinNumber
  481. minNumber = prescribingNumber % drup.MinNumber
  482. } else {
  483. maxNumber = prescribingNumber
  484. }
  485. if warehouse.StockMaxNumber < maxNumber {
  486. return errors.New("库存数量不足")
  487. }
  488. warehouse.StockMaxNumber = warehouse.StockMaxNumber - maxNumber
  489. warehouse.Mtime = time.Now().Unix()
  490. if warehouse.StockMinNumber < minNumber {
  491. warehouse.StockMaxNumber = warehouse.StockMaxNumber - 1
  492. warehouse.StockMinNumber = warehouse.StockMinNumber + drup.MinNumber - minNumber
  493. } else {
  494. warehouse.StockMinNumber = warehouse.StockMinNumber - minNumber
  495. }
  496. if warehouse.StockMaxNumber < 0 {
  497. return errors.New("库存数量不足")
  498. }
  499. errThree := UpDateDrugWarehouseInfoByStock(&warehouse)
  500. if errThree != nil {
  501. return errThree
  502. }
  503. return nil
  504. } else {
  505. // 当该批次的库存数量小于出库数量的话,则先把该批次出库完后,再进行递归出库
  506. warehouseOutInfo := &models.DrugWarehouseOutInfo{
  507. WarehouseOutOrderNumber: warehouseout.WarehouseOutOrderNumber,
  508. WarehouseOutId: warehouseout.ID,
  509. Status: 1,
  510. Ctime: time.Now().Unix(),
  511. Remark: "",
  512. OrgId: orgID,
  513. Type: 1,
  514. Manufacturer: 0,
  515. Dealer: 0,
  516. IsSys: 0,
  517. SysRecordTime: advice.Ctime,
  518. DrugId: advice.DrugId,
  519. }
  520. warehouseOutInfo.Count = stock_number
  521. warehouseOutInfo.CountUnit = advice.CountUnit
  522. errOne := AddSigleDrugWarehouseOutInfo(warehouseOutInfo)
  523. if errOne != nil {
  524. return errOne
  525. }
  526. // 出库完成后,要将该批次库存清零
  527. warehouse.StockMaxNumber = 0
  528. warehouse.StockMinNumber = 0
  529. warehouse.Mtime = time.Now().Unix()
  530. errThree := UpDateDrugWarehouseInfoByStock(&warehouse)
  531. if errThree != nil {
  532. return errThree
  533. }
  534. // 清零完该库存后,还有剩余出库未出完,进行对应的递归操作
  535. prescribingNumber_two_temp := deliver_number - stock_number
  536. advice.CountUnit = drup.MinUnit
  537. AutoDrugDeliverInfo(orgID, prescribingNumber_two_temp, warehouseout, drup, advice)
  538. }
  539. return
  540. }
  541. // 耗材出库
  542. func ConsumablesDeliveryTotal(orgID int64, patient_id int64, record_time int64, goods []*models.DialysisBeforePrepareGoods, goodOne []*models.NewDialysisBeforePrepareGoods) (err error) {
  543. //查询该患者当天已经出库的耗材信息
  544. goods_yc, _ := FindConsumablesByDateTwo(orgID, patient_id, record_time)
  545. //for _, item := range goods_yc {
  546. // fmt.Println("已经出库的耗材新消息 -----------------------------------------", item.Count)
  547. //}
  548. //for _, item := range goods {
  549. // fmt.Println("新传进来来的耗材新消息0000000000000000000000000000000000000000000000000", item.Count)
  550. //}
  551. //
  552. //for _,item :=range goodOne{
  553. // fmt.Println("hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh",item.Count)
  554. //}
  555. // 和新请求的出库数据进行对比,分出那些是继续出库的,那些是需要删除出库的
  556. for i := len(goods_yc) - 1; i >= 0; i-- {
  557. goods_yc_temp := goods_yc[i]
  558. for j := len(goods) - 1; j >= 0; j-- {
  559. goods_temp := goods[j]
  560. // 已经出库和新请求出库都存在该耗材,则判断出库数量,分成是继续出库,还是删除出库
  561. if goods_yc_temp.GoodTypeId == goods_temp.GoodTypeId && goods_yc_temp.GoodId == goods_temp.GoodId {
  562. // 已经出库和新请求出库的出库数量一致,则清除两个结构体里的数据(既不出库,也不删除出库)
  563. if goods_yc_temp.Count == goods_temp.Count {
  564. goods_yc = append(goods_yc[:i], goods_yc[i+1:]...)
  565. goods = append(goods[:j], goods[j+1:]...)
  566. break
  567. }
  568. //fmt.Println("998888888888888", goods_yc_temp.Count)
  569. //fmt.Println("7777777777777777", goods_temp.Count)
  570. // 如果已经出库的数量 大于 新请求出库的数量,则代表需要删除出库
  571. if goods_yc_temp.Count > goods_temp.Count {
  572. temp_count := goods_yc_temp.Count - goods_temp.Count
  573. goods_yc[i].Count = temp_count
  574. goods = append(goods[:j], goods[j+1:]...)
  575. break
  576. }
  577. //fmt.Println("555555555555", goods_yc_temp.Count)
  578. //fmt.Println("444444444444", goods_temp.Count)
  579. // 如果已经出库的数量 小于 新请求出库的梳理,则代表需要增加出库
  580. if goods_yc_temp.Count < goods_temp.Count {
  581. temp_count := goods_temp.Count - goods_yc_temp.Count
  582. //fmt.Println("988888888888888", temp_count)
  583. goods[j].Count = temp_count
  584. goods_yc = append(goods_yc[:i], goods_yc[i+1:]...)
  585. //fmt.Println("888888888", goods_yc)
  586. break
  587. }
  588. }
  589. }
  590. }
  591. // goods_yc 这个数据就是需要已经出库了,但是现在需要删除出库的耗材数据
  592. // goods 这个数据就是需要出库的耗材的数据(新增的数据)
  593. fmt.Println("goods222222222222", goods)
  594. fmt.Println("goodsy999999999999", goods_yc)
  595. if len(goods) > 0 {
  596. out, err := FindStockOutByIsSys(orgID, 1, record_time)
  597. if err == gorm.ErrRecordNotFound {
  598. //没有记录,则创建出库单
  599. timeStr := time.Now().Format("2006-01-02")
  600. timeArr := strings.Split(timeStr, "-")
  601. total, _ := FindAllWarehouseOut(orgID)
  602. total = total + 1
  603. warehousing_out_order := strconv.FormatInt(orgID, 10) + timeArr[0] + timeArr[1] + timeArr[2] + "000"
  604. number, _ := strconv.ParseInt(warehousing_out_order, 10, 64)
  605. number = number + total
  606. warehousing_out_order = "CKD" + strconv.FormatInt(number, 10)
  607. warehouseOut := models.WarehouseOut{
  608. WarehouseOutOrderNumber: warehousing_out_order,
  609. OperationTime: time.Now().Unix(),
  610. OrgId: orgID,
  611. Creater: 0,
  612. Ctime: time.Now().Unix(),
  613. Status: 1,
  614. WarehouseOutTime: record_time,
  615. Dealer: 0,
  616. Manufacturer: 0,
  617. Type: 1,
  618. IsSys: 1,
  619. }
  620. err := AddSigleWarehouseOut(&warehouseOut)
  621. if err != nil {
  622. utils.TraceLog("创建出库单失败 err = %v", err)
  623. return err
  624. } else {
  625. out = warehouseOut
  626. }
  627. }
  628. for _, item := range goods {
  629. var newCount int64 = 0
  630. for _, it := range goodOne {
  631. if item.GoodTypeId == it.GoodTypeId && item.GoodId == it.GoodId {
  632. newCount = it.Count
  633. }
  634. }
  635. prepare := models.DialysisBeforePrepare{
  636. GoodTypeId: item.GoodTypeId,
  637. GoodId: item.GoodId,
  638. Count: item.Count,
  639. }
  640. fmt.Println("count0000000000000000000000000000000", newCount)
  641. ConsumablesDelivery(orgID, patient_id, record_time, &prepare, &out, newCount)
  642. }
  643. }
  644. if len(goods_yc) > 0 {
  645. for _, good_yc := range goods_yc {
  646. out, _ := FindStockOutByIsSys(orgID, 1, record_time)
  647. ConsumablesDeliveryDelete(orgID, patient_id, record_time, good_yc, &out)
  648. }
  649. }
  650. return nil
  651. }
  652. //耗材出库
  653. func ConsumablesDelivery(orgID int64, patient_id int64, record_time int64, goods *models.DialysisBeforePrepare, warehouseOut *models.WarehouseOut, count int64) (err error) {
  654. // 判断处方里药品单位是拆零单位还是包装单位, 如果是包装单位,则根据规格,将包装数量转为拆零数量
  655. var deliver_number int64 = 0
  656. var stock_number int64 = 0
  657. var maxNumber int64 = 0
  658. deliver_number = goods.Count
  659. fmt.Println("添加数量", deliver_number)
  660. // 根据先进先出原则,查询最先入库的批次,进行出库
  661. // 如果没有对应的库存,则报错
  662. warehouse, err := FindFirstWarehousingInfoByStock(goods.GoodId, goods.GoodTypeId)
  663. fmt.Println("库存数量", warehouse.StockCount)
  664. if err != nil {
  665. return
  666. }
  667. // 将该批次的剩余库存数量转换为拆零数量
  668. stock_number = warehouse.StockCount
  669. // 当库存数量大于或等于出库数量的话,则正常出库该批次
  670. if stock_number >= deliver_number {
  671. warehouseOutInfo := &models.WarehouseOutInfo{
  672. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  673. WarehouseOutId: warehouseOut.ID,
  674. WarehouseInfotId: warehouse.ID,
  675. Status: 1,
  676. Ctime: time.Now().Unix(),
  677. Remark: "",
  678. OrgId: orgID,
  679. Type: 1,
  680. Manufacturer: 0,
  681. Dealer: 0,
  682. IsSys: 1,
  683. SysRecordTime: record_time,
  684. GoodTypeId: goods.GoodTypeId,
  685. GoodId: goods.GoodId,
  686. PatientId: patient_id,
  687. }
  688. warehouseOutInfo.Count = count
  689. stockInInfo, _ := FindLastStockInInfoRecord(goods.GoodId, orgID)
  690. warehouseOutInfo.Price = stockInInfo.Price
  691. //判断当前耗材是否有出库详情记录
  692. _, errcode := GetStockInRecoredByGoodId(goods.GoodId, goods.GoodTypeId, record_time, patient_id, orgID)
  693. if errcode == gorm.ErrRecordNotFound {
  694. errOne := AddSigleWarehouseOutInfo(warehouseOutInfo)
  695. if errOne != nil {
  696. return errOne
  697. } else {
  698. details := models.BloodAutomaticReduceDetail{
  699. WarehouseOutId: warehouseOutInfo.ID,
  700. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  701. PatientId: patient_id,
  702. Ctime: time.Now().Unix(),
  703. Mtime: time.Now().Unix(),
  704. Status: 1,
  705. RecordTime: record_time,
  706. OrgId: orgID,
  707. GoodId: goods.GoodId,
  708. GoodTypeId: goods.GoodTypeId,
  709. Count: count,
  710. }
  711. //查询当天耗材是否已经存在数据
  712. _, errcode := GetAutoMaticReduceDetail(orgID, patient_id, record_time, goods.GoodId, goods.GoodTypeId)
  713. fmt.Println("errcode00000000000000000000000000000000000000", errcode)
  714. if errcode == gorm.ErrRecordNotFound {
  715. errTwo := CreateAutoReduceRecord(&details)
  716. if errTwo != nil {
  717. return errTwo
  718. }
  719. } else if errcode == nil {
  720. fmt.Println("sql999999999999999999999999999999999999999")
  721. DeleteAutoRedeceDetailTwo(orgID, patient_id, record_time, goods.GoodId, goods.GoodTypeId)
  722. CreateAutoReduceRecord(&details)
  723. }
  724. }
  725. } else if errcode == nil {
  726. warehouseOutInfo := &models.WarehouseOutInfo{
  727. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  728. WarehouseOutId: warehouseOut.ID,
  729. WarehouseInfotId: warehouse.ID,
  730. Status: 1,
  731. Ctime: time.Now().Unix(),
  732. Remark: "",
  733. OrgId: orgID,
  734. Type: 1,
  735. Manufacturer: 0,
  736. Dealer: 0,
  737. IsSys: 1,
  738. SysRecordTime: record_time,
  739. GoodTypeId: goods.GoodTypeId,
  740. GoodId: goods.GoodId,
  741. PatientId: patient_id,
  742. }
  743. fmt.Println("count9999999999999999999999999999999999999", count)
  744. warehouseOutInfo.Count = count
  745. //更新入库详情表
  746. UpdateAutoMaticReduceDetail(goods.GoodId, goods.GoodTypeId, record_time, patient_id, orgID, warehouseOutInfo)
  747. details := models.BloodAutomaticReduceDetail{
  748. WarehouseOutId: warehouseOutInfo.ID,
  749. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  750. PatientId: patient_id,
  751. Ctime: time.Now().Unix(),
  752. Mtime: time.Now().Unix(),
  753. Status: 1,
  754. RecordTime: record_time,
  755. OrgId: orgID,
  756. GoodId: goods.GoodId,
  757. GoodTypeId: goods.GoodTypeId,
  758. Count: count,
  759. }
  760. //查询当天耗材是否已经存在数据
  761. _, errcode := GetAutoMaticReduceDetail(orgID, patient_id, record_time, goods.GoodId, goods.GoodTypeId)
  762. if errcode == gorm.ErrRecordNotFound {
  763. errTwo := CreateAutoReduceRecord(&details)
  764. if errTwo != nil {
  765. return errTwo
  766. }
  767. } else if errcode == nil {
  768. DeleteAutoRedeceDetailTwo(orgID, patient_id, record_time, goods.GoodId, goods.GoodTypeId)
  769. CreateAutoReduceRecord(&details)
  770. }
  771. }
  772. fmt.Println("newcount9999999999999999999999999999999", goods.NewCount)
  773. maxNumber = goods.Count
  774. if warehouse.StockCount < maxNumber {
  775. return errors.New("库存数量不足")
  776. }
  777. warehouse.StockCount = warehouse.StockCount - maxNumber
  778. fmt.Println("999999999999999", warehouse.StockCount)
  779. warehouse.Mtime = time.Now().Unix()
  780. if warehouse.StockCount < 0 {
  781. return errors.New("库存数量不足")
  782. }
  783. errThree := UpDateWarehouseInfoByStock(&warehouse)
  784. if errThree != nil {
  785. return errThree
  786. }
  787. return nil
  788. } else {
  789. // 当改批次的库存数量小于出库数量的话,则先把该批次出库完后,再进行递归出库
  790. // warehouseOutInfo := &models.WarehouseOutInfo{
  791. // WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  792. // WarehouseOutId: warehouseOut.ID,
  793. // WarehouseInfotId: warehouse.ID,
  794. // Status: 1,
  795. // Ctime: time.Now().Unix(),
  796. // Remark: "",
  797. // OrgId: orgID,
  798. // Type: 1,
  799. // Manufacturer: 0,
  800. // Dealer: 0,
  801. // IsSys: 1,
  802. // SysRecordTime: record_time,
  803. // GoodTypeId: goods.GoodTypeId,
  804. // GoodId: goods.GoodId,
  805. // PatientId: patient_id,
  806. // }
  807. // warehouseOutInfo.Count = stock_number
  808. // stockInInfo, _ := service.FindLastStockInInfoRecord(goods.GoodId, orgID)
  809. // warehouseOutInfo.Price = stockInInfo.Price
  810. // cur_warehouse, errcode := GetStockInRecoredByGoodId(goods.GoodId, goods.GoodTypeId, record_time, patient_id, orgID)
  811. // if errcode == gorm.ErrRecordNotFound {
  812. // errOne := service.AddSigleWarehouseOutInfo(warehouseOutInfo)
  813. // if errOne != nil {
  814. // return errOne
  815. // } else {
  816. // details := &models.AutomaticReduceDetail{
  817. // WarehouseOutId: warehouseOutInfo.ID,
  818. // WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  819. // PatientId: patient_id,
  820. // Ctime: time.Now().Unix(),
  821. // Mtime: time.Now().Unix(),
  822. // Status: 1,
  823. // RecordTime: record_time,
  824. // OrgId: orgID,
  825. // GoodId: goods.GoodId,
  826. // GoodTypeId: goods.GoodTypeId,
  827. // Count: goods.Count,
  828. // }
  829. // errTwo := service.AddSigleAutoReduceRecordInfo(details)
  830. // if errTwo != nil {
  831. // return errTwo
  832. // }
  833. // }
  834. // // 出库完成后,要将该批次库存清零
  835. // warehouse.StockCount = 0
  836. // warehouse.Mtime = time.Now().Unix()
  837. // errThree := UpDateWarehouseInfoByStock(&warehouse)
  838. // if errThree != nil {
  839. // return errThree
  840. // }
  841. // // 清零完该库存后,还有剩余出库未出完,进行对应的递归操作
  842. // goods.Count = deliver_number - stock_number
  843. // ConsumablesDelivery(orgID, patient_id, record_time, goods, warehouseOut, count)
  844. // } else if errcode == nil {
  845. // warehouseOutInfo := &models.WarehouseOutInfo{
  846. // WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  847. // WarehouseOutId: warehouseOut.ID,
  848. // WarehouseInfotId: warehouse.ID,
  849. // Status: 1,
  850. // Ctime: time.Now().Unix(),
  851. // Remark: "",
  852. // OrgId: orgID,
  853. // Type: 1,
  854. // Manufacturer: 0,
  855. // Dealer: 0,
  856. // IsSys: 1,
  857. // SysRecordTime: record_time,
  858. // GoodTypeId: goods.GoodTypeId,
  859. // GoodId: goods.GoodId,
  860. // PatientId: patient_id,
  861. // }
  862. // warehouseOutInfo.Count = cur_warehouse.Count + StockCount
  863. // //更新入库详情表
  864. // UpdateAutoMaticReduceDetail(goods.GoodId, goods.GoodTypeId, record_time, patient_id, orgID, warehouseOutInfo)
  865. // details := models.BloodAutomaticReduceDetail{
  866. // WarehouseOutId: warehouseOutInfo.ID,
  867. // WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  868. // PatientId: patient_id,
  869. // Ctime: time.Now().Unix(),
  870. // Mtime: time.Now().Unix(),
  871. // Status: 1,
  872. // RecordTime: record_time,
  873. // OrgId: orgID,
  874. // GoodId: goods.GoodId,
  875. // GoodTypeId: goods.GoodTypeId,
  876. // Count: count,
  877. // }
  878. // //查询当天耗材是否已经存在数据
  879. // _, errcode := GetAutoMaticReduceDetail(orgID, patient_id, record_time, goods.GoodId, goods.GoodTypeId)
  880. // if errcode == gorm.ErrRecordNotFound {
  881. // errTwo := CreateAutoReduceRecord(&details)
  882. // if errTwo != nil {
  883. // return errTwo
  884. // }
  885. // } else if errcode == nil {
  886. // DeleteAutoRedeceDetailTwo(orgID, patient_id, record_time, goods.GoodId, goods.GoodTypeId)
  887. // CreateAutoReduceRecord(&details)
  888. // }
  889. // }
  890. // 出库完成后,要将该批次库存清零
  891. warehouse.StockCount = 0
  892. warehouse.Mtime = time.Now().Unix()
  893. errThree := UpDateWarehouseInfoByStock(&warehouse)
  894. if errThree != nil {
  895. return errThree
  896. }
  897. // 清零完该库存后,还有剩余出库未出完,进行对应的递归操作
  898. goods.Count = deliver_number - stock_number
  899. ConsumablesDelivery(orgID, patient_id, record_time, goods, warehouseOut, count)
  900. }
  901. return nil
  902. }
  903. //耗材出库删除
  904. func ConsumablesDeliveryDelete(orgID int64, patient_id int64, record_time int64, good_yc *models.DialysisBeforePrepare, warehouseOut *models.WarehouseOut) (err error) {
  905. // 先根据相关信息查询当天该耗材的出库信息
  906. warehouseOutInfos, err := FindStockOutInfoByStock(orgID, good_yc.GoodTypeId, good_yc.GoodId, record_time, patient_id)
  907. if err != nil {
  908. return err
  909. }
  910. var delete_count int64 = 0
  911. for _, ware := range warehouseOutInfos {
  912. // 判断当前出库的数据和删除出库数量
  913. if good_yc.Count <= ware.Count {
  914. delete_count = good_yc.Count
  915. } else {
  916. delete_count = ware.Count
  917. }
  918. // 在出库记录表里记录退库详情
  919. warehouseOutInfo := &models.WarehouseOutInfo{
  920. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  921. WarehouseOutId: warehouseOut.ID,
  922. //WarehouseInfoId: warehouse.ID,
  923. Status: 1,
  924. Ctime: time.Now().Unix(),
  925. Remark: "",
  926. OrgId: orgID,
  927. Type: 1,
  928. Manufacturer: 0,
  929. Dealer: 0,
  930. IsSys: 2,
  931. SysRecordTime: record_time,
  932. GoodTypeId: good_yc.GoodTypeId,
  933. GoodId: good_yc.GoodId,
  934. PatientId: patient_id,
  935. }
  936. warehouseOutInfo.Count = ware.Count - delete_count
  937. stockInInfo, _ := FindLastStockInInfoRecord(good_yc.GoodId, orgID)
  938. warehouseOutInfo.Price = stockInInfo.Price
  939. errOne := UpdateAutoMaticReduceDetail(good_yc.GoodId, good_yc.GoodTypeId, record_time, patient_id, orgID, warehouseOutInfo)
  940. if errOne != nil {
  941. return errOne
  942. } else {
  943. details := &models.AutomaticReduceDetail{
  944. WarehouseOutId: warehouseOutInfo.ID,
  945. WarehouseOutOrderNumber: warehouseOutInfo.WarehouseOutOrderNumber,
  946. PatientId: patient_id,
  947. Ctime: time.Now().Unix(),
  948. Mtime: time.Now().Unix(),
  949. Status: 1,
  950. RecordTime: record_time,
  951. OrgId: orgID,
  952. GoodId: good_yc.GoodId,
  953. GoodTypeId: good_yc.GoodTypeId,
  954. Count: warehouseOutInfo.Count,
  955. Type: 2,
  956. }
  957. //查询是否当天已经存在数据
  958. _, errcode := GetAutoMaticReduceDetail(orgID, patient_id, record_time, good_yc.GoodId, good_yc.GoodTypeId)
  959. if errcode == gorm.ErrRecordNotFound {
  960. errTwo := AddSigleAutoReduceRecordInfo(details)
  961. if errTwo != nil {
  962. return errTwo
  963. }
  964. } else if errcode == nil {
  965. DeleteAutoRedeceDetailTwo(orgID, patient_id, record_time, good_yc.GoodId, good_yc.GoodTypeId)
  966. AddSigleAutoReduceRecordInfo(details)
  967. }
  968. }
  969. // 删除出库完成后,要增加对应批次的库存数量
  970. errThree := UpDateWarehouseInfoByStockDelete(ware.WarehouseInfotId, delete_count)
  971. if errThree != nil {
  972. return errThree
  973. }
  974. // 增加了对应的库存后,看看还有多少需要退库的
  975. good_yc.Count = good_yc.Count - delete_count
  976. if good_yc.Count == 0 {
  977. return nil
  978. }
  979. }
  980. if good_yc.Count == 0 {
  981. return nil
  982. } else {
  983. return errors.New("退库和出库数据不匹配")
  984. }
  985. }
  986. func GetAutoReduceRecordInfoByPatientId(orgid int64, patient_id int64, recordTime int64) (autoReduce []*models.AutomaticReduceDetail, err error) {
  987. err = XTReadDB().Model(&autoReduce).Where("org_id = ? and patient_id = ? and record_time = ? and status = 1", orgid, patient_id, recordTime).Find(&autoReduce).Error
  988. return autoReduce, err
  989. }
  990. func DeleteDialysisBefor(orgid int64, patient_id int64, record_date int64, goodid int64, goodtypeid int64) error {
  991. prepare := models.DialysisBeforePrepare{}
  992. err := XTWriteDB().Model(&prepare).Where("user_org_id = ? and patient_id = ? and record_date = ? and status = 1 and good_id = ? and good_type_id = ?", orgid, patient_id, record_date, goodid, goodtypeid).Updates(map[string]interface{}{"status": 0, "count": 0, "mtime": time.Now().Unix()}).Error
  993. return err
  994. }
  995. func DeleteDialysisBeforThree(orgid int64, patient_id int64, record_date int64) error {
  996. prepare := models.DialysisBeforePrepare{}
  997. err := XTWriteDB().Model(&prepare).Where("user_org_id = ? and patient_id = ? and record_date = ? and status = 1", orgid, patient_id, record_date).Updates(map[string]interface{}{"status": 0, "count": 0, "mtime": time.Now().Unix()}).Error
  998. return err
  999. }
  1000. func ConsumablesDeliveryTotalOne(orgID int64, patient_id int64, record_time int64, goods []*models.DialysisBeforePrepareGoods, goodOne []*models.NewDialysisBeforePrepareGoods) (err error) {
  1001. //查询该患者当天已经出库的耗材信息
  1002. goods_yc, _ := FindConsumablesByDateTwo(orgID, patient_id, record_time)
  1003. for _, item := range goods_yc {
  1004. fmt.Println("item0000000000000", item.Count)
  1005. }
  1006. for _, item := range goods {
  1007. fmt.Println("item1111111111111", item.Count)
  1008. }
  1009. // 和新请求的出库数据进行对比,分出那些是继续出库的,那些是需要删除出库的
  1010. for i := len(goods_yc) - 1; i >= 0; i-- {
  1011. goods_yc_temp := goods_yc[i]
  1012. for j := len(goods) - 1; j >= 0; j-- {
  1013. goods_temp := goods[j]
  1014. // 已经出库和新请求出库都存在该耗材,则判断出库数量,分成是继续出库,还是删除出库
  1015. if goods_yc_temp.GoodTypeId == goods_temp.GoodTypeId && goods_yc_temp.GoodId == goods_temp.GoodId {
  1016. // 已经出库和新请求出库的出库数量一致,则清除两个结构体里的数据(既不出库,也不删除出库)
  1017. if goods_yc_temp.Count == goods_temp.Count {
  1018. goods_yc = append(goods_yc[:i], goods_yc[i+1:]...)
  1019. goods = append(goods[:j], goods[j+1:]...)
  1020. }
  1021. fmt.Println("998888888888888", goods_yc_temp.Count)
  1022. fmt.Println("7777777777777777", goods_temp.Count)
  1023. // 如果已经出库的数量 大于 新请求出库的数量,则代表需要删除出库
  1024. if goods_yc_temp.Count > goods_temp.Count {
  1025. fmt.Println("进来了是吗")
  1026. temp_count := goods_yc_temp.Count - goods_temp.Count
  1027. goods_yc[i].Count = temp_count
  1028. goods = append(goods[:j], goods[j+1:]...)
  1029. break
  1030. }
  1031. fmt.Println("555555555555", goods_yc_temp.Count)
  1032. fmt.Println("444444444444", goods_temp.Count)
  1033. // 如果已经出库的数量 小于 新请求出库的梳理,则代表需要增加出库
  1034. if goods_yc_temp.Count < goods_temp.Count {
  1035. temp_count := goods_temp.Count - goods_yc_temp.Count
  1036. fmt.Println("988888888888888", temp_count)
  1037. goods[j].Count = temp_count
  1038. goods_yc = append(goods_yc[:i], goods_yc[i+1:]...)
  1039. fmt.Println("888888888", goods_yc)
  1040. }
  1041. }
  1042. }
  1043. }
  1044. // goods_yc 这个数据就是需要已经出库了,但是现在需要删除出库的耗材数据
  1045. // goods 这个数据就是需要出库的耗材的数据
  1046. fmt.Println("goods222222222222", goods)
  1047. fmt.Println("goodsy999999999999", goods_yc)
  1048. if len(goods) > 0 {
  1049. out, err := FindStockOutByIsSys(orgID, 1, record_time)
  1050. if err == gorm.ErrRecordNotFound {
  1051. //没有记录,则创建出库单
  1052. timeStr := time.Now().Format("2006-01-02")
  1053. timeArr := strings.Split(timeStr, "-")
  1054. total, _ := FindAllWarehouseOut(orgID)
  1055. total = total + 1
  1056. warehousing_out_order := strconv.FormatInt(orgID, 10) + timeArr[0] + timeArr[1] + timeArr[2] + "000"
  1057. number, _ := strconv.ParseInt(warehousing_out_order, 10, 64)
  1058. number = number + total
  1059. warehousing_out_order = "CKD" + strconv.FormatInt(number, 10)
  1060. warehouseOut := models.WarehouseOut{
  1061. WarehouseOutOrderNumber: warehousing_out_order,
  1062. OperationTime: time.Now().Unix(),
  1063. OrgId: orgID,
  1064. Creater: 0,
  1065. Ctime: time.Now().Unix(),
  1066. Status: 1,
  1067. WarehouseOutTime: record_time,
  1068. Dealer: 0,
  1069. Manufacturer: 0,
  1070. Type: 1,
  1071. IsSys: 1,
  1072. }
  1073. err := AddSigleWarehouseOut(&warehouseOut)
  1074. if err != nil {
  1075. utils.TraceLog("创建出库单失败 err = %v", err)
  1076. return err
  1077. } else {
  1078. out = warehouseOut
  1079. }
  1080. }
  1081. for _, item := range goods {
  1082. for _, it := range goodOne {
  1083. if item.GoodTypeId == it.GoodTypeId && it.GoodId == it.GoodId {
  1084. it.NewCount = item.Count
  1085. }
  1086. }
  1087. }
  1088. for _, item := range goodOne {
  1089. //prepare := models.DialysisBeforePrepare{
  1090. // GoodTypeId: item.GoodTypeId,
  1091. // GoodId: item.GoodId,
  1092. // Count: item.Count,
  1093. // NewCount: item.NewCount,
  1094. //}
  1095. info := models.WarehouseOutInfo{
  1096. GoodTypeId: item.GoodTypeId,
  1097. GoodId: item.GoodId,
  1098. Count: item.Count,
  1099. }
  1100. ConsumablesDeliveryOne(orgID, record_time, &info, &out, item.Count)
  1101. }
  1102. }
  1103. if len(goods_yc) > 0 {
  1104. for _, good_yc := range goods_yc {
  1105. out, _ := FindStockOutByIsSys(orgID, 1, record_time)
  1106. ConsumablesDeliveryDeleteOne(orgID, record_time, good_yc, &out)
  1107. }
  1108. }
  1109. return nil
  1110. }
  1111. //耗材出库
  1112. func ConsumablesDeliveryOne(orgID int64, record_time int64, goods *models.WarehouseOutInfo, warehouseOut *models.WarehouseOut, count int64) (err error) {
  1113. // 判断处方里药品单位是拆零单位还是包装单位, 如果是包装单位,则根据规格,将包装数量转为拆零数量
  1114. var deliver_number int64 = 0
  1115. var stock_number int64 = 0
  1116. var maxNumber int64 = 0
  1117. deliver_number = goods.Count
  1118. fmt.Println("添加数量", deliver_number)
  1119. // 根据先进先出原则,查询最先入库的批次,进行出库
  1120. // 如果没有对应的库存,则报错
  1121. warehouse, err := FindFirstWarehousingInfoByStock(goods.GoodId, goods.GoodTypeId)
  1122. if err != nil {
  1123. return err
  1124. }
  1125. // 将该批次的剩余库存数量转换为拆零数量
  1126. stock_number = warehouse.StockCount
  1127. // 当库存数量大于或等于出库数量的话,则正常出库该批次
  1128. if stock_number >= deliver_number {
  1129. warehouseOutInfo := &models.WarehouseOutInfo{
  1130. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  1131. WarehouseOutId: warehouseOut.ID,
  1132. WarehouseInfotId: warehouse.ID,
  1133. Status: 1,
  1134. Ctime: time.Now().Unix(),
  1135. Remark: "",
  1136. OrgId: orgID,
  1137. Type: 1,
  1138. Manufacturer: 0,
  1139. Dealer: 0,
  1140. IsSys: 0,
  1141. SysRecordTime: record_time,
  1142. GoodTypeId: goods.GoodTypeId,
  1143. GoodId: goods.GoodId,
  1144. }
  1145. warehouseOutInfo.Count = goods.Count
  1146. stockInInfo, _ := FindLastStockInInfoRecord(goods.GoodId, orgID)
  1147. warehouseOutInfo.Price = stockInInfo.Price
  1148. //判断当前耗材是否有出库详情记录
  1149. _, errcode := GetStockInRecoredByGoodIdOne(goods.GoodId, goods.GoodTypeId, record_time, orgID)
  1150. if errcode == gorm.ErrRecordNotFound {
  1151. errOne := AddSigleWarehouseOutInfo(warehouseOutInfo)
  1152. if errOne != nil {
  1153. return errOne
  1154. }
  1155. } else if errcode == nil {
  1156. warehouseOutInfo := &models.WarehouseOutInfo{
  1157. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  1158. WarehouseOutId: warehouseOut.ID,
  1159. WarehouseInfotId: warehouse.ID,
  1160. Status: 1,
  1161. Ctime: time.Now().Unix(),
  1162. Remark: "",
  1163. OrgId: orgID,
  1164. Type: 1,
  1165. Manufacturer: 0,
  1166. Dealer: 0,
  1167. IsSys: 0,
  1168. SysRecordTime: record_time,
  1169. GoodTypeId: goods.GoodTypeId,
  1170. GoodId: goods.GoodId,
  1171. }
  1172. warehouseOutInfo.Count = goods.Count
  1173. }
  1174. maxNumber = goods.Count
  1175. if warehouse.StockCount < maxNumber {
  1176. return errors.New("库存数量不足")
  1177. }
  1178. warehouse.StockCount = warehouse.StockCount - maxNumber
  1179. warehouse.Mtime = time.Now().Unix()
  1180. if warehouse.StockCount < 0 {
  1181. return errors.New("库存数量不足")
  1182. }
  1183. errThree := UpDateWarehouseInfoByStock(&warehouse)
  1184. if errThree != nil {
  1185. return errThree
  1186. }
  1187. return nil
  1188. } else {
  1189. // 当改批次的库存数量小于出库数量的话,则先把该批次出库完后,再进行递归出库
  1190. warehouseOutInfo := &models.WarehouseOutInfo{
  1191. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  1192. WarehouseOutId: warehouseOut.ID,
  1193. WarehouseInfotId: warehouse.ID,
  1194. Status: 1,
  1195. Ctime: time.Now().Unix(),
  1196. Remark: "",
  1197. OrgId: orgID,
  1198. Type: 1,
  1199. Manufacturer: 0,
  1200. Dealer: 0,
  1201. IsSys: 0,
  1202. SysRecordTime: record_time,
  1203. GoodTypeId: goods.GoodTypeId,
  1204. GoodId: goods.GoodId,
  1205. }
  1206. warehouseOutInfo.Count = stock_number
  1207. stockInInfo, _ := service.FindLastStockInInfoRecord(goods.GoodId, orgID)
  1208. warehouseOutInfo.Price = stockInInfo.Price
  1209. _, errcode := GetStockInRecoredByGoodIdOne(goods.GoodId, goods.GoodTypeId, record_time, orgID)
  1210. if errcode == gorm.ErrRecordNotFound {
  1211. errOne := service.AddSigleWarehouseOutInfo(warehouseOutInfo)
  1212. if errOne != nil {
  1213. return errOne
  1214. }
  1215. // 出库完成后,要将该批次库存清零
  1216. warehouse.StockCount = 0
  1217. warehouse.Mtime = time.Now().Unix()
  1218. errThree := UpDateWarehouseInfoByStock(&warehouse)
  1219. if errThree != nil {
  1220. return errThree
  1221. }
  1222. // 清零完该库存后,还有剩余出库未出完,进行对应的递归操作
  1223. goods.Count = deliver_number - stock_number
  1224. ConsumablesDeliveryOne(orgID, record_time, goods, warehouseOut, count)
  1225. } else if errcode == nil {
  1226. warehouseOutInfo := &models.WarehouseOutInfo{
  1227. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  1228. WarehouseOutId: warehouseOut.ID,
  1229. WarehouseInfotId: warehouse.ID,
  1230. Status: 1,
  1231. Ctime: time.Now().Unix(),
  1232. Remark: "",
  1233. OrgId: orgID,
  1234. Type: 1,
  1235. Manufacturer: 0,
  1236. Dealer: 0,
  1237. IsSys: 0,
  1238. SysRecordTime: record_time,
  1239. GoodTypeId: goods.GoodTypeId,
  1240. GoodId: goods.GoodId,
  1241. }
  1242. warehouseOutInfo.Count = goods.Count
  1243. //更新入库详情表
  1244. UpdateAutoMaticReduceDetailOne(goods.GoodId, goods.GoodTypeId, record_time, orgID, warehouseOutInfo)
  1245. }
  1246. }
  1247. return nil
  1248. }
  1249. //耗材出库删除
  1250. func ConsumablesDeliveryDeleteOne(orgID int64, record_time int64, good_yc *models.DialysisBeforePrepare, warehouseOut *models.WarehouseOut) (err error) {
  1251. // 先根据相关信息查询当天该耗材的出库信息
  1252. warehouseOutInfos, err := FindStockOutInfoByStockOne(orgID, good_yc.GoodTypeId, good_yc.GoodId, record_time)
  1253. if err != nil {
  1254. return err
  1255. }
  1256. var delete_count int64 = 0
  1257. for _, ware := range warehouseOutInfos {
  1258. // 判断当前出库的数据和删除出库数量
  1259. if good_yc.Count <= ware.Count {
  1260. delete_count = good_yc.Count
  1261. } else {
  1262. delete_count = ware.Count
  1263. }
  1264. // 在出库记录表里记录退库详情
  1265. warehouseOutInfo := &models.WarehouseOutInfo{
  1266. WarehouseOutOrderNumber: warehouseOut.WarehouseOutOrderNumber,
  1267. WarehouseOutId: warehouseOut.ID,
  1268. //WarehouseInfoId: warehouse.ID,
  1269. Status: 1,
  1270. Ctime: time.Now().Unix(),
  1271. Remark: "",
  1272. OrgId: orgID,
  1273. Type: 1,
  1274. Manufacturer: 0,
  1275. Dealer: 0,
  1276. IsSys: 0,
  1277. SysRecordTime: record_time,
  1278. GoodTypeId: good_yc.GoodTypeId,
  1279. GoodId: good_yc.GoodId,
  1280. }
  1281. warehouseOutInfo.Count = delete_count
  1282. stockInInfo, _ := FindLastStockInInfoRecord(good_yc.GoodId, orgID)
  1283. warehouseOutInfo.Price = stockInInfo.Price
  1284. errOne := AddSigleWarehouseOutInfo(warehouseOutInfo)
  1285. if errOne != nil {
  1286. return errOne
  1287. }
  1288. // 删除出库完成后,要增加对应批次的库存数量
  1289. errThree := UpDateWarehouseInfoByStockDelete(ware.WarehouseInfotId, delete_count)
  1290. if errThree != nil {
  1291. return errThree
  1292. }
  1293. // 增加了对应的库存后,看看还有多少需要退库的
  1294. good_yc.Count = good_yc.Count - delete_count
  1295. if good_yc.Count == 0 {
  1296. return nil
  1297. }
  1298. }
  1299. if good_yc.Count == 0 {
  1300. return nil
  1301. } else {
  1302. return errors.New("退库和出库数据不匹配")
  1303. }
  1304. }