rb@217
|
1 |
# -*- coding: utf-8 -*-
|
rb@846
|
2 |
#!/usr/bin/env python
|
rb@280
|
3 |
import logging
|
rb@217
|
4 |
|
rb@186
|
5 |
import os
|
rb@186
|
6 |
import codecs
|
rb@275
|
7 |
import vobject
|
rb@275
|
8 |
import StringIO
|
rb@275
|
9 |
|
rb@275
|
10 |
import qrcode
|
rb@186
|
11 |
|
rb@186
|
12 |
from docutils.core import publish_parts
|
rb@275
|
13 |
from flask import Flask
|
peter@1
|
14 |
from flask import render_template
|
peter@1
|
15 |
from flask import request
|
rb@848
|
16 |
from flask_babel import gettext as _
|
rb@848
|
17 |
from flask_babel import Babel
|
rb@95
|
18 |
from config import LANGUAGES
|
rb@161
|
19 |
|
rb@275
|
20 |
import base64
|
rb@275
|
21 |
|
rb@122
|
22 |
LANGUAGE_SELECTED = "de"
|
icg179@496
|
23 |
# ToDo after engelish is implemented set LANGUAGE_SELECTED = None
|
rb@95
|
24 |
|
rb@246
|
25 |
# We need the path of this file to find templates to translate
|
rb@246
|
26 |
ESKP_PATH = os.path.dirname(os.path.abspath(__file__))
|
icg179@496
|
27 |
logging.basicConfig(filename=os.path.join(ESKP_PATH, 'eskp-app.log'), level=logging.DEBUG)
|
Tim@498
|
28 |
FILES= os.listdir(os.path.join(ESKP_PATH, 'static/images/uvmap'))
|
rb@246
|
29 |
|
peter@1
|
30 |
app = Flask(__name__)
|
peter@1
|
31 |
babel = Babel(app)
|
peter@1
|
32 |
|
peter@145
|
33 |
app.config['BABEL_DEFAULT_LOCALE'] = 'de'
|
peter@145
|
34 |
|
rb@275
|
35 |
|
rb@275
|
36 |
def get_vcard(filename):
|
rb@281
|
37 |
filename = os.path.join(ESKP_PATH, filename)
|
rb@275
|
38 |
with codecs.open(filename, 'r', 'utf-8') as f:
|
icg179@496
|
39 |
vcard = f.read()
|
rb@275
|
40 |
return vobject.readOne(vcard)
|
rb@275
|
41 |
|
rb@275
|
42 |
|
rb@214
|
43 |
def get_content(filename, overrides=None):
|
rb@162
|
44 |
content = u""
|
rb@246
|
45 |
filename = os.path.join(ESKP_PATH, filename)
|
rb@162
|
46 |
if os.path.isfile(filename):
|
rb@162
|
47 |
with codecs.open(filename, 'r', 'utf-8') as f:
|
rb@162
|
48 |
rst_data = f.read()
|
rb@214
|
49 |
content = publish_parts(rst_data, writer_name='html', settings_overrides=overrides)['html_body']
|
rb@162
|
50 |
return content
|
rb@162
|
51 |
|
icg179@496
|
52 |
|
icg179@496
|
53 |
def get_newest_date():
|
Tim@498
|
54 |
getdates = get_valid_dates(FILES)
|
icg179@496
|
55 |
newest_date = getdates[-1]
|
icg179@496
|
56 |
|
icg179@496
|
57 |
return newest_date
|
icg179@496
|
58 |
|
icg179@496
|
59 |
|
peter@142
|
60 |
def get_topmenue():
|
icg179@496
|
61 |
newest_date = get_newest_date()
|
rb@238
|
62 |
menue = [
|
icg179@496
|
63 |
('/ozoneloss', _(u'Ozoneloss'),
|
icg179@496
|
64 |
(('/ozoneloss', _(u'overview')),
|
j@733
|
65 |
('/ozoneloss/clams/2018', _(u'calculations')),
|
j@733
|
66 |
('/ozoneloss/vpsc/2018', _(u'estimations')),
|
icg179@496
|
67 |
('/ozoneloss/uvi', _(u'uv increase')),
|
icg179@496
|
68 |
('/ozoneloss/uvmap/' + newest_date, _(u'uv index map')))),
|
icg179@496
|
69 |
('/eskp', _(u'ESKP'), (None, None)),
|
icg179@496
|
70 |
('/iek-7', _(u'IEK-7'), (None, None))
|
icg179@496
|
71 |
]
|
peter@145
|
72 |
return menue
|
peter@142
|
73 |
|
icg179@496
|
74 |
|
peter@142
|
75 |
app.jinja_env.globals.update(get_topmenue=get_topmenue)
|
peter@142
|
76 |
|
icg179@496
|
77 |
|
j@250
|
78 |
def get_o3lossclams_dates():
|
rb@302
|
79 |
menue = [
|
j@733
|
80 |
('/ozoneloss/clams/2018', _(u'2018')),
|
j@584
|
81 |
('/ozoneloss/clams/2017', _(u'2017')),
|
j@441
|
82 |
('/ozoneloss/clams/2016', _(u'2016')),
|
j@380
|
83 |
('/ozoneloss/clams/2015', _(u'2015')),
|
j@380
|
84 |
('/ozoneloss/clams/2012', _(u'2012')),
|
j@380
|
85 |
('/ozoneloss/clams/2011', _(u'2011')),
|
j@380
|
86 |
('/ozoneloss/clams/2010', _(u'2010')),
|
icg179@496
|
87 |
]
|
j@249
|
88 |
return menue
|
j@249
|
89 |
|
icg179@496
|
90 |
|
icg179@496
|
91 |
def get_valid_dates(files):
|
icg179@496
|
92 |
dates = []
|
icg179@496
|
93 |
|
icg179@496
|
94 |
for file in files:
|
icg179@496
|
95 |
if file.endswith('.png') and file.find('uvi') >= 0:
|
icg179@496
|
96 |
date = file[-12:-6]
|
icg179@496
|
97 |
dates.append(date)
|
icg179@496
|
98 |
dates.sort()
|
icg179@496
|
99 |
for date in dates:
|
icg179@496
|
100 |
i = 0
|
icg179@496
|
101 |
for param in ['uvi', 'o3col', 'do3col']:
|
icg179@496
|
102 |
testfile = 'clams_' + param + '_' + date + '12.png'
|
icg179@496
|
103 |
if files.count(testfile) > 0:
|
icg179@496
|
104 |
i = i + 1
|
icg179@496
|
105 |
if i <> 3:
|
icg179@496
|
106 |
dates.remove(date)
|
icg179@496
|
107 |
return dates
|
icg179@496
|
108 |
|
icg179@496
|
109 |
|
icg179@496
|
110 |
def get_o3lossuvmap_dates(date_show):
|
Tim@498
|
111 |
dates = get_valid_dates(FILES)
|
icg179@496
|
112 |
ndates = len(dates)
|
icg179@496
|
113 |
ind = dates.index(date_show)
|
j@503
|
114 |
navitexts = []
|
icg179@496
|
115 |
if ind ==0 :
|
icg179@496
|
116 |
chosendates = [dates[ind+1]]
|
j@503
|
117 |
navitexts.append('next ->')
|
Tim@501
|
118 |
|
icg179@496
|
119 |
elif ind >= ndates - 1:
|
icg179@496
|
120 |
chosendates= [dates[ind-1]]
|
j@503
|
121 |
navitexts.append('<- prev')
|
icg179@496
|
122 |
else:
|
icg179@496
|
123 |
chosendates = [dates[ind-1], dates[ind+1]]
|
j@503
|
124 |
navitexts.append('<- prev')
|
j@503
|
125 |
navitexts.append('next ->')
|
icg179@496
|
126 |
menue = []
|
Tim@501
|
127 |
|
Tim@501
|
128 |
for i in range(len(chosendates)):
|
Tim@501
|
129 |
date = chosendates[i]
|
j@503
|
130 |
navitext = navitexts[i]
|
j@503
|
131 |
menue.append(('/ozoneloss/uvmap/' + date, _(navitext)))
|
icg179@496
|
132 |
return menue
|
icg179@496
|
133 |
|
icg179@496
|
134 |
|
j@249
|
135 |
def get_vpsc_dates():
|
rb@302
|
136 |
menue = [
|
j@733
|
137 |
('/ozoneloss/vpsc/2018', _(u'2018')),
|
j@584
|
138 |
('/ozoneloss/vpsc/2017', _(u'2017')),
|
j@441
|
139 |
('/ozoneloss/vpsc/2016', _(u'2016')),
|
j@380
|
140 |
('/ozoneloss/vpsc/2015', _(u'2015')),
|
j@380
|
141 |
('/ozoneloss/vpsc/2014', _(u'2014')),
|
j@380
|
142 |
('/ozoneloss/vpsc/2013', _(u'2013')),
|
j@380
|
143 |
('/ozoneloss/vpsc/2012', _(u'2012')),
|
j@380
|
144 |
('/ozoneloss/vpsc/2011', _(u'2011')),
|
j@380
|
145 |
('/ozoneloss/vpsc/2010', _(u'2010')),
|
icg179@496
|
146 |
]
|
rb@241
|
147 |
return menue
|
rb@241
|
148 |
|
icg179@496
|
149 |
|
j@250
|
150 |
app.jinja_env.globals.update(get_o3lossclams_dates=get_o3lossclams_dates)
|
j@249
|
151 |
app.jinja_env.globals.update(get_vpsc_dates=get_vpsc_dates)
|
icg179@496
|
152 |
app.jinja_env.globals.update(get_o3lossuvmap_dates=get_o3lossuvmap_dates)
|
peter@142
|
153 |
|
rb@267
|
154 |
|
rb@267
|
155 |
def modal_info(template, act, title, filename):
|
rb@264
|
156 |
content = get_content(filename)
|
rb@267
|
157 |
html = render_template(template, act=act, title=title, content=content, exit=_(u"Close"))
|
rb@264
|
158 |
return html
|
rb@264
|
159 |
|
rb@264
|
160 |
|
peter@1
|
161 |
@babel.localeselector
|
peter@1
|
162 |
def get_locale():
|
peter@109
|
163 |
"""ToDo: if translation is completed, switch to en """
|
rb@118
|
164 |
return LANGUAGE_SELECTED or request.accept_languages.best_match(LANGUAGES.keys()) or 'de'
|
peter@1
|
165 |
|
rb@100
|
166 |
|
peter@1
|
167 |
@app.route("/")
|
rb@100
|
168 |
@app.route("/index")
|
peter@1
|
169 |
def index():
|
rb@213
|
170 |
return render_template("/index.html",
|
rb@233
|
171 |
eskp_info=_(u'About ESKP'),
|
rb@233
|
172 |
)
|
peter@1
|
173 |
|
rb@241
|
174 |
|
rb@257
|
175 |
@app.route('/ozoneloss/clams/<year>')
|
rb@257
|
176 |
def ozoneloss_clams_year(year):
|
j@249
|
177 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_clams.rst")
|
rb@241
|
178 |
content = get_content(filename)
|
rb@257
|
179 |
return render_template("/ozoneloss_clams.html", act="ozoneloss/clams/%s" % year, content=content, year=year)
|
rb@241
|
180 |
|
rb@241
|
181 |
|
icg179@496
|
182 |
@app.route('/ozoneloss/uvmap/<date>')
|
icg179@496
|
183 |
def ozoneloss_uvmap_date(date):
|
icg179@496
|
184 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvmap.rst")
|
icg179@496
|
185 |
content = get_content(filename)
|
icg179@496
|
186 |
return render_template("/ozoneloss_uvmap.html", act="ozoneloss/uvmap/%s" % date, content=content, date=date)
|
icg179@496
|
187 |
|
icg179@496
|
188 |
|
rb@257
|
189 |
@app.route('/ozoneloss/vpsc/<year>')
|
rb@257
|
190 |
def ozoneloss_vspc_year(year):
|
j@249
|
191 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_vpsc.rst")
|
j@249
|
192 |
content = get_content(filename)
|
j@250
|
193 |
filename = os.path.join("templates", get_locale(), "rst", "explanation_vpsc.rst")
|
j@250
|
194 |
explanation = get_content(filename)
|
j@250
|
195 |
|
rb@257
|
196 |
return render_template("/ozoneloss_vpsc.html", act="ozoneloss/vpsc/%s" % year, content=content,
|
rb@257
|
197 |
content_explanation=explanation, year=year)
|
rb@241
|
198 |
|
j@317
|
199 |
|
rb@118
|
200 |
@app.route('/de')
|
rb@118
|
201 |
def de():
|
rb@118
|
202 |
global LANGUAGE_SELECTED
|
rb@118
|
203 |
LANGUAGE_SELECTED = "de"
|
peter@160
|
204 |
return render_template("/index.html",
|
rb@233
|
205 |
eskp_info=_(u'About ESKP'),
|
rb@233
|
206 |
)
|
rb@118
|
207 |
|
icg179@496
|
208 |
|
rb@118
|
209 |
@app.route('/en')
|
rb@118
|
210 |
def en():
|
rb@118
|
211 |
global LANGUAGE_SELECTED
|
rb@118
|
212 |
LANGUAGE_SELECTED = "en"
|
peter@160
|
213 |
return render_template("/index.html",
|
rb@233
|
214 |
eskp_info=_(u'About ESKP'),
|
rb@233
|
215 |
)
|
rb@118
|
216 |
|
icg179@496
|
217 |
|
rb@233
|
218 |
@app.route("/eskp")
|
rb@233
|
219 |
def eskp():
|
rb@233
|
220 |
filename = os.path.join("templates", get_locale(), "rst", "eskp.rst")
|
rb@162
|
221 |
content = get_content(filename)
|
j@272
|
222 |
filename = os.path.join("templates", get_locale(), "rst", "eskp_title.rst")
|
j@272
|
223 |
headline = get_content(filename)
|
j@272
|
224 |
return render_template("/eskp.html", act="eskp", content=content, headline=headline)
|
j@252
|
225 |
|
peter@1
|
226 |
|
rb@275
|
227 |
def qr_image_data(card):
|
icg179@496
|
228 |
buf = StringIO.StringIO()
|
rb@275
|
229 |
qr = qrcode.QRCode(
|
rb@275
|
230 |
version=1,
|
rb@275
|
231 |
error_correction=qrcode.constants.ERROR_CORRECT_L,
|
rb@275
|
232 |
box_size=2,
|
rb@275
|
233 |
border=2,
|
icg179@496
|
234 |
)
|
rb@275
|
235 |
qr.add_data(card.serialize())
|
rb@275
|
236 |
qr.make(fit=True)
|
rb@275
|
237 |
img = qr.make_image()
|
rb@275
|
238 |
img.save(buf)
|
rb@275
|
239 |
image = buf.getvalue()
|
rb@275
|
240 |
return base64.b64encode(image)
|
rb@275
|
241 |
|
rb@275
|
242 |
|
rb@233
|
243 |
@app.route("/ozoneloss")
|
rb@257
|
244 |
def ozoneloss():
|
rb@233
|
245 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss.rst")
|
rb@165
|
246 |
content = get_content(filename)
|
rb@275
|
247 |
|
rb@275
|
248 |
vcard_file = os.path.join("vcards", "jug.vcf")
|
rb@275
|
249 |
author = u""
|
icg179@496
|
250 |
|
rb@275
|
251 |
try:
|
rb@275
|
252 |
card = get_vcard(vcard_file)
|
rb@275
|
253 |
except IOError:
|
rb@275
|
254 |
card = None
|
rb@275
|
255 |
if card is not None:
|
rb@275
|
256 |
qr_image = qr_image_data(card)
|
rb@275
|
257 |
author = render_template("/author_info.html", act="author", title=_(u"Ozoneloss"),
|
icg179@496
|
258 |
card=card, image=qr_image, contact=_(u"Contact"), exit=_(u"Close"))
|
rb@275
|
259 |
|
rb@267
|
260 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_publications.rst")
|
rb@267
|
261 |
publications = modal_info("/publications_info.html", "publications", _(u"Ozoneloss"), filename)
|
rb@264
|
262 |
return render_template("/ozoneloss.html", act="ozoneloss", content=content,
|
icg179@496
|
263 |
author=author, card=card, publications=publications)
|
icg179@496
|
264 |
|
j@247
|
265 |
|
j@249
|
266 |
@app.route("/ozoneloss/clams")
|
rb@257
|
267 |
def ozoneloss_clams():
|
j@249
|
268 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_clams.rst")
|
j@247
|
269 |
content = get_content(filename)
|
j@249
|
270 |
return render_template("/ozoneloss_clams.html", act="ozoneloss/clams", content=content)
|
j@249
|
271 |
|
icg179@496
|
272 |
|
icg179@496
|
273 |
@app.route("/ozoneloss/uvmap")
|
icg179@496
|
274 |
def ozoneloss_uvmap():
|
icg179@496
|
275 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvmap.rst")
|
icg179@496
|
276 |
content = get_content(filename)
|
icg179@496
|
277 |
return render_template("/ozoneloss_uvmap.html", act="ozoneloss/uvmap", content=content)
|
icg179@496
|
278 |
|
icg179@496
|
279 |
|
j@249
|
280 |
@app.route("/ozoneloss/vpsc")
|
rb@257
|
281 |
def ozoneloss_vspc():
|
j@249
|
282 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_vpsc.rst")
|
j@249
|
283 |
content = get_content(filename)
|
j@249
|
284 |
return render_template("/ozoneloss_vpsc.html", act="ozoneloss/vpsc", content=content)
|
peter@1
|
285 |
|
j@380
|
286 |
|
j@380
|
287 |
@app.route("/ozoneloss/uvi", methods=['GET'])
|
j@317
|
288 |
def ozoneloss_uvi():
|
j@380
|
289 |
# XXX check 'POST' does not work
|
j@317
|
290 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvi.rst")
|
j@317
|
291 |
content = get_content(filename)
|
j@380
|
292 |
|
j@380
|
293 |
lat = 50.
|
j@380
|
294 |
o3offset = 50.
|
j@380
|
295 |
figname = "uvincr_lat%0.3i_do3%0.3i.svg" % (lat, o3offset)
|
j@380
|
296 |
|
j@380
|
297 |
return render_template('ozoneloss_uvi.html', act="ozoneloss/uvi", content=content, figname=figname,
|
j@380
|
298 |
alt=_(u"UV increase at {{lat}} degrees N for {{o3offset}} DU ozone depletion"))
|
j@380
|
299 |
|
icg179@496
|
300 |
|
j@380
|
301 |
@app.route("/ozoneloss/uvi_graph", methods=['POST'])
|
j@380
|
302 |
def ozoneloss_uvi_graph():
|
j@380
|
303 |
filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvi.rst")
|
j@380
|
304 |
content = get_content(filename)
|
j@380
|
305 |
|
j@380
|
306 |
latstr = request.form['Gradzahl']
|
j@380
|
307 |
o3offsetstr = request.form['Dobson-Unit']
|
j@380
|
308 |
|
j@380
|
309 |
latstr2 = latstr.replace(u'\xb0', '')
|
j@380
|
310 |
|
j@380
|
311 |
if latstr2.endswith(u'N'):
|
j@380
|
312 |
latstr2 = latstr2.replace(u'N', '')
|
j@380
|
313 |
lat = float(latstr2)
|
j@380
|
314 |
if latstr2.endswith(u'S'):
|
j@380
|
315 |
latstr2 = latstr2.replace(u'S', '')
|
j@380
|
316 |
lat = -float(latstr2)
|
j@380
|
317 |
|
j@380
|
318 |
figname = "uvincr_lat%0.3i_do3%0.3i.svg" % (lat, float(o3offsetstr))
|
j@380
|
319 |
|
j@380
|
320 |
return render_template('graph.html', act="ozoneloss_uvi_graph", content=content, figname=figname,
|
j@380
|
321 |
o3offsetstr=o3offsetstr, latstr=latstr)
|
j@380
|
322 |
|
j@380
|
323 |
|
rb@233
|
324 |
@app.route("/iek-7")
|
rb@257
|
325 |
def institute():
|
rb@233
|
326 |
filename = os.path.join("templates", get_locale(), "rst", "iek-7.rst")
|
rb@167
|
327 |
content = get_content(filename)
|
rb@277
|
328 |
vcard_file = os.path.join("vcards", "sas.vcf")
|
rb@277
|
329 |
author = u""
|
rb@277
|
330 |
try:
|
rb@277
|
331 |
card = get_vcard(vcard_file)
|
rb@277
|
332 |
except IOError:
|
rb@277
|
333 |
card = None
|
rb@277
|
334 |
if card is not None:
|
rb@277
|
335 |
qr_image = qr_image_data(card)
|
rb@277
|
336 |
author = render_template("/author_info.html", act="author", title=_(u"IEK-7"),
|
icg179@496
|
337 |
card=card, image=qr_image, contact=_(u"Contact"), exit=_(u"Close"))
|
rb@277
|
338 |
|
rb@264
|
339 |
return render_template("/iek-7.html", act="iek-7", content=content,
|
j@380
|
340 |
author=author, card=card, contact=u"IEK-7")
|
peter@1
|
341 |
|
peter@1
|
342 |
|
rb@100
|
343 |
@app.route("/imprint")
|
peter@88
|
344 |
def imprint():
|
rb@163
|
345 |
filename = os.path.join("templates", get_locale(), "rst", "imprint.rst")
|
rb@163
|
346 |
content = get_content(filename)
|
rb@170
|
347 |
return render_template("/content.html", act="imprint", content=content)
|
peter@88
|
348 |
|
peter@109
|
349 |
|
rb@846
|
350 |
@app.route("/dataprotection")
|
rb@846
|
351 |
def dataprotection():
|
rb@846
|
352 |
filename = os.path.join("templates", get_locale(), "rst", "datenschutz.rst")
|
rb@846
|
353 |
content = get_content(filename)
|
rb@846
|
354 |
return render_template("/content.html", act="datenschutz", content=content)
|
rb@845
|
355 |
|
rb@845
|
356 |
|
peter@1
|
357 |
@app.errorhandler(404)
|
peter@1
|
358 |
def page_not_found(e):
|
rb@213
|
359 |
msg = _(u"Url: %(url)s not found", url=request.url)
|
rb@157
|
360 |
info = _(u"This information is not available!")
|
rb@157
|
361 |
return render_template("404.html", msg=msg, info=info)
|
peter@1
|
362 |
|
icg179@496
|
363 |
|
peter@1
|
364 |
if __name__ == "__main__":
|
j@503
|
365 |
app.run(host='localhost', port=5014)
|