2018-06-24 19:42:34 +00:00
|
|
|
# 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
|
2018-07-02 08:48:10 +00:00
|
|
|
from functools import wraps
|
2018-06-28 06:31:11 +00:00
|
|
|
import os
|
|
|
|
|
2018-07-02 08:53:15 +00:00
|
|
|
from flask import Flask, current_app, redirect, render_template, request, url_for
|
2018-07-02 11:04:05 +00:00
|
|
|
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
|
2018-06-24 19:42:34 +00:00
|
|
|
|
|
|
|
|
2018-06-29 12:55:57 +00:00
|
|
|
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():
|
2018-07-02 08:53:15 +00:00
|
|
|
now = datetime.utcnow()
|
|
|
|
|
2018-06-29 12:55:57 +00:00
|
|
|
return {
|
|
|
|
'lang': babel_get_locale().language,
|
2018-07-02 08:53:15 +00:00
|
|
|
'now': now,
|
|
|
|
'now_ts': now.timestamp(),
|
2018-06-29 12:55:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
def route(*args, **kwargs):
|
|
|
|
def decorator(func):
|
|
|
|
setattr(func, 'routing', (args, kwargs))
|
|
|
|
|
|
|
|
return func
|
|
|
|
|
|
|
|
return decorator
|
|
|
|
|
|
|
|
|
2018-06-24 19:42:34 +00:00
|
|
|
class CalendarSocialApp(Flask):
|
2018-06-28 06:31:11 +00:00
|
|
|
def __init__(self, name, config=None):
|
2018-07-02 06:33:21 +00:00
|
|
|
from .models import db, User, Role
|
|
|
|
from .security import security
|
2018-06-28 12:41:14 +00:00
|
|
|
|
2018-06-24 19:42:34 +00:00
|
|
|
Flask.__init__(self, name)
|
|
|
|
|
2018-06-28 06:31:11 +00:00
|
|
|
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)
|
2018-06-29 08:38:18 +00:00
|
|
|
babel = Babel(app=self)
|
2018-06-29 12:55:57 +00:00
|
|
|
babel.localeselector(get_locale)
|
2018-06-28 06:31:11 +00:00
|
|
|
|
2018-06-29 06:03:49 +00:00
|
|
|
user_store = SQLAlchemyUserDatastore(db, User, Role)
|
|
|
|
security.init_app(self, datastore=user_store)
|
|
|
|
|
2018-06-29 12:55:57 +00:00
|
|
|
self.context_processor(template_vars)
|
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
for attr_name in self.__dir__():
|
|
|
|
attr = getattr(self, attr_name)
|
2018-06-24 19:42:34 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
if not callable(attr):
|
|
|
|
continue
|
|
|
|
|
|
|
|
args, kwargs = getattr(attr, 'routing', (None, None))
|
2018-06-24 19:42:34 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
if args is None:
|
|
|
|
continue
|
2018-06-29 07:04:24 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
self.route(*args, **kwargs)(attr)
|
2018-06-25 07:01:13 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
@route('/')
|
|
|
|
def hello(self):
|
|
|
|
from .calendar_system.gregorian import GregorianCalendar
|
2018-06-29 06:03:49 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
if not current_user.is_authenticated:
|
|
|
|
return render_template('welcome.html')
|
2018-06-25 07:01:13 +00:00
|
|
|
|
2018-07-02 08:53:15 +00:00
|
|
|
try:
|
|
|
|
timestamp = datetime.fromtimestamp(float(request.args.get('date')))
|
|
|
|
except TypeError:
|
|
|
|
timestamp = datetime.utcnow()
|
|
|
|
|
|
|
|
calendar = GregorianCalendar(timestamp.timestamp())
|
2018-06-24 19:42:34 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
return render_template('index.html', calendar=calendar)
|
2018-06-24 19:42:34 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
@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
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
from .forms import RegistrationForm
|
|
|
|
from .models import db, User
|
2018-06-29 12:00:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
form = RegistrationForm()
|
2018-06-29 12:00:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +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
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
form.populate_obj(user)
|
|
|
|
db.session.add(user)
|
|
|
|
db.session.commit()
|
2018-06-29 12:00:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
return redirect(url_for('hello'))
|
2018-06-29 12:00:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
return render_template('registration.html', form=form)
|
2018-06-29 12:00:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +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
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
form = EventForm()
|
2018-06-30 04:44:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
if form.validate_on_submit():
|
|
|
|
event = Event(user=current_user)
|
|
|
|
form.populate_obj(event)
|
2018-06-30 04:44:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
db.session.add(event)
|
|
|
|
db.session.commit()
|
2018-06-30 04:44:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
return redirect(url_for('hello'))
|
2018-06-30 04:44:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
return render_template('event-edit.html', form=form)
|
2018-06-30 04:44:45 +00:00
|
|
|
|
2018-07-02 08:48:10 +00:00
|
|
|
|
|
|
|
app = CalendarSocialApp(__name__)
|