From 8a27076ae4c2b9faf743e71b113b889b318625d5 Mon Sep 17 00:00:00 2001 From: Miguel Grinberg Date: Sun, 16 Dec 2012 00:35:16 -0800 Subject: [PATCH] full text search --- app/forms.py | 2 ++ app/models.py | 8 +++++++- app/templates/base.html | 1 + app/templates/search_results.html | 9 +++++++++ app/views.py | 23 +++++++++++++++++++++-- config.py | 3 ++- 6 files changed, 42 insertions(+), 4 deletions(-) create mode 100644 app/templates/search_results.html diff --git a/app/forms.py b/app/forms.py index 42b6b11..a39fe06 100644 --- a/app/forms.py +++ b/app/forms.py @@ -28,3 +28,5 @@ class EditForm(Form): class PostForm(Form): post = TextField('post', validators = [Required()]) +class SearchForm(Form): + search = TextField('search', validators = [Required()]) diff --git a/app/models.py b/app/models.py index f11e92e..3b1fd48 100644 --- a/app/models.py +++ b/app/models.py @@ -1,5 +1,7 @@ from hashlib import md5 from app import db +from app import app +import flask.ext.whooshalchemy as whooshalchemy ROLE_USER = 0 ROLE_ADMIN = 1 @@ -71,10 +73,14 @@ class User(db.Model): return '' % (self.nickname) class Post(db.Model): + __searchable__ = ['body'] + id = db.Column(db.Integer, primary_key = True) body = db.Column(db.String(140)) timestamp = db.Column(db.DateTime) user_id = db.Column(db.Integer, db.ForeignKey('user.id')) def __repr__(self): - return '' % (self.text) \ No newline at end of file + return '' % (self.body) + +whooshalchemy.whoosh_index(app, Post) diff --git a/app/templates/base.html b/app/templates/base.html index c0f2e7b..f73a146 100644 --- a/app/templates/base.html +++ b/app/templates/base.html @@ -11,6 +11,7 @@ Home {% if g.user.is_authenticated() %} | Your Profile + |
{{g.search_form.hidden_tag()}}{{g.search_form.search(size=20)}}
| Logout {% endif %} diff --git a/app/templates/search_results.html b/app/templates/search_results.html new file mode 100644 index 0000000..96f0824 --- /dev/null +++ b/app/templates/search_results.html @@ -0,0 +1,9 @@ + +{% extends "base.html" %} + +{% block content %} +

Search results for "{{query}}":

+{% for post in results %} + {% include 'post.html' %} +{% endfor %} +{% endblock %} \ No newline at end of file diff --git a/app/views.py b/app/views.py index 12fdeb9..e47363b 100644 --- a/app/views.py +++ b/app/views.py @@ -1,10 +1,10 @@ from flask import render_template, flash, redirect, session, url_for, request, g from flask.ext.login import login_user, logout_user, current_user, login_required from app import app, db, lm, oid -from forms import LoginForm, EditForm, PostForm +from forms import LoginForm, EditForm, PostForm, SearchForm from models import User, ROLE_USER, ROLE_ADMIN, Post from datetime import datetime -from config import POSTS_PER_PAGE +from config import POSTS_PER_PAGE, MAX_SEARCH_RESULTS @lm.user_loader def load_user(id): @@ -17,6 +17,7 @@ def before_request(): g.user.last_seen = datetime.utcnow() db.session.add(g.user) db.session.commit() + g.search_form = SearchForm() @app.errorhandler(404) def internal_error(error): @@ -118,6 +119,7 @@ def edit(): form = form) @app.route('/follow/') +@login_required def follow(nickname): user = User.query.filter_by(nickname = nickname).first() if user == None: @@ -136,6 +138,7 @@ def follow(nickname): return redirect(url_for('user', nickname = nickname)) @app.route('/unfollow/') +@login_required def unfollow(nickname): user = User.query.filter_by(nickname = nickname).first() if user == None: @@ -152,3 +155,19 @@ def unfollow(nickname): db.session.commit() flash('You have stopped following ' + nickname + '.') return redirect(url_for('user', nickname = nickname)) + +@app.route('/search', methods = ['POST']) +@login_required +def search(): + if not g.search_form.validate_on_submit(): + return redirect(url_for('index')) + return redirect(url_for('search_results', query = g.search_form.search.data)) + +@app.route('/search_results/') +@login_required +def search_results(query): + results = Post.query.whoosh_search(query, MAX_SEARCH_RESULTS).all() + return render_template('search_results.html', + query = query, + results = results) + diff --git a/config.py b/config.py index 6de81f1..9adb163 100644 --- a/config.py +++ b/config.py @@ -13,6 +13,7 @@ OPENID_PROVIDERS = [ SQLALCHEMY_DATABASE_URI = 'sqlite:///' + os.path.join(basedir, 'app.db') SQLALCHEMY_MIGRATE_REPO = os.path.join(basedir, 'db_repository') +WHOOSH_BASE = os.path.join(basedir, 'search.db') # mail server settings MAIL_SERVER = 'localhost' @@ -25,4 +26,4 @@ ADMINS = ['you@example.com'] # pagination POSTS_PER_PAGE = 3 - +MAX_SEARCH_RESULTS = 50