eskp.py
author Jens-Uwe Grooss <j.-u.grooss@fz-juelich.de>
Wed, 29 Jan 2025 16:54:42 +0100
changeset 1577 bc114089a220
parent 1557 b9ea1aa08de5
child 1598 c03d9beb0451
permissions -rwxr-xr-x
update of ESKP pages to new institute name
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
j@1053
    52
def get_html_content(filename, overrides=None):
j@1053
    53
    html_content = u""
j@1053
    54
    filename = os.path.join(ESKP_PATH, filename)
j@1053
    55
    if os.path.isfile(filename):
j@1053
    56
        with codecs.open(filename, 'r', 'utf-8') as f:
j@1053
    57
            html_content = f.read()
j@1053
    58
j@1053
    59
    return html_content
icg179@496
    60
icg179@496
    61
def get_newest_date():
Tim@498
    62
    getdates = get_valid_dates(FILES)
icg179@496
    63
    newest_date = getdates[-1]
icg179@496
    64
icg179@496
    65
    return newest_date
icg179@496
    66
icg179@496
    67
peter@142
    68
def get_topmenue():
icg179@496
    69
    newest_date = get_newest_date()
rb@238
    70
    menue = [
icg179@496
    71
        ('/ozoneloss', _(u'Ozoneloss'),
icg179@496
    72
         (('/ozoneloss', _(u'overview')),
j@1557
    73
          ('/ozoneloss/clams/2025', _(u'calculations')),
j@1557
    74
          ('/ozoneloss/vpsc/2025', _(u'estimations')),
j@1019
    75
          ('/ozoneloss/uvmap/' + newest_date, _(u'uv index map')),
icg179@496
    76
          ('/ozoneloss/uvi', _(u'uv increase')),
j@1019
    77
          ('/ozoneloss/decadal', _(u'decadal')))),
icg179@496
    78
        ('/eskp', _(u'ESKP'), (None, None)),
j@1577
    79
        ('/ice-4', _(u'ICE-4'), (None, None))
icg179@496
    80
    ]
peter@145
    81
    return menue
peter@142
    82
icg179@496
    83
peter@142
    84
app.jinja_env.globals.update(get_topmenue=get_topmenue)
peter@142
    85
icg179@496
    86
j@250
    87
def get_o3lossclams_dates():
rb@302
    88
    menue = [
j@1557
    89
        ('/ozoneloss/clams/2025', _(u'2025')),
j@1526
    90
        ('/ozoneloss/clams/2024', _(u'2024')),
j@1295
    91
        ('/ozoneloss/clams/2022', _(u'2022')),
j@989
    92
        ('/ozoneloss/clams/2020', _(u'2020')),
j@851
    93
        ('/ozoneloss/clams/2019', _(u'2019')),
j@733
    94
        ('/ozoneloss/clams/2018', _(u'2018')),
j@584
    95
        ('/ozoneloss/clams/2017', _(u'2017')),
j@441
    96
        ('/ozoneloss/clams/2016', _(u'2016')),
j@380
    97
        ('/ozoneloss/clams/2015', _(u'2015')),
j@380
    98
        ('/ozoneloss/clams/2012', _(u'2012')),
j@380
    99
        ('/ozoneloss/clams/2011', _(u'2011')),
j@380
   100
        ('/ozoneloss/clams/2010', _(u'2010')),
icg179@496
   101
    ]
j@249
   102
    return menue
j@249
   103
icg179@496
   104
icg179@496
   105
def get_valid_dates(files):
icg179@496
   106
    dates = []
icg179@496
   107
icg179@496
   108
    for file in files:
icg179@496
   109
        if file.endswith('.png') and file.find('uvi') >= 0:
icg179@496
   110
            date = file[-12:-6]
icg179@496
   111
            dates.append(date)
icg179@496
   112
    dates.sort()
icg179@496
   113
    for date in dates:
icg179@496
   114
        i = 0
icg179@496
   115
        for param in ['uvi', 'o3col', 'do3col']:
icg179@496
   116
            testfile = 'clams_' + param + '_' + date + '12.png'
icg179@496
   117
            if files.count(testfile) > 0:
icg179@496
   118
                i = i + 1
icg179@496
   119
        if i <> 3:
icg179@496
   120
            dates.remove(date)
icg179@496
   121
    return dates
icg179@496
   122
icg179@496
   123
icg179@496
   124
def get_o3lossuvmap_dates(date_show):
Tim@498
   125
    dates = get_valid_dates(FILES)
icg179@496
   126
    ndates = len(dates)
icg179@496
   127
    ind = dates.index(date_show)
j@503
   128
    navitexts = []
icg179@496
   129
    if ind ==0 :
icg179@496
   130
        chosendates = [dates[ind+1]]
j@503
   131
        navitexts.append('next ->')
Tim@501
   132
icg179@496
   133
    elif ind >= ndates - 1:
icg179@496
   134
        chosendates= [dates[ind-1]]
j@503
   135
        navitexts.append('<- prev')
icg179@496
   136
    else:
icg179@496
   137
        chosendates = [dates[ind-1], dates[ind+1]]
j@503
   138
        navitexts.append('<- prev')
j@503
   139
        navitexts.append('next ->')
icg179@496
   140
    menue = []
Tim@501
   141
Tim@501
   142
    for i in range(len(chosendates)):
Tim@501
   143
        date = chosendates[i]
j@503
   144
        navitext = navitexts[i]
j@503
   145
        menue.append(('/ozoneloss/uvmap/' + date, _(navitext)))
icg179@496
   146
    return menue
icg179@496
   147
icg179@496
   148
j@249
   149
def get_vpsc_dates():
rb@302
   150
    menue = [
j@1557
   151
        ('/ozoneloss/vpsc/2025', _(u'2025')),
j@1526
   152
        ('/ozoneloss/vpsc/2024', _(u'2024')),
j@1295
   153
        ('/ozoneloss/vpsc/2022', _(u'2022')),
j@989
   154
        ('/ozoneloss/vpsc/2020', _(u'2020')),
j@851
   155
        ('/ozoneloss/vpsc/2019', _(u'2019')),
j@733
   156
        ('/ozoneloss/vpsc/2018', _(u'2018')),
j@584
   157
        ('/ozoneloss/vpsc/2017', _(u'2017')),
j@441
   158
        ('/ozoneloss/vpsc/2016', _(u'2016')),
j@380
   159
        ('/ozoneloss/vpsc/2015', _(u'2015')),
j@380
   160
        ('/ozoneloss/vpsc/2014', _(u'2014')),
j@380
   161
        ('/ozoneloss/vpsc/2013', _(u'2013')),
j@380
   162
        ('/ozoneloss/vpsc/2012', _(u'2012')),
j@380
   163
        ('/ozoneloss/vpsc/2011', _(u'2011')),
j@380
   164
        ('/ozoneloss/vpsc/2010', _(u'2010')),
icg179@496
   165
    ]
rb@241
   166
    return menue
rb@241
   167
icg179@496
   168
j@250
   169
app.jinja_env.globals.update(get_o3lossclams_dates=get_o3lossclams_dates)
j@249
   170
app.jinja_env.globals.update(get_vpsc_dates=get_vpsc_dates)
icg179@496
   171
app.jinja_env.globals.update(get_o3lossuvmap_dates=get_o3lossuvmap_dates)
peter@142
   172
rb@267
   173
rb@267
   174
def modal_info(template, act, title, filename):
rb@264
   175
    content = get_content(filename)
rb@267
   176
    html = render_template(template, act=act, title=title, content=content, exit=_(u"Close"))
rb@264
   177
    return html
rb@264
   178
rb@264
   179
peter@1
   180
@babel.localeselector
peter@1
   181
def get_locale():
j@1040
   182
    requested_language = request.accept_languages.best_match(LANGUAGES.keys())
j@1040
   183
    if requested_language in ("de", "en"):
j@1040
   184
        return requested_language or "de"
rb@100
   185
peter@1
   186
@app.route("/")
rb@100
   187
@app.route("/index")
peter@1
   188
def index():
rb@213
   189
    return render_template("/index.html",
rb@233
   190
                           eskp_info=_(u'About ESKP'),
rb@233
   191
                           )
peter@1
   192
rb@241
   193
rb@257
   194
@app.route('/ozoneloss/clams/<year>')
rb@257
   195
def ozoneloss_clams_year(year):
j@249
   196
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_clams.rst")
rb@241
   197
    content = get_content(filename)
rb@257
   198
    return render_template("/ozoneloss_clams.html", act="ozoneloss/clams/%s" % year, content=content, year=year)
rb@241
   199
rb@241
   200
icg179@496
   201
@app.route('/ozoneloss/uvmap/<date>')
icg179@496
   202
def ozoneloss_uvmap_date(date):
icg179@496
   203
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvmap.rst")
icg179@496
   204
    content = get_content(filename)
j@1056
   205
    htmlfile = os.path.join("templates", get_locale(), "html", "uvi_table.html")
j@1056
   206
    html_table = get_html_content(htmlfile)
j@1056
   207
j@1056
   208
    return render_template("/ozoneloss_uvmap.html", act="ozoneloss/uvmap/%s" % date, content=content,
j@1056
   209
                           html_table=html_table, date=date)
icg179@496
   210
icg179@496
   211
rb@257
   212
@app.route('/ozoneloss/vpsc/<year>')
j@1019
   213
def ozoneloss_vpsc_year(year):
j@249
   214
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_vpsc.rst")
j@249
   215
    content = get_content(filename)
j@250
   216
    filename = os.path.join("templates", get_locale(), "rst", "explanation_vpsc.rst")
j@250
   217
    explanation = get_content(filename)
j@250
   218
rb@257
   219
    return render_template("/ozoneloss_vpsc.html", act="ozoneloss/vpsc/%s" % year, content=content,
rb@257
   220
                           content_explanation=explanation, year=year)
rb@241
   221
j@317
   222
rb@118
   223
@app.route('/de')
rb@118
   224
def de():
rb@118
   225
    global LANGUAGE_SELECTED
rb@118
   226
    LANGUAGE_SELECTED = "de"
peter@160
   227
    return render_template("/index.html",
rb@233
   228
                           eskp_info=_(u'About ESKP'),
rb@233
   229
                           )
rb@118
   230
icg179@496
   231
rb@118
   232
@app.route('/en')
rb@118
   233
def en():
rb@118
   234
    global LANGUAGE_SELECTED
rb@118
   235
    LANGUAGE_SELECTED = "en"
peter@160
   236
    return render_template("/index.html",
rb@233
   237
                           eskp_info=_(u'About ESKP'),
rb@233
   238
                           )
rb@118
   239
icg179@496
   240
rb@233
   241
@app.route("/eskp")
rb@233
   242
def eskp():
rb@233
   243
    filename = os.path.join("templates", get_locale(), "rst", "eskp.rst")
rb@162
   244
    content = get_content(filename)
j@272
   245
    filename = os.path.join("templates", get_locale(), "rst", "eskp_title.rst")
j@272
   246
    headline = get_content(filename)
j@272
   247
    return render_template("/eskp.html", act="eskp", content=content, headline=headline)
j@252
   248
peter@1
   249
rb@275
   250
def qr_image_data(card):
icg179@496
   251
    buf = StringIO.StringIO()
rb@275
   252
    qr = qrcode.QRCode(
rb@275
   253
        version=1,
rb@275
   254
        error_correction=qrcode.constants.ERROR_CORRECT_L,
rb@275
   255
        box_size=2,
rb@275
   256
        border=2,
icg179@496
   257
    )
rb@275
   258
    qr.add_data(card.serialize())
rb@275
   259
    qr.make(fit=True)
rb@275
   260
    img = qr.make_image()
rb@275
   261
    img.save(buf)
rb@275
   262
    image = buf.getvalue()
rb@275
   263
    return base64.b64encode(image)
rb@275
   264
rb@275
   265
rb@233
   266
@app.route("/ozoneloss")
rb@257
   267
def ozoneloss():
rb@233
   268
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss.rst")
rb@165
   269
    content = get_content(filename)
j@1116
   270
    ref_date = "200119"
j@1116
   271
    newest_date = get_newest_date()
j@1116
   272
    content = content.replace(ref_date, newest_date, 1)
rb@275
   273
rb@275
   274
    vcard_file = os.path.join("vcards", "jug.vcf")
rb@275
   275
    author = u""
icg179@496
   276
rb@275
   277
    try:
rb@275
   278
        card = get_vcard(vcard_file)
rb@275
   279
    except IOError:
rb@275
   280
        card = None
rb@275
   281
    if card is not None:
rb@275
   282
        qr_image = qr_image_data(card)
rb@275
   283
        author = render_template("/author_info.html", act="author", title=_(u"Ozoneloss"),
icg179@496
   284
                                 card=card, image=qr_image, contact=_(u"Contact"), exit=_(u"Close"))
rb@275
   285
rb@267
   286
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_publications.rst")
rb@267
   287
    publications = modal_info("/publications_info.html", "publications", _(u"Ozoneloss"), filename)
rb@264
   288
    return render_template("/ozoneloss.html", act="ozoneloss", content=content,
icg179@496
   289
                           author=author, card=card, publications=publications)
icg179@496
   290
j@247
   291
j@249
   292
@app.route("/ozoneloss/clams")
rb@257
   293
def ozoneloss_clams():
j@249
   294
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_clams.rst")
j@247
   295
    content = get_content(filename)
j@249
   296
    return render_template("/ozoneloss_clams.html", act="ozoneloss/clams", content=content)
j@249
   297
icg179@496
   298
icg179@496
   299
@app.route("/ozoneloss/uvmap")
icg179@496
   300
def ozoneloss_uvmap():
icg179@496
   301
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvmap.rst")
icg179@496
   302
    content = get_content(filename)
j@1053
   303
j@1053
   304
    htmlfile = os.path.join("templates", get_locale(), "html", "uvi_table.html")
j@1053
   305
    html_content = get_html_content(htmlfile)
j@1053
   306
    return render_template("/ozoneloss_uvmap.html", act="ozoneloss/uvmap", content=content,
j@1053
   307
                           html_content=html_content)
icg179@496
   308
icg179@496
   309
j@249
   310
@app.route("/ozoneloss/vpsc")
j@1019
   311
def ozoneloss_vpsc():
j@249
   312
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_vpsc.rst")
j@249
   313
    content = get_content(filename)
j@249
   314
    return render_template("/ozoneloss_vpsc.html", act="ozoneloss/vpsc", content=content)
peter@1
   315
j@380
   316
j@380
   317
@app.route("/ozoneloss/uvi", methods=['GET'])
j@317
   318
def ozoneloss_uvi():
j@317
   319
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvi.rst")
j@317
   320
    content = get_content(filename)
j@1057
   321
    htmlfile = os.path.join("templates", get_locale(), "html", "ozoneloss_uvi_form_start.html")
j@1053
   322
    html_content = get_html_content(htmlfile)
j@1053
   323
    htmlfile = os.path.join("templates", get_locale(), "html", "uvi_table.html")
j@1053
   324
    html_table = get_html_content(htmlfile)
j@380
   325
j@380
   326
    lat = 50.
j@380
   327
    o3offset = 50.
j@380
   328
    figname = "uvincr_lat%0.3i_do3%0.3i.svg" % (lat, o3offset)
j@380
   329
j@380
   330
    return render_template('ozoneloss_uvi.html', act="ozoneloss/uvi", content=content, figname=figname,
j@1053
   331
                           alt=_(u"UV increase at {{lat}} degrees N for {{o3offset}} DU ozone depletion"),
j@1057
   332
                                 html_content=html_content, html_table=html_table)
j@380
   333
icg179@496
   334
j@380
   335
@app.route("/ozoneloss/uvi_graph", methods=['POST'])
j@380
   336
def ozoneloss_uvi_graph():
j@380
   337
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_uvi.rst")
j@380
   338
    content = get_content(filename)
j@1053
   339
    htmlfile = os.path.join("templates", get_locale(), "html", "ozoneloss_uvi_form.html")
j@1053
   340
    html_content = get_html_content(htmlfile)
j@1053
   341
    htmlfile = os.path.join("templates", get_locale(), "html", "uvi_table.html")
j@1053
   342
    html_table = get_html_content(htmlfile)
j@380
   343
j@380
   344
    latstr = request.form['Gradzahl']
j@380
   345
    o3offsetstr = request.form['Dobson-Unit']
j@380
   346
j@380
   347
    latstr2 = latstr.replace(u'\xb0', '')
j@380
   348
    if latstr2.endswith(u'N'):
j@380
   349
        latstr2 = latstr2.replace(u'N', '')
j@380
   350
        lat = float(latstr2)
j@380
   351
    if latstr2.endswith(u'S'):
j@380
   352
        latstr2 = latstr2.replace(u'S', '')
j@380
   353
        lat = -float(latstr2)
j@1057
   354
        
j@1057
   355
    html_content = html_content.replace('{{latstr}}', latstr)
j@1057
   356
    html_content = html_content.replace('{{o3offsetstr}}', o3offsetstr)
j@1057
   357
    figname = "uvincr_lat%0.3i_do3%0.3i.svg" % (lat, float(o3offsetstr))
j@380
   358
j@1057
   359
    return render_template('ozoneloss_uvi.html', act="ozoneloss_uvi_graph", content=content, figname=figname,
j@1053
   360
                           alt=_(u"UV increase at {{lat}} degrees N for {{o3offset}} DU ozone depletion"),
j@1057
   361
                           html_content=html_content, html_table=html_table)
j@380
   362
j@1019
   363
@app.route("/ozoneloss/decadal")
j@1019
   364
def ozoneloss_decadal():
j@1019
   365
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_decadal.rst")
j@1019
   366
    content = get_content(filename)
j@1019
   367
    filename = os.path.join("templates", get_locale(), "rst", "ozoneloss_decadal_bottom.rst")
j@1019
   368
    explanation = get_content(filename)
j@1019
   369
    
j@1019
   370
    return render_template("/ozoneloss_decadal.html", act="ozoneloss_decadal", content=content,
j@1019
   371
                           content_explanation=explanation)
j@1019
   372
j@1019
   373
j@380
   374
j@1577
   375
@app.route("/ice-4")
rb@257
   376
def institute():
j@1577
   377
    filename = os.path.join("templates", get_locale(), "rst", "ice-4.rst")
rb@167
   378
    content = get_content(filename)
rb@277
   379
    vcard_file = os.path.join("vcards", "sas.vcf")
rb@277
   380
    author = u""
rb@277
   381
    try:
rb@277
   382
        card = get_vcard(vcard_file)
rb@277
   383
    except IOError:
rb@277
   384
        card = None
rb@277
   385
    if card is not None:
rb@277
   386
        qr_image = qr_image_data(card)
j@1577
   387
        author = render_template("/author_info.html", act="author", title=_(u"ICE-4"),
icg179@496
   388
                                 card=card, image=qr_image, contact=_(u"Contact"), exit=_(u"Close"))
rb@277
   389
j@1577
   390
    return render_template("/ice-4.html", act="ice-4", content=content,
j@1577
   391
                           author=author, card=card, contact=u"ICE-4")
peter@1
   392
peter@1
   393
rb@100
   394
@app.route("/imprint")
peter@88
   395
def imprint():
rb@163
   396
    filename = os.path.join("templates", get_locale(), "rst", "imprint.rst")
rb@163
   397
    content = get_content(filename)
rb@170
   398
    return render_template("/content.html", act="imprint", content=content)
peter@88
   399
peter@109
   400
rb@846
   401
@app.route("/dataprotection")
rb@846
   402
def dataprotection():
rb@846
   403
    filename = os.path.join("templates", get_locale(), "rst", "datenschutz.rst")
rb@846
   404
    content = get_content(filename)
rb@846
   405
    return render_template("/content.html", act="datenschutz", content=content)
rb@845
   406
rb@845
   407
peter@1
   408
@app.errorhandler(404)
peter@1
   409
def page_not_found(e):
rb@213
   410
    msg = _(u"Url: %(url)s not found", url=request.url)
rb@157
   411
    info = _(u"This information is not available!")
rb@157
   412
    return render_template("404.html", msg=msg, info=info)
peter@1
   413
icg179@496
   414
peter@1
   415
if __name__ == "__main__":
j@503
   416
    app.run(host='localhost', port=5014)
Impressum Datenschutzerklärung