schedule.py 43 KB

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