Support custom db file. Optimizations.

This commit is contained in:
Arun Prakash Jana 2016-11-12 08:08:28 +05:30
parent b387fc95e1
commit 5e0bc15e28
No known key found for this signature in database
GPG Key ID: A75979F35C080412
2 changed files with 140 additions and 77 deletions

View File

@ -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:

147
buku.py
View File

@ -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)
try:
# Get SHA256 hash of DB file # Get SHA256 hash of DB file
dbhash = BukuCrypt.get_filehash(dbpath) 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,8 +283,9 @@ 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)
@ -277,46 +304,55 @@ class BukuCrypt:
# 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) sys.exit(1)
else: else:
os.remove(encpath) os.remove(encfile)
print('File decrypted') print('File decrypted')
except struct.error:
logger.error('Tainted file')
sys.exit(1)
except Exception as e:
logger.error(e)
sys.exit(1)
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:
dbpath = BukuDb.get_default_dbdir()
filename = 'bookmarks.db'
dbfile = os.path.join(dbpath, filename)
else:
dbfile = os.path.abspath(dbfile)
dbpath, filename = os.path.split(dbfile)
encfile = dbfile + '.enc'
try:
if not os.path.exists(dbpath): if not os.path.exists(dbpath):
os.makedirs(dbpath) os.makedirs(dbpath)
except Exception as e:
logger.error(e)
os.exit(1)
dbfile = os.path.join(dbpath, 'bookmarks.db') db_exists = os.path.exists(dbfile)
enc_exists = os.path.exists(encfile)
encpath = os.path.join(dbpath, 'bookmarks.db.enc') if db_exists and not enc_exists:
# Notify if DB file needs to be decrypted first pass
if os.path.exists(encpath) and not os.path.exists(dbfile): 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: