forked from charlesroelli/org-board
-
Notifications
You must be signed in to change notification settings - Fork 0
/
README
303 lines (227 loc) · 11.4 KB
/
README
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
org-board
=========
Last updated: Wed 30 May 2018 20:06:55 CEST
* Motivation
org-board is a bookmarking and web archival system for Emacs Org
mode, building on ideas from Pinboard <https://pinboard.in>. It
archives your bookmarks so that you can access them even when
you're not online, or when the site hosting them goes down.
`wget' is used as a backend for archival, so any of its options
can be used directly from org-board. This means you can download
whole sites for archival with a couple of keystrokes, while
keeping track of your archives from a simple Org file.
* Summary
In org-board, a bookmark is represented by an Org heading of any
level, with a `URL' property containing one or more URLs. Once
such a heading is created, a call to `org-board-archive' creates a
unique ID and directory for the entry via `org-attach', archives
the contents and requisites of the page(s) listed in the `URL'
property using `wget', and saves them inside the entry's
directory. A link to the (timestamped) root archive folder is
created in the property `ARCHIVED_AT'. Multiple archives can be
made for each entry. Additional options to pass to `wget' can be
specified via the property `WGET_OPTIONS'. The variable
`org-board-after-archive-functions' (defaulting to nil) holds a
list of functions to run after each archival operation.
* User commands
`org-board-archive' archives the current entry, creating a unique
ID and directory via `org-attach' if necessary.
`org-board-archive-dry-run' shows the `wget' invocation that will
run for this entry in the echo area.
`org-board-new' prompts for a URL to add to the current entry's
properties, then archives the entry immediately.
`org-board-delete-all' deletes all the archives for this entry by
deleting the `org-attach' directory.
`org-board-open' opens the bookmark at point in a browser.
Default to the built-in browser, `eww', and with prefix, the
native operating system browser.
`org-board-diff' uses `zdiff' (if available) or `ediff' to
recursively diff two archives of the same entry.
`org-board-diff3' uses `ediff' to recursively diff three archives
of the same entry.
`org-board-cancel' cancels the current org-board archival process.
`org-board-run-after-archive-function' prompts for a function and
an archive in the current entry, and applies the function to the
archive.
These are all bound in the `org-board-keymap' variable (not bound
to any key by default).
* Customizable options
`org-board-wget-program' is the path to the wget program.
`org-board-wget-switches' are the command line options to use with
`wget'. By default these are included as:
"-e robots=off" ignores robots.txt files.
"--page-requisites" downloads all page requisites (CSS, images).
"--adjust-extension" add a ".html" extension where needed.
"--convert-links" convert external links to internal.
`org-board-agent-header-alist' is an alist mapping agent names to
their respective header/user-agent arguments. Set a
`WGET_OPTIONS' property to a key of this alist (say,
`Mac-OS-10.8') and org-board will replace the key with its
corresponding value before calling wget. This is useful for some
sites that refuse to serve pages to `wget'.
`org-board-wget-show-buffer' controls whether the archival process
buffer is shown in a window (defaults to true).
`org-board-log-wget-invocation' controls whether to log the
archival process command in the root of the archival directory
(defaults to true).
`org-board-domain-regexp-alist' applies certain options when a
domain matches a regular expression. See the docstring for
details. As an example, this is used to make sure that `wget'
does not send a User Agent string when archiving from Google
Cache, which will not normally serve pages to it.
`org-board-after-archive-functions' (default nil) holds a list of
functions to run after an archival takes place. This is intended
for user extensions to `org-board'. The functions receive three
arguments: a list of URLs downloaded, the folder name where they
were downloaded and the process filter event string (see the Elisp
manual for details on the possible values of this string). For an
example use of `org-board-after-archive-functions', see the
"Example usage" section below.
* Known limitations
Options like "--header: 'Agent X" cannot be specified as
properties, because the property API splits on spaces, and such an
option has to be passed to `wget' as one argument. To work around
this, add these types of options to `org-board-agent-header-alist'
instead, where the property API is not involved.
At the moment, only one archive can be done at a time.
* Example usage
** Archiving
I recently found a list of articles on linkers that I wanted to
bookmark and keep locally for offline reading. In a dedicated org
file for bookmarks I created this entry:
** TODO Linkers (20-part series)
:PROPERTIES:
:URL: http://a3f.at/lists/linkers
:WGET_OPTIONS: --recursive -l 1 --span-hosts
:END:
Where the `URL' property is a page that already lists the URLs
that I wanted to download. I specified the recursive property for
`wget' along with a depth of 1 ("-l 1") so that each linked page
would be downloaded. With point inside the entry, I run "M-x
org-board-archive". An `org-attach' directory is created and
`wget' starts downloading the pages to it. Afterwards, the end
the entry looks like this:
** TODO Linkers (20-part series)
:PROPERTIES:
:URL: http://a3f.at/lists/linkers
:WGET_OPTIONS: --recursive -l 1 --span-hosts
:ID: D3BCE79F-C465-45D5-847E-7733684B9812
:ARCHIVED_AT: [2016-08-30-Tue-15-03-56]
:END:
The value in the `ARCHIVED_AT' property is a link that points to
the root of the timestamped archival directory. The ID property
was automatically generated by `org-attach'.
** Diffing
You can diff between two archives done for the same entry using
`org-board-diff', so you can see how a page has changed over time.
The diff recurses through the directory structure of an archive
and will highlight any changes that have been made. `ediff' is
used if `zdiff' is not available (both are capable of recursing
through a directory structure, but `zdiff' is possibly more
intuitive to use). `org-board-diff3' also offers diffing between
three different archive directories.
** `org-board-after-archive-functions'
`org-board-after-archive-functions' is a list of functions run
after an archive is finished. You can use it to do anything you
like with newly archived pages. For example, you could add a
function that copies the new archive to an external hard disk, or
opens the archived page in your browser as soon as it is done
downloading. You could also, for instance, copy all of the media
files that were downloaded to your own media folder, and pop up a
Dired buffer inside that folder to give you the chance to
organize them.
Here is an example function that copies the archived page to an
external service called `IPFS' <http://ipfs.io/>, a decentralized
versioning and storage system geared towards web content (thanks
to Alan Schmitt):
(defun org-board-add-to-ipfs (urls output-folder event &rest _rest)
"Add the downloaded site to IPFS."
(unless (string-match "exited abnormally" event)
(let* ((parsed-url (url-generic-parse-url (car urls)))
(domain (url-host parsed-url))
(path (url-filename parsed-url))
(output (shell-command-to-string
(concat "ipfs add -r "
(concat output-folder domain))))
(ipref
(nth 1 (split-string
(car (last (split-string output "\n" t))) " "))))
(with-current-buffer (get-buffer-create "*org-board-post-archive*")
(princ (format "your file is at %s\n"
(concat "http://localhost:8080/ipfs/" ipref path))
(current-buffer))))))
(eval-after-load "org-board"
'(add-hook 'org-board-after-archive-functions 'org-board-add-to-ipfs))
Note that for forward compatibility, it's best to add to a final
`&rest' argument to every function listed in
`org-board-after-archive-functions', since a future update may
provide each function with additional arguments (like a marker
pointing to a buffer position where the archive was initiated, for
example).
For more information on `org-board-after-archive-functions', see
its docstring and the docstring of
`org-board-test-after-archive-function'.
You can also interactively run an after-archive function with the
command `org-board-run-after-archive-function'. See its docstring
for details.
* Getting started
** Installation
There are two ways to install the package. One way is to clone
this repository and add the directory to your load-path manually.
(add-to-list 'load-path "/path/to/org-board")
(require 'org-board)
Alternatively, you can download the package directly from Emacs
using MELPA <https://github.com/melpa/melpa>. M-x
package-install RET org-board RET will take care of it.
** Keybindings
The following keymap is defined in `org-board-keymap':
| Key | Command |
| a | org-board-archive |
| r | org-board-archive-dry-run |
| n | org-board-new |
| k | org-board-delete-all |
| o | org-board-open |
| d | org-board-diff |
| 3 | org-board-diff3 |
| c | org-board-cancel |
| x | org-board-run-after-archive-function |
| O | org-attach-reveal-in-emacs |
| ? | Show help for this keymap. |
To install the keymap give it a prefix key, e.g.:
(global-set-key (kbd "C-c o") org-board-keymap)
Then typing `C-c o a' would run `org-board-archive', for example.
* Miscellaneous
The location of `wget' should be picked up automatically from the
`PATH' environment variable. If it is not, then the variable
`org-board-wget-program' can be customized.
Other options are already set so that archiving bookmarks is done
pretty much automatically. With no `WGET_OPTIONS' specified, by
default `org-board-archive' will just download the page and its
requisites (images and CSS), and nothing else.
** Support for org-capture from Firefox (thanks to Alan Schmitt):
On the Firefox side, install org-capture from here:
http://chadok.info/firefox-org-capture/
Alternatively, you can do it manually by following the
instructions here:
http://weblog.zamazal.org/org-mode-firefox/
(in the “The advanced way” section)
When org-capture is installed, add `(require 'org-protocol)' to
your init file (`~/.emacs').
Then create a capture template like this:
(setq org-board-capture-file "my-org-board.org")
(setq org-capture-templates
`(...
("c" "capture through org protocol" entry
(file+headline ,org-board-capture-file "Unsorted")
"* %?%:description\n:PROPERTIES:\n:URL: %:link\n:END:\n\n Added %U")
...))
And add a hook to `org-capture-before-finalize-hook':
(defun do-org-board-dl-hook ()
(when (equal (buffer-name)
(concat "CAPTURE-" org-board-capture-file))
(org-board-archive)))
(add-hook 'org-capture-before-finalize-hook 'do-org-board-dl-hook)
* Acknowledgements
Thanks to Alan Schmitt for the code to combine `org-board' and
`org-capture', and for the example function used in the
documentation of `org-board-after-archive-functions' above.