Wiki

Version 182 (Etienne Pallier, 03/22/2016 10:08 am)

1 1 Etienne Pallier
h1. Technical Documentation for the PYROS project (FGFT-CC)
2 1 Etienne Pallier
3 1 Etienne Pallier
4 1 Etienne Pallier
HOWTO Format Redmine Wiki : http://www.redmine.org/projects/redmine/wiki/FrRedmineWikiFormatting
5 1 Etienne Pallier
6 1 Etienne Pallier
{{>toc}}
7 1 Etienne Pallier
8 1 Etienne Pallier
9 121 Etienne Pallier
---
10 1 Etienne Pallier
11 121 Etienne Pallier
h2. %{margin-left:0px; font-weight:bold; font-size:25px;  display:block; color:red;}I - TODO%
12 121 Etienne Pallier
13 126 Etienne Pallier
 * installation sur windows
14 125 Etienne Pallier
15 126 Etienne Pallier
 * gitlab
16 1 Etienne Pallier
17 126 Etienne Pallier
 * séparation des BD Django et Pyros
18 125 Etienne Pallier
19 126 Etienne Pallier
 * intégration dans Eclipse
20 126 Etienne Pallier
21 174 Etienne Pallier
 * Intégration des modules Django déjà développés
22 125 Etienne Pallier
23 121 Etienne Pallier
24 3 Etienne Pallier
---
25 1 Etienne Pallier
26 140 Etienne Pallier
h2. %{margin-left:0px; font-weight:bold; font-size:25px;  display:block; color:red;}I - DATABASE SCHEMA (v0.2.1)%
27 116 Etienne Pallier
28 116 Etienne Pallier
29 117 Etienne Pallier
{{thumbnail(PYROS_PDM_v021.png, size=300, title=Pyros data model)}}
30 121 Etienne Pallier
31 121 Etienne Pallier
32 121 Etienne Pallier
---
33 116 Etienne Pallier
34 141 Etienne Pallier
h2. %{margin-left:0px; font-weight:bold; font-size:25px;  display:block; color:red;}II - Get the project (from gitlab)%
35 141 Etienne Pallier
36 141 Etienne Pallier
https://gitlab.irap.omp.eu/epallier/pyros
37 141 Etienne Pallier
38 161 Etienne Pallier
https://gitlab.irap.omp.eu/epallier/pyros.git
39 161 Etienne Pallier
40 161 Etienne Pallier
ssh git@gitlab.irap.omp.eu:epallier/pyros.git
41 141 Etienne Pallier
42 1 Etienne Pallier
43 169 Etienne Pallier
44 169 Etienne Pallier
h3. Get the project from the terminal
45 169 Etienne Pallier
46 174 Etienne Pallier
<pre>
47 1 Etienne Pallier
git clone https://gitlab.irap.omp.eu/epallier/pyros.git PYROS
48 1 Etienne Pallier
49 174 Etienne Pallier
(or also : git clone git@gitlab.irap.omp.eu:epallier/pyros.git)
50 174 Etienne Pallier
</pre>
51 174 Etienne Pallier
52 174 Etienne Pallier
If you just want a static copy of the project (without synchronization) just remove the .git/ folder :
53 174 Etienne Pallier
<pre>
54 174 Etienne Pallier
$ rm -r .git/
55 174 Etienne Pallier
</pre>
56 169 Etienne Pallier
57 169 Etienne Pallier
h3. Get the project from Eclipse
58 169 Etienne Pallier
59 169 Etienne Pallier
TODO:
60 169 Etienne Pallier
61 141 Etienne Pallier
h2. %{margin-left:0px; font-weight:bold; font-size:25px;  display:block; color:red;}III - INSTALLATION%
62 2 Etienne Pallier
63 120 Etienne Pallier
64 120 Etienne Pallier
---
65 120 Etienne Pallier
66 51 Etienne Pallier
h3. If necessary, install MySql
67 2 Etienne Pallier
68 49 Etienne Pallier
 * Linux Ubuntu
69 49 Etienne Pallier
<pre>
70 107 Paul Carensac
71 107 Paul Carensac
$ sudo apt-get install mysql-server
72 107 Paul Carensac
$ sudo apt-get install mysql-client
73 107 Paul Carensac
74 49 Etienne Pallier
</pre>
75 49 Etienne Pallier
76 49 Etienne Pallier
 * Linux CentOS
77 49 Etienne Pallier
<pre>
78 49 Etienne Pallier
TODO:
79 49 Etienne Pallier
$ sudo yum install mysql
80 49 Etienne Pallier
...
81 49 Etienne Pallier
</pre>
82 49 Etienne Pallier
83 49 Etienne Pallier
 * Mac OS X
84 1 Etienne Pallier
Install XAMPP
85 1 Etienne Pallier
(but you could also use the pre-installed Mac OS MySql)
86 177 Etienne Pallier
<pre>
87 177 Etienne Pallier
TODO:
88 49 Etienne Pallier
</pre>
89 1 Etienne Pallier
90 49 Etienne Pallier
 * Windows
91 177 Etienne Pallier
Install XAMPP
92 49 Etienne Pallier
<pre>
93 49 Etienne Pallier
TODO:
94 49 Etienne Pallier
</pre>
95 49 Etienne Pallier
96 49 Etienne Pallier
97 49 Etienne Pallier
---
98 1 Etienne Pallier
99 6 Etienne Pallier
h3. Install python3.5
100 1 Etienne Pallier
101 11 Etienne Pallier
102 11 Etienne Pallier
 * Mac OS X :
103 23 Etienne Pallier
<pre>
104 24 Etienne Pallier
1) Installer MacPort
105 24 Etienne Pallier
(TODO: doc)
106 24 Etienne Pallier
107 24 Etienne Pallier
2) Installer le "port" python35
108 1 Etienne Pallier
$ sudo port install python35
109 1 Etienne Pallier
</pre>
110 1 Etienne Pallier
111 1 Etienne Pallier
 * Linux (Ubuntu) :
112 1 Etienne Pallier
<pre>
113 107 Paul Carensac
sudo add-apt-repository ppa:fkrull/deadsnakes
114 107 Paul Carensac
sudo apt-get update
115 107 Paul Carensac
sudo apt-get install python3.5
116 107 Paul Carensac
117 107 Paul Carensac
sudo pip install virtualenv
118 23 Etienne Pallier
</pre>
119 1 Etienne Pallier
120 5 Etienne Pallier
121 3 Etienne Pallier
122 4 Etienne Pallier
---
123 5 Etienne Pallier
124 59 Etienne Pallier
h3. Get the project (from git)
125 59 Etienne Pallier
126 78 Etienne Pallier
<pre>
127 59 Etienne Pallier
TODO:
128 79 Etienne Pallier
129 130 Etienne Pallier
$ git http://pyros... PYROS
130 78 Etienne Pallier
</pre>
131 1 Etienne Pallier
132 130 Etienne Pallier
This creates a PYROS folder containing the project
133 59 Etienne Pallier
134 59 Etienne Pallier
135 59 Etienne Pallier
---
136 59 Etienne Pallier
137 131 Etienne Pallier
h3. Create project structure
138 131 Etienne Pallier
139 131 Etienne Pallier
<pre>
140 131 Etienne Pallier
$ mkdir PYROS
141 132 Etienne Pallier
</pre>
142 131 Etienne Pallier
143 131 Etienne Pallier
Example of a good organization :
144 131 Etienne Pallier
145 131 Etienne Pallier
1 project = N applis
146 158 Etienne Pallier
1 appli = N models
147 158 Etienne Pallier
A! L’appli est à côté du projet, PAS DEDANS,
148 158 Etienne Pallier
cela facilite la REUTILISATION
149 158 Etienne Pallier
(an appli can be part of many projects => reuse)
150 131 Etienne Pallier
1 appli = 1 Python module, organized for Django, by default = appli web (but not mandatory)
151 131 Etienne Pallier
152 131 Etienne Pallier
<pre>
153 131 Etienne Pallier
154 131 Etienne Pallier
MYPROJECT/
155 171 Etienne Pallier
	REQUIREMENTS.txt
156 131 Etienne Pallier
	src/
157 131 Etienne Pallier
		myproject/
158 131 Etienne Pallier
		appli1/
159 131 Etienne Pallier
		appli2/
160 131 Etienne Pallier
161 131 Etienne Pallier
		appliN/
162 131 Etienne Pallier
163 131 Etienne Pallier
	public/
164 131 Etienne Pallier
		static/
165 131 Etienne Pallier
166 131 Etienne Pallier
	private/
167 148 Etienne Pallier
		venv_py35_pyros/
168 131 Etienne Pallier
169 131 Etienne Pallier
</pre>
170 131 Etienne Pallier
171 131 Etienne Pallier
172 131 Etienne Pallier
Set needed folders:
173 131 Etienne Pallier
<pre>
174 131 Etienne Pallier
$ cd PYROS/
175 131 Etienne Pallier
$ mkdir private public
176 131 Etienne Pallier
$ mkdir public/static
177 131 Etienne Pallier
</pre>
178 131 Etienne Pallier
179 131 Etienne Pallier
180 131 Etienne Pallier
181 131 Etienne Pallier
---
182 131 Etienne Pallier
183 3 Etienne Pallier
h3. Create virtualenv with python3.5 dedicated to pyros project (inside the project folder)
184 127 Etienne Pallier
185 131 Etienne Pallier
<pre>
186 1 Etienne Pallier
187 133 Etienne Pallier
$ cd private/
188 3 Etienne Pallier
189 3 Etienne Pallier
$ which python3.5
190 3 Etienne Pallier
/opt/local/bin/python3.5
191 15 Etienne Pallier
192 127 Etienne Pallier
$ virtualenv-3.5 venv_py35_pyros -p /opt/local/bin/python3.5
193 3 Etienne Pallier
=> creates a venv_py35_pyros/ folder inside PYROS/private/
194 1 Etienne Pallier
195 131 Etienne Pallier
</pre>
196 1 Etienne Pallier
197 7 Etienne Pallier
---
198 3 Etienne Pallier
199 65 Etienne Pallier
h3. Activate the python virtual environment (from inside the project)
200 7 Etienne Pallier
201 55 Etienne Pallier
<pre>
202 1 Etienne Pallier
203 65 Etienne Pallier
$ pwd
204 134 Etienne Pallier
.../PYROS/private
205 1 Etienne Pallier
206 65 Etienne Pallier
$ source ./venv_py35_pyros/bin/activate
207 65 Etienne Pallier
208 3 Etienne Pallier
$ python -V
209 1 Etienne Pallier
Python 3.5.1
210 3 Etienne Pallier
211 3 Etienne Pallier
$ which pip
212 77 Etienne Pallier
.../PYROS/venv_py35_pyros/bin/pip
213 16 Etienne Pallier
214 18 Etienne Pallier
Upgrade pip to last version available :
215 3 Etienne Pallier
$ pip install --upgrade pip
216 3 Etienne Pallier
Collecting pip
217 3 Etienne Pallier
  Downloading pip-8.1.1-py2.py3-none-any.whl (1.2MB)
218 3 Etienne Pallier
Installing collected packages: pip
219 3 Etienne Pallier
  Found existing installation: pip 7.1.2
220 3 Etienne Pallier
    Uninstalling pip-7.1.2:
221 3 Etienne Pallier
      Successfully uninstalled pip-7.1.2
222 1 Etienne Pallier
Successfully installed pip-8.1.1
223 55 Etienne Pallier
224 55 Etienne Pallier
</pre>
225 18 Etienne Pallier
226 27 Etienne Pallier
h3. Install needed python packages (from within the virtual environment)
227 18 Etienne Pallier
228 66 Etienne Pallier
First, be sure that the virtual environment is activated:
229 66 Etienne Pallier
<pre>
230 66 Etienne Pallier
$ python -V
231 66 Etienne Pallier
Python 3.5.1
232 66 Etienne Pallier
</pre>
233 66 Etienne Pallier
234 62 Etienne Pallier
 * *Automatic Installation of all packages*
235 62 Etienne Pallier
<pre>
236 90 Etienne Pallier
$ pip install -r REQUIREMENTS.txt
237 62 Etienne Pallier
</pre>
238 62 Etienne Pallier
239 93 Etienne Pallier
 * *Or, manual installation of each package*
240 62 Etienne Pallier
241 62 Etienne Pallier
  * *Install Django* :
242 32 Etienne Pallier
<pre>
243 3 Etienne Pallier
$ pip install django
244 3 Etienne Pallier
Collecting django
245 3 Etienne Pallier
  Downloading Django-1.9.4-py2.py3-none-any.whl (6.6MB)
246 3 Etienne Pallier
Installing collected packages: django
247 3 Etienne Pallier
Successfully installed django-1.9.4
248 3 Etienne Pallier
249 3 Etienne Pallier
$ pip install django-admin-tools
250 3 Etienne Pallier
Collecting django-admin-tools
251 3 Etienne Pallier
  Downloading django_admin_tools-0.7.2-py2.py3-none-any.whl (289kB)
252 3 Etienne Pallier
Installing collected packages: django-admin-tools
253 3 Etienne Pallier
Successfully installed django-admin-tools-0.7.2
254 3 Etienne Pallier
255 21 Etienne Pallier
$ pip install django-debug-toolbar
256 21 Etienne Pallier
Collecting django-debug-toolbar
257 21 Etienne Pallier
  Downloading django_debug_toolbar-1.4-py2.py3-none-any.whl (212kB)
258 21 Etienne Pallier
Requirement already satisfied (use --upgrade to upgrade): Django>=1.7 in ./venv_py35_pyros/lib/python3.5/site-packages (from django-debug-toolbar)
259 21 Etienne Pallier
Collecting sqlparse (from django-debug-toolbar)
260 21 Etienne Pallier
  Downloading sqlparse-0.1.19.tar.gz (58kB)
261 21 Etienne Pallier
Building wheels for collected packages: sqlparse
262 21 Etienne Pallier
  Running setup.py bdist_wheel for sqlparse ... done
263 21 Etienne Pallier
  Stored in directory: /Users/epallier/Library/Caches/pip/wheels/7b/d4/72/6011bb100dd5fc213164e4bbee13d4e03261dd54ce6a5de6b8
264 21 Etienne Pallier
Successfully built sqlparse
265 21 Etienne Pallier
Installing collected packages: sqlparse, django-debug-toolbar
266 21 Etienne Pallier
Successfully installed django-debug-toolbar-1.4 sqlparse-0.1.19
267 21 Etienne Pallier
268 21 Etienne Pallier
$ pip install django-extensions
269 21 Etienne Pallier
Collecting django-extensions
270 21 Etienne Pallier
  Downloading django_extensions-1.6.1-py2.py3-none-any.whl (202kB)
271 21 Etienne Pallier
Collecting six>=1.2 (from django-extensions)
272 21 Etienne Pallier
  Downloading six-1.10.0-py2.py3-none-any.whl
273 21 Etienne Pallier
Installing collected packages: six, django-extensions
274 21 Etienne Pallier
Successfully installed django-extensions-1.6.1 six-1.10.0
275 21 Etienne Pallier
276 21 Etienne Pallier
$ pip install django-suit
277 21 Etienne Pallier
Collecting django-suit
278 21 Etienne Pallier
  Downloading django-suit-0.2.18.tar.gz (587kB)
279 21 Etienne Pallier
Building wheels for collected packages: django-suit
280 1 Etienne Pallier
  Running setup.py bdist_wheel for django-suit ... done
281 1 Etienne Pallier
  Stored in directory: /Users/epallier/Library/Caches/pip/wheels/12/8b/9a/e02ab0ad9229881638aa040d47d77c8f562999533811927d41
282 1 Etienne Pallier
Successfully built django-suit
283 1 Etienne Pallier
Installing collected packages: django-suit
284 1 Etienne Pallier
Successfully installed django-suit-0.2.18
285 21 Etienne Pallier
286 32 Etienne Pallier
</pre>
287 32 Etienne Pallier
288 63 Etienne Pallier
  * *Install the web application server gunicorn (will be used in production instead of the dev django web server)* :
289 32 Etienne Pallier
<pre>
290 25 Etienne Pallier
$ pip install gunicorn
291 25 Etienne Pallier
Collecting gunicorn
292 25 Etienne Pallier
  Downloading gunicorn-19.4.5-py2.py3-none-any.whl (112kB)
293 21 Etienne Pallier
Installing collected packages: gunicorn
294 26 Etienne Pallier
Successfully installed gunicorn-19.4.5
295 1 Etienne Pallier
</pre>
296 1 Etienne Pallier
297 63 Etienne Pallier
  * *Install the python mysql client*:
298 32 Etienne Pallier
<pre>
299 32 Etienne Pallier
$ pip install mysqlclient
300 72 Etienne Pallier
...
301 72 Etienne Pallier
</pre>
302 33 Etienne Pallier
303 73 Etienne Pallier
   * => Issue under Mac OS X:
304 72 Etienne Pallier
<pre>
305 3 Etienne Pallier
$ pip install mysqlclient
306 3 Etienne Pallier
Collecting mysqlclient
307 1 Etienne Pallier
  Downloading mysqlclient-1.3.7.tar.gz (79kB)
308 22 Etienne Pallier
Building wheels for collected packages: mysqlclient
309 3 Etienne Pallier
  Running setup.py bdist_wheel for mysqlclient ... error
310 3 Etienne Pallier
311 1 Etienne Pallier
  ----------------------------------------
312 1 Etienne Pallier
  Failed building wheel for mysqlclient
313 1 Etienne Pallier
  Running setup.py clean for mysqlclient
314 1 Etienne Pallier
Failed to build mysqlclient
315 1 Etienne Pallier
Installing collected packages: mysqlclient
316 1 Etienne Pallier
  Running setup.py install for mysqlclient ... done
317 1 Etienne Pallier
Successfully installed mysqlclient-1.3.7
318 1 Etienne Pallier
319 1 Etienne Pallier
BOUH !!!
320 1 Etienne Pallier
321 1 Etienne Pallier
$ pip install --upgrade wheel
322 1 Etienne Pallier
Collecting wheel
323 1 Etienne Pallier
  Downloading wheel-0.29.0-py2.py3-none-any.whl (66kB)
324 1 Etienne Pallier
Installing collected packages: wheel
325 1 Etienne Pallier
  Found existing installation: wheel 0.24.0
326 1 Etienne Pallier
    Uninstalling wheel-0.24.0:
327 1 Etienne Pallier
      Successfully uninstalled wheel-0.24.0
328 1 Etienne Pallier
Successfully installed wheel-0.29.0
329 1 Etienne Pallier
330 1 Etienne Pallier
$ pip uninstall mysqlclient
331 1 Etienne Pallier
332 1 Etienne Pallier
$ pip install mysqlclient
333 1 Etienne Pallier
Collecting mysqlclient
334 1 Etienne Pallier
  Using cached mysqlclient-1.3.7.tar.gz
335 1 Etienne Pallier
Building wheels for collected packages: mysqlclient
336 1 Etienne Pallier
  Running setup.py bdist_wheel for mysqlclient ... done
337 1 Etienne Pallier
  Stored in directory: /Users/epallier/Library/Caches/pip/wheels/9b/06/50/d11418c26cf8f2156b13d4363b5afde8e7e75ebb8540d0228d
338 1 Etienne Pallier
Successfully built mysqlclient
339 1 Etienne Pallier
Installing collected packages: mysqlclient
340 1 Etienne Pallier
Successfully installed mysqlclient-1.3.7
341 1 Etienne Pallier
342 1 Etienne Pallier
YES !!!
343 1 Etienne Pallier
344 1 Etienne Pallier
</pre>
345 1 Etienne Pallier
346 107 Paul Carensac
   * => Issues under Ubuntu:
347 107 Paul Carensac
<pre>
348 107 Paul Carensac
$ pip install mysqlclient
349 107 Paul Carensac
Collecting mysqlclient
350 107 Paul Carensac
  Downloading mysqlclient-1.3.7.tar.gz (79kB)
351 107 Paul Carensac
    100% |████████████████████████████████| 81kB 1.5MB/s
352 107 Paul Carensac
    Complete output from command python setup.py egg_info:
353 107 Paul Carensac
    /bin/sh: 1: mysql_config: not found
354 107 Paul Carensac
    Traceback (most recent call last):
355 107 Paul Carensac
      File "<string>", line 1, in <module>
356 107 Paul Carensac
    [...]
357 107 Paul Carensac
    ----------------------------------------
358 107 Paul Carensac
Command "python setup.py egg_info" failed with error code 1 in /tmp/pip-build-q6j4inuz/mysqlclient/
359 1 Etienne Pallier
360 108 Paul Carensac
BOUH !!!
361 108 Paul Carensac
362 107 Paul Carensac
$ sudo apt-get install libmysqlclient-dev
363 108 Paul Carensac
364 107 Paul Carensac
$ pip install mysqlclient
365 107 Paul Carensac
Collecting mysqlclient
366 107 Paul Carensac
  Using cached mysqlclient-1.3.7.tar.gz
367 107 Paul Carensac
Building wheels for collected packages: mysqlclient
368 107 Paul Carensac
  Running setup.py bdist_wheel for mysqlclient ... error
369 107 Paul Carensac
370 107 Paul Carensac
    _mysql.c:40:20: fatal error: Python.h: No such file or directory
371 107 Paul Carensac
     #include "Python.h"
372 107 Paul Carensac
                        ^
373 107 Paul Carensac
    compilation terminated.
374 107 Paul Carensac
    error: command 'x86_64-linux-gnu-gcc' failed with exit status 1
375 107 Paul Carensac
    
376 107 Paul Carensac
    ----------------------------------------
377 107 Paul Carensac
Command "/home/carens_p/pyros/venv_py35_pyros/bin/python3.5 -u -c "import setuptools, tokenize;__file__='/tmp/pip-build-k3klv92j/mysqlclient/setup.py';exec(compile(getattr(tokenize, 'open', open)(__file__).read().replace('\r\n', '\n'), __file__, 'exec'))" install --record /tmp/pip-gz242xxs-record/install-record.txt --single-version-externally-managed --compile --install-headers /home/carens_p/pyros/venv_py35_pyros/include/site/python3.5/mysqlclient" failed with error code 1 in /tmp/pip-build-k3klv92j/mysqlclient/
378 107 Paul Carensac
379 108 Paul Carensac
BOUH !!!
380 108 Paul Carensac
381 107 Paul Carensac
$ sudo apt-get install python3.5-dev
382 108 Paul Carensac
383 107 Paul Carensac
$ pip install mysqlclient
384 107 Paul Carensac
385 107 Paul Carensac
YES !!!
386 107 Paul Carensac
387 107 Paul Carensac
</pre>
388 12 Etienne Pallier
389 82 Etienne Pallier
 * *Set Requirements*
390 82 Etienne Pallier
391 82 Etienne Pallier
<pre>
392 82 Etienne Pallier
$ pip freeze > REQUIREMENTS.txt
393 82 Etienne Pallier
</pre>
394 12 Etienne Pallier
395 12 Etienne Pallier
---
396 28 Etienne Pallier
397 7 Etienne Pallier
h3. Create Django project pyros
398 38 Etienne Pallier
399 38 Etienne Pallier
<pre>
400 1 Etienne Pallier
401 83 Etienne Pallier
From inside the project:
402 83 Etienne Pallier
$ pwd
403 83 Etienne Pallier
.../PYROS/
404 83 Etienne Pallier
405 1 Etienne Pallier
$ django-admin startproject pyros
406 1 Etienne Pallier
407 149 Etienne Pallier
Rename the project folder "pyros/" as "src/"
408 149 Etienne Pallier
409 84 Etienne Pallier
$ mv pyros src
410 84 Etienne Pallier
411 1 Etienne Pallier
We have then this architecture:
412 86 Etienne Pallier
413 86 Etienne Pallier
PYROS
414 172 Etienne Pallier
├── REQUIREMENTS.txt
415 86 Etienne Pallier
├── private
416 137 Etienne Pallier
│   └── venv_py35_pyros
417 83 Etienne Pallier
├── public
418 83 Etienne Pallier
│   └── static
419 83 Etienne Pallier
├── src
420 83 Etienne Pallier
│   ├── manage.py
421 1 Etienne Pallier
│   ├── pyros
422 84 Etienne Pallier
│   │   ├── __init__.py
423 84 Etienne Pallier
│   │   ├── settings.py
424 84 Etienne Pallier
│   │   ├── urls.py
425 1 Etienne Pallier
│   │   └── wsgi.py
426 137 Etienne Pallier
427 142 Etienne Pallier
</pre>
428 3 Etienne Pallier
429 3 Etienne Pallier
430 1 Etienne Pallier
431 142 Etienne Pallier
432 142 Etienne Pallier
---
433 142 Etienne Pallier
434 142 Etienne Pallier
h3. Test the project
435 142 Etienne Pallier
436 142 Etienne Pallier
<pre>
437 142 Etienne Pallier
438 150 Etienne Pallier
$ cd src/
439 150 Etienne Pallier
440 1 Etienne Pallier
$ ./manage.py runserver
441 38 Etienne Pallier
(or gunicorn pyros.wsgi)
442 3 Etienne Pallier
==> http://localhost:8000
443 87 Etienne Pallier
...
444 87 Etienne Pallier
...
445 87 Etienne Pallier
Ctrl-c
446 87 Etienne Pallier
447 1 Etienne Pallier
</pre>
448 142 Etienne Pallier
449 142 Etienne Pallier
450 142 Etienne Pallier
451 142 Etienne Pallier
---
452 142 Etienne Pallier
453 142 Etienne Pallier
h3. The Web server
454 142 Etienne Pallier
455 142 Etienne Pallier
456 142 Etienne Pallier
Apache : gère tous les fichiers statiques (images, html…), et délègue les fichiers python au serveur django (par défaut)
457 142 Etienne Pallier
458 143 Etienne Pallier
Le fichier pyros/urls.py prend le relai pour tout ce qui est django
459 143 Etienne Pallier
460 142 Etienne Pallier
Le moteur web django sera soit du wsgi soit du unicorn
461 142 Etienne Pallier
462 142 Etienne Pallier
Par défaut, 1 seul worker, mais on peut en configurer plusieurs, l’idéal étant de faire "nb coeurs + 1" 
463 142 Etienne Pallier
(le worker maître qui fait le dispatching aux autres)
464 142 Etienne Pallier
465 142 Etienne Pallier
Frontend : Apache ou Ngininx
466 142 Etienne Pallier
467 142 Etienne Pallier
Backend : gunicorn (gère facilement des workers) ou uwsgi
468 142 Etienne Pallier
469 152 Etienne Pallier
<pre>
470 152 Etienne Pallier
$ gunicorn pyros.wsgi
471 1 Etienne Pallier
(à la place de manage runserver => A EVITER EN PROD)
472 152 Etienne Pallier
473 1 Etienne Pallier
Ou encore:
474 1 Etienne Pallier
475 152 Etienne Pallier
$ gunicorn --workers 5 library.wsgi
476 152 Etienne Pallier
</pre>
477 3 Etienne Pallier
478 42 Etienne Pallier
---
479 7 Etienne Pallier
480 3 Etienne Pallier
h3. Set Database engine as MySql
481 3 Etienne Pallier
482 41 Etienne Pallier
Edit src/pyros/settings.py
483 3 Etienne Pallier
484 3 Etienne Pallier
<pre>
485 3 Etienne Pallier
DATABASES = {
486 1 Etienne Pallier
    'default': {
487 3 Etienne Pallier
        'ENGINE': 'django.db.backends.mysql',
488 1 Etienne Pallier
        'NAME': 'pyros',
489 1 Etienne Pallier
        'USER': 'root',
490 1 Etienne Pallier
        'PASSWORD': ''
491 41 Etienne Pallier
    }
492 3 Etienne Pallier
}
493 7 Etienne Pallier
</pre>
494 3 Etienne Pallier
495 43 Etienne Pallier
---
496 7 Etienne Pallier
497 43 Etienne Pallier
h3. Import database into Django (with inspectdb)
498 3 Etienne Pallier
499 101 Etienne Pallier
From src/ :
500 94 Etienne Pallier
501 3 Etienne Pallier
<pre>
502 44 Etienne Pallier
$ ./manage.py inspectdb > models.py
503 94 Etienne Pallier
</pre>
504 44 Etienne Pallier
505 97 Etienne Pallier
Issue on Mac OS X:
506 3 Etienne Pallier
<pre>
507 3 Etienne Pallier
Traceback (most recent call last):
508 3 Etienne Pallier
  File "/Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/django/db/backends/mysql/base.py", line 25, in <module>
509 3 Etienne Pallier
    import MySQLdb as Database
510 3 Etienne Pallier
  File "/Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/MySQLdb/__init__.py", line 19, in <module>
511 3 Etienne Pallier
    import _mysql
512 3 Etienne Pallier
ImportError: dlopen(/Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so, 2): Library not loaded: libmysqlclient.18.dylib
513 3 Etienne Pallier
  Referenced from: /Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so
514 3 Etienne Pallier
  Reason: image not found
515 3 Etienne Pallier
516 3 Etienne Pallier
=> BOUH !!!
517 3 Etienne Pallier
518 3 Etienne Pallier
LA SOLUTION EST ICI : http://stackoverflow.com/questions/6383310/python-mysqldb-library-not-loaded-libmysqlclient-18-dylib
519 3 Etienne Pallier
520 124 Etienne Pallier
Il suffit de faire ceci:
521 3 Etienne Pallier
522 124 Etienne Pallier
$ sudo mkdir -p /usr/local/lib   
523 124 Etienne Pallier
$ sudo ln -s /Applications/XAMPP/xamppfiles/lib/libmysql* /usr/local/lib/
524 124 Etienne Pallier
525 124 Etienne Pallier
Mais on peut aussi faire ceci:
526 3 Etienne Pallier
527 123 Etienne Pallier
Okay, so the offending file is /Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so
528 123 Etienne Pallier
529 123 Etienne Pallier
Next, figure out where _mysql.so thinks it should find libmysqlclient.18.dylib:
530 123 Etienne Pallier
531 1 Etienne Pallier
$ otool -L /Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so
532 10 Etienne Pallier
/Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so:
533 3 Etienne Pallier
	libmysqlclient.18.dylib (compatibility version 18.0.0, current version 18.0.0)
534 3 Etienne Pallier
...
535 3 Etienne Pallier
536 123 Etienne Pallier
So, it's looking for libmysqlclient.18.dylib with no path information, let's fix that:
537 123 Etienne Pallier
538 1 Etienne Pallier
$ locate libmysqlclient.18.dylib
539 1 Etienne Pallier
/Applications/XAMPP/xamppfiles/lib/libmysqlclient.18.dylib
540 1 Etienne Pallier
/Library/SystemMigration/History/Migration-68137DFB-CB6A-4FBB-81E2-11BDB5D01E48/QuarantineRoot/usr/lib/libmysqlclient.18.dylib
541 1 Etienne Pallier
542 10 Etienne Pallier
$ sudo install_name_tool -change libmysqlclient.18.dylib /Applications/XAMPP/xamppfiles/lib/libmysqlclient.18.dylib /Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so
543 123 Etienne Pallier
544 123 Etienne Pallier
Now _mysql.so knows the full path to the library and everything works, regardless of environment variables.
545 10 Etienne Pallier
546 10 Etienne Pallier
$ otool -L /Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so
547 10 Etienne Pallier
/Users/epallier/Documents/_W_more/PROJECTS/GFT/SOFT/PYROS/pyros/venv_py35_pyros/lib/python3.5/site-packages/_mysql.cpython-35m-darwin.so:
548 3 Etienne Pallier
	/Applications/XAMPP/xamppfiles/lib/libmysqlclient.18.dylib (compatibility version 18.0.0, current version 18.0.0)
549 1 Etienne Pallier
...
550 1 Etienne Pallier
551 3 Etienne Pallier
$ ./manage.py inspectdb > models.py
552 44 Etienne Pallier
553 3 Etienne Pallier
=> YES !!!
554 3 Etienne Pallier
</pre>
555 7 Etienne Pallier
556 3 Etienne Pallier
557 48 Etienne Pallier
---
558 7 Etienne Pallier
559 3 Etienne Pallier
h3. Create a Django application pyrosapp
560 3 Etienne Pallier
561 102 Etienne Pallier
From src/ :
562 98 Etienne Pallier
563 102 Etienne Pallier
<pre>
564 1 Etienne Pallier
$ ./manage.py startapp pyrosapp
565 1 Etienne Pallier
</pre>
566 1 Etienne Pallier
567 103 Etienne Pallier
We obtain this structure:
568 1 Etienne Pallier
569 103 Etienne Pallier
<pre>
570 104 Etienne Pallier
571 103 Etienne Pallier
PYROS/
572 172 Etienne Pallier
├── REQUIREMENTS.txt
573 1 Etienne Pallier
├── private/
574 153 Etienne Pallier
│   └── venv_py35_pyros/
575 153 Etienne Pallier
├── public/
576 153 Etienne Pallier
│   └── static/
577 103 Etienne Pallier
├── src/
578 103 Etienne Pallier
│   ├── manage.py
579 153 Etienne Pallier
│   ├── pyros/
580 103 Etienne Pallier
│   │   ├── __init__.py
581 103 Etienne Pallier
│   │   ├── __pycache__
582 103 Etienne Pallier
│   │   ├── settings.py
583 103 Etienne Pallier
│   │   ├── urls.py
584 103 Etienne Pallier
│   │   └── wsgi.py
585 153 Etienne Pallier
│   └── pyrosapp/
586 1 Etienne Pallier
│       ├── __init__.py
587 103 Etienne Pallier
│       ├── admin.py
588 103 Etienne Pallier
│       ├── apps.py
589 103 Etienne Pallier
│       ├── migrations
590 103 Etienne Pallier
│       ├── models.py
591 103 Etienne Pallier
│       ├── tests.py
592 103 Etienne Pallier
│       └── views.py
593 153 Etienne Pallier
594 103 Etienne Pallier
595 7 Etienne Pallier
</pre>
596 1 Etienne Pallier
597 7 Etienne Pallier
---
598 7 Etienne Pallier
599 3 Etienne Pallier
h3. Replace the default pyrosapp models.py with the inspectdb generated one
600 3 Etienne Pallier
601 105 Etienne Pallier
From src/ :
602 1 Etienne Pallier
603 105 Etienne Pallier
<pre>
604 105 Etienne Pallier
$ mv models.py pyrosapp/
605 105 Etienne Pallier
</pre>
606 3 Etienne Pallier
607 139 Paul Carensac
Add pyrosapp to the project's applications :
608 7 Etienne Pallier
609 139 Paul Carensac
Edit src/pyros/settings.py
610 1 Etienne Pallier
611 139 Paul Carensac
<pre>
612 139 Paul Carensac
INSTALLED_APPS = [
613 139 Paul Carensac
    'django.contrib.admin',
614 139 Paul Carensac
    'django.contrib.auth',
615 139 Paul Carensac
    'django.contrib.contenttypes',
616 139 Paul Carensac
    'django.contrib.sessions',
617 139 Paul Carensac
    'django.contrib.messages',
618 139 Paul Carensac
    'django.contrib.staticfiles',
619 139 Paul Carensac
    'pyrosapp',
620 139 Paul Carensac
]
621 139 Paul Carensac
</pre>
622 9 Etienne Pallier
623 47 Etienne Pallier
---
624 3 Etienne Pallier
625 145 Paul Carensac
h3. Fix and improve the pyrosapp models.py file (generated by inspectdb)
626 1 Etienne Pallier
627 145 Paul Carensac
Once models.py file generated, we need to delete the database and create an empty one :
628 145 Paul Carensac
 
629 145 Paul Carensac
<pre>
630 145 Paul Carensac
$ mysql -u root [-p (if password needed)]
631 106 Etienne Pallier
632 145 Paul Carensac
mysql> DROP DATABSE pyros
633 145 Paul Carensac
mysql> CREATE SCHEMA IF NOT EXISTS 'pyros' DEFAULT CHARACTER SET utf8;
634 1 Etienne Pallier
635 145 Paul Carensac
</pre>
636 1 Etienne Pallier
637 145 Paul Carensac
Then edit pyrosapp/models.py :
638 1 Etienne Pallier
639 145 Paul Carensac
 * Change 'managed = False' to 'managed = True' for every model
640 145 Paul Carensac
641 145 Paul Carensac
 * Change classes names to CamelCase (do not change the 'db_table = ...' lines). *Be careful* : it is needed to change all occurences :
642 145 Paul Carensac
643 145 Paul Carensac
    * NrtAlanysis
644 145 Paul Carensac
    * ScheduleHistory
645 145 Paul Carensac
    * ScientificProgram
646 145 Paul Carensac
    * SequenceType
647 145 Paul Carensac
    * SiteWatch
648 145 Paul Carensac
    * SiteWatchHistory
649 145 Paul Carensac
    * StrategyObs
650 154 Paul Carensac
    * UserLevel
651 145 Paul Carensac
    * WeatherWatch
652 145 Paul Carensac
    * WeatherWatchHistory
653 145 Paul Carensac
654 145 Paul Carensac
 * Change the deleting mode from 'models.DO_NOTHING' to 'models.CASCADE' for the following foreign keys :
655 145 Paul Carensac
656 145 Paul Carensac
    * Image.plan
657 145 Paul Carensac
    * Plan.album
658 145 Paul Carensac
    * Album.sequence
659 145 Paul Carensac
    * Sequence.request
660 145 Paul Carensac
661 145 Paul Carensac
 * Change the 'ForeignKey' liaisons to 'OneToOneField' liaisons (just replace ForeignKey by OneToOneField), and change deleting mode to 'models.CASCADE' for the following foreign keys :
662 145 Paul Carensac
663 145 Paul Carensac
    * Alert.request
664 145 Paul Carensac
    * Detector.device
665 145 Paul Carensac
    * Filter.device
666 145 Paul Carensac
    * Telescope.device
667 145 Paul Carensac
668 145 Paul Carensac
 * We need to redefine many to many relationships for the following classes :
669 145 Paul Carensac
670 145 Paul Carensac
    * User - ScientificProgram :
671 145 Paul Carensac
672 145 Paul Carensac
        * add 'users = models.ManyToManyField('User')' in ScientificProgram class
673 145 Paul Carensac
        * delete UserHasScientificProgram class
674 145 Paul Carensac
675 145 Paul Carensac
    * Sequence - ScheduleHistory
676 145 Paul Carensac
677 145 Paul Carensac
        * add 'sequences = models.ManyToManyField('Sequence')' in ScheduleHistory class
678 145 Paul Carensac
        * delete ScheduleHasSequences class
679 145 Paul Carensac
680 145 Paul Carensac
 * Finally apply modifications to the database :
681 145 Paul Carensac
682 145 Paul Carensac
<pre>
683 145 Paul Carensac
$ pwd
684 145 Paul Carensac
.../PYROS/src
685 145 Paul Carensac
$ python manage.py makemigrations pyrosapp
686 145 Paul Carensac
$ python manage.py migrate
687 145 Paul Carensac
</pre>
688 109 Etienne Pallier
689 109 Etienne Pallier
---
690 111 Etienne Pallier
691 156 Paul Carensac
h2. %{margin-left:0px; font-weight:bold; font-size:25px;  display:block; color:red;}IV - CONFIGURATION of the Django Back Office (administration interface)%
692 155 Paul Carensac
693 155 Paul Carensac
---
694 155 Paul Carensac
695 155 Paul Carensac
h3. Back Office setup
696 155 Paul Carensac
697 155 Paul Carensac
 * Prerequisites in src/pyros/settings.py :
698 155 Paul Carensac
699 155 Paul Carensac
    * INSTALLED_APPS must (at least) contain :
700 155 Paul Carensac
701 155 Paul Carensac
        * django.contrib.admin
702 155 Paul Carensac
        * django.contrib.auth
703 155 Paul Carensac
        * django.contrib.contenttypes
704 155 Paul Carensac
        * django.contrib.sessions
705 155 Paul Carensac
706 155 Paul Carensac
    * MIDDLEWARES must (at least) contain :
707 155 Paul Carensac
708 155 Paul Carensac
        * django.contrib.sessions.middleware.SessionMiddleware
709 155 Paul Carensac
        * django.middleware.common.CommonMiddleware
710 155 Paul Carensac
        * django.contrib.auth.middleware.AuthenticationMiddleware
711 155 Paul Carensac
712 155 Paul Carensac
 * At least one 'python manage.py migrate' must have been executed
713 155 Paul Carensac
714 155 Paul Carensac
 * Create a superuser for the administration :
715 155 Paul Carensac
716 155 Paul Carensac
<pre>
717 155 Paul Carensac
$ python manage.py createsuperuser
718 155 Paul Carensac
</pre>
719 155 Paul Carensac
720 156 Paul Carensac
* For each app of the project, fill the admin.py file :
721 155 Paul Carensac
722 155 Paul Carensac
<pre>
723 155 Paul Carensac
from django.contrib import admin
724 155 Paul Carensac
from app.models import Model1, Model2
725 155 Paul Carensac
726 155 Paul Carensac
admin.site.register(Model1)
727 155 Paul Carensac
admin.site.register(Model2)
728 155 Paul Carensac
</pre>
729 155 Paul Carensac
730 155 Paul Carensac
*Reminder* : each application must be registered in the settings.py INSTALLED_APPS variable.
731 157 Paul Carensac
732 168 Paul Carensac
* For each model in models.py, add a '__str__()' method in order to identify the object on the back office. Example :
733 157 Paul Carensac
734 157 Paul Carensac
<pre>
735 157 Paul Carensac
class UserLevel(models.Model):
736 157 Paul Carensac
    name = models.CharField(max_length=45, blank=True, null=True)
737 157 Paul Carensac
    desc = models.TextField(blank=True, null=True)
738 157 Paul Carensac
    priority = models.IntegerField(blank=True, null=True)
739 157 Paul Carensac
    quota = models.FloatField(blank=True, null=True)
740 157 Paul Carensac
741 157 Paul Carensac
    class Meta:
742 157 Paul Carensac
        managed = True
743 157 Paul Carensac
        db_table = 'userlevel'
744 157 Paul Carensac
745 157 Paul Carensac
    def __str__(self):
746 163 Paul Carensac
        return (str(self.name))
747 157 Paul Carensac
</pre>
748 157 Paul Carensac
749 159 Paul Carensac
*Naming convention* : Use self.name when possible, the creation time/date otherwise. Example :
750 159 Paul Carensac
751 159 Paul Carensac
<pre>
752 159 Paul Carensac
class SiteWatch(models.Model):
753 159 Paul Carensac
    updated = models.DateTimeField(blank=True, null=True)
754 159 Paul Carensac
    lights = models.CharField(max_length=45, blank=True, null=True)
755 159 Paul Carensac
    dome = models.CharField(max_length=45, blank=True, null=True)
756 159 Paul Carensac
    doors = models.CharField(max_length=45, blank=True, null=True)
757 159 Paul Carensac
    temperature = models.FloatField(blank=True, null=True)
758 159 Paul Carensac
759 159 Paul Carensac
    class Meta:
760 159 Paul Carensac
        managed = True
761 159 Paul Carensac
        db_table = 'sitewatch'
762 159 Paul Carensac
763 159 Paul Carensac
    def __str__(self):
764 160 Paul Carensac
        return (str(self.updated))
765 159 Paul Carensac
</pre>
766 173 Etienne Pallier
767 176 Etienne Pallier
h2. %{margin-left:0px; font-weight:bold; font-size:25px;  display:block; color:red;}V - INSTALLATION FROM THE BEGINNING (for dev only)%
768 173 Etienne Pallier
769 1 Etienne Pallier
770 176 Etienne Pallier
h3. How the git repository was created
771 176 Etienne Pallier
772 176 Etienne Pallier
*Git global setup:*
773 176 Etienne Pallier
774 176 Etienne Pallier
<pre>
775 176 Etienne Pallier
$ git config --global user.name "Etienne Pallier"
776 176 Etienne Pallier
$ git config --global user.email "etienne.pallier@irap.omp.eu"
777 176 Etienne Pallier
778 176 Etienne Pallier
$ cat ~/.gitconfig 
779 176 Etienne Pallier
[user]
780 176 Etienne Pallier
	name = Etienne Pallier
781 176 Etienne Pallier
	email = epallier@irap.omp.eu
782 176 Etienne Pallier
[http]
783 176 Etienne Pallier
	sslVerify = false
784 176 Etienne Pallier
</pre>
785 176 Etienne Pallier
786 176 Etienne Pallier
787 176 Etienne Pallier
*Create a new repository:*
788 176 Etienne Pallier
789 176 Etienne Pallier
<pre>
790 176 Etienne Pallier
$ cd PYROS/
791 176 Etienne Pallier
792 176 Etienne Pallier
Define files and folders to be ignored:
793 176 Etienne Pallier
$ vi .gitignore
794 176 Etienne Pallier
.DS_Store
795 176 Etienne Pallier
private
796 176 Etienne Pallier
__pycache__
797 176 Etienne Pallier
798 176 Etienne Pallier
$ touch README.md
799 176 Etienne Pallier
800 176 Etienne Pallier
$ git add README.md
801 176 Etienne Pallier
802 176 Etienne Pallier
$ git commit -m "first commit"
803 176 Etienne Pallier
804 176 Etienne Pallier
$ git remote add origin https://gitlab.irap.omp.eu/epallier/pyros.git
805 176 Etienne Pallier
806 176 Etienne Pallier
$ git push -u origin master
807 176 Etienne Pallier
808 176 Etienne Pallier
$ git add .
809 176 Etienne Pallier
810 176 Etienne Pallier
( if you want to be sure to add ALL files: 
811 176 Etienne Pallier
$ git add -A
812 176 Etienne Pallier
)
813 176 Etienne Pallier
814 176 Etienne Pallier
( if you wanted to remove added files, just type:
815 176 Etienne Pallier
$ git reset HEAD
816 176 Etienne Pallier
)
817 176 Etienne Pallier
818 176 Etienne Pallier
$ git commit -m "first full project commit"
819 176 Etienne Pallier
820 176 Etienne Pallier
$ git push -u origin master
821 176 Etienne Pallier
Counting objects: 43, done.
822 176 Etienne Pallier
Delta compression using up to 4 threads.
823 176 Etienne Pallier
Compressing objects: 100% (41/41), done.
824 176 Etienne Pallier
Writing objects: 100% (43/43), 575.13 KiB ö 0 bytes/s, done.
825 176 Etienne Pallier
Total 43 (delta 2), reused 0 (delta 0)
826 176 Etienne Pallier
To https://gitlab.irap.omp.eu/epallier/pyros.git
827 176 Etienne Pallier
   9c7128c..64501c9  master -> master
828 176 Etienne Pallier
Branch master set up to track remote branch master from origin.
829 176 Etienne Pallier
830 176 Etienne Pallier
$ git status
831 176 Etienne Pallier
On branch master
832 176 Etienne Pallier
Your branch is up-to-date with 'origin/master'.
833 176 Etienne Pallier
nothing to commit, working directory clean
834 176 Etienne Pallier
835 182 Etienne Pallier
</pre>
836 176 Etienne Pallier
837 182 Etienne Pallier
838 182 Etienne Pallier
h3. Create database
839 182 Etienne Pallier
840 182 Etienne Pallier
841 182 Etienne Pallier
 * Linux and Mac OS X:
842 182 Etienne Pallier
<pre>
843 182 Etienne Pallier
One liner:
844 182 Etienne Pallier
$ mysql -u root < pyros_create.sql
845 182 Etienne Pallier
846 182 Etienne Pallier
Or :
847 182 Etienne Pallier
$ mysql -u root
848 182 Etienne Pallier
mysql> create database pyros;
849 182 Etienne Pallier
mysql> use pyros;
850 182 Etienne Pallier
mysql> source pyros_create.sql;
851 182 Etienne Pallier
852 182 Etienne Pallier
(
853 182 Etienne Pallier
TODO:
854 182 Etienne Pallier
mysql> grant all on pyros.* to pyros@localhost identified by ‘pyros’;)
855 182 Etienne Pallier
mysql> flush privileges;
856 182 Etienne Pallier
)
857 176 Etienne Pallier
</pre>
858 182 Etienne Pallier
859 182 Etienne Pallier
860 182 Etienne Pallier
 * Windows:
861 182 Etienne Pallier
<pre>
862 182 Etienne Pallier
TODO: Use phpmyadmin ?
863 182 Etienne Pallier
</pre>
864 182 Etienne Pallier
865 182 Etienne Pallier
866 182 Etienne Pallier
---