Support custom db file. Optimizations.
This commit is contained in:
parent
b387fc95e1
commit
5e0bc15e28
14
README.md
14
README.md
@ -39,8 +39,9 @@ Though a terminal utility, it's possible to add bookmarks to `buku` without touc
|
|||||||
- [Cmdline options](#cmdline-options)
|
- [Cmdline options](#cmdline-options)
|
||||||
- [Operational notes](#operational-notes)
|
- [Operational notes](#operational-notes)
|
||||||
- [GUI integration](#gui-integration)
|
- [GUI integration](#gui-integration)
|
||||||
- [Add bookmarks to buku](#add-bookmarks-to-buku)
|
- [Add bookmarks from anywhere](#add-bookmarks-from-anywhere)
|
||||||
- [Import bookmarks to browser](#import-bookmarks-to-browser)
|
- [Import bookmarks to browser](#import-bookmarks-to-browser)
|
||||||
|
- [As a library](#as-a-library)
|
||||||
- [Examples](#examples)
|
- [Examples](#examples)
|
||||||
- [Contributions](#contributions)
|
- [Contributions](#contributions)
|
||||||
- [Mentions](#mentions)
|
- [Mentions](#mentions)
|
||||||
@ -246,7 +247,7 @@ Shell completion scripts for Bash, Fish and Zsh can be found in respective subdi
|
|||||||
|
|
||||||
`buku` can integrate in a GUI environment with simple tweaks.
|
`buku` can integrate in a GUI environment with simple tweaks.
|
||||||
|
|
||||||
### Add bookmarks to buku
|
### Add bookmarks from anywhere
|
||||||
|
|
||||||
With support for piped input, it's possible to add bookmarks to `buku` using keyboard shortcuts on Linux and OS X. CLIPBOARD (plus PRIMARY on Linux) text selections can be added directly this way. The additional utility required is `xsel` (on Linux) or `pbpaste` (on OS X).
|
With support for piped input, it's possible to add bookmarks to `buku` using keyboard shortcuts on Linux and OS X. CLIPBOARD (plus PRIMARY on Linux) text selections can be added directly this way. The additional utility required is `xsel` (on Linux) or `pbpaste` (on OS X).
|
||||||
|
|
||||||
@ -301,6 +302,15 @@ To export specific tags, run:
|
|||||||
$ buku --export path_to_bookmarks.html --tag tag 1, tag 2
|
$ buku --export path_to_bookmarks.html --tag tag 1, tag 2
|
||||||
Once exported, import the html file in your browser.
|
Once exported, import the html file in your browser.
|
||||||
|
|
||||||
|
## As a library
|
||||||
|
|
||||||
|
`buku` can be used as a powerful bookmark management library. All functionality are available through carefully designed APIs. `main()` is a good usage example. It's also possible to use a custom database file in multi-user scenarios. Check out the documentation for the following APIs which accept an optional argument as database file:
|
||||||
|
|
||||||
|
BukuDb.initdb(dbfile=None)
|
||||||
|
BukuCrypt.encrypt_file(iterations, dbfile=None)
|
||||||
|
BukuCrypt.decrypt_file(iterations, dbfile=None)
|
||||||
|
NOTE: This flexibility is not exposed in the program.
|
||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
|
|
||||||
1. **Add** a bookmark with **tags** `linux news` and `open source`, **comment** `Informative website on Linux and open source`, **fetch page title** from the web:
|
1. **Add** a bookmark with **tags** `linux news` and `open source`, **comment** `Informative website on Linux and open source`, **fetch page title** from the web:
|
||||||
|
203
buku.py
203
buku.py
@ -55,7 +55,7 @@ http_handler = None # urllib3 PoolManager handler
|
|||||||
htmlparser = None # Use a single HTML Parser instance
|
htmlparser = None # Use a single HTML Parser instance
|
||||||
|
|
||||||
# Disguise as Firefox on Ubuntu
|
# Disguise as Firefox on Ubuntu
|
||||||
USER_AGENT = ('Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:48.0) Gecko/20100101 Firefox/48.0')
|
USER_AGENT = 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:48.0) Gecko/20100101 Firefox/48.0'
|
||||||
|
|
||||||
# Crypto globals
|
# Crypto globals
|
||||||
BLOCKSIZE = 65536
|
BLOCKSIZE = 65536
|
||||||
@ -136,10 +136,11 @@ class BukuCrypt:
|
|||||||
return hasher.digest()
|
return hasher.digest()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def encrypt_file(iterations):
|
def encrypt_file(iterations, dbfile=None):
|
||||||
'''Encrypt the bookmarks database file
|
'''Encrypt the bookmarks database file
|
||||||
|
|
||||||
:param iterations: number of iterations for key generation
|
:param iterations: number of iterations for key generation
|
||||||
|
:param dbfile: custom database file path (including filename)
|
||||||
'''
|
'''
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -157,14 +158,20 @@ class BukuCrypt:
|
|||||||
logger.error('Iterations must be >= 1')
|
logger.error('Iterations must be >= 1')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
dbpath = os.path.join(BukuDb.get_dbdir_path(), 'bookmarks.db')
|
if not dbfile:
|
||||||
encpath = '%s.enc' % dbpath
|
dbfile = os.path.join(BukuDb.get_default_dbdir(), 'bookmarks.db')
|
||||||
if not os.path.exists(dbpath):
|
encfile = '%s.enc' % dbfile
|
||||||
logger.error('%s missing. Already encrypted?', dbpath)
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
# If both encrypted file and flat file exist, error out
|
db_exists = os.path.exists(dbfile)
|
||||||
if os.path.exists(dbpath) and os.path.exists(encpath):
|
enc_exists = os.path.exists(encfile)
|
||||||
|
|
||||||
|
if db_exists and not enc_exists:
|
||||||
|
pass
|
||||||
|
elif not db_exists:
|
||||||
|
logger.error('%s missing. Already encrypted?', dbfile)
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
# db_exists and enc_exists
|
||||||
logger.error('Both encrypted and flat DB files exist!')
|
logger.error('Both encrypted and flat DB files exist!')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
@ -178,8 +185,12 @@ class BukuCrypt:
|
|||||||
logger.error('Passwords do not match')
|
logger.error('Passwords do not match')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
# Get SHA256 hash of DB file
|
try:
|
||||||
dbhash = BukuCrypt.get_filehash(dbpath)
|
# Get SHA256 hash of DB file
|
||||||
|
dbhash = BukuCrypt.get_filehash(dbfile)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
# Generate random 256-bit salt and key
|
# Generate random 256-bit salt and key
|
||||||
salt = os.urandom(SALT_SIZE)
|
salt = os.urandom(SALT_SIZE)
|
||||||
@ -194,10 +205,10 @@ class BukuCrypt:
|
|||||||
modes.CBC(iv),
|
modes.CBC(iv),
|
||||||
backend=default_backend()
|
backend=default_backend()
|
||||||
).encryptor()
|
).encryptor()
|
||||||
filesize = os.path.getsize(dbpath)
|
filesize = os.path.getsize(dbfile)
|
||||||
|
|
||||||
with open(dbpath, 'rb') as infp:
|
try:
|
||||||
with open(encpath, 'wb') as outfp:
|
with open(dbfile, 'rb') as infp, open(encfile, 'wb') as outfp:
|
||||||
outfp.write(struct.pack('<Q', filesize))
|
outfp.write(struct.pack('<Q', filesize))
|
||||||
outfp.write(salt)
|
outfp.write(salt)
|
||||||
outfp.write(iv)
|
outfp.write(iv)
|
||||||
@ -214,15 +225,20 @@ class BukuCrypt:
|
|||||||
|
|
||||||
outfp.write(encryptor.update(chunk) + encryptor.finalize())
|
outfp.write(encryptor.update(chunk) + encryptor.finalize())
|
||||||
|
|
||||||
os.remove(dbpath)
|
os.remove(dbfile)
|
||||||
print('File encrypted')
|
print('File encrypted')
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def decrypt_file(iterations):
|
def decrypt_file(iterations, dbfile=None):
|
||||||
'''Decrypt the bookmarks database file
|
'''Decrypt the bookmarks database file
|
||||||
|
|
||||||
:param iterations: number of iterations for key generation
|
:param iterations: number of iterations for key generation
|
||||||
|
:param dbfile: custom database file path (including filename)
|
||||||
|
: The '.enc' suffix must be omitted.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -240,14 +256,24 @@ class BukuCrypt:
|
|||||||
logger.error('Decryption failed')
|
logger.error('Decryption failed')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
dbpath = os.path.join(BukuDb.get_dbdir_path(), 'bookmarks.db')
|
if not dbfile:
|
||||||
encpath = '%s.enc' % dbpath
|
dbfile = os.path.join(BukuDb.get_default_dbdir(), 'bookmarks.db')
|
||||||
if not os.path.exists(encpath):
|
else:
|
||||||
logger.error('%s missing', encpath)
|
dbfile = os.path.abspath(dbfile)
|
||||||
sys.exit(1)
|
dbpath, filename = os.path.split(dbfile)
|
||||||
|
|
||||||
# If both encrypted file and flat file exist, error out
|
encfile = '%s.enc' % dbfile
|
||||||
if os.path.exists(dbpath) and os.path.exists(encpath):
|
|
||||||
|
enc_exists = os.path.exists(encfile)
|
||||||
|
db_exists = os.path.exists(dbfile)
|
||||||
|
|
||||||
|
if enc_exists and not db_exists:
|
||||||
|
pass
|
||||||
|
elif not enc_exists:
|
||||||
|
logger.error('%s missing', encfile)
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
# db_exists and enc_exists
|
||||||
logger.error('Both encrypted and flat DB files exist!')
|
logger.error('Both encrypted and flat DB files exist!')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
@ -257,66 +283,76 @@ class BukuCrypt:
|
|||||||
logger.error('Decryption failed')
|
logger.error('Decryption failed')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
with open(encpath, 'rb') as infp:
|
try:
|
||||||
origsize = struct.unpack('<Q', infp.read(struct.calcsize('Q')))[0]
|
with open(encfile, 'rb') as infp:
|
||||||
|
size = struct.unpack('<Q', infp.read(struct.calcsize('Q')))[0]
|
||||||
|
|
||||||
# Read 256-bit salt and generate key
|
# Read 256-bit salt and generate key
|
||||||
salt = infp.read(32)
|
salt = infp.read(32)
|
||||||
key = ('%s%s' % (password,
|
key = ('%s%s' % (password,
|
||||||
salt.decode('utf-8', 'replace'))).encode('utf-8')
|
salt.decode('utf-8', 'replace'))).encode('utf-8')
|
||||||
for _ in range(iterations):
|
for _ in range(iterations):
|
||||||
key = sha256(key).digest()
|
key = sha256(key).digest()
|
||||||
|
|
||||||
iv = infp.read(16)
|
iv = infp.read(16)
|
||||||
decryptor = Cipher(
|
decryptor = Cipher(
|
||||||
algorithms.AES(key),
|
algorithms.AES(key),
|
||||||
modes.CBC(iv),
|
modes.CBC(iv),
|
||||||
backend=default_backend(),
|
backend=default_backend(),
|
||||||
).decryptor()
|
).decryptor()
|
||||||
|
|
||||||
# Get original DB file's SHA256 hash from encrypted file
|
# Get original DB file's SHA256 hash from encrypted file
|
||||||
enchash = infp.read(32)
|
enchash = infp.read(32)
|
||||||
|
|
||||||
with open(dbpath, 'wb') as outfp:
|
with open(dbfile, 'wb') as outfp:
|
||||||
while True:
|
while True:
|
||||||
chunk = infp.read(CHUNKSIZE)
|
chunk = infp.read(CHUNKSIZE)
|
||||||
if len(chunk) == 0:
|
if len(chunk) == 0:
|
||||||
break
|
break
|
||||||
|
|
||||||
outfp.write(decryptor.update(chunk) + decryptor.finalize())
|
outfp.write(
|
||||||
|
decryptor.update(chunk) + decryptor.finalize())
|
||||||
|
|
||||||
outfp.truncate(origsize)
|
outfp.truncate(size)
|
||||||
|
|
||||||
# Match hash of generated file with that of original DB file
|
# Match hash of generated file with that of original DB file
|
||||||
dbhash = BukuCrypt.get_filehash(dbpath)
|
dbhash = BukuCrypt.get_filehash(dbfile)
|
||||||
if dbhash != enchash:
|
if dbhash != enchash:
|
||||||
os.remove(dbpath)
|
os.remove(dbfile)
|
||||||
logger.error('Decryption failed')
|
logger.error('Decryption failed')
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
os.remove(encfile)
|
||||||
|
print('File decrypted')
|
||||||
|
except struct.error:
|
||||||
|
logger.error('Tainted file')
|
||||||
|
sys.exit(1)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
else:
|
|
||||||
os.remove(encpath)
|
|
||||||
print('File decrypted')
|
|
||||||
|
|
||||||
|
|
||||||
class BukuDb:
|
class BukuDb:
|
||||||
|
|
||||||
def __init__(self, json=False, field_filter=0, immutable=-1, chatty=False):
|
def __init__(self, json=False, field_filter=0, immutable=-1, chatty=False,
|
||||||
|
dbfile=None):
|
||||||
'''Database initialization API
|
'''Database initialization API
|
||||||
|
|
||||||
:param json: print results in json format
|
:param json: print results in json format
|
||||||
:param field_filter: bookmark print format specifier
|
:param field_filter: bookmark print format specifier
|
||||||
:param immutable: disable title fetch from web
|
:param immutable: disable title fetch from web
|
||||||
:param chatty: set the verbosity of the APIs
|
:param chatty: set the verbosity of the APIs
|
||||||
|
:param dbfile: custom database file path (including filename)
|
||||||
'''
|
'''
|
||||||
|
|
||||||
self.conn, self.cur = BukuDb.initdb()
|
self.conn, self.cur = BukuDb.initdb(dbfile)
|
||||||
self.json = json
|
self.json = json
|
||||||
self.field_filter = field_filter
|
self.field_filter = field_filter
|
||||||
self.immutable = immutable
|
self.immutable = immutable
|
||||||
self.chatty = chatty
|
self.chatty = chatty
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_dbdir_path():
|
def get_default_dbdir():
|
||||||
'''Determine the directory path where dbfile will be stored:
|
'''Determine the directory path where dbfile will be stored:
|
||||||
if $XDG_DATA_HOME is defined, use it
|
if $XDG_DATA_HOME is defined, use it
|
||||||
else if $HOME exists, use it
|
else if $HOME exists, use it
|
||||||
@ -336,30 +372,48 @@ class BukuDb:
|
|||||||
return os.path.join(data_home, 'buku')
|
return os.path.join(data_home, 'buku')
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def initdb():
|
def initdb(dbfile=None):
|
||||||
'''Initialize the database connection. Create DB
|
'''Initialize the database connection. Create DB
|
||||||
file and/or bookmarks table if they don't exist.
|
file and/or bookmarks table if they don't exist.
|
||||||
Alert on encryption options on first execution.
|
Alert on encryption options on first execution.
|
||||||
|
|
||||||
|
:param dbfile: custom database file path (including filename)
|
||||||
:return: (connection, cursor) tuple
|
:return: (connection, cursor) tuple
|
||||||
'''
|
'''
|
||||||
|
|
||||||
dbpath = BukuDb.get_dbdir_path()
|
if not dbfile:
|
||||||
if not os.path.exists(dbpath):
|
dbpath = BukuDb.get_default_dbdir()
|
||||||
os.makedirs(dbpath)
|
filename = 'bookmarks.db'
|
||||||
|
dbfile = os.path.join(dbpath, filename)
|
||||||
|
else:
|
||||||
|
dbfile = os.path.abspath(dbfile)
|
||||||
|
dbpath, filename = os.path.split(dbfile)
|
||||||
|
|
||||||
dbfile = os.path.join(dbpath, 'bookmarks.db')
|
encfile = dbfile + '.enc'
|
||||||
|
|
||||||
encpath = os.path.join(dbpath, 'bookmarks.db.enc')
|
try:
|
||||||
# Notify if DB file needs to be decrypted first
|
if not os.path.exists(dbpath):
|
||||||
if os.path.exists(encpath) and not os.path.exists(dbfile):
|
os.makedirs(dbpath)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
|
os.exit(1)
|
||||||
|
|
||||||
|
db_exists = os.path.exists(dbfile)
|
||||||
|
enc_exists = os.path.exists(encfile)
|
||||||
|
|
||||||
|
if db_exists and not enc_exists:
|
||||||
|
pass
|
||||||
|
elif enc_exists and not db_exists:
|
||||||
logger.error('Unlock database first')
|
logger.error('Unlock database first')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
# Show info on first creation
|
elif db_exists and enc_exists:
|
||||||
if not os.path.exists(dbfile):
|
logger.error('Both encrypted and flat DB files exist!')
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
# not db_exists and not enc_exists
|
||||||
print('DB file is being created at \x1b[1m%s\x1b[0m.' % dbfile)
|
print('DB file is being created at \x1b[1m%s\x1b[0m.' % dbfile)
|
||||||
print('You should \x1b[1mencrypt it\x1b[0m later.')
|
print('You should \x1b[1mencrypt it\x1b[0m later.\n')
|
||||||
|
|
||||||
try:
|
try:
|
||||||
# Create a connection
|
# Create a connection
|
||||||
@ -1816,9 +1870,6 @@ Webpage: https://github.com/jarun/Buku
|
|||||||
self.print_extended_help(file)
|
self.print_extended_help(file)
|
||||||
|
|
||||||
|
|
||||||
'''main starts here'''
|
|
||||||
|
|
||||||
|
|
||||||
# Handle piped input
|
# Handle piped input
|
||||||
def piped_input(argv, pipeargs=None):
|
def piped_input(argv, pipeargs=None):
|
||||||
if not sys.stdin.isatty():
|
if not sys.stdin.isatty():
|
||||||
@ -1827,6 +1878,9 @@ def piped_input(argv, pipeargs=None):
|
|||||||
pipeargs.extend(s.split())
|
pipeargs.extend(s.split())
|
||||||
|
|
||||||
|
|
||||||
|
'''main starts here'''
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
global tags_in, title_in, desc_in
|
global tags_in, title_in, desc_in
|
||||||
|
|
||||||
@ -2017,8 +2071,7 @@ def main():
|
|||||||
BukuCrypt.decrypt_file(args.unlock)
|
BukuCrypt.decrypt_file(args.unlock)
|
||||||
|
|
||||||
# Initialize the database and get handles, set verbose by default
|
# Initialize the database and get handles, set verbose by default
|
||||||
bdb = BukuDb(args.json, args.format, args.immutable,
|
bdb = BukuDb(args.json, args.format, args.immutable, not args.tacit)
|
||||||
not args.tacit)
|
|
||||||
|
|
||||||
# Add a record
|
# Add a record
|
||||||
if args.add is not None:
|
if args.add is not None:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user