| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 |
- import os
- from flask import Flask, url_for, redirect, render_template, request
- from flask_sqlalchemy import SQLAlchemy
- from wtforms import form, fields, validators
- import flask_admin as admin
- import flask_login as login
- from flask_admin.contrib import sqla
- from flask_admin import helpers, expose
- from werkzeug.security import generate_password_hash, check_password_hash
- # Create Flask application
- app = Flask(__name__)
- # Create dummy secrey key so we can use sessions
- app.config['SECRET_KEY'] = '123456790'
- # Create in-memory database
- app.config['DATABASE_FILE'] = 'sample_db.sqlite'
- app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///' + app.config['DATABASE_FILE']
- app.config['SQLALCHEMY_ECHO'] = True
- db = SQLAlchemy(app)
- # Create user model.
- class User(db.Model):
- id = db.Column(db.Integer, primary_key=True)
- first_name = db.Column(db.String(100))
- last_name = db.Column(db.String(100))
- login = db.Column(db.String(80), unique=True)
- email = db.Column(db.String(120))
- password = db.Column(db.String(64))
- # Flask-Login integration
- def is_authenticated(self):
- return True
- def is_active(self):
- return True
- def is_anonymous(self):
- return False
- def get_id(self):
- return self.id
- # Required for administrative interface
- def __unicode__(self):
- return self.username
- # Define login and registration forms (for flask-login)
- class LoginForm(form.Form):
- login = fields.StringField(validators=[validators.required()])
- password = fields.PasswordField(validators=[validators.required()])
- def validate_login(self, field):
- user = self.get_user()
- if user is None:
- raise validators.ValidationError('Invalid user')
- # we're comparing the plaintext pw with the the hash from the db
- if not check_password_hash(user.password, self.password.data):
- # to compare plain text passwords use
- # if user.password != self.password.data:
- raise validators.ValidationError('Invalid password')
- def get_user(self):
- return db.session.query(User).filter_by(login=self.login.data).first()
- class RegistrationForm(form.Form):
- login = fields.StringField(validators=[validators.required()])
- email = fields.StringField()
- password = fields.PasswordField(validators=[validators.required()])
- def validate_login(self, field):
- if db.session.query(User).filter_by(login=self.login.data).count() > 0:
- raise validators.ValidationError('Duplicate username')
- # Initialize flask-login
- def init_login():
- login_manager = login.LoginManager()
- login_manager.init_app(app)
- # Create user loader function
- @login_manager.user_loader
- def load_user(user_id):
- return db.session.query(User).get(user_id)
- # Create customized model view class
- class MyModelView(sqla.ModelView):
- def is_accessible(self):
- return login.current_user.is_authenticated
- # Create customized index view class that handles login & registration
- class MyAdminIndexView(admin.AdminIndexView):
- @expose('/')
- def index(self):
- if not login.current_user.is_authenticated:
- return redirect(url_for('.login_view'))
- return super(MyAdminIndexView, self).index()
- @expose('/login/', methods=('GET', 'POST'))
- def login_view(self):
- # handle user login
- form = LoginForm(request.form)
- if helpers.validate_form_on_submit(form):
- user = form.get_user()
- login.login_user(user)
- if login.current_user.is_authenticated:
- return redirect(url_for('.index'))
- link = '<p>Don\'t have an account? <a href="' + url_for('.register_view') + '">Click here to register.</a></p>'
- self._template_args['form'] = form
- self._template_args['link'] = link
- return super(MyAdminIndexView, self).index()
- @expose('/register/', methods=('GET', 'POST'))
- def register_view(self):
- form = RegistrationForm(request.form)
- if helpers.validate_form_on_submit(form):
- user = User()
- form.populate_obj(user)
- # we hash the users password to avoid saving it as plaintext in the db,
- # remove to use plain text:
- user.password = generate_password_hash(form.password.data)
- db.session.add(user)
- db.session.commit()
- login.login_user(user)
- return redirect(url_for('.index'))
- link = '<p>Already have an account? <a href="' + url_for('.login_view') + '">Click here to log in.</a></p>'
- self._template_args['form'] = form
- self._template_args['link'] = link
- return super(MyAdminIndexView, self).index()
- @expose('/logout/')
- def logout_view(self):
- login.logout_user()
- return redirect(url_for('.index'))
- # Flask views
- @app.route('/')
- def index():
- return render_template('index.html')
- # Initialize flask-login
- init_login()
- # Create admin
- admin = admin.Admin(app, 'Example: Auth', index_view=MyAdminIndexView(), base_template='my_master.html')
- # Add view
- admin.add_view(MyModelView(User, db.session))
- def build_sample_db():
- """
- Populate a small db with some example entries.
- """
- import string
- import random
- db.drop_all()
- db.create_all()
- # passwords are hashed, to use plaintext passwords instead:
- # test_user = User(login="test", password="test")
- test_user = User(login="test", password=generate_password_hash("test"))
- db.session.add(test_user)
- first_names = [
- 'Harry', 'Amelia', 'Oliver', 'Jack', 'Isabella', 'Charlie','Sophie', 'Mia',
- 'Jacob', 'Thomas', 'Emily', 'Lily', 'Ava', 'Isla', 'Alfie', 'Olivia', 'Jessica',
- 'Riley', 'William', 'James', 'Geoffrey', 'Lisa', 'Benjamin', 'Stacey', 'Lucy'
- ]
- last_names = [
- 'Brown', 'Smith', 'Patel', 'Jones', 'Williams', 'Johnson', 'Taylor', 'Thomas',
- 'Roberts', 'Khan', 'Lewis', 'Jackson', 'Clarke', 'James', 'Phillips', 'Wilson',
- 'Ali', 'Mason', 'Mitchell', 'Rose', 'Davis', 'Davies', 'Rodriguez', 'Cox', 'Alexander'
- ]
- for i in range(len(first_names)):
- user = User()
- user.first_name = first_names[i]
- user.last_name = last_names[i]
- user.login = user.first_name.lower()
- user.email = user.login + "@example.com"
- user.password = generate_password_hash(''.join(random.choice(string.ascii_lowercase + string.digits) for i in range(10)))
- db.session.add(user)
- db.session.commit()
- return
- if __name__ == '__main__':
- # Build a sample db on the fly, if one does not exist yet.
- app_dir = os.path.realpath(os.path.dirname(__file__))
- database_path = os.path.join(app_dir, app.config['DATABASE_FILE'])
- if not os.path.exists(database_path):
- build_sample_db()
- # Start app
- app.run(debug=True)
|