search pass three
This commit is contained in:
		
							parent
							
								
									13d4be8937
								
							
						
					
					
						commit
						08a4c71c10
					
				
					 9 changed files with 132 additions and 44 deletions
				
			
		
							
								
								
									
										36
									
								
								realms/modules/search/commands.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								realms/modules/search/commands.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,36 @@ | |||
| import click | ||||
| from realms import create_app, search | ||||
| from realms.modules.wiki.models import Wiki | ||||
| from realms.lib.util import filename_to_cname | ||||
| 
 | ||||
| 
 | ||||
| @click.group(short_help="Search Module") | ||||
| def cli(): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| @cli.command() | ||||
| def rebuild_index(): | ||||
|     """ Rebuild search index | ||||
|     """ | ||||
|     app = create_app() | ||||
| 
 | ||||
|     if app.config.get('SEARCH_TYPE') == 'simple': | ||||
|         click.echo("Search type is simple, try using elasticsearch.") | ||||
|         return | ||||
| 
 | ||||
|     with app.app_context(): | ||||
|         # Wiki | ||||
|         search.delete_index('wiki') | ||||
|         wiki = Wiki(app.config['WIKI_PATH']) | ||||
|         for entry in wiki.get_index(): | ||||
|             page = wiki.get_page(entry['name']) | ||||
|             name = filename_to_cname(page['name']) | ||||
|             # TODO add email? | ||||
|             body = dict(name=name, | ||||
|                         content=page['data'], | ||||
|                         message=page['info']['message'], | ||||
|                         username=page['info']['author'], | ||||
|                         updated_on=entry['mtime'], | ||||
|                         created_on=entry['ctime']) | ||||
|             search.index_wiki(name, body) | ||||
|  | @ -1,15 +1,21 @@ | |||
| from realms.modules.wiki.models import Wiki | ||||
| from realms.modules.search.models import Search | ||||
| from realms import search | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| @Wiki.after('write_page') | ||||
| def wiki_write_page(name, content, message=None, username=None, email=None, **kwargs): | ||||
| 
 | ||||
|     if not hasattr(search, 'index_wiki'): | ||||
|         # using simple search or none | ||||
|         return | ||||
| 
 | ||||
|     body = dict(name=name, | ||||
|                 content=content, | ||||
|                 message=message, | ||||
|                 email=email, | ||||
|                 username=username) | ||||
|     return Search.index('wiki', 'page', id_=name, body=body) | ||||
|     return search.index_wiki(name, body) | ||||
| 
 | ||||
| 
 | ||||
| @Wiki.after('rename_page') | ||||
|  |  | |||
|  | @ -1,26 +1,71 @@ | |||
| from realms import elastic | ||||
| from realms.lib.model import HookMixin | ||||
| from flask import g, current_app | ||||
| from realms.lib.util import filename_to_cname | ||||
| 
 | ||||
| 
 | ||||
| class Search(HookMixin): | ||||
| def simple(app): | ||||
|     return SimpleSearch() | ||||
| 
 | ||||
|     @classmethod | ||||
|     def index(cls, index, doc_type, id_=None, body=None): | ||||
|         return elastic.index(index=index, doc_type=doc_type, id=id_, body=body) | ||||
| 
 | ||||
|     @classmethod | ||||
|     def wiki(cls, query): | ||||
| def elasticsearch(app): | ||||
|     from flask.ext.elastic import Elastic | ||||
|     return ElasticSearch(Elastic(app)) | ||||
| 
 | ||||
| 
 | ||||
| class Search(object): | ||||
|     def __init__(self, app=None): | ||||
|         if app is not None: | ||||
|             self.init_app(app) | ||||
| 
 | ||||
|     def init_app(self, app): | ||||
|         search_obj = globals()[app.config['SEARCH_TYPE']] | ||||
|         app.extensions['search'] = search_obj(app) | ||||
| 
 | ||||
|     def __getattr__(self, item): | ||||
|         return getattr(current_app.extensions['search'], item) | ||||
| 
 | ||||
| 
 | ||||
| class BaseSearch(): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class SimpleSearch(BaseSearch): | ||||
|     def wiki(self, query): | ||||
|         res = [] | ||||
|         for entry in g.current_wiki.get_index(): | ||||
|             name = filename_to_cname(entry['name']) | ||||
|             if query in name.split('-'): | ||||
|                 page = g.current_wiki.get_page(name) | ||||
|                 res.append(dict(name=name, content=page['data'])) | ||||
|         return res | ||||
| 
 | ||||
|     def users(self, query): | ||||
|         pass | ||||
| 
 | ||||
| 
 | ||||
| class ElasticSearch(BaseSearch): | ||||
|     def __init__(self, elastic): | ||||
|         self.elastic = elastic | ||||
| 
 | ||||
|     def index(self, index, doc_type, id_=None, body=None): | ||||
|         return self.elastic.index(index=index, doc_type=doc_type, id=id_, body=body) | ||||
| 
 | ||||
|     def index_wiki(self, name, body): | ||||
|         self.index('wiki', 'page', id_=name, body=body) | ||||
| 
 | ||||
|     def delete_index(self, index): | ||||
|         return self.elastic.indices.delete(index=index, ignore=[400, 404]) | ||||
| 
 | ||||
|     def wiki(self, query): | ||||
|         if not query: | ||||
|             return [] | ||||
| 
 | ||||
|         res = elastic.search(index='wiki', body={"query": { | ||||
|         res = self.elastic.search(index='wiki', body={"query": { | ||||
|             "multi_match": { | ||||
|                 "query": query, | ||||
|                 "fields": ["name^3", "content"] | ||||
|                 "fields": ["name"] | ||||
|             }}}) | ||||
| 
 | ||||
|         return [hit["_source"] for hit in res['hits']['hits']] | ||||
| 
 | ||||
|     @classmethod | ||||
|     def users(cls, query): | ||||
|         pass | ||||
|     def users(self, query): | ||||
|         pass | ||||
|  |  | |||
|  | @ -1,10 +1,10 @@ | |||
| from flask import abort, g, render_template, request, redirect, Blueprint, flash, url_for, current_app | ||||
| from .models import Search | ||||
| from realms import search as search_engine | ||||
| 
 | ||||
| blueprint = Blueprint('search', __name__) | ||||
| 
 | ||||
| 
 | ||||
| @blueprint.route('/_search') | ||||
| def search(): | ||||
|     results = Search.wiki(request.args.get('q')) | ||||
|     results = search_engine.wiki(request.args.get('q')) | ||||
|     return render_template('search/search.html', results=results) | ||||
|  | @ -5,32 +5,11 @@ import gittle.utils | |||
| import yaml | ||||
| from gittle import Gittle | ||||
| from dulwich.repo import NotGitRepository | ||||
| from realms.lib.util import to_canonical | ||||
| from realms.lib.util import to_canonical, cname_to_filename, filename_to_cname | ||||
| from realms import cache | ||||
| from realms.lib.hook import HookMixin | ||||
| 
 | ||||
| 
 | ||||
| def cname_to_filename(cname): | ||||
|     """ Convert canonical name to filename | ||||
| 
 | ||||
|     :param cname: Canonical name | ||||
|     :return: str -- Filename | ||||
| 
 | ||||
|     """ | ||||
|     return cname + ".md" | ||||
| 
 | ||||
| 
 | ||||
| def filename_to_cname(filename): | ||||
|     """Convert filename to canonical name. | ||||
| 
 | ||||
|     .. note:: | ||||
| 
 | ||||
|     It's assumed filename is already canonical format | ||||
| 
 | ||||
|     """ | ||||
|     return os.path.splitext(filename)[0] | ||||
| 
 | ||||
| 
 | ||||
| class PageNotFound(Exception): | ||||
|     pass | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,7 +1,7 @@ | |||
| import json | ||||
| from nose.tools import * | ||||
| from flask import url_for | ||||
| from realms.modules.wiki.models import cname_to_filename, filename_to_cname | ||||
| from realms.lib.util import cname_to_filename, filename_to_cname | ||||
| from realms.lib.test import BaseTest | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue