Browse Source

Merge branch 'master' of git.labocleo.org:lodel2

prieto 8 years ago
parent
commit
dab11c3bdb
100 changed files with 2064 additions and 358 deletions
  1. 2
    0
      .gitignore
  2. 11
    3
      Makefile.am
  3. 24
    1
      README
  4. 2
    1
      configure.ac
  5. 1
    1
      debian/control
  6. 697
    0
      editorial_models/em_simple.py
  7. 7
    0
      em_test.py
  8. BIN
      examples/em_test.pickle
  9. 2
    2
      lodel/Makefile.am
  10. 9
    7
      lodel/auth/client.py
  11. 5
    2
      lodel/auth/exceptions.py
  12. 471
    0
      lodel/context.py
  13. 6
    5
      lodel/editorial_model/components.py
  14. 8
    6
      lodel/editorial_model/model.py
  15. 7
    3
      lodel/editorial_model/translator/xmlfile.py
  16. 1
    0
      lodel/exceptions.py
  17. 49
    69
      lodel/leapi/datahandlers/base_classes.py
  18. 9
    2
      lodel/leapi/datahandlers/datas.py
  19. 5
    2
      lodel/leapi/datahandlers/datas_base.py
  20. 3
    0
      lodel/leapi/datahandlers/exceptions.py
  21. 8
    3
      lodel/leapi/datahandlers/references.py
  22. 3
    1
      lodel/leapi/exceptions.py
  23. 13
    7
      lodel/leapi/lefactory.py
  24. 1
    2
      lodel/leapi/lefactory_common.py
  25. 32
    11
      lodel/leapi/leobject.py
  26. 9
    5
      lodel/leapi/query.py
  27. 12
    5
      lodel/logger.py
  28. 8
    6
      lodel/plugin/__init__.py
  29. 9
    5
      lodel/plugin/core_hooks.py
  30. 3
    1
      lodel/plugin/core_scripts.py
  31. 13
    8
      lodel/plugin/datasource_plugin.py
  32. 1
    1
      lodel/plugin/exceptions.py
  33. 8
    3
      lodel/plugin/extensions.py
  34. 4
    4
      lodel/plugin/hooks.py
  35. 7
    3
      lodel/plugin/interface.py
  36. 46
    24
      lodel/plugin/plugins.py
  37. 6
    2
      lodel/plugin/scripts.py
  38. 7
    3
      lodel/plugin/sessionhandler.py
  39. 0
    0
      lodel/plugins/Makefile.am
  40. 0
    0
      lodel/plugins/__init__.py
  41. 4
    2
      lodel/plugins/dummy/__init__.py
  42. 3
    1
      lodel/plugins/dummy/confspec.py
  43. 5
    8
      lodel/plugins/dummy/main.py
  44. 3
    1
      lodel/plugins/dummy_datasource/__init__.py
  45. 3
    1
      lodel/plugins/dummy_datasource/datasource.py
  46. 3
    1
      lodel/plugins/dummy_datasource/main.py
  47. 0
    0
      lodel/plugins/dummy_datasource/migration_handler.py
  48. 3
    1
      lodel/plugins/filesystem_session/__init__.py
  49. 3
    1
      lodel/plugins/filesystem_session/confspec.py
  50. 0
    0
      lodel/plugins/filesystem_session/filesystem_session.py
  51. 5
    3
      lodel/plugins/filesystem_session/main.py
  52. 1
    1
      lodel/plugins/mongodb_datasource/__init__.py
  53. 3
    1
      lodel/plugins/mongodb_datasource/confspec.py
  54. 21
    15
      lodel/plugins/mongodb_datasource/datasource.py
  55. 5
    2
      lodel/plugins/mongodb_datasource/exceptions.py
  56. 3
    1
      lodel/plugins/mongodb_datasource/main.py
  57. 10
    6
      lodel/plugins/mongodb_datasource/migration_handler.py
  58. 4
    2
      lodel/plugins/mongodb_datasource/utils.py
  59. 23
    0
      lodel/plugins/multisite/__init__.py
  60. 34
    0
      lodel/plugins/multisite/confspecs.py
  61. 21
    0
      lodel/plugins/multisite/example_lodel2.ini
  62. 69
    0
      lodel/plugins/multisite/loader.py
  63. 133
    0
      lodel/plugins/multisite/run.py
  64. 4
    2
      lodel/plugins/ram_sessions/__init__.py
  65. 6
    3
      lodel/plugins/ram_sessions/main.py
  66. 0
    0
      lodel/plugins/webui/__init__.py
  67. 2
    1
      lodel/plugins/webui/client.py
  68. 3
    1
      lodel/plugins/webui/confspec.py
  69. 42
    0
      lodel/plugins/webui/exceptions.py
  70. 0
    0
      lodel/plugins/webui/interface/__init__.py
  71. 0
    0
      lodel/plugins/webui/interface/controllers/__init__.py
  72. 101
    88
      lodel/plugins/webui/interface/controllers/admin.py
  73. 0
    0
      lodel/plugins/webui/interface/controllers/base.py
  74. 0
    0
      lodel/plugins/webui/interface/controllers/document.py
  75. 4
    2
      lodel/plugins/webui/interface/controllers/listing.py
  76. 4
    3
      lodel/plugins/webui/interface/controllers/users.py
  77. 0
    0
      lodel/plugins/webui/interface/lodelrequest.py
  78. 4
    1
      lodel/plugins/webui/interface/router.py
  79. 0
    0
      lodel/plugins/webui/interface/template/__init__.py
  80. 0
    0
      lodel/plugins/webui/interface/template/api/__init__.py
  81. 0
    0
      lodel/plugins/webui/interface/template/api/api_lodel_templates.py
  82. 0
    0
      lodel/plugins/webui/interface/template/exceptions/__init__.py
  83. 0
    0
      lodel/plugins/webui/interface/template/exceptions/not_allowed_custom_api_key_error.py
  84. 4
    2
      lodel/plugins/webui/interface/template/loader.py
  85. 0
    0
      lodel/plugins/webui/interface/urls.py
  86. 7
    3
      lodel/plugins/webui/main.py
  87. 15
    7
      lodel/plugins/webui/run.py
  88. 0
    0
      lodel/plugins/webui/templates/admin/admin.html
  89. 0
    0
      lodel/plugins/webui/templates/admin/admin_create.html
  90. 0
    0
      lodel/plugins/webui/templates/admin/admin_delete.html
  91. 0
    0
      lodel/plugins/webui/templates/admin/admin_edit.html
  92. 0
    0
      lodel/plugins/webui/templates/admin/admin_search.html
  93. 0
    0
      lodel/plugins/webui/templates/admin/editable_component.html
  94. 0
    0
      lodel/plugins/webui/templates/admin/list_classes_admin.html
  95. 0
    0
      lodel/plugins/webui/templates/admin/list_classes_create.html
  96. 0
    0
      lodel/plugins/webui/templates/admin/list_classes_delete.html
  97. 0
    0
      lodel/plugins/webui/templates/admin/show_class_admin.html
  98. 0
    0
      lodel/plugins/webui/templates/admin/show_class_delete.html
  99. 0
    0
      lodel/plugins/webui/templates/base.html
  100. 0
    0
      lodel/plugins/webui/templates/base_backend.html

+ 2
- 0
.gitignore View File

@@ -6,3 +6,5 @@ settings_local.py
6 6
 doc/html
7 7
 doc/*.sqlite3.db
8 8
 .*.swp
9
+Makefile
10
+Makefile.in

+ 11
- 3
Makefile.am View File

@@ -1,5 +1,5 @@
1
-SUBDIRS = lodel progs plugins
2
-EXTRA_DIST = plugins runtest examples tests runtest.sh debian
1
+SUBDIRS = lodel progs lodelsites
2
+EXTRA_DIST = runtest examples tests debian
3 3
 CLEANFILES = runtest
4 4
 
5 5
 lodel2_localstate_DATA =
@@ -27,7 +27,7 @@ doc_graphviz:
27 27
 
28 28
 do_subst = sed -e 's,\[@\]INSTALLMODEL_DIR\[@\],$(install_model_dir),g' 
29 29
 
30
-runtest: runtest.sh
30
+runtest: ./runtest.sh
31 31
 	$(do_subst) < $(srcdir)/runtest.sh > runtest
32 32
 	chmod +x runtest
33 33
 
@@ -43,10 +43,18 @@ deb: dist
43 43
 em_test: em_test.py
44 44
 	$(python) em_test.py
45 45
 
46
+# Test em update ( examples/em_test.pickle )
47
+em_simple: editorial_models/em_simple.py
48
+	$(python) editorial_models/em_simple.py
49
+	
46 50
 # generate leapi dynamic code
47 51
 dyncode: examples/em_test.pickle
48 52
 	$(python) scripts/refreshdyn.py examples/em_test.pickle $(dyncode_filename) && echo -e "\n\nCode generated in $(dyncode_filename)"
49 53
 
54
+# generate leapi dynamic code
55
+dyncode_simple: examples/em_simple.pickle
56
+	$(python) scripts/refreshdyn.py examples/em_simple.pickle $(dyncode_filename) && echo -e "\n\nCode generated in $(dyncode_filename)"
57
+
50 58
 # run tests
51 59
 checks: runtest
52 60
 	./runtest -v

+ 24
- 1
README View File

@@ -90,6 +90,9 @@ NOTE: You have to run in root to use this below
90 90
 		- generate a nginx conf for all instances :
91 91
 			slim -a --nginx-conf
92 92
 
93
+		- set logger to warning level and log in file :
94
+			slim -a -s --set-logger test:WARNING:/var/log/lodel2/%l.%s.log
95
+
93 96
 		- get some help for more options & actions :
94 97
 			slim -h
95 98
 
@@ -130,19 +133,39 @@ Mass deployments tricks & tips:
130 133
 	#Reactivate auth
131 134
 	sed -i -e 's/^noauth = /#noauth =/' -e 's/^#auth = /auth =/' /etc/mongodb.conf
132 135
 	#Test connection
133
-	echo "exit" | mongo --quiet -u $mongoadmin -p $mongopass admin && echo "Connection ok" || echo "connection fails"
136
+	echo "exit" | mongo --quiet -u $mongoadmin -p $mongopass admin && echo "Connection ok" || echo "connection fails"
134 137
 	#Indicate mongodb credentials to mass_deploy
135 138
 	echo -e "MONGODB_ADMIN_USER='$mongoadmin'\nMONGODB_ADMIN_PASSWORD='$mongopass'\n" >> /etc/lodel2/mass_deploy.cfg
136 139
 	#Deploying 50 instances
137 140
 	NINSTANCE=50
138 141
 	#Running mass_deploy
139 142
 	/usr/share/lodel2/scripts/mass_deploy $NINSTANCE
143
+
144
+	#FOLLOWING INSTRUCTIONS ARE FOR STARTING APPS WITH UWSGI
145
+	# for standalone instructions see bellow
146
+
140 147
 	#Updating nginx conf (delete /etc/nginx/sites-enabled/default if exists)
141 148
 	slim --nginx-conf -a > /etc/nginx/sites-enabled/lodel2
142 149
 	/etc/init.d/nginx reload
143 150
 	#Start all instances and check if they managed to start
144 151
 	slim --start -a && sleep 2 && slim -l
145 152
 
153
+	# FOLLOWING INSTRUCTIONS ARE FOR STANDALONE LODEL2 WEBSERVER
154
+	
155
+	#Configure nginx & restart it
156
+	export lodel2_install_dir="/usr/lib/python3/dist-packages";
157
+	echo -e "server {\n\tlisten 80 default_server;\n\tlisten [::]:80 default_server;\n\tlocation / {\n\t\tproxy_pass http://localhost:1337/;\n\t}\n\t\n\tlocation /static/ {\n\t\talias $lodel2_install_dir/lodel/plugins/webui/templates/;\n\t}\n\t\n}\n" > /etc/nginx/sites-enabled/default
158
+	/etc/init.d/nginx restart
159
+	#Build dyncode for instances
160
+	slim -a -m
161
+	#Copy the multisite loader in lodel instances root folder
162
+	cp /usr/lib/python3/dist-packages/lodel/plugins/multisite/loader.py /tmp/lodel2_instances/
163
+	cd /tmp/lodel2_instances/
164
+	#Start the server
165
+	python3 loader.py
166
+	#Now you can access to the app in HTTP via nginx throught port 80 
167
+	#or directly via the python server throught the port 1337
168
+
146 169
 
147 170
 	Cleaning mongodb + instances :
148 171
 	------------------------------

+ 2
- 1
configure.ac View File

@@ -15,7 +15,8 @@ AC_CONFIG_FILES([Makefile \
15 15
 	lodel/utils/Makefile \
16 16
 	progs/Makefile \
17 17
 	progs/slim/Makefile \
18
-	plugins/Makefile \
18
+	lodel/plugins/Makefile \
19
+	lodelsites/Makefile \
19 20
 ])
20 21
 
21 22
 

+ 1
- 1
debian/control View File

@@ -9,6 +9,6 @@ Package: lodel2
9 9
 Section: python
10 10
 Description: lodel2 debian package
11 11
 Architecture: any
12
-Depends: python3, python3-lxml, python3-jinja2, python3-werkzeug, python3-pymongo, uwsgi-plugin-python3
12
+Depends: python3, python3-lxml, python3-jinja2, python3-werkzeug, python3-pymongo, uwsgi-plugin-python3, make
13 13
 Suggests: pwgen, wamerican, mongodb-server
14 14
 Provides: lodel

+ 697
- 0
editorial_models/em_simple.py View File

@@ -0,0 +1,697 @@
1
+#!/usr/bin/python3
2
+#-*- coding: utf-8 -*-
3
+import sys
4
+import os, os.path
5
+
6
+sys.path.append(os.path.dirname(os.getcwd()+'/..'))
7
+from lodel.context import LodelContext
8
+LodelContext.init()
9
+
10
+from lodel.settings.settings import Settings as settings
11
+settings('globconf.d')
12
+from lodel.settings import Settings
13
+
14
+from lodel.editorial_model.components import *
15
+from lodel.editorial_model.exceptions import *
16
+from lodel.editorial_model.model import EditorialModel
17
+
18
+em = EditorialModel('simpleem', 'Simple editorial model')
19
+
20
+base_group = em.new_group(  'base_group',
21
+                            display_name = 'Base group',
22
+                            help_text = 'Base group that implements base EM features (like classtype)'
23
+)
24
+
25
+####################
26
+#   Lodel Object   #
27
+####################
28
+em_abstract = em.new_class( 'abstract_object',
29
+    display_name = 'Abstract lodel object',
30
+    help_text = 'For testing purpose',
31
+    group = base_group,
32
+    abstract = True)
33
+
34
+em_object = em.new_class(   'object',
35
+                            display_name = 'Object',
36
+                            help_text = 'Main class for all Em objects',
37
+                            group = base_group,
38
+                            abstract = True,
39
+                            parents = em_abstract,
40
+)
41
+em_object.new_field(    'lodel_id',
42
+                        display_name = 'Lodel identifier',
43
+                        help_text = 'Uniq ID that identify every lodel object',
44
+                        group = base_group,
45
+                        data_handler = 'uniqid',
46
+                        internal = True,
47
+)
48
+em_object.new_field(    'date_create',
49
+                        display_name = 'Creation date',
50
+                        group = base_group,
51
+                        data_handler = 'datetime',
52
+                        now_on_create = True,
53
+                        internal = True,
54
+)
55
+em_object.new_field(    'date_update',
56
+                        display_name = 'Last update',
57
+                        group = base_group,
58
+                        data_handler = 'datetime',
59
+                        now_on_update = True,
60
+                        internal = True,
61
+)
62
+entitie = em.new_class( 'entitie',
63
+                        display_name = 'entitie',
64
+                        help_text = 'Replace old entity classtype',
65
+                        abstract = True,
66
+                        group = base_group,
67
+                        parents = em_object,
68
+)
69
+########################
70
+# Base group
71
+########################
72
+
73
+person = em.new_class(  'person',
74
+                        display_name = 'Person',
75
+                        help_text = 'Person type',
76
+                        abstract = False,
77
+                        group = base_group,
78
+                        parents = em_object,
79
+)
80
+
81
+person.new_field(   'firstname',
82
+                    display_name = {
83
+                        'eng': 'Firstname',
84
+                        'fre': 'Prénom',
85
+                    },
86
+                    data_handler = 'varchar',
87
+                    group = base_group,
88
+)
89
+person.new_field(   'lastname',
90
+                    display_name = {
91
+                        'eng': 'Lastname',
92
+                        'fre': 'Nom de famille',
93
+                    },
94
+                    data_handler = 'varchar',
95
+                    group = base_group,
96
+)
97
+
98
+# person.new_field(   'role',
99
+#                     display_name = {
100
+#                         'eng': 'Role',
101
+#                         'fre': 'Rôle',
102
+#                     },
103
+#                     data_handler = 'varchar',
104
+#                     group = base_group,
105
+# )
106
+
107
+entry = em.new_class(   'entry',
108
+                        display_name = 'Entry',
109
+                        help_text = 'Entry type',
110
+                        abstract = False,
111
+                        group = base_group,
112
+                        parents = em_object,
113
+)
114
+entry.new_field(    'name',
115
+                    display_name = {
116
+                        'eng': 'Name',
117
+                        'fre': 'Nom',
118
+                    },
119
+                    data_handler = 'varchar',
120
+                    group = base_group,
121
+)
122
+entry.new_field(    'description',
123
+                    display_name = {
124
+                        'eng': 'Description',
125
+                        'fre': 'Description',
126
+                    },
127
+                    data_handler = 'text',
128
+                    group = base_group,
129
+)
130
+
131
+entry.new_field(    'role',
132
+                    display_name = {
133
+                        'eng': 'Role',
134
+                        'fre': 'Rôle',
135
+                    },
136
+                    data_handler = 'varchar',
137
+                    group = base_group,
138
+)
139
+
140
+#####################
141
+# Editorial classes #
142
+#####################
143
+
144
+editorial_group = em.new_group( 'editorial_abstract',
145
+                                display_name = 'Editorial base',
146
+                                help_text = {
147
+                                    'eng': 'Contains abstract class to handler editorial contents',
148
+                                    'fre': 'Contient les classes abstraites permetant la gestion de contenu éditorial'
149
+                                },
150
+                                depends = (base_group,)
151
+)
152
+
153
+################################### Texts ##########################################################
154
+# Classe texte
155
+text = em.new_class(   'text',
156
+                        display_name = 'Text',
157
+                        help_text = 'Abstract class that represent texts',
158
+                        group = editorial_group,
159
+                        abstract = True,
160
+                        parents = entitie,
161
+)
162
+
163
+text.new_field(    'title',
164
+                    display_name = {'eng': 'Title', 'fre': 'Titre'},
165
+                    group = editorial_group,
166
+                    data_handler = 'text',
167
+                    nullable = True,)
168
+text.new_field(    'subtitle',
169
+                    display_name = {
170
+                        'eng': 'Subtitle',
171
+                        'fre': 'Sous-titre',
172
+                    },
173
+                    group = editorial_group,
174
+                    data_handler = 'text',
175
+                    nullable = True,
176
+                    default = None)
177
+text.new_field(    'language',
178
+                    display_name = {
179
+                        'eng': 'Language',
180
+                        'fre': 'Langue',
181
+                    },
182
+                    group = editorial_group,
183
+                    data_handler = 'varchar',
184
+                    nullable = True,
185
+                    default = None)
186
+text.new_field(    'text',
187
+                    display_name = {
188
+                        'eng': 'Text',
189
+                        'fre': 'Texte',
190
+                    },
191
+                    group = editorial_group,
192
+                    data_handler = 'text',
193
+                    nullable = True,
194
+                    default = None)
195
+text.new_field(    'pub_date',
196
+                    display_name = {
197
+                        'eng': 'Publication date',
198
+                        'fre': 'Date de publication',
199
+                    },
200
+                    group = editorial_group,
201
+                    data_handler = 'datetime',
202
+                    nullable = True,
203
+                    default = None)
204
+text.new_field(    'footnotes',
205
+                    display_name = {
206
+                        'eng': 'Footnotes',
207
+                        'fre': 'Notes de bas de page',
208
+                    },
209
+                    group = editorial_group,
210
+                    data_handler = 'text',
211
+                    nullable = True,
212
+                    default = None)
213
+text.new_field(    'linked_entries',
214
+                    display_name = {
215
+                        'eng': 'Related entries',
216
+                        'fre': 'Indices liés',
217
+                    },
218
+                    group = editorial_group,
219
+                    data_handler = 'list',
220
+                    nullable = True,
221
+                    allowed_classes = [entry],
222
+                    back_reference = ('entry', 'linked_texts'),
223
+                    default = None
224
+)
225
+entry.new_field(    'linked_texts',
226
+                    display_name = {
227
+                        'eng': 'Related text',
228
+                        'fre': 'Texte lié ',
229
+                    },
230
+                    data_handler = 'list',
231
+                    nullable = True,
232
+                    allowed_classes = [text],
233
+                    group = editorial_group,
234
+                    back_reference = ('text', 'linked_entries'),
235
+                    default = None
236
+)
237
+# Classe article
238
+article = em.new_class( 'article',
239
+                        display_name = 'Article',
240
+                        group = editorial_group,
241
+                        abstract = False,
242
+                        parents = text)
243
+article.new_field(  'abstract',
244
+                    display_name = {
245
+                        'eng': 'Abstract',
246
+                        'fre': 'Résumé',
247
+                    },
248
+                    group = editorial_group,
249
+                    data_handler = 'text'
250
+)
251
+article.new_field(  'appendix',
252
+                    display_name = {
253
+                        'eng': 'Appendix',
254
+                        'fre': 'Appendice',
255
+                    },
256
+                    group = editorial_group,
257
+                    data_handler = 'text'
258
+)
259
+article.new_field(  'bibliography',
260
+                    display_name = {
261
+                        'eng': 'Bibliography',
262
+                        'fre': 'Bibliographie',
263
+                    },
264
+                    group = editorial_group,
265
+                    data_handler = 'text'
266
+)
267
+article.new_field(  'author_note',
268
+                    display_name = {
269
+                        'eng': 'Author note',
270
+                        'fre': "Note de l'auteur",
271
+                    },
272
+                    group = editorial_group,
273
+                    data_handler = 'text'
274
+)
275
+# Classe Review 
276
+review = em.new_class( 'review',
277
+                        display_name = 'Review',
278
+                        group = editorial_group,
279
+                        abstract = False,
280
+                        parents = text)
281
+review.new_field(   'reference',
282
+                    display_name = {
283
+                        'eng': 'Reference',
284
+                        'fre': "Référence",
285
+                    },
286
+                    group = editorial_group,
287
+                    data_handler = 'text'
288
+)
289
+
290
+###################################### Containers ###########################################
291
+# Classe container
292
+container = em.new_class(  'container',
293
+                            display_name = 'Container',
294
+                            group = editorial_group,
295
+                            abstract = True,
296
+                            parents = entitie)
297
+container.new_field(   'title',
298
+                        display_name = 'Title',
299
+                        group = editorial_group,
300
+                        data_handler = 'text'
301
+)
302
+container.new_field(   'subtitle',
303
+                        display_name = 'Subtitle',
304
+                        group = editorial_group,
305
+                        data_handler = 'text'
306
+)
307
+container.new_field(   'language',
308
+                        display_name = {
309
+                            'eng' : 'Language',
310
+                            'fre' : 'Langue',
311
+                        },
312
+                        group = editorial_group,
313
+                        data_handler = 'varchar'
314
+)
315
+container.new_field(    'linked_directors',
316
+                    display_name = {
317
+                        'eng': 'Directors',
318
+                        'fre': 'Directeurs',
319
+                    },
320
+                    group = editorial_group,
321
+                    data_handler = 'list',
322
+                    nullable = True,
323
+                    allowed_classes = [person],
324
+                    back_reference = ('person', 'linked_containers'),
325
+                    default = None
326
+)
327
+person.new_field(   'linked_containers',
328
+                    display_name = {
329
+                        'eng': 'Director of ',
330
+                        'fre': 'Directeur de ',
331
+                    },
332
+                    group = editorial_group,
333
+                    data_handler = 'list',
334
+                    nullable = True,
335
+                    allowed_classes = [container],
336
+                    back_reference = ('container', 'linked_directors'),
337
+                    default = None
338
+)
339
+container.new_field(    'description',
340
+                    display_name = {
341
+                        'eng': 'Description',
342
+                        'fre': 'Description',
343
+                    },
344
+                    data_handler = 'text',
345
+                    group = editorial_group,
346
+)
347
+container.new_field(    'publisher_note',
348
+                    display_name = {
349
+                        'eng': 'Publisher note',
350
+                        'fre': "Note de l'éditeur",
351
+                    },
352
+                    data_handler = 'text',
353
+                    group = editorial_group,
354
+)
355
+
356
+# Classe collection
357
+collection = em.new_class(  'collection',
358
+                            display_name = 'Collection',
359
+                            group = editorial_group,
360
+                            abstract = False,
361
+                            parents = container)
362
+collection.new_field(    'issn',
363
+                    display_name = {
364
+                        'eng': 'ISSN',
365
+                        'fre': "ISSN",
366
+                    },
367
+                    data_handler = 'varchar',
368
+                    group = editorial_group,
369
+)
370
+# Classe Publication : Pour gérer les back_references vers issue ou part
371
+publication = em.new_class(  'publication',
372
+                            display_name = 'Publication',
373
+                            group = editorial_group,
374
+                            abstract = True,
375
+                            parents = container)
376
+publication.new_field(  'linked_texts',
377
+                        display_name = {
378
+                            'eng': 'Text',
379
+                            'fre': 'Texte',
380
+                        },
381
+                      data_handler = 'list',
382
+                    nullable = True,
383
+                    allowed_classes = [text],
384
+                    group = editorial_group,
385
+                    back_reference = ('text', 'linked_container'),
386
+                    default = None
387
+)
388
+text.new_field(    'linked_container',
389
+                    display_name = {
390
+                        'eng': 'Container',
391
+                        'fre': 'Conteneur',
392
+                    },
393
+                    data_handler = 'link',
394
+                    nullable = True,
395
+                    allowed_classes = [publication],
396
+                    group = editorial_group,
397
+                    back_reference = ('publication', 'linked_texts'),
398
+                    default = None
399
+)
400
+# Classe Issue
401
+issue = em.new_class( 'issue',
402
+                        display_name = 'Issue',
403
+                        group = editorial_group,
404
+                        abstract = False,
405
+                        parents = publication)
406
+issue.new_field(    'isbn',
407
+                  display_name = 'ISBN',
408
+                  data_handler = 'varchar',
409
+                  group = editorial_group,
410
+)
411
+issue.new_field(    'print_isbn',
412
+                  display_name = {
413
+                    'eng': 'Printed ISBN',
414
+                    'fre': "ISBN imprimé",
415
+                  },
416
+                  data_handler = 'varchar',
417
+                  group = editorial_group,
418
+)
419
+issue.new_field(    'number',
420
+                    display_name = {
421
+                        'eng': 'Number',
422
+                        'fre': 'Numéro',
423
+                    },
424
+                  data_handler = 'varchar',
425
+                  group = editorial_group,
426
+)
427
+issue.new_field(    'cover',
428
+                    display_name = {
429
+                        'eng': 'Cover',
430
+                        'fre': 'Couverture',
431
+                    },
432
+                  data_handler = 'varchar',
433
+                  group = editorial_group,
434
+)
435
+issue.new_field(    'print_pub_date',
436
+                    display_name = {
437
+                        'eng': 'Print date',
438
+                        'fre': "Date d'impression",
439
+                    },
440
+                  data_handler = 'datetime',
441
+                  group = editorial_group,
442
+)     
443
+issue.new_field(    'e_pub_date',
444
+                    display_name = {
445
+                        'eng': 'Electronic publication date',
446
+                        'fre': 'Date de publication électronique',
447
+                    },
448
+                  data_handler = 'datetime',
449
+                  group = editorial_group,
450
+)  
451
+issue.new_field(    'abstract',
452
+                    display_name = {
453
+                        'eng': 'Abstract',
454
+                        'fre': 'Résumé',
455
+                    },
456
+                  data_handler = 'text',
457
+                  group = editorial_group,
458
+) 
459
+issue.new_field(    'collection',
460
+                    display_name = {
461
+                        'eng': 'Collection',
462
+                        'fre': 'Collection',
463
+                    },
464
+                    data_handler = 'link',
465
+                    nullable = True,
466
+                    allowed_classes = [collection],
467
+                    group = editorial_group,
468
+                    back_reference = ('collection', 'linked_issues'),
469
+                    default = None
470
+)
471
+collection.new_field(   'linked_issues',
472
+                        display_name = {
473
+                        'eng': 'Linked issues',
474
+                        'fre': 'Numéros',
475
+                    },
476
+                    data_handler = 'hierarch',
477
+                    back_reference = ('Issue', 'collection'),
478
+                    group = editorial_group,
479
+                    allowed_classes = [issue],
480
+                    default = None,
481
+                    nullable = True)
482
+# Classe Part
483
+part = em.new_class(  'part',
484
+                            display_name = 'Part',
485
+                            group = editorial_group,
486
+                            abstract = False,
487
+                            parents = publication,)
488
+part.new_field(     'publication',
489
+                    display_name = {
490
+                        'eng': 'Publication',
491
+                        'fre': 'Publication',
492
+                    },
493
+                    data_handler = 'link',
494
+                    nullable = True,
495
+                    allowed_classes = [publication],
496
+                    group = editorial_group,
497
+                    back_reference = ('publication', 'linked_parts'),
498
+                    default = None
499
+)
500
+publication.new_field(  'linked_parts',
501
+                        display_name = {
502
+                            'eng': 'Parts',
503
+                            'fre': 'Parties',
504
+                        },
505
+                      data_handler = 'hierarch',
506
+                    nullable = True,
507
+                    allowed_classes = [part],
508
+                    group = editorial_group,
509
+                    back_reference = ('part', 'publication'),
510
+                    default = None
511
+)
512
+
513
+#####################
514
+# Persons & authors #
515
+#####################
516
+
517
+editorial_person_group = em.new_group(  'editorial_person',
518
+                                        display_name = 'Editorial person',
519
+                                        help_text = {
520
+                                            'eng': 'Introduce the concept of editorial person (author, translator, director)',
521
+                                            'fre': 'Contient les classes servant à la gestion des personnes éditoriales (auteur, traducteur, directeur...)',
522
+                                        },
523
+                                        depends = (editorial_group,)
524
+)
525
+text_person = em.new_class( 'text_person',
526
+                            display_name = {
527
+                                'eng': 'TextPerson',
528
+                                'fre': 'TextePersonne',
529
+                            },
530
+                            help_text = {
531
+                                'eng': 'Represent a link between someone and a text',
532
+                                'fre': 'Représente un lien entre une personne et un texte',
533
+                            },
534
+                            group = editorial_person_group,
535
+                            abstract = True,
536
+                            parents = entitie,
537
+)
538
+bref_textperson_text = text_person.new_field(  'text',
539
+                                                display_name = {
540
+                                                    'eng': 'Linked text',
541
+                                                    'fre': 'Texte lié',
542
+                                                },
543
+                                                data_handler = 'link',
544
+                                                allowed_classes = [text],
545
+                                                group = editorial_person_group
546
+)
547
+bref_textperson_person = text_person.new_field( 'person',
548
+                                                display_name = {
549
+                                                    'eng': 'Linked person',
550
+                                                    'fre': 'Personne liée',
551
+                                                },
552
+                                                data_handler = 'link',
553
+                                                allowed_classes = [person],
554
+                                                group = editorial_person_group,
555
+)
556
+text_person.new_field(  'role',
557
+                        display_name = {
558
+                            'eng': 'Person role',
559
+                            'fre': 'Role de la personne',
560
+                        },
561
+                        data_handler = 'varchar',
562
+                        group = editorial_person_group
563
+)
564
+
565
+# simple example of linked text / person
566
+person.new_field(   'linked_texts',
567
+                    display_name = {
568
+                        'eng': 'Linked texts',
569
+                        'fre': 'Textes liés',
570
+                    },
571
+                    data_handler = 'list',
572
+                    back_reference = ('Text', 'linked_persons'),
573
+                    group = editorial_person_group,
574
+                    allowed_classes = [text],
575
+                    default = None,
576
+                    nullable = True)
577
+
578
+text.new_field( 'linked_persons',
579
+                display_name = {
580
+                    'eng': 'Linked persons',
581
+                    'fre': 'Personnes liées',
582
+                },
583
+                data_handler = 'list',
584
+                back_reference = ('Person', 'linked_texts'),
585
+                group = editorial_person_group,
586
+                allowed_classes = [person],
587
+                default = None,
588
+                nullable = True)
589
+
590
+#####################
591
+# Index classes     # <--- Note :   using a different datasource for testing
592
+#####################               purpose
593
+
594
+index_group = em.new_group( 'index_group',
595
+                            display_name = 'Indexes',
596
+                            help_text = {
597
+                                'eng': 'EM class that represents indexes'},
598
+                            depends = (editorial_group,))
599
+
600
+index_abstract = em.new_class(
601
+    'indexAbs',
602
+    display_name = {'eng': 'Abstract Index'},
603
+    help_text = {'eng': 'Abstract class common to each Index classes'},
604
+    abstract = True,
605
+    group = index_group,
606
+    datasources = 'dummy2',
607
+    parents = em_object)
608
+
609
+index_name = index_abstract.new_field(
610
+    'name',
611
+    display_name = {
612
+        'eng': 'name',
613
+        'fre': 'nom'},
614
+    data_handler = 'text',
615
+    group = index_group)
616
+
617
+index_value = index_abstract.new_field(
618
+    'value',
619
+    display_name = {
620
+        'eng': 'value',
621
+        'fre': 'valeur'},
622
+    data_handler = 'varchar',
623
+    group = index_group)
624
+
625
+text.new_field( 'indexes',
626
+    display_name = {
627
+        'eng': 'Indexes',
628
+        'fre': 'Indexes'},
629
+    data_handler = 'list',
630
+    back_reference = ('Indexabs', 'texts'),
631
+    allowed_classes = [index_abstract],
632
+    default = None,
633
+    nullable = True,
634
+    group = index_group)
635
+
636
+index_abstract.new_field( 'texts',
637
+    display_name = {
638
+        'eng': 'Text referenced by this index',
639
+        'fre': 'Texte contenant cette index'},
640
+    data_handler = 'list',
641
+    back_reference = ('Text', 'indexes'),
642
+    allowed_classes = [text],
643
+    group = index_group)
644
+
645
+index_theme = em.new_class(
646
+    'indexTheme',
647
+    display_name = {
648
+        'eng': 'Thematic index',
649
+        'fre': 'Index thématique'},
650
+    group = index_group,
651
+    datasources = 'dummy2',
652
+    parents = index_abstract)
653
+
654
+index_theme_theme = index_abstract.new_field(
655
+    'theme',
656
+    display_name = {
657
+        'eng': 'theme'},
658
+    data_handler = 'varchar',
659
+    group = index_group)
660
+
661
+#############
662
+#   USERS   #
663
+#############
664
+
665
+user_group = em.new_group(
666
+    'users', display_name = 'Lodel users',
667
+    help_text = 'Group that handle users en perm')
668
+
669
+user = em.new_class(
670
+    'user', display_name = 'Lodel user', help_text = 'Represent a lodel user',
671
+    group = user_group, abstract = False)
672
+
673
+user.new_field(
674
+    'id', display_name = 'user identifier', help_text = 'Uniq ID',
675
+    group = user_group, data_handler = 'uniqid', internal = True)
676
+
677
+user.new_field(
678
+    'firstname', display_name = 'Firstname',
679
+    group = user_group, data_handler = 'varchar', internal = False)
680
+
681
+user.new_field(
682
+    'lastname', display_name = 'Lastname',
683
+    group = user_group, data_handler = 'varchar', internal = False)
684
+
685
+user.new_field(
686
+    'login', display_name = 'user login', help_text = 'login',
687
+    group = user_group, data_handler = 'varchar', uniq = True, internal = False)
688
+
689
+user.new_field(
690
+    'password', display_name = 'Password',
691
+    group = user_group, data_handler = 'password', internal = False)
692
+
693
+
694
+#em.save('xmlfile', filename = 'examples/em_test.xml')
695
+pickle_file = 'examples/em_simple.pickle'
696
+em.save('picklefile', filename = pickle_file)
697
+print("Output written in %s" % pickle_file)

+ 7
- 0
em_test.py View File

@@ -1,6 +1,9 @@
1 1
 #!/usr/bin/python3
2 2
 #-*- coding: utf-8 -*-
3 3
 
4
+from lodel.context import LodelContext
5
+LodelContext.init()
6
+
4 7
 from lodel.settings.settings import Settings as settings
5 8
 settings('globconf.d')
6 9
 from lodel.settings import Settings
@@ -188,6 +191,8 @@ publication.new_field(  'collection',
188 191
                         display_name = 'Collection',
189 192
                         group = editorial_group,
190 193
                         data_handler = 'link',
194
+                        default = None,
195
+                        nullable = True,
191 196
                         allowed_classes = [collection],
192 197
                         back_reference = ('collection', 'publications'))
193 198
 collection.new_field(   'publications',
@@ -226,6 +231,8 @@ subsection.new_field(   'parent',
226 231
                         display_name = 'Parent',
227 232
                         group = editorial_group,
228 233
                         data_handler = 'link',
234
+                        default = None,
235
+                        nullable = True,
229 236
                         allowed_classes = [section],
230 237
                         back_reference = ('section', 'childs'))
231 238
 

BIN
examples/em_test.pickle View File


+ 2
- 2
lodel/Makefile.am View File

@@ -1,5 +1,5 @@
1
-SUBDIRS=auth editorial_model leapi plugin settings utils
2
-
1
+SUBDIRS=auth editorial_model leapi plugin settings utils plugins
2
+EXTRA_DIST = plugins
3 3
 lodel_PYTHON = *.py
4 4
 CLEANFILES = buildconf.py
5 5
 

+ 9
- 7
lodel/auth/client.py View File

@@ -5,12 +5,14 @@ import sys
5 5
 import warnings
6 6
 import inspect
7 7
 
8
-from lodel.settings import Settings
9
-from lodel import logger
10
-from lodel.plugin.hooks import LodelHook
11
-from lodel.plugin import SessionHandlerPlugin as SessionHandler
12
-from .exceptions import *
13
-from ..leapi.query import LeGetQuery
8
+from lodel.context import LodelContext
9
+LodelContext.expose_modules(globals(), {
10
+    'lodel.settings': ['Settings'],
11
+    'lodel.logger': 'logger',
12
+    'lodel.plugin': [('SessionHandlerPlugin', 'SessionHandler')],
13
+    'lodel.auth.exceptions': ['ClientError', 'ClientAuthenticationFailure',
14
+        'ClientPermissionDenied', 'ClientAuthenticationError'],
15
+    'lodel.leapi.query': ['LeGetQuery'],})
14 16
 
15 17
 ##@brief Client metaclass designed to implements container accessor on 
16 18
 #Client Class
@@ -241,7 +243,7 @@ a session is allready started !!!")
241 243
     #informations on login and password location (LeApi object & field)
242 244
     @classmethod
243 245
     def fetch_settings(cls):
244
-        from lodel import dyncode
246
+        LodelContext.expose_dyncode(globals(), 'dyncode')
245 247
         if cls._infos_fields is None:
246 248
             cls._infos_fields = list()
247 249
         else:

+ 5
- 2
lodel/auth/exceptions.py View File

@@ -1,5 +1,8 @@
1
-from lodel import logger
2
-from lodel.plugin.hooks import LodelHook
1
+from lodel.context import LodelContext
2
+
3
+LodelContext.expose_modules(globals(), {
4
+    'lodel.logger': 'logger',
5
+    'lodel.plugin.hooks': ['LodelHook']})
3 6
 
4 7
 ##@brief Handles common errors with a Client
5 8
 class ClientError(Exception):

+ 471
- 0
lodel/context.py View File

@@ -0,0 +1,471 @@
1
+import importlib
2
+import importlib.machinery
3
+import importlib.abc
4
+import sys
5
+import types
6
+import os
7
+import os.path
8
+import re
9
+import copy
10
+
11
+import warnings #For the moment no way to use the logger in this file (I guess)
12
+
13
+#A try to avoid circular dependencies problems
14
+if 'lodel' not in sys.modules: 
15
+    import lodel
16
+else:
17
+    globals()['lodel'] = sys.modules['lodel']
18
+
19
+if 'lodelsites' not in sys.modules:
20
+    import lodelsites
21
+else:
22
+    globals()['lodelsites'] = sys.modules['lodelsites']
23
+
24
+##@brief Name of the package that will contains all the virtual lodel
25
+#packages
26
+CTX_PKG = "lodelsites"
27
+
28
+##@brief Reserved context name for loading steps
29
+#@note This context is designed to be set at loading time, allowing lodel2
30
+#main process to use lodel packages
31
+LOAD_CTX = "__loader__"
32
+
33
+
34
+#
35
+#   Following exception classes are written here to avoid circular dependencies
36
+#   problems.
37
+#
38
+
39
+##@brief Designed to be raised by the context manager
40
+class ContextError(Exception):
41
+    pass
42
+
43
+##@brief Raised when an error concerning context modules occurs
44
+class ContextModuleError(ContextError):
45
+    pass
46
+
47
+def dir_for_context(site_identifier):
48
+    return os.path.join(lodelsites.__path__[0], site_identifier)
49
+    
50
+
51
+##@brief Designed to permit dynamic packages creation from the lodel package
52
+#
53
+#The class is added in first position in the sys.metapath variable. Doing this
54
+#we override the earlier steps of the import mechanism.
55
+#
56
+#When called the find_spec method determine wether the imported module is
57
+#a part of a virtual lodel package, else  it returns None and the standart
58
+#import mechanism go further.
59
+#If it's a submodule of a virtual lodel package we create a symlink
60
+#to represent the lodel package os the FS and then we make python import
61
+#files from the symlink.
62
+#
63
+#@note Current implementation is far from perfection. In fact no deletion
64
+#mechanisms is written and the virtual package cannot be a subpackage of
65
+#the lodel package for the moment...
66
+#@note Current implementation asserts that all plugins are in CWD
67
+#a symlink will be done to create a copy of the plugins folder in 
68
+#lodelsites/SITENAME/ folder
69
+class LodelMetaPathFinder(importlib.abc.MetaPathFinder):
70
+    
71
+    def find_spec(fullname, path, target = None):
72
+        if fullname.startswith(CTX_PKG):
73
+            spl = fullname.split('.')
74
+            site_identifier = spl[1]
75
+            #creating a symlink to represent the lodel site package
76
+            mod_path = dir_for_context(site_identifier)
77
+            if not os.path.exists(mod_path):
78
+                os.symlink(lodel.__path__[0], mod_path, True)
79
+            #Cache invalidation after we "created" the new package
80
+            #importlib.invalidate_caches()
81
+        return None
82
+
83
+
84
+##@brief Class designed to handle context switching and virtual module
85
+#exposure
86
+#
87
+#@note a dedicated context named LOAD_CTX is used as context for the 
88
+#loading process
89
+class LodelContext(object):
90
+    
91
+    ##@brief FLag telling that the context handler is in single context mode
92
+    MONOSITE = 1
93
+    ##@brief Flag telling that the context manager is in multi context mode
94
+    MULTISITE = 2
95
+
96
+    ##@brief Static property storing current context name
97
+    _current = None
98
+    ##@brief Stores the context type (single or multiple)
99
+    _type = None
100
+    ##@brief Stores the contexts
101
+    _contexts = None
102
+
103
+    ##@brief Flag indicating if the classe is initialized
104
+    __initialized = False
105
+    
106
+    ##@brief Create a new context
107
+    #@see LodelContext.new()
108
+    def __init__(self, site_id, instance_path = None):
109
+        if site_id is None and self.multisite():
110
+            site_id = LOAD_CTX
111
+        if self.multisite() and site_id is not LOAD_CTX:
112
+            with LodelContext.with_context(None) as ctx:
113
+                ctx.expose_modules(globals(), {'lodel.logger': 'logger'})
114
+                logger.info("New context instanciation named '%s'" % site_id)
115
+        if site_id is None:
116
+            self.__id = "MONOSITE"
117
+            #Monosite instanciation
118
+            if self.multisite():
119
+                raise ContextError("Cannot instanciate a context with \
120
+site_id set to None when we are in MULTISITE beahavior")
121
+            else:
122
+                #More verification can be done here (singleton specs ? )
123
+                self.__class__._current = self.__class__._contexts = self
124
+                self.__pkg_name = 'lodel'
125
+                self.__package = lodel
126
+                self.__instance_path = os.getcwd()
127
+                return
128
+        else:
129
+            #Multisite instanciation
130
+            if not self.multisite():
131
+                raise ContextError("Cannot instanciate a context with a \
132
+site_id when we are in MONOSITE beahvior")
133
+            if not self.validate_identifier(site_id):
134
+                raise ContextError("Given context name is not a valide identifier \
135
+    : '%s'" % site_id)
136
+            if site_id in self.__class__._contexts:
137
+                raise ContextError(
138
+                    "A context named '%s' allready exists." % site_id)
139
+            self.__id = site_id
140
+            self.__pkg_name = '%s.%s' % (CTX_PKG, site_id)
141
+
142
+            if instance_path is None:
143
+                """
144
+                raise ContextError("Cannot create a context without an \
145
+instance path")
146
+                """
147
+                warnings.warn("It can be a really BAD idea to create a \
148
+a context without a path......")
149
+                self.__instance_path = None
150
+            else:
151
+                self.__instance_path = os.path.realpath(instance_path)
152
+            #Importing the site package to trigger its creation
153
+            self.__package = importlib.import_module(self.__pkg_name)
154
+            self.__class__._contexts[site_id] = self
155
+        #Designed to be use by with statement
156
+        self.__previous_ctx = None
157
+    
158
+    ##@brief Expose a module from the context
159
+    #@param globs globals : globals where we have to expose the module
160
+    #@param spec tuple : first item is module name, second is the alias
161
+    def expose(self, globs, spec):
162
+        if len(spec) != 2:
163
+            raise ContextError("Invalid argument given. Expected a tuple of \
164
+length == 2 but got : %s" % spec)
165
+        module_fullname, exposure_spec = spec
166
+        module_fullname = self._translate(module_fullname)
167
+        if isinstance(exposure_spec, str):
168
+            self._expose_module(globs, module_fullname, exposure_spec)
169
+        else:
170
+            self._expose_objects(globs, module_fullname, exposure_spec)
171
+
172
+    ##@brief Return a module from current context
173
+    def get_module(self, fullname):
174
+        fullname = self._translate(fullname)
175
+        module = importlib.import_module(fullname)
176
+        return module
177
+        
178
+    
179
+    ##@brief Delete a site's context
180
+    #@param site_id str : the site's name to remove the context
181
+    def remove(cls, site_id):
182
+        if site_id is None:
183
+            if cls._type == cls.MULTISITE:
184
+                raise ContextError("Cannot have a context with \
185
+site_id set to None when we are in MULTISITE beahavior")
186
+            del cls._contexts
187
+        else:
188
+            if cls._type == cls.MULTISITE:
189
+                if site_id in cls._contexts:
190
+                    del cls._contexts[site_id]
191
+                else:
192
+                    raise ContextError("No site %s exist" % site_id)
193
+            else:
194
+                raise ContextError("Cannot have a context with \
195
+    site_id set when we are in MONOSITE beahavior")
196
+    
197
+    ##@return True if the class is in MULTISITE mode
198
+    @classmethod
199
+    def multisite(cls):
200
+        return cls._type == cls.MULTISITE
201
+    
202
+    ##@brief helper class to use LodeContext with with statement
203
+    #@note alias to get method
204
+    #@note maybe useless
205
+    #@todo delete me
206
+    @classmethod
207
+    def with_context(cls, target_ctx_id):
208
+        return cls.get(target_ctx_id)
209
+
210
+    ##@brief Set a context as active
211
+    #@param site_id str : site identifier (identify a context)
212
+    @classmethod
213
+    def set(cls, site_id):
214
+        if cls._type == cls.MONOSITE:
215
+            raise ContextError("Context cannot be set in MONOSITE beahvior")
216
+        site_id = LOAD_CTX if site_id is None else site_id
217
+        if not cls.validate_identifier(site_id):
218
+            raise ContextError("Given context name is not a valide identifier \
219
+: '%s'" % site_id)
220
+        if site_id not in cls._contexts:
221
+            raise ContextError("No context named '%s' found." % site_id)
222
+        wanted_ctx = cls._contexts[site_id]
223
+        if hasattr(wanted_ctx, '__instance_path'):
224
+            os.chdir(self.__instance_path) #May cause problems
225
+        cls._current = wanted_ctx
226
+        return cls._current
227
+    
228
+    ##@brief Getter for contexts
229
+    #@param ctx_id str | None | False : if False return the current context
230
+    #@return A LodelContext instance
231
+    @classmethod
232
+    def get(cls, ctx_id = False):
233
+        if ctx_id is False:
234
+            if cls._current is None:
235
+                raise ContextError("No context loaded")
236
+            return cls._current
237
+        ctx_id = LOAD_CTX if ctx_id is None else ctx_id
238
+        if ctx_id not in cls._contexts:
239
+            raise ContextError("No context identified by '%s'" % ctx_id)
240
+        return cls._contexts[ctx_id]
241
+    
242
+    ##@brief Returns the name of the loaded context
243
+    @classmethod
244
+    def get_name(cls):
245
+        if cls._current is None:
246
+            raise ContextError("No context loaded")
247
+        return copy.copy(cls._current.__id)
248
+        
249
+
250
+    ##@brief Create a new context given a context name
251
+    #
252
+    #@note It's just an alias to the LodelContext.__init__ method
253
+    #@param site_id str : context name
254
+    #@return the context instance
255
+    @classmethod
256
+    def new(cls, site_id, instance_path = None):
257
+        if site_id is None:
258
+            site_id = LOAD_CTX
259
+        return cls(site_id, instance_path)
260
+
261
+    ##@brief Helper function that import and expose specified modules
262
+    #
263
+    #The specs given is a dict. Each element is indexed by a module
264
+    #fullname. Items can be of two types :
265
+    #@par Simple import with alias
266
+    #In this case items of specs is a string representing the alias name
267
+    #for the module we are exposing
268
+    #@par from x import i,j,k equivalent
269
+    #In this case items are lists of object name to expose as it in globals.
270
+    #You can specify an alias by giving a tuple instead of a string as 
271
+    #list element. In this case the first element of the tuple is the object
272
+    #name and the second it's alias in the globals
273
+    #
274
+    #@todo make the specs format more consitant
275
+    #@param cls : bultin params
276
+    #@param globs dict : the globals dict of the caller module
277
+    #@param specs dict : specs of exposure (see comments of this method)
278
+    #@todo implements relative module imports. (maybe by looking for 
279
+    #"calling" package in globs dict)
280
+    @classmethod
281
+    def expose_modules(cls, globs, specs):
282
+        ctx = cls.get()
283
+        for spec in specs.items():
284
+            ctx.expose(globs, spec)
285
+    
286
+    ##@brief Return a module from current context
287
+    #@param fullname str : module fullname
288
+    @classmethod
289
+    def module(cls, fullname):
290
+        return cls.get().get_module(fullname)
291
+        
292
+    ##@brief Expose leapi_dyncode module
293
+    @classmethod
294
+    def expose_dyncode(cls, globs, alias = 'leapi_dyncode'):
295
+        cls.get()._expose_dyncode(globs, alias)
296
+
297
+    ##@brief Initialize the context manager
298
+    #
299
+    #@note Add the LodelMetaPathFinder class to sys.metapath if type is
300
+    #LodelContext.MULTISITE
301
+    #@param type FLAG : takes value in LodelContext.MONOSITE or
302
+    #LodelContext.MULTISITE
303
+    @classmethod
304
+    def init(cls, type=MONOSITE):
305
+        if cls._current is not None:
306
+            raise ContextError("Context allready started and used. Enable to \
307
+initialize it anymore")
308
+        if type not in ( cls.MONOSITE, cls.MULTISITE):
309
+            raise ContextError("Invalid flag given : %s" % type)
310
+        cls._type = type
311
+        if cls._type == cls.MULTISITE:
312
+            cls._contexts = dict()
313
+            #Add custom MetaPathFinder allowing implementing custom imports
314
+            sys.meta_path = [LodelMetaPathFinder] + sys.meta_path
315
+            #Create and set __loader__ context
316
+            cls.new(LOAD_CTX)
317
+            cls.set(LOAD_CTX)
318
+        else:
319
+            #Add a single context with no site_id
320
+            cls._contexts = cls._current = cls(None)
321
+        cls.__initialized = True
322
+    
323
+    ##@return True if the class is initialized
324
+    @classmethod
325
+    def is_initialized(cls):
326
+        return cls.__initialized
327
+    
328
+    ##@brief Return the directory of the package of the current loaded context
329
+    @classmethod
330
+    def context_dir(cls):
331
+        if cls._type == cls.MONOSITE:
332
+            return './'
333
+        return dir_for_context(cls._current.__id)
334
+        
335
+
336
+    ##@brief Validate a context identifier
337
+    #@param identifier str : the identifier to validate
338
+    #@return true if the name is valide else false
339
+    @staticmethod
340
+    def validate_identifier(identifier):
341
+        if identifier == LOAD_CTX:
342
+            return True
343
+        return identifier is None or \
344
+            re.match(r'^[a-zA-Z0-9][a-zA-Z0-9_]', identifier)
345
+    
346
+    ##@brief Safely expose a module in globals using an alias name
347
+    #
348
+    #@note designed to implements warning messages or stuff like that
349
+    #when doing nasty stuff
350
+    #
351
+    #@warning Logging stuffs may lead in a performance issue
352
+    #
353
+    #@todo try to use the logger module instead of warnings
354
+    #@param globs globals : the globals where we want to expose our
355
+    #module alias
356
+    #@param obj object : the object we want to expose
357
+    #@param alias str : the alias name for our module
358
+    @staticmethod
359
+    def safe_exposure(globs, obj, alias):
360
+        if alias in globs:
361
+            if globs[alias] != obj:
362
+                print("Context '%s' : A module exposure leads in globals overwriting for \
363
+key '%s' with a different value : %s != %s" % (LodelContext.get_name(), alias, globs[alias], obj))
364
+                """#Uncomment this bloc to display a stack trace for dangerous modules overwriting
365
+                print("DEBUG INFOS : ")
366
+                import traceback
367
+                traceback.print_stack()
368
+                """
369
+            else:
370
+                print("Context '%s' : A module exposure leads in a useless replacement for \
371
+key '%s'" % (LodelContext.get_name(),alias))
372
+        globs[alias] = obj
373
+        
374
+    ##@brief Create a context from a path and returns the context name
375
+    #@param path str : the path from which we extract a sitename
376
+    #@return the site identifier
377
+    @classmethod
378
+    def from_path(cls, path):
379
+        if cls._type != cls.MULTISITE:
380
+            raise ContextError("Cannot create a context from a path in \
381
+MONOSITE mode")
382
+        site_id = os.path.basename(path.strip('/'))
383
+        path = os.path.realpath(path)
384
+        if not cls.validate_identifier(site_id):
385
+            raise ContextError(
386
+                "Unable to create a context named '%s'" % site_id)
387
+        cls.new(site_id, path)
388
+        return site_id
389
+
390
+
391
+
392
+    ##@brief Utility method to expose a module with an alias name in globals
393
+    #@param globs globals() : concerned globals dict
394
+    #@param fullname str : module fullname
395
+    #@param alias str : alias name
396
+    @classmethod
397
+    def _expose_module(cls, globs, fullname, alias):
398
+        module = importlib.import_module(fullname)
399
+        cls.safe_exposure(globs, module, alias)
400
+    
401
+    ##@brief Utility mehod to expose objects like in a from x import y,z
402
+    #form
403
+    #@param globs globals() : dict of globals
404
+    #@param fullename str : module fullname
405
+    #@param objects list : list of object names to expose
406
+    @classmethod
407
+    def _expose_objects(cls, globs, fullname, objects):
408
+        errors = []
409
+        module = importlib.import_module(fullname)
410
+        for o_name in objects:
411
+            if isinstance(o_name, str):
412
+                alias = o_name
413
+            else:
414
+                o_name, alias = o_name
415
+            if not hasattr(module, o_name):
416
+                errors.append(o_name)
417
+            else:
418
+                cls.safe_exposure(globs, getattr(module, o_name), alias)
419
+        if len(errors) > 0:
420
+            msg = "Module %s does not have any of [%s] as attribute" % (
421
+                fullname, ','.join(errors))
422
+            raise ImportError(msg)
423
+    
424
+    ##@brief Implements LodelContext::expose_dyncode()
425
+    #@todo change hardcoded leapi_dyncode.py filename
426
+    def _expose_dyncode(self, globs, alias = 'leapi_dyncode'):
427
+        fullname = '%s.%s.dyncode' % (CTX_PKG, self.__id)
428
+        if fullname in sys.modules:
429
+            dyncode = sys.modules[fullname]
430
+        else:
431
+            path = os.path.join(self.__instance_path, 'leapi_dyncode.py')
432
+            sfl = importlib.machinery.SourceFileLoader(fullname, path)
433
+            dyncode = sfl.load_module()
434
+        self.safe_exposure(globs, dyncode, alias)
435
+    
436
+    ##@brief Translate a module fullname to the context equivalent
437
+    #@param module_fullname str : a module fullname
438
+    #@return The module name in the current context
439
+    def _translate(self, module_fullname):
440
+        if not module_fullname.startswith('lodel'):
441
+            raise ContextModuleError("Given module is not lodel or any \
442
+submodule : '%s'" % module_fullname)
443
+        return module_fullname.replace('lodel', self.__pkg_name)
444
+
445
+    ##@brief Implements the with statement behavior
446
+    #@see https://www.python.org/dev/peps/pep-0343/
447
+    #@see https://wiki.python.org/moin/WithStatement
448
+    def __enter__(self):
449
+        if not self.multisite:
450
+            warnings.warn("Using LodelContext with with statement in \
451
+MONOSITE mode")
452
+        if self.__previous_ctx is not None:
453
+            raise ContextError("__enter__ called but a previous context \
454
+is allready registered !!! Bailout")
455
+        current = LodelContext.get().__id
456
+        if current != self.__id:
457
+            #Only switch if necessary
458
+            self.__previous_ctx = LodelContext.get().__id
459
+            LodelContext.set(self.__id)
460
+        return self
461
+
462
+    ##@brief Implements the with statement behavior
463
+    #@see https://www.python.org/dev/peps/pep-0343/
464
+    #@see https://wiki.python.org/moin/WithStatement
465
+    def __exit__(self, exc_type, exc_val, exc_tb):
466
+        prev = self.__previous_ctx
467
+        self.__previous_ctx = None
468
+        if prev is not None:
469
+            #Only restore if needed
470
+            LodelContext.set(self.__previous_ctx)
471
+

+ 6
- 5
lodel/editorial_model/components.py View File

@@ -9,11 +9,12 @@ import warnings
9 9
 import copy
10 10
 import hashlib
11 11
 
12
-from lodel.utils.mlstring import MlString
13
-
14
-from lodel.settings import Settings
15
-from lodel.editorial_model.exceptions import *
16
-from lodel.leapi.leobject import CLASS_ID_FIELDNAME
12
+from lodel.context import LodelContext
13
+LodelContext.expose_modules(globals(), {
14
+    'lodel.utils.mlstring': ['MlString'],
15
+    'lodel.settings': ['Settings'],
16
+    'lodel.editorial_model.exceptions': ['EditorialModelError', 'assert_edit'],
17
+    'lodel.leapi.leobject': ['CLASS_ID_FIELDNAME']})
17 18
 
18 19
 ##@brief Abstract class to represent editorial model components
19 20
 # @see EmClass EmField

+ 8
- 6
lodel/editorial_model/model.py View File

@@ -4,13 +4,15 @@ import hashlib
4 4
 import importlib
5 5
 import copy
6 6
 
7
-from lodel.utils.mlstring import MlString
8
-from lodel import logger
9
-from lodel.settings import Settings
10
-from lodel.settings.utils import SettingsError
7
+from lodel.context import LodelContext
8
+LodelContext.expose_modules(globals(), {
9
+    'lodel.utils.mlstring': ['MlString'],
10
+    'lodel.logger': 'logger',
11
+    'lodel.settings': ['Settings'],
12
+    'lodel.settings.utils': ['SettingsError'],
13
+    'lodel.editorial_model.exceptions': ['EditorialModelError', 'assert_edit'],
14
+    'lodel.editorial_model.components': ['EmClass', 'EmField', 'EmGroup']})
11 15
 
12
-from lodel.editorial_model.exceptions import *
13
-from lodel.editorial_model.components import EmClass, EmField, EmGroup
14 16
 
15 17
 ##@brief Describe an editorial model
16 18
 #@ingroup lodel2_em

+ 7
- 3
lodel/editorial_model/translator/xmlfile.py View File

@@ -3,9 +3,13 @@
3 3
 import lxml
4 4
 import os
5 5
 from lxml import etree
6
-from lodel.editorial_model.model import EditorialModel
7
-from lodel.editorial_model.components import *
8
-from lodel.utils.mlstring import MlString
6
+
7
+from lodel.context import LodelContext
8
+LodelContext.expose_modules(globals(), {
9
+    'lodel.editorial_model.model': ['EditorialModel'],
10
+    'lodel.editorial_model.components': ['EmComponent', 'EmClass', 'EmField',
11
+        'EmGroup'],
12
+    'lodel.utils.mlstring': ['MlString']})
9 13
 
10 14
 ##@package lodel.editorial_model.translator.xmlfile Translator module designed
11 15
 #to load & save EM in XML

+ 1
- 0
lodel/exceptions.py View File

@@ -42,3 +42,4 @@ class DataNoneValid(Exception):
42 42
 #@note Designed to be raised in DataHandler
43 43
 class FieldValidationError(Exception):
44 44
     pass
45
+

+ 49
- 69
lodel/leapi/datahandlers/base_classes.py View File

@@ -2,14 +2,20 @@
2 2
 
3 3
 ## @package lodel.leapi.datahandlers.base_classes Define all base/abstract class for data handlers
4 4
 #
5
-# Contains custom exceptions too
5
+# Contains custom exceptions too
6 6
 
7 7
 import copy
8 8
 import importlib
9 9
 import inspect
10 10
 import warnings
11
-from lodel.exceptions import *
12
-from lodel import logger
11
+
12
+from lodel.context import LodelContext
13
+
14
+LodelContext.expose_modules(globals(), {
15
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
16
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError'],
17
+    'lodel.leapi.datahandlers.exceptions': ['LodelDataHandlerConsistencyException', 'LodelDataHandlerException'],
18
+    'lodel.logger': 'logger'})
13 19
 
14 20
 
15 21
 ##@brief Base class for all data handlers
@@ -20,7 +26,7 @@ class DataHandler(object):
20 26
     ##@brief Stores the DataHandler childs classes indexed by name
21 27
     _base_handlers = None
22 28
     ##@brief Stores custom datahandlers classes indexed by name
23
-    # @todo do it ! (like plugins, register handlers... blablabla)
29
+    # @todo do it ! (like plugins, register handlers... blablabla)
24 30
     __custom_handlers = dict()
25 31
 
26 32
     help_text = 'Generic Field Data Handler'
@@ -330,6 +336,12 @@ class Reference(DataHandler):
330 336
             logger.warning('Object referenced does not exist')
331 337
             return False
332 338
         return True
339
+    
340
+    ##@brief Utility method designed to fetch referenced objects
341
+    #@param value mixed : the field value
342
+    #@throw NotImplementedError
343
+    def get_referenced(self, value):
344
+        raise NotImplementedError
333 345
 
334 346
 
335 347
 ##@brief This class represent a data_handler for single reference to another object
@@ -338,7 +350,7 @@ class Reference(DataHandler):
338 350
 class SingleRef(Reference):
339 351
 
340 352
     def __init__(self, allowed_classes=None, **kwargs):
341
-        super().__init__(allowed_classes=allowed_classes)
353
+        super().__init__(allowed_classes=allowed_classes, **kwargs)
342 354
 
343 355
 
344 356
     ##@brief Check and cast value in appropriate type
@@ -352,6 +364,17 @@ class SingleRef(Reference):
352 364
         #    raise FieldValidationError("List or string expected for a set field")
353 365
         return value
354 366
 
367
+    ##@brief Utility method designed to fetch referenced objects
368
+    #@param value mixed : the field value
369
+    #@return A LeObject child class instance
370
+    #@throw LodelDataHandlerConsistencyException if no referenced object found
371
+    def get_referenced(self, value):
372
+        for leo_cls in self.linked_classes:
373
+            res = leo_cls.get_from_uid(value)
374
+            if res is not None:
375
+                return res
376
+        raise LodelDataHandlerConsistencyException("Unable to find \
377
+referenced object with uid %s" % value)
355 378
 
356 379
 
357 380
 ##@brief This class represent a data_handler for multiple references to another object
@@ -397,70 +420,27 @@ class MultipleRef(Reference):
397 420
             raise FieldValidationError("MultipleRef have for invalid values [%s]  :" % (",".join(error_list)))
398 421
         return new_val
399 422
 
400
-    ##@brief Construct a multiple ref data
401
-    def construct_data(self, emcomponent, fname, datas, cur_value):
402
-        cur_value = super().construct_data(emcomponent, fname, datas, cur_value)
403
-        if cur_value is not None:
404
-            if self.back_reference is not None:
405
-                br_class = self.back_reference[0]
406
-                for br_id in cur_value:
407
-                    query_filters = list()
408
-                    query_filters.append((br_class.uid_fieldname()[0], '=', br_id))
409
-                    br_obj = br_class.get(query_filters)
410
-                    if len(br_obj) != 0:
411
-                        br_list = br_obj[0].data(self.back_reference[1])
412
-                        if br_list is None:
413
-                            br_list = list()
414
-                        if br_id not in br_list:
415
-                            br_list.append(br_id)
416
-        return cur_value
417
-
418
-    ## @brief Checks the backreference, updates it if it is not complete
419
-    # @param emcomponent EmComponent : An EmComponent child class instance
420
-    # @param fname : the field name
421
-    # @param datas dict : dict storing fields values
422
-    # @note Not done in case of delete
423
-    def make_consistency(self, emcomponent, fname, datas, type_query):
424
-        dh = emcomponent.field(fname)
425
-        logger.info('Warning : multiple uid capabilities are broken here')
426
-        uid = datas[emcomponent.uid_fieldname()[0]]
427
-        if self.back_reference is not None:
428
-            target_class = self.back_reference[0]
429
-            target_field = self.back_reference[1]
430
-            target_uidfield = target_class.uid_fieldname()[0]
431
-            l_value = datas[fname]
432
-
433
-            if l_value is not None:
434
-                for value in l_value:
435
-                    query_filters = list()
436
-                    query_filters.append((target_uidfield, '=', value))
437
-                    obj = target_class.get(query_filters)
438
-                    if len(obj) == 0:
439
-                        logger.warning('Object referenced does not exist')
440
-                        return False
441
-                    l_uids_ref = obj[0].data(target_field)
442
-                    if l_uids_ref is None:
443
-                        l_uids_ref = list()
444
-                    if uid not in l_uids_ref:
445
-                        l_uids_ref.append(uid)
446
-                        obj[0].set_data(target_field, l_uids_ref)
447
-                        obj[0].update()
448
-
449
-            if type_query == 'update':
450
-                query_filters = list()
451
-                query_filters.append((uid, ' in ', target_field))
452
-                objects = target_class.get(query_filters)
453
-                if l_value is None:
454
-                    l_value = list()
455
-                if len(objects) != len(l_value):
456
-                    for obj in objects:
457
-                        l_uids_ref = obj.data(target_field)
458
-                        if obj.data(target_uidfield) not in l_value:
459
-                            l_uids_ref.remove(uid)
460
-                            obj.set_data(target_field, l_uids_ref)
461
-                            obj.update()
462
-
463
-
423
+    ##@brief Utility method designed to fetch referenced objects
424
+    #@param value mixed : the field value
425
+    #@return A list of LeObject child class instance
426
+    #@throw LodelDataHandlerConsistencyException if some referenced objects
427
+    #were not found
428
+    def get_referenced(self, values):
429
+        if values is None or len(values) == 0:
430
+            return list()
431
+        left = set(values)
432
+        values = set(values)
433
+        res = list()
434
+        for leo_cls in self.linked_classes:
435
+            uidname = leo_cls.uid_fieldname()[0] #MULTIPLE UID BROKEN HERE
436
+            tmp_res = leo_cls.get(('%s in (%s)' % (uidname, ','.join(
437
+                [str(l) for l in left]))))
438
+            left ^= set(( leo.uid() for leo in tmp_res))
439
+            res += tmp_res
440
+            if len(left) == 0:
441
+                return res
442
+        raise LodelDataHandlerConsistencyException("Unable to find \
443
+some referenced objects. Followinf uid were not found : %s" % ','.join(left))
464 444
 
465 445
 ## @brief Class designed to handle datas access will fieldtypes are constructing datas
466 446
 #@ingroup lodel2_datahandlers

+ 9
- 2
lodel/leapi/datahandlers/datas.py View File

@@ -1,10 +1,17 @@
1 1
 #-*- coding: utf-8 -*-
2 2
 import warnings
3 3
 import inspect
4
-from lodel.leapi.datahandlers.datas_base import *
5
-from lodel.exceptions import *
6 4
 import re
7 5
 
6
+from lodel.context import LodelContext
7
+
8
+LodelContext.expose_modules(globals(), {
9
+    'lodel.leapi.datahandlers.datas_base': ['Boolean', 'Integer', 'Varchar',
10
+        'DateTime', 'Text', 'File'],
11
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
12
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
13
+
14
+
8 15
 ##@brief Data field designed to handle formated strings
9 16
 class FormatString(Varchar):
10 17
 

+ 5
- 2
lodel/leapi/datahandlers/datas_base.py View File

@@ -4,8 +4,11 @@ import datetime
4 4
 import time
5 5
 import os
6 6
 
7
-from lodel.leapi.datahandlers.base_classes import DataField
8
-from lodel.exceptions import *
7
+from lodel.context import LodelContext
8
+LodelContext.expose_modules(globals(), {
9
+    'lodel.leapi.datahandlers.base_classes': ['DataField'],
10
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
11
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
9 12
 
10 13
 
11 14
 ##@brief Data field designed to handle boolean values

+ 3
- 0
lodel/leapi/datahandlers/exceptions.py View File

@@ -1,2 +1,5 @@
1 1
 def LodelDataHandlerException(Exception):
2 2
     pass
3
+
4
+def LodelDataHandlerConsistencyException(LodelDataHandlerException):
5
+    pass

+ 8
- 3
lodel/leapi/datahandlers/references.py View File

@@ -1,7 +1,12 @@
1 1
 # -*- coding: utf-8 -*-
2
-from lodel.leapi.datahandlers.base_classes import Reference, MultipleRef, SingleRef
3
-from lodel.exceptions import *
4
-from lodel import logger
2
+
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.leapi.datahandlers.base_classes': ['Reference', 'MultipleRef',
6
+        'SingleRef'],
7
+    'lodel.logger': 'logger',
8
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
9
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
5 10
 
6 11
 class Link(SingleRef):
7 12
     pass

+ 3
- 1
lodel/leapi/exceptions.py View File

@@ -1,6 +1,8 @@
1 1
 #-*- coding: utf-8 -*-
2 2
 
3
-from lodel.exceptions import LodelExceptions, LodelException
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.exceptions': ['LodelExceptions', 'LodelException']})
4 6
 
5 7
 class LeApiError(LodelException):
6 8
     pass

+ 13
- 7
lodel/leapi/lefactory.py View File

@@ -2,10 +2,14 @@
2 2
 
3 3
 import os, os.path
4 4
 import functools
5
-from lodel.editorial_model.components import *
6
-from lodel.leapi.leobject import LeObject
7
-from lodel.leapi.datahandlers.base_classes import DataHandler
8
-from lodel import logger
5
+
6
+from lodel.context import LodelContext
7
+LodelContext.expose_modules(globals(), {
8
+    'lodel.editorial_model.components': ['EmComponent', 'EmClass', 'EmField',
9
+        'EmGroup'],
10
+    'lodel.leapi.leobject': ['LeObject'],
11
+    'lodel.leapi.datahandlers.base_classes': ['DataHandler'],
12
+    'lodel.logger': 'logger'})
9 13
 
10 14
 ##@brief Generate python module code from a given model
11 15
 # @param model lodel.editorial_model.model.EditorialModel
@@ -15,9 +19,11 @@ def dyncode_from_em(model):
15 19
     cls_code, modules, bootstrap_instr = generate_classes(model)
16 20
 
17 21
     # Header
18
-    imports = """from lodel.leapi.leobject import LeObject
19
-from lodel.leapi.datahandlers.base_classes import DataField
20
-from lodel.plugin.hooks import LodelHook
22
+    imports = """from lodel.context import LodelContext
23
+LodelContext.expose_modules(globals(), {
24
+    'lodel.leapi.leobject': ['LeObject'],
25
+    'lodel.leapi.datahandlers.base_classes': ['DataField'],
26
+    'lodel.plugin.hooks': ['LodelHook']})
21 27
 """
22 28
     for module in modules:
23 29
         imports += "import %s\n" % module

+ 1
- 2
lodel/leapi/lefactory_common.py View File

@@ -5,7 +5,6 @@
5 5
 #- All lines that begins with #- will be deleted from dynamically generated
6 6
 #- code...
7 7
 
8
-
9 8
 ##@brief Return a dynamically generated class given it's name
10 9
 #@param name str : The dynamic class name
11 10
 #@return False or a child class of LeObject
@@ -32,6 +31,6 @@ def lowername2class(name):
32 31
 def lodel2_dyncode_datasources_init(self, caller, payload):
33 32
     for cls in dynclasses:
34 33
         cls._init_datasources()
35
-    from lodel.plugin.hooks import LodelHook
34
+    LodelContext.expose_modules(globals(), {'lodel.plugin.hooks': ['LodelHook']})
36 35
     LodelHook.call_hook("lodel2_dyncode_loaded", __name__, dynclasses)
37 36
 

+ 32
- 11
lodel/leapi/leobject.py View File

@@ -4,16 +4,23 @@ import importlib
4 4
 import warnings
5 5
 import copy
6 6
 
7
-from lodel import logger
8
-from lodel.settings import Settings
9
-from lodel.settings.utils import SettingsError
10
-from .query import LeInsertQuery, LeUpdateQuery, LeDeleteQuery, LeGetQuery
11
-from .exceptions import *
12
-from lodel.plugin.exceptions import *
13
-from lodel.plugin.hooks import LodelHook
14
-from lodel.plugin import Plugin, DatasourcePlugin
15
-from lodel.leapi.datahandlers.base_classes import DatasConstructor
16
-from lodel.leapi.datahandlers.base_classes import Reference
7
+from lodel.context import LodelContext
8
+
9
+LodelContext.expose_modules(globals(), {
10
+    'lodel.logger': 'logger',
11
+    'lodel.settings': 'Settings',
12
+    'lodel.settings.utils': 'SettingsError',
13
+    'lodel.leapi.query': ['LeInsertQuery', 'LeUpdateQuery', 'LeDeleteQuery',
14
+        'LeGetQuery'],
15
+    'lodel.leapi.exceptions': ['LeApiError', 'LeApiErrors',
16
+        'LeApiDataCheckError', 'LeApiDataCheckErrors', 'LeApiQueryError',
17
+        'LeApiQueryErrors'],
18
+    'lodel.plugin.exceptions': ['PluginError', 'PluginTypeError',
19
+        'LodelScriptError', 'DatasourcePluginError'],
20
+    'lodel.exceptions': ['LodelFatalError'],
21
+    'lodel.plugin.hooks': ['LodelHook'],
22
+    'lodel.plugin': ['Plugin', 'DatasourcePlugin'],
23
+    'lodel.leapi.datahandlers.base_classes': ['DatasConstructor', 'Reference']})
17 24
 
18 25
 ##@brief Stores the name of the field present in each LeObject that indicates
19 26
 #the name of LeObject subclass represented by this object
@@ -160,7 +167,7 @@ class LeObject(object):
160 167
     def reference_handlers(cls, with_backref = True):
161 168
         return {    fname: fdh 
162 169
                     for fname, fdh in cls.fields(True).items()
163
-                    if issubclass(fdh.__class__, Reference) and \
170
+                    if fdh.is_reference() and \
164 171
                         (not with_backref or fdh.back_reference is not None)}
165 172
     
166 173
     ##@brief Return a LeObject child class from a name
@@ -621,8 +628,22 @@ construction and consitency when datas are not complete\n")
621 628
     #@todo broken multiple UID
622 629
     @classmethod
623 630
     def get_from_uid(cls, uid):
631
+        if cls.uid_fieldname() is None:
632
+            raise LodelFatalError(
633
+                "No uid defined for class %s" % cls.__name__)
624 634
         uidname = cls.uid_fieldname()[0] #Brokes composed UID
625 635
         res = cls.get([(uidname,'=', uid)])
636
+        
637
+        #dedoublonnage vu que query ou la datasource est bugué
638
+        if len(res) > 1:
639
+            res_cp = res
640
+            res = []
641
+            while len(res_cp) > 0:
642
+                cur_res = res_cp.pop()
643
+                if cur_res.uid() in [ r.uid() for r in res_cp]:
644
+                    logger.error("DOUBLON detected in query results !!!")
645
+                else:
646
+                    res.append(cur_res)
626 647
         if len(res) > 1:
627 648
             raise LodelFatalError("Get from uid returned more than one \
628 649
 object ! For class %s with uid value = %s" % (cls, uid))

+ 9
- 5
lodel/leapi/query.py View File

@@ -5,9 +5,13 @@ import copy
5 5
 import inspect
6 6
 import warnings
7 7
 
8
-from .exceptions import *
9
-from lodel.plugin.hooks import LodelHook
10
-from lodel import logger
8
+from lodel.context import LodelContext
9
+LodelContext.expose_modules(globals(), {
10
+    'lodel.leapi.exceptions': ['LeApiError', 'LeApiErrors', 
11
+        'LeApiDataCheckError', 'LeApiDataCheckErrors', 'LeApiQueryError',
12
+        'LeApiQueryErrors'],
13
+    'lodel.plugin.hooks': ['LodelHook'],
14
+    'lodel.logger': ['logger']})
11 15
 
12 16
 
13 17
 ##@todo check datas when running query
@@ -44,11 +48,11 @@ class LeQuery(object):
44 48
             self._target_class.prepare_datas(datas) #not yet implemented
45 49
         if self._hook_prefix is None:
46 50
             raise NotImplementedError("Abstract method")
47
-        LodelHook.call_hook(self._hook_prefix+'_pre',
51
+        LodelHook.call_hook(self._hook_prefix+'pre',
48 52
                                 self._target_class,
49 53
                                 datas)
50 54
         ret = self._query(datas=datas)
51
-        ret = LodelHook.call_hook(self._hook_prefix+'_post',
55
+        ret = LodelHook.call_hook(self._hook_prefix+'post',
52 56
                                     self._target_class,
53 57
                                     ret)
54 58
         return ret

+ 12
- 5
lodel/logger.py View File

@@ -13,17 +13,21 @@ handlers = dict() # Handlers list (generated from settings)
13 13
 ##@brief Stores sent messages until module is able to be initialized
14 14
 msg_buffer = []
15 15
 
16
-# Fetching default root logger
17
-logger = logging.getLogger()
16
+# Fetching logger for current context
17
+from lodel.context import LodelContext
18
+logger = logging.getLogger(LodelContext.get_name())
18 19
 
19 20
 ##@brief Module initialisation from settings
20 21
 #@return True if inited else False
21 22
 def __init_from_settings():
23
+    from lodel.context import LodelContext
22 24
     try:
23
-        from lodel.settings import Settings
25
+        LodelContext.expose_modules(globals(), {
26
+            'lodel.settings': ['Settings']})
24 27
     except Exception:
25 28
         return False
26
-    from lodel.settings.settings import Settings as Lodel2Settings
29
+    LodelContext.expose_modules(globals(), {
30
+        'lodel.settings.settings': [('Settings', 'Lodel2Settings')]})
27 31
     if not Lodel2Settings.started():
28 32
         return False
29 33
     # capture warning disabled, because the custom format raises error (unable
@@ -49,7 +53,7 @@ def __init_from_settings():
49 53
 # @param name str : The handler name
50 54
 # @param logging_opt dict : dict containing options ( see above )
51 55
 def add_handler(name, logging_opt):
52
-    logger = logging.getLogger()
56
+    logger = logging.getLogger(LodelContext.get_name())
53 57
     if name in handlers:
54 58
         raise KeyError("A handler named '%s' allready exists")
55 59
     
@@ -115,6 +119,9 @@ def log(lvl, msg, *args, **kwargs):
115 119
         else:
116 120
             for s_kwargs, args in msg_buffer:
117 121
                 log(*args, **s_kwargs)
122
+    from lodel.context import LodelContext
123
+    if LodelContext.multisite():
124
+        msg = "CTX(%s) %s" % (LodelContext.get_name(), msg)
118 125
     caller = logger.findCaller() # Opti warning : small overhead
119 126
     extra = {
120 127
         '_pathname': os.path.abspath(caller[0]),

+ 8
- 6
lodel/plugin/__init__.py View File

@@ -93,9 +93,11 @@
93 93
 # - @ref lodel.plugin.sessionhandler.SessionHandlerPlugin "SessionHandlerPlugin"
94 94
 #
95 95
 
96
-from .hooks import LodelHook
97
-from .plugins import Plugin, CustomMethod
98
-from .datasource_plugin import DatasourcePlugin
99
-from .sessionhandler import SessionHandlerPlugin
100
-from .interface import InterfacePlugin
101
-from .extensions import Extension
96
+from lodel.context import LodelContext
97
+LodelContext.expose_modules(globals(), {
98
+    'lodel.plugin.hooks': ['LodelHook'],
99
+    'lodel.plugin.plugins': ['Plugin', 'CustomMethod'],
100
+    'lodel.plugin.datasource_plugin': ['DatasourcePlugin'],
101
+    'lodel.plugin.sessionhandler': ['SessionHandlerPlugin'],
102
+    'lodel.plugin.interface': ['InterfacePlugin'],
103
+    'lodel.plugin.extensions': ['Extension']})

+ 9
- 5
lodel/plugin/core_hooks.py View File

@@ -1,8 +1,10 @@
1 1
 #-*- coding: utf-8 -*-
2 2
 
3
-from lodel.plugin import LodelHook
4
-from lodel.settings import Settings
5
-from lodel import logger
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.plugin': ['LodelHook'],
6
+    'lodel.settings': ['Settings'],
7
+    'lodel.logger': 'logger'})
6 8
 
7 9
 ##@package lodel.plugin.core_hooks
8 10
 #@brief Lodel2 internal hooks declaration
@@ -38,7 +40,8 @@ def datasources_bootstrap_hook(hook_name, caller, payload):
38 40
 ##@brief Bootstrap hook that print debug infos about registered hooks
39 41
 @LodelHook('lodel2_bootstraped')
40 42
 def list_hook_debug_hook(name, caller, payload):
41
-    from lodel import logger
43
+    LodelContext.expose_modules(globals(), {
44
+        'lodel.logger': 'logger'})
42 45
     hlist = LodelHook.hook_list()
43 46
     for name, reg_hooks in hlist.items():
44 47
         for hook, priority in reg_hooks:
@@ -55,5 +58,6 @@ def list_hook_debug_hook(name, caller, payload):
55 58
 ##@brief Hooks that trigger custom methods injection in dynmic classes
56 59
 @LodelHook("lodel2_dyncode_loaded")
57 60
 def lodel2_plugins_custom_methods(self, caller, dynclasses):
58
-    from lodel.plugin.plugins import CustomMethod
61
+    LodelContext.expose_modules(globals(), {
62
+        'lodel.plugin.plugins': ['CustomMethod']})
59 63
     CustomMethod.set_registered(dynclasses)

+ 3
- 1
lodel/plugin/core_scripts.py View File

@@ -1,4 +1,6 @@
1
-import lodel.plugin.scripts as lodel_script
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.plugin.scripts': 'lodel_script'})
2 4
 
3 5
 ##@package lodel.plugin.core_scripts
4 6
 #@brief Lodel2 internal scripts declaration

+ 13
- 8
lodel/plugin/datasource_plugin.py View File

@@ -1,8 +1,11 @@
1
-from .plugins import Plugin
2
-from .exceptions import *
3
-from lodel.exceptions import *
4
-from lodel.settings.validator import SettingValidator
5
-
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.plugin.plugins': ['Plugin'],
4
+    'lodel.plugin.exceptions': ['PluginError', 'PluginTypeError',
5
+        'LodelScriptError', 'DatasourcePluginError'],
6
+    'lodel.settings.validator': ['SettingValidator'],
7
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
8
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
6 9
 
7 10
 _glob_typename = 'datasource'
8 11
 
@@ -93,7 +96,7 @@ class DatasourcePlugin(Plugin):
93 96
     _plist_confspecs = {
94 97
         'section': 'lodel2',
95 98
         'key': 'datasource_connectors',
96
-        'default': None,
99
+        'default': 'dummy_datasource',
97 100
         'validator': SettingValidator(
98 101
             'custom_list', none_is_valid = False,
99 102
             validator_name = 'plugin', validator_kwargs = {
@@ -162,7 +165,8 @@ migration handler !!!")
162 165
     #false
163 166
     @staticmethod
164 167
     def plugin_name(ds_name, ro):
165
-        from lodel.settings import Settings
168
+        LodelContext.expose_modules(globals(), {
169
+            'lodel.settings': ['Settings']})
166 170
         # fetching connection identifier given datasource name
167 171
         try:
168 172
             ds_identifier = getattr(Settings.datasources, ds_name)
@@ -198,7 +202,8 @@ DS_PLUGIN_NAME.DS_INSTANCE_NAME. But got %s" % ds_identifier)
198 202
     #@throw NameError if a datasource plugin or instance cannot be found
199 203
     @staticmethod
200 204
     def _get_ds_connection_conf(ds_identifier,ds_plugin_name):
201
-        from lodel.settings import Settings
205
+        LodelContext.expose_modules(globals(), {
206
+            'lodel.settings': ['Settings']})
202 207
         if ds_plugin_name not in Settings.datasource._fields:
203 208
             msg = "Unknown or unconfigured datasource plugin %s"
204 209
             msg %= ds_plugin_name

+ 1
- 1
lodel/plugin/exceptions.py View File

@@ -1,7 +1,7 @@
1 1
 class PluginError(Exception):
2 2
     pass
3 3
 
4
-class PluginTypeErrror(PluginError):
4
+class PluginTypeError(PluginError):
5 5
     pass
6 6
 
7 7
 class LodelScriptError(Exception):

+ 8
- 3
lodel/plugin/extensions.py View File

@@ -1,8 +1,13 @@
1
-from .plugins import Plugin
2
-from .exceptions import *
3
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.plugin.plugins': ['Plugin'],
4
+    'lodel.plugin.exceptions': ['PluginError', 'PluginTypeError',
5
+        'LodelScriptError', 'DatasourcePluginError'],
6
+    'lodel.settings.validator': ['SettingValidator']})
4 7
 
5 8
 _glob_typename = 'extension'
9
+
10
+
6 11
 class Extension(Plugin):
7 12
     
8 13
     _plist_confspecs = {

+ 4
- 4
lodel/plugin/hooks.py View File

@@ -2,7 +2,7 @@
2 2
 
3 3
 import os
4 4
 import copy
5
-from importlib.machinery import SourceFileLoader
5
+from lodel.context import LodelContext
6 6
 
7 7
 
8 8
 ##@brief Class designed to handle a hook's callback with a priority
@@ -64,11 +64,11 @@ class LodelHook(object):
64 64
     # @return modified payload
65 65
     @classmethod
66 66
     def call_hook(cls, hook_name, caller, payload):
67
-        from lodel import logger
68
-        logger.info("Calling hook '%s'" % hook_name)
67
+        LodelContext.expose_modules(globals(), {'lodel.logger': 'logger'})
68
+        logger.debug("Calling hook '%s'" % hook_name)
69 69
         if hook_name in cls._hooks:
70 70
             for hook in cls._hooks[hook_name]:
71
-                logger.info("Lodel hook '%s' calls %s" % (
71
+                logger.debug("Lodel hook '%s' calls %s" % (
72 72
                     hook_name, hook))
73 73
                 payload = hook(hook_name, caller, payload)
74 74
         return payload

+ 7
- 3
lodel/plugin/interface.py View File

@@ -1,9 +1,13 @@
1
-from .plugins import Plugin
2
-from .exceptions import *
3
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.plugin.plugins': ['Plugin'],
4
+    'lodel.plugin.exceptions': ['PluginError', 'PluginTypeError',
5
+        'LodelScriptError', 'DatasourcePluginError'],
6
+    'lodel.settings.validator': ['SettingValidator']})
4 7
 
5 8
 _glob_typename = 'ui'
6 9
 
10
+
7 11
 ##@brief Handles interfaces plugin
8 12
 #@note It's a singleton class. Only 1 interface allowed by instance.
9 13
 class InterfacePlugin(Plugin):

+ 46
- 24
lodel/plugin/plugins.py View File

@@ -5,12 +5,17 @@ import os.path
5 5
 import importlib
6 6
 import copy
7 7
 import json
8
-from importlib.machinery import SourceFileLoader, SourcelessFileLoader
9
-
10
-from lodel import logger
11
-from lodel.settings.utils import SettingsError
12
-from .exceptions import *
13
-from lodel.exceptions import *
8
+from importlib.machinery import SourceFileLoader
9
+
10
+from lodel.context import LodelContext
11
+LodelContext.expose_modules(globals(), {
12
+    'lodel.logger': 'logger',
13
+    'lodel.settings.utils': ['SettingsError'],
14
+    'lodel.plugin.hooks': ['LodelHook'],
15
+    'lodel.plugin.exceptions': ['PluginError', 'PluginTypeError',
16
+        'LodelScriptError', 'DatasourcePluginError'],
17
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
18
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
14 19
 
15 20
 ## @package lodel.plugins Lodel2 plugins management
16 21
 #@ingroup lodel2_plugins
@@ -50,7 +55,7 @@ ACTIVATE_METHOD_NAME = '_activate'
50 55
 ##@brief Discover stage cache filename
51 56
 DISCOVER_CACHE_FILENAME = '.plugin_discover_cache.json'
52 57
 ##@brief Default & failover value for plugins path list
53
-DEFAULT_PLUGINS_PATH_LIST = ['./plugins']
58
+DEFAULT_PLUGINS_PATH_LIST = [os.path.join(LodelContext.context_dir(),'plugins')]
54 59
 
55 60
 ##@brief List storing the mandatory variables expected in a plugin __init__.py
56 61
 #file
@@ -292,13 +297,7 @@ class Plugin(object, metaclass=MetaPlugType):
292 297
         # Importing __init__.py infos in it
293 298
         plugin_module = '%s.%s' % (VIRTUAL_PACKAGE_NAME,
294 299
                                     plugin_name)
295
-
296
-        init_source = os.path.join(self.path, INIT_FILENAME)
297
-        try:
298
-            loader = SourceFileLoader(plugin_module, init_source)
299
-            self.module = loader.load_module()
300
-        except (ImportError,FileNotFoundError) as e:
301
-             raise PluginError("Failed to load plugin '%s'. It seems that the plugin name is not valid or the plugin do not exists" % plugin_name)
300
+        self.module = LodelContext.module(plugin_module)
302 301
 
303 302
         # loading confspecs
304 303
         try:
@@ -374,6 +373,11 @@ name differ from the one found in plugin's init file"
374 373
     #@throw AttributeError if varname not found
375 374
     #@throw ImportError if the file fails to be imported
376 375
     #@throw PluginError if the filename was not valid
376
+    #@todo Some strange things append :
377
+    #when loading modules in test self.module.__name__ does not contains
378
+    #the package... but in prod cases the self.module.__name__ is 
379
+    #the module fullname... Just a reminder note to explain the dirty
380
+    #if on self_modname
377 381
     def _import_from_init_var(self, varname):
378 382
         # Read varname
379 383
         try:
@@ -392,11 +396,15 @@ name differ from the one found in plugin's init file"
392 396
                 fname = filename,
393 397
                 name = self.name)
394 398
             raise PluginError(msg)
395
-        # importing the file in varname
396
-        module_name = self.module.__name__+"."+varname
397
-        filename = os.path.join(self.path, filename)
398
-        loader = SourceFileLoader(module_name, filename)
399
-        return loader.load_module()
399
+        #See the todo
400
+        if len(self.module.__name__.split('.')) == 1:
401
+            self_modname = self.module.__package__
402
+        else:
403
+            self_modname = self.module.__name__
404
+        #extract module name from filename
405
+        base_mod = '.'.join(filename.split('.')[:-1])
406
+        module_name = self_modname+"."+base_mod
407
+        return importlib.import_module(module_name)
400 408
    
401 409
     ##@brief Check dependencies of plugin
402 410
     #@return A list of plugin name to be loaded before
@@ -426,7 +434,6 @@ name differ from the one found in plugin's init file"
426 434
     #
427 435
     # @note Maybe we have to exit everything if a plugin cannot be loaded...
428 436
     def activable(self):
429
-        from lodel import logger
430 437
         try:
431 438
             test_fun = getattr(self.module, ACTIVATE_METHOD_NAME)
432 439
         except AttributeError:
@@ -448,7 +455,6 @@ name differ from the one found in plugin's init file"
448 455
     def _load(self):
449 456
         if self.loaded:
450 457
             return
451
-        from lodel import logger
452 458
         #Test that plugin "wants" to be activated
453 459
         activable = self.activable()
454 460
         if not(activable is True):
@@ -520,7 +526,6 @@ name differ from the one found in plugin's init file"
520 526
                 msg += "\n\t%20s : %s" % (name,e)
521 527
             msg += "\n"
522 528
             raise PluginError(msg)
523
-        from lodel.plugin.hooks import LodelHook
524 529
         LodelHook.call_hook(
525 530
             "lodel2_plugins_loaded", cls, cls._plugin_instances)
526 531
    
@@ -548,7 +553,8 @@ name differ from the one found in plugin's init file"
548 553
     # etc...
549 554
     @classmethod
550 555
     def plugin_list_confspec(cls):
551
-        from lodel.settings.validator import confspec_append
556
+        LodelContext.expose_modules(globals(), {
557
+            'lodel.settings.validator': ['confspec_append']})
552 558
         res = dict()
553 559
         for pcls in cls.plugin_types():
554 560
             plcs = pcls.plist_confspec()
@@ -705,7 +711,6 @@ file : '%s'. Running discover again..." % DISCOVER_CACHE_FILENAME)
705 711
                 #dropped
706 712
                 pass
707 713
         result = {'path_list': paths, 'plugins': result}
708
-        print("DEUG ",result['plugins'])
709 714
         #Writing to cache
710 715
         if not no_cache:
711 716
             with open(DISCOVER_CACHE_FILENAME, 'w+') as pdcache:
@@ -810,6 +815,8 @@ file : '%s'. Running discover again..." % DISCOVER_CACHE_FILENAME)
810 815
     ##@brief Import init file from a plugin path
811 816
     #@param path str : Directory path
812 817
     #@return a tuple (init_module, module_name)
818
+    #@todo replace by LodelContext usage !!! (not mandatory, this fun
819
+    #is only used in plugin discover method)
813 820
     @classmethod
814 821
     def import_init(cls, path):
815 822
         cls._mod_cnt += 1 # in order to ensure module name unicity
@@ -828,11 +835,21 @@ file : '%s'. Running discover again..." % DISCOVER_CACHE_FILENAME)
828 835
             raise PluginError("Unable to import initfile")
829 836
         return (res_module, temp_module)
830 837
 
838
+    @classmethod
839
+    def debug_wrapper(cls, updglob = None):
840
+        if updglob is not None:
841
+            for k, v in updglob.items():
842
+                globals()[k] = v
843
+        print(logger)
844
+
831 845
     ##@brief Reccursiv plugin discover given a path
832 846
     #@param path str : the path to walk through
833 847
     #@return A dict with plugin_name as key and {'path':..., 'version':...} as value
834 848
     @classmethod
835 849
     def _discover(cls, path):
850
+        #Ensure plugins symlink creation
851
+        LodelContext.expose_modules(globals(), {
852
+            'lodel.plugins': 'plugins'})
836 853
         res = []
837 854
         to_explore = [path]
838 855
         while len(to_explore) > 0:
@@ -849,6 +866,8 @@ file : '%s'. Running discover again..." % DISCOVER_CACHE_FILENAME)
849 866
                         to_explore.append(f_path)
850 867
         return res
851 868
 
869
+def debug_wrapper_mod():
870
+    print("MOD : ",logger)
852 871
 
853 872
 ##@brief Decorator class designed to allow plugins to add custom methods
854 873
 #to LeObject childs (dyncode objects)
@@ -961,3 +980,6 @@ with %s" % (custom_method._method_name, custom_method))
961 980
                         custom_method.__get_method())
962 981
                     logger.debug(
963 982
                         "Custom method %s added to target" % custom_method)
983
+
984
+def wrapper_debug_fun():
985
+    print(logger)

+ 6
- 2
lodel/plugin/scripts.py View File

@@ -1,8 +1,11 @@
1 1
 import argparse
2 2
 import sys
3 3
 
4
-from lodel import logger
5
-from lodel.exceptions import *
4
+from lodel.context import LodelContext
5
+LodelContext.expose_modules(globals(), {
6
+    'lodel.logger': 'logger',
7
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
8
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
6 9
 
7 10
 ##@defgroup lodel2_script Administration scripts
8 11
 #@ingroup lodel2_plugins
@@ -16,6 +19,7 @@ from lodel.exceptions import *
16 19
 #@todo store it in MetaLodelScript
17 20
 __registered_scripts = dict()
18 21
 
22
+
19 23
 ##@brief LodelScript metaclass that allows to "catch" child class
20 24
 #declaration
21 25
 #@ingroup lodel2_script

+ 7
- 3
lodel/plugin/sessionhandler.py View File

@@ -1,6 +1,10 @@
1
-from .plugins import Plugin, MetaPlugType
2
-from .exceptions import *
3
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.plugin.plugins': ['Plugin', 'MetaPlugType'],
4
+    'lodel.plugin.exceptions': ['PluginError', 'PluginTypeError',
5
+        'LodelScriptError', 'DatasourcePluginError'],
6
+    'lodel.settings.validator': ['SettingValidator']})
7
+
4 8
 
5 9
 ##@brief SessionHandlerPlugin metaclass designed to implements a wrapper
6 10
 #between SessionHandlerPlugin classmethod and plugin loader functions

plugins/Makefile.am → lodel/plugins/Makefile.am View File


plugins/__init__.py → lodel/plugins/__init__.py View File


plugins/dummy/__init__.py → lodel/plugins/dummy/__init__.py View File

@@ -1,4 +1,6 @@
1
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.settings.validator': ['SettingValidator']})
2 4
 
3 5
 __plugin_name__ = "dummy"
4 6
 __version__ = '0.0.1' #or __version__ = [0,0,1]
@@ -6,7 +8,7 @@ __loader__ = "main.py"
6 8
 __confspec__ = "confspec.py"
7 9
 __author__ = "Lodel2 dev team"
8 10
 __fullname__ = "Dummy plugin"
9
-__name__ = 'yweber.dummy'
11
+__name__ = 'dummy'
10 12
 __plugin_type__ = 'extension'
11 13
 
12 14
 

plugins/dummy/confspec.py → lodel/plugins/dummy/confspec.py View File

@@ -1,6 +1,8 @@
1 1
 #-*- coding: utf-8 -*-
2 2
 
3
-from lodel.settings.validator import SettingValidator
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.settings.validator': ['SettingValidator']})
4 6
 
5 7
 CONFSPEC = {
6 8
     'lodel2.section1': {

plugins/dummy/main.py → lodel/plugins/dummy/main.py View File

@@ -1,6 +1,9 @@
1 1
 #-*- coding: utf-8 -*-
2 2
 
3
-from lodel.plugin import LodelHook, CustomMethod
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.plugin': ['LodelHook', 'CustomMethod'],
6
+    'lodel.settings' : 'settings'})
4 7
 
5 8
 @LodelHook('leapi_get_post')
6 9
 @LodelHook('leapi_update_pre')
@@ -10,7 +13,7 @@ from lodel.plugin import LodelHook, CustomMethod
10 13
 @LodelHook('leapi_insert_pre')
11 14
 @LodelHook('leapi_insert_post')
12 15
 def dummy_callback(hook_name, caller, payload):
13
-    if Lodel.settings.Settings.debug:
16
+    if settings.Settings.debug:
14 17
         print("\tHook %s\tcaller %s with %s" % (hook_name, caller, payload))
15 18
     return payload
16 19
 
@@ -26,9 +29,3 @@ def dummy_instance_method(self):
26 29
     print("Hello world !\
27 30
 I'm a custom method on class %s" % self.__class__)
28 31
 
29
-
30
-@LodelHook('lodel2_loader_main')
31
-def foofun(hname, caller, payload):
32
-    from lodel import dyncode
33
-    print("Hello world ! I read dyncode from lodel.dyncode : ",
34
-        dyncode.dynclasses)

plugins/dummy_datasource/__init__.py → lodel/plugins/dummy_datasource/__init__.py View File

@@ -1,4 +1,6 @@
1
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.settings.validator': ['SettingValidator']})
2 4
 from .datasource import DummyDatasource as Datasource
3 5
 
4 6
 __plugin_type__ = 'datasource'

plugins/dummy_datasource/datasource.py → lodel/plugins/dummy_datasource/datasource.py View File

@@ -1,6 +1,8 @@
1 1
 #-*- coding:utf-8 -*-
2 2
 
3
-from lodel.plugin.datasource_plugin import AbstractDatasource
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.plugin.datasource_plugin': ['AbstractDatasource']})
4 6
 
5 7
 class DummyDatasource(AbstractDatasource):
6 8
     

plugins/dummy_datasource/main.py → lodel/plugins/dummy_datasource/main.py View File

@@ -1,6 +1,8 @@
1 1
 #-*- coding:utf-8 -*-
2 2
 
3
-from lodel.plugin import LodelHook
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.plugin': ['LodelHook']})
4 6
 from .datasource import DummyDatasource as Datasource
5 7
 
6 8
 def migration_handler_class():

plugins/dummy_datasource/migration_handler.py → lodel/plugins/dummy_datasource/migration_handler.py View File


plugins/filesystem_session/__init__.py → lodel/plugins/filesystem_session/__init__.py View File

@@ -1,4 +1,6 @@
1
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.settings.validator': ['SettingValidator']})
2 4
 
3 5
 __plugin_name__ = 'filesystem_session'
4 6
 __version__ = [0,0,1]

plugins/filesystem_session/confspec.py → lodel/plugins/filesystem_session/confspec.py View File

@@ -1,6 +1,8 @@
1 1
 # -*- coding: utf-8 -*-
2 2
 
3
-from lodel.settings.validator import SettingValidator
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.settings.validator': ['SettingValidator']})
4 6
 
5 7
 CONFSPEC = {
6 8
     'lodel2.sessions':{

plugins/filesystem_session/filesystem_session.py → lodel/plugins/filesystem_session/filesystem_session.py View File


plugins/filesystem_session/main.py → lodel/plugins/filesystem_session/main.py View File

@@ -6,9 +6,11 @@ import pickle
6 6
 import re
7 7
 import time
8 8
 
9
-from lodel import logger
10
-from lodel.auth.exceptions import ClientAuthenticationFailure
11
-from lodel.settings import Settings
9
+from lodel.context import LodelContext
10
+LodelContext.expose_modules(globals(), {
11
+    'lodel.logger': 'logger',
12
+    'lodel.auth.exceptions': ['ClientAuthenticationFailure'],
13
+    'lodel.settings': ['Settings']})
12 14
 
13 15
 from .filesystem_session import FileSystemSession
14 16
 

plugins/mongodb_datasource/__init__.py → lodel/plugins/mongodb_datasource/__init__.py View File

@@ -17,7 +17,7 @@ __fullname__ = "MongoDB plugin"
17 17
 #
18 18
 # @return bool|str : True if all the checks are OK, an error message if not
19 19
 def _activate():
20
-    from lodel import buildconf
20
+    from lodel import buildconf #NOTE : this one do not have to pass through the context
21 21
     return buildconf.PYMONGO
22 22
 
23 23
 #

plugins/mongodb_datasource/confspec.py → lodel/plugins/mongodb_datasource/confspec.py View File

@@ -1,6 +1,8 @@
1 1
 # -*- coding: utf-8 -*-
2 2
 
3
-from lodel.settings.validator import SettingValidator
3
+from lodel.context import LodelContext
4
+LodelContext.expose_modules(globals(), {
5
+    'lodel.settings.validator': ['SettingValidator']})
4 6
 
5 7
 ##@brief Mongodb datasource plugin confspec
6 8
 #@ingroup plugin_mongodb_datasource

plugins/mongodb_datasource/datasource.py → lodel/plugins/mongodb_datasource/datasource.py View File

@@ -9,11 +9,13 @@ from collections import OrderedDict
9 9
 import pymongo
10 10
 from pymongo.errors import BulkWriteError
11 11
 
12
-from lodel import logger
13
-from lodel.leapi.leobject import CLASS_ID_FIELDNAME
14
-from lodel.leapi.datahandlers.base_classes import Reference, MultipleRef
15
-from lodel.exceptions import LodelException, LodelFatalError
16
-from lodel.plugin.datasource_plugin import AbstractDatasource
12
+from lodel.context import LodelContext
13
+LodelContext.expose_modules(globals(), {
14
+    'lodel.logger': 'logger',
15
+    'lodel.leapi.leobject': ['CLASS_ID_FIELDNAME'],
16
+    'lodel.leapi.datahandlers.base_classes': ['Reference', 'MultipleRef'],
17
+    'lodel.exceptions': ['LodelException', 'LodelFatalError'],
18
+    'lodel.plugin.datasource_plugin': ['AbstractDatasource']})
17 19
 
18 20
 from . import utils
19 21
 from .exceptions import *
@@ -113,9 +115,9 @@ class MongoDbDatasource(AbstractDatasource):
113 115
             #Here we may implement the group
114 116
             #If sorted query we have to sort again
115 117
             if order is not None:
116
-                results = sorted(results,
117
-                    key=functools.cmp_to_key(
118
-                        self.__generate_lambda_cmp_order(order)))
118
+                key_fun = functools.cmp_to_key(
119
+                    self.__generate_lambda_cmp_order(order))
120
+                results = sorted(results, key=key_fun)
119 121
             #If limit given apply limit again
120 122
             if offset > len(results):
121 123
                 results = list()
@@ -301,7 +303,7 @@ abstract, preparing reccursiv calls" % (target, filters, relational_filters))
301 303
             old_datas_l = self.__collection(target).find(
302 304
                 mongo_filters)
303 305
             old_datas_l = list(old_datas_l)
304
-            
306
+        
305 307
         uidname = target.uid_fieldname()[0] #MULTIPLE UID BROKEN HERE
306 308
         for old_datas in old_datas_l:
307 309
             self.__update_backref(
@@ -364,7 +366,7 @@ abstract, preparing reccursiv calls" % (target, filters, relational_filters))
364 366
                 continue
365 367
             bref_cls = fdh.back_reference[0]
366 368
             bref_fname = fdh.back_reference[1]
367
-            if issubclass(fdh.__class__, MultipleRef):
369
+            if not fdh.is_singlereference():
368 370
                 #fdh is a multiple ref. So the update preparation will be
369 371
                 #divided into two loops :
370 372
                 #- one loop for deleting old datas
@@ -477,7 +479,7 @@ abstract, preparing reccursiv calls" % (target, filters, relational_filters))
477 479
         newdd = 'new' in values
478 480
         if bref_val is None:
479 481
             bref_val = bref_dh.empty()
480
-        if issubclass(bref_dh.__class__, MultipleRef):
482
+        if not bref_dh.is_singlereference():
481 483
             if oldd and newdd:
482 484
                 if tuid not in bref_val:
483 485
                     raise MongoDbConsistencyError("The value we want to \
@@ -544,7 +546,7 @@ value : in %s field %s" % (bref_leo,fname))
544 546
             raise MongoDbConsistencyError("Unable to get the object we make \
545 547
 reference to : %s with uid = %s" % (bref_cls, repr(uidv)))
546 548
         bref_dh = bref_leo.data_handler(bref_fname)
547
-        if not isinstance(bref_dh, Reference):
549
+        if not bref_dh.is_reference():
548 550
             raise LodelFatalError("Found a back reference field that \
549 551
 is not a reference : '%s' field '%s'" % (bref_leo, bref_fname))
550 552
         bref_val = bref_leo.data(bref_fname)
@@ -564,13 +566,17 @@ is not a reference : '%s' field '%s'" % (bref_leo, bref_fname))
564 566
     def __act_on_abstract(self,
565 567
         target, filters, relational_filters, act, **kwargs):
566 568
 
569
+        logger.debug("Abstract %s, running reccursiv select \
570
+on non abstract childs" % act.__name__)
567 571
         result = list() if act == self.select else 0
568 572
         if not target.is_abstract():
569
-            target_childs = target
573
+            target_childs = [target]
570 574
         else:
571 575
             target_childs = [tc for tc in target.child_classes()
572 576
                 if not tc.is_abstract()]
573 577
         for target_child in target_childs:
578
+            logger.debug(
579
+                "Abstract %s on %s" % (act.__name__, target_child.__name__))
574 580
             #Add target_child to filter
575 581
             new_filters = copy.copy(filters)
576 582
             for i in range(len(filters)):
@@ -807,7 +813,7 @@ by an equality filter")
807 813
         #Converting lodel2 wildcarded string into a case insensitive
808 814
         #mongodb re
809 815
         if mongop in cls.mongo_op_re:
810
-            if value.startswith('(') and value.endswith(')') and ',' in value:
816
+            if value.startswith('(') and value.endswith(')'):
811 817
                 if (dhdl.cast_type is not None):
812 818
                     mongoval = [ dhdl.cast_type(item) for item in mongoval[1:-1].split(',') ]
813 819
                 else:
@@ -849,7 +855,7 @@ field/operator couple in a query. We will keep only the first one")
849 855
         glco = cls.__generate_lambda_cmp_order
850 856
         fname, cmpdir = order[0]
851 857
         order = order[1:]
852
-        return lambda a,b: glco(order) if a[fname] == b[fname] else (\
858
+        return lambda a,b: glco(order)(a,b) if a[fname] == b[fname] else (\
853 859
             1 if (a[fname]>b[fname] if cmpdir == 'ASC' else a[fname]<b[fname])\
854 860
             else -1)
855 861
 

plugins/mongodb_datasource/exceptions.py → lodel/plugins/mongodb_datasource/exceptions.py View File

@@ -1,6 +1,9 @@
1
-from lodel.exceptions import *
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.exceptions': ['LodelException', 'LodelExceptions',
4
+        'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
2 5
 
3
-##@ingroup plugin_mongodb_datasource
6
+#@ingroup plugin_mongodb_datasource
4 7
 class MongoDbDataSourceError(Exception):
5 8
     pass
6 9
 

plugins/mongodb_datasource/main.py → lodel/plugins/mongodb_datasource/main.py View File

@@ -1,4 +1,6 @@
1
-from lodel.plugin import LodelHook
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.plugin': ['LodelHook']})
2 4
 
3 5
 from .datasource import MongoDbDatasource as Datasource
4 6
 

plugins/mongodb_datasource/migration_handler.py → lodel/plugins/mongodb_datasource/migration_handler.py View File

@@ -1,15 +1,19 @@
1 1
 # -*- coding: utf-8 -*-
2 2
 import datetime
3 3
 
4
-from lodel.editorial_model.components import EmClass, EmField
5
-from lodel.editorial_model.model import EditorialModel
4
+from lodel.context import LodelContext
5
+LodelContext.expose_modules(globals(), {
6
+    'lodel.editorial_model.components': ['EmClass', 'EmField'],
7
+    'lodel.editorial_model.model': ['EditorialModel'],
8
+    'lodel.leapi.datahandlers.base_classes': ['DataHandler'],
9
+    'lodel.plugin': ['LodelHook'],
10
+    'lodel.logger': 'logger'})
11
+
12
+from leapi_dyncode import * #<-- TODO : handle this !!!
13
+
6 14
 from .utils import connect, object_collection_name, mongo_fieldname
7
-from lodel.leapi.datahandlers.base_classes import DataHandler
8
-from lodel.plugin import LodelHook
9
-from leapi_dyncode import *
10 15
 from .datasource import MongoDbDatasource
11 16
 from .exceptions import *
12
-from lodel import logger
13 17
 
14 18
 class MigrationHandler(object):
15 19
 

plugins/mongodb_datasource/utils.py → lodel/plugins/mongodb_datasource/utils.py View File

@@ -3,8 +3,10 @@
3 3
 import pymongo
4 4
 from pymongo import MongoClient
5 5
 
6
-from lodel.settings.settings import Settings as settings
7
-from lodel import logger
6
+from lodel.context import LodelContext
7
+LodelContext.expose_modules(globals(), {
8
+    'lodel.settings.settings': [('Settings', 'settings')],
9
+    'lodel.logger': 'logger'})
8 10
 
9 11
 common_collections = {
10 12
     'object': 'objects',

+ 23
- 0
lodel/plugins/multisite/__init__.py View File

@@ -0,0 +1,23 @@
1
+from lodel.context import LodelContext, ContextError
2
+try:
3
+    LodelContext.expose_modules(globals(), {
4
+        'lodel.settings.validator': ['SettingValidator']})
5
+
6
+    __plugin_name__ = "multisite"
7
+    __version__ = '0.0.1' #or __version__ = [0,0,1]
8
+    __loader__ = "main.py"
9
+    __author__ = "Lodel2 dev team"
10
+    __fullname__ = "Multisite plugin"
11
+    __name__ = 'yweber.dummy'
12
+    __plugin_type__ = 'extension'
13
+
14
+    CONFSPEC = {
15
+        'lodel2.server': {
16
+            'port': (80,SettingValidator('int')),
17
+            'listen_addr': ('', SettingValidator('string')),
18
+        }
19
+    }
20
+
21
+except ContextError:
22
+    pass
23
+

+ 34
- 0
lodel/plugins/multisite/confspecs.py View File

@@ -0,0 +1,34 @@
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.settings.validator': ['SettingValidator']})
4
+
5
+#Define a minimal confspec used by multisite loader
6
+LODEL2_CONFSPECS = {
7
+    'lodel2': {
8
+        'debug': (True, SettingValidator('bool'))
9
+    },
10
+    'lodel2.server': {
11
+        'listen_address': ('127.0.0.1', SettingValidator('dummy')),
12
+        #'listen_address': ('', SettingValidator('ip')), #<-- not implemented
13
+        'listen_port': ( 1337, SettingValidator('int')),
14
+        'uwsgi_workers': (8, SettingValidator('int')),
15
+        'uwsgicmd': ('/usr/bin/uwsgi', SettingValidator('dummy')),
16
+        'virtualenv': (None, SettingValidator('path', none_is_valid = True)),
17
+    },
18
+    'lodel2.logging.*' : {
19
+        'level': (  'ERROR',
20
+                    SettingValidator('loglevel')),
21
+        'context': (    False,
22
+                        SettingValidator('bool')),
23
+        'filename': (   None,
24
+                        SettingValidator('errfile', none_is_valid = True)),
25
+        'backupcount': (    10,
26
+                            SettingValidator('int', none_is_valid = False)),
27
+        'maxbytes': (   1024*10,
28
+                        SettingValidator('int', none_is_valid = False)),
29
+    },
30
+    'lodel2.datasources.*': {
31
+        'read_only': (False, SettingValidator('bool')),
32
+        'identifier': ( None, SettingValidator('string')),
33
+    }
34
+}

+ 21
- 0
lodel/plugins/multisite/example_lodel2.ini View File

@@ -0,0 +1,21 @@
1
+[lodel2.logging.templog]
2
+level = INFO
3
+filename = /tmp/log
4
+context = True
5
+
6
+[lodel2.server]
7
+listen_address = 
8
+listen_port = 1337
9
+max_children = 100
10
+
11
+[lodel2.logging.stderr]
12
+level = WARNING
13
+filename = -
14
+context = True
15
+
16
+
17
+[lodel2.datasources.default]
18
+identifier = dummy_datasource.default
19
+
20
+[lodel2.datasource.dummy_datasource.default]
21
+dummy = dummy

+ 69
- 0
lodel/plugins/multisite/loader.py View File

@@ -0,0 +1,69 @@
1
+# -*- coding: utf-8 -*-
2
+import os
3
+import sys
4
+import shlex
5
+import warnings
6
+
7
+#Here we have to bootstrap a minimal __loader__ context in order
8
+#to be able to load the settings
9
+#
10
+#This file (once bootstraped) start a new process for uWSGI. uWSGI then
11
+#run lodel.plugins.multisite.run.application function
12
+try:
13
+    from lodel.context import LodelContext
14
+except ImportError:
15
+    LODEL_BASE_DIR = os.path.dirname(
16
+        os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
17
+    from lodel.context import LodelContext
18
+
19
+from lodel import buildconf
20
+
21
+LodelContext.init(LodelContext.MULTISITE)
22
+LodelContext.set(None) #Loading context creation
23
+#Multisite instance settings loading
24
+CONFDIR = os.path.join(os.getcwd(), 'conf.d')
25
+if not os.path.isdir(CONFDIR):
26
+    warnings.warn('%s do not exists, default settings used' % CONFDIR)
27
+LodelContext.expose_modules(globals(), {
28
+    'lodel.settings.settings': [('Settings', 'settings')],
29
+    'lodel.plugins.multisite.confspecs': 'multisite_confspecs'})
30
+if not settings.started():
31
+    settings('./conf.d', multisite_confspecs.LODEL2_CONFSPECS)
32
+
33
+LodelContext.expose_modules(globals(), {
34
+    'lodel.settings': ['Settings']})
35
+
36
+##@brief Starts uwsgi in background using settings
37
+def uwsgi_fork():
38
+    
39
+    sockfile = os.path.join(buildconf.LODEL2VARDIR, 'uwsgi_sockets/')
40
+    if not os.path.isdir(sockfile):
41
+        os.mkdir(sockfile)
42
+    sockfile = os.path.join(sockfile, 'uwsgi_lodel2_multisite.sock')
43
+    logfile = os.path.join(
44
+        buildconf.LODEL2LOGDIR, 'uwsgi_lodel2_multisite.log')
45
+        
46
+    cmd='{uwsgi} --plugin python3 --http-socket {addr}:{port} --module \
47
+lodel.plugins.multisite.run --socket {sockfile} --logto {logfile} -p {uwsgiworkers}'
48
+    cmd = cmd.format(
49
+                addr = Settings.server.listen_address,
50
+                port = Settings.server.listen_port,
51
+                uwsgi= Settings.server.uwsgicmd,
52
+                sockfile=sockfile,
53
+                logfile = logfile,
54
+                uwsgiworkers = Settings.server.uwsgi_workers)
55
+    if Settings.server.virtualenv is not None:
56
+        cmd += " --virtualenv %s" % Settings.webui.virtualenv
57
+
58
+    try:
59
+        args = shlex.split(cmd)
60
+        print("Running %s" % cmd)
61
+        exit(os.execl(args[0], *args))
62
+    except Exception as e:
63
+        print("Webui plugin uwsgi execl fails cmd was '%s' error : " % cmd,
64
+            e, file=sys.stderr)
65
+        exit(1)
66
+
67
+if __name__ == '__main__':
68
+    uwsgi_fork()
69
+        

+ 133
- 0
lodel/plugins/multisite/run.py View File

@@ -0,0 +1,133 @@
1
+# -*- coding: utf-8 -*-
2
+import os
3
+import os.path
4
+import warnings
5
+
6
+#This file expose common function to process a wsgi request and the
7
+#uWSGI application callback
8
+
9
+
10
+#preloading all instances
11
+FAST_APP_EXPOSAL_CACHE = dict()
12
+
13
+LODEL2_INSTANCES_DIR = '.'
14
+EXCLUDE_DIR = {'conf.d', '__pycache__'}
15
+
16
+try:
17
+    from lodel.context import LodelContext
18
+except ImportError:
19
+    LODEL_BASE_DIR = os.path.dirname(
20
+        os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
21
+    from lodel.context import LodelContext, ContextError
22
+
23
+LodelContext.init(LodelContext.MULTISITE)
24
+LodelContext.set(None) #Loading context creation
25
+
26
+#Multisite instance settings loading
27
+CONFDIR = os.path.join(os.getcwd(), 'conf.d')
28
+if not os.path.isdir(CONFDIR):
29
+    warnings.warn('%s do not exists, default settings used' % CONFDIR)
30
+LodelContext.expose_modules(globals(), {
31
+    'lodel.settings.settings': [('Settings', 'settings')],
32
+    'lodel.plugins.multisite.confspecs': 'multisite_confspecs'})
33
+if not settings.started():
34
+    settings('./conf.d', multisite_confspecs.LODEL2_CONFSPECS)
35
+
36
+#Fetching insrtance list from subdirectories
37
+lodelsites_list = [ os.path.realpath(os.path.join(LODEL2_INSTANCES_DIR,sitename)) 
38
+    for sitename in os.listdir(LODEL2_INSTANCES_DIR)
39
+    if os.path.isdir(sitename) and sitename not in EXCLUDE_DIR]
40
+
41
+#Bootstraping instances
42
+for lodelsite_path in lodelsites_list:
43
+    ctx_name = LodelContext.from_path(lodelsite_path)
44
+    #Switch to new context
45
+    LodelContext.set(ctx_name)
46
+    os.chdir(lodelsite_path)
47
+    # Loading settings
48
+    LodelContext.expose_modules(globals(), {
49
+        'lodel.settings.settings': [('Settings', 'settings')]})
50
+    if not settings.started():
51
+        settings('./conf.d')
52
+    LodelContext.expose_modules(globals(), {'lodel.settings': ['Settings']})
53
+
54
+    # Loading hooks & plugins
55
+    LodelContext.expose_modules(globals(), {
56
+        'lodel.plugin': ['LodelHook'],
57
+        'lodel.plugin.core_hooks': 'core_hooks',
58
+        'lodel.plugin.core_scripts': 'core_scripts'
59
+    })
60
+
61
+    #Load plugins
62
+    LodelContext.expose_modules(globals(), {
63
+        'lodel.logger': 'logger',
64
+        'lodel.plugin': ['Plugin']})
65
+    logger.debug("Loader.start() called")
66
+    Plugin.load_all()
67
+    #Import & expose dyncode
68
+    LodelContext.expose_dyncode(globals())
69
+    #Next hook triggers dyncode datasource instanciations
70
+    LodelHook.call_hook('lodel2_plugins_loaded', '__main__', None)
71
+    #Next hook triggers call of interface's main loop
72
+    LodelHook.call_hook('lodel2_bootstraped', '__main__', None)
73
+    #FAST_APP_EXPOSAL_CACHE populate
74
+    FAST_APP_EXPOSAL_CACHE[ctx_name] = LodelContext.module(
75
+    	'lodel.plugins.webui.run')
76
+    LodelContext
77
+    #a dirty & quick attempt to fix context unwanted exite via
78
+    #hooks
79
+    for name in ( 'LodelHook', 'core_hooks', 'core_scripts',
80
+            'Settings', 'settings', 'logger', 'Plugin'):
81
+        del(globals()[name])
82
+    #switch back to loader context
83
+    LodelContext.set(None)
84
+
85
+#
86
+# From here lodel2 multisite instances are loaded and ready to run
87
+#
88
+
89
+
90
+##@brief Utility function to return quickly an error
91
+def http_error(env, start_response, status = '500 internal server error', \
92
+        extra = None):
93
+    headers = [('Content-type', 'text/plain; charset=utf-8')]
94
+    start_response(status, headers)
95
+    msg = status
96
+    if extra is not None:
97
+        msg = extra
98
+    return [msg.encode('utf-8')]
99
+
100
+
101
+##@brief utility function to extract site id from an url
102
+#@param url str : 
103
+def site_id_from_url(url):
104
+    res = ''
105
+    for c in url[1:]:
106
+        if c == '/':
107
+            break
108
+        res += c
109
+    if len(res) == 0:
110
+        return None
111
+    return res
112
+
113
+##@brief This method is run in a child process by the handler
114
+def application(env, start_response):
115
+    #Attempt to load a context
116
+    site_id = site_id_from_url(env['PATH_INFO'])
117
+    if site_id is None:
118
+        #It can be nice to provide a list of instances here
119
+        return http_error(env, start_response, '404 Not Found')
120
+    try:
121
+        LodelContext.set(site_id)
122
+        #We are in the good context
123
+
124
+    except ContextError as e:
125
+        print(e)
126
+        return http_error(env, start_response, '404 Not found',
127
+            "No site named '%s'" % site_id)
128
+    #Calling webui
129
+    return FAST_APP_EXPOSAL_CACHE[site_id].application(env, start_response)
130
+    #LodelContext.expose_modules(globals(), {
131
+    #    'lodel.plugins.webui.run': ['application']})
132
+    #return application(env, start_response)
133
+

plugins/ram_sessions/__init__.py → lodel/plugins/ram_sessions/__init__.py View File

@@ -1,6 +1,8 @@
1
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.settings.validator': ['SettingValidator']})
2 4
 
3
-__plugin_name__ = 'ram_session'
5
+__plugin_name__ = 'ram_sessions'
4 6
 __version__ = [0,0,1]
5 7
 __plugin_type__ = 'session_handler'
6 8
 __loader__ = 'main.py'

plugins/ram_sessions/main.py → lodel/plugins/ram_sessions/main.py View File

@@ -3,9 +3,12 @@ import os
3 3
 import copy
4 4
 import binascii
5 5
 
6
-from lodel import logger
7
-from lodel.settings import Settings
8
-from lodel.auth.exceptions import *
6
+from lodel.context import LodelContext
7
+LodelContext.expose_modules(globals(), {
8
+    'lodel.logger': 'logger',
9
+    'lodel.settings': ['Settings'],
10
+    'lodel.auth.exceptions': ['ClientError', 'ClientAuthenticationFailure',
11
+        'ClientPermissionDenied', 'ClientAuthenticationError']})
9 12
 
10 13
 __sessions = dict()
11 14
 

plugins/webui/__init__.py → lodel/plugins/webui/__init__.py View File


plugins/webui/client.py → lodel/plugins/webui/client.py View File

@@ -1,4 +1,5 @@
1
-from lodel.auth.client import Client
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {'lodel.auth.client': ['Client']})
2 3
 
3 4
 class WebUiClient(Client):
4 5
     

plugins/webui/confspec.py → lodel/plugins/webui/confspec.py View File

@@ -1,4 +1,6 @@
1
-from lodel.settings.validator import SettingValidator
1
+from lodel.context import LodelContext
2
+LodelContext.expose_modules(globals(), {
3
+    'lodel.settings.validator': ['SettingValidator']})
2 4
 
3 5
 CONFSPEC = {
4 6
     'lodel2.webui': {

plugins/webui/exceptions.py → lodel/plugins/webui/exceptions.py View File

@@ -1,6 +1,7 @@
1 1
 #-*- coding: utf-8 -*-
2 2
 
3 3
 from werkzeug.wrappers import Response
4
+from lodel.context import LodelContext
4 5
 
5 6
 class HttpException(Exception):
6 7
 
@@ -23,8 +24,23 @@ class HttpException(Exception):
23 24
         self.status_code = status_code
24 25
         self.tpl = tpl
25 26
         self.custom = custom
27
+    
28
+    ##@brief Log exception with lodel logger
29
+    def log(self):
30
+        LodelContext.expose_modules(globals(), {'lodel.logger': 'logger'})
31
+        msg = "Webui HTTP exception : %s" % self
32
+        if self.status_code / 100 == 4:
33
+            logger.security(msg)
34
+        elif self.status_code / 100 == 5:
35
+            logger.error(msg)
36
+        else:
37
+            logger.warning(msg)
38
+    
39
+    def __str__(self):
40
+        return "HTTP:%d '%s'" % (self.status_code, self.custom)
26 41
 
27 42
     def render(self, request):
43
+        self.log()
28 44
         from .interface.template.loader import TemplateLoader
29 45
         loader = TemplateLoader()
30 46
         tpl_vars = {
@@ -47,5 +63,31 @@ class HttpException(Exception):
47 63
             return HttpException.STATUS_STR[status_fam][status_code]
48 64
 
49 65
 
66
+##@brief Render multiple errors
67
+class HttpErrors(HttpException):
68
+    
69
+    def __init__(self, errors, title = None, status_code = 400):
70
+        super().__init__(status_code = status_code, tpl = 'errors.html',
71
+            custom = title)
72
+        self.errors = errors
73
+
74
+    def __str__(self):
75
+        ret = super().__str__()
76
+        ret += ', '.join([ '%s: %s' % val for val in self.errors.items()])
77
+        return ret
78
+
79
+    def render(self, request):
80
+        self.log()
81
+        from .interface.template.loader import TemplateLoader
82
+        loader = TemplateLoader()
83
+        tpl_vars = {
84
+            'status_code': self.status_code,
85
+            'errors': self.errors,
86
+            'title': self.custom }
87
+        response = Response(
88
+            loader.render_to_response(self.tpl, template_vars = tpl_vars),
89
+            mimetype = 'text/html')
90
+        response.status_code = self.status_code
91
+        return response
50 92
 
51 93
         

plugins/webui/interface/__init__.py → lodel/plugins/webui/interface/__init__.py View File


plugins/webui/interface/controllers/__init__.py → lodel/plugins/webui/interface/controllers/__init__.py View File


plugins/webui/interface/controllers/admin.py → lodel/plugins/webui/interface/controllers/admin.py View File

@@ -2,13 +2,17 @@
2 2
 from ...exceptions import *
3 3
 from .base import get_response
4 4
 
5
-from lodel.leapi.exceptions import *
6
-from lodel import logger
5
+from lodel.context import LodelContext
6
+LodelContext.expose_modules(globals(), {
7
+    'lodel.leapi.exceptions': [],
8
+    'lodel.logger': 'logger',
9
+    'lodel.leapi.datahandlers.base_classes': ['MultipleRef'],
10
+    'lodel.leapi.exceptions': ['LeApiDataCheckErrors'],
11
+    'lodel.exceptions': ['LodelExceptions']})
12
+LodelContext.expose_dyncode(globals(), 'dyncode')
7 13
 
8 14
 from ...client import WebUiClient
9
-import leapi_dyncode as dyncode
10 15
 import warnings
11
-from lodel.leapi.datahandlers.base_classes import MultipleRef
12 16
 
13 17
 LIST_SEPARATOR = ','
14 18
 
@@ -35,54 +39,24 @@ def admin_update(request):
35 39
     #    return get_response('users/signin.html')
36 40
     msg=''
37 41
     
38
-    # If the form has been submitted
39
-    if request.method == 'POST':
40
-        error = None
41
-        datas = list()
42
-        classname = request.form['classname']
43
-        logger.warning('Composed uids broken here')
44
-        uid = request.form['uid']
42
+    datas = process_form(request)
43
+    if not(datas is False):
44
+        if 'lodel_id' not in datas:
45
+            raise HttpException(400)
46
+        target_leo = dyncode.Object.name2class(datas['classname'])
47
+        leo = target_leo.get_from_uid(datas['lodel_id'])
48
+        if leo is None:
49
+            raise HttpException(404,
50
+                custom = 'No %s with id %s' % (
51
+                    target_leo.__name__, datas['lodel_id']))
45 52
         try:
46
-            target_leo = dyncode.Object.name2class(classname)
47
-        except LeApiError:
48
-            classname = None
49
-        if classname is None or target_leo.is_abstract():
50
-            raise HttpException(400, custom = "Bad classname given")
51
-
52
-        leo_to_update = target_leo.get_from_uid(uid)
53
-        
54
-        errors = dict()
55
-        for fieldname, value in request.form.items():
56
-            #We want to drop 2 input named 'classname' and 'uid'
57
-            if len(fieldname) > 12:
58
-                #Other input names are : field_input_FIELDNAME
59
-                #Extract the fieldname
60
-                fieldname = fieldname[12:]
61
-                try:
62
-                    dh = leo_to_update.data_handler(fieldname)
63
-                except NameError as e:
64
-                    errors[fieldname] = e
65
-                    continue
66
-                #Multiple ref list preparation
67
-                if issubclass(dh.__class__, MultipleRef):
68
-                    value=[spl for spl in [
69
-                           v.strip() for v in value.split(LIST_SEPARATOR)]
70
-                        if len(spl) > 0]
71
-                elif len(value.strip()) == 0:
72
-                    value = None
73
-                try:
74
-                    leo_to_update.set_data(fieldname, value)
75
-                except Exception as e:
76
-                    errors[fieldname] = e
77
-                    continue
78
-        if len(errors) > 0:
79
-            custom_msg = '<h1>Errors in datas</h1><ul>'
80
-            for fname, error in errors.items():
81
-                custom_msg += '<li>%s : %s</li>' % (
82
-                    fname, error)
83
-            custom_msg += '</ul>'
84
-            raise HttpException(400, custom = custom_msg)
85
-        leo_to_update.update()
53
+            leo.update(
54
+                { f:datas[f] for f in datas if f not in ('classname', 'lodel_id')})
55
+        except LeApiDataCheckErrors as e:
56
+            raise HttpErrors(
57
+                title='Form validation errors', errors = e._exceptions)
58
+            
59
+            
86 60
 
87 61
     # Display of the form with the object's values to be updated
88 62
     if 'classname' in request.GET:
@@ -132,45 +106,25 @@ def admin_create(request):
132 106
     # temporary, the acl will be more restrictive
133 107
     #if WebUiClient.is_anonymous():
134 108
     #    return get_response('users/signin.html')
135
-    classname = None
136
-     # If the form has been submitted
137
-    if request.method == 'POST':
138
-        error = None
139
-        datas = list()
140
-        classname = request.form['classname']
141
-        try:
142
-            target_leo = dyncode.Object.name2class(classname)
143
-        except LeApiError:
144
-            classname = None
145
-        if classname is None or target_leo.is_abstract():
146
-            raise HttpException(400)
147
-        fieldnames = target_leo.fieldnames()
148
-        fields = dict()
149 109
 
150
-        for in_put, in_value in request.form.items():
151
-            # The classname is handled by the datasource, we are not allowed to modify it
152
-            # both are hidden in the form, to identify the object here
153
-            if in_put != 'classname' and in_value != '':
154
-                dhl = target_leo.data_handler(in_put[12:])
155
-                if dhl.is_reference() and in_value != '' and not dhl.is_singlereference():
156
-                    logger.info(in_value)
157
-                    in_value.replace(" ","")
158
-                    in_value=in_value.split(',')
159
-                    in_value=list(in_value)
160
-                fields[in_put[12:]] = in_value
161
-            if in_value == '':
162
-                fields[in_put[12:]] = None             
163
-
164
-        # Insertion in the database of the values corresponding to a new object
165
-        new_uid = target_leo.insert(fields)
166
-        
167
-        # reurn to the form with a confirmation or error message
168
-        if not new_uid is None:
169
-            msg = 'Successfull creation';
110
+    datas = process_form(request)
111
+    if not(datas is False):
112
+        target_leo = dyncode.Object.name2class(datas['classname'])
113
+        if 'lodel_id' in datas:
114
+            raise HttpException(400)
115
+        try:
116
+            new_uid = target_leo.insert(
117
+                { f:datas[f] for f in datas if f != 'classname'})
118
+        except LeApiDataCheckErrors as e:
119
+            raise HttpErrors(
120
+                title='Form validation errors', errors = e._exceptions)
121
+        if new_uid is None:
122
+            raise HttpException(400, "Creation fails")
170 123
         else:
171
-            msg = 'Oops something wrong happened...object not saved'
172
-        return get_response('admin/admin_create.html', target=target_leo, msg = msg)
173
-    
124
+            return get_response(
125
+                'admin/admin_create.html', target=target_leo,
126
+                msg = "Created with uid %s" % new_uid)
127
+
174 128
     # Display of an empty form
175 129
     if 'classname' in request.GET:
176 130
         # We need the class to create an object in
@@ -318,4 +272,63 @@ def search_object(request):
318 272
         # TODO The get method must be implemented here
319 273
     return get_response('admin/admin_search.html', my_classes = dyncode.dynclasses)
320 274
 
275
+##@brief Process a form POST and return the posted datas
276
+#@param request : the request object
277
+#@return a dict with datas as value and fieldname as key
278
+def process_form(request):
279
+    if request.method != 'POST':
280
+        return False
281
+    res = dict()
282
+    errors = dict()
283
+    #Fetch concerned LeObject
284
+    if 'classname' not in request.form:
285
+        logger.error("Received a form without classname !")
286
+        raise HttpException(400)
287
+    res['classname'] = classname = request.form['classname']
288
+    try:
289
+        target_leo = dyncode.Object.name2class(classname)
290
+    except LeApiError:
291
+        logger.error(
292
+            "Received a form with an invalid leo name : '%s'" % classname)
293
+        raise HttpException(400, "No leobject named '%s'" % classname)
294
+    if target_leo.is_abstract():
295
+        logger.error(
296
+            "Received a form with an abstract leo : '%s'" % classname)
297
+        raise HttpException(400, '%s is abstract' % classname)
298
+    #Process input fields
299
+    for fieldname, value in request.form.items():
300
+        if fieldname == 'classname':
301
+            continue
302
+        elif fieldname == 'uid':
303
+            fieldname = 'lodel_id' #wow
304
+        elif fieldname.startswith('field_input_'):
305
+            fieldname = fieldname[12:]
306
+        try:
307
+            dh = target_leo.data_handler(fieldname)
308
+        except NameError as e:
309
+            errors[fieldname] = e
310
+            continue
311
+        if dh.is_reference() and not dh.is_singlereference():
312
+            #Converting multiple references fields
313
+            value = value.strip()
314
+            if len(value) == 0:
315
+                #handling default value for empty string
316
+                if hasattr(dh, 'default'):
317
+                    value = dh.default
318
+                else:
319
+                    #if not explicit default value, enforcing default as 
320
+                    #an empty list
321
+                    value = []
322
+            else:
323
+                value = [ v.strip() for v in value.split(LIST_SEPARATOR) ]
324
+                value = [ v for v in value if len(v) > 0]
325
+        else:
326
+            #Handling default value for empty string
327
+            if len(value.strip()) == 0 and hasattr(dh, 'default'):
328
+                value = dh.default
329
+        res[fieldname] = value
330
+    if len(errors) > 0:
331
+        del(res)
332
+        raise HttpErrors(errors, title="Form validation error")
333
+    return res
321 334
 

plugins/webui/interface/controllers/base.py → lodel/plugins/webui/interface/controllers/base.py View File


plugins/webui/interface/controllers/document.py → lodel/plugins/webui/interface/controllers/document.py View File


plugins/webui/interface/controllers/listing.py → lodel/plugins/webui/interface/controllers/listing.py View File

@@ -1,8 +1,10 @@
1 1
 # -*- coding: utf-8 -*-
2
+from lodel.context import LodelContext
3
+LodelContext.expose_modules(globals(), {'lodel.logger': 'logger'})
4
+LodelContext.expose_dyncode(globals(), 'dyncode')
5
+
2 6
 from .base import get_response
3 7
 from ...exceptions import *
4
-from lodel import logger
5
-import leapi_dyncode as dyncode
6 8
 
7 9
 ##@brief These functions are called by the rules defined in ../urls.py
8 10
 ## To browse the editorial model

plugins/webui/interface/controllers/users.py → lodel/plugins/webui/interface/controllers/users.py View File

@@ -3,8 +3,9 @@ from .base import get_response
3 3
 from ...exceptions import *
4 4
 from ...client import WebUiClient as WebUiClient
5 5
 
6
-from lodel import logger
7
-import leapi_dyncode as dyncode
6
+from lodel.context import LodelContext
7
+LodelContext.expose_modules(globals(), {'lodel.logger': 'logger'})
8
+LodelContext.expose_dyncode(globals(), 'dyncode')
8 9
 
9 10
 ##@brief These functions are called by the rules defined in ../urls.py
10 11
 ## Their goal is to handle the user authentication
@@ -33,4 +34,4 @@ def signin(request):
33 34
 # @note the response is given in the login html page 
34 35
 def signout(request):
35 36
     WebUiClient.destroy()
36
-    return get_response('users/signin.html')
37
+    return get_response('users/signin.html')

plugins/webui/interface/lodelrequest.py → lodel/plugins/webui/interface/lodelrequest.py View File


plugins/webui/interface/router.py → lodel/plugins/webui/interface/router.py View File

@@ -4,7 +4,10 @@ import re
4 4
 from .controllers import *
5 5
 from .urls import urls
6 6
 from ..main import root_url
7
-from lodel.settings import Settings
7
+
8
+from lodel.context import LodelContext
9
+LodelContext.expose_modules(globals(), {
10
+    'lodel.settings': ['Settings']})
8 11
 
9 12
 def format_url_rule(url_rule):
10 13
     if url_rule.startswith('^'):

plugins/webui/interface/template/__init__.py → lodel/plugins/webui/interface/template/__init__.py View File


plugins/webui/interface/template/api/__init__.py → lodel/plugins/webui/interface/template/api/__init__.py View File


plugins/webui/interface/template/api/api_lodel_templates.py → lodel/plugins/webui/interface/template/api/api_lodel_templates.py View File


plugins/webui/interface/template/exceptions/__init__.py → lodel/plugins/webui/interface/template/exceptions/__init__.py View File


plugins/webui/interface/template/exceptions/not_allowed_custom_api_key_error.py → lodel/plugins/webui/interface/template/exceptions/not_allowed_custom_api_key_error.py View File


plugins/webui/interface/template/loader.py → lodel/plugins/webui/interface/template/loader.py View File

@@ -2,9 +2,11 @@
2 2
 import jinja2
3 3
 import os
4 4
 
5
-from lodel.settings import Settings
5
+from lodel.context import LodelContext
6
+LodelContext.expose_modules(globals(), {'lodel.settings': ['Settings']})
7
+LodelContext.expose_dyncode(globals())
8
+
6 9
 from ...client import WebUiClient as WebUiClient
7
-import leapi_dyncode
8 10
 
9 11
 from .api import api_lodel_templates
10 12
 from .exceptions.not_allowed_custom_api_key_error import NotAllowedCustomAPIKeyError

plugins/webui/interface/urls.py → lodel/plugins/webui/interface/urls.py View File


plugins/webui/main.py → lodel/plugins/webui/main.py View File

@@ -3,9 +3,13 @@
3 3
 import os, os.path
4 4
 import sys
5 5
 import shlex
6
-from lodel.plugin import LodelHook
7
-from lodel.settings import Settings
8
-from lodel import buildconf
6
+
7
+from lodel.context import LodelContext
8
+LodelContext.expose_modules(globals(), {
9
+    'lodel.plugin': ['LodelHook'],
10
+    'lodel.settings': ['Settings']})
11
+
12
+from lodel import buildconf #<-- This one is common to the build
9 13
 
10 14
 PLUGIN_PATH = os.path.dirname(__file__)
11 15
 

plugins/webui/run.py → lodel/plugins/webui/run.py View File

@@ -1,18 +1,26 @@
1 1
 # -*- coding: utf-8 -*-
2
-import loader # Lodel2 loader
2
+from lodel.context import LodelContext
3
+
4
+if not LodelContext.is_initialized():
5
+    import loader # Lodel2 loader
3 6
 
4 7
 import os
5 8
 import hashlib
6 9
 import time
10
+import sys
7 11
 
8 12
 from werkzeug.wrappers import Response
9 13
 
10
-from lodel.settings import Settings
14
+LodelContext.expose_modules(globals(), {
15
+    'lodel.settings': ['Settings'],
16
+    'lodel.logger': 'logger',
17
+    'lodel.auth.exceptions': ['ClientError', 'ClientAuthenticationFailure',
18
+        'ClientPermissionDenied', 'ClientAuthenticationError']})
19
+
11 20
 from .interface.router import get_controller
12 21
 from .interface.lodelrequest import LodelRequest
13 22
 from .exceptions import *
14 23
 from .client import WebUiClient
15
-from lodel.auth.exceptions import *
16 24
 
17 25
 try:
18 26
     SESSION_FILES_BASE_DIR = Settings.webui.sessions.directory
@@ -55,11 +63,11 @@ def empty_cookie(response):
55 63
     response.set_cookie(COOKIE_SESSION_ID, '')
56 64
 
57 65
 #Starting instance
58
-loader.start()
66
+try:
67
+    loader.start() #Works only in MONOSITE mode
68
+except NameError:
69
+    pass
59 70
 #providing access to dyncode
60
-import lodel
61
-import leapi_dyncode as dyncode
62
-lodel.dyncode = dyncode
63 71
 
64 72
 
65 73
 # WSGI Application

plugins/webui/templates/admin/admin.html → lodel/plugins/webui/templates/admin/admin.html View File


plugins/webui/templates/admin/admin_create.html → lodel/plugins/webui/templates/admin/admin_create.html View File


plugins/webui/templates/admin/admin_delete.html → lodel/plugins/webui/templates/admin/admin_delete.html View File


plugins/webui/templates/admin/admin_edit.html → lodel/plugins/webui/templates/admin/admin_edit.html View File


plugins/webui/templates/admin/admin_search.html → lodel/plugins/webui/templates/admin/admin_search.html View File


plugins/webui/templates/admin/editable_component.html → lodel/plugins/webui/templates/admin/editable_component.html View File


plugins/webui/templates/admin/list_classes_admin.html → lodel/plugins/webui/templates/admin/list_classes_admin.html View File


plugins/webui/templates/admin/list_classes_create.html → lodel/plugins/webui/templates/admin/list_classes_create.html View File


plugins/webui/templates/admin/list_classes_delete.html → lodel/plugins/webui/templates/admin/list_classes_delete.html View File


plugins/webui/templates/admin/show_class_admin.html → lodel/plugins/webui/templates/admin/show_class_admin.html View File


plugins/webui/templates/admin/show_class_delete.html → lodel/plugins/webui/templates/admin/show_class_delete.html View File


plugins/webui/templates/base.html → lodel/plugins/webui/templates/base.html View File


plugins/webui/templates/base_backend.html → lodel/plugins/webui/templates/base_backend.html View File


Some files were not shown because too many files changed in this diff

Loading…
Cancel
Save