anki/rslib/src/notes.rs

420 lines
13 KiB
Rust
Raw Normal View History

// Copyright: Ankitects Pty Ltd and contributors
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
use crate::{
backend_proto as pb,
collection::Collection,
decks::DeckID,
define_newtype,
err::{AnkiError, Result},
notetype::{CardGenContext, NoteField, NoteType, NoteTypeID},
text::{ensure_string_in_nfc, strip_html_preserving_image_filenames},
timestamp::TimestampSecs,
types::Usn,
};
use itertools::Itertools;
use num_integer::Integer;
use std::{collections::HashSet, convert::TryInto};
2020-03-26 05:42:43 +01:00
define_newtype!(NoteID, i64);
2020-04-14 07:41:01 +02:00
// fixme: ensure nulls and x1f not in field contents
#[derive(Debug)]
2020-04-14 07:41:01 +02:00
pub struct Note {
2020-03-26 05:42:43 +01:00
pub id: NoteID,
2020-04-14 07:41:01 +02:00
pub guid: String,
2020-03-26 06:00:24 +01:00
pub ntid: NoteTypeID,
pub mtime: TimestampSecs,
pub usn: Usn,
2020-04-14 07:41:01 +02:00
pub tags: Vec<String>,
pub(crate) fields: Vec<String>,
2020-04-14 07:41:01 +02:00
pub(crate) sort_field: Option<String>,
pub(crate) checksum: Option<u32>,
}
impl Note {
pub(crate) fn new(notetype: &NoteType) -> Self {
Note {
id: NoteID(0),
guid: guid(),
ntid: notetype.id,
mtime: TimestampSecs(0),
usn: Usn(0),
tags: vec![],
fields: vec!["".to_string(); notetype.fields.len()],
sort_field: None,
checksum: None,
}
}
pub fn fields(&self) -> &Vec<String> {
&self.fields
}
pub fn set_field(&mut self, idx: usize, text: impl Into<String>) -> Result<()> {
if idx >= self.fields.len() {
return Err(AnkiError::invalid_input(
"field idx out of range".to_string(),
));
}
self.fields[idx] = text.into();
Ok(())
}
2020-04-14 07:41:01 +02:00
/// Prepare note for saving to the database. Does not mark it as modified.
pub fn prepare_for_update(&mut self, nt: &NoteType, normalize_text: bool) -> Result<()> {
assert!(nt.id == self.ntid);
if nt.fields.len() != self.fields.len() {
return Err(AnkiError::invalid_input(format!(
"note has {} fields, expected {}",
self.fields.len(),
nt.fields.len()
)));
}
if normalize_text {
for field in &mut self.fields {
ensure_string_in_nfc(field);
}
}
2020-04-14 07:41:01 +02:00
let field1_nohtml = strip_html_preserving_image_filenames(&self.fields()[0]);
let checksum = field_checksum(field1_nohtml.as_ref());
let sort_field = if nt.config.sort_field_idx == 0 {
2020-04-14 07:41:01 +02:00
field1_nohtml
} else {
strip_html_preserving_image_filenames(
self.fields
.get(nt.config.sort_field_idx as usize)
2020-04-14 07:41:01 +02:00
.map(AsRef::as_ref)
.unwrap_or(""),
)
};
self.sort_field = Some(sort_field.into());
self.checksum = Some(checksum);
Ok(())
2020-04-14 07:41:01 +02:00
}
pub(crate) fn set_modified(&mut self, usn: Usn) {
self.mtime = TimestampSecs::now();
self.usn = usn;
}
pub(crate) fn nonempty_fields<'a>(&self, fields: &'a [NoteField]) -> HashSet<&'a str> {
self.fields
.iter()
.enumerate()
.filter_map(|(ord, s)| {
if s.trim().is_empty() {
None
} else {
fields.get(ord).map(|f| f.name.as_str())
}
})
.collect()
}
}
impl From<Note> for pb::Note {
fn from(n: Note) -> Self {
pb::Note {
id: n.id.0,
guid: n.guid,
ntid: n.ntid.0,
mtime_secs: n.mtime.0 as u32,
usn: n.usn.0,
tags: n.tags,
fields: n.fields,
}
}
}
impl From<pb::Note> for Note {
fn from(n: pb::Note) -> Self {
Note {
id: NoteID(n.id),
guid: n.guid,
ntid: NoteTypeID(n.ntid),
mtime: TimestampSecs(n.mtime_secs as i64),
usn: Usn(n.usn),
tags: n.tags,
fields: n.fields,
sort_field: None,
checksum: None,
}
}
}
/// Text must be passed to strip_html_preserving_image_filenames() by
/// caller prior to passing in here.
2020-03-17 03:31:54 +01:00
pub(crate) fn field_checksum(text: &str) -> u32 {
let digest = sha1::Sha1::from(text).digest().bytes();
u32::from_be_bytes(digest[..4].try_into().unwrap())
}
fn guid() -> String {
anki_base91(rand::random())
}
fn anki_base91(mut n: u64) -> String {
let table = b"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ\
0123456789!#$%&()*+,-./:;<=>?@[]^_`{|}~";
let mut buf = String::new();
while n > 0 {
let (q, r) = n.div_rem(&(table.len() as u64));
buf.push(table[r as usize] as char);
n = q;
}
buf.chars().rev().collect()
}
impl Collection {
fn canonify_note_tags(&self, note: &mut Note, usn: Usn) -> Result<()> {
// fixme: avoid the excess split/join
note.tags = self
.canonify_tags(&note.tags.join(" "), usn)?
.0
.split(' ')
.map(Into::into)
.collect();
Ok(())
}
pub fn add_note(&mut self, note: &mut Note, did: DeckID) -> Result<()> {
self.transact(None, |col| {
let nt = col
2020-04-18 00:48:10 +02:00
.get_notetype(note.ntid)?
.ok_or_else(|| AnkiError::invalid_input("missing note type"))?;
let ctx = CardGenContext::new(&nt, col.usn()?);
let norm = col.normalize_note_text();
col.add_note_inner(&ctx, note, did, norm)
})
}
pub(crate) fn add_note_inner(
&mut self,
ctx: &CardGenContext,
note: &mut Note,
did: DeckID,
normalize_text: bool,
) -> Result<()> {
self.canonify_note_tags(note, ctx.usn)?;
note.prepare_for_update(&ctx.notetype, normalize_text)?;
note.set_modified(ctx.usn);
self.storage.add_note(note)?;
self.generate_cards_for_new_note(ctx, note, did)
}
pub fn update_note(&mut self, note: &mut Note) -> Result<()> {
self.transact(None, |col| {
let nt = col
.get_notetype(note.ntid)?
.ok_or_else(|| AnkiError::invalid_input("missing note type"))?;
let ctx = CardGenContext::new(&nt, col.usn()?);
let norm = col.normalize_note_text();
col.update_note_inner_generating_cards(&ctx, note, true, norm)
})
}
pub(crate) fn update_note_inner_generating_cards(
&mut self,
ctx: &CardGenContext,
note: &mut Note,
mark_note_modified: bool,
normalize_text: bool,
) -> Result<()> {
self.update_note_inner_without_cards(
note,
ctx.notetype,
ctx.usn,
mark_note_modified,
normalize_text,
)?;
self.generate_cards_for_existing_note(ctx, note)
}
pub(crate) fn update_note_inner_without_cards(
&mut self,
note: &mut Note,
nt: &NoteType,
usn: Usn,
mark_note_modified: bool,
normalize_text: bool,
) -> Result<()> {
self.canonify_note_tags(note, usn)?;
note.prepare_for_update(nt, normalize_text)?;
if mark_note_modified {
note.set_modified(usn);
}
self.storage.update_note(note)
}
/// Remove a note. Cards must already have been deleted.
pub(crate) fn remove_note_only(&mut self, nid: NoteID, usn: Usn) -> Result<()> {
if let Some(_note) = self.storage.get_note(nid)? {
// fixme: undo
self.storage.remove_note(nid)?;
self.storage.add_note_grave(nid, usn)?;
}
Ok(())
}
/// Update cards and field cache after notes modified externally.
/// If gencards is false, skip card generation.
pub(crate) fn after_note_updates(
&mut self,
nids: &[NoteID],
usn: Usn,
generate_cards: bool,
mark_notes_modified: bool,
) -> Result<()> {
let nids_by_notetype = self.storage.note_ids_by_notetype(nids)?;
let norm = self.normalize_note_text();
for (ntid, group) in &nids_by_notetype.into_iter().group_by(|tup| tup.0) {
let nt = self
.get_notetype(ntid)?
.ok_or_else(|| AnkiError::invalid_input("missing note type"))?;
let genctx = CardGenContext::new(&nt, usn);
for (_, nid) in group {
let mut note = self.storage.get_note(nid)?.unwrap();
if generate_cards {
self.update_note_inner_generating_cards(
&genctx,
&mut note,
mark_notes_modified,
norm,
)?;
} else {
self.update_note_inner_without_cards(
&mut note,
&genctx.notetype,
usn,
mark_notes_modified,
norm,
)?;
}
}
}
Ok(())
}
}
#[cfg(test)]
mod test {
use super::{anki_base91, field_checksum};
use crate::{collection::open_test_collection, config::ConfigKey, decks::DeckID, err::Result};
#[test]
fn test_base91() {
// match the python implementation for now
assert_eq!(anki_base91(0), "");
assert_eq!(anki_base91(1), "b");
assert_eq!(anki_base91(u64::max_value()), "Rj&Z5m[>Zp");
assert_eq!(anki_base91(1234567890), "saAKk");
}
#[test]
fn test_field_checksum() {
assert_eq!(field_checksum("test"), 2840236005);
assert_eq!(field_checksum("今日"), 1464653051);
}
#[test]
fn adding_cards() -> Result<()> {
let mut col = open_test_collection();
let nt = col
.get_notetype_by_name("basic (and reversed card)")?
.unwrap();
let mut note = nt.new_note();
// if no cards are generated, 1 card is added
col.add_note(&mut note, DeckID(1)).unwrap();
let existing = col.storage.existing_cards_for_note(note.id)?;
assert_eq!(existing.len(), 1);
assert_eq!(existing[0].ord, 0);
// nothing changes if the first field is filled
note.fields[0] = "test".into();
col.update_note(&mut note).unwrap();
let existing = col.storage.existing_cards_for_note(note.id)?;
assert_eq!(existing.len(), 1);
assert_eq!(existing[0].ord, 0);
// second field causes another card to be generated
note.fields[1] = "test".into();
col.update_note(&mut note).unwrap();
let existing = col.storage.existing_cards_for_note(note.id)?;
assert_eq!(existing.len(), 2);
assert_eq!(existing[1].ord, 1);
// cloze cards also generate card 0 if no clozes are found
2020-04-21 05:49:40 +02:00
let nt = col.get_notetype_by_name("cloze")?.unwrap();
let mut note = nt.new_note();
col.add_note(&mut note, DeckID(1)).unwrap();
2020-04-21 05:49:40 +02:00
let existing = col.storage.existing_cards_for_note(note.id)?;
assert_eq!(existing.len(), 1);
assert_eq!(existing[0].ord, 0);
2020-04-21 06:24:19 +02:00
assert_eq!(existing[0].original_deck_id, DeckID(1));
2020-04-21 05:49:40 +02:00
// and generate cards for any cloze deletions
2020-04-21 05:49:40 +02:00
note.fields[0] = "{{c1::foo}} {{c2::bar}} {{c3::baz}} {{c0::quux}} {{c501::over}}".into();
col.update_note(&mut note)?;
let existing = col.storage.existing_cards_for_note(note.id)?;
let mut ords = existing.iter().map(|a| a.ord).collect::<Vec<_>>();
ords.sort();
assert_eq!(ords, vec![0, 1, 2, 499]);
Ok(())
}
#[test]
fn normalization() -> Result<()> {
let mut col = open_test_collection();
let nt = col.get_notetype_by_name("Basic")?.unwrap();
let mut note = nt.new_note();
note.fields[0] = "\u{fa47}".into();
col.add_note(&mut note, DeckID(1))?;
assert_eq!(note.fields[0], "\u{6f22}");
// non-normalized searches should be converted
assert_eq!(
col.search_cards("\u{fa47}", crate::search::SortMode::NoOrder)?
.len(),
1
);
assert_eq!(
col.search_cards("front:\u{fa47}", crate::search::SortMode::NoOrder)?
.len(),
1
);
col.remove_note_only(note.id, col.usn()?)?;
// if normalization turned off, note text is entered as-is
let mut note = nt.new_note();
note.fields[0] = "\u{fa47}".into();
col.set_config(ConfigKey::NormalizeNoteText, &false)
.unwrap();
col.add_note(&mut note, DeckID(1))?;
assert_eq!(note.fields[0], "\u{fa47}");
// normalized searches won't match
assert_eq!(
col.search_cards("\u{6f22}", crate::search::SortMode::NoOrder)?
.len(),
0
);
// but original characters will
assert_eq!(
col.search_cards("\u{fa47}", crate::search::SortMode::NoOrder)?
.len(),
1
);
Ok(())
}
}