2009-07-21 00:12:25 -07:00
|
|
|
# encoding: utf8
|
2009-08-16 21:03:49 -07:00
|
|
|
import os, os.path
|
2009-08-18 23:42:59 -07:00
|
|
|
import pkg_resources
|
2009-07-25 01:28:33 -07:00
|
|
|
import re
|
|
|
|
|
2009-07-21 00:12:25 -07:00
|
|
|
from sqlalchemy.sql import func
|
2009-07-25 01:28:33 -07:00
|
|
|
import whoosh
|
2009-08-16 21:03:49 -07:00
|
|
|
import whoosh.filedb.filestore
|
|
|
|
import whoosh.filedb.fileindex
|
|
|
|
import whoosh.index
|
2009-07-25 01:28:33 -07:00
|
|
|
from whoosh.qparser import QueryParser
|
|
|
|
import whoosh.spelling
|
2009-07-21 00:12:25 -07:00
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
from pokedex.db import connect
|
2009-07-21 00:12:25 -07:00
|
|
|
import pokedex.db.tables as tables
|
|
|
|
|
2009-07-25 01:28:33 -07:00
|
|
|
# Dictionary of table name => table class.
|
|
|
|
# Need the table name so we can get the class from the table name after we
|
|
|
|
# retrieve something from the index
|
|
|
|
indexed_tables = {}
|
|
|
|
for cls in [
|
2009-07-26 23:18:11 -07:00
|
|
|
tables.Ability,
|
|
|
|
tables.Item,
|
|
|
|
tables.Move,
|
2009-07-25 01:28:33 -07:00
|
|
|
tables.Pokemon,
|
2009-07-26 23:18:11 -07:00
|
|
|
tables.Type,
|
2009-07-25 01:28:33 -07:00
|
|
|
]:
|
|
|
|
indexed_tables[cls.__tablename__] = cls
|
|
|
|
|
2009-07-26 00:37:37 -07:00
|
|
|
# Dictionary of extra keys to file types of objects under, e.g. Pokémon can
|
|
|
|
# also be looked up purely by number
|
|
|
|
extra_keys = {
|
2009-07-26 23:18:11 -07:00
|
|
|
tables.Move: [
|
|
|
|
lambda row: u"move %d" % row.id,
|
|
|
|
],
|
2009-07-26 00:37:37 -07:00
|
|
|
tables.Pokemon: [
|
|
|
|
lambda row: unicode(row.id),
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
def open_index(directory=None, session=None, recreate=False):
|
|
|
|
"""Opens the whoosh index stored in the named directory and returns (index,
|
|
|
|
speller). If the index doesn't already exist, it will be created.
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
`directory`
|
|
|
|
Directory containing the index. Defaults to a location within the
|
|
|
|
`pokedex` egg directory.
|
|
|
|
|
|
|
|
`session`
|
|
|
|
If the index needs to be created, this database session will be used.
|
|
|
|
Defaults to an attempt to connect to the default SQLite database
|
|
|
|
installed by `pokedex setup`.
|
|
|
|
|
|
|
|
`recreate`
|
|
|
|
If set to True, the whoosh index will be created even if it already
|
|
|
|
exists.
|
2009-07-25 01:28:33 -07:00
|
|
|
"""
|
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
# Defaults
|
|
|
|
if not directory:
|
|
|
|
directory = pkg_resources.resource_filename('pokedex',
|
|
|
|
'data/whoosh_index')
|
|
|
|
|
|
|
|
if not session:
|
|
|
|
session = connect()
|
|
|
|
|
|
|
|
# Attempt to open or create the index
|
|
|
|
directory_exists = os.path.exists(directory)
|
|
|
|
if directory_exists and not recreate:
|
|
|
|
# Already exists; should be an index!
|
|
|
|
try:
|
|
|
|
index = whoosh.index.open_dir(directory, indexname='pokedex')
|
|
|
|
speller = whoosh.index.open_dir(directory, indexname='spelling')
|
|
|
|
return index, speller
|
|
|
|
except whoosh.index.EmptyIndexError as e:
|
|
|
|
# Apparently not a real index. Fall out of the if and create it
|
|
|
|
pass
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
if not directory_exists:
|
|
|
|
os.mkdir(directory)
|
|
|
|
|
|
|
|
|
|
|
|
# Create index
|
2009-07-25 01:28:33 -07:00
|
|
|
schema = whoosh.fields.Schema(
|
|
|
|
name=whoosh.fields.ID(stored=True),
|
|
|
|
table=whoosh.fields.STORED,
|
|
|
|
row_id=whoosh.fields.STORED,
|
2009-08-16 21:03:49 -07:00
|
|
|
language=whoosh.fields.STORED,
|
|
|
|
|
|
|
|
# Whoosh 0.2 explodes when using a file-stored schema with no TEXT
|
|
|
|
# columns. Appease it
|
|
|
|
dummy=whoosh.fields.TEXT,
|
2009-07-25 01:28:33 -07:00
|
|
|
)
|
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
index = whoosh.index.create_in(directory, schema=schema,
|
|
|
|
indexname='pokedex')
|
2009-07-25 01:28:33 -07:00
|
|
|
writer = index.writer()
|
|
|
|
|
|
|
|
# Index every name in all our tables of interest
|
2009-07-26 00:37:37 -07:00
|
|
|
speller_entries = []
|
2009-07-25 01:28:33 -07:00
|
|
|
for cls in indexed_tables.values():
|
|
|
|
q = session.query(cls)
|
|
|
|
|
|
|
|
# Only index base Pokémon formes
|
|
|
|
if hasattr(cls, 'forme_base_pokemon_id'):
|
|
|
|
q = q.filter_by(forme_base_pokemon_id=None)
|
|
|
|
|
|
|
|
for row in q.yield_per(5):
|
2009-07-26 00:37:37 -07:00
|
|
|
row_key = dict(table=cls.__tablename__, row_id=row.id)
|
|
|
|
|
|
|
|
# Spelling index only indexes strings of letters, alas, so we
|
|
|
|
# reduce every name to this to make the index work. However, exact
|
|
|
|
# matches are not returned, so e.g. 'nidoran' would neither match
|
|
|
|
# exactly nor fuzzy-match. Solution: add the spelling-munged name
|
|
|
|
# as a regular index row too.
|
2009-07-25 01:28:33 -07:00
|
|
|
name = row.name.lower()
|
2009-07-26 00:37:37 -07:00
|
|
|
writer.add_document(name=name, **row_key)
|
|
|
|
|
|
|
|
speller_entries.append(name)
|
|
|
|
|
2009-07-26 23:18:11 -07:00
|
|
|
for extra_key_func in extra_keys.get(cls, []):
|
2009-07-26 00:37:37 -07:00
|
|
|
extra_key = extra_key_func(row)
|
|
|
|
writer.add_document(name=extra_key, **row_key)
|
2009-07-25 01:28:33 -07:00
|
|
|
|
|
|
|
writer.commit()
|
|
|
|
|
2009-08-16 21:03:49 -07:00
|
|
|
# XXX GIHWEGREHKG
|
|
|
|
old__schema = whoosh.spelling.SpellChecker._schema
|
|
|
|
def new__schema(self):
|
|
|
|
schema = old__schema(self)
|
|
|
|
schema.add('dummy', whoosh.fields.TEXT)
|
|
|
|
return schema
|
|
|
|
whoosh.spelling.SpellChecker._schema = new__schema
|
|
|
|
|
2009-07-26 00:37:37 -07:00
|
|
|
# Construct and populate a spell-checker index. Quicker to do it all
|
|
|
|
# at once, as every call to add_* does a commit(), and those seem to be
|
|
|
|
# expensive
|
2009-08-18 23:42:59 -07:00
|
|
|
speller = whoosh.spelling.SpellChecker(index.storage, indexname='spelling')
|
2009-07-26 00:37:37 -07:00
|
|
|
# WARNING: HERE BE DRAGONS
|
|
|
|
# whoosh.spelling refuses to index things that don't look like words.
|
|
|
|
# Unfortunately, this doesn't work so well for Pokémon (Mr. Mime,
|
|
|
|
# Porygon-Z, etc.), and attempts to work around it lead to further
|
|
|
|
# complications.
|
|
|
|
# The below is copied from SpellChecker.add_scored_words without the check
|
|
|
|
# for isalpha(). XXX get whoosh patched to make this unnecessary!
|
2009-08-16 21:03:49 -07:00
|
|
|
writer = speller.index(create=True).writer()
|
2009-07-26 00:37:37 -07:00
|
|
|
for word in speller_entries:
|
|
|
|
fields = {"word": word, "score": 1}
|
|
|
|
for size in xrange(speller.mingram, speller.maxgram + 1):
|
|
|
|
nga = whoosh.analysis.NgramAnalyzer(size)
|
|
|
|
gramlist = [t.text for t in nga(word)]
|
|
|
|
if len(gramlist) > 0:
|
|
|
|
fields["start%s" % size] = gramlist[0]
|
|
|
|
fields["end%s" % size] = gramlist[-1]
|
|
|
|
fields["gram%s" % size] = " ".join(gramlist)
|
|
|
|
writer.add_document(**fields)
|
|
|
|
writer.commit()
|
|
|
|
# end copy-pasta
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
return index, speller
|
|
|
|
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
def lookup(name, session=None, exact_only=False):
|
2009-07-21 00:12:25 -07:00
|
|
|
"""Attempts to find some sort of object, given a database session and name.
|
|
|
|
|
2009-07-25 01:28:33 -07:00
|
|
|
Returns (objects, exact) where `objects` is a list of database objects, and
|
|
|
|
`exact` is True iff the given name matched the returned objects exactly.
|
2009-07-21 00:12:25 -07:00
|
|
|
|
2009-07-25 01:28:33 -07:00
|
|
|
This function ONLY does fuzzy matching if there are no exact matches.
|
2009-07-21 00:12:25 -07:00
|
|
|
|
|
|
|
Formes are not returned; "Shaymin" will return only grass Shaymin.
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-07-21 00:12:25 -07:00
|
|
|
Currently recognizes:
|
|
|
|
- Pokémon names: "Eevee"
|
2009-08-18 23:42:59 -07:00
|
|
|
|
|
|
|
`name`
|
|
|
|
Name of the thing to look for.
|
|
|
|
|
|
|
|
`session`
|
|
|
|
A database session to use for retrieving objects. As with get_index,
|
|
|
|
if this is not provided, a connection to the default database will be
|
|
|
|
attempted.
|
|
|
|
|
|
|
|
`exact_only`
|
|
|
|
If True, only exact matches are returned. If set to False (the
|
|
|
|
default), and the provided `name` doesn't match anything exactly,
|
|
|
|
spelling correction will be attempted.
|
2009-07-21 00:12:25 -07:00
|
|
|
"""
|
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
if not session:
|
|
|
|
session = connect()
|
|
|
|
|
|
|
|
index, speller = open_index()
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-08-18 23:42:59 -07:00
|
|
|
exact = True
|
2009-07-25 01:28:33 -07:00
|
|
|
|
2009-07-26 00:37:37 -07:00
|
|
|
# Look for exact name. A Term object does an exact match, so we don't have
|
|
|
|
# to worry about a query parser tripping on weird characters in the input
|
|
|
|
searcher = index.searcher()
|
2009-07-27 18:45:43 -07:00
|
|
|
query = whoosh.query.Term('name', name.lower())
|
2009-07-26 00:37:37 -07:00
|
|
|
results = searcher.search(query)
|
2009-07-25 01:28:33 -07:00
|
|
|
|
|
|
|
if not exact_only:
|
|
|
|
# Look for some fuzzy matches
|
|
|
|
if not results:
|
|
|
|
exact = False
|
2009-07-26 00:37:37 -07:00
|
|
|
results = []
|
2009-07-25 01:28:33 -07:00
|
|
|
|
|
|
|
for suggestion in speller.suggest(name, 3):
|
2009-07-26 00:37:37 -07:00
|
|
|
query = whoosh.query.Term('name', suggestion)
|
|
|
|
results.extend(searcher.search(query))
|
2009-07-25 01:28:33 -07:00
|
|
|
|
|
|
|
# Convert results to db objects
|
|
|
|
objects = []
|
|
|
|
seen = {}
|
|
|
|
for result in results:
|
|
|
|
# Skip dupe results
|
|
|
|
seen_key = result['table'], result['row_id']
|
|
|
|
if seen_key in seen:
|
|
|
|
continue
|
|
|
|
seen[seen_key] = True
|
|
|
|
|
|
|
|
cls = indexed_tables[result['table']]
|
|
|
|
obj = session.query(cls).get(result['row_id'])
|
|
|
|
objects.append(obj)
|
2009-07-21 00:12:25 -07:00
|
|
|
|
2009-07-25 01:28:33 -07:00
|
|
|
return objects, exact
|