-
-
Notifications
You must be signed in to change notification settings - Fork 297
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
support for retaining bookmarks order on deletion + printing multiline descriptions #818
base: master
Are you sure you want to change the base?
Conversation
fillwidth = columns - INDENT | ||
desc_lines = [s for line in row.desc.splitlines() | ||
for s in textwrap.wrap(line, width=fillwidth) or ['']] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Splitting into lines then wrapping each separately (the fallback value is for empty lines since wrap()
swallows them)
desc_lines = [s for line in row.desc.splitlines() | ||
for s in textwrap.wrap(line, width=fillwidth) or ['']] | ||
TR = str.maketrans(',-', '-,') # we want breaks after commas rather than hyphens | ||
tag_lines = textwrap.wrap(row.tags.translate(TR), width=fillwidth) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
wrap()
inserts linebreaks after hyphens by default; which is not really a good thing when displaying tags. On the other hand, it doesn't insert linebreaks after commas (unless those are followed by whitespace); so what we want is to replace -
with ,
and vice versa before passing tags string to wrap()
(then revert the change during the printing).
print(DESC_STR % line, end='') | ||
ln_num += 1 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Removed ln_num
since it's simpler to just use enumerate()
with self.lock: | ||
for pos, row in reversed(list(enumerate(results))): | ||
self.delete_rec(row[0], delay_commit=True) | ||
for pos, id in reversed(list(enumerate(ids))): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There's no actual guarantee that the input list is sorted by index and contains unique entries; which led to a bunch of warnings about missing indices (likely accompanied by removing wrong records) when I was testing this on certain searches.
if idx not in ids: | ||
ids += (idx,) | ||
|
||
ids = set(args.delete) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We don't actually care about the order, so using set()
is a simpler approach here
@@ -1456,17 +1464,18 @@ class BukuDb: | |||
if delay_delete: | |||
self.conn.commit() | |||
else: | |||
self.commit_delete() | |||
self.commit_delete(retain_order=retain_order) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In most cases, this value is passed directly as the respective parameter of calls which may result in record deletion
self.conn.commit() | ||
self.cur.execute('VACUUM') |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
VACUUM
is necessary to shrink the file after deletion (particularly when deleting a large amount of records). Note that it effectively creates (and commits) a transaction of its own, so placing it before conn.commit()
results in an error.
@@ -1841,28 +1850,34 @@ class BukuDb: | |||
delay_commit : bool | |||
True if record should not be committed to the DB, | |||
leaving commit responsibility to caller. Default is False. | |||
upto : int, optional | |||
If specified, multiple indices are moved at once. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This one is used specifically for optimization of range deletions (whether we're using retain_order
or not, there's no reason to make multiple updates when a single one would suffice, without a worse outcome).
""" | ||
|
||
# Return if the last index left in DB was just deleted | ||
max_id = self.get_max_id() | ||
if not max_id: | ||
if not max_id or (upto and upto < index): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If a range was specified, it needs to be non-empty
with self.lock: | ||
if retain_order or (upto or 0) > index: | ||
step = (max(max_id - upto, upto + 1 - index) if not retain_order else | ||
1 if not upto else upto + 1 - index) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There's 3 possibility here:
- retain-order without upto
- retain-order with upto
- upto without retain-order
The 3rd case only differs from the 2nd one in that we want to support the sub-case where there's more records with larger indices remaining than was in the deleted range ("Indices 9998-10000 moved to 9990-9992"). This is covered by the max()
subexpression.
msg = f'Indices {index+step}-{max_id} moved to {index}-{max_id-step}' | ||
else: | ||
self.cur.execute('UPDATE bookmarks SET id = ? WHERE id = ?', (index, max_id)) | ||
msg = f'Index {max_id} moved to {index}' |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A single UPDATE
should suffice for our purpose
@@ -2114,9 +2138,6 @@ class BukuDb: | |||
return False | |||
|
|||
if self.delete_rec_all(): | |||
with self.lock: | |||
self.cur.execute('VACUUM') | |||
self.conn.commit() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
VACUUM
is now invoked directly by delete_rec_all()
after the commit (same as for other deletion operations)
@@ -6332,22 +6355,17 @@ POSITIONAL ARGUMENTS: | |||
try: | |||
vals = [int(x) for x in args.delete[0].split('-')] | |||
if len(vals) == 2: | |||
bdb.delete_rec(0, vals[0], vals[1], True) | |||
bdb.delete_rec(0, vals[0], vals[1], is_range=True, retain_order=args.retain_order) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's better to specify explicitly what this True
actually stands for
a5462c1
to
6a993a5
Compare
retained_order
parameter for deletion-related operations (when truthy,compactdb()
updates multiple record indices at once in a way that prevents their reordering)--retained-order
CLI option to allow for using said functionality from CLI(...there doesn't seem to be any deletion support in the interactive shell 🤔)
delete_resultset()
@jarun honestly, I'd say it would make sense to always retain ordering in CLI deletions as well, since this at least guarantees consistent bookmarks order (and IDs are not consistent either way). …Alternatively we could drop the "old" DB compression logic altogether; though the new one is slower when deleting a large arbitrary resultset. (Then again, it's not like such updates need to be done very often.)
Screenshots
printing records with multiline data
(in case of tags, we're tricking
data:image/s3,"s3://crabby-images/4660f/4660f189cbfea93fb39ceb9f543f47ec579db4e3" alt="sample data"
data:image/s3,"s3://crabby-images/72635/72635c8771f0d732f31850193470b9d2a737c3ba" alt="usecase"
textwrap
into inserting linebreaks after commas rather than hyphens)(an example of multiline description which looks better when rendered as such)
batch deletion of search results (with retained order of remaining records) + resized DB file
(tail output after removing 9987 records)
data:image/s3,"s3://crabby-images/cdd79/cdd79505fe4cb2a2687c9eac1057f753abeb84c2" alt="deletion log"
data:image/s3,"s3://crabby-images/c6eb9/c6eb9f699fbbf00500358584ddae767788b28e84" alt="remaining records"
data:image/s3,"s3://crabby-images/a0f1f/a0f1fdcb8a15e6abbfa11f4ae9c970d993ce4c9b" alt="file size"
(remaining records still retain their original order)
(the file size got reduced compared to the original database)
batch deletion of random records (with retained order)
(tail output after removing 9995 records)
data:image/s3,"s3://crabby-images/9225a/9225a91958ace6fdb83a666c5cd64fcaa617b36f" alt="deletion log"
data:image/s3,"s3://crabby-images/46a7e/46a7e55302f34e49835c202fb671132c1867fb82" alt="remaining records"
(remaining records still retain their original order)
batch deletion of an index range (with retained order)
yes | buku --nostdin --db generated.db --np --delete 3-9997 --retain-order
(tail output after removing 9995 sequential records)
data:image/s3,"s3://crabby-images/7ccc4/7ccc49429ca28ab48795142b8e1e3b139432cf2e" alt="deletion log"
data:image/s3,"s3://crabby-images/be152/be1524a8c729637b753957bbe0fb9c295267cf41" alt="remaining records"
(remaining records still retain their original order)
batch deletion of an index range without retained order
(removed indices are replaced in a single DB update)
data:image/s3,"s3://crabby-images/54db2/54db249194357b8c829c4d1e2f4c51418f132ada" alt="deletion log"
data:image/s3,"s3://crabby-images/43acf/43acfcd495e2c850c04f417b56a036042a60df13" alt="resulting database"
data:image/s3,"s3://crabby-images/1f7c3/1f7c3071ee4e0fb212c6d90ca3410c7ecbd8f02b" alt="deletion log 2"
data:image/s3,"s3://crabby-images/591c4/591c455b1104fbba2d9897b79081bdcd0998edd4" alt="resulting database 2"
(the last 10 records of resulting database)
(this also works when there's not enough records with larger indices to replace all that got removed)
(the last 4 records of resulting database)
webUI
(GUI deletion uses retained order now since it's a more convenient and intuitively expected UX)
data:image/s3,"s3://crabby-images/26ece/26ece7acb0d03c45e48c33559e30282066cb942a" alt="image"
data:image/s3,"s3://crabby-images/80bb9/80bb95e4412b4dce1ba25fe643b7c6beb8c78084" alt="image"
data:image/s3,"s3://crabby-images/7e63d/7e63de7cd2574719dc6544e325d17ab50c85ab8a" alt="image"