mirror of
https://github.com/ankitects/anki.git
synced 2025-09-19 22:42:25 -04:00

The way we were handling 1 day+ learning intervals was not great - they'd show up at the start of a new day before normal reviews, meaning the hardest cards came first. In previous Anki versions we deliberately sorted the queue in the opposite order to prevent that. When relearning the cards the next day, if you failed a card and expected to see it in 10 minutes that wouldn't happen either, as all the overdue cards took precedence. To fix this, we put cards that are due tomorrow or later into a separate queue (queue 3), and pull cards from that queue only after the reviews are done. In the future it might also be nice to move overdue learning cards into that queue automatically at the start of a session.
651 lines
22 KiB
Python
651 lines
22 KiB
Python
# -*- coding: utf-8 -*-
|
|
# Copyright: Damien Elmes <anki@ichi2.net>
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
import time, os, random, re, stat, datetime, copy, shutil, sys
|
|
from anki.lang import _, ngettext
|
|
from anki.utils import ids2str, hexifyID, checksum, fieldChecksum, stripHTML, \
|
|
intTime, splitFields, joinFields, maxID, json
|
|
from anki.hooks import runHook, runFilter
|
|
from anki.sched import Scheduler
|
|
from anki.models import ModelManager
|
|
from anki.media import MediaManager
|
|
from anki.decks import DeckManager
|
|
from anki.tags import TagManager
|
|
from anki.consts import *
|
|
from anki.errors import AnkiError
|
|
|
|
import anki.latex # sets up hook
|
|
import anki.cards, anki.notes, anki.template, anki.find
|
|
|
|
defaultConf = {
|
|
# review options
|
|
'activeDecks': [1],
|
|
'curDeck': 1,
|
|
'newSpread': NEW_CARDS_DISTRIBUTE,
|
|
'collapseTime': 1200,
|
|
'timeLim': 0,
|
|
'estTimes': True,
|
|
'dueCounts': True,
|
|
# other config
|
|
'curModel': None,
|
|
'nextPos': 1,
|
|
'sortType': "noteFld",
|
|
'sortBackwards': False,
|
|
'addToCur': True, # add new to currently selected deck?
|
|
'replayBoth': True, # include question when replaying answer?
|
|
}
|
|
|
|
# this is initialized by storage.Collection
|
|
class _Collection(object):
|
|
|
|
def __init__(self, db, server=False):
|
|
self.db = db
|
|
self.path = db._path
|
|
self.server = server
|
|
self._lastSave = time.time()
|
|
self.clearUndo()
|
|
self.media = MediaManager(self)
|
|
self.models = ModelManager(self)
|
|
self.decks = DeckManager(self)
|
|
self.tags = TagManager(self)
|
|
self.load()
|
|
if not self.crt:
|
|
d = datetime.datetime.today()
|
|
d -= datetime.timedelta(hours=4)
|
|
d = datetime.datetime(d.year, d.month, d.day)
|
|
d += datetime.timedelta(hours=4)
|
|
self.crt = int(time.mktime(d.timetuple()))
|
|
self.sched = Scheduler(self)
|
|
# check for improper shutdown
|
|
self.cleanup()
|
|
|
|
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.scm,
|
|
self.dty,
|
|
self._usn,
|
|
self.ls,
|
|
self.conf,
|
|
models,
|
|
decks,
|
|
dconf,
|
|
tags) = self.db.first("""
|
|
select crt, mod, scm, dty, usn, ls,
|
|
conf, models, decks, dconf, tags from col""")
|
|
self.conf = json.loads(self.conf)
|
|
self.models.load(models)
|
|
self.decks.load(decks, dconf)
|
|
self.tags.load(tags)
|
|
|
|
def setMod(self):
|
|
"""Mark DB modified.
|
|
|
|
DB operations and the deck/tag/model managers do this automatically, so this
|
|
is only necessary if you modify properties of this object or the conf dict."""
|
|
self.db.mod = True
|
|
|
|
def flush(self, mod=None):
|
|
"Flush state to DB, updating mod time."
|
|
self.mod = intTime(1000) if mod is None else mod
|
|
self.db.execute(
|
|
"""update col set
|
|
crt=?, mod=?, scm=?, dty=?, usn=?, ls=?, conf=?""",
|
|
self.crt, self.mod, self.scm, self.dty,
|
|
self._usn, self.ls, json.dumps(self.conf))
|
|
|
|
def save(self, name=None, mod=None):
|
|
"Flush, commit DB, and take out another write lock."
|
|
# let the managers conditionally flush
|
|
self.models.flush()
|
|
self.decks.flush()
|
|
self.tags.flush()
|
|
# and flush deck + bump mod if db has been changed
|
|
if self.db.mod:
|
|
self.flush(mod=mod)
|
|
self.db.commit()
|
|
self.lock()
|
|
self.db.mod = False
|
|
self._markOp(name)
|
|
self._lastSave = time.time()
|
|
|
|
def autosave(self):
|
|
"Save if 5 minutes has passed since last save."
|
|
if time.time() - self._lastSave > 300:
|
|
self.save()
|
|
|
|
def lock(self):
|
|
# make sure we don't accidentally bump mod time
|
|
mod = self.db.mod
|
|
self.db.execute("update col set mod=mod")
|
|
self.db.mod = mod
|
|
|
|
def close(self, save=True):
|
|
"Disconnect from DB."
|
|
if self.db:
|
|
self.cleanup()
|
|
if save:
|
|
self.save()
|
|
else:
|
|
self.rollback()
|
|
if not self.server:
|
|
self.db.execute("pragma journal_mode = delete")
|
|
self.db.close()
|
|
self.db = None
|
|
self.media.close()
|
|
|
|
def reopen(self):
|
|
"Reconnect to DB (after changing threads, etc)."
|
|
import anki.db
|
|
if not self.db:
|
|
self.db = anki.db.DB(self.path)
|
|
self.media.connect()
|
|
|
|
def rollback(self):
|
|
self.db.rollback()
|
|
self.load()
|
|
self.lock()
|
|
|
|
def modSchema(self, check=True):
|
|
"Mark schema modified. Call this first so user can abort if necessary."
|
|
if not self.schemaChanged():
|
|
if check and not runFilter("modSchema", True):
|
|
raise AnkiError("abortSchemaMod")
|
|
self.scm = intTime(1000)
|
|
|
|
def schemaChanged(self):
|
|
"True if schema changed since last sync."
|
|
return self.scm > self.ls
|
|
|
|
def setDirty(self):
|
|
"Signal there are temp. suspended cards that need cleaning up on close."
|
|
self.dty = True
|
|
|
|
def cleanup(self):
|
|
"Unsuspend any temporarily suspended cards."
|
|
if self.dty:
|
|
self.sched.onClose()
|
|
self.dty = False
|
|
|
|
def usn(self):
|
|
return self._usn if self.server else -1
|
|
|
|
def beforeUpload(self):
|
|
"Called before a full upload."
|
|
tbls = "notes", "cards", "revlog", "graves"
|
|
for t in tbls:
|
|
self.db.execute("update %s set usn=0 where usn=-1" % t)
|
|
self._usn += 1
|
|
self.models.beforeUpload()
|
|
self.tags.beforeUpload()
|
|
self.decks.beforeUpload()
|
|
self.modSchema()
|
|
self.ls = self.scm
|
|
self.close()
|
|
|
|
# Object creation helpers
|
|
##########################################################################
|
|
|
|
def getCard(self, id):
|
|
return anki.cards.Card(self, id)
|
|
|
|
def getNote(self, id):
|
|
return anki.notes.Note(self, id=id)
|
|
|
|
# Utils
|
|
##########################################################################
|
|
|
|
def nextID(self, type, inc=True):
|
|
type = "next"+type.capitalize()
|
|
id = self.conf.get(type, 1)
|
|
if inc:
|
|
self.conf[type] = id+1
|
|
return id
|
|
|
|
def reset(self):
|
|
"Rebuild the queue and reload data after DB modified."
|
|
self.sched.reset()
|
|
|
|
# Deletion logging
|
|
##########################################################################
|
|
|
|
def _logRem(self, ids, type):
|
|
self.db.executemany("insert into graves values (%d, ?, %d)" % (
|
|
self.usn(), type), ([x] for x in ids))
|
|
|
|
# Notes
|
|
##########################################################################
|
|
|
|
def noteCount(self):
|
|
return self.db.scalar("select count() from notes")
|
|
|
|
def newNote(self):
|
|
"Return a new note with the current model."
|
|
return anki.notes.Note(self, self.models.current())
|
|
|
|
def addNote(self, note):
|
|
"Add a note to the collection. Return number of new cards."
|
|
# check we have card models available, then save
|
|
cms = self.findTemplates(note)
|
|
if not cms:
|
|
return 0
|
|
note.flush()
|
|
# deck conf governs which of these are used
|
|
due = self.nextID("pos")
|
|
# add cards
|
|
ncards = 0
|
|
for template in cms:
|
|
self._newCard(note, template, due)
|
|
ncards += 1
|
|
return ncards
|
|
|
|
def remNotes(self, ids):
|
|
self.remCards(self.db.list("select id from cards where nid in "+
|
|
ids2str(ids)))
|
|
|
|
def _remNotes(self, ids):
|
|
"Bulk delete notes by ID. Don't call this directly."
|
|
if not ids:
|
|
return
|
|
strids = ids2str(ids)
|
|
# we need to log these independently of cards, as one side may have
|
|
# more card templates
|
|
self._logRem(ids, REM_NOTE)
|
|
self.db.execute("delete from notes where id in %s" % strids)
|
|
|
|
# Card creation
|
|
##########################################################################
|
|
|
|
def findTemplates(self, note):
|
|
"Return (active), non-empty templates."
|
|
model = note.model()
|
|
avail = self.models.availOrds(model, joinFields(note.fields))
|
|
return self._tmplsFromOrds(model, avail)
|
|
|
|
def _tmplsFromOrds(self, model, avail):
|
|
ok = []
|
|
if model['type'] == MODEL_STD:
|
|
for t in model['tmpls']:
|
|
if t['ord'] in avail:
|
|
ok.append(t)
|
|
else:
|
|
# cloze - generate temporary templates from first
|
|
for ord in avail:
|
|
t = copy.copy(model['tmpls'][0])
|
|
t['ord'] = ord
|
|
ok.append(t)
|
|
return ok
|
|
|
|
def genCards(self, nids):
|
|
"Generate cards for non-empty templates, return ids to remove."
|
|
# build map of (nid,ord) so we don't create dupes
|
|
snids = ids2str(nids)
|
|
have = {}
|
|
dids = {}
|
|
for id, nid, ord, did in self.db.execute(
|
|
"select id, nid, ord, did from cards where nid in "+snids):
|
|
# existing cards
|
|
if nid not in have:
|
|
have[nid] = {}
|
|
have[nid][ord] = id
|
|
# and their dids
|
|
if nid in dids:
|
|
if dids[nid] and dids[nid] != did:
|
|
# cards are in two or more different decks; revert to
|
|
# model default
|
|
dids[nid] = None
|
|
else:
|
|
# first card or multiple cards in same deck
|
|
dids[nid] = did
|
|
# build cards for each note
|
|
data = []
|
|
ts = maxID(self.db)
|
|
now = intTime()
|
|
rem = []
|
|
usn = self.usn()
|
|
for nid, mid, flds in self.db.execute(
|
|
"select id, mid, flds from notes where id in "+snids):
|
|
model = self.models.get(mid)
|
|
avail = self.models.availOrds(model, flds)
|
|
did = dids.get(nid) or model['did']
|
|
# add any missing cards
|
|
for t in self._tmplsFromOrds(model, avail):
|
|
doHave = nid in have and t['ord'] in have[nid]
|
|
if not doHave:
|
|
# check deck is not a cram deck
|
|
did = t['did'] or did
|
|
if self.decks.isDyn(did):
|
|
did = 1
|
|
# if the deck doesn't exist, use default instead
|
|
did = self.decks.get(did)['id']
|
|
# we'd like to use the same due# as sibling cards, but we
|
|
# can't retrieve that quickly, so we give it a new id
|
|
# instead
|
|
data.append((ts, nid, did, t['ord'],
|
|
now, usn, self.nextID("pos")))
|
|
ts += 1
|
|
# note any cards that need removing
|
|
if nid in have:
|
|
for ord, id in have[nid].items():
|
|
if ord not in avail:
|
|
rem.append(id)
|
|
# bulk update
|
|
self.db.executemany("""
|
|
insert into cards values (?,?,?,?,?,?,0,0,?,0,0,0,0,0,0,0,0,"")""",
|
|
data)
|
|
return rem
|
|
|
|
# type 0 - when previewing in add dialog, only non-empty
|
|
# type 1 - when previewing edit, only existing
|
|
# type 2 - when previewing in models dialog, all templates
|
|
def previewCards(self, note, type=0):
|
|
if type == 0:
|
|
cms = self.findTemplates(note)
|
|
elif type == 1:
|
|
cms = [c.template() for c in note.cards()]
|
|
else:
|
|
cms = note.model()['tmpls']
|
|
if not cms:
|
|
return []
|
|
cards = []
|
|
for template in cms:
|
|
cards.append(self._newCard(note, template, 1, flush=False))
|
|
return cards
|
|
|
|
def _newCard(self, note, template, due, flush=True):
|
|
"Create a new card."
|
|
card = anki.cards.Card(self)
|
|
card.nid = note.id
|
|
card.ord = template['ord']
|
|
card.did = template['did'] or note.model()['did']
|
|
# if invalid did, use default instead
|
|
card.did = self.decks.get(card.did)['id']
|
|
card.due = self._dueForDid(card.did, due)
|
|
if flush:
|
|
card.flush()
|
|
return card
|
|
|
|
def _dueForDid(self, did, due):
|
|
conf = self.decks.confForDid(did)
|
|
# in order due?
|
|
if conf['new']['order'] == NEW_CARDS_DUE:
|
|
return due
|
|
else:
|
|
# random mode; seed with note ts so all cards of this note get the
|
|
# same random number
|
|
r = random.Random()
|
|
r.seed(due)
|
|
return r.randrange(1, max(due, 1000))
|
|
|
|
# Cards
|
|
##########################################################################
|
|
|
|
def isEmpty(self):
|
|
return not self.db.scalar("select 1 from cards limit 1")
|
|
|
|
def cardCount(self):
|
|
return self.db.scalar("select count() from cards")
|
|
|
|
def remCards(self, ids):
|
|
"Bulk delete cards by ID."
|
|
if not ids:
|
|
return
|
|
sids = ids2str(ids)
|
|
nids = self.db.list("select nid from cards where id in "+sids)
|
|
# remove cards
|
|
self._logRem(ids, REM_CARD)
|
|
self.db.execute("delete from cards where id in "+sids)
|
|
self.db.execute("delete from revlog where cid in "+sids)
|
|
# then notes
|
|
nids = self.db.list("""
|
|
select id from notes where id in %s and id not in (select nid from cards)""" %
|
|
ids2str(nids))
|
|
self._remNotes(nids)
|
|
|
|
def emptyCids(self):
|
|
rem = []
|
|
for m in self.models.all():
|
|
rem += self.genCards(self.models.nids(m))
|
|
return rem
|
|
|
|
def emptyCardReport(self, cids):
|
|
rep = ""
|
|
for ords, cnt, flds in self.db.all("""
|
|
select group_concat(ord+1), count(), flds from cards c, notes n
|
|
where c.nid = n.id and c.id in %s group by nid""" % ids2str(cids)):
|
|
rep += _("Cards: %(c)s\nFields: %(f)s\n\n") % dict(
|
|
c=ords, f=flds.replace("\x1f", " / "))
|
|
return rep
|
|
|
|
# Field checksums and sorting fields
|
|
##########################################################################
|
|
|
|
def _fieldData(self, snids):
|
|
return self.db.execute(
|
|
"select id, mid, flds from notes where id in "+snids)
|
|
|
|
def updateFieldCache(self, nids):
|
|
"Update field checksums and sort cache, after find&replace, etc."
|
|
snids = ids2str(nids)
|
|
r = []
|
|
for (nid, mid, flds) in self._fieldData(snids):
|
|
fields = splitFields(flds)
|
|
model = self.models.get(mid)
|
|
r.append((stripHTML(fields[self.models.sortIdx(model)]),
|
|
fieldChecksum(fields[0]),
|
|
nid))
|
|
# apply, relying on calling code to bump usn+mod
|
|
self.db.executemany("update notes set sfld=?, csum=? where id=?", r)
|
|
|
|
# Q/A generation
|
|
##########################################################################
|
|
|
|
def renderQA(self, ids=None, type="card"):
|
|
# gather metadata
|
|
if type == "card":
|
|
where = "and c.id in " + ids2str(ids)
|
|
elif type == "note":
|
|
where = "and f.id in " + ids2str(ids)
|
|
elif type == "model":
|
|
where = "and m.id in " + ids2str(ids)
|
|
elif type == "all":
|
|
where = ""
|
|
else:
|
|
raise Exception()
|
|
return [self._renderQA(row)
|
|
for row in self._qaData(where)]
|
|
|
|
def _renderQA(self, data):
|
|
"Returns hash of id, question, answer."
|
|
# data is [cid, nid, mid, did, ord, tags, flds]
|
|
# unpack fields and create dict
|
|
flist = splitFields(data[6])
|
|
fields = {}
|
|
model = self.models.get(data[2])
|
|
for (name, (idx, conf)) in self.models.fieldMap(model).items():
|
|
fields[name] = flist[idx]
|
|
fields['Tags'] = data[5]
|
|
fields['Type'] = model['name']
|
|
fields['Deck'] = self.decks.name(data[3])
|
|
if model['type'] == MODEL_STD:
|
|
template = model['tmpls'][data[4]]
|
|
else:
|
|
template = model['tmpls'][0]
|
|
fields['Card'] = template['name']
|
|
fields['c%d' % (data[4]+1)] = "1"
|
|
# render q & a
|
|
d = dict(id=data[0])
|
|
for (type, format) in (("q", template['qfmt']), ("a", template['afmt'])):
|
|
if type == "q":
|
|
format = format.replace("{{cloze:", "{{cq:%d:" % (
|
|
data[4]+1))
|
|
else:
|
|
format = format.replace("{{cloze:", "{{ca:%d:" % (
|
|
data[4]+1))
|
|
fields = runFilter("mungeFields", fields, model, data, self)
|
|
html = anki.template.render(format, fields)
|
|
d[type] = runFilter(
|
|
"mungeQA", html, type, fields, model, data, self)
|
|
return d
|
|
|
|
def _qaData(self, where=""):
|
|
"Return [cid, nid, mid, did, ord, tags, flds] db query"
|
|
return self.db.execute("""
|
|
select c.id, f.id, f.mid, c.did, c.ord, f.tags, f.flds
|
|
from cards c, notes f
|
|
where c.nid == f.id
|
|
%s""" % where)
|
|
|
|
# Finding cards
|
|
##########################################################################
|
|
|
|
def findCards(self, query, full=False, order=None):
|
|
return anki.find.Finder(self).findCards(query, full, order)
|
|
|
|
def findReplace(self, nids, src, dst, regex=None, field=None, fold=True):
|
|
return anki.find.findReplace(self, nids, src, dst, regex, field, fold)
|
|
|
|
def findDuplicates(self, fmids):
|
|
return anki.find.findDuplicates(self, fmids)
|
|
|
|
# Stats
|
|
##########################################################################
|
|
|
|
def cardStats(self, card):
|
|
from anki.stats import CardStats
|
|
return CardStats(self, card).report()
|
|
|
|
def stats(self):
|
|
from anki.stats import CollectionStats
|
|
return CollectionStats(self)
|
|
|
|
# Timeboxing
|
|
##########################################################################
|
|
|
|
def startTimebox(self):
|
|
self._startTime = time.time()
|
|
self._startReps = self.sched.reps
|
|
|
|
def timeboxReached(self):
|
|
"Return (elapsedTime, reps) if timebox reached, or False."
|
|
if not self.conf['timeLim']:
|
|
# timeboxing disabled
|
|
return False
|
|
elapsed = time.time() - self._startTime
|
|
if elapsed > self.conf['timeLim']:
|
|
return (self.conf['timeLim'], self.sched.reps - self._startReps)
|
|
|
|
# Undo
|
|
##########################################################################
|
|
|
|
def clearUndo(self):
|
|
# [type, undoName, data]
|
|
# type 1 = review; type 2 = checkpoint
|
|
self._undo = None
|
|
|
|
def undoName(self):
|
|
"Undo menu item name, or None if undo unavailable."
|
|
if not self._undo:
|
|
return None
|
|
return self._undo[1]
|
|
|
|
def undo(self):
|
|
if self._undo[0] == 1:
|
|
return self._undoReview()
|
|
else:
|
|
self._undoOp()
|
|
|
|
def markReview(self, card):
|
|
old = []
|
|
if self._undo:
|
|
if self._undo[0] == 1:
|
|
old = self._undo[2]
|
|
self.clearUndo()
|
|
self._undo = [1, _("Review"), old + [copy.copy(card)]]
|
|
|
|
def _undoReview(self):
|
|
data = self._undo[2]
|
|
c = data.pop()
|
|
if not data:
|
|
self.clearUndo()
|
|
# write old data
|
|
c.flush()
|
|
# and delete revlog entry
|
|
last = self.db.scalar(
|
|
"select id from revlog where cid = ? "
|
|
"order by id desc limit 1", c.id)
|
|
self.db.execute("delete from revlog where id = ?", last)
|
|
# and finally, update daily counts
|
|
# fixme: what to do in cramming case?
|
|
n = 1 if c.queue == 3 else c.queue
|
|
type = ("new", "lrn", "rev")[n]
|
|
self.sched._updateStats(c, type, -1)
|
|
return c.id
|
|
|
|
def _markOp(self, name):
|
|
"Call via .save()"
|
|
if name:
|
|
self._undo = [2, name]
|
|
else:
|
|
# saving disables old checkpoint, but not review undo
|
|
if self._undo and self._undo[0] == 2:
|
|
self.clearUndo()
|
|
|
|
def _undoOp(self):
|
|
self.rollback()
|
|
self.clearUndo()
|
|
|
|
# DB maintenance
|
|
##########################################################################
|
|
|
|
def fixIntegrity(self):
|
|
"Fix possible problems and rebuild caches."
|
|
problems = []
|
|
self.save()
|
|
oldSize = os.stat(self.path)[stat.ST_SIZE]
|
|
if self.db.scalar("pragma integrity_check") != "ok":
|
|
return _("Collection is corrupt. Please see the manual.")
|
|
# delete any notes with missing cards
|
|
ids = self.db.list("""
|
|
select id from notes where id not in (select distinct nid from cards)""")
|
|
self._remNotes(ids)
|
|
# tags
|
|
self.tags.registerNotes()
|
|
# field cache
|
|
for m in self.models.all():
|
|
self.updateFieldCache(self.models.nids(m))
|
|
# new card position
|
|
self.conf['nextPos'] = self.db.scalar(
|
|
"select max(due)+1 from cards where type = 0") or 0
|
|
# reviews should have a reasonable due #
|
|
ids = self.db.list(
|
|
"select id from cards where queue = 2 and due > 10000")
|
|
if ids:
|
|
problems.append("Reviews had incorrect due date.")
|
|
self.db.execute(
|
|
"update cards set due = 0, mod = ?, usn = ? where id in %s"
|
|
% ids2str(ids), intTime(), self.usn())
|
|
# and finally, optimize
|
|
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
|
|
ok = not problems
|
|
problems.append(txt)
|
|
self.save()
|
|
return ("\n".join(problems), ok)
|
|
|
|
def optimize(self):
|
|
self.db.execute("vacuum")
|
|
self.db.execute("analyze")
|
|
self.lock()
|