# -*- coding: utf-8 -*- # Copyright: Damien Elmes # License: GNU GPL, version 3 or later; http://www.gnu.org/copyleft/gpl.html import time, os, random, re, stat, simplejson from anki.lang import _, ngettext from anki.utils import parseTags, tidyHTML, ids2str, hexifyID, \ checksum, fieldChecksum, addTags, deleteTags, stripHTML, intTime, \ splitFields from anki.hooks import runHook, runFilter from anki.sched import Scheduler from anki.media import MediaRegistry from anki.consts import * import anki.latex # sets up hook import anki.cards, anki.facts, anki.models, anki.template # Settings related to queue building. These may be loaded without the rest of # the config to check due counts faster on mobile clients. defaultQconf = { 'revGroups': [], 'newGroups': [], 'newPerDay': 20, 'newToday': [0, 0], # currentDay, count 'newTodayOrder': NEW_TODAY_ORD, 'newCardOrder': 1, 'newCardSpacing': NEW_CARDS_DISTRIBUTE, 'revCardOrder': REV_CARDS_RANDOM, } # scheduling and other options defaultConf = { 'collapseTime': 600, 'sessionRepLimit': 0, 'sessionTimeLimit': 600, 'currentModelId': None, 'currentGroupId': 1, 'nextFid': 1, 'nextCid': 1, 'nextGid': 2, 'mediaURL': "", 'latexPre': """\ \\documentclass[12pt]{article} \\special{papersize=3in,5in} \\usepackage[utf8]{inputenc} \\usepackage{amssymb,amsmath} \\pagestyle{empty} \\setlength{\\parindent}{0in} \\begin{document} """, 'latexPost': "\\end{document}", 'fontFamilies': [ [u'MS 明朝',u'ヒラギノ明朝 Pro W3',u'Kochi Mincho', u'東風明朝'] ] } # this is initialized by storage.Deck class _Deck(object): # fixme: make configurable? factorFour = 1.3 def __init__(self, db): self.db = db self.path = db._path self.load() if self.utcOffset == -2: # shared deck; reset timezone and creation date self.utcOffset = time.timezone + 60*60*4 self.crt = intTime() self.undoEnabled = False self.sessionStartReps = 0 self.sessionStartTime = 0 self.lastSessionStart = 0 # counter for reps since deck open self.reps = 0 self.sched = Scheduler(self) self.media = MediaRegistry(self) def name(self): n = os.path.splitext(os.path.basename(self.path))[0] return n # DB-related ########################################################################## def load(self): (self.crt, self.mod, self.schema, self.syncName, self.lastSync, self.utcOffset, self.qconf, self.conf, self.data) = self.db.first(""" select crt, mod, schema, syncName, lastSync, utcOffset, qconf, conf, data from deck""") self.qconf = simplejson.loads(self.qconf) self.conf = simplejson.loads(self.conf) self.data = simplejson.loads(self.data) def flush(self): "Flush state to DB, updating mod time." self.mod = intTime() self.db.execute( """update deck set mod=?, schema=?, syncName=?, lastSync=?, utcOffset=?, qconf=?, conf=?, data=?""", self.mod, self.schema, self.syncName, self.lastSync, self.utcOffset, simplejson.dumps(self.qconf), simplejson.dumps(self.conf), simplejson.dumps(self.data)) def save(self): "Flush, then commit DB." self.flush() self.db.commit() def close(self, save=True): "Disconnect from DB." if self.db: if save: self.save() else: self.rollback() self.db.close() self.db = None runHook("deckClosed", self) def reopen(self): "Reconnect to DB (after changing threads, etc). Doesn't reload." import anki.db if not self.db: self.db = anki.db.DB(self.path) def rollback(self): self.db.rollback() def modSchema(self): if not self.schemaDirty(): # next sync will be full self.emptyTrash() self.schema = intTime() def schemaDirty(self): "True if schema changed since last sync, or syncing off." return self.schema > self.lastSync # Object creation helpers ########################################################################## def getCard(self, id): return anki.cards.Card(self, id) def getFact(self, id): return anki.facts.Fact(self, id=id) def getTemplate(self, id): return anki.models.Template(self, self.deck.db.first( "select * from templates where id = ?", id)) # unsorted ########################################################################## def nextID(self, type): type = "next"+type.capitalize() id = self.conf.get(type, 1) self.conf[type] = id+1 return id def reset(self): self.sched.reset() # Times ########################################################################## def nextDueMsg(self): next = self.earliestTime() if next: # all new cards except suspended newCount = self.newCardsDueBy(self.dayCutoff + 86400) newCardsTomorrow = min(newCount, self.newCardsPerDay) cards = self.cardsDueBy(self.dayCutoff + 86400) msg = _('''\ At this time tomorrow:
%(wait)s
%(new)s''') % { 'new': ngettext("There will be %d new card.", "There will be %d new cards.", newCardsTomorrow) % newCardsTomorrow, 'wait': ngettext("There will be %s review.", "There will be %s reviews.", cards) % cards, } if next > (self.dayCutoff+86400) and not newCardsTomorrow: msg = (_("The next review is in %s.") % self.earliestTimeStr()) else: msg = _("No cards are due.") return msg def earliestTime(self): """Return the time of the earliest card. This may be in the past if the deck is not finished. If the deck has no (enabled) cards, return None. Ignore new cards.""" earliestRev = self.db.scalar(self.cardLimit("revActive", "revInactive", """ select due from cards c where queue = 1 order by due limit 1""")) earliestFail = self.db.scalar(self.cardLimit("revActive", "revInactive", """ select due+%d from cards c where queue = 0 order by due limit 1""" % self.delay0)) if earliestRev and earliestFail: return min(earliestRev, earliestFail) elif earliestRev: return earliestRev else: return earliestFail def earliestTimeStr(self, next=None): """Return the relative time to the earliest card as a string.""" if next == None: next = self.earliestTime() if not next: return _("unknown") diff = next - time.time() return anki.utils.fmtTimeSpan(diff) def cardsDueBy(self, time): "Number of cards due at TIME. Ignore new cards" return self.db.scalar( self.cardLimit( "revActive", "revInactive", "select count(*) from cards c where queue between 0 and 1 " "and due < :lim"), lim=time) def newCardsDueBy(self, time): "Number of new cards due at TIME." return self.db.scalar( self.cardLimit( "newActive", "newInactive", "select count(*) from cards c where queue = 2 " "and due < :lim"), lim=time) def deckFinishedMsg(self): spaceSusp = "" c= self.spacedCardCount() if c: spaceSusp += ngettext( 'There is %d delayed card.', 'There are %d delayed cards.', c) % c c2 = self.hiddenCards() if c2: if spaceSusp: spaceSusp += "
" spaceSusp += _( "Some cards are inactive or suspended.") if spaceSusp: spaceSusp = "

" + spaceSusp return _('''\

Congratulations!

You have finished for now.

%(next)s %(spaceSusp)s
''') % { "next": self.nextDueMsg(), "spaceSusp": spaceSusp, } # Suspending ########################################################################## def suspendCards(self, ids): "Suspend cards." self.startProgress() self.db.execute(""" update cards set queue = -1, mod = :t where id in %s""" % ids2str(ids), t=time.time()) self.finishProgress() def unsuspendCards(self, ids): "Unsuspend cards." self.startProgress() self.db.execute(""" update cards set queue = type, mod=:t where queue = -1 and id in %s""" % ids2str(ids), t=time.time()) self.finishProgress() def buryFact(self, fact): "Bury all cards for fact until next session." for card in fact.cards: if card.queue in (0,1,2): card.queue = -2 # Counts ########################################################################## def hiddenCards(self): "Assumes queue finished. True if some due cards have not been shown." return self.db.scalar(""" select 1 from cards where due < :now and queue between 0 and 1 limit 1""", now=self.dayCutoff) def spacedCardCount(self): "Number of spaced cards." print "spacedCardCount" return 0 return self.db.scalar(""" select count(cards.id) from cards where due > :now and due < :now""", now=time.time()) def isEmpty(self): return not self.cardCount def matureCardCount(self): return self.db.scalar( "select count(id) from cards where interval >= :t ", t=MATURE_THRESHOLD) def youngCardCount(self): return self.db.scalar( "select count(id) from cards where interval < :t " "and reps != 0", t=MATURE_THRESHOLD) def newCountAll(self): "All new cards, including spaced." return self.db.scalar( "select count(id) from cards where type = 2") def seenCardCount(self): return self.db.scalar( "select count(id) from cards where type between 0 and 1") # Card predicates ########################################################################## def cardState(self, card): if self.cardIsNew(card): return "new" elif card.interval > MATURE_THRESHOLD: return "mature" return "young" def cardIsNew(self, card): "True if a card has never been seen before." return card.reps == 0 def cardIsYoung(self, card): "True if card is not new and not mature." return (not self.cardIsNew(card) and not self.cardIsMature(card)) def cardIsMature(self, card): return card.interval >= MATURE_THRESHOLD # Stats ########################################################################## def getETA(self, stats): # rev + new cards first, account for failures import traceback; traceback.print_stack() count = stats['rev'] + stats['new'] count *= 1 + stats['gYoungNo%'] / 100.0 left = count * stats['dAverageTime'] # failed - higher time per card for higher amount of cards failedBaseMulti = 1.5 failedMod = 0.07 failedBaseCount = 20 factor = (failedBaseMulti + (failedMod * (stats['failed'] - failedBaseCount))) left += stats['failed'] * stats['dAverageTime'] * factor return left # Facts ########################################################################## def factCount(self): return self.db.scalar("select count() from facts") def newFact(self): "Return a new fact with the current model." return anki.facts.Fact(self, self.currentModel()) def addFact(self, fact, gid=1): "Add a fact to the deck. Return number of new cards." # check we have card models available cms = self.findTemplates(fact) if not cms: return None # flush the fact fact.id = self.nextID("fid") fact.flush() # notice any new tags self.registerTags(fact.tags) # if random mode, determine insertion point isRandom = self.qconf['newCardOrder'] == NEW_CARDS_RANDOM if isRandom: due = random.randrange(0, 1000000) # add cards ncards = 0 for template in cms: card = anki.cards.Card(self) card.id = self.nextID("cid") card.fid = fact.id card.ord = template['ord'] card.gid = template['gid'] or gid if isRandom: card.due = due else: card.due = fact.id card.flush() ncards += 1 return ncards def findTemplates(self, fact, checkActive=True): "Return active, non-empty templates." ok = [] for c, template in enumerate(fact.model.templates): if template['actv'] or not checkActive: # [cid, fid, mid, gid, ord, tags, flds, data] data = [1, 1, fact.model.id, 1, c, "", fact.joinedFields(), ""] now = self._renderQA(fact.model, "", data) data[6] = "\x1f".join([""]*len(fact._fields)) empty = self._renderQA(fact.model, "", data) if now['q'] == empty['q']: continue if not template['emptyAns']: if now['a'] == empty['a']: continue # add ordinal template['ord'] = c ok.append(template) return ok def addCards(self, fact, tids): ids = [] for template in self.findTemplates(fact, False): if template.id not in tids: continue if self.db.scalar(""" select count(id) from cards where fid = :fid and tid = :cmid""", fid=fact.id, cmid=template.id) == 0: # enough for 10 card models assuming 0.00001 timer precision card = anki.cards.Card( fact, template, fact.created+0.0001*template.ord) raise Exception("incorrect; not checking selective study") self.newAvail += 1 ids.append(card.id) if ids: fact.setMod(textChanged=True, deck=self) self.setMod() return ids def factIsInvalid(self, fact): "True if existing fact is invalid. Returns the error." try: fact.assertValid() fact.assertUnique(self.db) except FactInvalidError, e: return e def factUseCount(self, fid): "Return number of cards referencing a given fact id." return self.db.scalar("select count(id) from cards where fid = :id", id=fid) def _deleteFacts(self, ids): "Bulk delete facts by ID. Don't call this directly." if not ids: return strids = ids2str(ids) self.db.execute("delete from facts where id in %s" % strids) self.db.execute("delete from fsums where fid in %s" % strids) def _deleteDanglingFacts(self): "Delete any facts without cards. Don't call this directly." ids = self.db.list(""" select id from facts where id not in (select distinct fid from cards)""") self._deleteFacts(ids) return ids def previewFact(self, oldFact, cms=None): "Duplicate fact and generate cards for preview. Don't add to deck." # check we have card models available if cms is None: cms = self.findTemplates(oldFact, checkActive=True) if not cms: return [] fact = self.cloneFact(oldFact) # proceed cards = [] for template in cms: card = anki.cards.Card(fact, template) cards.append(card) fact.setMod(textChanged=True, deck=self, media=False) return cards def cloneFact(self, oldFact): "Copy fact into new session." model = self.db.query(Model).get(oldFact.model.id) fact = self.newFact(model) for field in fact.fdata: fact[field.name] = oldFact[field.name] fact._tags = oldFact._tags return fact # Cards ########################################################################## def cardCount(self): all = self.db.scalar("select count() from cards") trash = self.db.scalar("select count() from cards where queue = -4") return all - trash def deleteCard(self, id): "Delete a card given its id. Delete any unused facts." self.deleteCards([id]) def deleteCards(self, ids): "Bulk delete cards by ID." if not ids: return sids = ids2str(ids) self.startProgress() if self.schemaDirty(): # immediate delete? self.db.execute("delete from cards where id in %s" % sids) # remove any dangling facts self._deleteDanglingFacts() else: # trash sfids = ids2str( self.db.list("select fid from cards where id in "+sids)) # need to handle delete of fsums/revlog remotely after sync self.db.execute( "update cards set crt = 0, mod = ? where id in "+sids, intTime()) self.db.execute( "update facts set crt = 0, mod = ? where id in "+sfids, intTime()) self.db.execute("delete from fsums where fid in "+sfids) self.db.execute("delete from revlog where cid in "+sids) self.finishProgress() def emptyTrash(self): self.db.executescript(""" delete from facts where id in (select fid from cards where queue = -4); delete from cards where queue = -4;""") def resetCards(self, ids=None): "Reset progress on cards in IDS." print "position in resetCards()" sql = """ update cards set mod=:now, position=0, type=2, queue=2, lastInterval=0, interval=0, due=created, factor=2.5, reps=0, successive=0, lapses=0, flags=0""" sql2 = "delete from revlog" if ids is None: lim = "" else: sids = ids2str(ids) sql += " where id in "+sids sql2 += " where cardId in "+sids self.db.execute(sql, now=time.time()) self.db.execute(sql2) if self.qconf['newCardOrder'] == NEW_CARDS_RANDOM: # we need to re-randomize now self.randomizeNewCards(ids) def randomizeNewCards(self, cardIds=None): "Randomize 'due' on all new cards." now = time.time() query = "select distinct fid from cards where reps = 0" if cardIds: query += " and id in %s" % ids2str(cardIds) fids = self.db.list(query) data = [{'fid': fid, 'rand': random.uniform(0, now), 'now': now} for fid in fids] self.db.executemany(""" update cards set due = :rand + ord, mod = :now where fid = :fid and type = 2""", data) def orderNewCards(self): "Set 'due' to card creation time." self.db.execute(""" update cards set due = created, mod = :now where type = 2""", now=time.time()) def rescheduleCards(self, ids, min, max): "Reset cards and schedule with new interval in days (min, max)." self.resetCards(ids) vals = [] for id in ids: r = random.uniform(min*86400, max*86400) vals.append({ 'id': id, 'due': r + time.time(), 'int': r / 86400.0, 't': time.time(), }) self.db.executemany(""" update cards set interval = :int, due = :due, reps = 1, successive = 1, yesCount = 1, firstAnswered = :t, queue = 1, type = 1, where id = :id""", vals) # Models ########################################################################## def currentModel(self): return self.getModel(self.conf['currentModelId']) def models(self): "Return a dict of mid -> model." mods = {} for m in [self.getModel(id) for id in self.db.list( "select id from models")]: mods[m.id] = m return mods def getModel(self, mid): return anki.models.Model(self, mid) def addModel(self, model): model.flush() self.conf['currentModelId'] = model.id def deleteModel(self, mid): "Delete MODEL, and all its cards/facts." # do a direct delete self.modSchema() # delete facts/cards self.deleteCards(self.db.list(""" select id from cards where fid in (select id from facts where mid = ?)""", mid)) # then the model self.db.execute("delete from models where id = ?", mid) # GUI should ensure last model is not deleted if self.conf['currentModelId'] == mid: self.conf['currentModelId'] = self.db.scalar( "select id from models limit 1") def modelUseCount(self, model): "Return number of facts using model." return self.db.scalar("select count() from facts " "where facts.mid = :id", id=model.id) # Field checksums and sorting fields ########################################################################## def _fieldData(self, sfids): return self.db.execute( "select id, mid, flds from facts where id in "+sfids) def updateFieldCache(self, fids, csum=True): "Update field checksums and sort cache, after find&replace, etc." sfids = ids2str(fids) mods = self.models() r = [] r2 = [] for (fid, mid, flds) in self._fieldData(sfids): fields = splitFields(flds) model = mods[mid] if csum: for c, f in enumerate(model.fields): if f['uniq'] and fields[c]: r.append((fid, mid, fieldChecksum(fields[c]))) r2.append((stripHTML(fields[model.sortIdx()])[ :SORT_FIELD_LEN], fid)) if csum: self.db.execute("delete from fsums where fid in "+sfids) self.db.executemany("insert into fsums values (?,?,?)", r) self.db.executemany("update facts set sfld = ? where id = ?", r2) # Q/A generation ########################################################################## def renderQA(self, ids=None, type="card"): # gather metadata if type == "card": where = "and c.id in " + ids2str(ids) elif type == "fact": where = "and f.id in " + ids2str(ids) elif type == "model": where = "and m.id in " + ids2str(ids) elif type == "all": where = "" else: raise Exception() mods = self.models() groups = dict(self.db.all("select id, name from groups")) return [self._renderQA(mods[row[2]], groups[row[3]], row) for row in self._qaData(where)] def _renderQA(self, model, gname, data, filters=True): "Returns hash of id, question, answer." # data is [cid, fid, mid, gid, ord, tags, flds, data] # unpack fields and create dict flist = data[6].split("\x1f") fields = {} for (name, (idx, conf)) in model.fieldMap().items(): fields[name] = flist[idx] fields["text:"+name] = stripHTML(fields[name]) if fields[name]: fields["text:"+name] = stripHTML(fields[name]) fields[name] = '%s' % ( hexifyID(data[2]), hexifyID(idx), fields[name]) else: fields["text:"+name] = "" fields[name] = "" fields['Tags'] = data[5] fields['Model'] = model.name fields['Group'] = gname template = model.templates[data[4]] fields['Template'] = template['name'] # render q & a d = dict(id=data[0]) for (type, format) in (("q", template['qfmt']), ("a", template['afmt'])): # if filters: # fields = runFilter("renderQA.pre", fields, , self) html = anki.template.render(format, fields) # if filters: # d[type] = runFilter("renderQA.post", html, fields, meta, self) self.media.registerText(html) d[type] = html return d def _qaData(self, where=""): "Return [cid, fid, mid, gid, ord, tags, flds, data] db query" return self.db.execute(""" select c.id, f.id, m.id, g.id, c.ord, f.tags, f.flds, f.data from cards c, facts f, models m, groups g where c.fid == f.id and f.mid == m.id and c.gid = g.id %s""" % where) # Tags ########################################################################## def tagList(self): return self.db.list("select name from tags order by name") def cardsWithNoTags(self): return self.db.list(""" select cards.id from cards, facts where facts.tags = "" and cards.fid = facts.id""") def cardHasTag(self, card, tag): tags = self.db.scalar("select tags from fact where id = :fid", fid=card.fid) return tag.lower() in parseTags(tags.lower()) def updateFactTags(self, fids=None): "Add any missing tags to the tags list." if fids: lim = " where id in " + ids2str(fids) else: lim = "" self.registerTags(set(parseTags( " ".join(self.db.list("select distinct tags from facts"+lim))))) def registerTags(self, tags): r = [] for t in tags: r.append({'t': t}) self.db.executemany(""" insert or ignore into tags (mod, name) values (%d, :t)""" % intTime(), r) def addTags(self, ids, tags, add=True): "Add tags in bulk. TAGS is space-separated." self.startProgress() newTags = parseTags(tags) # cache tag names self.registerTags(newTags) # find facts missing the tags if add: l = "tags not " fn = addTags else: l = "tags " fn = deleteTags lim = " or ".join( [l+"like :_%d" % c for c, t in enumerate(newTags)]) res = self.db.all( "select id, tags from facts where " + lim, **dict([("_%d" % x, '%% %s %%' % y) for x, y in enumerate(newTags)])) # update tags fids = [] def fix(row): fids.append(row[0]) return {'id': row[0], 't': fn(tags, row[1]), 'n':intTime()} self.db.executemany(""" update facts set tags = :t, mod = :n where id = :id""", [fix(row) for row in res]) # update q/a cache self.registerTags(parseTags(tags)) self.finishProgress() def deleteTags(self, ids, tags): self.addTags(ids, tags, False) # Finding cards ########################################################################## def findCards(self, query): import anki.find return anki.find.findCards(self, query) def findReplace(self, *args, **kwargs): import anki.find return anki.find.findReplace(self, *args, **kwargs) def findDuplicates(self, fmids): import anki.find return anki.find.findDuplicates(self, fmids) # Progress info ########################################################################## def startProgress(self, max=0, min=0, title=None): self.enableProgressHandler() runHook("startProgress", max, min, title) def updateProgress(self, label=None, value=None): runHook("updateProgress", label, value) def finishProgress(self): runHook("updateProgress") runHook("finishProgress") self.disableProgressHandler() def progressHandler(self): if (time.time() - self.progressHandlerCalled) < 0.2: return self.progressHandlerCalled = time.time() if self.progressHandlerEnabled: runHook("dbProgress") def setupProgressHandler(self): self.progressHandlerCalled = 0 self.progressHandlerEnabled = False try: self.engine.raw_connection().set_progress_handler( deck.progressHandler, 100) except: pass def enableProgressHandler(self): self.progressHandlerEnabled = True def disableProgressHandler(self): self.progressHandlerEnabled = False # Timeboxing ########################################################################## def startTimebox(self): self.lastSessionStart = self.sessionStartTime self.sessionStartTime = time.time() self.sessionStartReps = self.repsToday def stopTimebox(self): self.sessionStartTime = 0 def timeboxStarted(self): return self.sessionStartTime def timeboxReached(self): if not self.sessionStartTime: # not started return False if (self.sessionTimeLimit and time.time() > (self.sessionStartTime + self.sessionTimeLimit)): return True if (self.sessionRepLimit and self.sessionRepLimit <= self.repsToday - self.sessionStartReps): return True return False # Syncing ########################################################################## def enableSyncing(self): self.syncName = self.getSyncName() def disableSyncing(self): self.syncName = u"" def syncingEnabled(self): return self.syncName def genSyncName(self): return unicode(checksum(self.path.encode("utf-8"))) def syncHashBad(self): if self.syncName and self.syncName != self.genSyncName(): self.disableSyncing() return True # DB maintenance ########################################################################## def fixIntegrity(self, quick=False): "Fix possible problems and rebuild caches." self.save() self.resetUndo() problems = [] recover = False if quick: num = 4 else: num = 10 oldSize = os.stat(self.path)[stat.ST_SIZE] self.startProgress(num) self.updateProgress(_("Checking database...")) if self.db.scalar("pragma integrity_check") != "ok": self.finishProgress() return _("Database file is damaged.\n" "Please restore from automatic backup (see FAQ).") # ensure correct views and indexes are available self.updateProgress() updateIndices(self) # does the user have a model? self.updateProgress() if not self.db.scalar("select count(id) from models"): self.addModel(BasicModel()) problems.append(_("Deck was missing a model")) # is currentModel pointing to a valid model? if not self.db.all(""" select decks.id from decks, models where decks.currentModelId = models.id"""): self.currentModelId = self.models[0].id problems.append(_("The current model didn't exist")) # fdata missing a field model ids = self.db.list(""" select id from fdata where fmid not in ( select distinct id from fields)""") if ids: self.db.execute("delete from fdata where id in %s" % ids2str(ids)) problems.append(ngettext("Deleted %d field with missing field model", "Deleted %d fdata with missing field model", len(ids)) % len(ids)) # facts missing a field? ids = self.db.list(""" select distinct facts.id from facts, fields where facts.mid = fields.mid and fields.id not in (select fmid from fdata where fid = facts.id)""") if ids: self.deleteFacts(ids) problems.append(ngettext("Deleted %d fact with missing fields", "Deleted %d facts with missing fields", len(ids)) % len(ids)) # cards missing a fact? ids = self.db.list(""" select id from cards where fid not in (select id from facts)""") if ids: recover = True self.recoverCards(ids) problems.append(ngettext("Recovered %d card with missing fact", "Recovered %d cards with missing fact", len(ids)) % len(ids)) # cards missing a card model? ids = self.db.list(""" select id from cards where tid not in (select id from templates)""") if ids: recover = True self.recoverCards(ids) problems.append(ngettext("Recovered %d card with no card template", "Recovered %d cards with no card template", len(ids)) % len(ids)) # cards with a card model from the wrong model ids = self.db.list(""" select id from cards where tid not in (select cm.id from templates cm, facts f where cm.mid = f.mid and f.id = cards.fid)""") if ids: recover = True self.recoverCards(ids) problems.append(ngettext("Recovered %d card with wrong card template", "Recovered %d cards with wrong card template", len(ids)) % len(ids)) # facts missing a card? ids = self.deleteDanglingFacts() if ids: problems.append(ngettext("Deleted %d fact with no cards", "Deleted %d facts with no cards", len(ids)) % len(ids)) # dangling fields? ids = self.db.list(""" select id from fdata where fid not in (select id from facts)""") if ids: self.db.execute( "delete from fdata where id in %s" % ids2str(ids)) problems.append(ngettext("Deleted %d dangling field", "Deleted %d dangling fields", len(ids)) % len(ids)) if not quick: self.updateProgress() # these sometimes end up null on upgrade self.db.execute("update models set source = 0 where source is null") self.db.execute( "update templates set allowEmptyAnswer = 1, typeAnswer = '' " "where allowEmptyAnswer is null or typeAnswer is null") # fix tags self.updateProgress() self.db.execute("delete from tags") self.updateFactTags() print "should ensure tags having leading/trailing space" # make sure ords are correct self.updateProgress() self.db.execute(""" update fdata set ord = (select ord from fields where id = fmid)""") self.db.execute(""" update cards set ord = (select ord from templates where cards.tid = templates.id)""") # fix problems with stripping html self.updateProgress() fdata = self.db.all("select id, val from fdata") newFdata = [] for (id, val) in fdata: newFdata.append({'id': id, 'val': tidyHTML(val)}) self.db.executemany( "update fdata set val=:val where id=:id", newFdata) # and field checksums self.updateProgress() self.updateAllFieldChecksums() # regenerate question/answer cache for m in self.models: self.updateCardsFromModel(m, dirty=False) # rebuild self.updateProgress() self.rebuildTypes() # force a full sync self.modSchema() # and finally, optimize self.updateProgress() self.optimize() newSize = os.stat(self.path)[stat.ST_SIZE] save = (oldSize - newSize)/1024 txt = _("Database rebuilt and optimized.") if save > 0: txt += "\n" + _("Saved %dKB.") % save problems.append(txt) self.save() self.finishProgress() if problems: if recover: problems.append("\n" + _("""\ Cards with corrupt or missing facts have been placed into new facts. \ Your scheduling info and card content has been preserved, but the \ original layout of the facts has been lost.""")) return "\n".join(problems) return "ok" def optimize(self): self.db.execute("vacuum") self.db.execute("analyze") # Undo/redo ########################################################################## def initUndo(self): # note this code ignores 'unique', as it's an sqlite reserved word self.undoStack = [] self.redoStack = [] self.undoEnabled = True self.db.execute( "create temporary table undoLog (seq integer primary key not null, sql text)") tables = self.db.list( "select name from sqlite_master where type = 'table'") for table in tables: if table in ("undoLog", "sqlite_stat1"): continue columns = [r[1] for r in self.db.all("pragma table_info(%s)" % table)] # insert self.db.execute(""" create temp trigger _undo_%(t)s_it after insert on %(t)s begin insert into undoLog values (null, 'delete from %(t)s where rowid = ' || new.rowid); end""" % {'t': table}) # update sql = """ create temp trigger _undo_%(t)s_ut after update on %(t)s begin insert into undoLog values (null, 'update %(t)s """ % {'t': table} sep = "set " for c in columns: if c == "unique": continue sql += "%(s)s%(c)s=' || quote(old.%(c)s) || '" % { 's': sep, 'c': c} sep = "," sql += " where rowid = ' || old.rowid); end" self.db.execute(sql) # delete sql = """ create temp trigger _undo_%(t)s_dt before delete on %(t)s begin insert into undoLog values (null, 'insert into %(t)s (rowid""" % {'t': table} for c in columns: sql += ",\"%s\"" % c sql += ") values (' || old.rowid ||'" for c in columns: if c == "unique": sql += ",1" continue sql += ",' || quote(old.%s) ||'" % c sql += ")'); end" self.db.execute(sql) def undoName(self): for n in reversed(self.undoStack): if n: return n[0] def redoName(self): return self.redoStack[-1][0] def undoAvailable(self): if not self.undoEnabled: return for r in reversed(self.undoStack): if r: return True def redoAvailable(self): return self.undoEnabled and self.redoStack def resetUndo(self): try: self.db.execute("delete from undoLog") except: pass self.undoStack = [] self.redoStack = [] def setUndoBarrier(self): if not self.undoStack or self.undoStack[-1] is not None: self.undoStack.append(None) def setUndoStart(self, name, merge=False): if not self.undoEnabled: return if merge and self.undoStack: if self.undoStack[-1] and self.undoStack[-1][0] == name: # merge with last entry? return start = self._latestUndoRow() self.undoStack.append([name, start, None]) def setUndoEnd(self, name): if not self.undoEnabled: return end = self._latestUndoRow() while self.undoStack[-1] is None: # strip off barrier self.undoStack.pop() self.undoStack[-1][2] = end if self.undoStack[-1][1] == self.undoStack[-1][2]: self.undoStack.pop() else: self.redoStack = [] runHook("undoEnd") def _latestUndoRow(self): return self.db.scalar("select max(rowid) from undoLog") or 0 def _undoredo(self, src, dst): while 1: u = src.pop() if u: break (start, end) = (u[1], u[2]) if end is None: end = self._latestUndoRow() sql = self.db.list(""" select sql from undoLog where seq > :s and seq <= :e order by seq desc""", s=start, e=end) mod = len(sql) / 35 if mod: self.startProgress(36) self.updateProgress(_("Processing...")) newstart = self._latestUndoRow() for c, s in enumerate(sql): if mod and not c % mod: self.updateProgress() self.engine.execute(s) newend = self._latestUndoRow() dst.append([u[0], newstart, newend]) if mod: self.finishProgress() def undo(self): "Undo the last action(s)." self._undoredo(self.undoStack, self.redoStack) runHook("postUndoRedo") def redo(self): "Redo the last action(s)." self._undoredo(self.redoStack, self.undoStack) runHook("postUndoRedo") # Dynamic indices ########################################################################## def updateDynamicIndices(self): # determine required columns required = [] if self.qconf['revCardOrder'] in (REV_CARDS_OLD_FIRST, REV_CARDS_NEW_FIRST): required.append("interval") cols = ["queue", "due", "gid"] + required # update if changed if self.db.scalar( "select 1 from sqlite_master where name = 'ix_cards_multi'"): rows = self.db.all("pragma index_info('ix_cards_multi')") else: rows = None if not (rows and cols == [r[2] for r in rows]): self.db.execute("drop index if exists ix_cards_multi") self.db.execute("create index ix_cards_multi on cards (%s)" % ", ".join(cols)) self.db.execute("analyze")