schedule.py 44 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. from fastapi import APIRouter, Request, Depends, Query, HTTPException, status, BackgroundTasks
  4. from common.security import valid_access_token
  5. from fastapi.responses import JSONResponse, FileResponse,StreamingResponse
  6. from sqlalchemy.orm import Session
  7. from sqlalchemy import and_, or_
  8. from sqlalchemy.sql import func
  9. from common.auth_user import *
  10. from pydantic import BaseModel
  11. from common.db import db_dept
  12. from common.db import db_czrz
  13. from exceptions import AppException, HmacException
  14. from database import get_db
  15. from typing import List
  16. from models import *
  17. import xlrd
  18. from utils import *
  19. from utils.ry_system_util import *
  20. from utils.video_util import *
  21. import traceback
  22. import os
  23. import json
  24. router = APIRouter()
  25. def name_get_user_id(db,keywords,islike=True):
  26. query = db.query(EmergencyContactUser)
  27. query = query.filter(EmergencyContactUser.del_flag != '2')
  28. if islike:
  29. query = query.filter(EmergencyContactUser.name.like(f"%{keywords}%"))
  30. else:
  31. query = query.filter(EmergencyContactUser.name==keywords)
  32. data = query.all()
  33. return [info.id for info in data]
  34. def user_id_get_duty_id(db,user_id_list):
  35. query = db.query(DutyPersonnelArrangement)
  36. query = query.filter(DutyPersonnelArrangement.del_flag != '2')
  37. query = query.filter(DutyPersonnelArrangement.personnel_id.in_(user_id_list))
  38. data = query.all()
  39. return [info.duty_id for info in data]
  40. def duty_id_get_user_id(db,duty_id):
  41. query = db.query(DutyPersonnelArrangement)
  42. query = query.filter(DutyPersonnelArrangement.del_flag != '2')
  43. query = query.filter(DutyPersonnelArrangement.duty_id==duty_id)
  44. data = query.all()
  45. return data
  46. def user_id_get_info(db,user_id):
  47. query = db.query(EmergencyContactUser)
  48. query = query.filter(EmergencyContactUser.del_flag != '2')
  49. query = query.filter(EmergencyContactUser.id==user_id)
  50. data = query.first()
  51. return data
  52. def position_id_get_info(db,position_id):
  53. query = db.query(DutyPosition)
  54. query = query.filter(DutyPosition.del_flag != '2')
  55. query = query.filter(DutyPosition.id==position_id)
  56. data = query.first()
  57. return data
  58. def dept_id_get_info(db,dept_id):
  59. query = db.query(EmergencyContactDepartment)
  60. query = query.filter(EmergencyContactDepartment.del_flag != '2')
  61. query = query.filter(EmergencyContactDepartment.id==dept_id)
  62. data = query.first()
  63. return data
  64. def get_duty_unit_id_list(db):
  65. query = db.query(SysDictData)
  66. query = query.filter(SysDictData.del_flag != '2')
  67. query = query.filter(SysDictData.dict_type=='duty_unit')
  68. data = query.all()
  69. return [int(i.dict_value) for i in data]
  70. def dept_id_get_duty_unit_id(db,dept_id,duty_unit_id_list):
  71. if dept_id == 0:
  72. return 0
  73. if dept_id in duty_unit_id_list:
  74. return dept_id
  75. query = db.query(EmergencyContactDepartment)
  76. query = query.filter(EmergencyContactDepartment.del_flag != '2')
  77. query = query.filter(EmergencyContactDepartment.id==dept_id)
  78. data = query.first()
  79. if data:
  80. dept_id_get_duty_unit_id(db, data.parent_department_id, duty_unit_id_list)
  81. else:
  82. return 0
  83. def mobile_phone_get_dept_id(db,mobile_phone,duty_unit_id_list):
  84. query = db.query(EmergencyContactUser)
  85. query = query.filter(EmergencyContactUser.del_flag != '2')
  86. query = query.filter(EmergencyContactUser.mobile_phone.like(f'%{mobile_phone}%'))
  87. data = query.first()
  88. if data:
  89. dept_id = data.department_id
  90. dept_id = dept_id_get_duty_unit_id(db, dept_id, duty_unit_id_list)
  91. return dept_id
  92. else:
  93. return 0
  94. @router.get('/list')
  95. async def get_dict_data_by_type(
  96. keywords:str =Query(None),
  97. duty_type:str =Query(None),
  98. start_time:str =Query(None),
  99. end_time:str =Query(None),
  100. duty_unit:int =Query(None),
  101. page: int = Query(1, gt=0),
  102. pageSize: int = Query(10, gt=0),
  103. db: Session = Depends(get_db),
  104. body = Depends(remove_xss_json),
  105. user_id = Depends(valid_access_token)
  106. ):
  107. try:
  108. # 根据 dict_type 查询字典数据
  109. # dict_data = db.query(SysDictData).filter_by(dict_type=dictType).all()
  110. query = db.query(DutySchedule)
  111. query = query.filter(DutySchedule.del_flag != '2')
  112. # 添加查询条件
  113. if keywords:
  114. user_list= name_get_user_id(db,keywords)
  115. duty_list = user_id_get_duty_id(db,user_list)
  116. query = query.filter(DutySchedule.id.in_(duty_list))
  117. if duty_type:
  118. query = query.filter(DutySchedule.duty_type==duty_type)
  119. if start_time:
  120. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  121. query = query.filter(DutySchedule.duty_date>=start_time)
  122. if end_time:
  123. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  124. query = query.filter(DutySchedule.duty_date<=end_time)
  125. if duty_unit:
  126. query = query.filter(DutySchedule.duty_unit==duty_unit)
  127. # 计算总记录数
  128. total_count = query.count()
  129. # 计算分页
  130. offset = (page - 1) * pageSize
  131. query = query.order_by(DutySchedule.start_time.desc(),DutySchedule.end_time.desc(),DutySchedule.update_time.desc())
  132. duty_data = query.offset(offset).limit(pageSize).all()
  133. # 转换为字典
  134. data_list = []
  135. for d in duty_data:
  136. user_data = []
  137. user_list = duty_id_get_user_id(db,d.id)
  138. for user_info in user_list:
  139. contact_info = user_id_get_info(db, user_info.personnel_id)
  140. user_data.append({"position_id":user_info.position_id,
  141. "personnel_id":user_info.personnel_id,
  142. "name":contact_info.name,
  143. "position":contact_info.position,
  144. "mobile_phone":contact_info.mobile_phone,
  145. "office_phone":contact_info.office_phone,
  146. "department_id":contact_info.department_id,
  147. "yzy_userid":contact_info.userid})
  148. data_list.append({
  149. "id": d.id,
  150. "start_time": d.start_time.strftime('%Y-%m-%d %H:%M:%S') if d.start_time else '',
  151. "end_time":d.end_time.strftime('%Y-%m-%d %H:%M:%S') if d.end_time else '',
  152. "duty_date": d.duty_date,
  153. "shift_type":d.shift_type,
  154. "duty_unit": d.duty_unit,
  155. "duty_type": d.duty_type,
  156. "createTime": d.create_time.strftime('%Y-%m-%d %H:%M:%S') if d.create_time else '',
  157. "user_data":user_data
  158. })
  159. # 构建返回结果
  160. result = {
  161. "total": total_count,
  162. "page": page,
  163. "pageSize": pageSize,
  164. "totalPages": (total_count + pageSize - 1) // pageSize,
  165. "data": data_list,
  166. "code": 200,
  167. "msg": "查询成功"
  168. }
  169. return result
  170. except Exception as e:
  171. # 处理异常
  172. traceback.print_exc()
  173. return JSONResponse(status_code=404, content={
  174. 'code': 404,
  175. 'msg': str(e)
  176. })
  177. @router.get('/latest_by_group')
  178. async def get_dict_data_by_type(
  179. keywords:str =Query(None),
  180. start_time:str =Query(None),
  181. end_time:str =Query(None),
  182. duty_unit:int =Query(None),
  183. page: int = Query(1, gt=0),
  184. pageSize: int = Query(10, gt=0),
  185. db: Session = Depends(get_db),
  186. body = Depends(remove_xss_json),
  187. user_id = Depends(valid_access_token)
  188. ):
  189. try:
  190. # 根据 dict_type 查询字典数据
  191. # dict_data = db.query(SysDictData).filter_by(dict_type=dictType).all()
  192. query = db.query(DutySchedule)
  193. query = query.filter(DutySchedule.del_flag != '2')
  194. # 添加查询条件
  195. if keywords:
  196. user_list= name_get_user_id(db,keywords)
  197. duty_list = user_id_get_duty_id(db,user_list)
  198. query = query.filter(DutySchedule.id.in_(duty_list))
  199. if start_time:
  200. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  201. query = query.filter(DutySchedule.duty_date>=start_time)
  202. if end_time:
  203. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  204. query = query.filter(DutySchedule.duty_date<=end_time)
  205. if duty_unit:
  206. query = query.filter(DutySchedule.duty_unit==duty_unit)
  207. else:
  208. now_user_info=user_id_get_user_info(db,user_id)
  209. print(user_id,now_user_info.phonenumber,get_duty_unit_id_list(db))
  210. if now_user_info:
  211. query = query.filter(DutySchedule.duty_unit==mobile_phone_get_dept_id(db,mpfun.dec_data(now_user_info.phonenumber),get_duty_unit_id_list(db)))
  212. else:
  213. return {
  214. "total": 0,
  215. "page": page,
  216. "pageSize": pageSize,
  217. "totalPages": (0 + pageSize - 1) // pageSize,
  218. "data": [],
  219. "code": 200,
  220. "msg": "查询成功"
  221. }
  222. subquery = db.query(
  223. DutySchedule.duty_date,
  224. DutySchedule.shift_type,
  225. func.max(DutySchedule.update_time).label("latest_create_time")
  226. ).filter(DutySchedule.del_flag != '2').group_by(
  227. DutySchedule.duty_date,
  228. DutySchedule.shift_type
  229. ).subquery()
  230. query = query.join(
  231. subquery,
  232. and_(
  233. DutySchedule.duty_date == subquery.c.duty_date,
  234. DutySchedule.shift_type == subquery.c.shift_type,
  235. DutySchedule.update_time == subquery.c.latest_create_time
  236. )
  237. )
  238. # 计算总记录数
  239. total_count = query.count()
  240. # 计算分页
  241. offset = (page - 1) * pageSize
  242. query = query.order_by(DutySchedule.update_time.desc())
  243. print(query)
  244. duty_data = query.offset(offset).limit(pageSize).all()
  245. # 转换为字典
  246. data_list = []
  247. for d in duty_data:
  248. user_data = []
  249. user_list = duty_id_get_user_id(db,d.id)
  250. for user_info in user_list:
  251. contact_info = user_id_get_info(db, user_info.personnel_id)
  252. user_data.append({"position_id":user_info.position_id,
  253. "personnel_id":user_info.personnel_id,
  254. "name":contact_info.name,
  255. "position":contact_info.position,
  256. "mobile_phone":contact_info.mobile_phone,
  257. "office_phone":contact_info.office_phone,
  258. "department_id":contact_info.department_id,
  259. "yzy_userid":contact_info.userid})
  260. data_list.append({
  261. "id": d.id,
  262. "start_time": d.start_time.strftime('%Y-%m-%d %H:%M:%S') if d.start_time else '',
  263. "end_time":d.end_time.strftime('%Y-%m-%d %H:%M:%S') if d.end_time else '',
  264. "duty_date": d.duty_date,
  265. "shift_type":d.shift_type,
  266. "duty_unit": d.duty_unit,
  267. "duty_type": d.duty_type,
  268. "createTime": d.create_time.strftime('%Y-%m-%d %H:%M:%S') if d.create_time else '',
  269. "user_data":user_data
  270. })
  271. # 构建返回结果
  272. result = {
  273. "total": total_count,
  274. "page": page,
  275. "pageSize": pageSize,
  276. "totalPages": (total_count + pageSize - 1) // pageSize,
  277. "data": data_list,
  278. "code": 200,
  279. "msg": "查询成功"
  280. }
  281. return result
  282. except Exception as e:
  283. # 处理异常
  284. traceback.print_exc()
  285. return JSONResponse(status_code=404, content={
  286. 'code': 404,
  287. 'msg': str(e)
  288. })
  289. @router.get('/info/{id}')
  290. async def get_dict_data_by_type(
  291. id: str ,
  292. db: Session = Depends(get_db),
  293. body = Depends(remove_xss_json),
  294. user_id = Depends(valid_access_token)
  295. ):
  296. try:
  297. # 根据 dict_type 查询字典数据
  298. # dict_data = dict_type_get_dict_data_info(db,'three_proofing')
  299. query = db.query(DutySchedule)
  300. # 添加查询条件
  301. # if dictType:
  302. query = query.filter(DutySchedule.id==id)
  303. query = query.filter(DutySchedule.del_flag != '2')
  304. d = query.first()
  305. user_data = []
  306. user_list = duty_id_get_user_id(db, d.id)
  307. for user_info in user_list:
  308. contact_info = user_id_get_info(db, user_info.personnel_id)
  309. user_data.append({"position_id": user_info.position_id,
  310. "personnel_id": user_info.personnel_id,
  311. "name": contact_info.name,
  312. "position": contact_info.position,
  313. "mobile_phone": contact_info.mobile_phone,
  314. "office_phone": contact_info.office_phone,
  315. "department_id": contact_info.department_id,
  316. "yzy_userid": contact_info.userid})
  317. data_list={
  318. "id": d.id,
  319. "start_time": d.start_time,
  320. "end_time": d.end_time,
  321. "duty_date": d.duty_date,
  322. "shift_type": d.shift_type,
  323. "duty_unit": d.duty_unit,
  324. "duty_type": d.duty_type,
  325. "createTime": d.create_time.strftime('%Y-%m-%d %H:%M:%S') if d.create_time else '',
  326. "user_data": user_data
  327. }
  328. # 构建返回结果
  329. result = {
  330. "data": data_list,
  331. "code": 200,
  332. "msg": "查询成功"
  333. }
  334. return result
  335. except Exception as e:
  336. # 处理异常
  337. traceback.print_exc()
  338. return JSONResponse(status_code=404, content={
  339. 'code': 404,
  340. 'msg': str(e)
  341. })
  342. @router.post('/create')
  343. async def create_dict_data(
  344. db: Session = Depends(get_db),
  345. body = Depends(remove_xss_json),
  346. user_id = Depends(valid_access_token)
  347. ):
  348. try:
  349. # 创建一个新的 SysDictData 实例
  350. user_data = body['user_data']
  351. if len(user_data)==0:
  352. return JSONResponse(status_code=404, content={
  353. 'errcode': 404,
  354. 'errmsg': '值班人员不能为空'
  355. })
  356. new_duty_data = DutySchedule(
  357. start_time=body['start_time'],
  358. end_time=body['end_time'],
  359. duty_date=body['duty_date'],
  360. shift_type=body['shift_type'],
  361. duty_unit=body['duty_unit'],
  362. duty_type=body['duty_type'],
  363. create_by=user_id
  364. )
  365. # 添加到会话并提交
  366. db.add(new_duty_data)
  367. db.commit()
  368. db.refresh(new_duty_data)
  369. user_list = []
  370. for user_info in user_data:
  371. user_list.append( DutyPersonnelArrangement(
  372. duty_id=new_duty_data.id,
  373. position_id=user_info['position_id'],
  374. personnel_id=user_info['personnel_id'],
  375. create_by=user_id
  376. ))
  377. db.add_all(user_list)
  378. db.commit()
  379. # 构建返回结果
  380. result = {
  381. "code": 200,
  382. "msg": "操作成功",
  383. "data": None
  384. }
  385. return result
  386. except Exception as e:
  387. # 处理异常
  388. traceback.print_exc()
  389. return JSONResponse(status_code=404, content={
  390. 'code': 404,
  391. 'msg': str(e)
  392. })
  393. @router.post('/list_create')
  394. async def create_dict_data(
  395. db: Session = Depends(get_db),
  396. body = Depends(remove_xss_json),
  397. user_id = Depends(valid_access_token)
  398. ):
  399. try:
  400. # 创建一个新的 SysDictData 实例
  401. if isinstance(body,list) == False:
  402. return JSONResponse(status_code=500,content={'msg':"请求体非列表","code":500})
  403. for data in body:
  404. user_data = data['user_data']
  405. if len(user_data)==0:
  406. return JSONResponse(status_code=404, content={
  407. 'errcode': 404,
  408. 'errmsg': '值班人员不能为空'
  409. })
  410. new_duty_data = DutySchedule(
  411. start_time=data['start_time'],
  412. end_time=data['end_time'],
  413. duty_date=data['duty_date'],
  414. shift_type=data['shift_type'],
  415. duty_unit=data['duty_unit'],
  416. duty_type=data['duty_type'],
  417. create_by=user_id
  418. )
  419. # 添加到会话并提交
  420. db.add(new_duty_data)
  421. db.commit()
  422. db.refresh(new_duty_data)
  423. user_list = []
  424. for user_info in user_data:
  425. user_list.append( DutyPersonnelArrangement(
  426. duty_id=new_duty_data.id,
  427. position_id=user_info['position_id'],
  428. personnel_id=user_info['personnel_id'],
  429. create_by=user_id
  430. ))
  431. db.add_all(user_list)
  432. db.commit()
  433. # 构建返回结果
  434. result = {
  435. "code": 200,
  436. "msg": "操作成功",
  437. "data": None
  438. }
  439. return result
  440. except Exception as e:
  441. # 处理异常
  442. traceback.print_exc()
  443. return JSONResponse(status_code=404, content={
  444. 'code': 404,
  445. 'msg': str(e)
  446. })
  447. @router.put("/update")
  448. async def updata_dict_type(
  449. db: Session = Depends(get_db),
  450. body = Depends(remove_xss_json),
  451. user_id = Depends(valid_access_token)
  452. ):
  453. try:
  454. # 从请求数据创建一个新的 SysDictType 实例
  455. query = db.query(DutySchedule)
  456. query = query.filter(DutySchedule.id == body['id'])
  457. query = query.filter(DutySchedule.del_flag != '2')
  458. # query = db.query(SysDictData).filter(SysDictData.dict_code == form_data.dictCode)
  459. # query = db.query(SysDictData).filter(SysDictData.del_flag != '2')
  460. duty_data = query.first()
  461. old_user_list = duty_id_get_user_id(db,duty_data.id)
  462. for info in old_user_list:
  463. info.del_flag = '2'
  464. db.commit()
  465. user_data = body['user_data']
  466. if len(user_data)==0:
  467. return JSONResponse(status_code=404, content={
  468. 'errcode': 404,
  469. 'errmsg': '值班人员不能为空'
  470. })
  471. user_list = []
  472. for user_info in user_data:
  473. user_list.append(DutyPersonnelArrangement(
  474. duty_id=duty_data.id,
  475. position_id=user_info['position_id'],
  476. personnel_id=user_info['personnel_id'],
  477. create_by=user_id
  478. ))
  479. db.add_all(user_list)
  480. if not duty_data:
  481. return JSONResponse(status_code=404, content={
  482. 'errcode': 404,
  483. 'errmsg': '值班不存在'
  484. })
  485. duty_data.start_time=body['start_time']
  486. duty_data.end_time = body['end_time']
  487. duty_data.duty_date = body['duty_date']
  488. duty_data.shift_type=body['shift_type']
  489. duty_data.duty_unit = body['duty_unit']
  490. duty_data.duty_type = body['duty_type']
  491. duty_data.update_by = user_id
  492. # 添加到数据库会话并提交
  493. db.commit()
  494. # db.refresh(new_dict_type) # 可选,如果需要刷新实例状态
  495. # 构建并返回响应
  496. return {
  497. "code": 200,
  498. "msg": "操作成功",
  499. "data": None # 根据你的响应示例,data 为 null
  500. }
  501. except Exception as e:
  502. # 处理异常
  503. traceback.print_exc()
  504. return JSONResponse(status_code=404, content={
  505. 'code': 404,
  506. 'msg': str(e)
  507. })
  508. @router.delete("/delete/{id}") # 使用 ID 来标识要删除的接口
  509. async def delete_dict_data(
  510. id: str,
  511. db: Session = Depends(get_db),
  512. body = Depends(remove_xss_json),
  513. user_id = Depends(valid_access_token)
  514. ):
  515. try:
  516. # 从数据库中获取要删除的 OneShareApiEntity 实例
  517. query = db.query(DutySchedule)
  518. query = query.filter(DutySchedule.id == id)
  519. query = query.filter(DutySchedule.del_flag != '2')
  520. position_data = query.first()
  521. # dict_data = db.query(SysDictData).filter(SysDictData.dict_code == dictCode and SysDictData.del_flag != '2').first()
  522. if not position_data:
  523. return JSONResponse(status_code=404, content={
  524. 'code': 404,
  525. 'msg': '值班不存在'
  526. })
  527. position_data.del_flag = '2'
  528. # 删除实例
  529. # db.delete(api)
  530. db.commit()
  531. # 构建并返回响应
  532. return {
  533. "code": 200,
  534. "msg": "操作成功",
  535. "data": None
  536. }
  537. except Exception as e:
  538. traceback.print_exc()
  539. return JSONResponse(status_code=404, content={
  540. 'code': 404,
  541. 'msg': str(e)
  542. })
  543. @router.delete("/delete_list/{id_list}") # 使用 ID 来标识要删除的接口
  544. async def delete_dict_data(
  545. id_list: str,
  546. db: Session = Depends(get_db),
  547. body = Depends(remove_xss_json),
  548. user_id = Depends(valid_access_token)
  549. ):
  550. try:
  551. # 从数据库中获取要删除的 OneShareApiEntity 实例
  552. id_list = [int(i) for i in id_list.split(',')]
  553. query = db.query(DutySchedule)
  554. query = query.filter(DutySchedule.id.in_(id_list))
  555. query = query.filter(DutySchedule.del_flag != '2')
  556. position_data = query.all()
  557. # dict_data = db.query(SysDictData).filter(SysDictData.dict_code == dictCode and SysDictData.del_flag != '2').first()
  558. if not position_data:
  559. return JSONResponse(status_code=404, content={
  560. 'code': 404,
  561. 'msg': '值班不存在'
  562. })
  563. for info in position_data:
  564. info.del_flag = '2'
  565. # 删除实例
  566. # db.delete(api)
  567. db.commit()
  568. # 构建并返回响应
  569. return {
  570. "code": 200,
  571. "msg": "操作成功",
  572. "data": None
  573. }
  574. except Exception as e:
  575. traceback.print_exc()
  576. return JSONResponse(status_code=404, content={
  577. 'code': 404,
  578. 'msg': str(e)
  579. })
  580. def foastr_to_datetime(datedata,timedata,book):
  581. if isinstance(datedata, str) and isinstance(timedata, str):
  582. result = datetime.strptime(datedata+" "+timedata,'%Y-%m-%d %H:%M')
  583. elif isinstance(datedata, str) and isinstance(timedata, float):
  584. date_part = datetime.strptime(datedata, "%Y-%m-%d").date()
  585. time_part = (datetime.min + timedelta(days=timedata)).time()
  586. result = datetime.combine(date_part, time_part)
  587. elif isinstance(datedata, float) and isinstance(timedata, str):
  588. date_part = xlrd.xldate.xldate_as_datetime(datedata, book.datemode).date()
  589. time_part = datetime.strptime(timedata, "%H:%M:%S").time()
  590. result = datetime.combine(date_part, time_part)
  591. else:
  592. result = xlrd.xldate.xldate_as_datetime(datedata+timedata, book.datemode)
  593. return result
  594. # 自定义排班 导出
  595. @router.post('/zdypb/createImport')
  596. # 值班排班 导入
  597. @router.post('/zbpb/createImport')
  598. async def create_contact(
  599. request: Request,
  600. background_tasks: BackgroundTasks,
  601. db: Session = Depends(get_db),
  602. body=Depends(remove_xss_json),
  603. auth_user: AuthUser = Depends(find_auth_user),
  604. user_id=Depends(valid_access_token)
  605. ):
  606. try:
  607. # 提取请求数据
  608. filename = body['filename']
  609. file_name_desc = body['file_name_desc']
  610. duty_unit = body['duty_unit']
  611. duty_type = ''
  612. if 'duty_type' in body:
  613. duty_type = body['duty_type']
  614. if len(filename) == 0:
  615. raise Exception()
  616. file_path = f'/data/upload/mergefile/uploads/{filename}'
  617. # 检查文件是否存在
  618. if not os.path.isfile(file_path):
  619. return JSONResponse(status_code=404, content={
  620. 'errcode': 404,
  621. 'errmsg': f'{filename}不存在'
  622. })
  623. msg = '成功'
  624. code =200
  625. try:
  626. book = xlrd.open_workbook(file_path)
  627. sheet = book.sheet_by_index(0)
  628. except Exception as e:
  629. traceback.print_exc()
  630. msg = f'\n文件打开失败,请核实文件格式为xlsx/xlx>{str(e)}'
  631. code = 500
  632. return JSONResponse(status_code=code, content={
  633. "code": code,
  634. "msg": msg,
  635. "data": None
  636. })
  637. duty_data = {}
  638. duty_persion_data = {}
  639. import_status = True
  640. for row in range(5, sheet.nrows):
  641. user_data = []
  642. start_date = sheet.cell(row, 0).value
  643. if start_date == '' or start_date==0:
  644. msg = f'\n行<{row + 1}>开始日期不能为空<{start_date}>'
  645. code = 500
  646. start_time = sheet.cell(row, 1).value
  647. if start_time == '':
  648. msg = f'\n行<{row + 1}>开始时间不能为空<{start_time}>'
  649. code = 500
  650. start_time = foastr_to_datetime(start_date,start_time,book)
  651. end_date = sheet.cell(row, 2).value
  652. if end_date == '':
  653. msg = f'\n行<{row + 1}>结束日期不能为空<{end_date}>'
  654. code = 500
  655. end_time = sheet.cell(row, 3).value
  656. if end_time == '':
  657. msg = f'\n行<{row + 1}>结束时间不能为空<{end_time}>'
  658. code = 500
  659. end_time = foastr_to_datetime(end_date,end_time,book)
  660. dbld = sheet.cell(row, 4).value
  661. # if dbld == '':
  662. # msg = f'\n行<{row + 1}>带班领导不能为空<{dbld}>'
  663. # code = 500
  664. for name in dbld.split(','):
  665. if name=='':
  666. continue
  667. personnel_id_list = name_get_user_id(db, name,False)
  668. if len(personnel_id_list)==0:
  669. import_status = False
  670. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  671. code = 500
  672. for personnel_id in personnel_id_list:
  673. user_data.append({"position_id":1,"personnel_id":personnel_id})
  674. kjdb = sheet.cell(row, 5).value
  675. # if kjdb == '':
  676. # msg = f'\n行<{row + 1}>科级带班不能为空<{kjdb}>'
  677. # code = 500
  678. for name in kjdb.split(','):
  679. if name=='':
  680. continue
  681. personnel_id_list = name_get_user_id(db, name,False)
  682. if len(personnel_id_list)==0:
  683. import_status = False
  684. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  685. code = 500
  686. for personnel_id in personnel_id_list:
  687. user_data.append({"position_id":2,"personnel_id":personnel_id})
  688. zb = sheet.cell(row, 6).value
  689. # if zb == '':
  690. # msg = f'\n行<{row + 1}>主班不能为空<{zb}>'
  691. # code = 500
  692. for name in zb.split(','):
  693. if name=='':
  694. continue
  695. personnel_id_list = name_get_user_id(db, name,False)
  696. if len(personnel_id_list)==0:
  697. import_status = False
  698. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  699. code = 500
  700. for personnel_id in personnel_id_list:
  701. user_data.append({"position_id":8,"personnel_id":personnel_id})
  702. zz = sheet.cell(row, 7).value
  703. # if zz == '':
  704. # msg = f'\n行<{row + 1}>专职不能为空<{zz}>'
  705. # code = 500
  706. for name in zz.split(','):
  707. if name=='':
  708. continue
  709. personnel_id_list = name_get_user_id(db, name,False)
  710. if len(personnel_id_list)==0:
  711. import_status = False
  712. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  713. code = 500
  714. for personnel_id in personnel_id_list:
  715. user_data.append({"position_id":9,"personnel_id":personnel_id})
  716. start_date = start_time.strftime('%Y-%m-%d')
  717. end_date = end_time.strftime('%Y-%m-%d')
  718. if start_date == end_date:
  719. shift_type = '1'
  720. else:
  721. if start_time==end_time:
  722. shift_type = '3'
  723. else:
  724. shift_type = '2'
  725. new_duty_data = DutySchedule(
  726. start_time=start_time, #f'{start_date} {start_time}',
  727. end_time=end_time, #f'{end_date} {end_time}',
  728. duty_date=start_date,
  729. shift_type=shift_type,
  730. duty_unit=duty_unit,
  731. duty_type=duty_type,
  732. create_by=user_id
  733. )
  734. # print(start_date+str(shift_type))
  735. duty_data[start_date+str(shift_type)] = new_duty_data
  736. # 添加到会话并提交
  737. duty_persion_data[start_date+str(shift_type)] = user_data
  738. # print(duty_persion_data)
  739. # db.add(new_duty_data)
  740. # db.commit()
  741. # db.refresh(new_duty_data)
  742. # user_list = []
  743. # for user_info in user_data:
  744. # user_list.append(DutyPersonnelArrangement(
  745. # duty_id=new_duty_data.id,
  746. # position_id=user_info['position_id'],
  747. # personnel_id=user_info['personnel_id'],
  748. # create_by=user_id
  749. # ))
  750. # db.add_all(user_list)
  751. # db.commit()
  752. if import_status:
  753. for duty in duty_data:
  754. new_duty_data = duty_data[duty]
  755. db.add(new_duty_data)
  756. db.commit()
  757. db.refresh(new_duty_data)
  758. user_list = []
  759. for user_info in duty_persion_data[duty]:
  760. # print(new_duty_data.id,user_info)
  761. user_list.append(DutyPersonnelArrangement(
  762. duty_id=new_duty_data.id,
  763. position_id=user_info['position_id'],
  764. personnel_id=user_info['personnel_id'],
  765. create_by=user_id
  766. ))
  767. db.add_all(user_list)
  768. db.commit()
  769. except Exception as e:
  770. traceback.print_exc()
  771. # 处理异常
  772. db.rollback()
  773. code = 500
  774. msg = str(e)
  775. return JSONResponse(status_code=code, content={
  776. "code": code,
  777. "msg": msg,
  778. "data": None
  779. })
  780. # 值班排班 导出
  781. @router.get("/zbpb/export")
  782. async def download_file(keywords:str =Query(None),
  783. duty_type:str =Query(None),
  784. start_time:str =Query(None),
  785. end_time:str =Query(None),
  786. duty_unit:int =Query(None),
  787. db: Session = Depends(get_db),
  788. body = Depends(remove_xss_json),
  789. user_id = Depends(valid_access_token)):
  790. """
  791. 根据提供的文件名下载文件。
  792. :param filename: 要下载的文件的名称。
  793. """
  794. try:
  795. query = db.query(DutySchedule)
  796. query = query.filter(DutySchedule.del_flag != '2')
  797. # 添加查询条件
  798. if keywords:
  799. user_list = name_get_user_id(db, keywords)
  800. duty_list = user_id_get_duty_id(db, user_list)
  801. query = query.filter(DutySchedule.id.in_(duty_list))
  802. if duty_type:
  803. query = query.filter(DutySchedule.duty_type == duty_type)
  804. if start_time:
  805. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  806. query = query.filter(DutySchedule.duty_date >= start_time)
  807. if end_time:
  808. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  809. query = query.filter(DutySchedule.duty_date <= end_time)
  810. if duty_unit:
  811. query = query.filter(DutySchedule.duty_unit == duty_unit)
  812. subquery = db.query(
  813. DutySchedule.duty_date,
  814. DutySchedule.shift_type,
  815. func.max(DutySchedule.update_time).label("latest_create_time")
  816. ).filter(DutySchedule.del_flag != '2').group_by(
  817. DutySchedule.duty_date,
  818. DutySchedule.shift_type
  819. ).subquery()
  820. query = query.join(
  821. subquery,
  822. and_(
  823. DutySchedule.duty_date == subquery.c.duty_date,
  824. DutySchedule.shift_type == subquery.c.shift_type,
  825. DutySchedule.update_time == subquery.c.latest_create_time
  826. )
  827. )
  828. query = query.order_by(DutySchedule.duty_date)
  829. duty_data = query.all()
  830. # 转换为字典
  831. data_list = []
  832. for d in duty_data:
  833. data = {
  834. "值班开始日期": d.start_time.strftime('%Y-%m-%d') if d.start_time else '',
  835. "值班开始时间": d.start_time.strftime('%H:%M') if d.start_time else '',
  836. "值班结束日期": d.end_time.strftime('%Y-%m-%d') if d.end_time else '',
  837. "值班结束时间": d.end_time.strftime('%H:%M') if d.end_time else '',
  838. "带班领导": ""
  839. }
  840. user_data = {}
  841. user_list = duty_id_get_user_id(db, d.id)
  842. for user_info in user_list:
  843. contact_info = user_id_get_info(db, user_info.personnel_id)
  844. position= position_id_get_info(db,user_info.position_id)
  845. if position.position_name in user_data:
  846. user_data[position.position_name].append(contact_info.name)
  847. else:
  848. user_data[position.position_name]=[contact_info.name]
  849. for position_name in user_data:
  850. data[position_name] = ",".join(user_data[position_name])
  851. data_list.append(data)
  852. # 构造文件的完整路径
  853. import pandas as pd
  854. from io import BytesIO
  855. # 将查询结果转换为 DataFrame
  856. df = pd.DataFrame(data_list)
  857. # 将 DataFrame 导出为 Excel 文件
  858. output = BytesIO()
  859. with pd.ExcelWriter(output, engine='openpyxl') as writer:
  860. df.to_excel(writer, index=False)
  861. # 设置响应头
  862. output.seek(0)
  863. from urllib.parse import quote
  864. encoded_filename = f'值班排班{datetime.now().strftime("%Y%m%d%H%mi%s")}.xlsx'
  865. encoded_filename = quote(encoded_filename, encoding='utf-8')
  866. headers = {
  867. 'Content-Disposition': f'attachment; filename*=UTF-8\'\'{encoded_filename}',
  868. 'Content-Type': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
  869. }
  870. # 返回文件流
  871. return StreamingResponse(output, headers=headers)
  872. except HTTPException as e:
  873. raise e
  874. except Exception as e:
  875. # 处理其他异常情况
  876. raise HTTPException(status_code=500, detail=str(e))
  877. # 本单位值班 导出
  878. @router.get("/bdwzb/export")
  879. # 各级单位值班表 导出
  880. @router.get("/gjdwzbb/export")
  881. async def download_file(keywords:str =Query(None),
  882. start_time:str =Query(None),
  883. end_time:str =Query(None),
  884. duty_unit:int =Query(None),
  885. db: Session = Depends(get_db),
  886. body = Depends(remove_xss_json),
  887. user_id = Depends(valid_access_token)):
  888. """
  889. 根据提供的文件名下载文件。
  890. :param filename: 要下载的文件的名称。
  891. """
  892. try:
  893. query = db.query(DutySchedule)
  894. query = query.filter(DutySchedule.del_flag != '2')
  895. # 添加查询条件
  896. if keywords:
  897. user_list = name_get_user_id(db, keywords)
  898. duty_list = user_id_get_duty_id(db, user_list)
  899. query = query.filter(DutySchedule.id.in_(duty_list))
  900. if start_time:
  901. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  902. query = query.filter(DutySchedule.duty_date >= start_time)
  903. if end_time:
  904. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  905. query = query.filter(DutySchedule.duty_date <= end_time)
  906. if duty_unit:
  907. query = query.filter(DutySchedule.duty_unit == duty_unit)
  908. else:
  909. now_user_info = user_id_get_user_info(db, user_id)
  910. print(user_id, now_user_info.phonenumber, get_duty_unit_id_list(db))
  911. if now_user_info:
  912. query = query.filter(
  913. DutySchedule.duty_unit == mobile_phone_get_dept_id(db, mpfun.dec_data(now_user_info.phonenumber),
  914. get_duty_unit_id_list(db)))
  915. else:
  916. return {
  917. "data": [],
  918. "code": 200,
  919. "msg": "查询成功"
  920. }
  921. subquery = db.query(
  922. DutySchedule.duty_date,
  923. DutySchedule.shift_type,
  924. func.max(DutySchedule.update_time).label("latest_create_time")
  925. ).filter(DutySchedule.del_flag != '2').group_by(
  926. DutySchedule.duty_date,
  927. DutySchedule.shift_type
  928. ).subquery()
  929. query = query.join(
  930. subquery,
  931. and_(
  932. DutySchedule.duty_date == subquery.c.duty_date,
  933. DutySchedule.shift_type == subquery.c.shift_type,
  934. DutySchedule.update_time == subquery.c.latest_create_time
  935. )
  936. )
  937. query = query.order_by(DutySchedule.update_time.desc())
  938. duty_data = query.all()
  939. # 转换为字典
  940. data_list = []
  941. for d in duty_data:
  942. user_data = []
  943. leaders = []
  944. user_list = duty_id_get_user_id(db, d.id)
  945. for user_info in user_list:
  946. contact_info = user_id_get_info(db, user_info.personnel_id)
  947. position= position_id_get_info(db,user_info.position_id)
  948. if position.position_name=='带班领导':
  949. leaders.append(contact_info.name)
  950. user_data.append(contact_info.name)
  951. data_list.append({
  952. "单位名称": dept_id_get_info(db,d.duty_unit).department_name,
  953. "开始时间": d.start_time.strftime('%Y-%m-%d %H:%M:%S') if d.start_time else '',
  954. "结束时间": d.end_time.strftime('%Y-%m-%d %H:%M:%S') if d.end_time else '',
  955. "带班领导": "、".join(leaders),
  956. "值班人员": "、".join(user_data)
  957. })
  958. # 构造文件的完整路径
  959. import pandas as pd
  960. from io import BytesIO
  961. # 将查询结果转换为 DataFrame
  962. df = pd.DataFrame(data_list)
  963. # 将 DataFrame 导出为 Excel 文件
  964. output = BytesIO()
  965. with pd.ExcelWriter(output, engine='openpyxl') as writer:
  966. df.to_excel(writer, index=False)
  967. # 设置响应头
  968. output.seek(0)
  969. from urllib.parse import quote
  970. encoded_filename = f'值班表{datetime.now().strftime("%Y%m%d%H%mi%s")}.xlsx'
  971. encoded_filename = quote(encoded_filename, encoding='utf-8')
  972. headers = {
  973. 'Content-Disposition': f'attachment; filename*=UTF-8\'\'{encoded_filename}',
  974. 'Content-Type': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
  975. }
  976. # 返回文件流
  977. return StreamingResponse(output, headers=headers)
  978. except HTTPException as e:
  979. raise e
  980. except Exception as e:
  981. # 处理其他异常情况
  982. raise HTTPException(status_code=500, detail=str(e))
  983. # 各级单位值班人 导出
  984. @router.get("/gjdwzbr/export")
  985. async def download_file(keywords:str =Query(None),
  986. start_time:str =Query(None),
  987. end_time:str =Query(None),
  988. duty_unit:int =Query(None),
  989. db: Session = Depends(get_db),
  990. body = Depends(remove_xss_json),
  991. user_id = Depends(valid_access_token)):
  992. """
  993. 根据提供的文件名下载文件。
  994. :param filename: 要下载的文件的名称。
  995. """
  996. try:
  997. query = db.query(DutySchedule)
  998. query = query.filter(DutySchedule.del_flag != '2')
  999. # 添加查询条件
  1000. if keywords:
  1001. user_list = name_get_user_id(db, keywords)
  1002. duty_list = user_id_get_duty_id(db, user_list)
  1003. query = query.filter(DutySchedule.id.in_(duty_list))
  1004. if start_time:
  1005. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  1006. query = query.filter(DutySchedule.duty_date >= start_time)
  1007. if end_time:
  1008. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  1009. query = query.filter(DutySchedule.duty_date <= end_time)
  1010. if duty_unit:
  1011. query = query.filter(DutySchedule.duty_unit == duty_unit)
  1012. else:
  1013. now_user_info = user_id_get_user_info(db, user_id)
  1014. print(user_id, now_user_info.phonenumber, get_duty_unit_id_list(db))
  1015. if now_user_info:
  1016. query = query.filter(
  1017. DutySchedule.duty_unit == mobile_phone_get_dept_id(db, mpfun.dec_data(now_user_info.phonenumber),
  1018. get_duty_unit_id_list(db)))
  1019. else:
  1020. return {
  1021. "data": [],
  1022. "code": 200,
  1023. "msg": "查询成功"
  1024. }
  1025. subquery = db.query(
  1026. DutySchedule.duty_date,
  1027. DutySchedule.shift_type,
  1028. func.max(DutySchedule.update_time).label("latest_create_time")
  1029. ).filter(DutySchedule.del_flag != '2').group_by(
  1030. DutySchedule.duty_date,
  1031. DutySchedule.shift_type
  1032. ).subquery()
  1033. query = query.join(
  1034. subquery,
  1035. and_(
  1036. DutySchedule.duty_date == subquery.c.duty_date,
  1037. DutySchedule.shift_type == subquery.c.shift_type,
  1038. DutySchedule.update_time == subquery.c.latest_create_time
  1039. )
  1040. )
  1041. query = query.order_by(DutySchedule.update_time.desc())
  1042. duty_data = query.all()
  1043. # 转换为字典
  1044. data_list = []
  1045. yz_list = []
  1046. for d in duty_data:
  1047. user_list = duty_id_get_user_id(db, d.id)
  1048. for user_info in user_list:
  1049. contact_info = user_id_get_info(db, user_info.personnel_id)
  1050. position = position_id_get_info(db, user_info.position_id)
  1051. department = dept_id_get_info(db,contact_info.department_id)
  1052. yz = d.duty_date.strftime('%Y-%m-%d')+str(contact_info.id)
  1053. if yz not in yz_list:
  1054. yz_list.append(yz)
  1055. data_list.append({"值班日期":d.duty_date,
  1056. "值班机构": dept_id_get_info(db,d.duty_unit).department_name,
  1057. "组织机构": department.department_name,
  1058. "值班岗位": position.position_name,
  1059. "值班人员": contact_info.name,
  1060. "职务": contact_info.position,
  1061. "联系电话": contact_info.mobile_phone,
  1062. "办公电话": contact_info.office_phone})
  1063. # 构造文件的完整路径
  1064. import pandas as pd
  1065. from io import BytesIO
  1066. # 将查询结果转换为 DataFrame
  1067. df = pd.DataFrame(data_list)
  1068. # 将 DataFrame 导出为 Excel 文件
  1069. output = BytesIO()
  1070. with pd.ExcelWriter(output, engine='openpyxl') as writer:
  1071. df.to_excel(writer, index=False)
  1072. # 设置响应头
  1073. output.seek(0)
  1074. from urllib.parse import quote
  1075. encoded_filename = f'值班人员{datetime.now().strftime("%Y%m%d%H%mi%s")}.xlsx'
  1076. encoded_filename = quote(encoded_filename, encoding='utf-8')
  1077. headers = {
  1078. 'Content-Disposition': f'attachment; filename*=UTF-8\'\'{encoded_filename}',
  1079. 'Content-Type': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
  1080. }
  1081. # 返回文件流
  1082. return StreamingResponse(output, headers=headers)
  1083. except HTTPException as e:
  1084. raise e
  1085. except Exception as e:
  1086. # 处理其他异常情况
  1087. raise HTTPException(status_code=500, detail=str(e))