librarian
[wolnelektury.git] / src / search / management / commands / reindex.py
1 # -*- coding: utf-8 -*-
2 # This file is part of Wolnelektury, licensed under GNU Affero GPLv3 or later.
3 # Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
4 #
5 import sys
6 from django.core.management.base import BaseCommand
7
8 from optparse import make_option
9
10
11 def query_yes_no(question, default="yes"):
12     """Ask a yes/no question via raw_input() and return their answer.
13
14     "question" is a string that is presented to the user.
15     "default" is the presumed answer if the user just hits <Enter>.
16         It must be "yes" (the default), "no" or None (meaning
17         an answer is required of the user).
18
19     The "answer" return value is one of "yes" or "no".
20     """
21     valid = {"yes": True, "y": True, "ye": True,
22              "no": False, "n": False}
23     if default is None:
24         prompt = " [y/n] "
25     elif default == "yes":
26         prompt = " [Y/n] "
27     elif default == "no":
28         prompt = " [y/N] "
29     else:
30         raise ValueError("invalid default answer: '%s'" % default)
31
32     while True:
33         sys.stdout.write(question + prompt)
34         choice = raw_input().lower()
35         if default is not None and choice == '':
36             return valid[default]
37         elif choice in valid:
38             return valid[choice]
39         else:
40             sys.stdout.write("Please respond with 'yes' or 'no' (or 'y' or 'n').\n")
41
42
43 class Command(BaseCommand):
44     help = 'Reindex everything.'
45     args = ''
46     
47     option_list = BaseCommand.option_list + (
48         make_option('-n', '--book-id', action='store_true', dest='book_id', default=False,
49                     help='book id instead of slugs'),
50         make_option('-t', '--just-tags', action='store_true', dest='just_tags', default=False,
51                     help='just reindex tags'),
52     )
53
54     def handle(self, *args, **opts):
55         from catalogue.models import Book
56         from search.index import Index
57         idx = Index()
58         
59         if not opts['just_tags']:
60             if args:
61                 books = []
62                 for a in args:
63                     if opts['book_id']:
64                         books += Book.objects.filter(id=int(a)).all()
65                     else:
66                         books += Book.objects.filter(slug=a).all()
67             else:
68                 books = list(Book.objects.all())
69
70             while books:
71                 try:
72                     b = books[0]
73                     print b.title
74                     idx.index_book(b)
75                     idx.index.commit()
76                     books.pop(0)
77                 except Exception, e:
78                     print "Error occured: %s" % e
79                     try:
80                         # we might not be able to rollback
81                         idx.index.rollback()
82                     except:
83                         pass
84                     retry = query_yes_no("Retry?")
85                     if not retry:
86                         break
87
88         print 'Reindexing tags.'
89         idx.index_tags()
90         idx.index.commit()