Le repo des sources pour le site web des JM2L
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

views.py 80 KiB

9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808
  1. # -*- coding: utf8 -*-
  2. from pyramid.httpexceptions import HTTPFound, HTTPNotFound, HTTPForbidden
  3. from pyramid.httpexceptions import HTTPBadRequest, HTTPUnauthorized
  4. from pyramid.renderers import render_to_response
  5. from pyramid.view import notfound_view_config, forbidden_view_config
  6. from pyramid.view import view_config
  7. from mako.template import Template
  8. from .upload import IMAGEPATH
  9. # Import Web Forms
  10. from .forms import *
  11. # Database access imports
  12. from .models import *
  13. from .helpers import Orga_helpers
  14. from sqlalchemy import func, or_
  15. from os import path, makedirs
  16. # Usefull tools
  17. from slugify import slugify
  18. from icalendar import Calendar
  19. from pytz import timezone
  20. from icalendar import Event as Evt
  21. from pyramid_mailer.message import Message
  22. # Then, standard libs
  23. import webhelpers.paginate as paginate
  24. import unicodedata
  25. import datetime
  26. import re
  27. import shutil
  28. CurrentYear = 2015
  29. ## =-=- Here, We keep some usefull function -=-=
  30. def remove_accents(input_str):
  31. """ This function is intended to remove all accent from input unicode string """
  32. nkfd_form = unicodedata.normalize('NFKD', input_str)
  33. only_ascii = nkfd_form.encode('ASCII', 'ignore')
  34. return only_ascii
  35. def embeed_video(mime_type, link):
  36. Container = "<video controls='controls' preload='metadata' style='width:60%'>"
  37. Container += "<source type='%s' " % mime_type
  38. Container += "src='%s' />" % link
  39. Container += "</video>"
  40. return Container
  41. ## =-=- Here, We handle ICal requests -=-=
  42. @view_config(route_name='progr_iCal', renderer="string")
  43. def ICal_Progamme_Request(request):
  44. year = int(request.matchdict.get('year', CurrentYear))
  45. # Initialization
  46. # Compute days used by all events matching the specified input year
  47. Events = DBSession.query(Event)\
  48. .filter(Event.for_year == year)\
  49. .filter(Event.event_type != 'Stand')\
  50. .order_by(Event.start_time)
  51. cal = Calendar()
  52. cal.add('prodid', '-//Programme %d//jm2l.linux-azur.org//' % year)
  53. cal.add('version', '2.0')
  54. tz = timezone('Europe/Paris')
  55. for ev in Events:
  56. if ev.event_type:
  57. event = Evt()
  58. event['uid'] = "%d/%d" % ( year, ev.uid )
  59. event.add('summary', ev.name )
  60. event.add('dtstart', ev.start_time.replace(tzinfo=tz) )
  61. event.add('dtend', ev.end_time.replace(tzinfo=tz) )
  62. event.add('created', ev.last_change.replace(tzinfo=tz) )
  63. event.add('description', "http://www.linux-azur.org/event/%s/%s" % (ev.for_year, ev.slug) )
  64. event.add('url', "http://www.linux-azur.org/event/%s/%s" % (ev.for_year, ev.slug) )
  65. event.add('priority', 5)
  66. cal.add_component(event)
  67. request.response.content_type = "text/calendar"
  68. return cal.to_ical()
  69. ## =-=- Here, We handle Json requests -=-=
  70. @view_config(route_name='users_json', renderer="json")
  71. def JSON_User_Request(request):
  72. """ Build a JSON answer with active users and pagination handling """
  73. # Check arguments consitency
  74. pageSize = request.params.get('pageSize',"8")
  75. current_page = request.params.get('pageNum',"1")
  76. UserQuery = request.params.get('searchTerm', u"")
  77. # Don't answer to users that aren't logged
  78. if not request.user:
  79. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  80. # Check consistancy of parameters
  81. if pageSize.isdigit() and current_page.isdigit():
  82. current_page = int(current_page)
  83. pageSize = int(pageSize)
  84. else:
  85. return HTTPBadRequest('pageSize and pageNum accept only digits.')
  86. # Query database
  87. Users = DBSession.query(User.uid, User.nom, User.prenom)\
  88. .filter(User.slug.contains( remove_accents(UserQuery) ))
  89. page_url = paginate.PageURL_WebOb(request)
  90. records = paginate.Page(Users, current_page, url=page_url, items_per_page=pageSize)
  91. ListMatchUser = map( lambda u:{"id": u.uid, "text":"%s %s" % ( u.prenom, u.nom )}, records )
  92. return { "Results": ListMatchUser, "Total":records.item_count,
  93. "logged_in":request.authenticated_userid }
  94. @view_config(route_name='tiers_json', renderer="json")
  95. def JSON_Tiers_Request(request):
  96. """ Build a JSON answer with active users and pagination handling """
  97. # Check arguments consitency
  98. pageSize = request.params.get('pageSize',"8")
  99. current_page = request.params.get('pageNum',"1")
  100. TiersQuery = request.params.get('searchTerm', u"")
  101. # Don't answer to users that aren't logged
  102. if not request.user:
  103. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  104. # Check consistancy of parameters
  105. if pageSize.isdigit() and current_page.isdigit():
  106. current_page = int(current_page)
  107. pageSize = int(pageSize)
  108. else:
  109. return HTTPBadRequest('pageSize and pageNum accept only digits.')
  110. # Query database
  111. JTiers = DBSession.query(Tiers.uid, Tiers.name)\
  112. .filter(Tiers.slug.contains( remove_accents(TiersQuery) ))
  113. page_url = paginate.PageURL_WebOb(request)
  114. records = paginate.Page(JTiers, current_page, url=page_url, items_per_page=pageSize)
  115. ListMatchTiers = map( lambda t:{"id": t.uid, "text": t.name }, records )
  116. return { "Results": ListMatchTiers, "Total":records.item_count,
  117. "logged_in":request.authenticated_userid }
  118. @view_config(route_name='progr_json', renderer="json")
  119. def JSON_Progamme_Request(request):
  120. year = int(request.matchdict.get('year', CurrentYear))
  121. # Initialization
  122. DicResult = dict()
  123. # Query database
  124. # Compute days used by all events matching the specified input year
  125. Days = DBSession.query( func.strftime('%d', Event.start_time).label('day') )\
  126. .filter(Event.for_year == year)\
  127. .filter(Event.event_type != None)\
  128. .group_by(func.strftime('%d', Event.start_time)).all()
  129. for Day in Days:
  130. Events = DBSession.query(Event)\
  131. .filter(Event.for_year == year)\
  132. .filter(Event.event_type != 'Stand')\
  133. .filter("strftime('%d', start_time) = :dow").params(dow=Day.day)\
  134. .order_by(Event.start_time)
  135. ListEv = []
  136. for ev in Events:
  137. if ev.event_type:
  138. ListEv.append( {
  139. "uid":"%d/%d" % ( year, ev.uid ),
  140. "desc":ev.name,
  141. "startDate":ev.start_time.strftime('%Y-%m-%dT%H:%M:%S'),
  142. "endDate":ev.end_time.strftime('%Y-%m-%dT%H:%M:%S'),
  143. "placeName":ev.Salle and (ev.Salle.name or "unk") ,
  144. "status":ev.event_type
  145. } )
  146. DicResult[Day.day] = ListEv
  147. return { 'all':DicResult }
  148. @view_config(route_name='timeline_json', renderer="json")
  149. def JSON_TimeLine_Request(request):
  150. year = int(request.matchdict.get('year', CurrentYear))
  151. # Initialization
  152. DicResult = dict()
  153. # Query database
  154. # Compute days used by all events matching the specified input year
  155. Days = DBSession.query( func.strftime('%d', Event.start_time).label('day') )\
  156. .filter(Event.for_year == year)\
  157. .filter(Event.event_type != None)\
  158. .group_by(func.strftime('%d', Event.start_time)).all()
  159. ListEv = []
  160. for Day in Days:
  161. Events = DBSession.query(Event)\
  162. .filter(Event.for_year == year)\
  163. .filter(Event.event_type != 'Stand')\
  164. .filter("strftime('%d', start_time) = :dow").params(dow=Day.day)\
  165. .order_by(Event.start_time)
  166. #ListEv = []
  167. for ev in Events:
  168. if ev.event_type:
  169. CurMedia = ev.video.first() or ""
  170. if CurMedia:
  171. Container = "<video controls='controls' preload='metadata' style='width:60%'>"
  172. Container += "<source type='%s' " % CurMedia.mime_type
  173. Container += "src='%s' />" % CurMedia.get_path
  174. Container += "</video>"
  175. else:
  176. Container = ""
  177. ListEv.append( {
  178. #"uid":"%d/%d" % ( year, ev.uid ),
  179. "headline":ev.name,
  180. "startDate":ev.start_time.strftime('%Y,%m,%d,%H,%M'),
  181. "endDate":ev.end_time.strftime('%Y,%m,%d,%H,%M'),
  182. "text":ev.Salle and (ev.Salle.name or "unk"),
  183. #"text":ev.description[:100],
  184. "tags":ev.Salle and (ev.Salle.salle_id or "unk") ,
  185. #"status":ev.event_type,
  186. "asset": {
  187. "media": Container,
  188. "credit": ",".join(["%s %s" % (i.prenom, i.nom) for i in ev.intervenants]),
  189. "caption":"" }
  190. } )
  191. if year==2015:
  192. DicResult = {
  193. "lang":"fr",
  194. "headline":"JM2L 2015",
  195. "type":"default",
  196. "startDate":"2015,11,28,10",
  197. "text":"<i><span class='c1'>9ème Édition</span></i>",
  198. "asset":
  199. {
  200. "media":"https://www.youtube.com/watch?v=DnfjrxVoLao",
  201. "credit":"JM2L",
  202. "caption":""
  203. }
  204. }
  205. elif year==2011:
  206. DicResult = {
  207. "lang":"fr",
  208. "headline":"JM2L 2011",
  209. "type":"default",
  210. "startDate":"2011,11,25,10",
  211. "text":"<i><span class='c1'>6ème Édition</span></i>",
  212. "asset":
  213. {
  214. "media":"https://www.youtube.com/embed/rcaNeXuAEhs",
  215. "credit":"JM2L",
  216. "caption":"Reportage FR3"
  217. },
  218. }
  219. elif year==2010:
  220. DicResult = {
  221. "lang":"fr",
  222. "headline":"JM2L 2010",
  223. "type":"default",
  224. "text":"<i><span class='c1'>5ème Édition</span></i>",
  225. "asset":
  226. {
  227. "media":embeed_video("video/ogg","http://jm2l.linux-azur.org/resources/2010/Video/reportages/JM2L2010-PleinSudTV.ogv"),
  228. "credit":"<a href='http://pleinsudtv.com/index.php/casa/affichage-en-vignettes/93-casa-culture/582-les-jm-du-logiciel-libre-2010'>Le reportage Plein-sud TV</a>",
  229. "caption":"JM2L",
  230. }
  231. }
  232. elif year==2007:
  233. DicResult = {
  234. "lang":"fr",
  235. "headline":"JM2L 2007",
  236. "type":"default",
  237. "text":"<i><span class='c1'>2ème Édition</span></i>",
  238. "asset":
  239. {
  240. "media":embeed_video("video/ogg","http://jm2l.linux-azur.org/resources/2007/Video/20071110-linux.ogv"),
  241. "credit":"<a href='http://pleinsudtv.com/index.php/casa/affichage-en-vignettes/93-casa-culture/245-logiciel-libre-linux-jm2l'>Le reportage Plein-sud TV</a>",
  242. "caption":"JM2L 2007",
  243. }
  244. }
  245. else:
  246. DicResult = {
  247. "lang":"fr",
  248. "headline":"JM2L %d" % year,
  249. "type":"default",
  250. "asset":
  251. {
  252. "media":"",
  253. "credit":"JM2L",
  254. "caption":""
  255. }
  256. }
  257. DicResult["date"] = ListEv
  258. return { 'timeline':DicResult }
  259. ## =-=- Here, We handle HTTP requests - Public Part -=-=
  260. @view_config(route_name='home', renderer="jm2l:templates/NewIndex.mako")
  261. def index_page(request):
  262. year = request.matchdict.get('year')
  263. if year:
  264. year=int(year[:-1])
  265. content = DBSession.query(JM2L_Year).filter(JM2L_Year.year_uid==year).first()
  266. if content:
  267. content = content.description
  268. else:
  269. content = ""
  270. if 2004<year<=CurrentYear:
  271. if year==2006:
  272. return {'year': year, 'content':content, 'edition':u"1<sup>ère</sup>" }
  273. elif year==2015:
  274. return {'year': year, 'content':content, 'edition':u"9<sup>ème</sup>" }
  275. else:
  276. edition = year - 2005
  277. return {'year': year, 'content':content, 'edition':u"%d<sup>ème</sup>" % edition }
  278. else:
  279. raise HTTPNotFound()
  280. else:
  281. content = DBSession.query(JM2L_Year).filter(JM2L_Year.year_uid==CurrentYear).first().description
  282. return {'year': CurrentYear, 'content':content, 'edition':u"9<sup>ème</sup>"}
  283. @view_config(route_name='edit_index', renderer="jm2l:templates/Staff/EditIndex.mako")
  284. def edit_index(request):
  285. year = int(request.matchdict.get('year', None))
  286. if not request.user.Staff:
  287. # Don't answer to users that aren't logged
  288. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  289. content = DBSession.query(JM2L_Year).filter(JM2L_Year.year_uid==year).first()
  290. form = IndexForm(request.POST, content, meta={'csrf_context': request.session})
  291. if request.method == 'POST' and form.validate():
  292. form.populate_obj(content)
  293. return HTTPFound(location=request.route_url('home', year="%d/" % year))
  294. MainTab = {'home':'active', "logged_in":request.authenticated_userid,
  295. 'form':form, 'DisplayYear':year}
  296. return MainTab
  297. @view_config(route_name='programme', renderer="jm2l:templates/Public/Programme.mako")
  298. def programme(request):
  299. year = int(request.matchdict.get('year'))
  300. if 2006 > year:
  301. return HTTPBadRequest('The first JM2L event was in 2006.')
  302. # Query database about selected Year.
  303. Events = DBSession.query(Event)\
  304. .filter(Event.for_year == year)\
  305. .order_by(Event.start_time)
  306. Days = DBSession.query(func.strftime('%d-%m-%Y', Event.start_time))\
  307. .filter(Event.for_year == year)\
  308. .filter(Event.event_type != None)\
  309. .group_by(func.strftime('%d', Event.start_time)).all()
  310. ListDay = []
  311. for day in Days:
  312. RefDay = datetime.datetime.strptime(day[0],'%d-%m-%Y')
  313. ListDay.append( ( RefDay.strftime('%A %d %b %Y'),
  314. RefDay.strftime('%d') ) )
  315. MainTab = {'programme':'active','DisplayYear':year, \
  316. 'Events':Events, 'Event':Event, 'Days':ListDay, "logged_in":request.authenticated_userid }
  317. return MainTab
  318. @view_config(route_name='presse', renderer="jm2l:templates/Public/Presse.mako")
  319. def static_presse(request):
  320. year = int(request.matchdict.get('year', None))
  321. content = DBSession.query(JM2L_Year).filter(JM2L_Year.year_uid==year).first()
  322. MainTab = {'presse':'active', "logged_in":request.authenticated_userid, 'content':content, 'DisplayYear':year}
  323. return MainTab
  324. @view_config(route_name='edit_presse', renderer="jm2l:templates/Staff/EditPresse.mako")
  325. def edit_presse(request):
  326. year = int(request.matchdict.get('year', None))
  327. if request.user is None:
  328. # Don't answer to users that aren't logged
  329. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  330. if not request.user.Staff:
  331. # Don't answer to users that aren't logged
  332. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  333. content = DBSession.query(JM2L_Year).filter(JM2L_Year.year_uid==year).first()
  334. form = DossPresse(request.POST, content, meta={'csrf_context': request.session})
  335. if request.method == 'POST' and form.validate():
  336. form.populate_obj(content)
  337. MainTab = {'presse':'active', "logged_in":request.authenticated_userid, 'form':form, 'DisplayYear':year}
  338. return MainTab
  339. @view_config(route_name='plan', renderer="jm2l:templates/Public/Plan.mako")
  340. def static_plan(request):
  341. session = request.session
  342. session['year'] = 2015
  343. MainTab = {'plan':'active', "logged_in":request.authenticated_userid }
  344. return MainTab
  345. ## =-=- Here, We handle HTTP requests - Staff Logged Part -=-=
  346. @view_config(route_name='list_task', renderer='jm2l:templates/Staff/list.mako')
  347. def list_view(request):
  348. if request.user is None:
  349. # Don't answer to users that aren't logged
  350. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  351. if not request.user.Staff:
  352. # Don't answer to users that aren't logged
  353. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  354. DicTask = {}
  355. taskgroup = DBSession.query( TasksArea ).all()
  356. for grp in taskgroup:
  357. tasks = DBSession.query( Tasks )\
  358. .filter( Tasks.area_uid==grp.uid )\
  359. .order_by(Tasks.closed, Tasks.due_date).all()
  360. DicTask[grp] = tasks
  361. return {'tasks': DicTask }
  362. @view_config(route_name='handle_task', renderer='jm2l:templates/Staff/tasks.mako')
  363. def tasks(request):
  364. if request.user is None:
  365. # Don't answer to users that aren't logged
  366. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  367. if not request.user.Staff:
  368. # Don't answer to users that aren't logged
  369. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  370. task_id = request.matchdict.get('task_id')
  371. # Convert the pole_id GET parameter to int or 0
  372. if request.params.get('pole_id') and request.params.get('pole_id').isdigit():
  373. pole_id = int(request.params.get('pole_id'))
  374. else:
  375. pole_id = None
  376. # Get areas from db
  377. Areas = DBSession.query(TasksArea.uid, TasksArea.name)\
  378. .order_by('name').all()
  379. # Get users from db
  380. Users = DBSession.query(User)\
  381. .filter(User.Staff==1)\
  382. .order_by('nom').all()
  383. if task_id:
  384. TmpTask = Tasks.by_id(int(task_id))
  385. if not TmpTask:
  386. raise HTTPNotFound()
  387. form = EditStaffTasks(request.POST, TmpTask, meta={'csrf_context': request.session})
  388. else:
  389. TmpTask = Tasks()
  390. # Check if the supplied pole_id is in the Areas' range
  391. form = StaffTasks(request.POST, TmpTask, meta={'csrf_context': request.session})
  392. Area = TasksArea.by_id(pole_id)
  393. if Area:
  394. form.area_uid.data = Area.uid
  395. # Put some areas on form
  396. form.area_uid.choices = Areas
  397. # Put some users on form
  398. form.closed_by.choices = [(u.uid, "%s %s" % (u.nom, u.prenom))
  399. for u in Users]
  400. form.due_date.type = "date"
  401. if request.method=='POST' and form.validate():
  402. form.populate_obj(TmpTask)
  403. TmpTask.closed = False
  404. if 'uid' in form._fields.keys():
  405. DBSession.merge(TmpTask)
  406. else:
  407. DBSession.add(TmpTask)
  408. DBSession.flush()
  409. return HTTPFound(location=request.route_url('list_task')+"#"+slugify(TmpTask.area.name))
  410. return {'form':form, 'area':TmpTask.area and slugify(TmpTask.area.name) or ''}
  411. @view_config(route_name='handle_pole', renderer='jm2l:templates/Staff/pole.mako')
  412. def tasks_area(request):
  413. if request.user is None:
  414. # Don't answer to users that aren't logged
  415. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  416. if not request.user.Staff:
  417. # Don't answer to users that aren't logged
  418. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  419. pole_id = request.matchdict.get('pole_id')
  420. if pole_id:
  421. Pole = TasksArea.by_id(int(pole_id))
  422. if not Pole:
  423. raise HTTPNotFound()
  424. form = EditStaffArea(request.POST, Pole, meta={'csrf_context': request.session})
  425. else:
  426. Pole = TasksArea()
  427. form = StaffArea(request.POST, Pole, meta={'csrf_context': request.session})
  428. if request.method == 'POST' and form.validate():
  429. form.populate_obj(Pole)
  430. if 'uid' in form._fields.keys():
  431. DBSession.merge(Pole)
  432. else:
  433. DBSession.add(Pole)
  434. return HTTPFound(location=request.route_url('list_task')+"#"+slugify(Pole.name))
  435. return {'form':form }
  436. @view_config(route_name='action_task')
  437. def action_task(request):
  438. if request.user is None:
  439. # Don't answer to users that aren't logged
  440. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  441. if not request.user.Staff:
  442. # Don't answer to users that aren't logged
  443. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  444. action = request.matchdict.get('action')
  445. task_id = request.matchdict.get('task_id')
  446. Task = Tasks.by_id(int(task_id))
  447. if action=='close':
  448. Task.closed = True
  449. request.session.flash(('info', u'La tâche a été fermé, Félicitations !'))
  450. DBSession.merge(Task)
  451. if action=='open':
  452. Task.closed = False
  453. request.session.flash(('info', u'La tâche a été ré-ouverte !'))
  454. DBSession.merge(Task)
  455. if action=='delete':
  456. request.session.flash(('info', u'La tâche a été supprimée !'))
  457. DBSession.delete(Task)
  458. return HTTPFound(location=request.route_url('list_task')+"#"+slugify(Task.area.name))
  459. @view_config(route_name='action_task_area')
  460. def action_task_area(request):
  461. if request.user is None:
  462. # Don't answer to users that aren't logged
  463. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  464. if not request.user.Staff:
  465. # Don't answer to users that aren't logged
  466. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  467. action = request.matchdict.get('action')
  468. pole_id = request.matchdict.get('pole_id')
  469. Pole = TasksArea.by_id(int(pole_id))
  470. if not Pole:
  471. raise HTTPNotFound()
  472. if action=='delete':
  473. request.session.flash(('info', u'Le pôle a été supprimé !'))
  474. DBSession.delete(Pole)
  475. return HTTPFound(location=request.route_url('list_task'))
  476. @view_config(route_name='list_salles', renderer='jm2l:templates/Salles/list.mako')
  477. def list_salles(request):
  478. if request.user is None:
  479. # Don't answer to users that aren't logged
  480. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  481. if not request.user.Staff:
  482. # Don't answer to users that aren't logged
  483. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  484. DicSalle = {}
  485. years = DBSession.query( JM2L_Year ).all()
  486. for year in years:
  487. salles = DBSession.query( Salles )\
  488. .filter( Salles.year_uid==year.year_uid )\
  489. .order_by(Salles.name).all()
  490. DicSalle[year] = salles
  491. return {'DicSalle': DicSalle }
  492. @view_config(route_name='handle_salle', renderer='jm2l:templates/Salles/salle.mako')
  493. def handle_salle(request):
  494. if request.user is None:
  495. # Don't answer to users that aren't logged
  496. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  497. if not request.user.Staff:
  498. # Don't answer to users that aren't logged
  499. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  500. salle_id = request.matchdict.get('salle_id')
  501. if salle_id:
  502. Salle = Salles.by_id(int(salle_id))
  503. if not Salle:
  504. raise HTTPNotFound()
  505. form = EditSalleForm(request.POST, Salle, meta={'csrf_context': request.session})
  506. else:
  507. Salle = Salles()
  508. form = SalleForm(request.POST, Salle, meta={'csrf_context': request.session})
  509. form.year_uid.choices = map(tuple, DBSession.query(JM2L_Year.year_uid, JM2L_Year.year_uid).all())
  510. form.phy_salle_id.choices = map(tuple, DBSession.query(SallePhy.uid, SallePhy.name).all())
  511. if request.method == 'POST' and form.validate():
  512. form.populate_obj(Salle)
  513. if 'uid' in form._fields.keys():
  514. DBSession.merge(Salle)
  515. else:
  516. DBSession.add(Salle)
  517. return HTTPFound(location=request.route_url('list_salles'))
  518. return {'form':form }
  519. @view_config(route_name='handle_salle_phy', renderer='jm2l:templates/Salles/salle_phy.mako')
  520. def handle_salle_phy(request):
  521. if request.user is None:
  522. # Don't answer to users that aren't logged
  523. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  524. if not request.user.Staff:
  525. # Don't answer to users that aren't logged
  526. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  527. salle_id = request.matchdict.get('salle_id')
  528. if salle_id:
  529. Salle = SallePhy.by_id(int(salle_id))
  530. if not Salle:
  531. raise HTTPNotFound()
  532. form = EditSallePhyForm(request.POST, Salle, meta={'csrf_context': request.session})
  533. else:
  534. Salle = SallePhy()
  535. form = SallePhyForm(request.POST, Salle, meta={'csrf_context': request.session})
  536. if request.method == 'POST' and form.validate():
  537. form.populate_obj(Salle)
  538. Salle.slug = slugify(Salle.name)
  539. if 'uid' in form._fields.keys():
  540. DBSession.merge(Salle)
  541. else:
  542. DBSession.add(Salle)
  543. return HTTPFound(location=request.route_url('list_salles'))
  544. return {'form':form }
  545. @view_config(route_name='action_salle')
  546. def action_salle(request):
  547. if request.user is None:
  548. # Don't answer to users that aren't logged
  549. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  550. if not request.user.Staff:
  551. # Don't answer to users that aren't logged
  552. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  553. action = request.matchdict.get('action')
  554. salle_id = request.matchdict.get('salle_id')
  555. Salle = Salles.by_id(int(salle_id))
  556. if not Salle:
  557. raise HTTPNotFound()
  558. if action=='delete':
  559. request.session.flash(('info', u'La Salle a été supprimée !'))
  560. DBSession.delete(Salle)
  561. return HTTPFound(location=request.route_url('list_salles'))
  562. ## =-=- Here, We handle HTTP requests - User Logged Part -=-=
  563. @view_config(route_name='exchange', renderer="jm2l:templates/Logistique/Logistique.mako")
  564. def exchange(request):
  565. modtype = request.matchdict.get('modtype', None)
  566. action = request.matchdict.get('action', None)
  567. uid = int(request.matchdict.get('id', -1))
  568. Exch = Exchange.by_id(uid)
  569. if not Exch:
  570. MainTab = {
  571. 'Exchanges':Exchange,
  572. 'Type':modtype[-1:],
  573. 'reload':True,
  574. 'logged_in':request.authenticated_userid
  575. }
  576. return MainTab
  577. if action in ['delete', 'accept', 'refuse', 'deal']:
  578. if action=='delete': # delete exchange
  579. DBSession.delete(Exch)
  580. elif action=='accept': # accept exchange
  581. Exch.exch_done=True
  582. DBSession.merge(Exch)
  583. elif action=='refuse': # refuse exchange
  584. Exch.exch_done=False
  585. if Exch.exch_state=="Ask":
  586. Exch.provider_id = None
  587. elif Exch.exch_state=="Proposal":
  588. Exch.asker_id = None
  589. DBSession.merge(Exch)
  590. elif action=='deal':
  591. # ask to deal the exchange
  592. if Exch.exch_state=="Ask":
  593. Exch.provider_id = request.user.uid
  594. elif Exch.exch_state=="Proposal":
  595. Exch.asker_id = request.user.uid
  596. # Return javascript to parent page
  597. response = render_to_response('jm2l:templates/modals_js.mako',
  598. {'modtype':modtype, 'action':action},
  599. request=request)
  600. response.content_type = 'text/javascript'
  601. return response
  602. else:
  603. MainTab = {
  604. 'Exchanges':Exchange,
  605. 'Type':modtype[-1:],
  606. 'reload':True,
  607. 'logged_in':request.authenticated_userid
  608. }
  609. return MainTab
  610. @view_config(route_name='miam')
  611. def miam(request):
  612. if request.user is None:
  613. # Don't answer to users that aren't logged
  614. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  615. miam_form = MiamForm(request.POST, request.user, meta={'csrf_context': request.session})
  616. if request.method == 'POST' and miam_form.validate():
  617. FicheSejour = Sejour.by_user(request.user.uid)
  618. if FicheSejour:
  619. Update=True
  620. else:
  621. FicheSejour = Sejour()
  622. FicheSejour.created = datetime.datetime.now()
  623. Update=False
  624. Repas=0
  625. for num, item in enumerate(['RepasVendredi', 'RepasSamediMidi', 'RepasSamediSoir']):
  626. if request.params.get(item)==u"1":
  627. Repas += 2**num
  628. FicheSejour.repas = Repas
  629. FicheSejour.repas_allerg = request.params.get('Allergies')
  630. FicheSejour.repas_contr = request.params.get('Contraintes')
  631. FicheSejour.user_id = request.user.uid
  632. FicheSejour.for_year = CurrentYear
  633. if Update:
  634. DBSession.merge(FicheSejour)
  635. else:
  636. DBSession.add(FicheSejour)
  637. request.session.flash(('info',u'Votre fiche a été mise à jour avec succès'))
  638. else:
  639. request.session.flash(('error',u'Un problème est survenu'))
  640. return HTTPFound(location='/MesJM2L#Miam')
  641. @view_config(route_name='sejour')
  642. def sejour(request):
  643. if request.user is None:
  644. # Don't answer to users that aren't logged
  645. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  646. if request.method == 'POST':
  647. FicheSejour = Sejour.by_user(request.user.uid)
  648. if FicheSejour:
  649. Update=True
  650. else:
  651. FicheSejour = Sejour()
  652. FicheSejour.created = datetime.datetime.now()
  653. Update=False
  654. FicheSejour.user_id = request.user.uid
  655. FicheSejour.last_change = datetime.datetime.now()
  656. FicheSejour.for_year = CurrentYear
  657. # Arrival
  658. ArrDate = datetime.datetime.strptime(request.params.get('Arrival:Day'),"%d/%m/%y")
  659. ArrTime = datetime.datetime.strptime(request.params.get('Arrival:Hour'),"%H:%M")
  660. FicheSejour.arrival_time = datetime.datetime.combine(ArrDate.date(), ArrTime.time())
  661. ArrivalCheck=0
  662. for num, item in enumerate(['Arrival:PMR', 'Arrival:Cov', 'Arrival:Bras', 'Arrival:Other']):
  663. if request.params.get(item):
  664. ArrivalCheck += 2**num
  665. FicheSejour.arrival_check = ArrivalCheck
  666. FicheSejour.arrival_text = request.params.get('Arrival:Comment')
  667. FicheSejour.arrival_place = request.params.get('Arrival:Place')
  668. # Departure
  669. DepDate = datetime.datetime.strptime(request.params.get('Departure:Day'),"%d/%m/%y")
  670. DepTime = datetime.datetime.strptime(request.params.get('Departure:Hour'),"%H:%M")
  671. FicheSejour.depart_time = datetime.datetime.combine(DepDate.date(), DepTime.time())
  672. DepartCheck=0
  673. for num, item in enumerate(['Departure:PMR', 'Departure:Cov', 'Departure:Bras', 'Departure:Other']):
  674. if request.params.get(item):
  675. DepartCheck += 2**num
  676. FicheSejour.depart_check = DepartCheck
  677. FicheSejour.depart_text = request.params.get('Departure:Comment')
  678. FicheSejour.depart_place = request.params.get('Departure:Place')
  679. if Update:
  680. DBSession.merge(FicheSejour)
  681. request.session.flash(('info',u'Vos modifications de séjour ont été pris en compte.'))
  682. else:
  683. DBSession.add(FicheSejour)
  684. request.session.flash(('info',u'\\o/ Votre séjour est enregistré ! Complétez la partie Logistique.'))
  685. return HTTPFound(location='/MesJM2L#Sejour')
  686. @view_config(route_name='orga')
  687. def orga(request):
  688. if request.user is None:
  689. # Don't answer to users that aren't logged
  690. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  691. if request.method == 'POST':
  692. FicheSejour = Sejour.by_user(request.user.uid)
  693. UpdateOrga=False
  694. if FicheSejour:
  695. Update=True
  696. if FicheSejour.orga_part:
  697. UpdateOrga=True
  698. else:
  699. FicheSejour = Sejour()
  700. FicheSejour.created = datetime.datetime.now()
  701. Update=False
  702. FicheSejour.user_id = request.user.uid
  703. FicheSejour.last_change = datetime.datetime.now()
  704. FicheSejour.for_year = CurrentYear
  705. OrgaPart=0
  706. for item in request.params:
  707. try:
  708. nb = int(item[1:])
  709. except:
  710. continue
  711. OrgaPart += 2**nb
  712. FicheSejour.orga_part = OrgaPart
  713. if UpdateOrga:
  714. request.session.flash(('info',u'Vos modifications de participation à l\'organisation ont été pris en compte.'))
  715. else:
  716. request.session.flash(('info',u'\\o/ Votre participation à l\'organisation est enregistrée !'))
  717. if Update:
  718. DBSession.merge(FicheSejour)
  719. else:
  720. DBSession.add(FicheSejour)
  721. return HTTPFound(location='/MesJM2L#Organisation')
  722. @view_config(route_name='vote_logo')
  723. def vote_logo(request):
  724. if request.user is None:
  725. # Don't answer to users that aren't logged
  726. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  727. else:
  728. vote = int(request.matchdict.get('num', -1))
  729. come = request.params.get('come_from')
  730. if vote:
  731. request.user.vote_logo=vote
  732. DBSession.merge(request.user)
  733. request.session.flash(('info',u'Votre vote à été pris en compte.'))
  734. return HTTPFound('/')
  735. else:
  736. request.session.flash(('warning',u"Votre vote n'a été pris en compte."))
  737. if come:
  738. return HTTPFound(location=come)
  739. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  740. @view_config(route_name='list_users', renderer="jm2l:templates/Participant/list_users.mako")
  741. def list_users(request):
  742. if request.user is None:
  743. # Don't answer to users that aren't logged
  744. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  745. if not request.user.Staff:
  746. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  747. Data = DBSession.query(User, Sejour).outerjoin(Sejour).all()
  748. Repas = DBSession.query(Sejour.repas).all()
  749. DicRepas = {"Ven":0, "Midi":0, "Soir":0}
  750. for r in Repas:
  751. if r[0] is None:
  752. continue
  753. if (r[0] & 1 == 1): DicRepas["Ven"]+=1
  754. if (r[0] & 2 == 2): DicRepas["Midi"]+=1
  755. if (r[0] & 4 == 4): DicRepas["Soir"]+=1
  756. return { 'Users':Data, 'UserEvent' : User_Event, "DicRepas":DicRepas }
  757. @view_config(route_name='list_orga', renderer="jm2l:templates/Participant/list_orga.mako")
  758. def list_orga(request):
  759. if request.user is None:
  760. # Don't answer to users that aren't logged
  761. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  762. if not request.user.Staff:
  763. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  764. Data = DBSession.query(User, Sejour).outerjoin(Sejour).all()
  765. return { 'Users':Data }
  766. @view_config(route_name='jm2l', renderer="jm2l:templates/jm2l.mako")
  767. def jm2l_page(request):
  768. if request.user is None:
  769. # Don't answer to users that aren't logged
  770. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  771. UserNum = request.params.get('user')
  772. if UserNum:
  773. profil = User.by_id(int(UserNum))
  774. if not profil:
  775. raise HTTPNotFound()
  776. if not request.user.Staff:
  777. raise HTTPForbidden(u'Vous n\'avez pas l\'autorité suffisante pour effectuer cette action.')
  778. else:
  779. profil = request.user
  780. FicheSejour = Sejour()
  781. FicheSejour.created = datetime.datetime.now()
  782. # Build Form
  783. profil_form = ProfilForm(request.POST, profil, meta={'csrf_context': request.session})
  784. miam_form = MiamForm(request.POST, profil, meta={'csrf_context': request.session})
  785. # Feed FicheSejour if any
  786. FicheSejour = Sejour.by_user(profil.uid)
  787. if FicheSejour:
  788. if FicheSejour.repas is not None:
  789. for num, item in enumerate(['RepasVendredi', 'RepasSamediMidi', 'RepasSamediSoir']):
  790. if FicheSejour.repas & 2**num:
  791. miam_form._fields[item].data = "1"
  792. else:
  793. miam_form._fields[item].data = "0"
  794. miam_form._fields['Allergies'].data = FicheSejour.repas_allerg
  795. miam_form._fields['Contraintes'].data = FicheSejour.repas_contr
  796. if request.method == 'POST' and profil_form.validate():
  797. ToDelete = list()
  798. # First, we remove entries no more present
  799. for obj in profil_form.tiersship.object_data:
  800. MatchEntry = filter( lambda x: x.object_data and x.object_data._sa_instance_state == obj._sa_instance_state,
  801. profil_form.tiersship.entries )
  802. if not MatchEntry:
  803. ToDelete.append(obj)
  804. # Then, it's time to consider new entries
  805. for entry in profil_form.tiersship.entries:
  806. if entry.object_data is None:
  807. TmpUser = User_Tiers()
  808. entry.object_data = TmpUser
  809. profil.tiersship.append(TmpUser)
  810. profil_form.tiersship.object_data = profil.tiersship
  811. profil_form.populate_obj(profil)
  812. # We should remove it as it's not in original data
  813. for obj in ToDelete:
  814. #profil.tiersship.remove(obj)
  815. DBSession.delete(obj)
  816. profil.last_change = datetime.datetime.utcnow()
  817. profil.slug = slugify(remove_accents('%s %s' % (profil.prenom, profil.nom)).lower().strip())
  818. DBSession.merge(profil)
  819. request.session.flash(('info',u'Votre fiche a été mise à jour avec succès'))
  820. MainTab = {'participer':'active',
  821. 'Places':Place.get_list(False),
  822. 'DBTiers':Tiers,
  823. 'DBTiersOpt':TiersOpt,
  824. 'Exchanges':Exchange,
  825. 'profil_form':profil_form,
  826. 'miam_form':miam_form,
  827. 'uprofil':profil,
  828. 'logged_in':request.authenticated_userid
  829. }
  830. return MainTab
  831. @view_config(route_name='modal', renderer="jm2l:templates/modals.mako")
  832. def Modal(request):
  833. year = int(request.matchdict.get('year', None))
  834. modtype = request.matchdict.get('modtype', None)
  835. uid = int(request.matchdict.get('id', -1))
  836. session = request.session
  837. if modtype=='Password':
  838. form = UserPasswordForm(request.POST, request.user, meta={'csrf_context': request.session})
  839. if request.method == 'POST' and form.validate():
  840. response = render_to_response('jm2l:templates/modals_js.mako',
  841. {'modtype':modtype},
  842. request=request)
  843. request.user.password = form.password.data
  844. DBSession.merge(request.user)
  845. response.content_type = 'text/javascript'
  846. return response
  847. if modtype=='UserPicture':
  848. form = None
  849. if request.method == 'POST':
  850. response = render_to_response('jm2l:templates/modals_js.mako',
  851. {'modtype':modtype},
  852. request=request)
  853. response.content_type = 'text/javascript'
  854. return response
  855. if modtype=='Place':
  856. if uid>0:
  857. place = Place.by_id(uid)
  858. if not place:
  859. raise HTTPNotFound()
  860. form = PlaceUpdateForm(request.POST, place, meta={'csrf_context': request.session})
  861. else:
  862. place = Place()
  863. form = PlaceCreateForm(request.POST, meta={'csrf_context': request.session})
  864. if request.method == 'POST' and form.validate():
  865. form.populate_obj(place)
  866. place.created_by=request.user.uid
  867. if uid>0:
  868. DBSession.merge(place)
  869. else:
  870. DBSession.add(place)
  871. response = render_to_response('jm2l:templates/modals_js.mako',
  872. {'modtype':modtype},
  873. request=request)
  874. response.content_type = 'text/javascript'
  875. return response
  876. if modtype in ['AskC', 'AskH', 'AskM', 'PropC', 'PropH', 'PropM']:
  877. if uid>0:
  878. Exch = Exchange.by_id(uid)
  879. if not Exch:
  880. raise HTTPNotFound()
  881. if modtype in ['AskC','PropC']:
  882. form = globals()["Update%sForm" % modtype](request.POST, Exch,
  883. start_place = Exch.Itin.start_place,
  884. arrival_place = Exch.Itin.arrival_place,
  885. Hour_start = Exch.start_time.strftime("%H:%M"),
  886. Day_start = Exch.start_time.strftime("%w"),
  887. exch_id = uid, meta={'csrf_context': request.session}
  888. )
  889. elif modtype in ['AskM','PropM']:
  890. form = globals()["Update%sForm" % modtype](request.POST, Exch,
  891. description = Exch.description,
  892. exch_categ = Exch.exch_categ,
  893. Hour_start = Exch.start_time.strftime("%H:%M"),
  894. Day_start = Exch.start_time.strftime("%w"),
  895. Hour_end = Exch.end_time.strftime("%H:%M"),
  896. Day_end = Exch.end_time.strftime("%w"),
  897. exch_id = uid, meta={'csrf_context': request.session}
  898. )
  899. elif modtype in ['AskH','PropH']:
  900. form = globals()["Update%sForm" % modtype](request.POST, Exch,
  901. description = Exch.description,
  902. exch_categ = Exch.exch_categ,
  903. Day_start = Exch.start_time.strftime("%w"),
  904. exch_id = uid, meta={'csrf_context': request.session}
  905. )
  906. # Itinerary, first get itinerary
  907. if 0:
  908. form.itin.form.start_place.data = Exch.Itin.start_place
  909. form.itin.form.arrival_place.data = Exch.Itin.arrival_place
  910. form.dateform.form.Hour.data = Exch.start_time.strftime("%H:%M")
  911. form.dateform.form.Day.data = Exch.start_time.strftime("%w")
  912. form.exch_id.data = uid
  913. else:
  914. Exch = Exchange()
  915. form = globals()["%sForm" % modtype](request.POST, meta={'csrf_context': request.session})
  916. if modtype in ['AskC', 'PropC']:
  917. # Put some place on form
  918. Places = DBSession.query(Place.place_id, Place.display_name)\
  919. .order_by('name').all()
  920. form.start_place.choices = Places
  921. form.arrival_place.choices = Places
  922. if modtype in ['PropH']:
  923. form.exch_categ.choices = DBSession.query( Exchange_Cat.cat_id, Exchange_Cat.exch_subtype)\
  924. .filter( Exchange_Cat.exch_type=='H' ).all()
  925. form.place_id.choices = DBSession.query( Place.place_id, Place.display_name)\
  926. .filter( Place.created_by==request.user.uid ).all()
  927. if modtype in ['AskM', 'PropM']:
  928. form.exch_categ.choices = DBSession.query( Exchange_Cat.cat_id, Exchange_Cat.exch_subtype)\
  929. .filter( Exchange_Cat.exch_type=='M' ).all()
  930. if request.method == 'POST' and form.validate():
  931. # Form has been validated, it's time to create our Exchange
  932. Exch.for_year = year
  933. Exch.exch_state = {'Ask':'Ask', 'Prop':'Proposal'}[modtype[:-1]]
  934. Exch.exch_type = modtype[-1:]
  935. if modtype in ['AskC', 'PropC']:
  936. # Itinerary, first Let's see if itinerary exist
  937. Itinerary = DBSession.query(Itineraire)\
  938. .filter(Itineraire.start_place==form.start_place.data) \
  939. .filter(Itineraire.arrival_place==form.arrival_place.data) \
  940. .filter(Itineraire.tr_voiture==True) \
  941. .first()
  942. if not Itinerary: # Not exist yet !
  943. Itinerary = Itineraire(start_place=form.start_place.data, \
  944. arrival_place=form.arrival_place.data, \
  945. tr_voiture=True, \
  946. created_by=1
  947. )
  948. DBSession.add(Itinerary)
  949. DBSession.flush()
  950. Exch.itin_id = Itinerary.itin_id
  951. # Start Time
  952. StartEvent = DBSession.query(JM2L_Year.start_time).filter(JM2L_Year.year_uid==year).first()
  953. Week = StartEvent[0].strftime("%W")
  954. # populate
  955. form.populate_obj(Exch)
  956. if modtype in ['AskC', 'PropC']:
  957. Exch.itin_id = Itinerary.itin_id
  958. if form._fields.has_key("Hour_start"):
  959. TargetTime = datetime.datetime.strptime('%d %d %d %s' % (year, int(Week), \
  960. int(form.Day_start.data), form.Hour_start.data), "%Y %W %w %H:%M")
  961. Exch.start_time = TargetTime
  962. elif form._fields.has_key("Day_start"):
  963. TargetTime = datetime.datetime.strptime('%d %d %d' % (year, int(Week), \
  964. int(form.Day_start.data)), "%Y %W %w")
  965. Exch.start_time = TargetTime
  966. if form._fields.has_key("Hour_end"):
  967. TargetTime = datetime.datetime.strptime('%d %d %d %s' % (year, int(Week), \
  968. int(form.Day_end.data), form.Hour_end.data), "%Y %W %w %H:%M")
  969. Exch.end_time = TargetTime
  970. elif form._fields.has_key("Day_end"):
  971. TargetTime = datetime.datetime.strptime('%d %d %d' % (year, int(Week), \
  972. int(form.Day_end.data)), "%Y %W %w")
  973. Exch.end_time = TargetTime
  974. Exch.last_change = datetime.datetime.utcnow()
  975. if Exch.exch_state=='Ask':
  976. Exch.asker_id = request.user.uid
  977. elif Exch.exch_state=='Proposal':
  978. Exch.provider_id = request.user.uid
  979. #print vars(form.itin.form)
  980. if uid>0:
  981. DBSession.merge(Exch)
  982. else:
  983. DBSession.add(Exch)
  984. response = render_to_response('jm2l:templates/modals_js.mako',
  985. {'modtype':modtype},
  986. request=request)
  987. response.content_type = 'text/javascript'
  988. return response
  989. # Fallback to HTML Display with errors
  990. return {'modtype':modtype, 'form':form, 'update':uid>0,
  991. 'logged_in':request.authenticated_userid }
  992. if modtype in ['ShowC', 'ShowH', 'ShowM']:
  993. if uid>0:
  994. Exch = Exchange.by_id(uid)
  995. if not Exch:
  996. raise HTTPNotFound()
  997. else:
  998. raise HTTPNotFound()
  999. # Show Details around the Current Exchange
  1000. return {'modtype':modtype, 'Exch':Exch, 'logged_in':request.authenticated_userid }
  1001. MainTab = {'modtype':modtype, 'form':form, 'update':uid>0, 'uid':uid,
  1002. 'DisplayYear':year, 'session':session,
  1003. 'logged_in':request.authenticated_userid }
  1004. return MainTab
  1005. @view_config(route_name='participer', renderer="jm2l:templates/Participer.mako")
  1006. def participer(request):
  1007. session = request.session
  1008. session['year'] = 2015
  1009. TmpUsr = User()
  1010. form = UserRegisterForm(request.POST, TmpUsr, meta={'csrf_context': request.session})
  1011. MyLink=None
  1012. if request.method == 'POST' and form.validate():
  1013. # Prepare mailer
  1014. form.populate_obj(TmpUsr)
  1015. TmpUsr.nom = TmpUsr.nom.capitalize()
  1016. TmpUsr.prenom = TmpUsr.prenom.capitalize()
  1017. TmpUsr.slug = slugify(remove_accents('%s %s' % (form.prenom.data, form.nom.data)).lower().strip())
  1018. TmpUsr.password = TmpUsr.my_hash
  1019. if len(TmpUsr.slug):
  1020. CheckExist = DBSession.query(User)\
  1021. .filter(User.slug==TmpUsr.slug)\
  1022. .first()
  1023. else:
  1024. CheckExist=None
  1025. if CheckExist:
  1026. MyLink = CheckExist.my_hash
  1027. NewUser = CheckExist
  1028. else:
  1029. DBSession.add(TmpUsr)
  1030. DBSession.flush()
  1031. MyLink = TmpUsr.my_hash
  1032. NewUser = TmpUsr
  1033. # Send the Welcome Mail
  1034. mailer = request.registry['mailer']
  1035. # Prepare Plain Text Message :
  1036. Mail_template = Template(filename='jm2l/templates/mail_plain.mako')
  1037. mail_plain = Mail_template.render(request=request, User=NewUser, action="Welcome")
  1038. # Prepare HTML Message :
  1039. Mail_template = Template(filename='jm2l/templates/mail_html.mako')
  1040. mail_html = Mail_template.render(request=request, User=NewUser, action="Welcome")
  1041. # Prepare Message
  1042. message = Message(subject="[JM2L] Mon inscription au site web JM2L",
  1043. sender="contact@jm2l.linux-azur.org",
  1044. recipients=[NewUser.mail or TmpUsr.mail],
  1045. body=mail_plain, html=mail_html)
  1046. message.add_bcc("spam@style-python.fr")
  1047. mailer.send(message)
  1048. request.session.flash(('info',u"Un mail vous a été envoyé afin de continuer votre quête !"))
  1049. MainTab = {'programme':'','presse':'', 'plan':'',
  1050. 'participer':'active', 'form':form, "link": MyLink,
  1051. 'logged_in':request.authenticated_userid }
  1052. return MainTab
  1053. @view_config(route_name='year')
  1054. def change_year(request):
  1055. year = int(request.matchdict.get('year', -1))
  1056. session = request.session
  1057. if year>-1:
  1058. session['year'] = year
  1059. return HTTPFound(location='/%s/' % year)
  1060. return HTTPFound(location=request.route_url('home', year=''))
  1061. @view_config(route_name='pict_user', renderer="jm2l:templates/Profil/pict_user.mako")
  1062. def pict_user(request):
  1063. return {"uprofil":request.user}
  1064. @view_config(route_name='pict_salle', renderer="jm2l:templates/Salles/pict_salle.mako")
  1065. def pict_salle(request):
  1066. salle_id = int(request.matchdict.get('salle_id', -1))
  1067. return {"Salles":Salles, "IdSalle":salle_id}
  1068. @view_config(route_name='event', renderer="jm2l:templates/view_event.mako")
  1069. def show_event(request):
  1070. year = int(request.matchdict.get('year', -1))
  1071. event_id = request.matchdict.get('event_id')
  1072. if event_id.isdigit():
  1073. TheEvent = Event.by_id(event_id)
  1074. if TheEvent is None:
  1075. raise HTTPNotFound()
  1076. else:
  1077. TheEvent = Event.by_slug(event_id, year)
  1078. if TheEvent is None:
  1079. raise HTTPNotFound()
  1080. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1081. 'event':TheEvent, 'logged_in':request.authenticated_userid, "Salles":Salles }
  1082. return MainTab
  1083. @view_config(route_name='link_event_user')
  1084. def link_event_user(request):
  1085. """ Get user and add it to current event """
  1086. if request.user is None:
  1087. # Don't answer to users that aren't logged
  1088. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1089. year = int(request.matchdict.get('year', -1))
  1090. form = AddIntervenant(request.POST, meta={'csrf_context': request.session})
  1091. intervention = request.matchdict.get('intervention', None)
  1092. TargetEvent = Event.by_id(form.event_uid.data)
  1093. Exist = User.by_id(form.intervenant.data)
  1094. if not Exist:
  1095. request.session.flash(('error',u"Une erreur s'est produite lors de l'ajout de votre intervenant !"))
  1096. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1097. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid)))
  1098. else:
  1099. TargetUser = Exist
  1100. uev = User_Event(year_uid=year, role=u"Animateur d'un évenement JM2L", user_uid=TargetUser.uid)
  1101. TargetEvent.interventions.append( uev )
  1102. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1103. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid)))
  1104. @view_config(route_name='link_event_tiers')
  1105. def link_event_tiers(request):
  1106. """ Create user if not exist, add it to current event """
  1107. if request.user is None:
  1108. # Don't answer to users that aren't logged
  1109. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1110. year = int(request.matchdict.get('year', -1))
  1111. form = AddTiers(request.POST, meta={'csrf_context': request.session})
  1112. intervention = request.matchdict.get('intervention', None)
  1113. TargetEvent = Event.by_id(form.event_uid.data)
  1114. Exist = Tiers.by_id(form.tiers.data)
  1115. if not Exist:
  1116. request.session.flash(('error',u"Une erreur s'est produite lors de l'ajout de votre entité !"))
  1117. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1118. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid)))
  1119. else:
  1120. TargetTiers = Exist
  1121. Matching = DBSession.query(Role_Tiers)\
  1122. .filter(Role_Tiers.year_uid==year)\
  1123. .filter(Role_Tiers.tiers_role=="Exposant")\
  1124. .filter(Role_Tiers.tiers_uid==TargetTiers.uid)\
  1125. .filter(Role_Tiers.event_uid==TargetEvent.uid)\
  1126. .all()
  1127. if len(Matching)==0:
  1128. tev = Role_Tiers(year_uid=year, tiers_role="Exposant", tiers_uid=TargetTiers.uid, event_uid=TargetEvent.uid)
  1129. DBSession.add(tev)
  1130. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1131. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid), _anchor="Tiers"))
  1132. @view_config(route_name='delete_link_u')
  1133. def delete_link_event_user(request):
  1134. """ Create user if not exist, add it to current event """
  1135. if request.user is None:
  1136. # Don't answer to users that aren't logged
  1137. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1138. year = int(request.matchdict.get('year', -1))
  1139. intervention = request.matchdict.get('intervention', None)
  1140. TargetEvent = Event.by_id( request.params.get('eid') )
  1141. Exist = User.by_id( request.params.get('uid') )
  1142. if not Exist:
  1143. request.session.flash(('error',u"Une erreur s'est produite lors de votre suppression !"))
  1144. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1145. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid)))
  1146. else:
  1147. TargetUser = Exist
  1148. Matching = DBSession.query(User_Event)\
  1149. .filter(User_Event.year_uid==year)\
  1150. .filter(User_Event.user_uid==TargetUser.uid)\
  1151. .filter(User_Event.event_uid==TargetEvent.uid)\
  1152. .all()
  1153. if len(Matching)==0:
  1154. request.session.flash(('error',u"Une erreur s'est produite lors de la suppression !"))
  1155. else:
  1156. for item in Matching:
  1157. DBSession.delete(item)
  1158. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1159. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid), _anchor="Tiers"))
  1160. @view_config(route_name='delete_link_t')
  1161. def delete_link_event_tiers(request):
  1162. """ Create user if not exist, add it to current event """
  1163. if request.user is None:
  1164. # Don't answer to users that aren't logged
  1165. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1166. year = int(request.matchdict.get('year', -1))
  1167. intervention = request.matchdict.get('intervention', None)
  1168. TargetEvent = Event.by_id( request.params.get('uid') )
  1169. Exist = Tiers.by_id( request.params.get('tid') )
  1170. if not Exist:
  1171. request.session.flash(('error',u"Une erreur s'est produite lors de l'ajout de votre entité !"))
  1172. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1173. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid)))
  1174. else:
  1175. TargetTiers = Exist
  1176. Matching = DBSession.query(Role_Tiers)\
  1177. .filter(Role_Tiers.year_uid==year)\
  1178. .filter(Role_Tiers.tiers_role=="Exposant")\
  1179. .filter(Role_Tiers.tiers_uid==TargetTiers.uid)\
  1180. .filter(Role_Tiers.event_uid==TargetEvent.uid)\
  1181. .all()
  1182. if len(Matching)==0:
  1183. request.session.flash(('error',u"Une erreur s'est produite lors de la suppression de votre entité !"))
  1184. else:
  1185. for item in Matching:
  1186. DBSession.delete(item)
  1187. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1188. year=str(year), intervention=intervention, event_id=str(TargetEvent.uid), _anchor="Tiers"))
  1189. @view_config(route_name='delete_event')
  1190. def delete_event(request):
  1191. if request.user is None:
  1192. # Don't answer to users that aren't logged
  1193. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1194. year = int(request.matchdict.get('year', -1))
  1195. event_id = request.matchdict.get('event_id')
  1196. intervention = request.matchdict.get('intervention', None)
  1197. # Check intervention
  1198. if not intervention in ['Stand', 'Table_ronde', 'Atelier', 'Conference', 'Concert']:
  1199. raise HTTPNotFound(u"Ce type d'évenement n'est pas reconnu")
  1200. # We should remove all links before to remove the event
  1201. if event_id.isdigit():
  1202. TheEvent = Event.by_id(event_id)
  1203. if TheEvent is None:
  1204. raise HTTPNotFound(u"Cette réference n'existe pas")
  1205. else:
  1206. TheEvent = Event.by_slug(event_id, year)
  1207. if TheEvent is None:
  1208. raise HTTPNotFound(u"Cette réference n'existe pas")
  1209. # Remove Roles
  1210. Roles_Link = DBSession.query(Role_Tiers).filter(Role_Tiers.event_uid==TheEvent.uid).all()
  1211. for tmp in Roles_Link:
  1212. DBSession.delete(tmp)
  1213. # Remove Intervenant
  1214. User_Link = DBSession.query(User_Event).filter(User_Event.event_uid==TheEvent.uid).all()
  1215. for tmp in User_Link:
  1216. DBSession.delete(tmp)
  1217. # Remove attachment if any
  1218. SRCPath = path.join('jm2l/upload', *(IMAGEPATH + ['event'] + [ str(year) ] + [ TheEvent.slug ]) )
  1219. shutil.rmtree(SRCPath, ignore_errors=True)
  1220. # Remove Event
  1221. DBSession.delete(TheEvent)
  1222. return HTTPFound(location=request.route_url('jm2l', _anchor="Interventions"))
  1223. @view_config(route_name='edit_event', renderer="jm2l:templates/edit_event.mako")
  1224. def edit_event(request):
  1225. if request.user is None:
  1226. # Don't answer to users that aren't logged
  1227. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1228. year = int(request.matchdict.get('year', -1))
  1229. event_id = request.matchdict.get('event_id')
  1230. intervention = request.matchdict.get('intervention', None)
  1231. IntervLabel = intervention.replace('_',' ').lower()
  1232. if intervention=='Conference':
  1233. IntervLabel = u'conférence'
  1234. # Check intervention
  1235. if not intervention in ['Stand', 'Table_ronde', 'Atelier', 'Conference', 'Concert']:
  1236. raise HTTPNotFound(u"Ce type d'évenement n'est pas reconnu")
  1237. TheYear = DBSession.query(JM2L_Year)\
  1238. .filter(JM2L_Year.year_uid==year)\
  1239. .first()
  1240. # Check year avaibility
  1241. if not TheYear:
  1242. raise HTTPNotFound(u"Cette année n'est pas pris en charge")
  1243. # Generate Timeslots for current year
  1244. TimeSlots = list(enumerate( [ x.strftime('%a %d %b %H:%M') for x in
  1245. TheYear.AvailableTimeSlots ] ))
  1246. if event_id:
  1247. # We try to update an existing record
  1248. if event_id.isdigit():
  1249. TheEvent = Event.by_id(event_id)
  1250. if TheEvent is None:
  1251. raise HTTPNotFound(u"Cette réference n'existe pas")
  1252. else:
  1253. TheEvent = Event.by_slug(event_id, year)
  1254. if TheEvent is None:
  1255. raise HTTPNotFound(u"Cette réference n'existe pas")
  1256. if request.user is None or not (request.user.Staff or request.user in TheEvent.intervenants):
  1257. raise HTTPForbidden(u"Vous n'êtes pas identifié comme étant un participant à cette intervention.")
  1258. # Compute some field value from selected event
  1259. if TheEvent.start_time in TheYear.AvailableTimeSlots:
  1260. start_sel = TheYear.AvailableTimeSlots.index(TheEvent.start_time)
  1261. else:
  1262. start_sel = len(TimeSlots)
  1263. TimeSlots.append( (len(TimeSlots), TheEvent.start_time.strftime('%a %d %b %H:%M')))
  1264. duration = (TheEvent.end_time - TheEvent.start_time).total_seconds()/60
  1265. end = TheEvent.start_time + datetime.timedelta(minutes=duration)
  1266. # prepare the form with update
  1267. form = ConfUpdateForm(request.POST, TheEvent, start_sel=start_sel, duration=duration, end_time=end,
  1268. meta={'csrf_context': request.session} )
  1269. # Customize labels
  1270. form.name.label.text += IntervLabel
  1271. form.description.label.text += IntervLabel
  1272. # Each event can get severals members
  1273. formAdd = AddIntervenant(event_uid=TheEvent.uid)
  1274. # Build list of intervenant
  1275. # Get users from db
  1276. Users = DBSession.query(User)\
  1277. .filter(User.Staff==1)\
  1278. .order_by('nom').all()
  1279. # Put some users on form
  1280. formAdd.intervenant.choices = [(u.uid, "%s %s" % (u.nom, u.prenom))
  1281. for u in Users]
  1282. # Each event can get severals entities
  1283. formAddT = AddTiers(event_uid=TheEvent.uid)
  1284. # Build list of entities
  1285. # Get entities from db
  1286. TmpTiers = DBSession.query(Tiers)\
  1287. .order_by('name').limit(10)
  1288. # Put some entities on form
  1289. formAddT.tiers.choices = [(u.uid, "%s %s" % (u.nom, u.prenom))
  1290. for u in Users]
  1291. else:
  1292. TheEvent = Event()
  1293. # prepare the form for creation
  1294. form = ConfCreateForm(request.POST,
  1295. event_type=intervention,
  1296. for_year=str(year), meta={'csrf_context': request.session}
  1297. )
  1298. # Customize labels
  1299. form.name.label.text += IntervLabel
  1300. form.description.label.text += IntervLabel
  1301. duration=60
  1302. # No intervenant
  1303. formAdd = None
  1304. formAddT = None
  1305. SalleDispo = DBSession.query(Salles)\
  1306. .filter(Salles.year_uid==year)\
  1307. .order_by('name')
  1308. if intervention=="Conference":
  1309. form.duration.choices =[
  1310. (15,u'Lighting talk ( 5 min)'),
  1311. (30,u'Conférence (20 min)'),
  1312. (60,u'Conférence (50 min)'),
  1313. (90,u'Conférence (75 min)'),
  1314. ]
  1315. if not duration in [15, 30, 60, 90]:
  1316. form.duration.choices.append( (duration,u'Conférence (%d min)' % duration) )
  1317. if not form._fields.has_key("uid"):
  1318. form.duration.data=60
  1319. SalleDispo = SalleDispo.filter(Salles.place_type.in_(['Conference', 'MAO']))
  1320. elif intervention=="Stand":
  1321. form.duration.choices =[
  1322. (8*60, u'Toute la journée'),
  1323. (4*60, u'une demi-journée')
  1324. ]
  1325. SalleDispo = SalleDispo.filter(Salles.place_type=='Stand')
  1326. elif intervention=="Atelier":
  1327. form.duration.choices = map( lambda d:(d, u'Atelier (%dh%.2d)' % (d/60, d%60) ), \
  1328. [60, 90, 120, 150, 180, 210, 240] )
  1329. if not duration in map(lambda (d,y): d, form.duration.choices):
  1330. form.duration.choices.append( (duration,u'Atelier (%dh%.2d)' % (duration/60, duration%60) ) )
  1331. SalleDispo = SalleDispo.filter(Salles.place_type.in_(['Ateliers', 'MAO']))
  1332. elif intervention=="Table_ronde":
  1333. form.duration.choices = map( lambda d:(d, u'Table ronde (%dh%.2d)' % (d/60, d%60) ), \
  1334. [60, 90, 120, 150] )
  1335. if not duration in map(lambda (d,y): d, form.duration.choices):
  1336. form.duration.choices.append( (duration,u'Table ronde (%dh%.2d)' % (duration/60, duration%60) ) )
  1337. SalleDispo = SalleDispo.filter(Salles.place_type=='Table ronde')
  1338. elif intervention=="Concert":
  1339. form.duration.choices = map( lambda d:(d, u'Concert (%dh%.2d)' % (d/60, d%60) ), \
  1340. [60, 90, 120, 150, 180, 210, 240] )
  1341. if not duration in map(lambda (d,y): d, form.duration.choices):
  1342. form.duration.choices.append( (duration,u'Concert (%dh%.2d)' % (duration/60, duration%60) ) )
  1343. SalleDispo = SalleDispo.filter(Salles.place_type.in_(['Stand', 'MAO']))
  1344. else:
  1345. raise HTTPForbidden(u"Pas encore disponible.")
  1346. form.salle_uid.choices = [(s.salle_id, s.name) for s in SalleDispo]
  1347. form.start_sel.choices = TimeSlots
  1348. if request.method == 'POST' and form.validate():
  1349. form.populate_obj(TheEvent)
  1350. TheEvent.start_time = TheYear.AvailableTimeSlots[form.start_sel.data]
  1351. TheEvent.end_time = TheEvent.start_time + datetime.timedelta(minutes=form.duration.data)
  1352. # Ok, time to put in database
  1353. if not form._fields.has_key("uid"):
  1354. TheEvent.slug = slugify(TheEvent.name)
  1355. DBSession.add(TheEvent)
  1356. # Append creator by default
  1357. if request.user.uid!=1:
  1358. uev = User_Event(year_uid=TheYear.year_uid, role="Animateur")
  1359. uev.user_uid = request.user.uid
  1360. TheEvent.interventions.append( uev )
  1361. DBSession.flush()
  1362. request.session.flash(('sucess',u'Votre intervention a été créee ! Vous pouvez la compléter à tout moment.'))
  1363. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1364. year=str(year), intervention=intervention, event_id=str(TheEvent.slug)))
  1365. else:
  1366. if slugify(TheEvent.name)!=TheEvent.slug:
  1367. # We should move some file as slug have been changed
  1368. # First we ensure there is no related event that already exist with that slug
  1369. CheckEvent = Event.by_slug( slugify(TheEvent.name), year)
  1370. if CheckEvent:
  1371. request.session.flash(('warning',u'Choisissez un autre titre pour votre évenement, il est en conflit avec un autre.'))
  1372. return {'event':TheEvent, 'form':form, 'formAdd':formAdd, 'formAddT':formAddT, 'Salles':Salles }
  1373. else:
  1374. SRCPath = path.join('jm2l/upload', *(IMAGEPATH + ['event'] + [ str(year) ] + [ TheEvent.slug ]) )
  1375. TheEvent.slug=slugify(TheEvent.name)
  1376. DSTPath = path.join('jm2l/upload', *(IMAGEPATH + ['event'] + [ str(year) ] + [ TheEvent.slug ]) )
  1377. if not path.isdir(path.dirname(DSTPath)):
  1378. makedirs(path.dirname(DSTPath))
  1379. # Then we should move event attachments to the new slug (if any)
  1380. if path.exists(SRCPath):
  1381. shutil.move(SRCPath, DSTPath)
  1382. DBSession.merge(TheEvent)
  1383. request.session.flash(('sucess',u'Votre intervention a été mis à jour !'))
  1384. return HTTPFound(location=request.route_url('edit_event', sep='/',
  1385. year=str(year), intervention=intervention, event_id=str(TheEvent.slug)))
  1386. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1387. 'event':TheEvent, 'form':form, 'formAdd':formAdd, 'formAddT':formAddT,
  1388. 'Salles':Salles }
  1389. return MainTab
  1390. @view_config(route_name='entities', renderer="jm2l:templates/list_tiers.mako")
  1391. def list_tiers(request):
  1392. Entities = dict()
  1393. EntityType = DBSession.query(TiersOpt.entity_type)\
  1394. .group_by(TiersOpt.entity_type).all()
  1395. for EType in EntityType:
  1396. Entities[EType.entity_type] = DBSession.query(Tiers).join(TiersOpt)\
  1397. .filter(TiersOpt.entity_type==EType.entity_type)\
  1398. .order_by(TiersOpt.entity_subtype, Tiers.name)
  1399. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1400. 'entities':Entities, 'logged_in':request.authenticated_userid }
  1401. return MainTab
  1402. @view_config(route_name='show_entity', renderer="jm2l:templates/view_tiers.mako")
  1403. def show_tiers(request):
  1404. tiers_type = request.matchdict.get('tiers_type')
  1405. entity_id = request.matchdict.get('entity_id')
  1406. if entity_id.isdigit():
  1407. TheTiers = Tiers.by_id(entity_id)
  1408. if TheTiers is None:
  1409. raise HTTPNotFound()
  1410. else:
  1411. TheTiers = Tiers.by_slug(entity_id)
  1412. if TheTiers is None:
  1413. raise HTTPNotFound()
  1414. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1415. 'entity':TheTiers, 'logged_in':request.authenticated_userid }
  1416. return MainTab
  1417. @view_config(route_name='delete_entity')
  1418. def delete_tiers(request):
  1419. if request.user is None:
  1420. # Don't answer to users that aren't logged
  1421. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1422. entity_id = request.matchdict.get('entity_id', None)
  1423. if entity_id:
  1424. if entity_id.isdigit():
  1425. TheTiers = Tiers.by_id(int(entity_id))
  1426. if TheTiers is None:
  1427. raise HTTPNotFound()
  1428. else:
  1429. TheTiers = Tiers.by_slug(entity_id)
  1430. if TheTiers is None:
  1431. raise HTTPNotFound()
  1432. if len(TheTiers.membership)!=0:
  1433. request.session.flash(('error', u"Vous devez supprimer tous les membres liés avant la suppression d'une entité."))
  1434. return HTTPFound(location=request.route_url('show_entity', entity_id=TheTiers.slug, tiers_type=TheTiers.get_entity_type.slug_entity_type))
  1435. if len(TheTiers.membership)!=0:
  1436. request.session.flash(('error', u"Vous devez supprimer tous les roles liés avant la suppression d'une entité."))
  1437. return HTTPFound(location=request.route_url('show_entity', entity_id=TheTiers.slug, tiers_type=TheTiers.get_entity_type.slug_entity_type))
  1438. DBSession.delete(TheTiers)
  1439. request.session.flash(('info', u"L'entité a bien été supprimée"))
  1440. return HTTPFound(location=request.route_url('entities'))
  1441. else:
  1442. raise HTTPNotFound()
  1443. @view_config(route_name='add_entity', renderer="jm2l:templates/edit_tiers.mako")
  1444. @view_config(route_name='edit_entity', renderer="jm2l:templates/edit_tiers.mako")
  1445. def edit_tiers(request):
  1446. entity_id = request.matchdict.get('entity_id', None)
  1447. TargetList = list()
  1448. if request.user is None:
  1449. # Don't answer to users that aren't logged
  1450. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1451. entity_types = DBSession.query(TiersOpt.entity_type).group_by(TiersOpt.entity_type).all()
  1452. for entity_type in entity_types:
  1453. entity_subtypes = DBSession.query(TiersOpt)\
  1454. .filter(TiersOpt.entity_type==entity_type.entity_type)\
  1455. .group_by(TiersOpt.entity_subtype).all()
  1456. ListType = [(i.uid, i.entity_subtype) for i in entity_subtypes]
  1457. TargetList.append( (entity_type.entity_type, ListType) )
  1458. if entity_id:
  1459. if entity_id.isdigit():
  1460. TheTiers = Tiers.by_id(entity_id)
  1461. if TheTiers is None:
  1462. raise HTTPNotFound()
  1463. else:
  1464. TheTiers = Tiers.by_slug(entity_id)
  1465. if TheTiers is None:
  1466. raise HTTPNotFound()
  1467. form = UpdateTiersForm(request.POST, TheTiers, meta={'csrf_context': request.session})
  1468. UserOptions = DBSession.query(TiersOpt)\
  1469. .filter(TiersOpt.entity_type==TheTiers.tiers_type)\
  1470. .all()
  1471. form.tiers_type.choices = TargetList
  1472. else:
  1473. TheTiers = Tiers()
  1474. # prepare the form for creation
  1475. form = TiersForm(request.POST, TheTiers, meta={'csrf_context': request.session})
  1476. form.tiers_type.choices = TargetList
  1477. UserOptions = list()
  1478. #test_form = TiersForm(request.POST, TheTiers, meta={'csrf_context': request.session})
  1479. if request.method == 'POST' and form.validate():
  1480. ToDelete = list()
  1481. ToDeleteR = list()
  1482. # First, we remove entries no more present
  1483. for obj in form.membership.object_data:
  1484. MatchEntry = filter( lambda x: x.object_data and x.object_data._sa_instance_state == obj._sa_instance_state,
  1485. form.membership.entries )
  1486. if not MatchEntry:
  1487. ToDelete.append(obj)
  1488. # For roles too
  1489. for obj in form.roles.object_data:
  1490. MatchEntry = filter( lambda x: x.object_data and x.object_data._sa_instance_state == obj._sa_instance_state,
  1491. form.roles.entries )
  1492. if not MatchEntry:
  1493. ToDeleteR.append(obj)
  1494. # We should remove it as it's not in original data
  1495. for obj in ToDelete:
  1496. TheTiers.membership.remove(obj)
  1497. DBSession.delete(obj)
  1498. # For roles too
  1499. for obj in ToDeleteR:
  1500. TheTiers.roles.remove(obj)
  1501. DBSession.delete(obj)
  1502. # Then, it's time to consider new entries
  1503. for entry in form.membership.entries:
  1504. if entry.object_data is None:
  1505. TmpUser = User_Tiers()
  1506. entry.object_data = TmpUser
  1507. TheTiers.membership.append(TmpUser)
  1508. form.membership.object_data = TheTiers.membership
  1509. # For roles too
  1510. for entry in form.roles.entries:
  1511. if entry.object_data is None:
  1512. TmpRole = Role_Tiers()
  1513. entry.object_data = TmpRole
  1514. TheTiers.roles.append(TmpRole)
  1515. form.roles.object_data = TheTiers.roles
  1516. form.populate_obj(TheTiers)
  1517. # Handle Remove of accents
  1518. OriginalSlug = TheTiers.slug
  1519. if not form._fields.has_key('uid'):
  1520. TheTiers.slug = slugify(form.name.data)
  1521. TheTiers.creator_id = request.user.uid
  1522. DBSession.add(TheTiers)
  1523. DBSession.flush()
  1524. return HTTPFound(location=request.route_url('edit_entity', sep='/',
  1525. entity_id=str(TheTiers.slug), tiers_type=TheTiers.get_entity_type.entity_type))
  1526. else:
  1527. if OriginalSlug!=slugify(form.name.data):
  1528. # We should move some file as slug have been changed
  1529. # First we ensure there is no related event that already exist with that slug
  1530. CheckTiers = Tiers.by_slug( slugify(form.name.data) )
  1531. if CheckTiers:
  1532. request.session.flash(('warning',u'Attention, Choisissez un autre titre pour votre entitée, elle est en conflit avec une autre.'))
  1533. DBSession.rollback()
  1534. return HTTPFound(location=request.route_url('edit_entity', sep='/',
  1535. entity_id=str(OriginalSlug), tiers_type=TheTiers.get_entity_type.entity_type))
  1536. else:
  1537. TheTiers.slug = slugify(form.name.data)
  1538. SRCPath = path.join('jm2l/upload', *(IMAGEPATH + ['tiers'] + [ OriginalSlug ]) )
  1539. DSTPath = path.join('jm2l/upload', *(IMAGEPATH + ['tiers'] + [ TheTiers.slug ]) )
  1540. if not path.isdir(path.dirname(DSTPath)):
  1541. makedirs(path.dirname(DSTPath))
  1542. # Then we should move event attachments to the new slug (if any)
  1543. if path.exists(SRCPath):
  1544. shutil.move(SRCPath, DSTPath)
  1545. DBSession.merge(TheTiers)
  1546. return HTTPFound(location=request.route_url('show_entity', entity_id=TheTiers.slug,
  1547. tiers_type=TheTiers.get_entity_type.slug_entity_type))
  1548. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1549. 'form':form, 'DBUser':User, 'UserOptions':UserOptions,
  1550. 'logged_in':request.authenticated_userid }
  1551. return MainTab
  1552. @view_config(route_name='edit_entity_cat', renderer="jm2l:templates/edit_tiers_categ.mako")
  1553. def edit_tiers_category(request):
  1554. if request.user is None:
  1555. # Don't answer to users that aren't logged
  1556. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1557. DicResult = dict()
  1558. ListChanges = list()
  1559. if request.method == 'POST':
  1560. # Reformat data
  1561. RegExist = re.compile('collection\[(?P<slug>[\w-]+)\]\[(?P<num>\d+)\]\[(?P<id>\d+)\]')
  1562. RegTitle = re.compile('collection\[(?P<slug>[\w-]+)\]\[title]')
  1563. RegNew = re.compile('collection\[(?P<slug>[\w-]+)\]\[(?P<num>\d+)\]\[id\]')
  1564. for key, value in request.POST.iteritems():
  1565. regN= RegNew.match(key)
  1566. regT= RegTitle.match(key)
  1567. reg = RegExist.match(key)
  1568. if reg:
  1569. if not DicResult.has_key(reg.group('slug')):
  1570. DicResult[reg.group('slug')] = dict()
  1571. if DicResult[reg.group('slug')].has_key('items'):
  1572. DicResult[reg.group('slug')]['items'].append( ( int(reg.group('id')), value ) )
  1573. else:
  1574. DicResult[reg.group('slug')]['items'] = [ ( int(reg.group('id')), value ) ]
  1575. elif regN:
  1576. if not DicResult.has_key(regN.group('slug')):
  1577. DicResult[regN.group('slug')] = dict()
  1578. if DicResult[regN.group('slug')].has_key('items'):
  1579. DicResult[regN.group('slug')]['items'].append( ( 'id', value ) )
  1580. else:
  1581. DicResult[regN.group('slug')]['items'] = [ ( 'id', value ) ]
  1582. ListChanges.append(('add', 0, DicResult[regN.group('slug')]['title'], value))
  1583. elif regT:
  1584. if not DicResult.has_key(regT.group('slug')):
  1585. DicResult[regT.group('slug')] = dict()
  1586. DicResult[regT.group('slug')]['title'] = value
  1587. else:
  1588. raise
  1589. for opt in DBSession.query(TiersOpt).all():
  1590. if DicResult.has_key(opt.slug_entity_type):
  1591. found = filter( lambda (x,y): opt.uid==x,
  1592. DicResult[opt.slug_entity_type].get('items', []))
  1593. if not found:
  1594. ListChanges.append(('remove', opt.uid, opt.entity_type, opt.entity_subtype))
  1595. else:
  1596. for tst in found:
  1597. # Check changes on Cat Name
  1598. if DicResult[opt.slug_entity_type]['title']!=opt.entity_type or \
  1599. tst[1]!=opt.entity_subtype:
  1600. ListChanges.append(('changed', opt.uid,
  1601. DicResult[opt.slug_entity_type]['title'],
  1602. tst[1]))
  1603. else:
  1604. ListChanges.append(('remove', opt.uid, opt.entity_type, opt.entity_subtype))
  1605. # Do The change
  1606. for action, uid, entity, subentity in ListChanges:
  1607. if action=="changed":
  1608. opt = TiersOpt.by_id(uid)
  1609. opt.entity_type = entity
  1610. opt.entity_subtype = subentity
  1611. elif action=="remove":
  1612. opt = TiersOpt.by_id(uid)
  1613. DBSession.delete(opt)
  1614. elif action=="add":
  1615. opt = TiersOpt()
  1616. opt.entity_type = entity
  1617. opt.entity_subtype = subentity
  1618. DBSession.add(opt)
  1619. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1620. 'logged_in':request.authenticated_userid, 'TiersOpt':TiersOpt }
  1621. return MainTab
  1622. @view_config(route_name='show_user', renderer="jm2l:templates/view_user.mako")
  1623. def show_user(request):
  1624. user_slug = request.matchdict.get('user_slug', None)
  1625. if user_slug is None or len(user_slug)==0:
  1626. raise HTTPNotFound(u"Cet utilisateur n'a pas été reconnu")
  1627. # Query database
  1628. DispUser = User.by_slug(user_slug)
  1629. if DispUser is None:
  1630. raise HTTPNotFound()
  1631. MainTab = {'programme':'','presse':'', 'plan':'', 'participer':'',
  1632. 'DispUser':DispUser, 'logged_in':request.authenticated_userid }
  1633. return MainTab
  1634. #@view_config(route_name='link_user_entity')
  1635. def link_user_entity(request):
  1636. if request.user is None:
  1637. # Don't answer to users that aren't logged
  1638. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1639. uid = int(request.matchdict.get('uid', -1))
  1640. year = int(request.matchdict.get('year', -1))
  1641. user_id = int(request.matchdict.get('uid', -1))
  1642. TheTiers = Tiers.by_id(uid)
  1643. if TheTiers is None:
  1644. raise HTTPNotFound()
  1645. return HTTPFound(location=request.route_url('edit_entity', uid=uid) )
  1646. #@view_config(route_name='link_role_entity')
  1647. def link_role_entity(request):
  1648. if request.user is None:
  1649. # Don't answer to users that aren't logged
  1650. raise HTTPForbidden(u'Vous devez vous identifier pour obtenir une réponse.')
  1651. uid = int(request.matchdict.get('uid', -1))
  1652. year = int(request.matchdict.get('year', -1))
  1653. role_id = int(request.matchdict.get('role_id', -1))
  1654. TheTiers = Tiers.by_id(uid)
  1655. if TheTiers is None:
  1656. raise HTTPNotFound()
  1657. return HTTPFound(location=request.route_url('edit_entity', uid=uid) )
  1658. @forbidden_view_config()
  1659. def forbidden(reason, request):
  1660. if 'ident' in reason.detail:
  1661. request.session.flash(('info', reason.detail))
  1662. return HTTPFound(location='/sign/login?from='+request.environ['PATH_INFO'] )
  1663. else:
  1664. request.response.status = 403
  1665. return render_to_response('jm2l:templates/Errors/403.mako', { "reason":reason },
  1666. request=request)
  1667. @notfound_view_config()
  1668. def notfound(reason, request):
  1669. request.response.status = 404
  1670. return render_to_response('jm2l:templates/Errors/404.mako', { "reason":reason },
  1671. request=request)