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

Anki used random 64bit IDs for cards, facts and fields. This had some nice properties: - merging data in syncs and imports was simply a matter of copying each way, as conflicts were astronomically unlikely - it made it easy to identify identical cards and prevent them from being reimported But there were some negatives too: - they're more expensive to store - javascript can't handle numbers > 2**53, which means AnkiMobile, iAnki and so on have to treat the ids as strings, which is slow - simply copying data in a sync or import can lead to corruption, as while a duplicate id indicates the data was originally the same, it may have diverged. A more intelligent approach is necessary. - sqlite was sorting the fields table based on the id, which meant the fields were spread across the table, and costly to fetch So instead, we'll move to incremental ids. In the case of model changes we'll declare that a schema change and force a full sync to avoid having to deal with conflicts, and in the case of cards and facts, we'll need to update the ids on one end to merge. Identical cards can be detected by checking to see if their id is the same and their creation time is the same. Creation time has been added back to cards and facts because it's necessary for sync conflict merging. That means facts.pos is not required. The graves table has been removed. It's not necessary for schema related changes, and dead cards/facts can be represented as a card with queue=-4 and created=0. Because we will record schema modification time and can ensure a full sync propagates to all endpoints, it means we can remove the dead cards/facts on schema change. Tags have been removed from the facts table and are represented as a field with ord=-1 and fmid=0. Combined with the locality improvement for fields, it means that fetching fields is not much more expensive than using the q/a cache. Because of the above, removing the q/a cache is a possibility now. The q and a columns on cards has been dropped. It will still be necessary to render the q/a on fact add/edit, since we need to record media references. It would be nice to avoid this in the future. Perhaps one way would be the ability to assign a type to fields, like "image", "audio", or "latex". LaTeX needs special consider anyway, as it was being rendered into the q/a cache.
202 lines
5.5 KiB
Python
202 lines
5.5 KiB
Python
# -*- coding: utf-8 -*-
|
|
# Copyright: Damien Elmes <anki@ichi2.net>
|
|
# License: GNU GPL, version 3 or later; http://www.gnu.org/copyleft/gpl.html
|
|
|
|
"""\
|
|
Models load their templates and fields when they are loaded. If you update a
|
|
template or field, you should call model.flush(), rather than trying to save
|
|
the subobject directly.
|
|
"""
|
|
|
|
import simplejson
|
|
from anki.utils import intTime
|
|
from anki.lang import _
|
|
|
|
# Models
|
|
##########################################################################
|
|
|
|
defaultConf = {
|
|
}
|
|
|
|
class Model(object):
|
|
|
|
def __init__(self, deck, id=None):
|
|
self.deck = deck
|
|
if id:
|
|
self.id = id
|
|
self.load()
|
|
else:
|
|
self.id = None
|
|
self.name = u""
|
|
self.mod = intTime()
|
|
self.conf = defaultConf.copy()
|
|
self.fields = []
|
|
self.templates = []
|
|
|
|
def load(self):
|
|
(self.mod,
|
|
self.name,
|
|
self.conf) = self.deck.db.first("""
|
|
select mod, name, conf from models where id = ?""", self.id)
|
|
self.conf = simplejson.loads(self.conf)
|
|
self.loadFields()
|
|
self.loadTemplates()
|
|
|
|
def flush(self):
|
|
self.mod = intTime()
|
|
ret = self.deck.db.execute("""
|
|
insert or replace into models values (?, ?, ?, ?)""",
|
|
self.id, self.mod, self.name,
|
|
simplejson.dumps(self.conf))
|
|
self.id = ret.lastrowid
|
|
[f._flush() for f in self.fields]
|
|
[t._flush() for t in self.templates]
|
|
|
|
def updateCache(self):
|
|
self.deck.updateCache([self.id], "model")
|
|
|
|
def _getID(self):
|
|
if not self.id:
|
|
# flush so we can get our DB id
|
|
self.flush()
|
|
return self.id
|
|
|
|
# Fields
|
|
##################################################
|
|
|
|
def loadFields(self):
|
|
sql = "select * from fields where mid = ? order by ord"
|
|
self.fields = [Field(self.deck, data)
|
|
for data in self.deck.db.all(sql, self.id)]
|
|
|
|
def addField(self, field):
|
|
self.deck.modSchema()
|
|
field.mid = self._getID()
|
|
field.ord = len(self.fields)
|
|
self.fields.append(field)
|
|
|
|
def fieldMap(self):
|
|
"Mapping of field name -> (fmid, ord)."
|
|
return dict([(f.name, (f.id, f.ord, f.conf)) for f in self.fields])
|
|
|
|
# Templates
|
|
##################################################
|
|
|
|
def loadTemplates(self):
|
|
sql = "select * from templates where mid = ? order by ord"
|
|
self.templates = [Template(self.deck, data)
|
|
for data in self.deck.db.all(sql, self.id)]
|
|
|
|
def addTemplate(self, template):
|
|
self.deck.modSchema()
|
|
template.mid = self._getID()
|
|
template.ord = len(self.templates)
|
|
self.templates.append(template)
|
|
|
|
# Copying
|
|
##################################################
|
|
|
|
def copy(self):
|
|
"Copy, flush and return."
|
|
new = Model(self.deck, self.id)
|
|
new.id = None
|
|
new.name += _(" copy")
|
|
# get new id
|
|
f = new.fields; new.fields = []
|
|
t = new.templates; new.templates = []
|
|
new.flush()
|
|
# then put back
|
|
new.fields = f
|
|
new.templates = t
|
|
for f in new.fields:
|
|
f.id = None
|
|
f.mid = new.id
|
|
f._flush()
|
|
for t in new.templates:
|
|
t.id = None
|
|
t.mid = new.id
|
|
t._flush()
|
|
return new
|
|
|
|
# Field model object
|
|
##########################################################################
|
|
|
|
defaultFieldConf = {
|
|
'rtl': False, # features
|
|
'required': False,
|
|
'unique': False,
|
|
'font': "Arial",
|
|
'quizSize': 20,
|
|
'editSize': 20,
|
|
'quizColour': "#fff",
|
|
'pre': True,
|
|
}
|
|
|
|
class Field(object):
|
|
|
|
def __init__(self, deck, data=None):
|
|
self.deck = deck
|
|
if data:
|
|
self.initFromData(data)
|
|
else:
|
|
self.id = None
|
|
self.numeric = 0
|
|
self.conf = defaultFieldConf.copy()
|
|
|
|
def initFromData(self, data):
|
|
(self.id,
|
|
self.mid,
|
|
self.ord,
|
|
self.name,
|
|
self.numeric,
|
|
self.conf) = data
|
|
self.conf = simplejson.loads(self.conf)
|
|
|
|
def _flush(self):
|
|
ret = self.deck.db.execute("""
|
|
insert or replace into fields values (?, ?, ?, ?, ?, ?)""",
|
|
self.id, self.mid, self.ord,
|
|
self.name, self.numeric,
|
|
simplejson.dumps(self.conf))
|
|
self.id = ret.lastrowid
|
|
|
|
# Template object
|
|
##########################################################################
|
|
|
|
defaultTemplateConf = {
|
|
'hideQ': False,
|
|
'align': 0,
|
|
'bg': "#000",
|
|
'allowEmptyAns': None,
|
|
'typeAnswer': None,
|
|
}
|
|
|
|
class Template(object):
|
|
|
|
def __init__(self, deck, data=None):
|
|
self.deck = deck
|
|
if data:
|
|
self.initFromData(data)
|
|
else:
|
|
self.id = None
|
|
self.active = True
|
|
self.conf = defaultTemplateConf.copy()
|
|
|
|
def initFromData(self, data):
|
|
(self.id,
|
|
self.mid,
|
|
self.ord,
|
|
self.name,
|
|
self.active,
|
|
self.qfmt,
|
|
self.afmt,
|
|
self.conf) = data
|
|
self.conf = simplejson.loads(self.conf)
|
|
|
|
def _flush(self):
|
|
ret = self.deck.db.execute("""
|
|
insert or replace into templates values (?, ?, ?, ?, ?, ?, ?, ?)""",
|
|
self.id, self.mid, self.ord, self.name,
|
|
self.active, self.qfmt, self.afmt,
|
|
simplejson.dumps(self.conf))
|
|
self.id = ret.lastrowid
|