#!/usr/bin/env python3 # # Bookmark management utility # # Copyright (C) 2015-2016 Arun Prakash Jana # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Buku. If not, see . import sys import os import sqlite3 import re import argparse import webbrowser import html.parser as HTMLParser import urllib3 from urllib.parse import urlparse, unquote import signal import json import logging import inspect import atexit try: import readline readline except ImportError: pass __version__ = '2.6' __author__ = 'Arun Prakash Jana ' __license__ = 'GPLv3' # Globals update = False # Update a bookmark in DB tags_in = None # Input tags specified at cmdline title_in = None # Input title specified at cmdline description = None # Description of the bookmark tagsearch = False # Search bookmarks by tag title_data = None # Title fetched from a webpage interrupted = False # Received SIGINT DELIM = ',' # Delimiter used to store tags in DB SKIP_MIMES = {'.pdf', '.txt'} http_handler = None # urllib3 PoolManager handler # Crypto globals BLOCKSIZE = 65536 SALT_SIZE = 32 CHUNKSIZE = 0x80000 # Read/write 512 KB chunks # Set up logging logging.basicConfig(format='[%(levelname)s] %(message)s') logger = logging.getLogger() class BMHTMLParser(HTMLParser.HTMLParser): '''Class to parse and fetch the title from a HTML page, if available ''' def __init__(self): HTMLParser.HTMLParser.__init__(self) self.inTitle = False self.data = '' self.lasttag = None def handle_starttag(self, tag, attrs): self.inTitle = False if tag == 'title': self.inTitle = True self.lasttag = tag def handle_endtag(self, tag): global title_data if tag == 'title': self.inTitle = False if self.data != '': title_data = self.data self.reset() # We have received title data, exit parsing def handle_data(self, data): if self.lasttag == 'title' and self.inTitle: self.data = '%s%s' % (self.data, data) def error(self, message): pass class BukuCrypt: '''Class to handle encryption and decryption of the database file. Functionally a separate entity. Involves late imports in the static functions but it saves ~100ms each time. Given that encrypt/decrypt are not done automatically and any one should be called at a time, this doesn't seem to be an outrageous approach. ''' @staticmethod def get_filehash(filepath): '''Get the SHA256 hash of a file :param filepath: path to the file :return: hash digest of the file ''' from hashlib import sha256 with open(filepath, 'rb') as fp: hasher = sha256() buf = fp.read(BLOCKSIZE) while len(buf) > 0: hasher.update(buf) buf = fp.read(BLOCKSIZE) return hasher.digest() @staticmethod def encrypt_file(iterations): '''Encrypt the bookmarks database file :param iterations: number of iterations for key generation ''' try: from getpass import getpass import struct from hashlib import sha256 from cryptography.hazmat.backends import default_backend from cryptography.hazmat.primitives.ciphers import (Cipher, modes, algorithms) except ImportError: logger.error('cryptography lib(s) missing') sys.exit(1) if iterations < 1: logger.error('Iterations must be >= 1') sys.exit(1) dbpath = os.path.join(BukuDb.get_dbdir_path(), 'bookmarks.db') encpath = '%s.enc' % dbpath if not os.path.exists(dbpath): logger.error('%s missing. Already encrypted?', dbpath) sys.exit(1) # If both encrypted file and flat file exist, error out if os.path.exists(dbpath) and os.path.exists(encpath): logger.error('Both encrypted and flat DB files exist!') sys.exit(1) password = '' password = getpass() passconfirm = getpass() if password == '': logger.error('Empty password') sys.exit(1) if password != passconfirm: logger.error('Passwords do not match') sys.exit(1) # Get SHA256 hash of DB file dbhash = BukuCrypt.get_filehash(dbpath) # Generate random 256-bit salt and key salt = os.urandom(SALT_SIZE) key = ('%s%s' % (password, salt.decode('utf-8', 'replace'))).encode('utf-8') for _ in range(iterations): key = sha256(key).digest() iv = os.urandom(16) encryptor = Cipher( algorithms.AES(key), modes.CBC(iv), backend=default_backend() ).encryptor() filesize = os.path.getsize(dbpath) with open(dbpath, 'rb') as infp: with open(encpath, 'wb') as outfp: outfp.write(struct.pack(' index: self.cur.execute(query1, (row[0],)) results = self.cur.fetchall() for row in results: self.cur.execute(query2, (row[0],)) self.cur.execute(query3, (index, row[1], row[2], row[3], row[4],)) if not delay_commit: self.conn.commit() print('Index %d moved to %d' % (row[0], index)) def delete_bm(self, index, low=0, high=0, is_range=False, delay_commit=False): '''Delete a single record or remove the table if index is None :param index: DB index of deleted entry :param low: lower index of range :param low: higher index of range :param is_range: a range is passed using low and high arguments :param delay_commit: do not commit to DB, caller's responsibility :return: True on success, False on failure ''' if is_range: # Delete a range of indices # If range starts from 0, delete all records if low == 0: return self.cleardb() try: query = 'DELETE from bookmarks where id BETWEEN ? AND ?' self.cur.execute(query, (low, high)) if not delay_commit: self.conn.commit() print('Bookmarks from index %s to %s deleted' % (low, high)) # Compact DB by ascending order of index to ensure # the existing higher indices move only once # Delayed commit is forced for index in range(low, high + 1): self.compactdb(index, delay_commit=True) if not delay_commit: self.conn.commit() except IndexError: logger.error('Index out of bound') return False elif index == 0: # Remove the table return self.cleardb() else: # Remove a single entry try: query = 'DELETE FROM bookmarks WHERE id = ?' self.cur.execute(query, (index,)) if not delay_commit: self.conn.commit() if self.cur.rowcount == 1: print('Removed index %d' % index) self.compactdb(index, delay_commit) else: logger.error('No matching index') return False except IndexError: logger.error('Index out of bound') return False return True def delete_resultset(self, results): '''Delete search results in descending order of DB index. Indices are expected to be unique and in ascending order. This API forces a delayed commit. :param results: set of results to delete :return: True on success, False on failure ''' resp = input('Delete the search results? (y/n): ') if resp != 'y': return False # delete records in reverse order pos = len(results) - 1 while pos >= 0: idx = results[pos][0] self.delete_bm(idx, delay_commit=True) # Commit at every 200th removal if pos % 200 == 0: self.conn.commit() pos -= 1 return True def cleardb(self): '''Drops the bookmark table if it exists :return: True on success, False on failure ''' resp = input('Remove ALL bookmarks? (y/n): ') if resp != 'y': print('No bookmarks deleted') return False self.cur.execute('DROP TABLE if exists bookmarks') self.conn.commit() print('All bookmarks deleted') return True def print_bm(self, index, empty=False, immutable=False): '''Print bookmark details at index or all bookmarks if index is 0 Print only bookmarks with blank title or tag if empty is True Note: URL is printed on top because title may be blank :param index: index to print (0 for all) :param empty: flag to show only bookmarks with no title or tags :param immutable: flag to show only bookmarks with immutable titles ''' if index == 0: # Show all entries if empty: qry = "SELECT * FROM bookmarks WHERE metadata = '' OR tags = ?" self.cur.execute(qry, (DELIM,)) resultset = self.cur.fetchall() print('\x1b[1m%s records found\x1b[21m\n' % len(resultset)) elif immutable: qry = "SELECT * FROM bookmarks WHERE flags & 1 == 1" self.cur.execute(qry) resultset = self.cur.fetchall() print('\x1b[1m%s records found\x1b[21m\n' % len(resultset)) else: self.cur.execute('SELECT * FROM bookmarks') resultset = self.cur.fetchall() if not self.json: if self.field_filter == 0: for row in resultset: print_record(row) elif self.field_filter == 1: for row in resultset: print('%s\t%s' % (row[0], row[1])) elif self.field_filter == 2: for row in resultset: print('%s\t%s\t%s' % (row[0], row[1], row[3][1:-1])) elif self.field_filter == 3: for row in resultset: print('%s\t%s' % (row[0], row[2])) else: print(format_json(resultset, field_filter=self.field_filter)) else: # Show record at index try: query = 'SELECT * FROM bookmarks WHERE id = ?' self.cur.execute(query, (index,)) results = self.cur.fetchall() if len(results) == 0: logger.error('No matching index') return except IndexError: logger.error('Index out of bound') return if not self.json: for row in results: if self.field_filter == 0: print_record(row) elif self.field_filter == 1: print('%s\t%s' % (row[0], row[1])) elif self.field_filter == 2: print('%s\t%s\t%s' % (row[0], row[1], row[3][1:-1])) elif self.field_filter == 3: print('%s\t%s' % (row[0], row[2])) else: print(format_json(results, True, self.field_filter)) def list_tags(self): '''Print all unique tags ordered alphabetically''' count = 1 tags = [] unique_tags = [] query = 'SELECT DISTINCT tags FROM bookmarks ORDER BY tags' for row in self.cur.execute(query): tagset = row[0].strip(DELIM).split(DELIM) for tag in tagset: if tag not in tags: tags += (tag,) if tags[0] == '': unique_tags = sorted(tags[1:], key=str.lower) else: unique_tags = sorted(tags, key=str.lower) for tag in unique_tags: print('%6d. %s' % (count, tag)) count += 1 def replace_tag(self, orig, new=None): '''Replace orig tags with new tags in DB for all records. Remove orig tag if new tag is empty. :param orig: original tags :param new: replacement tags :return: True on success, False on failure ''' update = False delete = False newtags = DELIM orig = '%s%s%s' % (DELIM, orig, DELIM) if new is None: delete = True else: newtags = parse_tags(new) if newtags == DELIM: delete = True if orig == newtags: print('Tags are same.') return False query = 'SELECT id, tags FROM bookmarks WHERE tags LIKE ?' self.cur.execute(query, ('%' + orig + '%',)) results = self.cur.fetchall() query = 'UPDATE bookmarks SET tags = ? WHERE id = ?' for row in results: if not delete: # Check if tag newtags is already added if row[1].find(newtags) >= 0: newtags = DELIM tags = row[1].replace(orig, newtags) tags = parse_tags([tags]) self.cur.execute(query, (tags, row[0],)) print('Index %d updated' % row[0]) update = True if update: self.conn.commit() return update def browse_by_index(self, index): '''Open URL at index in browser :param index: DB index :return: True on success, False on failure ''' if index == 0: query = 'SELECT id from bookmarks ORDER BY RANDOM() LIMIT 1' self.cur.execute(query) result = self.cur.fetchone() # Return if no entries in DB if result is None: print('No bookmarks added yet ...') return False index = result[0] logger.debug('Opening random index ' + str(index)) query = 'SELECT URL FROM bookmarks WHERE id = ?' try: for row in self.cur.execute(query, (index,)): url = unquote(row[0]) open_in_browser(url) return True logger.error('No matching index') except IndexError: logger.error('Index out of bound') return False def exportdb(self, filepath, markdown=False, taglist=None): '''Export bookmarks to a Firefox bookmarks formatted html file. :param filepath: path to file to export to :param markdown: use markdown syntax :param taglist: list of specific tags to export :return: True on success, False on failure ''' import time count = 0 timestamp = int(time.time()) arguments = [] query = 'SELECT * FROM bookmarks' is_tag_valid = False if taglist is not None: tagstr = parse_tags(taglist) if len(tagstr) == 0 or tagstr == DELIM: logger.error('Invalid tag') return False if len(tagstr) > 0: tags = tagstr.split(DELIM) query = '%s WHERE' % query for tag in tags: if tag != '': is_tag_valid = True query += " tags LIKE '%' || ? || '%' OR" tag = '%s%s%s' % (DELIM, tag, DELIM) arguments += (tag,) if is_tag_valid: query = query[:-3] else: query = query[:-6] logger.debug('(%s), %s' % (query, arguments)) self.cur.execute(query, arguments) resultset = self.cur.fetchall() if len(resultset) == 0: print('No bookmarks exported') return False if os.path.exists(filepath): resp = input('%s exists. Overwrite? (y/n): ' % filepath) if resp != 'y': return False try: outfp = open(filepath, mode='w', encoding='utf-8') except Exception as e: logger.error(e) return False if not markdown: outfp.write(''' Bookmarks

Bookmarks

Buku bookmarks

''' % (timestamp, timestamp)) for row in resultset: out = '%s

\n

') else: outfp.write('List of buku bookmarks:\n\n') for row in resultset: if row[2] == '': out = '- [Untitled](%s)\n' % (row[1]) else: out = '- [%s](%s)\n' % (row[2], row[1]) outfp.write(out) count += 1 outfp.close() print('%s exported' % count) return True def importdb(self, filepath, markdown=False): '''Import bookmarks from a html file. Supports Firefox, Google Chrome and IE imports :param filepath: path to file to import :param markdown: use markdown syntax :return: True on success, False on failure ''' if not markdown: try: import bs4 with open(filepath, mode='r', encoding='utf-8') as infp: soup = bs4.BeautifulSoup(infp, 'html.parser') except ImportError: logger.error('Beautiful Soup not found') return False except Exception as e: logger.error(e) return False html_tags = soup.findAll('a') for tag in html_tags: # Extract comment from

tag desc = None comment_tag = tag.findNextSibling('dd') if comment_tag: desc = comment_tag.text[0:comment_tag.text.find('\n')] self.add_bm(tag['href'], tag.string, ('%s%s%s' % (DELIM, tag['tags'], DELIM)) if tag.has_attr('tags') else None, desc, True) self.conn.commit() infp.close() else: with open(filepath, mode='r', encoding='utf-8') as infp: for line in infp: # Supported markdown format: [title](url) # Find position of title end, url start delimiter combo index = line.find('](') if index != -1: # Find title start delimiter title_start_delim = line[:index].find('[') # Reverse find the url end delimiter url_end_delim = line[index + 2:].rfind(')') if title_start_delim != -1 and url_end_delim > 0: # Parse title title = line[title_start_delim + 1:index] # Parse url url = line[index + 2:index + 2 + url_end_delim] self.add_bm(url, title, None, None, True) self.conn.commit() infp.close() return True def mergedb(self, path): '''Merge bookmarks from another Buku database file :param path: path to DB file to merge :return: True on success, False on failure ''' try: # Connect to input DB if sys.version_info >= (3, 4, 4): # Python 3.4.4 and above indb_conn = sqlite3.connect('file:%s?mode=ro' % path, uri=True) else: indb_conn = sqlite3.connect(path) indb_cur = indb_conn.cursor() indb_cur.execute('SELECT * FROM bookmarks') except Exception as e: logger.error(e) return False resultset = indb_cur.fetchall() for row in resultset: self.add_bm(row[1], row[2], row[3], row[4], True) if len(resultset): self.conn.commit() try: indb_cur.close() indb_conn.close() except Exception: pass return True def close_quit(self, exitval=0): '''Close a DB connection and exit :param exitval: program exit value ''' if self.conn is not None: try: self.cur.close() self.conn.close() except Exception: # ignore errors here, we're closing down pass sys.exit(exitval) # Generic functions def is_bad_url(url): '''Check if URL is malformed This API is not bulletproof but works in most cases. :param url: URL to scan :return: True or False ''' # Get the netloc token netloc = urlparse(url).netloc if not netloc: # Try of prepend '//' and get netloc netloc = urlparse('//' + url).netloc if not netloc: return True logger.debug('netloc: %s' % netloc) # netloc cannot start or end with a '.' if netloc.startswith('.') or netloc.endswith('.'): return True # netloc should have at least one '.' index = netloc.rfind('.') if index < 0: return True return False def is_ignored_mime(url): '''Check if URL links to ignored mime Only a 'HEAD' request is made for these URLs :param url: URL to scan :return: True or False ''' for mime in SKIP_MIMES: if url.lower().endswith(mime): return True return False def get_page_title(resp): '''Invoke HTML parser and extract title from HTTP response The page title is set in a global variable :param resp: HTTP(S) GET response ''' parser = BMHTMLParser() try: parser.feed(resp.data.decode(errors='replace')) except Exception as e: # Suppress Exception due to intentional self.reset() in HTMLParser if logger.isEnabledFor(logging.DEBUG) \ and str(e) != 'we should not get here!': _, _, linenumber, func, _, _ = inspect.stack()[0] logger.error('%s(), ln %d: %s', func, linenumber, e) def network_handler(url): '''Handle server connection and redirections :param url: URL to fetch :return: {title, recognized mime, bad url} tuple ''' global title_data, http_handler title_data = None resp = None method = 'GET' if is_bad_url(url): return ('', 0, 1) if is_ignored_mime(url): method = 'HEAD' if not http_handler: http_handler = urllib3.PoolManager() try: while True: resp = http_handler.request( method, url, timeout=40, headers={'Accept-Encoding': 'gzip,deflate', 'DNT': '1'} ) if resp.status == 200: get_page_title(resp) break elif resp.status == 403: # HTTP response Forbidden # Handle URLs in the form of https://www.domain.com/ # which fail when trying to fetch resource '/' # retry without trailing '/' logger.debug('Received status 403: retrying...') # Remove trailing / if url[-1] == '/': url = url[:-1] else: break else: logger.error('[%s] %s', resp.status, resp.reason) break resp.release_conn() except Exception as e: _, _, linenumber, func, _, _ = inspect.stack()[0] logger.error('%s(), ln %d: %s', func, linenumber, e) finally: if resp: resp.release_conn() if method == 'HEAD': return ('', 1, 0) if title_data is None: return ('', 0, 0) return (title_data.strip().replace('\n', ''), 0, 0) def parse_tags(keywords=None): '''Format and get tag string from tokens :param keywords: list of tags :return: comma-delimited string of tags :return: just delimiter, if no keywords :return: None, if keyword is None ''' if keywords is None: return None tags = DELIM orig_tags = [] unique_tags = [] # Cleanse and get the tags tagstr = ' '.join(keywords) marker = tagstr.find(DELIM) while marker >= 0: token = tagstr[0:marker] tagstr = tagstr[marker + 1:] marker = tagstr.find(DELIM) token = token.strip() if token == '': continue tags = '%s%s%s' % (tags, token, DELIM) tagstr = tagstr.strip() if tagstr != '': tags = '%s%s%s' % (tags, tagstr, DELIM) logger.debug('keywords: %s', keywords) logger.debug('parsed tags: [%s]', tags) if tags == DELIM: return tags orig_tags += tags.strip(DELIM).split(DELIM) for tag in orig_tags: if tag not in unique_tags: unique_tags += (tag, ) # Select unique tags # Sort the tags sorted_tags = sorted(unique_tags, key=str.lower) # Wrap with delimiter return '%s%s%s' % (DELIM, DELIM.join(sorted_tags), DELIM) def prompt(results, noninteractive=False): '''Show each matching result from a search and prompt :param noninteractive: do not seek user input ''' count = 0 for row in results: count += 1 print_record(row, count) if noninteractive: return while True: try: nav = input('Results, ranges (x-y,(a)ll) to open: ') if not nav: nav = input('Results, ranges (x-y,(a)ll) to open: ') if not nav: # Quit on double enter break except EOFError: return # open all results and re-prompt if 'a' is pressed if nav == 'a': for index in range(0, count): try: open_in_browser(unquote(results[index][1])) except Exception as e: _, _, linenumber, func, _, _ = inspect.stack()[0] logger.error('%s(), ln %d: %s', func, linenumber, e) continue # iterate over white-space separated indices for nav in (' '.join(nav.split())).split(): if is_int(nav): index = int(nav) - 1 if index < 0 or index >= count: logger.error('Index out of bound') continue try: open_in_browser(unquote(results[index][1])) except Exception as e: _, _, linenumber, func, _, _ = inspect.stack()[0] logger.error('%s(), ln %d: %s', func, linenumber, e) elif '-' in nav and is_int(nav.split('-')[0]) \ and is_int(nav.split('-')[1]): lower = int(nav.split('-')[0]) upper = int(nav.split('-')[1]) if lower > upper: lower, upper = upper, lower for index in range(lower-1, upper): try: open_in_browser(unquote(results[index][1])) except Exception as e: _, _, linenumber, func, _, _ = inspect.stack()[0] logger.error('%s(), ln %d: %s', func, linenumber, e) else: break def print_record(row, idx=0): '''Print a single DB record Handles both search result and individual record :param idx: search result index. If 0, print with DB index ''' # Start with index and URL if idx != 0: pr = '\x1b[1m\x1b[93m%d. \x1b[0m\x1b[92m%s\x1b[0m \ \x1b[1m[%s]\x1b[0m\n' % (idx, row[1], row[0]) else: pr = '\x1b[1m\x1b[93m%d. \x1b[0m\x1b[92m%s\x1b[0m' % (row[0], row[1]) # Indicate if record is immutable if row[5] & 1: pr = '%s \x1b[1m(L)\x1b[0m\n' % (pr) else: pr = '%s\n' % (pr) # Append title if row[2] != '': pr = '%s \x1b[91m>\x1b[0m %s\n' % (pr, row[2]) # Append description if row[4] != '': pr = '%s \x1b[91m+\x1b[0m %s\n' % (pr, row[4]) # Append tags IF not default (delimiter) if row[3] != DELIM: pr = '%s \x1b[91m#\x1b[0m %s\n' % (pr, row[3][1:-1]) print(pr) def format_json(resultset, single_record=False, field_filter=0): '''Return results in Json format :param single_record: indicates only one record :param field_filter: determines fields to show :return: record(s) in Json format ''' if single_record: marks = {} for row in resultset: if field_filter == 1: marks['uri'] = row[1] elif field_filter == 2: marks['uri'] = row[1] marks['tags'] = row[3][1:-1] elif field_filter == 3: marks['title'] = row[2] else: marks['uri'] = row[1] marks['title'] = row[2] marks['description'] = row[4] marks['tags'] = row[3][1:-1] else: marks = [] for row in resultset: if field_filter == 1: record = {'uri': row[1]} elif field_filter == 2: record = {'uri': row[1], 'tags': row[3][1:-1]} elif field_filter == 3: record = {'title': row[2]} else: record = {'uri': row[1], 'title': row[2], 'description': row[4], 'tags': row[3][1:-1]} marks.append(record) return json.dumps(marks, sort_keys=True, indent=4) def is_int(string): '''Check if a string is a digit :param string: input string :return: True on success, False on exception ''' try: int(string) return True except Exception: return False def open_in_browser(url): '''Duplicate stdin, stdout (to suppress showing errors on the terminal) and open URL in default browser :param url: URL to open ''' url = url.replace('%22', '\"') _stderr = os.dup(2) os.close(2) _stdout = os.dup(1) os.close(1) fd = os.open(os.devnull, os.O_RDWR) os.dup2(fd, 2) os.dup2(fd, 1) try: webbrowser.open(url) except Exception as e: _, _, linenumber, func, _, _ = inspect.stack()[0] logger.error('%s(), ln %d: %s', func, linenumber, e) finally: os.close(fd) os.dup2(_stderr, 2) os.dup2(_stdout, 1) def check_upstream_release(): '''Check and report the latest upstream release version''' import requests r = requests.get('https://api.github.com/repos/jarun/buku/tags?per_page=1') if r.status_code != 200: logger.error('[%s] %s', r.status_code, r.reason) else: latest = r.json()[0]['name'] if latest == 'v' + __version__: print('This is the latest release') else: print('Latest upstream release is %s' % latest) def sigint_handler(signum, frame): '''Custom SIGINT handler''' global interrupted interrupted = True print('\nInterrupted.', file=sys.stderr) sys.exit(1) signal.signal(signal.SIGINT, sigint_handler) def regexp(expr, item): '''Perform a regular expression search''' return re.search(expr, item, re.IGNORECASE) is not None # Custom Action classes for argparse class CustomUpdateAction(argparse.Action): '''Class to capture if optional param 'update' is actually used, even if sans arguments ''' def __call__(self, parser, args, values, option_string=None): global update update = True # NOTE: the following converts a None argument to an empty array [] setattr(args, self.dest, values) class CustomTagAction(argparse.Action): '''Class to capture if optional param 'tag' is actually used, even if sans arguments ''' def __call__(self, parser, args, values, option_string=None): global tags_in tags_in = [DELIM, ] setattr(args, self.dest, values) class CustomTitleAction(argparse.Action): '''Class to capture if optional param 'title' is actually used, even if sans arguments ''' def __call__(self, parser, args, values, option_string=None): global title_in title_in = '' setattr(args, self.dest, values) class CustomDescAction(argparse.Action): '''Class to capture if optional param 'comment' is actually used, even if sans arguments ''' def __call__(self, parser, args, values, option_string=None): global description description = '' setattr(args, self.dest, values) class CustomTagSearchAction(argparse.Action): '''Class to capture if optional param 'stag' is actually used, even if sans arguments ''' def __call__(self, parser, args, values, option_string=None): global tagsearch tagsearch = True setattr(args, self.dest, values) class ExtendedArgumentParser(argparse.ArgumentParser): '''Extend classic argument parser''' # Print additional help and info @staticmethod def print_extended_help(file=None): file.write(''' prompt keys: 1-N open the Nth search result in web browser ranges, space-separated result indices work double Enter exit buku symbols: > title + comment # tags Version %s Copyright (C) 2015-2016 Arun Prakash Jana License: GPLv3 Webpage: https://github.com/jarun/Buku ''' % __version__) # Help def print_help(self, file=None): super(ExtendedArgumentParser, self).print_help(file) self.print_extended_help(file) '''main starts here''' # Handle piped input def piped_input(argv, pipeargs=None): if not sys.stdin.isatty(): pipeargs.extend(argv) for s in sys.stdin.readlines(): pipeargs.extend(s.split()) def main(): global tags_in, title_in, description pipeargs = [] atexit.register(logging.shutdown) try: piped_input(sys.argv, pipeargs) except KeyboardInterrupt: pass # If piped input, set argument vector if len(pipeargs) > 0: sys.argv = pipeargs # Setup custom argument parser argparser = ExtendedArgumentParser( description='A powerful command-line bookmark manager. Your mini web!', formatter_class=argparse.RawTextHelpFormatter, usage='''buku [OPTIONS] [KEYWORD [KEYWORD ...]]''', add_help=False ) HIDE = argparse.SUPPRESS # --------------------- # GENERAL OPTIONS GROUP # --------------------- general_grp = argparser.add_argument_group( title='general options', description='''-a, --add URL [tags ...] bookmark URL with comma-separated tags -u, --update [...] update fields of bookmark at DB indices accepts indices and ranges refresh all titles, if no arguments refresh titles of bookmarks at indices, if no edit options are specified -d, --delete [...] delete bookmarks. Valid inputs: either a hyphenated single range (100-200), OR space-separated indices (100 15 200) delete search results with search options delete all bookmarks, if no arguments -h, --help show this information and exit''') addarg = general_grp.add_argument addarg('-a', '--add', nargs='+', help=HIDE) addarg('-u', '--update', nargs='*', action=CustomUpdateAction, help=HIDE) addarg('-d', '--delete', nargs='*', help=HIDE) addarg('-h', '--help', action='store_true', help=HIDE) # ------------------ # EDIT OPTIONS GROUP # ------------------ edit_grp = argparser.add_argument_group( title='edit options', description='''--url keyword specify url, works with -u only --tag [+|-] [...] set comma-separated tags, works with -a, -u clear tags, if no arguments append specified tags, if preceded by '+' remove specified tags, if preceded by '-' -t, --title [...] manually set title, works with -a, -u if no arguments: -a: do not set title, -u: clear title -c, --comment [...] description of the bookmark, works with -a, -u; clears comment, if no arguments --immutable N disable title fetch from web during update works with -a, -u N=0: mutable (default), N=1: immutable''') addarg = edit_grp.add_argument addarg('--url', nargs=1, help=HIDE) addarg('--tag', nargs='*', action=CustomTagAction, help=HIDE) addarg('-t', '--title', nargs='*', action=CustomTitleAction, help=HIDE) addarg('-c', '--comment', nargs='*', action=CustomDescAction, help=HIDE) addarg('--immutable', type=int, default=-1, choices={0, 1}, help=HIDE) # -------------------- # SEARCH OPTIONS GROUP # -------------------- search_grp = argparser.add_argument_group( title='search options', description='''-s, --sany keyword [...] search bookmarks for ANY matching keyword -S, --sall keyword [...] search bookmarks with ALL keywords special keywords - "blank": list entries with empty title/tag "immutable": list entries with locked title --deep match substrings ('pen' matches 'opened') --sreg expr run a regex search --stag [...] search bookmarks by tag list tags alphabetically, if no arguments''') addarg = search_grp.add_argument addarg('-s', '--sany', nargs='+', help=HIDE) addarg('-S', '--sall', nargs='+', help=HIDE) addarg('--sreg', nargs=1, help=HIDE) addarg('--deep', action='store_true', help=HIDE) addarg('--stag', nargs='*', action=CustomTagSearchAction, help=HIDE) # ------------------------ # ENCRYPTION OPTIONS GROUP # ------------------------ crypto_grp = argparser.add_argument_group( title='encryption options', description='''-l, --lock [N] encrypt DB file with N (> 0, default 8) hash iterations to generate key -k, --unlock [N] decrypt DB file with N (> 0, default 8) hash iterations to generate key''') addarg = crypto_grp.add_argument addarg('-k', '--unlock', nargs='?', type=int, const=8, help=HIDE) addarg('-l', '--lock', nargs='?', type=int, const=8, help=HIDE) # ---------------- # POWER TOYS GROUP # ---------------- power_grp = argparser.add_argument_group( title='power toys', description='''-e, --export file export bookmarks to Firefox format html use --tag to export only specific tags -i, --import file import bookmarks from html file; Firefox, Google Chrome and IE formats supported --markdown use markdown with -e and -i supported format: [title](url), 1 per line -m, --merge file merge bookmarks from another buku database -p, --print [...] show details of bookmark by DB index accepts indices and ranges show all bookmarks, if no arguments -f, --format N modify -p, search output. N=1: show only URL, N=2: show URL and tag, N=3: show only title -r, --replace oldtag [newtag ...] replace oldtag with newtag everywhere delete oldtag, if no newtag -j, --json Json formatted output for -p and search --noprompt do not show the prompt, run and exit -o, --open [N] open bookmark at DB index N in web browser open a random index if N is omitted --tacit reduce verbosity --upstream check latest upstream version available -z, --debug show debug information and additional logs''') addarg = power_grp.add_argument addarg('-e', '--export', nargs=1, help=HIDE) addarg('-i', '--import', nargs=1, dest='importfile', help=HIDE) addarg('--markdown', action='store_true', help=HIDE) addarg('-m', '--merge', nargs=1, help=HIDE) addarg('-p', '--print', nargs='*', help=HIDE) addarg('-f', '--format', type=int, default=0, choices={1, 2, 3}, help=HIDE) addarg('-r', '--replace', nargs='+', help=HIDE) addarg('-j', '--json', action='store_true', help=HIDE) addarg('--noprompt', action='store_true', help=HIDE) addarg('-o', '--open', nargs='?', type=int, const=0, help=HIDE) addarg('--tacit', action='store_true', help=HIDE) addarg('--upstream', action='store_true', help=HIDE) addarg('-z', '--debug', action='store_true', help=HIDE) # Show help and exit if no arguments if len(sys.argv) < 2: argparser.print_help(sys.stdout) sys.exit(1) # Parse the arguments args = argparser.parse_args() # Show help and exit if help requested if args.help: argparser.print_help(sys.stdout) sys.exit(0) # Assign the values to globals if tags_in is not None and len(args.tag) > 0: tags_in = args.tag if title_in is not None and len(args.title) > 0: title_in = ' '.join(args.title) if description is not None and len(args.comment) > 0: description = ' '.join(args.comment) if args.debug: logger.setLevel(logging.DEBUG) logger.debug('Version %s', __version__) else: logging.disable(logging.WARNING) # Handle encrypt/decrypt options at top priority if args.lock is not None: BukuCrypt.encrypt_file(args.lock) if args.unlock is not None: BukuCrypt.decrypt_file(args.unlock) # Initialize the database and get handles, set verbose by default bdb = BukuDb(args.json, args.format, args.immutable, not args.tacit) # Add a record if args.add is not None: # Parse tags into a comma-separated string tags = DELIM keywords = args.add if tags_in is not None: if tags_in[0] == '+' and len(tags_in) == 1: pass elif tags_in[0] == '+': tags_in = tags_in[1:] # In case of add, args.add may have URL followed by tags # Add delimiter as url+tags may not end with one keywords = args.add + [DELIM] + tags_in else: keywords = args.add + [DELIM] + tags_in if len(keywords) > 1: tags = parse_tags(keywords[1:]) bdb.add_bm(args.add[0], title_in, tags, description) # Update record if update: if args.url is not None: url_in = args.url[0] else: url_in = '' append = False delete = False if tags_in is not None: if (tags_in[0] == '+' or tags_in[0] == '-') \ and len(tags_in) == 1: logger.error('Please specify a tag') bdb.close_quit(1) elif tags_in[0] == '+': tags_in = tags_in[1:] append = True elif tags_in[0] == '-': tags_in = tags_in[1:] delete = True # Parse tags into a comma-separated string tags = parse_tags(tags_in) if len(args.update) == 0: bdb.update_bm(0, url_in, title_in, tags, description, append, delete) else: for idx in args.update: if is_int(idx): bdb.update_bm(int(idx), url_in, title_in, tags, description, append, delete) elif '-' in idx and is_int(idx.split('-')[0]) \ and is_int(idx.split('-')[1]): lower = int(idx.split('-')[0]) upper = int(idx.split('-')[1]) if lower > upper: lower, upper = upper, lower # Update only once if range starts from 0 (all) if lower == 0: bdb.update_bm(0, url_in, title_in, tags, description, append, delete) else: for _id in range(lower, upper + 1): bdb.update_bm(_id, url_in, title_in, tags, description, append, delete) if interrupted: break if interrupted: break # Search operations search_results = None search_opted = False # Search URLs, titles, tags for any keyword and delete if wanted if args.sany is not None: search_opted = True search_results = bdb.searchdb(args.sany, False, args.deep) # Search URLs, titles, tags with all keywords and delete if wanted elif args.sall is not None: search_opted = True if args.sall[0] == 'blank' and len(args.sall) == 1: bdb.print_bm(0, True) elif args.sall[0] == 'immutable' and len(args.sall) == 1: bdb.print_bm(0, False, True) else: search_results = bdb.searchdb(args.sall, True, args.deep) # Run a regular expression search elif args.sreg is not None: search_opted = True search_results = bdb.searchdb(args.sreg, False, regex=True) # Search bookmarks by tag and delete if wanted elif tagsearch: search_opted = True if len(args.stag) > 0: tag = '%s%s%s' % (DELIM, ' '.join(args.stag).strip(DELIM), DELIM) search_results = bdb.search_by_tag(tag) else: bdb.list_tags() if search_results: oneshot = args.noprompt # In case of search and delete, prompt should be non-interactive if args.delete is not None and len(args.delete) == 0: oneshot = True if not args.json: prompt(search_results, oneshot) else: print(format_json(search_results, field_filter=args.format)) # Delete search results if opted if args.delete is not None and len(args.delete) == 0: bdb.delete_resultset(search_results) # Delete record(s) if args.delete is not None: if len(args.delete) == 0: # Attempt delete-all only if search was not opted if not search_opted: bdb.cleardb() elif len(args.delete) == 1 and '-' in args.delete[0]: vals = str(args.delete[0]).split('-') if len(vals) == 2 and is_int(vals[0]) and is_int(vals[1]): if int(vals[0]) == int(vals[1]): bdb.delete_bm(int(vals[0])) elif int(vals[0]) < int(vals[1]): bdb.delete_bm(0, int(vals[0]), int(vals[1]), True) else: bdb.delete_bm(0, int(vals[1]), int(vals[0]), True) else: logger.error('Incorrect index or range') bdb.close_quit(1) else: ids = [] # Select the unique indices for idx in args.delete: if idx not in ids: ids += (idx,) try: # Index delete order - highest to lowest ids.sort(key=lambda x: int(x), reverse=True) for idx in ids: bdb.delete_bm(int(idx)) except ValueError: logger.error('Incorrect index or range') # Print records if args.print is not None: if len(args.print) == 0: bdb.print_bm(0) else: for idx in args.print: if is_int(idx): bdb.print_bm(int(idx)) elif '-' in idx and is_int(idx.split('-')[0]) \ and is_int(idx.split('-')[1]): lower = int(idx.split('-')[0]) upper = int(idx.split('-')[1]) if lower > upper: lower, upper = upper, lower for _id in range(lower, upper + 1): bdb.print_bm(_id) else: logger.error('Invalid index or range') bdb.close_quit(1) # Replace a tag in DB if args.replace is not None: if len(args.replace) == 1: bdb.replace_tag(args.replace[0]) else: bdb.replace_tag(args.replace[0], args.replace[1:]) # Export bookmarks if args.export is not None: if args.tag is None: bdb.exportdb(args.export[0], args.markdown) elif len(args.tag) == 0: logger.error('Missing tag') else: bdb.exportdb(args.export[0], args.markdown, args.tag) # Import bookmarks if args.importfile is not None: bdb.importdb(args.importfile[0], args.markdown) # Merge a database file and exit if args.merge is not None: bdb.mergedb(args.merge[0]) # Open URL in browser if args.open is not None: if args.open < 0: logger.error('Index must be >= 0') bdb.close_quit(1) bdb.browse_by_index(args.open) # Report upstream version if args.upstream: check_upstream_release() # Close DB connection and quit bdb.close_quit(0) if __name__ == '__main__': main()