From f8a7a805527d9c58e42d4d56db59730673f0012f Mon Sep 17 00:00:00 2001 From: Samuel Hym Date: Fri, 29 Nov 2024 19:32:15 +0100 Subject: [PATCH] Add support for directory listings in `Staticmod` Add a simple generator for directory listings: - the listings are unstyled - the generation must load the full directory (so that the entries can be sorted) - consequently the generation should probably not be enabled when there are huge directories (which are usually a bad idea anyhow) --- src/extensions/dune | 2 +- src/extensions/staticmod.ml | 69 +++++++++++++++++++++++++++++++------ 2 files changed, 59 insertions(+), 12 deletions(-) diff --git a/src/extensions/dune b/src/extensions/dune index 538a2b8b6..71d2acb5c 100644 --- a/src/extensions/dune +++ b/src/extensions/dune @@ -56,7 +56,7 @@ (name staticmod) (public_name ocsigenserver.ext.staticmod) (modules staticmod) - (libraries ocsigenserver)) + (libraries tyxml ocsigenserver)) (library (name userconf) diff --git a/src/extensions/staticmod.ml b/src/extensions/staticmod.ml index a6af87930..72a7b064f 100644 --- a/src/extensions/staticmod.ml +++ b/src/extensions/staticmod.ml @@ -116,6 +116,55 @@ let find_static_page ~request ~usermode ~dir ~(err : Cohttp.Code.status) (Ocsigen_extensions.Error_in_user_config_file "Staticmod: cannot use '..' in user paths") +let respond_dir relpath dname : (Cohttp.Response.t * Cohttp_lwt.Body.t) Lwt.t = + let readsortdir = + (* Read a complete directory and sort its entries *) + let chunk_size = 1024 in + let rec aux entries dir = + Lwt_unix.readdir_n dir chunk_size >>= fun chunk -> + let entries = chunk :: entries in + if Array.length chunk < chunk_size + then Lwt.return entries + else aux entries dir + in + Lwt_unix.opendir dname >>= fun dir -> + Lwt.finalize + (fun () -> + aux [] dir >|= fun entries -> + List.sort compare (List.concat_map Array.to_list entries)) + (fun () -> Lwt_unix.closedir dir) + in + Lwt.catch + (fun () -> + readsortdir >>= fun entries -> + let render e = Format.asprintf "%a" (Tyxml.Html.pp_elt ()) e in + let t = render (Tyxml.Html.txt ("Directory listing for " ^ relpath)) in + let entries = + let open Tyxml.Html in + List.filter_map + (function + | "." | ".." -> None + | e -> Some (render (li [a ~a:[a_href e] [txt e]]))) + entries + in + (* Chunks of [html (head (title t) []) (body [h1 [t]; ul entries])] *) + let chunk1 = + {| +|} + and chunk2 = {|

|} + and chunk3 = {|

|} in + let doc = + chunk1 :: t :: chunk2 :: t :: chunk3 :: (entries @ [chunkend]) + in + let headers = Cohttp.Header.init_with "content-type" "text/html" in + Lwt.return + ( Cohttp.Response.make ~status:`OK ~headers () + , Cohttp_lwt.Body.of_string_list doc )) + (function + | Unix.Unix_error _ -> Cohttp_lwt_unix.Server.respond_not_found () + | exn -> Lwt.fail exn) + let gen ~usermode ?cache dir = function | Ocsigen_extensions.Req_found _ -> Lwt.return Ocsigen_extensions.Ext_do_nothing @@ -123,20 +172,18 @@ let gen ~usermode ?cache dir = function (err, ({Ocsigen_extensions.request_info; _} as request)) -> let try_block () = Lwt_log.ign_info ~section "Is it a static file?"; + let pathstring = + Ocsigen_lib.Url.string_of_url_path ~encode:false + (Ocsigen_request.sub_path request_info) + in let status_filter, page = - let pathstring = - Ocsigen_lib.Url.string_of_url_path ~encode:false - (Ocsigen_request.sub_path request_info) - in find_static_page ~request ~usermode ~dir ~err ~pathstring in - let fname = - match page with - | Ocsigen_local_files.RFile fname -> fname - | Ocsigen_local_files.RDir _ -> - failwith "FIXME: staticmod dirs not implemented" - in - Cohttp_lwt_unix.Server.respond_file ~fname () >>= fun answer -> + (match page with + | Ocsigen_local_files.RFile fname -> + Cohttp_lwt_unix.Server.respond_file ~fname () + | Ocsigen_local_files.RDir dname -> respond_dir pathstring dname) + >>= fun answer -> let answer = Ocsigen_response.of_cohttp answer in let answer = if not status_filter