afpy.org/afpy.py

243 lines
8.0 KiB
Python
Raw Normal View History

2017-09-22 14:42:56 +00:00
import datetime
2018-04-29 13:47:47 +00:00
import email
2017-09-22 14:42:56 +00:00
import locale
2018-04-29 13:47:47 +00:00
import time
from pathlib import Path
from xml.etree import ElementTree
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 flask import Flask, abort, redirect, render_template, request, url_for
2018-04-29 13:47:47 +00:00
from flask_cache import Cache
2017-09-21 17:06:27 +00:00
from jinja2 import TemplateNotFound
2017-09-21 14:34:26 +00:00
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})
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-04-29 13:47:47 +00:00
PLANET = {
'Emplois AFPy': 'https://plone.afpy.org/rss-jobs/RSS',
'Nouvelles AFPy': 'https://plone.afpy.org/rss-actualites/RSS',
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',
'Yaal': 'https://www.yaal.fr/blog/feeds/all.atom.xml'
}
2017-09-21 16:32:31 +00:00
MEETUPS = {
2017-09-22 13:24:37 +00:00
'bruxelles': (
'https://www.meetup.com/fr-FR/'
'Belgium-Python-Meetup-aka-AperoPythonBe/'),
2017-09-22 13:37:15 +00:00
'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
}
POSTS = {
'actualites': 'Actualités',
'emplois': 'Offres demploi',
}
root = Path(__file__).parent / 'posts'
for category in POSTS:
for status in ('waiting', 'published'):
(root / category / status).mkdir(parents=True, exist_ok=True)
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 = {}
path = root / 'actualites' / 'published'
timestamps = sorted(path.iterdir(), reverse=True)[:4]
for timestamp in timestamps:
tree = ElementTree.parse(timestamp / 'post.xml')
posts[timestamp.name] = {item.tag: item.text for item in tree.iter()}
return render_template(
'index.html', body_id='index', name='actualites', posts=posts)
@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(),
settings_overrides={'initial_header_level': 2})
except FileNotFoundError:
abort(404)
2017-09-22 10:11:26 +00:00
return render_template(
'rst.html', body_id=name, html=parts['body'], title=parts['title'])
2017-09-21 15:20:00 +00:00
@app.route('/post/edit/<name>')
@app.route('/post/edit/<name>/<timestamp>')
def edit_post(name, timestamp=None):
if name not in POSTS:
abort(404)
if timestamp is None:
state = 'waiting'
post = {}
else:
for state in ('published', 'waiting'):
if (root / name / state / timestamp / 'post.xml').is_file():
path = (root / name / state / timestamp / 'post.xml')
break
else:
abort(404)
tree = ElementTree.parse(path)
post = {item.tag: (item.text or '').strip() for item in tree.iter()}
return render_template(
'edit_post.html', body_id='edit-post', post=post, name=name,
state=state)
@app.route('/post/edit/<name>', methods=['post'])
@app.route('/post/edit/<name>/<timestamp>', methods=['post'])
def save_post(name, timestamp=None):
original_timestamp = timestamp
if name not in POSTS:
abort(404)
if timestamp is None:
timestamp = str(int(time.time()))
status = 'waiting'
folder = root / name / 'waiting' / timestamp
folder.mkdir()
post = folder / 'post.xml'
elif (root / name / 'waiting' / timestamp / 'post.xml').is_file():
status = 'waiting'
elif (root / name / 'published' / timestamp / 'post.xml').is_file():
status = 'published'
else:
abort(404)
post = root / name / status / timestamp / 'post.xml'
tree = ElementTree.Element('item')
for key in ('title', 'description', 'content', 'email'):
element = ElementTree.SubElement(tree, key)
element.text = request.form[key]
element = ElementTree.SubElement(tree, 'pubDate')
element.text = email.utils.formatdate(
int(timestamp) if timestamp else time.time())
ElementTree.ElementTree(tree).write(post)
if 'publish' in request.form and status == 'waiting':
(root / name / 'waiting' / timestamp).rename(
root / name / 'published' / timestamp)
elif 'unpublish' in request.form and status == 'published':
(root / name / 'published' / timestamp).rename(
root / name / 'waiting' / timestamp)
return redirect(
'/' if original_timestamp else url_for('rest', name='confirmation'))
@app.route('/posts/<name>')
def posts(name):
if name not in POSTS:
2017-09-22 14:22:25 +00:00
abort(404)
path = root / name / 'published'
timestamps = sorted(path.iterdir(), reverse=True)[:12]
posts = {}
for timestamp in timestamps:
tree = ElementTree.parse(timestamp / 'post.xml')
posts[timestamp.name] = {item.tag: item.text for item in tree.iter()}
2017-09-22 10:11:26 +00:00
return render_template(
'posts.html', body_id=name, posts=posts, title=POSTS[name], name=name)
@app.route('/posts/<name>/<timestamp>')
def post(name, timestamp):
if name not in POSTS:
abort(404)
try:
tree = ElementTree.parse(
root / name / 'published' / timestamp / 'post.xml')
except Exception:
abort(404)
post = {item.tag: item.text for item in tree.iter()}
return render_template('post.html', body_id='post', post=post)
2017-09-21 15:40:18 +00:00
@app.route('/feed/<name>/rss.xml')
@cache.cached()
def feed(name):
if name not in POSTS:
abort(404)
path = root / name / 'published'
timestamps = sorted(path.iterdir(), reverse=True)[:12]
entries = []
for timestamp in timestamps:
tree = ElementTree.parse(timestamp / 'post.xml')
entry = {item.tag: item.text for item in tree.iter()}
entry['timestamp'] = int(timestamp.name)
entry['link'] = url_for(
'post', name=name, timestamp=timestamp.name, _external=True)
entries.append({'content': entry})
title = f'{POSTS[name]} AFPy.org'
return render_template(
'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:
date = getattr(entry, 'published_parsed', entry.updated_parsed)
entry['timestamp'] = time.mktime(date)
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(
'rss.xml', entries=entries, title='Planet Python francophone',
description='Nouvelles autour de Python en français',
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)
@app.template_filter('parse_rfc822_datetime')
def parse_rfc822_datetime(rfc822_datetime):
return email.utils.parsedate_tz(rfc822_datetime)
2017-09-22 14:42:56 +00:00
@app.template_filter('datetime')
def format_datetime(time_struct, format_):
return datetime.datetime(*time_struct[:6]).strftime(format_)
2018-04-29 13:47:47 +00:00
@app.template_filter('rfc822_datetime')
def format_rfc822_datetime(timestamp):
return email.utils.formatdate(timestamp)
2018-04-29 13:47:47 +00:00
if app.env == 'development': # pragma: no cover
2017-09-21 14:34:26 +00:00
from sassutils.wsgi import SassMiddleware
2017-09-22 09:30:21 +00:00
app.wsgi_app = SassMiddleware(
app.wsgi_app, {'afpy': ('sass', 'static/css', '/static/css')})