seppo_bin.ml 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. (*
  2. * _ _ ____ _
  3. * _| || |_/ ___| ___ _ __ _ __ ___ | |
  4. * |_ .. _\___ \ / _ \ '_ \| '_ \ / _ \| |
  5. * |_ _|___) | __/ |_) | |_) | (_) |_|
  6. * |_||_| |____/ \___| .__/| .__/ \___/(_)
  7. * |_| |_|
  8. *
  9. * Personal Social Web.
  10. *
  11. * Copyright (C) The #Seppo contributors. All rights reserved.
  12. *
  13. * This program is free software: you can redistribute it and/or modify
  14. * it under the terms of the GNU General Public License as published by
  15. * the Free Software Foundation, either version 3 of the License, or
  16. * (at your option) any later version.
  17. *
  18. * This program is distributed in the hope that it will be useful,
  19. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  20. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  21. * GNU General Public License for more details.
  22. *
  23. * You should have received a copy of the GNU General Public License
  24. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  25. *)
  26. module C = Cgi
  27. open Seppo_lib
  28. open Astring
  29. (**
  30. Being a CGI[1] means #Seppo is a binary executable reading from stdin
  31. writing to stdout in the first place. The Webserver puts incoming HTTP Header
  32. variables into the environment.
  33. Serving dynamic content via CGI can be as simple as putting the executable
  34. into the web directory of e.g. an Apache webserver and entering the according
  35. http url into your browser.
  36. Besides that, seppo.cgi has housekeeping commands for the commandline.
  37. [1] RFC3875 The Common Gateway Interface https://www.rfc-editor.org/rfc/rfc3875.html
  38. *)
  39. let () =
  40. Mirage_crypto_rng_unix.use_default ();
  41. let uuid = () |> (() |> Random.State.make_self_init |> Uuidm.v4_gen) in
  42. let resp_err ec ?(hdrs = [Http.H.ct_plain]) status msg =
  43. (try Logr.err (fun m -> m "%a %s %s" Uuidm.pp uuid ec msg);
  44. with _ -> ());
  45. Cgi.Response.flush uuid stdout (status, hdrs, fun oc ->
  46. let s = status |> Cohttp.Code.string_of_status in
  47. Printf.eprintf "FATAL: %s\r\nsee %s\r\n%s\r\n" s ec msg;
  48. Printf.fprintf oc "Status: %s\r\n\r\n%s\r\n%s" s ec msg ) in
  49. (try match Cgi.Request.(from_env () |> consolidate |> proxy) with
  50. | Error "Not Found." ->
  51. (* some CLI commands are fine without logging. At least -h and -V have to. *)
  52. Sys.argv
  53. |> Array.to_list
  54. |> Shell.exec
  55. | Error msg ->
  56. resp_err E.e1036 `Internal_server_error msg
  57. | Ok req ->
  58. req.script_name
  59. |> Cgi.cd_cgi_bin_twin_path
  60. |> Unix.chdir;
  61. match Shell.log_dir |> File.mkdir_p 0o770 with
  62. | Error msg -> resp_err E.e1035 `Internal_server_error msg
  63. | Ok log_dir ->
  64. assert (log_dir |> St.is_suffix ~affix:"/");
  65. let tnow = Ptime_clock.now () in
  66. Logr.open_out Shell.log_file;
  67. let r = req
  68. |> C.handle uuid tnow stdin
  69. |> Cgi.Response.flush uuid stdout in
  70. close_in stdin;
  71. (* closing kills the cgi: close_out stdout; *)
  72. flush stderr;
  73. (* L.close_out (); *)
  74. r
  75. with
  76. | Unix.(Unix_error(ENOENT, "chdir", dir)) -> resp_err E.e1037 `Internal_server_error ("cd: The directory '" ^ dir ^ "' does not exist")
  77. | Sys_error msg -> resp_err E.e1034 `Internal_server_error msg
  78. | e -> resp_err E.e1005 `Internal_server_error (Printexc.to_string e) )
  79. |> exit