2013-03-20 21:58:10 +01:00
|
|
|
(in-package :org.ryuslash.clark)
|
|
|
|
|
|
|
|
(export '(clark))
|
|
|
|
|
|
|
|
(defvar *db* nil
|
|
|
|
"The database connection.")
|
|
|
|
|
2013-03-21 23:41:13 +01:00
|
|
|
(defconstant *helps*
|
|
|
|
'(("add" "Add a bookmark to the database"
|
|
|
|
"add <url> <name> <description> [<tags> ...]")
|
|
|
|
("help" "Display this help and exit"
|
|
|
|
"help")
|
|
|
|
("version" "Output version information and exit"
|
|
|
|
"version"))
|
|
|
|
"Help texts for commands.")
|
|
|
|
|
2013-03-21 00:54:06 +01:00
|
|
|
(defconstant *version* "0.1.0"
|
|
|
|
"Clark's version.")
|
|
|
|
|
2013-03-21 22:30:55 +01:00
|
|
|
(defun add-command (args)
|
|
|
|
"Add a new bookmark to the database."
|
|
|
|
(with-transaction *db*
|
|
|
|
(destructuring-bind (url name description &rest tags) args
|
|
|
|
(insert-bookmark url name description)
|
|
|
|
(add-tags tags))))
|
|
|
|
|
|
|
|
(defun add-tags (tags)
|
|
|
|
"Add tags to the bookmark_tag table and possibly to tag."
|
|
|
|
(let ((bookmark-id (last-insert-rowid *db*)))
|
|
|
|
(map nil (lambda (tag)
|
|
|
|
(let ((tag-id (handler-case (insert-tag tag)
|
|
|
|
(sqlite-error () (get-tag-id tag)))))
|
|
|
|
(insert-bookmark-tag bookmark-id tag-id))) tags)))
|
|
|
|
|
2013-03-21 00:54:06 +01:00
|
|
|
(defun check-db (name)
|
2013-03-20 21:58:10 +01:00
|
|
|
"Connect to the database, possibly creating it."
|
|
|
|
(let ((db-exists (probe-file name)))
|
2013-03-21 00:54:06 +01:00
|
|
|
(setf *db* (connect name))
|
2013-03-20 21:58:10 +01:00
|
|
|
(unless db-exists
|
2013-03-21 00:54:06 +01:00
|
|
|
(execute-non-query *db* "CREATE TABLE bookmark (url VARCHAR(255) UNIQUE, date INTEGER, name VARCHAR(255), description TEXT)")
|
2013-03-21 22:30:55 +01:00
|
|
|
(execute-non-query *db* "CREATE TABLE tag (name VARCHAR(255) UNIQUE)")
|
2013-03-21 00:54:06 +01:00
|
|
|
(execute-non-query *db* "CREATE TABLE bookmark_tag (bookmark_id INTEGER REFERENCES bookmark(rowid), tag_id INTEGER REFERENCES tag(rowid), PRIMARY KEY (bookmark_id, tag_id))"))))
|
|
|
|
|
|
|
|
(defun get-bookmarks ()
|
|
|
|
"Get a list of all bookmarks.
|
|
|
|
|
|
|
|
The result contains the url and the name of the bookmark."
|
|
|
|
(let ((statement
|
|
|
|
(prepare-statement *db* "select url, name from bookmark")))
|
|
|
|
(loop
|
|
|
|
while (step-statement statement)
|
|
|
|
collect (list (statement-column-value statement 0)
|
|
|
|
(statement-column-value statement 1))
|
|
|
|
finally (finalize-statement statement))))
|
2013-03-20 21:58:10 +01:00
|
|
|
|
2013-03-21 22:30:55 +01:00
|
|
|
(defun get-tag-id (name)
|
|
|
|
"Get the rowid of tag NAME."
|
|
|
|
(execute-single *db* "SELECT rowid FROM tag WHERE name = ?" name))
|
|
|
|
|
2013-03-21 00:54:06 +01:00
|
|
|
(defun help-command (args)
|
|
|
|
"Show a help message."
|
|
|
|
(format t (concatenate
|
|
|
|
'string
|
2013-03-21 22:35:54 +01:00
|
|
|
"Usage: clark [<command> [<options> ...]]~%"
|
|
|
|
" clark add <url> <name> <description> [<tags> ...]~%"
|
2013-03-21 00:54:06 +01:00
|
|
|
"~%"
|
|
|
|
"Possible commands:~%"
|
2013-03-21 23:41:13 +01:00
|
|
|
"~%"))
|
|
|
|
(map nil (lambda (hlp)
|
|
|
|
(destructuring-bind (name short long) hlp
|
|
|
|
(format t " ~7A ~A~%" name short))) *helps*))
|
2013-03-21 00:54:06 +01:00
|
|
|
|
2013-03-21 22:30:55 +01:00
|
|
|
(defun insert-bookmark (url name description)
|
|
|
|
"Insert URL, NAME and DESCRIPTION into the bookmark table."
|
|
|
|
(execute-non-query *db* "INSERT INTO bookmark VALUES (?, ?, ?, ?)"
|
2013-03-21 23:30:52 +01:00
|
|
|
url (get-universal-time) name description))
|
2013-03-21 22:30:55 +01:00
|
|
|
|
|
|
|
(defun insert-bookmark-tag (bookmark-id tag-id)
|
|
|
|
"Insert BOOKMARK-ID and TAG-ID into the bookmark_tag table."
|
|
|
|
(execute-non-query *db* "INSERT INTO bookmark_tag VALUES (?, ?)"
|
|
|
|
bookmark-id tag-id))
|
|
|
|
|
|
|
|
(defun insert-tag (name)
|
|
|
|
"Insert tag NAME into the database and return its rowid."
|
|
|
|
(execute-non-query *db* "INSERT INTO tag VALUES (?)" name)
|
|
|
|
(last-insert-rowid *db*))
|
|
|
|
|
2013-03-21 21:29:43 +01:00
|
|
|
(defun make-command-name (base)
|
|
|
|
"Turn BASE into the name of a possible command."
|
|
|
|
(intern (concatenate 'string (string-upcase base) "-COMMAND")
|
|
|
|
:org.ryuslash.clark))
|
|
|
|
|
2013-03-21 21:39:56 +01:00
|
|
|
(defun parse-args (args)
|
|
|
|
"Parse command-line arguments ARGS.
|
|
|
|
|
|
|
|
The executable name should already have been removed."
|
|
|
|
(let ((cmd-name (make-command-name (car args))))
|
|
|
|
(if (fboundp cmd-name)
|
|
|
|
(funcall cmd-name (cdr args))
|
|
|
|
(progn
|
|
|
|
(format t "Unknown command: ~A~%" (car args))
|
|
|
|
(help-command nil)))))
|
|
|
|
|
2013-03-21 21:29:43 +01:00
|
|
|
(defun print-bookmark (bm)
|
|
|
|
"Print information about bookmark BM.
|
|
|
|
|
|
|
|
BM should be a list containing the url and name of the bookmark."
|
|
|
|
(destructuring-bind (url name) bm
|
|
|
|
(format t "~A~%~A~%~%" url name)))
|
|
|
|
|
2013-03-21 00:54:06 +01:00
|
|
|
(defun version-command (args)
|
|
|
|
"Display clark's version number."
|
|
|
|
(format t "clark version ~A~%" *version*))
|
|
|
|
|
|
|
|
(defun clark (args)
|
2013-03-21 21:29:43 +01:00
|
|
|
"Main function.
|
|
|
|
|
|
|
|
Connect to the database, parse command-line arguments, execute and
|
|
|
|
then disconnect."
|
2013-03-21 00:54:06 +01:00
|
|
|
(check-db "test2.db")
|
|
|
|
(if (> (length args) 1)
|
2013-03-21 21:39:56 +01:00
|
|
|
(parse-args (cdr args))
|
2013-03-21 00:54:06 +01:00
|
|
|
(map nil #'print-bookmark (get-bookmarks)))
|
2013-03-20 21:58:10 +01:00
|
|
|
(disconnect *db*))
|