No Description
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.

admin.py 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. # -*- coding: utf-8 -*-
  2. from ...exceptions import *
  3. from .base import get_response
  4. from lodel.context import LodelContext
  5. LodelContext.expose_modules(globals(), {
  6. 'lodel.leapi.exceptions': [],
  7. 'lodel.logger': 'logger',
  8. 'lodel.leapi.datahandlers.base_classes': ['MultipleRef'],
  9. 'lodel.leapi.exceptions': ['LeApiDataCheckErrors'],
  10. 'lodel.exceptions': ['LodelExceptions']})
  11. LodelContext.expose_dyncode(globals(), 'dyncode')
  12. from ...client import WebUiClient
  13. import warnings
  14. LIST_SEPARATOR = ','
  15. ##@brief These functions are called by the rules defined in ../urls.py
  16. ## To administrate the instance of the editorial model
  17. ##@brief Controller's function to redirect on the home page of the admin
  18. # @param request : the request (get or post)
  19. # @note the response is given in a html page called in get_response_function
  20. def index_admin(request):
  21. # We have to be identified to admin the instance
  22. # temporary, the acl will be more restrictive
  23. #if WebUiClient.is_anonymous():
  24. # return get_response('users/signin.html')
  25. return get_response('admin/admin.html')
  26. ##@brief Controller's function to update an object of the editorial model
  27. # @param request : the request (get or post)
  28. # @note the response is given in a html page (in templates/admin) called in get_response_function
  29. def admin_update(request):
  30. # We have to be identified to admin the instance
  31. # temporary, the acl will be more restrictive
  32. #if WebUiClient.is_anonymous():
  33. # return get_response('users/signin.html')
  34. msg=''
  35. datas = process_form(request)
  36. if not(datas is False):
  37. if 'lodel_id' not in datas:
  38. raise HttpException(400)
  39. target_leo = dyncode.Object.name2class(datas['classname'])
  40. leo = target_leo.get_from_uid(datas['lodel_id'])
  41. if leo is None:
  42. raise HttpException(404,
  43. custom = 'No %s with id %s' % (
  44. target_leo.__name__, datas['lodel_id']))
  45. try:
  46. leo.update(
  47. { f:datas[f] for f in datas if f not in ('classname', 'lodel_id')})
  48. except LeApiDataCheckErrors as e:
  49. raise HttpErrors(
  50. title='Form validation errors', errors = e._exceptions)
  51. # Display of the form with the object's values to be updated
  52. if 'classname' in request.GET:
  53. # We need the class of the object to update
  54. classname = request.GET['classname']
  55. if len(classname) > 1:
  56. raise HttpException(400)
  57. classname = classname[0]
  58. try:
  59. target_leo = dyncode.Object.name2class(classname)
  60. except LeApiError:
  61. # classname = None
  62. raise HttpException(400)
  63. logger.warning('Composed uids broken here')
  64. uid_field = target_leo.uid_fieldname()[0]
  65. # We need the uid of the object
  66. test_valid = 'lodel_id' in request.GET \
  67. and len(request.GET['lodel_id']) == 1
  68. if test_valid:
  69. try:
  70. dh = target_leo.field(uid_field)
  71. # we cast the uid extrated form the request to the adequate type
  72. # given by the datahandler of the uidfield's datahandler
  73. lodel_id = dh.cast_type(request.GET['lodel_id'][0])
  74. except (ValueError, TypeError):
  75. test_valid = False
  76. if not test_valid:
  77. raise HttpException(400)
  78. else:
  79. # Check if the object actually exists
  80. # We get it from the database
  81. query_filters = list()
  82. query_filters.append((uid_field,'=',lodel_id))
  83. obj = target_leo.get(query_filters)
  84. if len(obj) == 0:
  85. raise HttpException(404)
  86. return get_response('admin/admin_edit.html', target=target_leo, lodel_id =lodel_id)
  87. ##@brief Controller's function to create an object of the editorial model
  88. # @param request : the request (get or post)
  89. # @note the response is given in a html page (in templates/admin) called in get_response_function
  90. def admin_create(request):
  91. # We have to be identified to admin the instance
  92. # temporary, the acl will be more restrictive
  93. #if WebUiClient.is_anonymous():
  94. # return get_response('users/signin.html')
  95. datas = process_form(request)
  96. if not(datas is False):
  97. target_leo = dyncode.Object.name2class(datas['classname'])
  98. if 'lodel_id' in datas:
  99. raise HttpException(400)
  100. try:
  101. new_uid = target_leo.insert(
  102. { f:datas[f] for f in datas if f != 'classname'})
  103. except LeApiDataCheckErrors as e:
  104. raise HttpErrors(
  105. title='Form validation errors', errors = e._exceptions)
  106. if new_uid is None:
  107. raise HttpException(400, "Creation fails")
  108. else:
  109. return get_response(
  110. 'admin/admin_create.html', target=target_leo,
  111. msg = "Created with uid %s" % new_uid)
  112. # Display of an empty form
  113. if 'classname' in request.GET:
  114. # We need the class to create an object in
  115. classname = request.GET['classname']
  116. if len(classname) > 1:
  117. raise HttpException(400)
  118. classname = classname[0]
  119. try:
  120. target_leo = dyncode.Object.name2class(classname)
  121. except LeApiError:
  122. classname = None
  123. if classname is None or target_leo.is_abstract():
  124. raise HttpException(400)
  125. return get_response('admin/admin_create.html', target=target_leo)
  126. ##@brief Controller's function to delete an object of the editorial model
  127. # @param request : the request (get)
  128. # @note the response is given in a html page (in templates/admin) called in get_response_function
  129. def admin_delete(request):
  130. # We have to be identified to admin the instance
  131. # temporary, the acl will be more restrictive
  132. #if WebUiClient.is_anonymous():
  133. # return get_response('users/signin.html')
  134. classname = None
  135. if 'classname' in request.GET:
  136. # We need the class to delete an object in
  137. classname = request.GET['classname']
  138. if len(classname) > 1:
  139. raise HttpException(400)
  140. classname = classname[0]
  141. try:
  142. target_leo = dyncode.Object.name2class(classname)
  143. except LeApiError:
  144. # classname = None
  145. raise HttpException(400)
  146. logger.warning('Composed uids broken here')
  147. uid_field = target_leo.uid_fieldname()[0]
  148. # We also need the uid of the object to delete
  149. test_valid = 'lodel_id' in request.GET \
  150. and len(request.GET['lodel_id']) == 1
  151. if test_valid:
  152. try:
  153. dh = target_leo.field(uid_field)
  154. # we cast the uid extrated form the request to the adequate type
  155. # given by the datahandler of the uidfield's datahandler
  156. lodel_id = dh.cast_type(request.GET['lodel_id'][0])
  157. except (ValueError, TypeError):
  158. test_valid = False
  159. if not test_valid:
  160. raise HttpException(400)
  161. else:
  162. query_filters = list()
  163. query_filters.append((uid_field,'=',lodel_id))
  164. nb_deleted = target_leo.delete_bundle(query_filters)
  165. if nb_deleted == 1:
  166. msg = 'Object successfully deleted';
  167. else:
  168. msg = 'Oops something wrong happened...object still here'
  169. return get_response('admin/admin_delete.html', target=target_leo, lodel_id =lodel_id, msg = msg)
  170. def admin_classes(request):
  171. # We have to be identified to admin the instance
  172. # temporary, the acl will be more restrictive
  173. #if WebUiClient.is_anonymous():
  174. # return get_response('users/signin.html')
  175. return get_response('admin/list_classes_admin.html', my_classes = dyncode.dynclasses)
  176. def create_object(request):
  177. # We have to be identified to admin the instance
  178. # temporary, the acl will be more restrictive
  179. #if WebUiClient.is_anonymous():
  180. # return get_response('users/signin.html')
  181. return get_response('admin/list_classes_create.html', my_classes = dyncode.dynclasses)
  182. def delete_object(request):
  183. # We have to be identified to admin the instance
  184. # temporary, the acl will be more restrictive
  185. #if WebUiClient.is_anonymous():
  186. # return get_response('users/signin.html')
  187. return get_response('admin/list_classes_delete.html', my_classes = dyncode.dynclasses)
  188. def admin_class(request):
  189. # We have to be identified to admin the instance
  190. # temporary, the acl will be more restrictive
  191. #if WebUiClient.is_anonymous():
  192. # return get_response('users/signin.html')
  193. # We need the class we'll list to select the object to edit
  194. if 'classname' in request.GET:
  195. classname = request.GET['classname']
  196. if len(classname) > 1:
  197. raise HttpException(400)
  198. classname = classname[0]
  199. try:
  200. target_leo = dyncode.Object.name2class(classname)
  201. except LeApiError:
  202. classname = None
  203. if classname is None or target_leo.is_abstract():
  204. raise HttpException(400)
  205. return get_response('admin/show_class_admin.html', target=target_leo)
  206. def delete_in_class(request):
  207. # We have to be identified to admin the instance
  208. # temporary, the acl will be more restrictive
  209. #if WebUiClient.is_anonymous():
  210. # return get_response('users/signin.html')
  211. # We need the class we'll list to select the object to delete
  212. if 'classname' in request.GET:
  213. classname = request.GET['classname']
  214. if len(classname) > 1:
  215. raise HttpException(400)
  216. classname = classname[0]
  217. try:
  218. target_leo = dyncode.Object.name2class(classname)
  219. except LeApiError:
  220. classname = None
  221. if classname is None or target_leo.is_abstract():
  222. raise HttpException(400)
  223. return get_response('admin/show_class_delete.html', target=target_leo)
  224. def admin(request):
  225. # We have to be identified to admin the instance
  226. # temporary, the acl will be more restrictive
  227. #if WebUiClient.is_anonymous():
  228. # return get_response('users/signin.html')
  229. return get_response('admin/admin.html')
  230. def search_object(request):
  231. if request.method == 'POST':
  232. classname = request.POST['classname']
  233. searchstring = request.POST['searchstring']
  234. try:
  235. target_leo = dyncode.Object.name2class(classname)
  236. except LeApiError:
  237. raise HttpException(400)
  238. # TODO The get method must be implemented here
  239. return get_response('admin/admin_search.html', my_classes = dyncode.dynclasses)
  240. ##@brief Process a form POST and return the posted datas
  241. #@param request : the request object
  242. #@return a dict with datas as value and fieldname as key
  243. def process_form(request):
  244. if request.method != 'POST':
  245. return False
  246. res = dict()
  247. errors = dict()
  248. #Fetch concerned LeObject
  249. if 'classname' not in request.form:
  250. logger.error("Received a form without classname !")
  251. raise HttpException(400)
  252. res['classname'] = classname = request.form['classname']
  253. try:
  254. target_leo = dyncode.Object.name2class(classname)
  255. except LeApiError:
  256. logger.error(
  257. "Received a form with an invalid leo name : '%s'" % classname)
  258. raise HttpException(400, "No leobject named '%s'" % classname)
  259. if target_leo.is_abstract():
  260. logger.error(
  261. "Received a form with an abstract leo : '%s'" % classname)
  262. raise HttpException(400, '%s is abstract' % classname)
  263. #Process input fields
  264. for fieldname, value in request.form.items():
  265. if fieldname == 'classname':
  266. continue
  267. elif fieldname == 'uid':
  268. fieldname = 'lodel_id' #wow
  269. elif fieldname.startswith('field_input_'):
  270. fieldname = fieldname[12:]
  271. try:
  272. dh = target_leo.data_handler(fieldname)
  273. except NameError as e:
  274. errors[fieldname] = e
  275. continue
  276. if dh.is_reference() and not dh.is_singlereference():
  277. #Converting multiple references fields
  278. value = value.strip()
  279. if len(value) == 0:
  280. #handling default value for empty string
  281. if hasattr(dh, 'default'):
  282. value = dh.default
  283. else:
  284. #if not explicit default value, enforcing default as
  285. #an empty list
  286. value = []
  287. else:
  288. value = [ v.strip() for v in value.split(LIST_SEPARATOR) ]
  289. value = [ v for v in value if len(v) > 0]
  290. else:
  291. #Handling default value for empty string
  292. if len(value.strip()) == 0 and hasattr(dh, 'default'):
  293. value = dh.default
  294. res[fieldname] = value
  295. if len(errors) > 0:
  296. del(res)
  297. raise HttpErrors(errors, title="Form validation error")
  298. return res