package b0
sectionYPositions = computeSectionYPositions($el), 10)"
x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
>
Software construction and deployment kit
Install
dune-project
Dependency
Authors
Maintainers
Sources
b0-0.0.5.tbz
sha512=00a6868b4dfa34565d0141b335622a81a0e8d5b9e3c6dfad025dabfa3df2db2a1302b492953bbbce30c3a4406c324fcec25250a00b38f6d18a69e15605e3b07e
doc/src/b0_b00_kit/b00_pager.ml.html
Source file b00_pager.ml
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(*--------------------------------------------------------------------------- Copyright (c) 2018 The b0 programmers. All rights reserved. Distributed under the ISC license, see terms at the end of the file. ---------------------------------------------------------------------------*) open B0_std open B0_std.Result.Syntax open Cmdliner (* Environment variables *) module Env = struct let pager = "PAGER" let less = "LESS" let term = "TERM" end let envs = let vars = lazy begin Cmd.Env.info Env.pager ~doc:"The pager used to display content. This is a command invocation \ given to execvp(3)." :: Cmd.Env.info Env.term ~doc:"See options $(b,--color) and $(b,--no-pager)." :: [] end in fun () -> Lazy.force vars (* Paging *) let find ?win_exe ?search ~don't () = if don't then Ok None else match Os.Env.find ~empty_is_none:true Env.term with | Some "dumb" | None -> Ok None | Some _ -> let cmds = B0_std.[Cmd.atom "less"; Cmd.atom "more"] in let* cmds = let empty_is_none = true in match Os.Env.find' ~empty_is_none B0_std.Cmd.of_string Env.pager with | Error _ as e -> e | Ok None -> Ok cmds | Ok (Some cmd) -> Ok (cmd :: cmds) in let rec loop = function | [] -> Ok None | cmd :: cmds -> match Os.Cmd.find ?win_exe ?search cmd with | Ok None -> loop cmds | v -> v in loop cmds let pager_env () = match Os.Env.find ~empty_is_none:false Env.less with | Some _ -> Ok None | None -> Result.bind (Os.Env.current_assignments ()) @@ fun env -> Ok (Some ("LESS=FRX" :: env)) let page_stdout = function | None -> Ok () | Some pager -> let uerr = Unix.error_message in let err fmt = Fmt.error ("page stdout: " ^^ fmt) in let rec dup2 fd0 fd1 = match Unix.dup2 fd0 fd1 with | () -> Ok () | exception Unix.Unix_error (Unix.EINTR, _, _) -> dup2 fd0 fd1 | exception Unix.Unix_error (e, _, _) -> err "dup2: %s" (uerr e) in match pager_env () with | Error e -> err "%s" e | Ok env -> match Unix.pipe () with | exception Unix.Unix_error (e, _, _) -> err "pipe: %s" (uerr e) | (pager_read, parent_write) -> let stdin = Os.Cmd.in_fd ~close:true pager_read in Unix.set_close_on_exec parent_write; Os.Fd.apply ~close:Unix.close parent_write @@ fun parent_write -> Result.bind (Os.Cmd.spawn ?env ~stdin pager) @@ fun pid -> Result.bind (dup2 parent_write Unix.stdout) @@ fun () -> let parent_pid = Unix.getpid () in let on_parent_exit () = (* We need to be careful here, forked processes will also get execute to this. *) if parent_pid = Unix.getpid () then begin (* Before closing Unix.stdout it's better to flush formatter and channels. Otherwise it's done later by OCaml's standard shutdown procedure and it raises Sys_error as the fd is no longer valid. *) (try Fmt.flush Fmt.stdout () with Sys_error _ -> ()); (try flush stdout with Sys_error _ -> ()); (try Unix.close Unix.stdout with Unix.Unix_error _ -> ()); Log.if_error ~use:() @@ Result.map ignore @@ Os.Cmd.spawn_wait_status pid end in at_exit on_parent_exit; Ok () let page_files pager files = match pager with | Some pager when files = [] -> Ok () | Some pager -> Os.Cmd.run B0_std.Cmd.(pager %% paths files) | None -> let rec loop = function | [] -> Ok () | f :: fs -> match Os.File.read f with | Error _ as e -> e | Ok d -> Printf.printf "%s" d; if fs <> [] then Printf.printf "\x1C" (* U+001C FS *); flush stdout; loop fs in loop files (* Cli interaction *) let don't ?(docs = Manpage.s_common_options) () = let doc = "Do not display the output in a pager. This automatically happens \ if the $(b,TERM) environment variable is $(b,dumb) or unset." in Arg.(value & flag & info ["no-pager"] ~docs ~doc) (*--------------------------------------------------------------------------- Copyright (c) 2018 The b0 programmers Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. ---------------------------------------------------------------------------*)
sectionYPositions = computeSectionYPositions($el), 10)"
x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
>