afpy.org/afpy.py

345 lines
9.4 KiB
Python
Raw Normal View History

2018-04-29 13:47:47 +00:00
import email
2017-09-22 14:42:56 +00:00
import locale
2018-05-11 16:54:42 +00:00
import os
2018-04-29 13:47:47 +00:00
import time
2017-09-22 14:42:56 +00:00
2017-09-21 15:20:00 +00:00
import docutils.core
import docutils.writers.html5_polyglot
2017-09-21 15:40:18 +00:00
import feedparser
from dateutil.parser import parse
2018-06-05 10:34:49 +00:00
from flask import (Flask, abort, jsonify, redirect, render_template, request,
send_from_directory, url_for)
2018-04-29 13:47:47 +00:00
from flask_cache import Cache
2018-05-11 16:54:42 +00:00
from itsdangerous import BadSignature, URLSafeSerializer
2017-09-21 17:06:27 +00:00
from jinja2 import TemplateNotFound
2017-09-21 14:34:26 +00:00
2018-10-04 13:34:38 +00:00
import data_xml as data
2017-12-07 17:26:55 +00:00
locale.setlocale(locale.LC_ALL, 'fr_FR.UTF-8')
2017-09-22 14:42:56 +00:00
2018-04-29 13:47:47 +00:00
cache = Cache(config={'CACHE_TYPE': 'simple', 'CACHE_DEFAULT_TIMEOUT': 600})
2018-05-11 14:32:38 +00:00
signer = URLSafeSerializer(os.environ.get('SECRET', 'changeme!'))
2017-09-21 14:34:26 +00:00
app = Flask(__name__)
2018-04-29 13:47:47 +00:00
cache.init_app(app)
2017-09-21 14:34:26 +00:00
2018-10-04 13:34:38 +00:00
2018-05-11 15:07:13 +00:00
PAGINATION = 12
2018-04-29 13:47:47 +00:00
PLANET = {
2018-05-05 12:07:05 +00:00
'Emplois AFPy': 'https://www.afpy.org/feed/emplois/rss.xml',
'Nouvelles AFPy': 'https://www.afpy.org/feed/actualites/rss.xml',
2018-04-29 13:47:47 +00:00
'Anybox': 'https://anybox.fr/site-feed/RSS?set_language=fr',
'Ascendances': 'https://ascendances.wordpress.com/feed/',
'Code en Seine': 'https://codeenseine.fr/feeds/all.atom.xml',
2018-10-04 13:34:38 +00:00
'Yaal': 'https://www.yaal.fr/blog/feeds/all.atom.xml',
2018-04-29 13:47:47 +00:00
}
2017-09-21 16:32:31 +00:00
MEETUPS = {
2018-05-16 19:17:32 +00:00
'amiens': 'https://www.meetup.com/fr-FR/Python-Amiens',
2018-10-04 13:34:38 +00:00
'bruxelles': 'https://www.meetup.com/fr-FR/'
'Belgium-Python-Meetup-aka-AperoPythonBe/',
'grenoble': 'https://www.meetup.com/fr-FR/'
'Groupe-dutilisateurs-Python-Grenoble/',
'lille': 'https://www.meetup.com/fr-FR/Lille-py/',
'lyon': 'https://www.meetup.com/fr-FR/Python-AFPY-Lyon/',
'nantes': 'https://www.meetup.com/fr-FR/Nantes-Python-Meetup/',
'montpellier': 'https://www.meetup.com/fr-FR/Meetup-Python-Montpellier/',
2017-09-21 16:32:31 +00:00
}
2017-09-21 14:34:26 +00:00
2017-09-21 15:50:24 +00:00
@app.errorhandler(404)
def page_not_found(e):
return render_template('404.html'), 404
2017-09-21 14:34:26 +00:00
@app.route('/')
def index():
posts = {}
2018-10-04 13:34:38 +00:00
for post in data.get_posts(data.POST_ACTUALITIES, end=4):
timestamp = post[data.TIMESTAMP]
posts[timestamp] = post
return render_template(
2018-10-04 15:44:18 +00:00
'index.html',
body_id='index',
name=data.POST_ACTUALITIES,
posts=posts
2018-10-04 13:34:38 +00:00
)
@app.route('/<name>')
def pages(name):
if name == 'index':
return redirect(url_for('index'))
2017-09-21 17:06:27 +00:00
try:
return render_template(f'{name}.html', body_id=name, meetups=MEETUPS)
2017-09-21 17:06:27 +00:00
except TemplateNotFound:
abort(404)
2017-09-21 14:34:26 +00:00
2017-09-21 15:20:00 +00:00
@app.route('/docs/<name>')
def rest(name):
2017-09-22 14:22:25 +00:00
try:
with open(f'templates/{name}.rst') as fd:
parts = docutils.core.publish_parts(
source=fd.read(),
writer=docutils.writers.html5_polyglot.Writer(),
2018-10-04 13:34:38 +00:00
settings_overrides={'initial_header_level': 2},
)
2017-09-22 14:22:25 +00:00
except FileNotFoundError:
abort(404)
2017-09-22 10:11:26 +00:00
return render_template(
2018-10-04 15:44:18 +00:00
'rst.html',
body_id=name,
html=parts['body'],
title=parts['title']
2018-10-04 13:34:38 +00:00
)
2018-05-11 14:32:38 +00:00
@app.route('/post/edit/<name>')
2018-05-11 14:32:38 +00:00
@app.route('/post/edit/<name>/token/<token>')
def edit_post(name, token=None):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
abort(404)
2018-05-11 14:32:38 +00:00
if token:
try:
timestamp = signer.loads(token)
except BadSignature:
abort(401)
2018-10-04 13:34:38 +00:00
post = data.get_post(name, timestamp)
2018-05-11 14:32:38 +00:00
if not post:
abort(404)
2018-05-11 14:32:38 +00:00
else:
2018-10-04 13:34:38 +00:00
post = {data.STATE: data.STATE_WAITING}
if post[data.STATE] == data.STATE_TRASHED:
return redirect(url_for('rest', name='already_trashed'))
return render_template(
2018-10-04 13:34:38 +00:00
'edit_post.html',
body_id='edit-post',
post=post,
name=name,
admin=False,
)
2018-05-11 14:32:38 +00:00
@app.route('/admin/post/edit/<name>/<timestamp>')
def edit_post_admin(name, timestamp):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
abort(404)
2018-10-04 13:34:38 +00:00
post = data.get_post(name, timestamp)
2018-05-11 14:32:38 +00:00
if not post:
abort(404)
return render_template(
2018-10-04 15:44:18 +00:00
'edit_post.html',
body_id='edit-post',
post=post,
name=name,
admin=True
2018-10-04 13:34:38 +00:00
)
2018-05-11 14:32:38 +00:00
@app.route('/post/edit/<name>', methods=['post'])
@app.route('/post/edit/<name>/token/<token>', methods=['post'])
def save_post(name, token=None):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
2018-05-11 14:32:38 +00:00
abort(404)
if token:
try:
timestamp = signer.loads(token)
except BadSignature:
abort(401)
else:
timestamp = None
2018-10-04 13:34:38 +00:00
try:
post = data.save_post(
2018-10-04 15:34:58 +00:00
name, timestamp=timestamp, admin=False,
form=request.form, files=request.files
2018-10-04 13:34:38 +00:00
)
except data.DataException as e:
abort(e.http_code)
2018-05-11 14:32:38 +00:00
edit_post_url = url_for(
2018-10-04 13:34:38 +00:00
'edit_post', name=name, token=signer.dumps(post['_timestamp'])
)
if post[data.STATE] == data.STATE_TRASHED:
return redirect(url_for('rest', name='already_trashed'))
2018-10-04 15:44:18 +00:00
return render_template(
'confirmation.html',
edit_post_url=edit_post_url
)
2018-05-11 14:32:38 +00:00
@app.route('/admin/post/edit/<name>/<timestamp>', methods=['post'])
def save_post_admin(name, timestamp):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
2018-05-11 14:32:38 +00:00
abort(404)
2018-10-04 13:34:38 +00:00
try:
data.save_post(
2018-10-04 15:34:58 +00:00
name, timestamp=timestamp, admin=True,
form=request.form, files=request.files
2018-10-04 13:34:38 +00:00
)
except data.DataException as e:
abort(e.http_code)
2018-10-05 14:47:32 +00:00
if 'delete_image' in request.form:
return redirect(request.url)
2018-05-11 14:32:38 +00:00
return redirect(url_for('admin', name=name))
@app.route('/posts/<name>')
2018-05-11 15:07:13 +00:00
@app.route('/posts/<name>/page/<int:page>')
def posts(name, page=1):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
2017-09-22 14:22:25 +00:00
abort(404)
2018-05-11 15:07:13 +00:00
end = page * PAGINATION
start = end - PAGINATION
2018-10-04 13:34:38 +00:00
total_pages = (
data.count_posts(name, data.STATE_PUBLISHED) // PAGINATION
) + 1
posts = {}
2018-10-04 13:34:38 +00:00
for post in data.get_posts(
name, data.STATE_PUBLISHED, start=start, end=end
):
timestamp = post[data.TIMESTAMP]
posts[timestamp] = post
2017-09-22 10:11:26 +00:00
return render_template(
2018-10-04 13:34:38 +00:00
'posts.html',
body_id=name,
posts=posts,
title=data.POSTS[name],
name=name,
page=page,
total_pages=total_pages,
)
@app.route('/admin/posts/<name>')
def admin(name):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
abort(404)
2018-04-30 20:00:41 +00:00
posts = {}
2018-10-04 13:34:38 +00:00
for state in data.STATES:
posts[state] = state_posts = {}
2018-10-04 13:34:38 +00:00
for post in data.get_posts(name, state):
timestamp = post[data.TIMESTAMP]
state_posts[timestamp] = post
return render_template(
2018-10-04 13:34:38 +00:00
'admin.html',
body_id='admin',
posts=posts,
title=data.POSTS[name],
name=name,
)
2018-04-30 20:00:41 +00:00
@app.route('/posts/<name>/<timestamp>')
def post(name, timestamp):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
abort(404)
2018-10-04 13:34:38 +00:00
post = data.get_post(name, timestamp, data.STATE_PUBLISHED)
if not post:
abort(404)
2018-10-04 15:44:18 +00:00
return render_template(
'post.html',
body_id='post',
post=post,
name=name
)
2017-09-21 15:40:18 +00:00
2018-10-04 15:44:18 +00:00
@app.route('/post_image/<path:path>')
2018-05-04 20:54:23 +00:00
def post_image(path):
2018-10-04 15:44:18 +00:00
if path.count('/') != 3:
2018-05-04 20:54:23 +00:00
abort(404)
2018-10-04 15:44:18 +00:00
category, state, timestamp, name = path.split('/')
if category not in data.POSTS:
2018-05-04 20:54:23 +00:00
abort(404)
2018-10-04 15:44:18 +00:00
if state not in data.STATES:
2018-05-04 20:54:23 +00:00
abort(404)
2018-10-04 15:44:18 +00:00
return send_from_directory(data.root, path)
2018-05-04 20:54:23 +00:00
@app.route('/feed/<name>/rss.xml')
@cache.cached()
def feed(name):
2018-10-04 13:34:38 +00:00
if name not in data.POSTS:
abort(404)
entries = []
2018-10-04 13:34:38 +00:00
for post in data.get_posts(name, data.STATE_PUBLISHED, end=50):
2019-02-04 13:31:56 +00:00
post['timestamp'] = post[data.TIMESTAMP]
2018-10-04 13:34:38 +00:00
post['link'] = url_for(
2019-02-04 13:31:56 +00:00
'post', name=name, timestamp=post['timestamp'],
_external=True
2018-10-04 13:34:38 +00:00
)
entries.append({'content': post})
title = f'{data.POSTS[name]} AFPy.org'
return render_template(
2018-10-04 13:34:38 +00:00
'rss.xml',
entries=entries,
title=title,
description=title,
link=url_for('feed', name=name, _external=True),
)
2018-04-29 13:47:47 +00:00
@app.route('/planet/')
2018-04-26 13:23:02 +00:00
@app.route('/planet/rss.xml')
@cache.cached()
2018-04-26 13:23:02 +00:00
def planet():
2018-04-29 13:47:47 +00:00
entries = []
for name, url in PLANET.items():
for entry in feedparser.parse(url).entries:
if hasattr(entry, 'updated_parsed'):
date = entry.updated_parsed
elif hasattr(entry, 'published_parsed'):
date = entry.published_parsed
else:
date = time.time()
entry['timestamp'] = time.mktime(date) if date else time.time()
2018-04-29 13:47:47 +00:00
entries.append({'feed': name, 'content': entry})
entries.sort(reverse=True, key=lambda entry: entry['content']['timestamp'])
return render_template(
2018-10-04 13:34:38 +00:00
'rss.xml',
entries=entries,
title='Planet Python francophone',
description='Nouvelles autour de Python en français',
2018-10-04 13:34:38 +00:00
link=url_for('planet', _external=True),
)
2018-04-26 13:23:02 +00:00
@app.route('/rss-jobs/RSS')
def jobs():
return redirect('https://plone.afpy.org/rss-jobs/RSS', code=307)
2018-06-10 17:46:06 +00:00
@app.route('/status')
def status():
2018-06-04 10:04:13 +00:00
stats = {}
2018-10-04 13:34:38 +00:00
for category in data.POSTS:
2018-06-04 10:04:13 +00:00
stats[category] = {}
2018-10-04 13:34:38 +00:00
for state in data.STATES:
stats[category][state] = data.count_posts(category, state)
2018-06-04 10:04:13 +00:00
2018-06-10 17:46:06 +00:00
os_stats = os.statvfs(__file__)
stats['disk_free'] = os_stats.f_bavail * os_stats.f_frsize
stats['disk_total'] = os_stats.f_blocks * os_stats.f_frsize
stats['load_avg'] = os.getloadavg()
2018-06-04 10:04:13 +00:00
return jsonify(stats)
2018-04-29 13:47:47 +00:00
@app.template_filter('rfc822_datetime')
def format_rfc822_datetime(timestamp):
2019-02-04 13:31:56 +00:00
return email.utils.formatdate(int(timestamp))
2018-04-29 13:47:47 +00:00
@app.template_filter('parse_iso_datetime')
def parse_iso_datetime(iso_datetime, format_):
return parse(iso_datetime).strftime(format_)
if app.env == 'development': # pragma: no cover
2017-09-21 14:34:26 +00:00
from sassutils.wsgi import SassMiddleware
2018-10-04 13:34:38 +00:00
2017-09-22 09:30:21 +00:00
app.wsgi_app = SassMiddleware(
2018-10-04 13:34:38 +00:00
app.wsgi_app, {'afpy': ('sass', 'static/css', '/static/css')}
)