classes.py 97 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699
  1. from email.policy import default
  2. from fastapi import APIRouter, Form, Depends, HTTPException, File, UploadFile,Request
  3. from typing import List,Optional
  4. from fastapi.responses import FileResponse
  5. from random import randint
  6. from fastapi.security import OAuth2PasswordRequestForm
  7. from app.models.models import User,User_information,Favorite_course,Article_list,Class_date,Attend_record,One_day_class,Outter_class_list,Proposal
  8. from app.models.models import Class_list,Schools,Class_detail,Class_name,Registration,Group_name,Online_course,User_resume,User_point,Point_exchange_record
  9. from app.api import deps
  10. from sqlalchemy.orm import Session
  11. from typing import Any, Dict
  12. import secrets
  13. from fastapi_login.exceptions import InvalidCredentialsException
  14. from fastapi_login import LoginManager
  15. from datetime import timedelta,datetime
  16. from app.config import settings
  17. from pathlib import Path
  18. from jose import jwt
  19. from emails.template import JinjaTemplate
  20. from tortoise.queryset import Q
  21. from fastapi.responses import HTMLResponse
  22. import requests
  23. import json
  24. from itertools import chain
  25. from tortoise import fields
  26. import random
  27. from app.log import my_log
  28. import rpyc
  29. import emails
  30. from emails.template import JinjaTemplate
  31. from email.mime.text import MIMEText
  32. from email.mime.image import MIMEImage
  33. from email.mime.multipart import MIMEMultipart
  34. import re
  35. classes = APIRouter()
  36. # logging.basicConfig(
  37. # filename='app-basic.log',
  38. # level=logging.INFO,
  39. # format='%(asctime)s - %(name)s - %(levelname)s - %(message)s',
  40. # datefmt='%Y-%m-%d %H:%M:%S'
  41. # )
  42. # logger = logging.getLogger(__name__)
  43. def send_email(
  44. email_to: str,
  45. token: str,
  46. subject_template: str = "",
  47. html_template: str = "",
  48. environment: Dict[str, Any] = {},
  49. ):
  50. # message = emails.Message(
  51. # subject=JinjaTemplate(subject_template),
  52. # html=JinjaTemplate(html_template),
  53. # mail_from=(settings.EMAILS_FROM_NAME, settings.EMAILS_FROM_EMAIL),
  54. # )
  55. try:
  56. subject=subject_template
  57. html=html_template
  58. mailobj={}
  59. mailobj['toaddr']=email_to
  60. mailobj['title']=subject
  61. mailobj['totext']=html
  62. conn = rpyc.connect("192.168.192.80", 12345)
  63. conn.root.mailto(mailobj)
  64. return {"message":f"send email"}
  65. except Exception as e:
  66. return {"msg": str(e), "code": 500}
  67. @classes.post("/test_email")
  68. async def test_email(
  69. email_to: str,
  70. subject_template: str = "",
  71. html_template: str = ""
  72. ):
  73. email = email_to
  74. subject = subject_template
  75. message = html_template
  76. result = send_email(email,"",subject,message)
  77. return result
  78. IMAGEDIR = "/var/www/ntcri/assets/"
  79. IMAGEDIR_short = "assets/"
  80. encode_set = {
  81. "樹藝" : "ARB",
  82. "漆藝" : "LAC",
  83. "藍染" : "AIZ",
  84. "蠟雕" : "WAX",
  85. "竹工藝籃" :"BAM",
  86. "金工/飾品" :"MWT",
  87. "蠟燭/香氛/調香" : "CAN" ,
  88. "編織/羊毛氈/縫紉" : "KNI" ,
  89. "植栽/花藝" : "PLA" ,
  90. "陶藝/玻璃" : "CER" ,
  91. "皮件/皮革" : "LEA" ,
  92. "插畫/繪畫/寫字" : "Ill",
  93. "木工/竹藝" :"WOO",
  94. "安心活動" : "HER"
  95. }
  96. async def create_upload_files(files:Optional[List[UploadFile]] = File(None)):
  97. files_url = {}
  98. if files :
  99. file_num = 1
  100. print(file_num)
  101. for file in files:
  102. contents = await file.read()
  103. words = file.filename.split('.')
  104. # if words[len(words)-1] != ("png" or "jpg" or "pdf" or "jpeg"):
  105. # return {"msg":"wrong filetype"}
  106. #save the file
  107. with open(f"{IMAGEDIR}{file.filename}", "wb") as f:
  108. f.write(contents)
  109. file_name = "file" + str(file_num)
  110. print(file_name)
  111. files_url[file_name]=f"{IMAGEDIR_short}{file.filename}"
  112. file_num=file_num+1
  113. return files_url
  114. async def check_token(access_token: str):
  115. result = await User.get_or_none(token=access_token)
  116. if not result:
  117. print("no access")
  118. return None
  119. user_id = result.id
  120. return user_id
  121. async def update_location_time(location_id: int):
  122. if location_id:
  123. school = await Schools.get(id=location_id)
  124. school.update_time = datetime.now()
  125. await school.save()
  126. return {"msg": "success", "code": 200}
  127. @classes.post("/change_class_img")
  128. async def change_class_img(
  129. category: str ,
  130. files:Optional[List[UploadFile]] = File(None)
  131. ):
  132. try:
  133. files_url = {}
  134. file_num = 0
  135. for file in files:
  136. contents = await file.read()
  137. #save the file
  138. with open(f"{IMAGEDIR}{file.filename}", "wb") as f:
  139. f.write(contents)
  140. file_name = "file" + str(file_num)
  141. files_url[file_name]=f"{IMAGEDIR_short}{file.filename}"
  142. file_num=file_num+1
  143. class_name_list = await Class_name.filter(Q(category__icontains=category)& Q(is_inner = 0)).all()
  144. for class_name_obj in class_name_list:
  145. x = random.randrange(4)
  146. file_name = "file" + str(x)
  147. class_name_obj.cover_img = files_url[file_name]
  148. await class_name_obj.save()
  149. return {"msg": "success", "code": 200}
  150. except Exception as e:
  151. return {"msg": str(e), "code": 500}
  152. class resume():
  153. imgs = UploadFile
  154. teacher_name : str
  155. work_type : str
  156. experience : str
  157. expertise : str
  158. license : str
  159. media : str
  160. introduction: str
  161. @classes.post("/insert_school")
  162. async def insert_school(
  163. request: Request,
  164. location_name: str = Form(default=''),
  165. Lng: str = Form(default=None),
  166. Lat: str = Form(default=None),
  167. address : str = Form(default=''),
  168. school_introduction : str = Form(default=''),
  169. email : str = Form(default=''),
  170. phone : str = Form(default=''),
  171. access_token:str = Form(default=None),
  172. teachers_list :str = Form(default=None),
  173. is_pass_proposal : int = Form(default=0),
  174. principal_user_email : str = Form(default=None),
  175. ):
  176. try:
  177. user_id = None
  178. if access_token:
  179. user_id = await check_token(access_token)
  180. if user_id == None:
  181. return {"msg": "沒有權限", "code": 500}
  182. # teachers = []
  183. # for teacher in teachers_list:
  184. # try:
  185. # teacher_info = await User.get(email=teacher,is_avtive=1)
  186. # teachers.append(teacher_info.id)
  187. # except:
  188. # my_log("info",__name__,f"no this teacher")
  189. try:
  190. # 這裡response採用json格式,可以自由選擇為json?或是xml?
  191. if Lat == None and Lng == None:
  192. url = 'https://maps.googleapis.com/maps/api/geocode/json?'
  193. params = {'key':'AIzaSyDETbkTAmZNOECx2u4rmNudHvyoQTgDbc4', 'address':''}
  194. params['address'] = address
  195. #print(params['address'])
  196. res = requests.get(url, params=params)
  197. mes = json.loads(res.text)
  198. Lat = mes['results'][0]['geometry']['location']['lat']
  199. Lng = mes['results'][0]['geometry']['location']['lng']
  200. print(Lat,Lng)
  201. except :
  202. my_log("error",__name__,f"not get lat and lng")
  203. principal_user_id =None
  204. if principal_user_email:
  205. try:
  206. principal_user = await User.get(email=principal_user_email,is_avtive=1)
  207. principal_user_id = principal_user.id
  208. except:
  209. my_log("info",__name__,f"no this teacher")
  210. new_school = await Schools.create(
  211. name=location_name,
  212. longitude=Lng,
  213. latitude=Lat,
  214. address = address,
  215. update_time = datetime.now(),
  216. introduction = school_introduction ,
  217. email= email,
  218. phone = phone,
  219. create_user_id = user_id,
  220. is_delete = 0,
  221. is_pass_proposal = is_pass_proposal,
  222. principal_user_id = principal_user_id,
  223. teachers = teachers_list
  224. )
  225. client_ip = request.client.host
  226. my_log("info",__name__,f"Client IP: {client_ip} - new school:{new_school.id}")
  227. return {"msg": "success", "code": 200, "location_id": new_school.id}
  228. except Exception as e:
  229. return {"msg": str(e), "code": 500}
  230. @classes.post("/insert_class_name")
  231. async def insert_class_name(
  232. name: str = Form(default=''),
  233. location_id: int = Form(default=1),
  234. category: str = Form(default=''),
  235. introduction: str = Form(default=''),
  236. organizer: str = Form(default=''),
  237. cover_img_file:UploadFile = File(default=''),
  238. group_id : int = Form(default=1),
  239. group_sort :str = Form(default=''),
  240. recommend: int = Form(default=0),
  241. special_class_list_name : str = Form(default=None),
  242. is_inner : int = Form(default=0),
  243. is_check : int = Form(default=0),
  244. access_token:str = Form(default=None),
  245. teachers_resume : str = Form(default=None),
  246. syllabus : str = Form(default=None)
  247. ):
  248. try:
  249. cover_img = ''
  250. if cover_img_file != '':
  251. contents = await cover_img_file.read()
  252. #save the file
  253. with open(f"{IMAGEDIR}{cover_img_file.filename}", "wb") as f:
  254. f.write(contents)
  255. cover_img = f"{IMAGEDIR_short}{cover_img_file.filename}"
  256. user_id = None
  257. if access_token:
  258. user_id = await check_token(access_token)
  259. if user_id == None:
  260. return {"msg": "沒有權限", "code": 500}
  261. # 課程編碼
  262. encode = ""
  263. if encode_set.__contains__(category) :
  264. encode = encode_set[category]
  265. else:
  266. encode = "OTH"
  267. now = datetime.now()
  268. encode += now.strftime("%Y%m%d")
  269. new_class_name = await Class_name.create(
  270. name=name,
  271. school_id=location_id,
  272. category=category,
  273. introduction=introduction,
  274. organizer=organizer,
  275. cover_img=cover_img,
  276. group_id=group_id,
  277. group_sort=group_sort,
  278. recommend = recommend,
  279. special_class_list_name = special_class_list_name,
  280. is_inner = is_inner,
  281. is_check = is_check,
  282. create_user_id = user_id,
  283. create_time = datetime.now(),
  284. is_delete = 0,
  285. is_record_point =0,
  286. teachers_resume = teachers_resume,
  287. syllabus = syllabus
  288. )
  289. encode += str(new_class_name.id)
  290. new_class_name.encode = encode
  291. await new_class_name.save()
  292. update_location_time(location_id= location_id)
  293. return {"msg": "success", "code": 200, "new_class_name_id": new_class_name.id}
  294. except Exception as e:
  295. return {"msg": str(e), "code": 500}
  296. # @classes.get("/insert_encode")
  297. # async def insert_encode():
  298. # try:
  299. # class_name = await Class_name.all()
  300. # for classes in class_name:
  301. # encode = ""
  302. # if encode_set.__contains__(classes.category) :
  303. # encode = encode_set[classes.category]
  304. # else:
  305. # encode = "OTH"
  306. # now = classes.create_time
  307. # encode += now.strftime("%Y%m%d")
  308. # encode += str(classes.id)
  309. # classes.encode = encode
  310. # await classes.save()
  311. # except Exception as e:
  312. # return {"msg": str(e), "code": 500}
  313. # 創建場次
  314. @classes.post("/insert_event")
  315. async def insert_event(
  316. request : Request,
  317. name_id: int = Form(default=0),
  318. event: str = Form(default=''),
  319. start_time: datetime = Form(default=None),
  320. end_time: datetime = Form(default=None),
  321. contact: str = Form(default=''),
  322. lecturer: str = Form(default=''),
  323. location: str = Form(default=''),
  324. content: str = Form(default=''),
  325. URL: str = Form(default=''),
  326. people : str = Form(default=''),
  327. fee_method: str = Form(default=''),
  328. registration_way: str = Form(default=''),
  329. registration_start: datetime = Form(default=None),
  330. registration_end: datetime = Form(default=None),
  331. number_limit: int = Form(default=None),
  332. remark : str = Form(default=''),
  333. ATM_address: str = Form(default=''),
  334. access_token:str = Form(default=None),
  335. fee_payment: str = Form(default=''),
  336. number_minimum : int = Form(default=None),
  337. files_url = Depends(create_upload_files)
  338. ):
  339. try:
  340. user_id = None
  341. if access_token:
  342. user_id = await check_token(access_token)
  343. if user_id == None:
  344. return {"msg": "沒有權限", "code": 500}
  345. # 檢查是否有該課程
  346. class_name_list = await Class_name.filter(id=name_id).all()
  347. if class_name_list == []:
  348. return {"msg": "沒有此課程", "code": 500}
  349. if files_url:
  350. if files_url.__contains__("msg"):
  351. return {"msg": files_url["msg"], "code": 500}
  352. # 判斷時間合法
  353. if start_time and end_time :
  354. if start_time >= end_time :
  355. return {"msg": "時間輸入錯誤", "code": 500}
  356. new_class = await Class_list.create(
  357. name_id=name_id,
  358. event =event,
  359. start_time=start_time,
  360. end_time=end_time,
  361. contact=contact,
  362. lecturer=lecturer,
  363. location=location,
  364. content=content,
  365. URL=URL,
  366. people=people,
  367. fee_method=fee_method,
  368. registration_way=registration_way,
  369. remark=remark,
  370. ATM_address=ATM_address,
  371. create_user_id = user_id,
  372. fee_payment = fee_payment,
  373. event_create_time = datetime.now(),
  374. files=files_url
  375. )
  376. client_ip = request.client.host
  377. try:
  378. if (registration_start and registration_end and number_limit) or number_minimum:
  379. await Class_date.create(
  380. class_list_id = new_class.id,
  381. registration_start = registration_start,
  382. registration_end = registration_end,
  383. number_limit = number_limit,
  384. amount_left = number_limit,
  385. number_minimum = number_minimum
  386. )
  387. except:
  388. print("input class_date fail")
  389. # 寄開課信
  390. try :
  391. if user_id:
  392. try:
  393. user = await User.get_or_none(id = user_id)
  394. user_info = await User_information.get_or_none(user_id=user_id)
  395. if user is None or user_info is None:
  396. my_log("info",__name__,f"Client IP: {client_ip} - {user_id} has no user resume")
  397. return {"msg": "請去建立工藝家履歷", "code": 500}
  398. email = user.email
  399. subject = "創建課程通知"
  400. # message = f"親愛的工藝家{user_info.name}您好,<br>\
  401. # 感謝您埋下工藝種子!<br>\
  402. # 您創建的課程由管理員進行審核中,將於3-7個工作天內e-mail通知您審核結果。<br>\
  403. # 若7天後尚未收到課程審核通知,請將後台畫面、聯絡資訊等截圖寄送至客服信箱:<br>\
  404. # craftology@ntcri.gov.tw,以便客服為您查詢。<br>\
  405. # 註:此封信件為系統自動發送,請勿回信,謝謝。"
  406. with open("/var/www/ntcri/assets/edm/success_open_class/index.html", 'r', encoding='utf-8') as html_file:
  407. html_template = html_file.read()
  408. message = html_template.replace("{username}",user_info.name)
  409. send_email(email,"",subject,message)
  410. except Exception as e:
  411. return {"msg": str(e), "code": 500}
  412. except:
  413. print("no this user")
  414. return {"msg": "success", "code": 200, "class_id": new_class.id}
  415. except Exception as e:
  416. return {"msg": str(e), "code": 500}
  417. @classes.post("/auto_create_session")
  418. async def auto_create_session(
  419. class_event_id : int = Form(default=0),
  420. week_day_str : str = Form(default="[]"),
  421. start_week_time : str = Form(default="[]"),
  422. end_week_time : str = Form(default="[]"),
  423. ):
  424. try:
  425. week_day = eval(week_day_str)
  426. start_week_time= eval(start_week_time)
  427. end_week_time= eval(end_week_time)
  428. #print(week_day)
  429. class_obj = await Class_list.get_or_none(id = class_event_id)
  430. if class_obj is None :
  431. return {"msg": "no this class", "code": 500}
  432. time_stemp = class_obj.start_time
  433. session = 1
  434. session_list = await Class_detail.filter(class_list_id=class_event_id).delete()
  435. while time_stemp <= class_obj.end_time:
  436. if week_day[time_stemp.weekday()]:
  437. s_start_time = time_stemp.date().strftime("%Y-%m-%d")+" "+start_week_time[time_stemp.weekday()]
  438. s_end_time = time_stemp.date().strftime("%Y-%m-%d")+" "+end_week_time[time_stemp.weekday()]
  439. #time_del = class_obj.end_time.hour - time_stemp.hour
  440. #time_del_minutes = class_obj.end_time.minute - time_stemp.minute
  441. #time_end = time_stemp + timedelta(hours=time_del,minutes=time_del_minutes)
  442. #print(time_end,class_obj.end_time.hour)
  443. day1 = datetime.strptime(s_start_time, '%Y-%m-%d %H:%M:%S')
  444. day2 = datetime.strptime(s_end_time, '%Y-%m-%d %H:%M:%S')
  445. time_difference = day2 - day1
  446. total_minutes = int(time_difference.total_seconds() // 60)
  447. hour=total_minutes//60 + round((total_minutes%60)/60,1)
  448. new_session = await Class_detail.create(
  449. class_list_id=class_event_id,
  450. #start_time=time_stemp,
  451. #end_time=time_end,
  452. start_time=s_start_time,
  453. end_time=s_end_time,
  454. hour = hour,
  455. sessions=session,
  456. content = ""
  457. )
  458. session+=1
  459. time_stemp = time_stemp + timedelta(days=1)
  460. return {"msg": "success", "code": 200}
  461. except Exception as e:
  462. return {"msg": str(e), "code": 500}
  463. @classes.post("/insert_session")
  464. async def insert_session(
  465. class_event_id : int = Form(default=0),
  466. start_time: datetime = Form(default=datetime.now()),
  467. end_time: datetime = Form(default=datetime.now()),
  468. content : str = Form(default='')
  469. ):
  470. try:
  471. session_list = await Class_detail.filter(class_list_id=class_event_id).all()
  472. session = 0
  473. if session_list != []:
  474. for session_obj in session_list:
  475. if session < session_obj.sessions:
  476. session = session_obj.sessions
  477. time_difference = end_time - start_time
  478. session_list = await Class_detail.filter(class_list_id=class_event_id).delete()
  479. total_minutes = int(time_difference.total_seconds() / 60)
  480. hour=total_minutes//60 + round((total_minutes%60)/60,1)
  481. new_session = await Class_detail.create(
  482. class_list_id=class_event_id,
  483. start_time=start_time,
  484. end_time=end_time,
  485. sessions=session +1,
  486. content = content,
  487. hour = hour
  488. )
  489. return {"msg": "success", "code": 200, "new_session_id": new_session.id}
  490. except Exception as e:
  491. return {"msg": str(e), "code": 500}
  492. @classes.post("/update_school")
  493. async def update_school(
  494. location_id: int = Form(default=0),
  495. location_name: str = Form(default=''),
  496. Lng: str = Form(default=''),
  497. Lat: str = Form(default=''),
  498. address : str = Form(default=''),
  499. school_introduction : str = Form(default=None),
  500. email : str = Form(default=None),
  501. phone : str = Form(default=None),
  502. teachers_list : List[str] = Form(default=[]),
  503. is_pass_proposal : int = Form(default=None),
  504. principal_user_email : str = Form(default=None)
  505. ):
  506. try:
  507. school = await Schools.get(id=location_id)
  508. if location_name.strip() != '':
  509. school.name = location_name
  510. if Lng != '':
  511. school.longitude = Lng
  512. if Lat.strip() != '':
  513. school.latitude = Lat
  514. if address.strip() != '':
  515. school.address = address
  516. if school_introduction:
  517. school.introduction = school_introduction
  518. if email:
  519. school.email = email
  520. if phone:
  521. school.phone = phone
  522. if is_pass_proposal is not None:
  523. if is_pass_proposal == 0 and school.is_pass_proposal == 2:
  524. # 寄開課信
  525. try :
  526. if school.create_user_id:
  527. try:
  528. try :
  529. user = await User.get(id = school.create_user_id)
  530. user_info = await User_information.get(user_id = school.create_user_id)
  531. except:
  532. return {"msg": "請去建立工藝家履歷", "code": 500}
  533. email = user.email
  534. subject = "重新提案通知"
  535. # message = f"親愛的工藝家{user_info.name}您好,<br>\
  536. # 感謝您埋下工藝種子!<br>\
  537. # 您的提案由管理員進行審核中,將於3-7個工作天內e-mail通知您審核結果。<br>\
  538. # 若7天後尚未收到課程審核通知,請將後台畫面、聯絡資訊等截圖寄送至客服信箱:<br>\
  539. # craftology@ntcri.gov.tw,以便客服為您查詢。<br>\
  540. # 註:此封信件為系統自動發送,請勿回信,謝謝。"
  541. with open("/var/www/ntcri/assets/edm/porposal/index.html", 'r', encoding='utf-8') as html_file:
  542. html_template = html_file.read()
  543. message = html_template.replace("{username}",user_info.name)
  544. send_email(email,"",subject,message)
  545. except:
  546. print("have no email")
  547. except:
  548. print("no this user")
  549. if is_pass_proposal != 0 and is_pass_proposal ==1 :
  550. if school.create_user_id:
  551. try:
  552. user = await User.get_or_none(id = school.create_user_id)
  553. user_info = await User_information.get(user_id = school.create_user_id)
  554. if user is None :
  555. return {"msg": "請去建立工藝家履歷", "code": 500}
  556. email = user.email
  557. subject = "提案審核通過通知"
  558. # message = f"親愛的工藝家{user_info.name}您好,<br>\
  559. # 恭喜您埋下工藝種子已成功發芽,讓我們一起期待它的成長茁壯!<br>\
  560. # 您的提案已由管理員審核通過!<br>\
  561. # 您可透過「會員專區」→「我的提案」檢視提案內容,並開始開課。"
  562. with open("/var/www/ntcri/assets/edm/pass_porposal/index.html", 'r', encoding='utf-8') as html_file:
  563. html_template = html_file.read()
  564. message = html_template.replace("{username}",user_info.name)
  565. send_email(email,"",subject,message)
  566. except:
  567. print("have no email")
  568. school.is_pass_proposal = is_pass_proposal
  569. if principal_user_email:
  570. try:
  571. principal_user = await User.get(email=principal_user_email,is_avtive=1)
  572. school.principal_user_id = principal_user.id
  573. except:
  574. my_log("info",__name__,f"no this teacher")
  575. if teachers_list != []:
  576. teachers = []
  577. for teacher in teachers_list:
  578. try:
  579. teacher_info = await User.get(email=teacher,is_avtive=1)
  580. teachers.append(teacher_info.id)
  581. except:
  582. my_log("info",__name__,f"no this teacher")
  583. school.teachers = str(teachers)
  584. await school.save()
  585. update_location_time(location_id= location_id)
  586. return {"msg": "success", "code": 200}
  587. except Exception as e:
  588. return {"msg": str(e), "code": 500}
  589. @classes.post("/update_class_name")
  590. async def update_class_name(
  591. request : Request,
  592. class_name_id: int = Form(default=0),
  593. name: str = Form(default=None),
  594. location_id: int = Form(default=None),
  595. category: str = Form(default=None),
  596. introduction: str = Form(default=None),
  597. organizer: str = Form(default=None),
  598. cover_img_file:UploadFile = File(default=""),
  599. group_id : int = Form(default=None),
  600. group_sort : str = Form(default=None),
  601. recommend: int = Form(default=None),
  602. special_class_list_name : str = Form(default=None),
  603. is_inner : int = Form(default=None),
  604. is_check : int = Form(default=None),
  605. teachers_resume : str = Form(default=None),
  606. syllabus : str = Form(default=None)
  607. ):
  608. try:
  609. class_name = await Class_name.get(id=class_name_id)
  610. if name is not None:
  611. class_name.name = name
  612. if location_id is not None:
  613. class_name.school_id = location_id
  614. update_location_time(location_id= location_id)
  615. if category is not None:
  616. class_name.category = category
  617. if introduction is not None:
  618. class_name.introduction = introduction
  619. if organizer is not None:
  620. class_name.organizer = organizer
  621. if group_id is not None :
  622. class_name.group_id = group_id
  623. if is_inner is not None:
  624. class_name.is_inner = is_inner
  625. if syllabus is not None:
  626. class_name.syllabus = syllabus
  627. if teachers_resume is not None:
  628. class_name.teachers_resume = teachers_resume
  629. if cover_img_file != "":
  630. contents = await cover_img_file.read()
  631. with open(f"{IMAGEDIR}{cover_img_file.filename}", "wb") as f:
  632. f.write(contents)
  633. class_name.cover_img = f"{IMAGEDIR_short}{cover_img_file.filename}"
  634. if group_sort is not None:
  635. class_name.group_sort = group_sort
  636. if recommend is not None:
  637. class_name.recommend = recommend
  638. client_ip = request.client.host
  639. if is_check is not None:
  640. if class_name.is_check == 0 and is_check ==1:
  641. user_id = class_name.create_user_id
  642. if user_id:
  643. try:
  644. user = await User.get(id = user_id)
  645. user_info = await User_information.get(user_id = user_id)
  646. except:
  647. # my_log(type,name,msg)
  648. my_log("info",__name__,f"Client IP: {client_ip} - {user_id} has no user resume")
  649. return {"msg": "請去建立工藝家履歷", "code": 500}
  650. email = user.email
  651. subject = "課程核准通知"
  652. # message = f"親愛的工藝家{user_info.name}您好,<br>\
  653. # 恭喜您埋下工藝種子已成功發芽,讓我們一起期待它的成長茁壯!<br>\
  654. # 您創建的課程已由管理員審核通過並上架成功!<br>\
  655. # 您可透過「會員專區」→「我的開課」檢視跟管理學員報名狀況。<br>\
  656. # 註:此封信件為系統自動發送,請勿回信,謝謝。"
  657. with open("/var/www/ntcri/assets/edm/pass_check/index.html", 'r', encoding='utf-8') as html_file:
  658. html_template = html_file.read()
  659. message = html_template.replace("{username}",user_info.name)
  660. try:
  661. send_email(email,"",subject,message)
  662. except:
  663. my_log("error",__name__,f"Client IP: {client_ip} - send email fail")
  664. class_name.is_check = is_check
  665. if special_class_list_name:
  666. class_name.special_class_list_name = special_class_list_name
  667. await class_name.save()
  668. return {"msg": "success", "code": 200}
  669. except Exception as e:
  670. return {"msg": str(e), "code": 500}
  671. @classes.post("/cancel_proposal")
  672. async def cancel_proposal(
  673. request: Request,
  674. location_id: int = Form(default=0),
  675. access_token:str = Form(default=None),
  676. message:str = Form(default=None)
  677. ):
  678. try:
  679. user_id = None
  680. if access_token:
  681. user_id = await check_token(access_token)
  682. if user_id:
  683. user = await User.get(id=user_id)
  684. else :
  685. return {"msg": "no access", "code": 200}
  686. if user.is_superuser != 2:
  687. return {"msg": "no access", "code": 200}
  688. # 檢查是否有該課程
  689. class_name_list = await Schools.filter(id=location_id).all()
  690. if class_name_list == []:
  691. my_log("info",__name__,f"Client IP: {client_ip} - no this class_name_id:{location_id}")
  692. return {"msg": "沒有此課程", "code": 200}
  693. class_name = await Schools.get(id=location_id)
  694. class_name.is_pass_proposal = 2
  695. await class_name.save()
  696. if class_name.create_user_id:
  697. creater = await User.get(id=class_name.create_user_id)
  698. user_info = await User_information.get_or_none(user_id=user_id)
  699. email = creater.email
  700. subject = '提案駁回通知'
  701. # email_message = f'親愛的工藝家{creater.username}您好,<br>\
  702. # 很遺憾您埋下的工藝種子未能成功發芽。<br>\
  703. # 您的提案經管理員審核,因{message}未能通過審核,<br>\
  704. # 請別灰心!只要修正完善內容再次提交即有機會讓您的工藝種子發芽。<br>\
  705. # 若對於審核結果有疑慮,請將後台畫面、聯絡資訊等截圖寄送至客服信箱:<br>\
  706. # craftology@ntcri.gov.tw,以便客服為您查詢。<br>\
  707. # 註:此封信件為系統自動發送,請勿回信,謝謝。'
  708. with open("/var/www/ntcri/assets/edm/porposal_refuse/index.html", 'r', encoding='utf-8') as html_file:
  709. html_template = html_file.read()
  710. email_message = html_template.replace("{username}",user_info.name)
  711. email_message = email_message.replace("{msg}",message)
  712. send_email(email,"",subject,email_message)
  713. else :
  714. print("no creater")
  715. client_ip = request.client.host
  716. my_log("info",__name__,f"{client_ip} - school_id:{location_id} change is_pass_proposal to 2")
  717. return {"msg": "success", "code": 200}
  718. except Exception as e:
  719. client_ip = request.client.host
  720. my_log("error",__name__,f"{client_ip} - An exception occurred: {e} \n Request : {request}")
  721. return {"msg": str(e), "code": 500}
  722. @classes.post("/cancel_class_check")
  723. async def cancel_class_check(
  724. request: Request,
  725. class_name_id: int = Form(default=0),
  726. access_token:str = Form(default=None),
  727. message:str = Form(default=None)
  728. ):
  729. try:
  730. user_id = None
  731. if access_token:
  732. user_id = await check_token(access_token)
  733. if user_id:
  734. user = await User.get(id=user_id)
  735. else :
  736. return {"msg": "no access", "code": 200}
  737. if user.is_superuser != 2:
  738. return {"msg": "no access", "code": 200}
  739. # 檢查是否有該課程
  740. class_name_list = await Class_name.filter(id=class_name_id).all()
  741. if class_name_list == []:
  742. my_log("info",__name__,f"Client IP: {client_ip} - no this class_name_id:{class_name_id}")
  743. return {"msg": "沒有此課程", "code": 200}
  744. class_name = await Class_name.get(id=class_name_id)
  745. class_name.is_check = 2
  746. await class_name.save()
  747. if class_name.create_user_id:
  748. creater = await User.get(id=class_name.create_user_id)
  749. user_info = await User_information.get_or_none(user_id=user_id)
  750. email = creater.email
  751. subject = '開課駁回通知'
  752. # email_message = f'親愛的工藝家{creater.username}您好,<br>\
  753. # 很遺憾您埋下的工藝種子未能成功發芽。<br>\
  754. # 您創建的課程經管理員審核,因{message}未能通過審核,<br>\
  755. # 請別灰心!只要修正完善內容再次提交即有機會讓您的工藝種子發芽。<br>\
  756. # 若對於審核結果有疑慮,請將後台畫面、聯絡資訊等截圖寄送至客服信箱:<br>\
  757. # craftology@ntcri.gov.tw,以便客服為您查詢。<br>\
  758. # 註:此封信件為系統自動發送,請勿回信,謝謝。'
  759. with open("/var/www/ntcri/assets/edm/class_refuse/index.html", 'r', encoding='utf-8') as html_file:
  760. html_template = html_file.read()
  761. email_message = html_template.replace("{username}",user_info.name)
  762. email_message = email_message.replace("{msg}",message)
  763. send_email(email,"",subject,email_message)
  764. else :
  765. print("no creater")
  766. client_ip = request.client.host
  767. my_log("info",__name__,f"{client_ip} - class_name_id:{class_name_id} change is_check to 2")
  768. return {"msg": "success", "code": 200}
  769. except Exception as e:
  770. client_ip = request.client.host
  771. my_log("error",__name__,f"{client_ip} - An exception occurred: {e} \n Request : {request}")
  772. return {"msg": str(e), "code": 500}
  773. @classes.post("/update_event")
  774. async def update_event(
  775. id: int = Form(default=0),
  776. name_id: int = Form(default=None),
  777. event: str = Form(default=None),
  778. start_time: datetime = Form(default=None),
  779. end_time: datetime = Form(default=None),
  780. contact: str = Form(default=None),
  781. lecturer: str = Form(default=None),
  782. location: str = Form(default=None),
  783. content: str = Form(default=None),
  784. URL: str = Form(default=None),
  785. people : str = Form(default=None),
  786. fee_method: str = Form(default=None),
  787. registration_way: str = Form(default=None),
  788. registration_start: datetime = Form(default=None),
  789. registration_end: datetime = Form(default=None),
  790. number_limit: int = Form(default=None),
  791. remark : str = Form(default=None),
  792. ATM_address :str = Form(default=None),
  793. fee_payment : str = Form(default=None),
  794. number_minimum: int = Form(default=None),
  795. files_url = Depends(create_upload_files)
  796. ):
  797. try:
  798. class_obj = await Class_list.get_or_none(id=id)
  799. if class_obj is None :
  800. return {"msg": "no this event", "code": 500}
  801. if name_id != None:
  802. class_obj.name_id = name_id
  803. if event != None:
  804. class_obj.event = event
  805. if start_time != None:
  806. class_obj.start_time = start_time
  807. if end_time != None:
  808. class_obj.end_time = end_time
  809. if lecturer != None:
  810. class_obj.lecturer = lecturer
  811. if location != None:
  812. class_obj.location = location
  813. if contact != None:
  814. class_obj.contact = contact
  815. if content!= None:
  816. class_obj.content = content
  817. if URL!= None:
  818. class_obj.URL = URL
  819. if people!= None:
  820. class_obj.people = people
  821. if fee_method!= None:
  822. class_obj.fee_method = fee_method
  823. if registration_way!= None:
  824. class_obj.registration_way = registration_way
  825. if remark!= None:
  826. class_obj.remark = remark
  827. if ATM_address!= None:
  828. class_obj.ATM_address = ATM_address
  829. if fee_payment!= None:
  830. class_obj.fee_payment = fee_payment
  831. if files_url:
  832. if files_url.__contains__("msg"):
  833. return {"msg": files_url["msg"], "code": 500}
  834. class_obj.files_url = files_url
  835. class_date_obj,created = await Class_date.get_or_create(
  836. class_list_id=id,
  837. defaults={
  838. "registration_start" : registration_start,
  839. "registration_end" : registration_end,
  840. "number_limit" : number_limit,
  841. "amount_left" : number_limit,
  842. "number_minimum" : number_minimum
  843. }
  844. )
  845. if not created :
  846. if registration_start :
  847. class_date_obj.registration_start = registration_start
  848. if registration_end :
  849. class_date_obj.registration_end = registration_end
  850. if number_limit :
  851. class_date_obj.number_limit = number_limit
  852. if number_minimum != None:
  853. class_date_obj.number_minimum = number_minimum
  854. await class_date_obj.save()
  855. await class_obj.save()
  856. return {"msg": "success", "code": 200}
  857. except Exception as e:
  858. return {"msg": str(e), "code": 500}
  859. @classes.post("/update_session")
  860. async def update_session(
  861. session_id : int = Form(default=0),
  862. class_event_id : int = Form(default=0),
  863. start_time: datetime = Form(default=datetime.now()),
  864. end_time: datetime = Form(default=datetime.now()),
  865. sessions: str = Form(default=0),
  866. content : str = Form(default='')
  867. ):
  868. try:
  869. class_session_obj = await Class_detail.get(id=session_id)
  870. if class_event_id != 0:
  871. class_session_obj.class_list_id = class_event_id
  872. if start_time != '':
  873. class_session_obj.start_time = start_time
  874. if end_time != '':
  875. class_session_obj.end_time = end_time
  876. if sessions.strip() != '':
  877. class_session_obj.sessions = sessions
  878. if content.strip() != '':
  879. class_session_obj.content = content
  880. await class_session_obj.save()
  881. return {"msg": "success", "code": 200}
  882. except Exception as e:
  883. return {"msg": str(e), "code": 500}
  884. @classes.post("/delete_school")
  885. async def delete_school(location_id: int):
  886. if location_id:
  887. try:
  888. school = await Schools.get(id=location_id)
  889. except:
  890. return {"msg": "無法找到據點", "code": 500}
  891. school.is_delete = (school.is_delete+1)%2
  892. await school.save()
  893. return {"msg": "success", "code": 200}
  894. @classes.post("/delete_session")
  895. async def delete_session(id: int):
  896. if id:
  897. await Class_detail.filter(id=id).delete()
  898. return {"msg": "success", "code": 200}
  899. @classes.post("/delete_event")
  900. async def delete(id: int):
  901. if id:
  902. await Class_detail.filter(class_list_id=id).delete()
  903. await Class_list.filter(id=id).delete()
  904. await Class_date.filter(class_list_id=id).delete()
  905. return {"msg": "success", "code": 200}
  906. @classes.post("/delete_class_name")
  907. async def delete(id: int):
  908. if id:
  909. try:
  910. class_name = await Class_name.get(id=id)
  911. except:
  912. return {"msg": "無法找到課程", "code": 500}
  913. class_name.is_delete = (class_name.is_delete+1)%2 # 相反狀態(如果本來被刪除就恢復)
  914. await class_name.save()
  915. # class_event_list = await Class_list.filter(name_id=id).all()
  916. # for class_event_obj in class_event_list:
  917. # await Class_detail.filter(class_list_id=class_event_obj.id).delete()
  918. # await Class_date.filter(class_list_id=class_event_obj.id).delete()
  919. # await Class_list.filter(name_id=id).delete()
  920. # await Class_name.filter(id=id).delete()
  921. return {"msg": "success", "code": 200}
  922. @classes.get("/get_class_state")
  923. async def check_date_state(
  924. class_name_id : Optional[int] = None,
  925. class_event_id : Optional[int] = None,
  926. special_class_list_name : Optional[str] = None,
  927. ):
  928. if class_name_id:
  929. Q_word = Q(name_id=class_name_id)
  930. elif class_event_id:
  931. Q_word = Q(id=class_event_id)
  932. else:
  933. return {"msg": 'please input ID', "code": 500}
  934. if special_class_list_name == "one_day_class":
  935. class_list = await One_day_class.filter(Q_word).all()
  936. elif special_class_list_name == "outter_class_list":
  937. class_list = await Outter_class_list.filter(Q_word).all()
  938. else :
  939. class_list = await Class_list.filter(Q_word).all()
  940. result = {"state": "尚未開放報名"}
  941. try:
  942. if class_list == [] :
  943. result["state"] = "尚未創建場次"
  944. for class_obj in class_list:
  945. try:
  946. class_name = await Class_name.get_or_none(id = class_obj.name_id)
  947. if class_name.is_check == 0 :
  948. result["state"] = "課程等待審核"
  949. break
  950. if class_name.is_check == 2 :
  951. result["state"] = "課程審核未通過"
  952. break
  953. if class_name:
  954. print(class_name.is_record_point)
  955. print(class_name.name)
  956. school = await Schools.get_or_none(id = class_name.school_id)
  957. print(class_name.school_id)
  958. if school :
  959. print(school.is_pass_proposal)
  960. if school.is_pass_proposal == 0 :
  961. result["state"] = "正在提案階段"
  962. break
  963. if school.is_pass_proposal == 2 :
  964. result["state"] = "提案已被駁回"
  965. break
  966. if class_name.is_record_point == 1 :
  967. result["state"] = "課程已關閉"
  968. break
  969. if class_obj.start_time and class_obj.end_time:
  970. if class_obj.start_time.replace(tzinfo=None) <= datetime.now() and class_obj.end_time.replace(tzinfo=None) >= datetime.now():
  971. result["state"] = "開課中"
  972. break
  973. elif class_obj.end_time.replace(tzinfo=None) < datetime.now():
  974. result["state"] = "課程已結束"
  975. else:
  976. pass
  977. else :
  978. result["state"] = "尚未創建課程"
  979. except Exception as e:
  980. pass
  981. if special_class_list_name == "one_day_class":
  982. if class_obj.reg_deadline.replace(tzinfo=None) >= datetime.now() :
  983. result["state"] = "報名中"
  984. break
  985. elif class_obj.reg_deadline.replace(tzinfo=None) < datetime.now():
  986. result["state"] = "報名截止(報名額滿)"
  987. break
  988. else :
  989. try:
  990. class_date_obj = await Class_date.get(class_list_id=class_obj.id)
  991. if class_date_obj.registration_start.replace(tzinfo=None) <= datetime.now() and class_date_obj.registration_end.replace(tzinfo=None) >= datetime.now() :
  992. if class_date_obj.amount_left == 0:
  993. result["state"] = "報名截止(報名額滿)"
  994. break
  995. else:
  996. result["state"] = "報名中"
  997. break
  998. elif class_date_obj.registration_end.replace(tzinfo=None) < datetime.now():
  999. result["state"] = "報名截止(人數已滿)"
  1000. break
  1001. except Exception as e:
  1002. pass
  1003. return {"msg": "success", "code": 200, "result": result}
  1004. except Exception as e:
  1005. return {"msg": str(e), "code": 500}
  1006. @classes.get("/get_event")
  1007. async def search_event(
  1008. class_name_id: Optional[int] = None,
  1009. event_id : Optional[int] = None,
  1010. access_token: Optional[str] = None
  1011. ):
  1012. try:
  1013. try :
  1014. class_name_obj = await Class_name.get(id=class_name_id)
  1015. except:
  1016. return {"msg": "no this class id", "code": 500}
  1017. special_class_list_name = class_name_obj.special_class_list_name
  1018. Q_word = Q()
  1019. if class_name_id:
  1020. Q_word = Q_word | Q(name_id = class_name_id)
  1021. elif event_id:
  1022. Q_word = Q_word | Q(id = event_id)
  1023. else :
  1024. return {"msg": "please input class_name_id or event_id", "code": 500}
  1025. user_id = None
  1026. if access_token:
  1027. user_id = await check_token(access_token)
  1028. if user_id:
  1029. Q_word = Q_word & Q(create_user_id = user_id)
  1030. model_fields =[]
  1031. state = {}
  1032. if special_class_list_name==None:
  1033. class_list = Class_list.filter(Q_word).all().order_by("-start_time")
  1034. elif special_class_list_name=='one_day_class':
  1035. class_list = One_day_class.filter(Q_word).all().order_by("-start_time")
  1036. elif special_class_list_name=='outter_class_list':
  1037. class_list = Outter_class_list.filter(Q_word).all().order_by("-start_time")
  1038. else:
  1039. class_list = Class_list.filter(Q_word).all().order_by("-start_time")
  1040. class_list = await class_list.all()
  1041. classes = []
  1042. for class_obj in class_list:
  1043. class_name_obj = await Class_name.get(id=class_obj.name_id)
  1044. class_name = class_name_obj.name
  1045. try :
  1046. class_data = class_obj.show_data()
  1047. class_data["class_name"] = class_name
  1048. state = await check_date_state(class_event_id=class_obj.id,special_class_list_name=special_class_list_name)
  1049. class_data["state"] = state["result"]["state"]
  1050. try:
  1051. class_date_obj = await Class_date.get(class_list_id=class_obj.id)
  1052. class_data["registration_start"] = class_date_obj.registration_start
  1053. class_data["registration_end"] = class_date_obj.registration_end
  1054. class_data["number_limit"] = class_date_obj.number_limit
  1055. class_data["amount_left"] = class_date_obj.amount_left
  1056. class_data["number_minimum"] = class_date_obj.number_minimum
  1057. model_fields.append("registration_start","registration_end","number_limit","amount_left")
  1058. except:
  1059. print("沒有報名時間&人數限制的資料")
  1060. except:
  1061. class_data = {
  1062. "msg" : "fail to get data"
  1063. }
  1064. classes.append(class_data)
  1065. return {"msg": "success", "code": 200, "classes": classes}
  1066. except Exception as e:
  1067. return {"msg": str(e), "code": 500}
  1068. @classes.get("/get_school")
  1069. async def get_school(
  1070. location_id : Optional[int] = None,
  1071. keyword : Optional[str] = None,
  1072. location_keyword : Optional[str] = None,
  1073. page_num : Optional[int] = None,
  1074. page_amount : Optional[int] = None,
  1075. category : Optional[str] = None,
  1076. access_token: Optional[str] = None,
  1077. is_delete: Optional[int] = None,
  1078. is_pass_proposal : Optional[int] = None,
  1079. principal_user_email : Optional[str] = None
  1080. ):
  1081. try:
  1082. await copy()
  1083. my_log("info",__name__,"get_school")
  1084. Q_word = Q()
  1085. if location_id :
  1086. Q_word = Q_word & Q(id = location_id)
  1087. if keyword :
  1088. Q_word = Q_word & Q(name__icontains=keyword)
  1089. if location_keyword :
  1090. Q_word = Q_word & Q(address__icontains=location_keyword)
  1091. if is_pass_proposal is not None :
  1092. Q_word = Q_word &Q(is_pass_proposal=is_pass_proposal)
  1093. if principal_user_email :
  1094. try:
  1095. principal_user = await User.get(email=principal_user_email,is_avtive=1)
  1096. except:
  1097. my_log("info",__name__,f"no this teacher")
  1098. Q_word = Q_word & Q(principal_user_id=principal_user.id)
  1099. if is_delete is not None:
  1100. Q_word = Q_word &Q(is_delete=is_delete)
  1101. else :
  1102. Q_word = Q_word &Q(is_delete=0)
  1103. user_id = None
  1104. if access_token:
  1105. user_id = await check_token(access_token)
  1106. if user_id:
  1107. Q_word = Q_word &Q(create_user_id = user_id)
  1108. Q_word2 = Q()
  1109. if category:
  1110. for tmp_word in category.split(",") :
  1111. Q_word2 = Q_word2 | Q(category__icontains=tmp_word)
  1112. Q_word = Q_word & Q_word2
  1113. school_list = Schools.filter(Q_word)
  1114. # for school_obj in school_list_tmp:
  1115. # class_list = await Class_name.filter(Q(school_id = school_obj.id) & Q_word)
  1116. # if class_list == []:
  1117. # school_list = school_list.exclude(id = school_obj.id)
  1118. count = await school_list.all().count()
  1119. if page_num and page_amount:
  1120. school_list = school_list.offset((page_num-1)*page_amount).limit(page_amount)
  1121. school_list = await school_list.all().order_by("-update_time")
  1122. schools = []
  1123. for school_obj in school_list:
  1124. try :
  1125. school_data = school_obj.show_data()
  1126. if school_data["Lng"] == (None or '') or school_data["Lat"] == (None or ''):
  1127. try:
  1128. # 這裡response採用json格式,可以自由選擇為json?或是xml?
  1129. url = 'https://maps.googleapis.com/maps/api/geocode/json?'
  1130. params = {'key':'AIzaSyDETbkTAmZNOECx2u4rmNudHvyoQTgDbc4', 'address':''}
  1131. params['address'] = school_data["address"]
  1132. #print(params['address'])
  1133. res = requests.get(url, params=params)
  1134. mes = json.loads(res.text)
  1135. school_data["Lat"] = mes['results'][0]['geometry']['location']['lat']
  1136. school_data["Lng"] = mes['results'][0]['geometry']['location']['lng']
  1137. print(school_data["Lat"],school_data["Lng"])
  1138. school_obj.longitude = school_data["Lng"]
  1139. school_obj.latitude = school_data["Lng"]
  1140. await school_obj.save()
  1141. except :
  1142. my_log("error",__name__,f"not get lat and lng")
  1143. schools.append(school_data)
  1144. except:
  1145. schools.append({"msg : data wrong"})
  1146. return {"msg": "success", "code": 200, "total_num" : count,"schools": schools}
  1147. except Exception as e:
  1148. return {"msg": str(e), "code": 500}
  1149. @classes.get("/get_group_name")
  1150. async def get_school_group(
  1151. id : Optional[int] = 0
  1152. ):
  1153. try:
  1154. if id==0:
  1155. school_group_list = await Group_name.all()
  1156. print(school_group_list)
  1157. else:
  1158. school_group_list = [await Group_name.get(id=id)]
  1159. print(school_group_list)
  1160. school_groups = []
  1161. for school_obj in school_group_list:
  1162. school_data = {
  1163. "group_id": school_obj.id,
  1164. "group_name": school_obj.group_name,
  1165. "describe": school_obj.describe
  1166. }
  1167. school_groups.append(school_data)
  1168. return {"msg": "success", "code": 200, "school_groups": school_groups}
  1169. except Exception as e:
  1170. return {"msg": str(e), "code": 500}
  1171. @classes.post("/update_group_name")
  1172. async def update_school_group(
  1173. id : int = Form(default=0),
  1174. group_name : str = Form(default=''),
  1175. describe : str = Form(default=''),
  1176. ):
  1177. try:
  1178. group_name_obj = await Group_name.get(id=id)
  1179. if group_name.strip() != '':
  1180. group_name_obj.group_name = group_name
  1181. if describe.strip() != '':
  1182. group_name_obj.describe = describe
  1183. await group_name_obj.save()
  1184. return {"msg": "success", "code": 200}
  1185. except Exception as e:
  1186. return {"msg": str(e), "code": 500}
  1187. @classes.get("/get_class_name")
  1188. async def get_class_name(
  1189. location_id : str = None ,
  1190. class_name_id : Optional[int] = None,
  1191. group_id : Optional[int] = None,
  1192. group_sort :Optional[str] = None,
  1193. exclude_word: Optional[str] = None,
  1194. category :Optional[str] = None,
  1195. page_num : Optional[int] = None,
  1196. page_amount : Optional[int] = None,
  1197. recommend : Optional[int] = None,
  1198. is_inner : Optional[int] = None,
  1199. is_check : Optional[int] = None,
  1200. access_token: Optional[str] = None,
  1201. is_delete: Optional[int] = None,
  1202. keyword: Optional[str] = None,
  1203. has_user : Optional[int] = None,
  1204. is_record_point : Optional[int] = None,
  1205. location_keyword : Optional[str] = None,
  1206. encode : Optional[str] = None,
  1207. syllabus : Optional[str] = None
  1208. ):
  1209. try:
  1210. await copy()
  1211. class_name_list = Class_name.all()
  1212. Q_word = Q()
  1213. if location_keyword:
  1214. school_list = await Schools.filter(Q(address__icontains=location_keyword)).all()
  1215. schools = []
  1216. for school in school_list:
  1217. schools.append(school.id)
  1218. Q_word = Q_word & Q(school_id__in = schools)
  1219. if group_id:
  1220. Q_word = Q_word & Q(group_id = group_id)
  1221. if group_sort:
  1222. Q_word = Q_word & Q(group_sort = group_sort)
  1223. if encode:
  1224. Q_word = Q_word & Q(encode = encode)
  1225. if syllabus:
  1226. Q_word = Q_word & Q(syllabus__icontains = syllabus)
  1227. if category:
  1228. Q_word2 = Q()
  1229. for tmp_word in category.split(",") :
  1230. Q_word2 = Q_word2 | Q(category__icontains=tmp_word)
  1231. Q_word = Q_word &Q_word2
  1232. if location_id :
  1233. location_id_list = eval(location_id)
  1234. if not isinstance(location_id_list, list):
  1235. location_id_list = [location_id_list]
  1236. Q_word = Q_word & Q(school_id__in = location_id_list)
  1237. if class_name_id :
  1238. Q_word = Q_word & Q(id = class_name_id)
  1239. if recommend :
  1240. Q_word = Q_word & Q(recommend = recommend)
  1241. if is_inner!=None:
  1242. Q_word = Q_word & Q(is_inner = is_inner)
  1243. if is_check!=None:
  1244. if is_check == 3 :
  1245. Q_word = Q_word & Q(is_record_point =1) #已關閉
  1246. else:
  1247. Q_word = Q_word & Q(is_check = is_check,is_record_point__in = [0,None])
  1248. if is_delete is not None:
  1249. Q_word = Q_word & Q(is_delete = is_delete)
  1250. elif is_delete == 2:
  1251. pass
  1252. else :
  1253. Q_word = Q_word & Q(is_delete = 0)
  1254. if keyword:
  1255. Q_word = Q_word & ( Q(name__icontains=keyword)|
  1256. Q(category__icontains=keyword)|
  1257. Q(introduction__icontains=keyword)|
  1258. Q(organizer__icontains=keyword)|
  1259. Q(group_sort__icontains=keyword) |
  1260. Q(syllabus__icontains=keyword))
  1261. if is_record_point :
  1262. Q_word = Q_word & Q(is_record_point =is_record_point)
  1263. user_id = None
  1264. if access_token:
  1265. user_id = await check_token(access_token)
  1266. if user_id:
  1267. Q_word = Q_word & Q(create_user_id = user_id)
  1268. else :
  1269. return {"msg": "no access", "code": 500}
  1270. if has_user==1:
  1271. Q_word = Q_word & Q(create_user_id__isnull=False)
  1272. class_name_list = class_name_list.filter(Q_word).all()
  1273. if exclude_word:
  1274. for tmp_word in exclude_word.split(",") :
  1275. # print(tmp_word)
  1276. class_name_list = class_name_list.exclude(Q(category__icontains=tmp_word)|Q(group_sort__icontains=tmp_word)).all()
  1277. count = await class_name_list.all().count()
  1278. if page_num and page_amount:
  1279. class_name_list = class_name_list.offset((page_num-1)*page_amount).limit(page_amount)
  1280. class_name_list = await class_name_list.all().order_by("-id")
  1281. classes_name = []
  1282. state = {}
  1283. special_class_list_name = None
  1284. for class_name_obj in class_name_list:
  1285. try:
  1286. school_obj = await Schools.get(id=class_name_obj.school_id)
  1287. school_obj_id = school_obj.id
  1288. except:
  1289. school_obj_id = 0
  1290. try:
  1291. special_class_list_name = class_name_obj.special_class_list_name
  1292. if special_class_list_name:
  1293. state = await check_date_state(class_name_id=class_name_obj.id,special_class_list_name=special_class_list_name)
  1294. else:
  1295. state = await check_date_state(class_name_id=class_name_obj.id)
  1296. class_data = class_name_obj.show_data()
  1297. if school_obj_id:
  1298. for key, item in school_obj.show_data().items():
  1299. class_data[key] = item
  1300. class_data["state"]=state["result"]["state"]
  1301. if class_data["state"] == "課程已關閉" :
  1302. class_data["is_check"] = 3
  1303. except Exception as e:
  1304. class_data = {
  1305. "msg" : str(e)
  1306. }
  1307. classes_name.append(class_data)
  1308. return {"msg": "success", "code": 200,"total_num" : count,"classes": classes_name}
  1309. except Exception as e:
  1310. return {"msg": str(e), "code": 500}
  1311. @classes.get("/get_session")
  1312. async def get_session(
  1313. event_id : Optional[int] = None
  1314. ):
  1315. try:
  1316. if not event_id:
  1317. return {"msg": "please input event_id", "code": 500}
  1318. class_session_list = await Class_detail.filter(class_list_id=event_id).all().order_by("start_time")
  1319. classe_sessions = []
  1320. week_day = [0] * 7
  1321. start_week_time= [None] * 7
  1322. end_week_time= [None] * 7
  1323. for class_session_obj in class_session_list:
  1324. class_session_data = {
  1325. "session_id": class_session_obj.id,
  1326. "class_event_id": class_session_obj.class_list_id,
  1327. "start_time": class_session_obj.start_time,
  1328. "end_time": class_session_obj.end_time,
  1329. "sessions": class_session_obj.sessions,
  1330. "content": class_session_obj.content
  1331. }
  1332. week_day[class_session_obj.start_time.weekday()] = 1
  1333. start_week_time[class_session_obj.start_time.weekday()]=class_session_obj.start_time.time().strftime("%H:%M")
  1334. end_week_time[class_session_obj.end_time.weekday()]=class_session_obj.end_time.time().strftime("%H:%M")
  1335. classe_sessions.append(class_session_data)
  1336. if len(class_session_list)== 1 :
  1337. data_form = "one day class"
  1338. else :
  1339. data_form = {
  1340. "week_day" : week_day,
  1341. "start_week_time" : start_week_time,
  1342. "end_week_time" :end_week_time
  1343. }
  1344. return {"msg": "success", "code": 200, "data_form":data_form,"classe_sessions": classe_sessions}
  1345. except Exception as e:
  1346. return {"msg": str(e), "code": 500}
  1347. @classes.get("/search_class_like")
  1348. async def search_class_like(
  1349. keyword: str,
  1350. location_id : Optional[int] = None ,
  1351. group_id : Optional[int] = None,
  1352. group_sort :Optional[str] = None,
  1353. exclude_word: Optional[str] = None,
  1354. category :Optional[str] = None,
  1355. page_num : Optional[int] = None,
  1356. page_amount : Optional[int] = None,
  1357. recommend : Optional[int] = None,
  1358. is_inner : Optional[int] = None,
  1359. is_check : Optional[int] = None
  1360. ):
  1361. try:
  1362. Q_word = Q()
  1363. if group_id:
  1364. Q_word = Q_word & Q(group_id = group_id)
  1365. if group_sort:
  1366. Q_word = Q_word & Q(group_sort = group_sort)
  1367. if category:
  1368. for tmp_word in category.split(",") :
  1369. Q_word = Q_word | Q(category__icontains=tmp_word)
  1370. if location_id :
  1371. Q_word = Q_word & Q(school_id = location_id)
  1372. if recommend :
  1373. Q_word = Q_word & Q(recommend = recommend)
  1374. if is_inner!=None:
  1375. Q_word = Q_word & Q(is_inner = is_inner)
  1376. if is_check!=None:
  1377. Q_word = Q_word & Q(is_check = is_check)
  1378. class_name_id = []
  1379. for class_list in await Class_list.filter(Q(lecturer__icontains=keyword)):
  1380. class_name_id.append(class_list.name_id)
  1381. for class_list in await One_day_class.filter(Q(teacher__icontains=keyword)):
  1382. class_name_id.append(class_list.name_id)
  1383. if keyword :
  1384. Q_word = Q_word & ( Q(name__icontains=keyword)|
  1385. Q(category__icontains=keyword)|
  1386. Q(introduction__icontains=keyword)|
  1387. Q(organizer__icontains=keyword)|
  1388. Q(group_sort__icontains=keyword) |
  1389. Q(id__in=class_name_id)|
  1390. Q(syllabus__icontains=keyword))
  1391. class_name_list = Class_name.filter(Q_word).all()
  1392. count = await class_name_list.all().count()
  1393. if page_num and page_amount:
  1394. class_name_list = class_name_list.offset((page_num-1)*page_amount).limit(page_amount)
  1395. if exclude_word:
  1396. for tmp_word in exclude_word.split(",") :
  1397. # print(tmp_word)
  1398. class_name_list = class_name_list.exclude(Q(category__icontains=tmp_word)|Q(group_sort__icontains=tmp_word)).all()
  1399. class_name_list = await class_name_list.all().order_by("-id")
  1400. classes_name = []
  1401. state = {}
  1402. special_class_list_name = None
  1403. for class_name_obj in class_name_list:
  1404. try:
  1405. school_obj = await Schools.get(id=class_name_obj.school_id)
  1406. school_obj_id = school_obj.id
  1407. except:
  1408. school_obj_id = 0
  1409. try:
  1410. special_class_list_name = class_name_obj.special_class_list_name
  1411. if special_class_list_name:
  1412. state = await check_date_state(class_name_id=class_name_obj.id,special_class_list_name=special_class_list_name)
  1413. else:
  1414. state = await check_date_state(class_name_id=class_name_obj.id)
  1415. class_data = class_name_obj.show_data()
  1416. if school_obj_id:
  1417. for key, item in school_obj.show_data().items():
  1418. class_data[key] = item
  1419. class_data["state"]=state["result"]["state"]
  1420. except Exception as e:
  1421. class_data = {
  1422. "msg" : str(e)
  1423. }
  1424. classes_name.append(class_data)
  1425. return {"msg": "success", "code": 200, "total_num" : count,"classes": classes_name}
  1426. except Exception as e:
  1427. return {"msg": str(e), "code": 500}
  1428. @classes.post("/add_favorite_class")
  1429. async def add_favorite_class(
  1430. class_name_id: int,
  1431. user_id = Depends(check_token),
  1432. time_stemp: datetime = datetime.now()
  1433. ):
  1434. try:
  1435. if not user_id:
  1436. return {"msg": "no access", "code": 500}
  1437. new_favorite_class = await Favorite_course.get_or_create(
  1438. class_name_id=class_name_id,
  1439. user_id=user_id,
  1440. defaults={'time_stemp': time_stemp}
  1441. )
  1442. return {"msg": "success", "code": 200,"is exist": not new_favorite_class[1],"id":new_favorite_class[0].id}
  1443. except Exception as e:
  1444. return {"msg": str(e), "code": 500}
  1445. @classes.get("/get_favorite_class")
  1446. async def get_favorite_class(
  1447. user_id = Depends(check_token),
  1448. no_details : Optional[int] = None
  1449. ):
  1450. try:
  1451. class_list = await Favorite_course.filter(user_id = user_id).all()
  1452. favorite_courses = []
  1453. for class_obj in class_list:
  1454. if no_details:
  1455. class_data = {
  1456. "id": class_obj.id,
  1457. "user_id": class_obj.user_id,
  1458. "class_name_id" : class_obj.class_name_id,
  1459. "time_stemp":class_obj.time_stemp
  1460. }
  1461. favorite_courses.append(class_data)
  1462. else:
  1463. class_data = {
  1464. "id": class_obj.id,
  1465. "user_id": class_obj.user_id,
  1466. "class_name_id" : class_obj.class_name_id,
  1467. "time_stemp":class_obj.time_stemp
  1468. }
  1469. result = await get_class_name(class_name_id = class_obj.class_name_id)
  1470. try:
  1471. class_detail = result["classes"][0]
  1472. for key,item in class_detail.items():
  1473. class_data[key] = item
  1474. favorite_courses.append(class_data)
  1475. except:
  1476. class_data["msg"] = "this class doesn't exit"
  1477. favorite_courses.append(class_data)
  1478. return {"msg": "success", "code": 200, "favorite_courses": favorite_courses}
  1479. except Exception as e:
  1480. return {"msg": str(e), "code": 500}
  1481. @classes.post("/delete_favorite_class")
  1482. async def delete_favorite_class(
  1483. class_name_id: int,
  1484. user_id = Depends(check_token)
  1485. ):
  1486. try:
  1487. await Favorite_course.filter(class_name_id=class_name_id,user_id=user_id).delete()
  1488. return {"msg": "success", "code": 200}
  1489. except Exception as e:
  1490. return {"msg": str(e), "code": 500}
  1491. @classes.post("/insert_online_course")
  1492. async def insert_online_course(
  1493. title : str = Form(default=''),
  1494. category : str = Form(default=''),
  1495. create_time :str = Form(default=datetime.now()),
  1496. content : str = Form(default=''),
  1497. video_url :str = Form(default='')
  1498. ):
  1499. try:
  1500. new_online_course = await Online_course.create(
  1501. title=title,
  1502. create_time=create_time,
  1503. category=category,
  1504. content=content,
  1505. video_url=video_url,
  1506. group_id = 8
  1507. )
  1508. return {"msg": "success", "code": 200, "online_course_obj": new_online_course.id}
  1509. except Exception as e:
  1510. return {"msg": str(e), "code": 500}
  1511. @classes.post("/update_online_course")
  1512. async def update_online_course(
  1513. id : int = Form(default=0),
  1514. title : str = Form(default=''),
  1515. category : str = Form(default=''),
  1516. create_time :str = Form(default=datetime.now()),
  1517. content : str = Form(default=''),
  1518. video_url :str = Form(default='')
  1519. ):
  1520. try:
  1521. online_course_obj = await Online_course.get(id=id)
  1522. if title.strip() != '':
  1523. online_course_obj.title = title
  1524. if category.strip() != '':
  1525. online_course_obj.category = category
  1526. if create_time.strip() != '':
  1527. online_course_obj.create_time = create_time
  1528. if content.strip() != '':
  1529. online_course_obj.content = content
  1530. if video_url.strip() != '':
  1531. online_course_obj.video_url = video_url
  1532. await online_course_obj.save()
  1533. return {"msg": "success", "code": 200}
  1534. except Exception as e:
  1535. return {"msg": str(e), "code": 500}
  1536. @classes.get("/get_online_courese")
  1537. async def get_online_courese(
  1538. online_courese_id : Optional[int] = None,
  1539. category:Optional[str] = None,
  1540. group_id : Optional[int] = None,
  1541. page_num : Optional[int] = None,
  1542. page_amount : Optional[int] = None,
  1543. org :Optional[str] = None,
  1544. no_org:Optional[str] = None,
  1545. key_word:Optional[str] = None,
  1546. ):
  1547. try:
  1548. online_courese_list = Online_course.all()
  1549. Q_word = Q()
  1550. if group_id:
  1551. Q_word = Q_word & Q(group_id = group_id)
  1552. if online_courese_id :
  1553. Q_word = Q_word & Q(id = online_courese_id)
  1554. if org :
  1555. Q_word = Q_word & Q(org = org)
  1556. if no_org:
  1557. no_org_condition = Q(org__isnull=True) | ~Q(org=no_org)
  1558. Q_word = Q_word & no_org_condition
  1559. if category:
  1560. category_conditions = Q()
  1561. for tmp_word in category.split(",") :
  1562. category_conditions |= Q(category__icontains=tmp_word)
  1563. Q_word = Q_word & category_conditions
  1564. #Q_word = Q_word | Q(category__icontains=tmp_word)
  1565. if key_word:
  1566. key_word_condition = Q(title__icontains=key_word) | Q(category__icontains=key_word) | Q(content__icontains=key_word)
  1567. Q_word = Q_word & key_word_condition
  1568. count = await online_courese_list.all().filter(Q_word).count()
  1569. if page_num and page_amount:
  1570. online_courese_list = online_courese_list.offset((page_num-1)*page_amount).limit(page_amount)
  1571. online_courese_list = await online_courese_list.all().filter(Q_word).order_by("id")
  1572. online_coureses = []
  1573. for online_coures_obj in online_courese_list:
  1574. online_coures_data = online_coures_obj.show_data()
  1575. online_coureses.append(online_coures_data)
  1576. return {"msg": "success", "code": 200, "total_num" : count,"online_coures": online_coureses}
  1577. except Exception as e:
  1578. return {"msg": str(e), "code": 500}
  1579. @classes.post("/delete_online_course")
  1580. async def delete_online_course(
  1581. online_course_id : int
  1582. ):
  1583. try:
  1584. await Online_course.filter(id=online_course_id).delete()
  1585. return {"msg": "success", "code": 200}
  1586. except Exception as e:
  1587. return {"msg": str(e), "code": 500}
  1588. @classes.get("/get_group_classes_and_articles")
  1589. async def get_group_classes_and_articles(
  1590. group_id : int,
  1591. page_num : Optional[int] = None,
  1592. page_amount : Optional[int] = None
  1593. ):
  1594. try:
  1595. class_name_list = Class_name.filter(group_id = group_id,is_delete=0).all().order_by("-id")
  1596. article_list = Article_list.filter(group_id = group_id,is_del=0).all().order_by("-id")
  1597. class_count = await class_name_list.all().count()
  1598. article_count = await article_list.all().count()
  1599. if page_num and page_amount:
  1600. class_name_list = class_name_list.offset((page_num-1)*page_amount).limit(page_amount)
  1601. article_list = article_list.offset((page_num-1)*page_amount).limit(page_amount)
  1602. class_name_list = await class_name_list.all()
  1603. article_list = await article_list.all()
  1604. article_objs = []
  1605. for article_obj in article_list:
  1606. try :
  1607. article_tmp = {
  1608. "article_id": article_obj.id,
  1609. "title": article_obj.title,
  1610. "school_id" :article_obj.school_id,
  1611. "group_sort" :article_obj.group_sort,
  1612. "group_id" :article_obj.group_id,
  1613. "category": article_obj.category,
  1614. "create_time" : article_obj.create_time,
  1615. "click_time" : article_obj.click_time,
  1616. "depiction" : article_obj.depiction,
  1617. "content" : article_obj.content,
  1618. "files" : article_obj.files,
  1619. "vedio_url" : article_obj.vedio_url,
  1620. "tags" : article_obj.tags,
  1621. "cover_img": article_obj.cover_img
  1622. }
  1623. except:
  1624. article_tmp = {
  1625. "msg" : "fail to get data"
  1626. }
  1627. article_objs.append(article_tmp)
  1628. classes_name = []
  1629. for class_name_obj in class_name_list:
  1630. try:
  1631. school_obj = await Schools.get(id=class_name_obj.school_id)
  1632. school_obj_id = school_obj.id
  1633. except:
  1634. school_obj_id = 0
  1635. try:
  1636. special_class_list_name = class_name_obj.special_class_list_name
  1637. if special_class_list_name:
  1638. state = await check_date_state(class_name_id=class_name_obj.id,special_class_list_name=special_class_list_name)
  1639. else:
  1640. state = await check_date_state(class_name_id=class_name_obj.id)
  1641. class_data = class_name_obj.show_data()
  1642. if school_obj_id:
  1643. for key, item in school_obj.show_data().items():
  1644. class_data[key] = item
  1645. class_data["state"]=state["result"]["state"]
  1646. except Exception as e:
  1647. class_data = {
  1648. "msg" : str(e)
  1649. }
  1650. classes_name.append(class_data)
  1651. return {"msg": "success", "code": 200,"class_num" : class_count,"classes": classes_name,"article_num":article_count,"articles": article_objs}
  1652. except Exception as e:
  1653. return {"msg": str(e), "code": 500}
  1654. @classes.post("/add_attend_record_by_event")
  1655. async def add_attend_record(
  1656. class_id : int = Form(default=0),
  1657. ):
  1658. try:
  1659. if not class_id:
  1660. return {"msg": "Please input right ","code":500}
  1661. try:
  1662. class_detail_obj = await Class_detail.filter(class_list_id = class_id).all()
  1663. if not isinstance(class_detail_obj, list):
  1664. class_detail_obj = [class_detail_obj]
  1665. registration_obj = await Registration.filter(event_id = class_id,is_del = 0,).all()
  1666. if not isinstance(registration_obj, list):
  1667. registration_obj = [registration_obj]
  1668. if not class_detail_obj:
  1669. return {"msg": "no this class_detail_id session","code":500}
  1670. except:
  1671. return {"msg": "get class_detail_id error","code":500}
  1672. for obj in class_detail_obj:
  1673. class_event = await Class_list.get_or_none(id = class_id)
  1674. if class_event is None:
  1675. return {"msg": "no this class","code":500}
  1676. for obj1 in registration_obj:
  1677. new_record = await Attend_record.get_or_create(
  1678. class_detail_id = obj.id,
  1679. user_id = obj1.user_id,
  1680. defaults={'class_name_id':class_event.name_id,'class_event_id':class_id,'is_attend': 0}
  1681. )
  1682. return {"msg": "success", "code": 200}
  1683. except Exception as e:
  1684. return {"msg": str(e), "code": 500}
  1685. @classes.post("/add_attend_record")
  1686. async def add_attend_record(
  1687. class_detail_id : int = Form(default=0),
  1688. user_id : int = Form(default=0),
  1689. is_attend : int = Form(default=0)
  1690. ):
  1691. try:
  1692. if not class_detail_id or not user_id:
  1693. return {"msg": "Please input right ","code":500}
  1694. try:
  1695. class_detail_obj = await Class_detail.get_or_none(id = class_detail_id)
  1696. class_list = await Class_list.get_or_none(id = class_detail_obj.class_list_id)
  1697. if not class_detail_obj or not class_list:
  1698. return {"msg": "no this class_detail_id","code":500}
  1699. except:
  1700. return {"msg": "get class_detail_id error","code":500}
  1701. msg = ""
  1702. new_record,created = await Attend_record.get_or_create(
  1703. class_detail_id = class_detail_id,
  1704. user_id = user_id,
  1705. defaults = {
  1706. "class_name_id" : class_list.name_id,
  1707. "class_list_id" : class_detail_obj.class_list_id,
  1708. "is_attend" : is_attend
  1709. }
  1710. )
  1711. if not created:
  1712. try:
  1713. new_record.is_attend = is_attend
  1714. await new_record.save()
  1715. msg = "update success"
  1716. except:
  1717. msg = "update fail"
  1718. else:
  1719. msg = "created success"
  1720. return {"msg": msg, "code": 200,"new_record_id":new_record.id}
  1721. except Exception as e:
  1722. return {"msg": str(e), "code": 500}
  1723. @classes.post("/close_class")
  1724. async def close_class(user_id:int = Depends(check_token),class_name_id : int = None,cancel:int=None):
  1725. try:
  1726. if user_id :
  1727. user = await User.get(id=user_id)
  1728. class_name = await Class_name.get_or_none(id = class_name_id)
  1729. if not class_name :
  1730. return {"msg": "no this class", "code": 500}
  1731. elif class_name.is_record_point == 1:
  1732. if cancel==1:
  1733. user_attend_record = await Attend_record.filter(class_name_id=class_name_id,is_attend = 1)
  1734. for tmp in user_attend_record:
  1735. point = await User_point.get_or_none(user_id = tmp.user_id)
  1736. class_detail = await Class_detail.get_or_none(id = tmp.class_detail_id)
  1737. point.hours -= float(class_detail.hour)
  1738. await point.save()
  1739. class_name.is_record_point = 0
  1740. await class_name.save()
  1741. return {"msg": "class closed has cancel", "code": 500}
  1742. else:
  1743. return {"msg": "class has closed", "code": 500}
  1744. else :
  1745. if class_name.create_user_id:
  1746. if class_name.create_user_id != user_id:
  1747. print(class_name.create_user_id,user_id)
  1748. return {"msg": "no access", "code": 200}
  1749. user_attend_record = await Attend_record.filter(class_name_id=class_name_id,is_attend = 1)
  1750. for tmp in user_attend_record:
  1751. point,created = await User_point.get_or_create(
  1752. user_id = tmp.user_id,
  1753. defaults = {
  1754. "hours" : 0.0,
  1755. "points" : 0.0
  1756. }
  1757. )
  1758. class_detail = await Class_detail.get_or_none(id = tmp.class_detail_id)
  1759. point.hours += float(class_detail.hour)
  1760. await point.save()
  1761. result = await count_point(tmp.user_id)
  1762. class_name.is_record_point = 1
  1763. await class_name.save()
  1764. return {"msg":"success close class and record point", "code": 200}
  1765. except Exception as e:
  1766. return {"msg": str(e), "code": 500}
  1767. async def count_point(user_id) :
  1768. point = await User_point.get_or_none(user_id = user_id)
  1769. records = await Point_exchange_record.filter(user_id = user_id)
  1770. if point:
  1771. point.points = int(point.hours/3)
  1772. for record in records:
  1773. point.points -= record.point_exchange
  1774. await point.save()
  1775. return {"msg":"success", "code": 200}
  1776. else:
  1777. return {"msg":"沒有點數資料", "code": 500}
  1778. @classes.post("/delete_attend_record")
  1779. async def delete_attend_record(
  1780. id : int = 0
  1781. ):
  1782. try :
  1783. record = await Attend_record.get(id=id)
  1784. await Attend_record.filter(id=id).delete()
  1785. return {"msg": "success", "code": 200}
  1786. except Exception as e:
  1787. return {"msg": str(e), "code": 500}
  1788. @classes.get("/update_attend_record")
  1789. async def update_attend_record(
  1790. id : int = 0,
  1791. class_detail_id : Optional[int] = None,
  1792. user_id : Optional[int] = None,
  1793. is_attend : Optional[int] = None
  1794. ):
  1795. if not id :
  1796. return {"msg":"please input id"}
  1797. try :
  1798. tmp = await Attend_record.get(id=id)
  1799. if class_detail_id!=None:
  1800. tmp.class_detail_id = class_detail_id
  1801. if user_id!=None:
  1802. tmp.user_id = user_id
  1803. if is_attend!=None:
  1804. tmp.is_attend = is_attend
  1805. await tmp.save()
  1806. return {"msg": "success", "code": 200}
  1807. except Exception as e:
  1808. return {"msg": str(e), "code": 500}
  1809. @classes.get("/get_attend_record")
  1810. async def get_attend_record(
  1811. class_event_id : Optional[int] = None,
  1812. class_detail_id : Optional[int] = None,
  1813. user_id : Optional[int] = None,
  1814. is_attend : Optional[int] = None,
  1815. page_num : Optional[int] = None,
  1816. page_amount : Optional[int] = None
  1817. ):
  1818. try:
  1819. attend_record_list = Attend_record.all()
  1820. if class_event_id:
  1821. id_list = []
  1822. class_detail_list = await Class_detail.filter(class_list_id = class_event_id)
  1823. for tmp in class_detail_list :
  1824. id_list.append(tmp.id)
  1825. attend_record_list = Attend_record.filter(class_detail_id__in=id_list)
  1826. if class_detail_id:
  1827. attend_record_list = attend_record_list.filter(class_detail_id = class_detail_id)
  1828. if user_id :
  1829. attend_record_list = attend_record_list.filter(user_id = user_id)
  1830. if is_attend :
  1831. attend_record_list = attend_record_list.filter(is_attend = is_attend)
  1832. count = await attend_record_list.all().count()
  1833. if page_num and page_amount:
  1834. attend_record_list = attend_record_list.offset((page_num-1)*page_amount).limit(page_amount)
  1835. attend_record_list = await attend_record_list.all()
  1836. attend_records = []
  1837. for obj in attend_record_list:
  1838. class_detail_list = await Class_detail.get_or_none(id = obj.class_detail_id)
  1839. if class_detail_list == None:
  1840. return {"msg": f"class detail number error id={obj.class_detail_id}", "code": 500}
  1841. #registration_list = await Registration.filter(event_id = class_detail_list[0].class_list_id,user_id =obj.user_id)
  1842. user_list = await User.filter(id = obj.user_id)
  1843. user_information_list = await User_information.filter(user_id = obj.user_id)
  1844. class_list = await Class_list.get_or_none(id = obj.class_event_id)
  1845. if class_list != None :
  1846. class_name = await Class_name.get_or_none(id = class_list.name_id)
  1847. #if len(registration_list)==0:
  1848. # continue
  1849. data = {
  1850. "id" :obj.id,
  1851. "class_name_id" : obj.class_name_id,
  1852. "class_event_id":class_detail_list.class_list_id,
  1853. "class_detail_id" :obj.class_detail_id,
  1854. "user_id" :obj.user_id,
  1855. "is_attend" :obj.is_attend,
  1856. #"reg_confirm" : registration_list[0].reg_confirm,
  1857. #"payment_status": registration_list[0].payment_status,
  1858. #"five_digits": registration_list[0].five_digits,
  1859. "real_name": user_information_list[0].name,
  1860. "phone": user_information_list[0].phone,
  1861. "email": user_list[0].email
  1862. }
  1863. if class_list and class_name :
  1864. data["class_name"] = class_name.name
  1865. data["start_time"] = class_detail_list.start_time
  1866. data["end_time"]= class_detail_list.end_time
  1867. attend_records.append(data)
  1868. return {"msg": "success", "code": 200,"total_num" : count,"attend_record_list":attend_records}
  1869. except Exception as e:
  1870. return {"msg": str(e), "code": 500}
  1871. import pymysql
  1872. @classes.get("/copy")
  1873. async def copy(
  1874. from_id : Optional[int] = None
  1875. ):
  1876. try:
  1877. connection = pymysql.connect(
  1878. host='db.ptt.cx',
  1879. user='choozmo',
  1880. password='pAssw0rd',
  1881. database='test_copy'
  1882. )
  1883. class_list_ip = []
  1884. with connection.cursor() as cursor:
  1885. sql = "SELECT * FROM class_name"
  1886. cursor.execute(sql)
  1887. connection.commit()
  1888. rows = cursor.fetchall()
  1889. encode = "HOP"
  1890. now = datetime.now()
  1891. encode += now.strftime("%Y%m%d")
  1892. class_id_list = []
  1893. for row in rows:
  1894. # print(row)
  1895. class_name,created = await Class_name.update_or_create(
  1896. name = row[1],
  1897. defaults = {
  1898. "school_id":1,
  1899. "category" : row[3],
  1900. "introduction" : row[4],
  1901. "organizer" : row[5],
  1902. "cover_img" : row[6],
  1903. "group_id" :9,
  1904. "group_sort" : "希望工程",
  1905. "special_class_list_name" :None,
  1906. "recommend" : 0,
  1907. "is_inner" :1,
  1908. "is_check" :1,
  1909. "create_user_id" :None,
  1910. "create_time" : datetime.now(),
  1911. "is_delete" : row[10],
  1912. "is_record_point" :0,
  1913. "encode" :encode
  1914. }
  1915. )
  1916. #print(class_name.id)
  1917. sql = f"SELECT * FROM schools where id = {row[2]}"
  1918. cursor.execute(sql)
  1919. connection.commit()
  1920. school_row = cursor.fetchone()
  1921. # print(school_row)
  1922. try:
  1923. school,school_created = await Schools.update_or_create(
  1924. name = school_row[1],
  1925. defaults = {
  1926. "longitude" : school_row[2],
  1927. "latitude" : school_row[3],
  1928. "address" :school_row[4],
  1929. "update_time" :school_row[5],
  1930. "school_introduction":None,
  1931. "email":None,
  1932. "phone":None,
  1933. "school_create_user_id" :None,
  1934. "teachers" : None,
  1935. "is_delete" : 0,
  1936. "is_pass_proposal" : 1
  1937. }
  1938. )
  1939. class_name.school_id = school.id
  1940. await class_name.save()
  1941. #print("step 1 complete")
  1942. sql = f"SELECT * FROM class_list where name_id = {row[0]}"
  1943. cursor.execute(sql)
  1944. connection.commit()
  1945. list_row = cursor.fetchone()
  1946. # print(list_row)
  1947. event,event_created = await Class_list.update_or_create(
  1948. name_id = class_name.id,
  1949. defaults = {
  1950. "event": list_row[2],
  1951. "start_time": list_row[3],
  1952. "end_time": list_row[4],
  1953. "location": list_row[10],
  1954. "lecturer": list_row[9],
  1955. "contact": list_row[8],
  1956. "content": list_row[11],
  1957. "URL": list_row[12],
  1958. "people": None,
  1959. "fee_method": list_row[19],
  1960. "registration_way": list_row[15],
  1961. "remark": list_row[16],
  1962. "ATM_address":list_row[17],
  1963. "fee_payment" :list_row[18],
  1964. "create_user_id" :list_row[19],
  1965. "event_create_time" : datetime.now(),
  1966. "files" : list_row[20]
  1967. }
  1968. )
  1969. #print("step 2 complete")
  1970. date,date_create = await Class_date.update_or_create(
  1971. class_list_id = event.id,
  1972. defaults = {
  1973. "registration_start": list_row[5],
  1974. "registration_end": list_row[6],
  1975. "number_limit": list_row[13],
  1976. "amount_left": 0
  1977. }
  1978. )
  1979. #print("step 3 complete")
  1980. class_id_list.append(class_name.id)
  1981. except :
  1982. await Class_name.filter(id = class_name.id).delete()
  1983. my_log("error",__name__,f"{row[0]}copy error")
  1984. class_delete_list = await Class_name.filter(group_id = 9,group_sort = "希望工程").exclude(id__in = class_id_list).delete()
  1985. for class_ in class_delete_list :
  1986. print("class_delete:",class_.id)
  1987. cursor.close()
  1988. connection.close()
  1989. return {"msg": "success", "code": 500}
  1990. except Exception as e:
  1991. return {"msg": str(e), "code": 500}
  1992. @classes.post("/insert_proposal")
  1993. async def insert_class_name(
  1994. class_name: str = Form(default=None),
  1995. school_id : int = Form(default=None),
  1996. category : str = Form(default=None),
  1997. introduction: str = Form(default=None),
  1998. organizer: str = Form(default=None),
  1999. cover_img_file : UploadFile = File(default=None),
  2000. fee_method: str = Form(default=None),
  2001. number_limit: int = Form(default=None),
  2002. number_minimum : int = Form(default=None),
  2003. people : str = Form(default=None),
  2004. files = Depends(create_upload_files)
  2005. ):
  2006. try:
  2007. cover_img = ''
  2008. if cover_img_file != None:
  2009. contents = await cover_img_file.read()
  2010. #save the file
  2011. with open(f"{IMAGEDIR}{cover_img_file.filename}", "wb") as f:
  2012. f.write(contents)
  2013. cover_img = f"{IMAGEDIR_short}{cover_img_file.filename}"
  2014. if files:
  2015. if files.__contains__("msg"):
  2016. return {"msg": files["msg"], "code": 500}
  2017. school = await Schools.get_or_none(id = school_id)
  2018. if school is None :
  2019. return {"msg": f"no this school id {school_id}", "code": 500}
  2020. new_proposal,created = await Proposal.get_or_create(
  2021. school_id =school_id,
  2022. defaults = {
  2023. "class_name":class_name,
  2024. "category":category,
  2025. "introduction":introduction,
  2026. "organizer":organizer,
  2027. "cover_img":cover_img,
  2028. "fee_method":fee_method,
  2029. "number_limit":number_limit,
  2030. "number_minimum":number_minimum,
  2031. "people":people,
  2032. "files":files,
  2033. "create_time":datetime.now()
  2034. }
  2035. )
  2036. if not created:
  2037. if class_name:
  2038. new_proposal.class_name = class_name
  2039. if category:
  2040. new_proposal.category = category
  2041. if introduction:
  2042. new_proposal.introduction = introduction
  2043. if cover_img_file:
  2044. new_proposal.cover_img = cover_img
  2045. if fee_method:
  2046. new_proposal.fee_method = fee_method
  2047. if number_limit:
  2048. new_proposal.number_limit = number_limit
  2049. if number_minimum:
  2050. new_proposal.number_minimum = number_minimum
  2051. if people:
  2052. new_proposal.people = people
  2053. if files != "{}":
  2054. new_proposal.files = files
  2055. await new_proposal.save()
  2056. else:
  2057. user_id = school.create_user_id
  2058. if user_id:
  2059. try:
  2060. try :
  2061. user = await User.get(id = user_id)
  2062. user_info = await User_information.get(user_id = user_id)
  2063. except:
  2064. return {"msg": "請去建立工藝家履歷", "code": 500}
  2065. email = user.email
  2066. subject = "課程提案通知"
  2067. # message = f"親愛的工藝家{user_info.name}您好,<br>\
  2068. # 感謝您埋下工藝種子!<br>\
  2069. # 您的提案將由管理員進行審核中,將於3-7個工作天內e-mail通知您審核結果。<br>\
  2070. # 若7天後尚未收到課程審核通知,請將後台畫面、聯絡資訊等截圖寄送至客服信箱:<br>\
  2071. # craftology@ntcri.gov.tw,以便客服為您查詢。<br>\
  2072. # 註:此封信件為系統自動發送,請勿回信,謝謝。"
  2073. with open("/var/www/ntcri/assets/edm/porposal/index.html", 'r', encoding='utf-8') as html_file:
  2074. html_template = html_file.read()
  2075. message = html_template.replace("{username}",user_info.name)
  2076. send_email(email,"",subject,message)
  2077. except:
  2078. print("have no email")
  2079. return {"msg": "success", "code": 200, "new_proposal_id": new_proposal.id}
  2080. except Exception as e:
  2081. return {"msg": str(e), "code": 500}
  2082. @classes.get("/get_proposal")
  2083. async def insert_class_name(
  2084. id : Optional[int] = None,
  2085. school_id : Optional[int] = None
  2086. ):
  2087. try:
  2088. proposal_list = Proposal.all()
  2089. Q_word = Q()
  2090. if id :
  2091. Q_word = Q(id=id)
  2092. if school_id :
  2093. Q_word = Q_word & Q(school_id=school_id)
  2094. proposal_list = await proposal_list.filter(Q_word)
  2095. result = []
  2096. for proposal in proposal_list:
  2097. school = await Schools.get_or_none(id=proposal.school_id)
  2098. if school is None :
  2099. return {"msg": f"no this school,id = {proposal.school_id}", "code": 500}
  2100. info = proposal.show_data()
  2101. info["school"]=school.show_data()
  2102. result.append(info)
  2103. return {"msg": "success", "code": 200, "proposal_list": result}
  2104. except Exception as e:
  2105. return {"msg": str(e), "code": 500}
  2106. @classes.get("/get_favorite_course_count")
  2107. async def insert_class_name(
  2108. class_name_id_list : str = None
  2109. ):
  2110. try:
  2111. result = []
  2112. if class_name_id_list :
  2113. class_name_id = eval(class_name_id_list)
  2114. distinct_class_name_ids = list(map(int,class_name_id))
  2115. for entry in distinct_class_name_ids:
  2116. class_num = await Favorite_course.filter(class_name_id = entry).count()
  2117. result.append({"class_name_id":entry,"count":class_num})
  2118. else:
  2119. # 使用 Tortoise ORM 的 count 方法進行統計
  2120. distinct_class_name_ids = await Favorite_course.all().values_list('class_name_id')
  2121. # print(distinct_class_name_ids)
  2122. for entry in distinct_class_name_ids:
  2123. class_num = await Favorite_course.filter(class_name_id = entry[0]).count()
  2124. result.append({"class_name_id":entry[0],"count":class_num})
  2125. return {"msg": "success", "code": 200, "result": result}
  2126. except Exception as e:
  2127. return {"msg": str(e), "code": 500}