update before_upload()

This commit is contained in:
Damien Elmes 2020-05-17 14:13:21 +10:00
parent 6114836484
commit b9b837e7bd
13 changed files with 82 additions and 47 deletions

View File

@ -237,21 +237,8 @@ class _Collection:
def beforeUpload(self) -> None: def beforeUpload(self) -> None:
"Called before a full upload." "Called before a full upload."
tbls = "notes", "cards", "revlog"
for t in tbls:
self.db.execute("update %s set usn=0 where usn=-1" % t)
# we can save space by removing the log of deletions
self.db.execute("delete from graves")
self._usn += 1
self.models.beforeUpload()
self.decks.beforeUpload()
self.backend.before_upload()
self.modSchema(check=False)
self.ls = self.scm
# ensure db is compacted before upload
self.save(trx=False) self.save(trx=False)
self.db.execute("vacuum") self.backend.before_upload()
self.db.execute("analyze")
self.close(save=False, downgrade=True) self.close(save=False, downgrade=True)
# Object creation helpers # Object creation helpers

View File

@ -566,14 +566,6 @@ class DeckManager:
def nameMap(self) -> dict: def nameMap(self) -> dict:
return dict((d["name"], d) for d in self.all()) return dict((d["name"], d) for d in self.all())
# Sync handling
##########################################################################
def beforeUpload(self) -> None:
for d in self.all():
d["usn"] = 0
self.save()
# Dynamic decks # Dynamic decks
########################################################################## ##########################################################################

View File

@ -20,7 +20,7 @@ NoteType = Dict[str, Any]
Field = Dict[str, Any] Field = Dict[str, Any]
Template = Dict[str, Union[str, int, None]] Template = Dict[str, Union[str, int, None]]
# fixme: syncing, beforeUpload # fixme: syncing
class ModelsDictProxy: class ModelsDictProxy:
@ -479,11 +479,3 @@ and notes.mid = ? and cards.ord = ?""",
print("_availClozeOrds() is deprecated") print("_availClozeOrds() is deprecated")
note = anki.rsbackend.BackendNote(fields=[flds]) note = anki.rsbackend.BackendNote(fields=[flds])
return self.col.backend.cloze_numbers_in_note(note) return self.col.backend.cloze_numbers_in_note(note)
# Sync handling
##########################################################################
def beforeUpload(self) -> None:
for m in self.all():
m["usn"] = 0
self.save()

View File

@ -795,7 +795,7 @@ impl Backend {
} }
fn before_upload(&self) -> Result<()> { fn before_upload(&self) -> Result<()> {
self.with_col(|col| col.transact(None, |col| col.before_upload())) self.with_col(|col| col.before_upload())
} }
fn all_tags(&self) -> Result<pb::AllTagsOut> { fn all_tags(&self) -> Result<pb::AllTagsOut> {

View File

@ -144,10 +144,21 @@ impl Collection {
self.storage.usn(self.server) self.storage.usn(self.server)
} }
pub(crate) fn before_upload(&self) -> Result<()> { /// Prepare for upload. Caller should not create transaction.
self.storage.clear_tag_usns()?; pub(crate) fn before_upload(&mut self) -> Result<()> {
self.storage.clear_deck_conf_usns()?; self.transact(None, |col| {
col.storage.clear_all_graves()?;
Ok(()) col.storage.clear_pending_note_usns()?;
col.storage.clear_pending_card_usns()?;
col.storage.clear_pending_revlog_usns()?;
col.storage.clear_tag_usns()?;
col.storage.clear_deck_conf_usns()?;
col.storage.clear_deck_usns()?;
col.storage.clear_notetype_usns()?;
col.storage.increment_usn()?;
col.storage.set_schema_modified()?;
col.storage.set_last_sync(col.storage.get_schema_mtime()?)
})?;
self.storage.optimize()
} }
} }

View File

@ -184,6 +184,13 @@ impl super::SqliteStorage {
.optional() .optional()
.map_err(Into::into) .map_err(Into::into)
} }
pub(crate) fn clear_pending_card_usns(&self) -> Result<()> {
self.db
.prepare("update cards set usn = 0 where usn = -1")?
.execute(NO_PARAMS)?;
Ok(())
}
} }
#[cfg(test)] #[cfg(test)]

View File

@ -220,6 +220,13 @@ impl SqliteStorage {
.map_err(Into::into) .map_err(Into::into)
} }
pub(crate) fn clear_deck_usns(&self) -> Result<()> {
self.db
.prepare("update decks set usn = 0 where usn != 0")?
.execute(NO_PARAMS)?;
Ok(())
}
// Upgrading/downgrading/legacy // Upgrading/downgrading/legacy
pub(super) fn add_default_deck(&self, i18n: &I18n) -> Result<()> { pub(super) fn add_default_deck(&self, i18n: &I18n) -> Result<()> {

View File

@ -80,13 +80,9 @@ impl SqliteStorage {
} }
pub(crate) fn clear_deck_conf_usns(&self) -> Result<()> { pub(crate) fn clear_deck_conf_usns(&self) -> Result<()> {
for mut conf in self.all_deck_config()? { self.db
if conf.usn.0 != 0 { .prepare("update deck_config set usn = 0 where usn != 0")?
conf.usn.0 = 0; .execute(NO_PARAMS)?;
self.update_deck_conf(&conf)?;
}
}
Ok(()) Ok(())
} }

View File

@ -19,7 +19,6 @@ impl SqliteStorage {
Ok(()) Ok(())
} }
#[allow(dead_code)]
pub(crate) fn clear_all_graves(&self) -> Result<()> { pub(crate) fn clear_all_graves(&self) -> Result<()> {
self.db.execute("delete from graves", NO_PARAMS)?; self.db.execute("delete from graves", NO_PARAMS)?;
Ok(()) Ok(())

View File

@ -7,7 +7,7 @@ use crate::{
tags::{join_tags, split_tags}, tags::{join_tags, split_tags},
timestamp::TimestampMillis, timestamp::TimestampMillis,
}; };
use rusqlite::{params, OptionalExtension}; use rusqlite::{params, OptionalExtension, NO_PARAMS};
fn split_fields(fields: &str) -> Vec<String> { fn split_fields(fields: &str) -> Vec<String> {
fields.split('\x1f').map(Into::into).collect() fields.split('\x1f').map(Into::into).collect()
@ -88,4 +88,11 @@ impl super::SqliteStorage {
.query_row(&[nid], |r| r.get(0)) .query_row(&[nid], |r| r.get(0))
.map_err(Into::into) .map_err(Into::into)
} }
pub(crate) fn clear_pending_note_usns(&self) -> Result<()> {
self.db
.prepare("update notes set usn = 0 where usn = -1")?
.execute(NO_PARAMS)?;
Ok(())
}
} }

View File

@ -308,6 +308,13 @@ and ord in ",
.collect() .collect()
} }
pub(crate) fn clear_notetype_usns(&self) -> Result<()> {
self.db
.prepare("update notetypes set usn = 0 where usn != 0")?
.execute(NO_PARAMS)?;
Ok(())
}
// Upgrading/downgrading/legacy // Upgrading/downgrading/legacy
pub(crate) fn get_all_notetypes_as_schema11( pub(crate) fn get_all_notetypes_as_schema11(

View File

@ -12,4 +12,11 @@ impl SqliteStorage {
.execute(NO_PARAMS) .execute(NO_PARAMS)
.map_err(Into::into) .map_err(Into::into)
} }
pub(crate) fn clear_pending_revlog_usns(&self) -> Result<()> {
self.db
.prepare("update revlog set usn = 0 where usn = -1")?
.execute(NO_PARAMS)?;
Ok(())
}
} }

View File

@ -275,6 +275,13 @@ impl SqliteStorage {
} }
} }
pub(crate) fn increment_usn(&self) -> Result<()> {
self.db
.prepare_cached("update col set usn = usn + 1")?
.execute(NO_PARAMS)?;
Ok(())
}
pub(crate) fn creation_stamp(&self) -> Result<TimestampSecs> { pub(crate) fn creation_stamp(&self) -> Result<TimestampSecs> {
self.db self.db
.prepare_cached("select crt from col")? .prepare_cached("select crt from col")?
@ -296,6 +303,22 @@ impl SqliteStorage {
Ok(()) Ok(())
} }
pub(crate) fn get_schema_mtime(&self) -> Result<TimestampSecs> {
self.db
.prepare_cached("select scm from col")?
.query_and_then(NO_PARAMS, |r| r.get(0))?
.next()
.ok_or_else(|| AnkiError::invalid_input("missing col"))?
.map_err(Into::into)
}
pub(crate) fn set_last_sync(&self, stamp: TimestampSecs) -> Result<()> {
self.db
.prepare("update col set ls = ?")?
.execute(&[stamp])?;
Ok(())
}
////////////////////////////////////////// //////////////////////////////////////////
/// true if corrupt/can't access /// true if corrupt/can't access
@ -312,7 +335,7 @@ impl SqliteStorage {
} }
pub(crate) fn optimize(&self) -> Result<()> { pub(crate) fn optimize(&self) -> Result<()> {
self.db.execute_batch("vacuum")?; self.db.execute_batch("vacuum; analyze")?;
Ok(()) Ok(())
} }