1 -- This file is part of Diohsc
2 -- Copyright (C) 2020 Martin Bays <mbays@sdf.org>
4 -- This program is free software: you can redistribute it and/or modify
5 -- it under the terms of version 3 of the GNU General Public License as
6 -- published by the Free Software Foundation, or any later version.
8 -- You should have received a copy of the GNU General Public License
9 -- along with this program. If not, see http://www.gnu.org/licenses/.
12 {-# LANGUAGE LambdaCase #-}
24 import Control
.Monad
(msum)
25 import Data
.List
(isPrefixOf)
26 import Data
.Maybe (fromMaybe, isNothing)
27 import Safe
(atMay
, headMay
, maximumBound
, readMay
)
28 import System
.FilePath ((</>))
30 import qualified Data
.Text
.Lazy
as T
34 commands
:: Bool -> [String]
35 commands restricted
= metaCommands
++ navCommands
++ infoCommands
++
36 actionCommands restricted
++ otherCommands
38 metaCommands
= ["help", "quit"]
39 navCommands
= ["repeat", "mark", "inventory", "identify", "add", "fetch", "delete"]
40 infoCommands
= ["show", "page", "uri", "links", "mime"]
41 unsafeActionCommands
= ["save", "view", "browse", "!", "|", "||", "||-"]
42 safeActionCommands
= ["cat"]
43 actionCommands
True = safeActionCommands
44 actionCommands
False = unsafeActionCommands
++ safeActionCommands
45 otherCommands
= ["commands", "log", "query", "repl", "alias", "set", "at"]
47 normaliseCommand
:: String -> Maybe String
48 normaliseCommand partial
=
49 headMay
. filter (isPrefixOf partial
) $ commands
False
53 [ withBoldStr
"Navigation"
55 , "Enter a URI to go to it, then enter numbers to follow links."
57 , "You can navigate to other locations as follows:"
58 , " ../foo.gmi : Relative URI"
59 , " < , {-back} : Back"
60 , " > , {-forward} : Forward"
61 , " 'foo : Marked location"
62 , " ~ , {-next} : Next queue item"
63 , " } : Next unvisited link"
64 , " <} : Next unvisited link of parent"
66 , "See \"{help} {targets}\" for more."
68 , withBoldStr
"Commands"
72 , " {-help [TOPIC]} : This help, or help on a command or other topic"
73 , "Navigation commands:"
74 , " {-mark MARK} : Mark location as \"'MARK\""
75 , " 3-5 {-add} : Append links to queue"
76 , " {-repeat} : Make fresh request for current location"
77 , " {-inventory} : Show history and queue"
78 , " {-identify [ID]} : Select/create identity (client certificate)"
79 , " {-repl} : Enter loop making queries at current uri"
81 , " {-show} : Print text"
82 , " {-page} : Page text"
83 , " {-links} : Show links"
84 , " {-uri} : Show uri"
85 , " {-save [FILENAME]} : Save data, by default in {~/saves/}"
86 , " {-| CMD} : Pipe data to shell command"
87 , " {-! CMD} : Run shell command on data"
88 , " {-|| [CMD]} : Pipe rendered text to $PAGER or command"
89 , " {-view} : Run mailcap command on data"
90 , " {-browse [CMD] [ARGS]} : Run command on uri (default: $BROWSER)"
92 , "Commands and marks may be abbreviated; e.g. \"l\" is short for {links}."
94 , "Commands which act on the current location by default"
95 , "can also be given a target before the command, e.g.:"
96 , " 3 {|} mpv - : Request link 3 and pipe the stream to command 'mpv -'"
97 , " << {save} blah : Save data from history item before last"
98 , " 2 {browse} lynx : Run \"lynx [uri-of-link-2]\""
99 , " / {mark} root : Mark the root of the current capsule as 'root"
100 , " / {identify} asdf : Use identity \"asdf\" for all of current capsule"
101 , "Use \"{help} {targets}\" to get full details on this notation."
103 , "Spaces around the command can often be omitted, e.g.:"
104 , " 2a0 : Add link 2 to start of queue"
106 , withBoldStr
"Miscellaneous"
108 , "Use ^C to interrupt requests and abort prompts and so forth."
109 , "^C will never quit the program."
111 , "Use enter or 'q' to quit the pager, and space or 1-9 or 'h' to advance,"
112 , "and '>' to queue commands. See \"{help} {pager}\" for further details and commands."
114 , "There are a few more obscure commands;"
115 , "use {commands} to show them all, and \"{help} [command]\" for information."
117 , "Use \"{help} {topics}\" for a list of further help topics."
120 , withBoldStr
"Windows"
122 , "diohsc was written with unix-like systems in mind."
123 , "Various commands may not work well on Windows,"
124 , "and some of the help text may be unhelpful. Sorry about that."
125 , "Please do write to mbays@sdf.org with your complaints/bugreports."
130 topics
= ["targets", "queue", "pager", "trust", "configuration"
131 , "default_action", "proxies", "geminators", "render_filter"
132 , "pre_display", "link_desc_first", "log_length", "max_wrap_width", "no_confirm"
133 , "verbose_connection", "copying", "topics"]
135 helpOn
:: String -> [String]
137 fromMaybe ["Unknown command/topic; use {commands} and \"{help} {topics}\" for lists."] $ msum
138 [ commandHelp
<$> normaliseCommand s
139 , topicHelp
<$> completeTopic
]
141 completeTopic
= headMay
. filter (isPrefixOf s
) $ topics
++ topicSynonyms
142 topicSynonyms
= ["geminator", "proxy"]
144 "topics" -> ('{' :) . (++"}") <$> topics
146 [ "Many commands can be given a target or targets to operate on."
147 , "These are written before the command."
148 , "A target consists of an optional base target optionally followed by modifiers."
151 , " example.com , gemini://example.com : absolute URI"
152 , " 3 : link from current"
153 , " .. , ./foo , ../foo , /foo , ?foo : URI relative to current location"
154 , " 'example , 'ex , 'e : location marked with \"mark example\""
155 , " '' : location last jumped away from"
156 , " ~ : next queue item"
158 , " If no base target is given, the current location is used as the base."
161 , " /foo , ?foo : URI relative to base"
162 , " _3 : link from base"
163 , " < , {-back} : origin of base"
164 , " > , {-forward} : location of which base is the origin (if any)"
165 , " ] : link after \">\""
166 , " [ : link before \">\""
167 , " @ : root of base (i.e. recursive origin)"
168 , " The \"origin\" of a location reached via a link is the link's source."
169 , " Similarly for relative URIs. Other locations have no origin."
171 , "Example session demonstrating basic navigation:"
172 , " 'ex?foo : Go to mark \"'example\", but with query string set to \"foo\"."
173 , " 5 : Go to link 5 from 'ex?foo."
174 , " < : Go back to 'ex?foo."
175 , " ] : Go to link 6 from 'ex?foo."
176 , " <_2 : Go to link 2 from 'ex?foo."
177 , " <] : Go to link 3 from 'ex?foo. Call this location B."
178 , " 7 : Go to link 7 from B. Call this location C."
179 , " <<] : Go to link 4 from 'ex?foo. C is now set as the jump mark \"''\"."
180 , " ''<] : Go to link 8 from B."
181 , " @/bar : Go to 'ex/bar"
183 , "Numbered, range, and search targets:"
184 , " The specifiers \"$\", \"~\", \"<\", \">\", \"[\" and \"]\" all accept"
185 , " a number, or repetition of the symbol, to specify a particular item."
186 , " e.g \"~~~\" and \"~3\" both refer to the third queue entry."
188 , " Specify ranges with \"-\"; start and/or end may be omitted."
189 , " Specify first match of pattern with \"^pattern^\","
190 , " or all matches with \"^^pattern^\"."
191 , " Specify multiple items or ranges by separating with \",\"."
192 , " Last item is denoted by \"$\", and nth from last by \"$n\"."
193 , " Examples: \"1,3-5,^pattern^-$2\" refers to links 1,3,4,5 and all links from"
194 , " the first match of pattern to the penultimate item;"
195 , " \"~-\" refers to the whole queue."
197 , " Patterns are (extended) regular expressions (see `man 7 regex`)."
198 , " Matching is case-insensitive unless pattern contains an uppercase character."
199 , " Space and '^' in patterns must be backslash-escaped."
200 , " The terminating \"^\" may be omitted."
202 , "Restricting to unvisited:"
203 , " \"}\", \"{\", and \"*\" work like \"]\", \"[\", and \"_\","
204 , " but consider only unvisited links. Example: \"*-a\" adds all unvisited links."
206 , "See also: {mark}, {queue}, {alias}, {query}"
209 [ "The queue is a list of uris, which you can add to with \"add\""
210 , "and visit using {next} or by referring to them as \"~\", \"~3\", etc."
212 , "One way to use this: Whenever you see multiple links you would like to read,"
213 , "where in a tabbed browser you might open a new background tab for each,"
214 , "you can add them to the queue and then read each in turn."
215 , "A queue item is deleted by {delete} or any command which requests the uri;"
216 , "use marks and history (including \"$^pattern\") to revisit old entries."
218 , "You can use e.g. \"-a0\" to add all links to the *start* of the queue."
219 , "The queue can be manipulated with {add}; e.g."
220 , "\"~4-a0\" shifts queue entries ~4 onwards to the start of the queue."
221 , "The queue can also be used to build a list of targets for batch processing;"
222 , "e.g. you can add the desired uris to the queue then use \"~-|grep pattern\"."
224 , "Any uris written to {~/queue} (one uri per line)"
225 , "will be added to the queue, after which that file will be deleted."
226 , "This allows e.g. an rss reader to add to the queue of a diohsc instance."
228 , "You may also create named queues like \"foo~\" with a name argument to {add}."
229 , "The corresponding file is e.g. {~/queues/foo}."
232 [ "Keys for the inbuilt pager:"
233 , " space : advance one page"
234 , " h : advance half a page"
235 , " 1-9 : advance by the specified number of lines"
236 , " c : continue to end"
237 , " q, enter : quit pager"
238 , " :, > : enter a diohsc command to be executed immediately. Examples:"
239 , " \":3v\" views link 3 (e.g. an image)"
240 , " \":5a\" adds link 5 to the uri queue"
242 , "There is no way to go backwards; use your terminal's scrollback facility."
243 , "The {||} command can be used to invoke an alternative pager."
244 , "See also: {default_action}" ]
246 [ "{set} {proxy} SCHEME HOST:PORT : Set proxy for requests using given scheme"
247 , "{set} {proxy} SCHEME : unset proxy"
250 , " set proxy gopher 127.0.0.1:1965"
251 , "to use an Agena ({%Yellow%https://tildegit.org/solderpunk/agena}) instance"
252 , "running locally with its default configuration." ]
253 "proxy" -> topicHelp
"proxies"
255 [ "A valid certificate chain presented by a server will be trusted if the root"
256 , "is a Certificate Authority certificate found under {~/trusted_certs/}."
257 , "Otherwise you will be asked whether to trust the server certificate."
258 , "If you accept, it will be saved in {~/known_hosts/},"
259 , "and you will be warned if the server ever presents a different certificate." ]
261 [ "There are some commandline options; try \"diohsc --help\"."
263 , "There are also some options which can be set at run-time;"
264 , "use {set} for a list and their current values. Each option is a help topic."
266 , "{~/diohscrc} may contain commands to run at startup,"
267 , "e.g. setting aliases and identities and the options mentioned above;"
268 , "each line of the file is interpreted as a command."
269 , "See diohscrc.sample in the source distribution for some suggestions."
271 , "The files in {~/} can be edited."
272 , "To change the default save directory,"
273 , "make {~/saves} a symlink."
274 , "To disable command history, make {~/commandHistory}"
275 , "a symlink to /dev/null ; similarly for inputHistory and log."
276 , "(Alternatively, use the --ghost commandline option.)"
278 , "The line editor can be configured:"
279 , "see {%Yellow%https://github.com/judah/haskeline/wiki/UserPreferences}"
281 , "See also: {alias}, {identify}, {set}, {trust}" ]
283 [ "{set} {default_action} COMMAND [ARGS]: set action used on going to a new location."
284 , "The default is \"page\". You may prefer \"||\", or e.g. \"|| less\"."
285 , "See also: {configuration}" ]
286 "geminator" -> topicHelp
"geminators"
288 [ "{set} {geminators} MIMETYPE COMMAND: set shell command for conversion to text/gemini."
289 , "{set} {geminators} MIMETYPE: unset geminator."
291 , "The body of a response with a matching mimetype is piped through the command,"
292 , "and the output used as gemini text for rendering (\"page\", \"||\", etc),"
293 , "and for obtaining links."
295 , "Mimetype is a regular expression. The first matching geminator will be used."
298 , " set gem text/markdown md2gemini -l paragraph"
299 , " (see {%Yellow%https://github.com/makeworld-the-better-one/md2gemini})"
300 , " set gem (text|application)/(html|xml|xhtml.*) html2gmi -me"
301 , " (see {%Yellow%https://github.com/LukeEmmet/html2gmi})"
302 , " set gem image/jpeg echo '```' && jp2a --colors - && echo '```'"
303 , " set gem image/.* echo '```' && convert - jpeg:- | jp2a --colors - && echo '```'"
304 , " (ascii-art preview of images)" ]
306 [ "{set} {render_filter} COMMAND: set shell command to filter rendered text through."
307 , "{set} {render_filter}: unset render_filter."
309 , "Whenever the rendered text of a page would be used (\"page\", \"||\", etc),"
310 , "it will be piped through this command first."
313 , " set render_filter stdbuf -o0 uni2ascii -BPq"
314 , " (best-effort substitution of utf8 with pure-ascii equivalents)" ]
316 [ "{set} {pre_display} pre: suppress alt text of preformatted blocks"
317 , "{set} {pre_display} alt: display only alt text of preformatted blocks"
318 , "{set} {pre_display} both: display alt text and contents of preformatted blocks" ]
320 [ "{set} {link_desc_first} true: show link description before uri"
321 , "{set} {link_desc_first} false: show uri before link description" ]
323 [ "{set} {log_length} N: set number of items to store in log"
324 , "{set} {log_length} 0: clear log and disable logging"
325 , "See also: {log}" ]
327 [ "{set} {max_wrap_width} N: set maximum width for text wrapping" ]
329 [ "{set} {no_confirm} true: disable confirmation prompts for certain commands"
330 , "{set} {no_confirm} false: re-enable confirmation prompts"
332 , "You are advised not to enable this until you are familiar with the behaviour of"
333 , "the potentially dangerous commands like \"|\" and \"!\"" ]
334 "verbose_connection" ->
335 [ "{set} {verbose_connection} true: show extra information about connections"
336 , "{set} {verbose_connection} false: suppress extra information about connections"
339 [ "diohsc is free software, released under the terms of the GNU GPL v3 or later."
340 , "You should have obtained a copy of the licence as the file COPYING."
341 , "This version of diohsc is copyright Martin Bays <mbays@sdf.org> 2020." ]
342 t
-> ["No help on topic \"" <> t
<> "\"."]
345 [ "help: show general help"
346 , "help COMMAND: show help on command" ]
348 "repeat" -> ["TARGET repeat: request target"]
350 [ "TARGET {mark} MARK: mark target, which can subsequently be specified as 'MARK."
351 , "{mark}: list marks."
352 , "Marks are saved in {~/marks}. To delete a mark, remove the corresponding file."
354 , "The mark '' is a special \"jump back\" mark which is automatically set when"
355 , "navigating to a new uri without following a link."
357 , "Marks '0 to '9 are special per-session marks:"
358 , "they are not saved, and they are listed in the output of \"inventory\"."
360 , "The marks '' and '0-'9 refer to targets with their full history;"
361 , "they and their ancestors can be manipulated without causing network requests." ]
363 [ "{inventory}: show current queue (~N), path (<N,>N), and session marks ('N)."
364 , "See also: {log}" ]
367 , "TARGETS {log}: add targets to log."
369 , "The \"log\" is a list of visited URIs."
370 , "Its entries can be referenced with \"$\"."
372 , "URIs added to the log are considered \"visited\"; they are shown in a"
373 , "different colour and can be referenced with \"*\", \"{\", and \"}\"."
375 , "The log is saved in {~/log}."
376 , "To prevent excessive resource use and limit the privacy implications,"
377 , "the length of the log is bounded by the option {log_length}." ]
379 [ "TARGET {identify} [IDENTITY]: identify (as identity) for all future"
380 , " requests to target and to paths below target."
381 , " If identity doesn't exist, create a new identity."
383 , "An \"identity\" is a cryptographic certificate,"
384 , "sent to the server to securely identify you to the server."
385 , "An identity which will be used for a request is indicated as \"{%Yellow%uri}[{%Green%identity}]\"."
387 , "TARGET {identify} IDENTITY ed: create identity with an Ed25519 key pair"
388 , "Ed25519 uses much smaller keys than the default RSA algorithm,"
389 , "but some servers may fail to accept identities created using it."
390 , "See also: {configuration}" ]
392 [ "TARGETS {add}: add targets to the end of the queue."
393 , "TARGETS {add} 0: add targets to the start of the queue."
394 , "TARGETS {add} N: add targets to the queue after entry ~N."
396 , "TARGETS {add} foo: add targets to the named queue foo~."
397 , "TARGETS {add} foo N: add targets to the named queue after entry foo~N."
399 , "See also: {fetch}, {queue}, {targets}." ]
401 [ "{fetch} acts like {add}, but targets are fetched and cached before being added."
402 , "See {add} for syntax." ]
404 [ "TARGETS {delete}: delete specified uris from the queue."
405 , "e.g. \"~3-5,7d\" to delete certain entries, or \"~-d\" to clear the queue,"
406 , "or \"-d\" to delete all queue entries which are links from the current location."
408 , "TARGETS {delete} foo: delete specified uris from the named queue foo~."
410 "show" -> ["TARGET {show}: show rendered text of target, without paging."]
411 "page" -> [ "TARGET {page}: page rendered text of target."
412 , "See also: {pager}, {default_action}" ]
413 "uri" -> ["TARGET {uri}: show absolute uri of target."]
414 "links" -> ["TARGET {links}: show list of links of target."]
415 "mime" -> [ "TARGET {mime}: show mime type of target."
416 , "Note: any request this causes will be closed after receiving the header." ]
417 "save" -> [ "TARGET {save} [PATH]: save body."
418 , "If path is omitted or relative, it is based on {~/saves/} ."
419 , "The default filename is the last non-empty segment of the uri path,"
420 , "or the hostname if the path is empty." ]
421 "view" -> [ "TARGET {view}: run \"run-mailcap --view\" on body."
422 , "The action is determined by the mime-type; see the run-mailcap manpage." ]
423 "browse" -> [ "TARGET {browse}: run command given by environment variable $BROWSER on uri."
424 , "TARGET {browse} COMMAND: run given shell command on uri."
425 , "%s is substituted with the uri"
426 , "if no %s appears, the uri is used as an additional final argument."
427 , "A literal '%' can be escaped as '%%'." ]
428 "!" -> [ "TARGET {!} COMMAND: run shell command on body."
429 , "The line after '!' is used as a shell command after transforming as follows:"
430 , "%s is substituted with the path to a temporary file containing the target;"
431 , "if no %s appears, this path is appended to the end (separated by a space)."
432 , "A literal '%' can be escaped as '%%'."
433 , "Environment variables $URI and $MIMETYPE are set to correspond to the target." ]
434 "|" -> [ "TARGET {|} COMMAND: pipe body through shell command."
435 , "Environment variables $URI and $MIMETYPE are set to correspond to the target." ]
436 "||" -> [ "TARGET {||} [COMMAND]: pipe rendered text through shell command."
437 , "The default command is the contents of the environment variable $PAGER."
438 , "Environment variables $URI and $MIMETYPE are set to correspond to the target."
439 , "See {||-} for a variant which does not produce ansi escapes."
440 , "See also: {default_action}, {||-}" ]
441 "||-" -> [ "TARGET {||-} [COMMAND]: pipe plain rendered text through shell command."
442 , "This is the same as {||}, but no ansi escapes are included in the text." ]
443 "cat" -> [ "TARGET {cat}: print raw contents of location" ]
444 "commands" -> [ "{commands}: show list of commands and aliases,"
445 , "in order of priority when expanding abbreviations,"
446 , "and show the shortest permissible abbreviations."
447 , "See also: {alias}"]
448 "query" -> [ "TARGET {query} QUERY: request target with query set to QUERY."
449 , "Unlike TARGET?QUERY, this command does not require spaces to be escaped,"
450 , "and it can be aliased; e.g. if 'search is a mark set to a search engine:"
451 , " alias S 'search query"
454 , "The following backslash escape sequences will be interpreted"
455 , "(these can also be used with TARGET?QUERY by escaping the backslash):"
457 , " \\r : carriage return"
460 , " \\xHH : byte with given hex encoding"
461 , " \\\\ : backslash"
463 "repl" -> [ "TARGET {repl}: enter read-eval-print-loop,"
464 , "in which each line of input is used as a query string at the target."
465 , "To return to normal command mode, enter an empty query or ^C or ^D." ]
467 [ "{alias} ALIAS COMMANDLINE: add an alias"
468 , "{alias} ALIAS: delete an existing alias"
469 , "The commandline may include targets and/or a command."
471 , " alias up .. : then \"up\" translates to \"..\", and e.g. \"u add\" to \".. add\""
472 , " alias Mpv |mpv --cache-secs 5 - : then \"2M\" will stream link 2 to mpv"
473 , " with this sane caching (mpv's default cache size is 150M!)"
474 , "You can put alias commands in {~/diohscrc};"
475 , "see \"{help} {configuration}\"." ]
476 "set" -> [ "{set}: show settable options and their current values"
477 , "{set} OPTION VALUE [..]: set option"
478 , "Try using {help} on the options."
479 , "See also: {configuration}"
481 "at" -> [ "TARGET {at} COMMANDLINE: request target then execute commandline based there."
483 , "Example: 'example at *- add: add all unvisited links from 'example to queue." ]
484 c
-> ["No help on command \"" <> c
<> "\"."]
486 showMinPrefix
:: Bool -> [String] -> String -> String
487 showMinPrefix ansi ss s
=
488 let n
= maximumBound
0 $ commonPrefLen s
<$> takeWhile (/= s
) ss
489 (s
',s
'') = splitAt (n
+1) s
491 then s
<> applyIf ansi
(withColourStr Red
) " [Not typable!]"
492 else applyIf ansi withBoldStr s
' <>
493 applyIf
(not $ ansi ||
null s
'') (('[':) . (++"]")) s
''
495 commonPrefLen
:: Eq a
=> [a
] -> [a
] -> Int
496 commonPrefLen bs cs
= head [ n
498 , let mb
= atMay bs n
499 , let mc
= atMay cs n
500 , isNothing mb ||
isNothing mc || mb
/= mc
503 -- indicate initial prefix of commands/aliases in string, marked as {command},
504 -- and topics marked as {topic},
505 -- and path from userdir, marked as {~/path}.
506 -- Use {-command blah} in a table, it adds spaces as necessary.
507 -- e.g. {%Yellow%str} prints str in yellow (when ansi).
508 expandHelp
:: Bool -> [String] -> String -> String -> String
509 expandHelp ansi aliases userDir
= expandHelp
' where
510 cs
= aliases
++ commands
False
512 |
(pre
,'{':s
') <- break (== '{') s
513 , (twixt
,'}':post
) <- break (== '}') s
'
514 = let sub
= case twixt
of
515 '~
':'/':path
-> userDir
</> path
516 '-':cBlah |
(c
,blah
) <- break (== ' ') cBlah
518 let c
' = showMinPrefix ansi cs c
519 missing
= length c
+ 3 - visibleLength
(T
.pack c
')
520 in c
' <> blah
<> replicate missing
' '
521 '%':t
' |
(colStr
,'%':str
) <- break (== '%') t
'
522 , Just col
<- readMay colStr
->
523 applyIf ansi
(withColourStr col
) str
524 c | c `
elem` cs
-> showMinPrefix ansi cs c
525 t | t `
elem` topics
-> showMinPrefix ansi
(cs
++ topics
) t
526 _
-> '{' : twixt
++ "}"
527 in pre
<> sub
<> expandHelp
' post