2012-12-21 08:51:59 +01:00
|
|
|
# coding: utf-8
|
|
|
|
|
2019-12-25 22:36:26 +01:00
|
|
|
import os
|
|
|
|
import tempfile
|
2019-12-24 10:14:48 +01:00
|
|
|
|
2019-12-25 22:36:26 +01:00
|
|
|
from anki import Collection as aopen
|
2020-04-06 12:09:44 +02:00
|
|
|
from anki.dbproxy import emulate_named_args
|
2020-02-20 03:48:46 +01:00
|
|
|
from anki.lang import without_unicode_isolation
|
2020-02-27 03:25:19 +01:00
|
|
|
from anki.rsbackend import TR
|
2019-12-25 22:36:26 +01:00
|
|
|
from anki.stdmodels import addBasicModel, models
|
2019-12-24 10:14:48 +01:00
|
|
|
from anki.utils import isWin
|
2014-06-03 10:38:47 +02:00
|
|
|
from tests.shared import assertException, getEmptyCol
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2019-12-25 05:18:34 +01:00
|
|
|
|
2017-10-05 06:14:56 +02:00
|
|
|
def test_create_open():
|
2014-04-21 23:04:46 +02:00
|
|
|
(fd, path) = tempfile.mkstemp(suffix=".anki2", prefix="test_attachNew")
|
2012-12-21 08:51:59 +01:00
|
|
|
try:
|
2014-04-21 23:04:46 +02:00
|
|
|
os.close(fd)
|
2012-12-21 08:51:59 +01:00
|
|
|
os.unlink(path)
|
|
|
|
except OSError:
|
|
|
|
pass
|
|
|
|
deck = aopen(path)
|
|
|
|
# for open()
|
|
|
|
newPath = deck.path
|
|
|
|
newMod = deck.mod
|
2020-05-15 05:33:37 +02:00
|
|
|
deck.close()
|
2012-12-21 08:51:59 +01:00
|
|
|
del deck
|
|
|
|
|
2017-10-05 06:14:56 +02:00
|
|
|
# reopen
|
2012-12-21 08:51:59 +01:00
|
|
|
deck = aopen(newPath)
|
|
|
|
assert deck.mod == newMod
|
|
|
|
deck.close()
|
|
|
|
|
|
|
|
# non-writeable dir
|
2019-12-24 10:14:48 +01:00
|
|
|
if isWin:
|
|
|
|
dir = "c:\root.anki2"
|
|
|
|
else:
|
|
|
|
dir = "/attachroot.anki2"
|
2019-12-25 05:18:34 +01:00
|
|
|
assertException(Exception, lambda: aopen(dir))
|
2012-12-21 08:51:59 +01:00
|
|
|
# reuse tmp file from before, test non-writeable file
|
|
|
|
os.chmod(newPath, 0)
|
2019-12-25 05:18:34 +01:00
|
|
|
assertException(Exception, lambda: aopen(newPath))
|
2016-05-12 06:45:35 +02:00
|
|
|
os.chmod(newPath, 0o666)
|
2012-12-21 08:51:59 +01:00
|
|
|
os.unlink(newPath)
|
|
|
|
|
2019-12-25 05:18:34 +01:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
def test_noteAddDelete():
|
2014-06-03 10:38:47 +02:00
|
|
|
deck = getEmptyCol()
|
2012-12-21 08:51:59 +01:00
|
|
|
# add a note
|
|
|
|
f = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
f["Front"] = "one"
|
|
|
|
f["Back"] = "two"
|
2012-12-21 08:51:59 +01:00
|
|
|
n = deck.addNote(f)
|
|
|
|
assert n == 1
|
|
|
|
# test multiple cards - add another template
|
2019-12-25 05:18:34 +01:00
|
|
|
m = deck.models.current()
|
|
|
|
mm = deck.models
|
2012-12-21 08:51:59 +01:00
|
|
|
t = mm.newTemplate("Reverse")
|
2019-12-25 05:18:34 +01:00
|
|
|
t["qfmt"] = "{{Back}}"
|
|
|
|
t["afmt"] = "{{Front}}"
|
2012-12-21 08:51:59 +01:00
|
|
|
mm.addTemplate(m, t)
|
|
|
|
mm.save(m)
|
|
|
|
assert deck.cardCount() == 2
|
|
|
|
# creating new notes should use both cards
|
|
|
|
f = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
f["Front"] = "three"
|
|
|
|
f["Back"] = "four"
|
2012-12-21 08:51:59 +01:00
|
|
|
n = deck.addNote(f)
|
|
|
|
assert n == 2
|
|
|
|
assert deck.cardCount() == 4
|
|
|
|
# check q/a generation
|
|
|
|
c0 = f.cards()[0]
|
|
|
|
assert "three" in c0.q()
|
|
|
|
# it should not be a duplicate
|
|
|
|
assert not f.dupeOrEmpty()
|
|
|
|
# now let's make a duplicate
|
|
|
|
f2 = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
f2["Front"] = "one"
|
|
|
|
f2["Back"] = ""
|
2012-12-21 08:51:59 +01:00
|
|
|
assert f2.dupeOrEmpty()
|
|
|
|
# empty first field should not be permitted either
|
2019-12-25 05:18:34 +01:00
|
|
|
f2["Front"] = " "
|
2012-12-21 08:51:59 +01:00
|
|
|
assert f2.dupeOrEmpty()
|
|
|
|
|
2019-12-25 05:18:34 +01:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
def test_fieldChecksum():
|
2014-06-03 10:38:47 +02:00
|
|
|
deck = getEmptyCol()
|
2012-12-21 08:51:59 +01:00
|
|
|
f = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
f["Front"] = "new"
|
|
|
|
f["Back"] = "new2"
|
2012-12-21 08:51:59 +01:00
|
|
|
deck.addNote(f)
|
2019-12-25 05:18:34 +01:00
|
|
|
assert deck.db.scalar("select csum from notes") == int("c2a6b03f", 16)
|
2012-12-21 08:51:59 +01:00
|
|
|
# changing the val should change the checksum
|
2019-12-25 05:18:34 +01:00
|
|
|
f["Front"] = "newx"
|
2012-12-21 08:51:59 +01:00
|
|
|
f.flush()
|
2019-12-25 05:18:34 +01:00
|
|
|
assert deck.db.scalar("select csum from notes") == int("302811ae", 16)
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
def test_addDelTags():
|
2014-06-03 10:38:47 +02:00
|
|
|
deck = getEmptyCol()
|
2012-12-21 08:51:59 +01:00
|
|
|
f = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
f["Front"] = "1"
|
2012-12-21 08:51:59 +01:00
|
|
|
deck.addNote(f)
|
|
|
|
f2 = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
f2["Front"] = "2"
|
2012-12-21 08:51:59 +01:00
|
|
|
deck.addNote(f2)
|
|
|
|
# adding for a given id
|
|
|
|
deck.tags.bulkAdd([f.id], "foo")
|
2019-12-25 05:18:34 +01:00
|
|
|
f.load()
|
|
|
|
f2.load()
|
2012-12-21 08:51:59 +01:00
|
|
|
assert "foo" in f.tags
|
|
|
|
assert "foo" not in f2.tags
|
|
|
|
# should be canonified
|
|
|
|
deck.tags.bulkAdd([f.id], "foo aaa")
|
|
|
|
f.load()
|
|
|
|
assert f.tags[0] == "aaa"
|
|
|
|
assert len(f.tags) == 2
|
|
|
|
|
2019-12-25 05:18:34 +01:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
def test_timestamps():
|
2014-06-03 10:38:47 +02:00
|
|
|
deck = getEmptyCol()
|
2020-04-25 12:13:46 +02:00
|
|
|
assert len(deck.models.all_names_and_ids()) == len(models)
|
2012-12-21 08:51:59 +01:00
|
|
|
for i in range(100):
|
|
|
|
addBasicModel(deck)
|
2020-04-25 12:13:46 +02:00
|
|
|
assert len(deck.models.all_names_and_ids()) == 100 + len(models)
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2019-12-25 05:18:34 +01:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
def test_furigana():
|
2014-06-03 10:38:47 +02:00
|
|
|
deck = getEmptyCol()
|
2012-12-21 08:51:59 +01:00
|
|
|
mm = deck.models
|
|
|
|
m = mm.current()
|
|
|
|
# filter should work
|
2019-12-25 05:18:34 +01:00
|
|
|
m["tmpls"][0]["qfmt"] = "{{kana:Front}}"
|
2012-12-21 08:51:59 +01:00
|
|
|
mm.save(m)
|
|
|
|
n = deck.newNote()
|
2019-12-25 05:18:34 +01:00
|
|
|
n["Front"] = "foo[abc]"
|
2012-12-21 08:51:59 +01:00
|
|
|
deck.addNote(n)
|
|
|
|
c = n.cards()[0]
|
|
|
|
assert c.q().endswith("abc")
|
|
|
|
# and should avoid sound
|
2019-12-25 05:18:34 +01:00
|
|
|
n["Front"] = "foo[sound:abc.mp3]"
|
2012-12-21 08:51:59 +01:00
|
|
|
n.flush()
|
2020-01-24 02:06:11 +01:00
|
|
|
assert "anki:play" in c.q(reload=True)
|
2012-12-21 08:51:59 +01:00
|
|
|
# it shouldn't throw an error while people are editing
|
2019-12-25 05:18:34 +01:00
|
|
|
m["tmpls"][0]["qfmt"] = "{{kana:}}"
|
2012-12-21 08:51:59 +01:00
|
|
|
mm.save(m)
|
|
|
|
c.q(reload=True)
|
2020-02-16 12:07:40 +01:00
|
|
|
|
|
|
|
|
|
|
|
def test_translate():
|
|
|
|
d = getEmptyCol()
|
2020-02-20 03:48:46 +01:00
|
|
|
no_uni = without_unicode_isolation
|
2020-02-16 12:07:40 +01:00
|
|
|
|
2020-02-23 03:21:12 +01:00
|
|
|
assert (
|
2020-02-27 03:32:37 +01:00
|
|
|
d.tr(TR.CARD_TEMPLATE_RENDERING_FRONT_SIDE_PROBLEM)
|
2020-02-23 03:21:12 +01:00
|
|
|
== "Front template has a problem:"
|
|
|
|
)
|
2020-02-27 03:32:37 +01:00
|
|
|
assert no_uni(d.tr(TR.STATISTICS_REVIEWS, reviews=1)) == "1 review"
|
|
|
|
assert no_uni(d.tr(TR.STATISTICS_REVIEWS, reviews=2)) == "2 reviews"
|
2020-04-06 12:09:44 +02:00
|
|
|
|
|
|
|
|
2020-04-06 12:24:05 +02:00
|
|
|
def test_db_named_args(capsys):
|
2020-04-06 12:09:44 +02:00
|
|
|
sql = "select a, 2+:test5 from b where arg =:foo and x = :test5"
|
|
|
|
args = []
|
|
|
|
kwargs = dict(test5=5, foo="blah")
|
|
|
|
|
|
|
|
s, a = emulate_named_args(sql, args, kwargs)
|
|
|
|
assert s == "select a, 2+?1 from b where arg =?2 and x = ?1"
|
|
|
|
assert a == [5, "blah"]
|
2020-04-06 12:24:05 +02:00
|
|
|
|
|
|
|
# swallow the warning
|
|
|
|
_ = capsys.readouterr()
|