schedule.py 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158
  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==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. # 自定义排班 导出
  578. @router.post('/zdypb/createImport')
  579. # 值班排班 导入
  580. @router.post('/zbpb/createImport')
  581. async def create_contact(
  582. request: Request,
  583. background_tasks: BackgroundTasks,
  584. db: Session = Depends(get_db),
  585. body=Depends(remove_xss_json),
  586. auth_user: AuthUser = Depends(find_auth_user),
  587. user_id=Depends(valid_access_token)
  588. ):
  589. try:
  590. # 提取请求数据
  591. filename = body['filename']
  592. file_name_desc = body['file_name_desc']
  593. duty_unit = body['duty_unit']
  594. duty_type = ''
  595. if 'duty_type' in body:
  596. duty_type = body['duty_type']
  597. if len(filename) == 0:
  598. raise Exception()
  599. file_path = f'/data/upload/mergefile/uploads/{filename}'
  600. # 检查文件是否存在
  601. if not os.path.isfile(file_path):
  602. return JSONResponse(status_code=404, content={
  603. 'errcode': 404,
  604. 'errmsg': f'{filename}不存在'
  605. })
  606. msg = '成功'
  607. code =200
  608. try:
  609. book = xlrd.open_workbook(file_path)
  610. sheet = book.sheet_by_index(0)
  611. except Exception as e:
  612. traceback.print_exc()
  613. msg = f'\n文件打开失败,请核实文件格式为xlsx/xlx>{str(e)}'
  614. code = 500
  615. return JSONResponse(status_code=code, content={
  616. "code": code,
  617. "msg": msg,
  618. "data": None
  619. })
  620. duty_data = {}
  621. duty_persion_data = {}
  622. import_status = True
  623. for row in range(2, sheet.nrows):
  624. user_data = []
  625. start_date = sheet.cell(row, 0).value
  626. if start_date == '':
  627. msg = f'\n行<{row + 1}>开始日期不能为空<{start_date}>'
  628. code = 500
  629. start_time = sheet.cell(row, 1).value
  630. if start_time == '':
  631. msg = f'\n行<{row + 1}>开始时间不能为空<{start_time}>'
  632. code = 500
  633. end_date = sheet.cell(row, 2).value
  634. if end_date == '':
  635. msg = f'\n行<{row + 1}>结束日期不能为空<{end_date}>'
  636. code = 500
  637. end_time = sheet.cell(row, 3).value
  638. if end_time == '':
  639. msg = f'\n行<{row + 1}>结束时间不能为空<{end_time}>'
  640. code = 500
  641. dbld = sheet.cell(row, 4).value
  642. if dbld == '':
  643. msg = f'\n行<{row + 1}>带班领导不能为空<{dbld}>'
  644. code = 500
  645. for name in dbld.split(','):
  646. personnel_id_list = name_get_user_id(db, name)
  647. if len(personnel_id_list)==0:
  648. import_status = False
  649. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  650. code = 500
  651. for personnel_id in personnel_id_list:
  652. user_data.append({"position_id":1,"personnel_id":personnel_id})
  653. kjdb = sheet.cell(row, 5).value
  654. if kjdb == '':
  655. msg = f'\n行<{row + 1}>科级带班不能为空<{kjdb}>'
  656. code = 500
  657. for name in kjdb.split(','):
  658. personnel_id_list = name_get_user_id(db, name)
  659. if len(personnel_id_list)==0:
  660. import_status = False
  661. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  662. code = 500
  663. for personnel_id in personnel_id_list:
  664. user_data.append({"position_id":1,"personnel_id":personnel_id})
  665. zb = sheet.cell(row, 6).value
  666. if zb == '':
  667. msg = f'\n行<{row + 1}>主班不能为空<{zb}>'
  668. code = 500
  669. for name in zb.split(','):
  670. personnel_id_list = name_get_user_id(db, name)
  671. if len(personnel_id_list)==0:
  672. import_status = False
  673. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  674. code = 500
  675. for personnel_id in personnel_id_list:
  676. user_data.append({"position_id":1,"personnel_id":personnel_id})
  677. zz = sheet.cell(row, 7).value
  678. if zz == '':
  679. msg = f'\n行<{row + 1}>专职不能为空<{zz}>'
  680. code = 500
  681. for name in zz.split(','):
  682. personnel_id_list = name_get_user_id(db, name)
  683. if len(personnel_id_list)==0:
  684. import_status = False
  685. msg = f'\n行<{row + 1}>非应急通讯录人员<{name}>'
  686. code = 500
  687. for personnel_id in personnel_id_list:
  688. user_data.append({"position_id":1,"personnel_id":personnel_id})
  689. if start_date == end_date:
  690. shift_type = '1'
  691. else:
  692. if start_time==end_time:
  693. shift_type = '3'
  694. else:
  695. shift_type = '2'
  696. start_date1 = xlrd.xldate.xldate_as_datetime(start_date, book.datemode).strftime('%Y-%m-%d')
  697. new_duty_data = DutySchedule(
  698. start_time=xlrd.xldate.xldate_as_datetime(start_date+start_time, book.datemode).strftime('%Y-%m-%d %H:%M:%S'), #f'{start_date} {start_time}',
  699. end_time=xlrd.xldate.xldate_as_datetime(end_date+end_time, book.datemode).strftime('%Y-%m-%d %H:%M:%S'), #f'{end_date} {end_time}',
  700. duty_date=start_date1,
  701. shift_type=shift_type,
  702. duty_unit=duty_unit,
  703. duty_type=duty_type,
  704. create_by=user_id
  705. )
  706. duty_data[start_date1+str(shift_type)] = new_duty_data
  707. # 添加到会话并提交
  708. duty_persion_data[start_date1+ str(shift_type)] = user_data
  709. # db.add(new_duty_data)
  710. # db.commit()
  711. # db.refresh(new_duty_data)
  712. # user_list = []
  713. # for user_info in user_data:
  714. # user_list.append(DutyPersonnelArrangement(
  715. # duty_id=new_duty_data.id,
  716. # position_id=user_info['position_id'],
  717. # personnel_id=user_info['personnel_id'],
  718. # create_by=user_id
  719. # ))
  720. # db.add_all(user_list)
  721. # db.commit()
  722. if import_status:
  723. for duty in duty_data:
  724. new_duty_data = duty_data[duty]
  725. db.add(new_duty_data)
  726. db.commit()
  727. db.refresh(new_duty_data)
  728. user_list = []
  729. for user_info in duty_persion_data[duty]:
  730. user_list.append(DutyPersonnelArrangement(
  731. duty_id=new_duty_data.id,
  732. position_id=user_info['position_id'],
  733. personnel_id=user_info['personnel_id'],
  734. create_by=user_id
  735. ))
  736. db.add_all(user_list)
  737. db.commit()
  738. except Exception as e:
  739. traceback.print_exc()
  740. # 处理异常
  741. db.rollback()
  742. code = 500
  743. msg = str(e)
  744. return JSONResponse(status_code=code, content={
  745. "code": code,
  746. "msg": msg,
  747. "data": None
  748. })
  749. # 值班排班 导出
  750. @router.get("/zbpb/export")
  751. async def download_file(keywords:str =Query(None),
  752. duty_type:str =Query(None),
  753. start_time:str =Query(None),
  754. end_time:str =Query(None),
  755. duty_unit:int =Query(None),
  756. db: Session = Depends(get_db),
  757. body = Depends(remove_xss_json),
  758. user_id = Depends(valid_access_token)):
  759. """
  760. 根据提供的文件名下载文件。
  761. :param filename: 要下载的文件的名称。
  762. """
  763. try:
  764. query = db.query(DutySchedule)
  765. query = query.filter(DutySchedule.del_flag != '2')
  766. # 添加查询条件
  767. if keywords:
  768. user_list = name_get_user_id(db, keywords)
  769. duty_list = user_id_get_duty_id(db, user_list)
  770. query = query.filter(DutySchedule.id.in_(duty_list))
  771. if duty_type:
  772. query = query.filter(DutySchedule.duty_type == duty_type)
  773. if start_time:
  774. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  775. query = query.filter(DutySchedule.duty_date >= start_time)
  776. if end_time:
  777. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  778. query = query.filter(DutySchedule.duty_date <= end_time)
  779. if duty_unit:
  780. query = query.filter(DutySchedule.duty_unit == duty_unit)
  781. subquery = db.query(
  782. DutySchedule.duty_date,
  783. DutySchedule.shift_type,
  784. func.max(DutySchedule.update_time).label("latest_create_time")
  785. ).filter(DutySchedule.del_flag != '2').group_by(
  786. DutySchedule.duty_date,
  787. DutySchedule.shift_type
  788. ).subquery()
  789. query = query.join(
  790. subquery,
  791. and_(
  792. DutySchedule.duty_date == subquery.c.duty_date,
  793. DutySchedule.shift_type == subquery.c.shift_type,
  794. DutySchedule.update_time == subquery.c.latest_create_time
  795. )
  796. )
  797. query = query.order_by(DutySchedule.duty_date)
  798. duty_data = query.all()
  799. # 转换为字典
  800. data_list = []
  801. for d in duty_data:
  802. data = {
  803. "值班开始日期": d.start_time.strftime('%Y-%m-%d') if d.start_time else '',
  804. "值班开始时间": d.start_time.strftime('%H:%M') if d.start_time else '',
  805. "值班结束日期": d.end_time.strftime('%Y-%m-%d') if d.end_time else '',
  806. "值班结束时间": d.end_time.strftime('%H:%M') if d.end_time else '',
  807. "带班领导": ""
  808. }
  809. user_data = {}
  810. user_list = duty_id_get_user_id(db, d.id)
  811. for user_info in user_list:
  812. contact_info = user_id_get_info(db, user_info.personnel_id)
  813. position= position_id_get_info(db,user_info.position_id)
  814. if position.position_name in user_data:
  815. user_data[position.position_name].append(contact_info.name)
  816. else:
  817. user_data[position.position_name]=[contact_info.name]
  818. for position_name in user_data:
  819. data[position_name] = ",".join(user_data[position_name])
  820. data_list.append(data)
  821. # 构造文件的完整路径
  822. import pandas as pd
  823. from io import BytesIO
  824. # 将查询结果转换为 DataFrame
  825. df = pd.DataFrame(data_list)
  826. # 将 DataFrame 导出为 Excel 文件
  827. output = BytesIO()
  828. with pd.ExcelWriter(output, engine='openpyxl') as writer:
  829. df.to_excel(writer, index=False)
  830. # 设置响应头
  831. output.seek(0)
  832. from urllib.parse import quote
  833. encoded_filename = f'值班排班{datetime.now().strftime("%Y%m%d%H%mi%s")}.xlsx'
  834. encoded_filename = quote(encoded_filename, encoding='utf-8')
  835. headers = {
  836. 'Content-Disposition': f'attachment; filename*=UTF-8\'\'{encoded_filename}',
  837. 'Content-Type': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
  838. }
  839. # 返回文件流
  840. return StreamingResponse(output, headers=headers)
  841. except HTTPException as e:
  842. raise e
  843. except Exception as e:
  844. # 处理其他异常情况
  845. raise HTTPException(status_code=500, detail=str(e))
  846. # 本单位值班 导出
  847. @router.get("/bdwzb/export")
  848. # 各级单位值班表 导出
  849. @router.get("/gjdwzbb/export")
  850. async def download_file(keywords:str =Query(None),
  851. start_time:str =Query(None),
  852. end_time:str =Query(None),
  853. duty_unit:int =Query(None),
  854. db: Session = Depends(get_db),
  855. body = Depends(remove_xss_json),
  856. user_id = Depends(valid_access_token)):
  857. """
  858. 根据提供的文件名下载文件。
  859. :param filename: 要下载的文件的名称。
  860. """
  861. try:
  862. query = db.query(DutySchedule)
  863. query = query.filter(DutySchedule.del_flag != '2')
  864. # 添加查询条件
  865. if keywords:
  866. user_list = name_get_user_id(db, keywords)
  867. duty_list = user_id_get_duty_id(db, user_list)
  868. query = query.filter(DutySchedule.id.in_(duty_list))
  869. if start_time:
  870. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  871. query = query.filter(DutySchedule.duty_date >= start_time)
  872. if end_time:
  873. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  874. query = query.filter(DutySchedule.duty_date <= end_time)
  875. if duty_unit:
  876. query = query.filter(DutySchedule.duty_unit == duty_unit)
  877. else:
  878. now_user_info = user_id_get_user_info(db, user_id)
  879. print(user_id, now_user_info.phonenumber, get_duty_unit_id_list(db))
  880. if now_user_info:
  881. query = query.filter(
  882. DutySchedule.duty_unit == mobile_phone_get_dept_id(db, mpfun.dec_data(now_user_info.phonenumber),
  883. get_duty_unit_id_list(db)))
  884. else:
  885. return {
  886. "data": [],
  887. "code": 200,
  888. "msg": "查询成功"
  889. }
  890. subquery = db.query(
  891. DutySchedule.duty_date,
  892. DutySchedule.shift_type,
  893. func.max(DutySchedule.update_time).label("latest_create_time")
  894. ).filter(DutySchedule.del_flag != '2').group_by(
  895. DutySchedule.duty_date,
  896. DutySchedule.shift_type
  897. ).subquery()
  898. query = query.join(
  899. subquery,
  900. and_(
  901. DutySchedule.duty_date == subquery.c.duty_date,
  902. DutySchedule.shift_type == subquery.c.shift_type,
  903. DutySchedule.update_time == subquery.c.latest_create_time
  904. )
  905. )
  906. query = query.order_by(DutySchedule.update_time.desc())
  907. duty_data = query.all()
  908. # 转换为字典
  909. data_list = []
  910. for d in duty_data:
  911. user_data = []
  912. leaders = []
  913. user_list = duty_id_get_user_id(db, d.id)
  914. for user_info in user_list:
  915. contact_info = user_id_get_info(db, user_info.personnel_id)
  916. position= position_id_get_info(db,user_info.position_id)
  917. if position.position_name=='带班领导':
  918. leaders.append(contact_info.name)
  919. user_data.append(contact_info.name)
  920. data_list.append({
  921. "单位名称": dept_id_get_info(db,d.duty_unit).department_name,
  922. "开始时间": d.start_time.strftime('%Y-%m-%d %H:%M:%S') if d.start_time else '',
  923. "结束时间": d.end_time.strftime('%Y-%m-%d %H:%M:%S') if d.end_time else '',
  924. "带班领导": "、".join(leaders),
  925. "值班人员": "、".join(user_data)
  926. })
  927. # 构造文件的完整路径
  928. import pandas as pd
  929. from io import BytesIO
  930. # 将查询结果转换为 DataFrame
  931. df = pd.DataFrame(data_list)
  932. # 将 DataFrame 导出为 Excel 文件
  933. output = BytesIO()
  934. with pd.ExcelWriter(output, engine='openpyxl') as writer:
  935. df.to_excel(writer, index=False)
  936. # 设置响应头
  937. output.seek(0)
  938. from urllib.parse import quote
  939. encoded_filename = f'值班表{datetime.now().strftime("%Y%m%d%H%mi%s")}.xlsx'
  940. encoded_filename = quote(encoded_filename, encoding='utf-8')
  941. headers = {
  942. 'Content-Disposition': f'attachment; filename*=UTF-8\'\'{encoded_filename}',
  943. 'Content-Type': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
  944. }
  945. # 返回文件流
  946. return StreamingResponse(output, headers=headers)
  947. except HTTPException as e:
  948. raise e
  949. except Exception as e:
  950. # 处理其他异常情况
  951. raise HTTPException(status_code=500, detail=str(e))
  952. # 各级单位值班人 导出
  953. @router.get("/gjdwzbr/export")
  954. async def download_file(keywords:str =Query(None),
  955. start_time:str =Query(None),
  956. end_time:str =Query(None),
  957. duty_unit:int =Query(None),
  958. db: Session = Depends(get_db),
  959. body = Depends(remove_xss_json),
  960. user_id = Depends(valid_access_token)):
  961. """
  962. 根据提供的文件名下载文件。
  963. :param filename: 要下载的文件的名称。
  964. """
  965. try:
  966. query = db.query(DutySchedule)
  967. query = query.filter(DutySchedule.del_flag != '2')
  968. # 添加查询条件
  969. if keywords:
  970. user_list = name_get_user_id(db, keywords)
  971. duty_list = user_id_get_duty_id(db, user_list)
  972. query = query.filter(DutySchedule.id.in_(duty_list))
  973. if start_time:
  974. start_time = datetime.strptime(start_time, "%Y-%m-%d").date()
  975. query = query.filter(DutySchedule.duty_date >= start_time)
  976. if end_time:
  977. end_time = datetime.strptime(end_time, "%Y-%m-%d").date()
  978. query = query.filter(DutySchedule.duty_date <= end_time)
  979. if duty_unit:
  980. query = query.filter(DutySchedule.duty_unit == duty_unit)
  981. else:
  982. now_user_info = user_id_get_user_info(db, user_id)
  983. print(user_id, now_user_info.phonenumber, get_duty_unit_id_list(db))
  984. if now_user_info:
  985. query = query.filter(
  986. DutySchedule.duty_unit == mobile_phone_get_dept_id(db, mpfun.dec_data(now_user_info.phonenumber),
  987. get_duty_unit_id_list(db)))
  988. else:
  989. return {
  990. "data": [],
  991. "code": 200,
  992. "msg": "查询成功"
  993. }
  994. subquery = db.query(
  995. DutySchedule.duty_date,
  996. DutySchedule.shift_type,
  997. func.max(DutySchedule.update_time).label("latest_create_time")
  998. ).filter(DutySchedule.del_flag != '2').group_by(
  999. DutySchedule.duty_date,
  1000. DutySchedule.shift_type
  1001. ).subquery()
  1002. query = query.join(
  1003. subquery,
  1004. and_(
  1005. DutySchedule.duty_date == subquery.c.duty_date,
  1006. DutySchedule.shift_type == subquery.c.shift_type,
  1007. DutySchedule.update_time == subquery.c.latest_create_time
  1008. )
  1009. )
  1010. query = query.order_by(DutySchedule.update_time.desc())
  1011. duty_data = query.all()
  1012. # 转换为字典
  1013. data_list = []
  1014. yz_list = []
  1015. for d in duty_data:
  1016. user_list = duty_id_get_user_id(db, d.id)
  1017. for user_info in user_list:
  1018. contact_info = user_id_get_info(db, user_info.personnel_id)
  1019. position = position_id_get_info(db, user_info.position_id)
  1020. department = dept_id_get_info(db,contact_info.department_id)
  1021. yz = d.duty_date.strftime('%Y-%m-%d')+str(contact_info.id)
  1022. if yz not in yz_list:
  1023. yz_list.append(yz)
  1024. data_list.append({"值班日期":d.duty_date,
  1025. "值班机构": dept_id_get_info(db,d.duty_unit).department_name,
  1026. "组织机构": department.department_name,
  1027. "值班岗位": position.position_name,
  1028. "值班人员": contact_info.name,
  1029. "职务": contact_info.position,
  1030. "联系电话": contact_info.mobile_phone,
  1031. "办公电话": contact_info.office_phone})
  1032. # 构造文件的完整路径
  1033. import pandas as pd
  1034. from io import BytesIO
  1035. # 将查询结果转换为 DataFrame
  1036. df = pd.DataFrame(data_list)
  1037. # 将 DataFrame 导出为 Excel 文件
  1038. output = BytesIO()
  1039. with pd.ExcelWriter(output, engine='openpyxl') as writer:
  1040. df.to_excel(writer, index=False)
  1041. # 设置响应头
  1042. output.seek(0)
  1043. from urllib.parse import quote
  1044. encoded_filename = f'值班人员{datetime.now().strftime("%Y%m%d%H%mi%s")}.xlsx'
  1045. encoded_filename = quote(encoded_filename, encoding='utf-8')
  1046. headers = {
  1047. 'Content-Disposition': f'attachment; filename*=UTF-8\'\'{encoded_filename}',
  1048. 'Content-Type': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
  1049. }
  1050. # 返回文件流
  1051. return StreamingResponse(output, headers=headers)
  1052. except HTTPException as e:
  1053. raise e
  1054. except Exception as e:
  1055. # 处理其他异常情况
  1056. raise HTTPException(status_code=500, detail=str(e))