calendar-social/calsocial/__init__.py

153 lines
4.4 KiB
Python
Raw Normal View History

# Calendar.social
# Copyright (C) 2018 Gergely Polonkai
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
2018-06-25 07:01:13 +00:00
from datetime import datetime
from functools import wraps
import os
from flask import Flask, current_app, redirect, render_template, request, url_for
from flask_babelex import Babel, get_locale as babel_get_locale
2018-06-30 04:44:45 +00:00
from flask_security import SQLAlchemyUserDatastore, current_user, login_required
def get_locale():
"""Locale selector
Selects the best locale based on values sent by the browser.
"""
from flask import request
supported_languages = ['en', 'hu']
if 'l' in request.args and request.args['l'].lower() in supported_languages:
return request.args['l'].lower()
return request.accept_languages.best_match(supported_languages)
def template_vars():
now = datetime.utcnow()
return {
'lang': babel_get_locale().language,
'now': now,
'now_ts': now.timestamp(),
}
def route(*args, **kwargs):
def decorator(func):
setattr(func, 'routing', (args, kwargs))
return func
return decorator
class CalendarSocialApp(Flask):
def __init__(self, name, config=None):
from .models import db, User, Role
from .security import security
2018-06-28 12:41:14 +00:00
Flask.__init__(self, name)
config_name = os.environ.get('ENV', config or 'dev')
self.config.from_pyfile(f'config_{config_name}.py', True)
2018-06-29 06:03:49 +00:00
# Make sure we look up users both by their usernames and email addresses
self.config['SECURITY_USER_IDENTITY_ATTRIBUTES'] = ('username', 'email')
2018-06-28 12:41:14 +00:00
db.init_app(self)
babel = Babel(app=self)
babel.localeselector(get_locale)
2018-06-29 06:03:49 +00:00
user_store = SQLAlchemyUserDatastore(db, User, Role)
security.init_app(self, datastore=user_store)
self.context_processor(template_vars)
for attr_name in self.__dir__():
attr = getattr(self, attr_name)
if not callable(attr):
continue
args, kwargs = getattr(attr, 'routing', (None, None))
if args is None:
continue
self.route(*args, **kwargs)(attr)
2018-06-25 07:01:13 +00:00
@route('/')
def hello(self):
from .calendar_system.gregorian import GregorianCalendar
2018-06-29 06:03:49 +00:00
if not current_user.is_authenticated:
return render_template('welcome.html')
2018-06-25 07:01:13 +00:00
try:
timestamp = datetime.fromtimestamp(float(request.args.get('date')))
except TypeError:
timestamp = datetime.utcnow()
calendar = GregorianCalendar(timestamp.timestamp())
return render_template('index.html', calendar=calendar)
@route('/register', methods=['POST', 'GET'])
def register(self):
if not current_app.config['REGISTRATION_ENABLED']:
return render_template('registration-disabled.html')
2018-06-29 12:00:45 +00:00
from .forms import RegistrationForm
from .models import db, User
2018-06-29 12:00:45 +00:00
form = RegistrationForm()
2018-06-29 12:00:45 +00:00
if form.validate_on_submit():
# TODO: This might become False later, if we want registrations to be confirmed via E-mail
user = User(active=True)
2018-06-29 12:00:45 +00:00
form.populate_obj(user)
db.session.add(user)
db.session.commit()
2018-06-29 12:00:45 +00:00
return redirect(url_for('hello'))
2018-06-29 12:00:45 +00:00
return render_template('registration.html', form=form)
2018-06-29 12:00:45 +00:00
@route('/new-event', methods=['GET', 'POST'])
@login_required
def new_event(self):
from .forms import EventForm
from .models import db, Event
2018-06-29 12:00:45 +00:00
form = EventForm()
2018-06-30 04:44:45 +00:00
if form.validate_on_submit():
event = Event(user=current_user)
form.populate_obj(event)
2018-06-30 04:44:45 +00:00
db.session.add(event)
db.session.commit()
2018-06-30 04:44:45 +00:00
return redirect(url_for('hello'))
2018-06-30 04:44:45 +00:00
return render_template('event-edit.html', form=form)
2018-06-30 04:44:45 +00:00
app = CalendarSocialApp(__name__)