Skip to content

Commit

Permalink
Merge pull request ocaml-sf#597 from pfitaxel/feat/serve-during-rebuild
Browse files Browse the repository at this point in the history
Add option `learn-ocaml build serve --serve-during-build` and fix related minor issues
erikmd authored Apr 19, 2024

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
2 parents b7ed29c + cc6caa3 commit dc6f569
Showing 9 changed files with 203 additions and 59 deletions.
2 changes: 1 addition & 1 deletion Dockerfile
Original file line number Diff line number Diff line change
@@ -48,7 +48,7 @@ ENTRYPOINT ["dumb-init","/usr/bin/learn-ocaml-client"]
FROM alpine:3.13 as program

RUN apk update \
&& apk add ncurses-libs libev dumb-init git openssl \
&& apk add ncurses-libs libev dumb-init git openssl lsof \
&& addgroup learn-ocaml \
&& adduser learn-ocaml -DG learn-ocaml

2 changes: 1 addition & 1 deletion Dockerfile.test-server
Original file line number Diff line number Diff line change
@@ -54,7 +54,7 @@ LABEL org.label-schema.build-date="${BUILD_DATE}" \
org.label-schema.schema-version="1.0"

RUN apk update \
&& apk add ncurses-libs libev dumb-init git openssl \
&& apk add ncurses-libs libev dumb-init git openssl lsof \
&& addgroup learn-ocaml \
&& adduser learn-ocaml -DG learn-ocaml

3 changes: 3 additions & 0 deletions learn-ocaml.opam
Original file line number Diff line number Diff line change
@@ -17,6 +17,9 @@ license: "MIT"
homepage: "https://github.com/ocaml-sf/learn-ocaml"
bug-reports: "https://github.com/ocaml-sf/learn-ocaml/issues"
dev-repo: "git+https://github.com/ocaml-sf/learn-ocaml"
depexts: [
["lsof"] {os-distribution = "alpine"}
]
depends: [
"asak" { >= "0.4"}
"base64"
4 changes: 2 additions & 2 deletions src/app/learnocaml_index_main.ml
Original file line number Diff line number Diff line change
@@ -824,8 +824,8 @@ let () =
Lwt.async @@ fun () ->
set_string_translations ();
Dom_html.document##.title :=
Js.string ("Learn OCaml" ^ " v."^Learnocaml_api.version);
Manip.setInnerText El.version ("v."^Learnocaml_api.version);
Js.string ("Learn OCaml" ^ " v"^Learnocaml_api.version);
Manip.setInnerText El.version ("v"^Learnocaml_api.version);
Learnocaml_local_storage.init () ;
let sync_button_group = button_group () in
disable_button_group sync_button_group;
145 changes: 113 additions & 32 deletions src/main/learnocaml_main.ml
Original file line number Diff line number Diff line change
@@ -76,6 +76,23 @@ module Args = struct
Mandatory for '$(b,learn-ocaml build)' if the site is not hosted in path '/', \
which typically occurs for static deployment."

let serve_during_build =
value & flag &
info ["serve-during-build"] ~docs:"SERVER OPTIONS"
~env:(Cmd.Env.info "LEARNOCAML_SERVE_DURING_BUILD") ~doc:
"If the directory specified by $(b,--app-dir) already exists from a \
previous build, create a temporary child process to serve it \
while the build completes, in order to reduce server downtime. \
This flag requires to run both commands '$(b,learn-ocaml build serve)'. \
After the build, the child process stops and a new server starts. \
This flag is useful in a docker-compose context, and can be enabled \
by adding to the environment: '$(env)=true'."

let child_pid =
(* Note: option `--child-pid` is specific to the native learn-ocaml-server,
hence this dummy value here, to avoid copying it in "SERVER OPTIONS". *)
Term.const (None: int option)

module Grader = struct
let info = info ~docs:"GRADER OPTIONS"

@@ -263,17 +280,18 @@ module Args = struct
app_dir: string;
repo_dir: string;
build_dir: string;
serve_during_build: bool;
grader: Grader.t;
builder: Builder.t;
server: Server.t;
}

let term =
let apply commands app_dir repo_dir build_dir grader builder server =
{ commands; app_dir; repo_dir; build_dir; grader; builder; server }
let term child_pid =
let apply commands app_dir repo_dir build_dir grader builder server serve_during_build =
{ commands; app_dir; repo_dir; build_dir; grader; builder; server; serve_during_build }
in
Term.(const apply $commands $app_dir $repo_dir $build_dir
$Grader.term $Builder.term $Server.term app_dir base_url)
$Grader.term $Builder.term $Server.term app_dir base_url child_pid $serve_during_build)
end

open Args
@@ -319,7 +337,7 @@ let temp_app_dir o =
((basename o.app_dir) ^ ".temp")

let main o =
Printf.printf "Learnocaml v.%s running.\n%!" Learnocaml_api.version;
Printf.printf "Learnocaml v%s running.\n%!" Learnocaml_api.version;
let grade o =
if List.mem Grade o.commands then
(if List.mem Build o.commands || List.mem Serve o.commands then
@@ -391,23 +409,24 @@ let main o =
end
else Lwt.return_unit
in
let generate o =
let generate ?(check_port = true) o =
if List.mem Build o.commands then
(let get_app_dir o =
if not (List.mem Serve o.commands) then
Lwt.return o.app_dir
else if o.server.Server.replace then
let app_dir = temp_app_dir o in
(if Sys.file_exists app_dir then
else if o.server.Server.replace || o.serve_during_build then
let temp_dir = temp_app_dir o in
(if Sys.file_exists temp_dir then
(Printf.eprintf "Warning: temporary directory %s already exists\n%!"
app_dir;
temp_dir;
Lwt.return_unit)
else if Sys.file_exists o.app_dir then
Lwt_utils.copy_tree o.app_dir app_dir
Lwt_utils.copy_tree o.app_dir temp_dir
else
Lwt.return_unit)
>>= fun () -> Lwt.return app_dir
else if Learnocaml_server.check_running () <> None then
>>= fun () -> Lwt.return temp_dir
else if check_port && Learnocaml_server.check_running () <> None then
(* This server-specific check is here to fail earlier if need be *)
(Printf.eprintf
"Error: another server is already running on port %d \
(consider using option `--replace`)\n%!"
@@ -500,12 +519,29 @@ let main o =
else
Lwt.return true
in
let run_server o =
let kill_once pid =
let already = ref false in
fun () ->
if !already then () else
(already := true;
Unix.kill pid Sys.sigint;
Printf.eprintf "Waiting for child process %d to terminate... %!" pid;
ignore (Unix.waitpid [] pid);
prerr_endline "ok ")
in
(* child_pid = None => no --serve-during-build
child_pid = Some 0 => --serve-during-build, child process
child_pid = Some n, n>0 => --serve-during-build, main process *)
let run_server ~child_pid o =
if List.mem Serve o.commands then
let () =
if o.server.Server.replace then
let running = Learnocaml_server.check_running () in
Option.iter Learnocaml_server.kill_running running;
let int_child_pid = Option.value child_pid ~default:(-1) in
if o.server.Server.replace || (o.serve_during_build && int_child_pid > 0) then
let () =
(if int_child_pid > 0 then kill_once int_child_pid ()
else let running = Learnocaml_server.check_running () in
Option.iter Learnocaml_server.kill_running running)
in
let temp = temp_app_dir o in
let app_dir = absolute_filename o.app_dir in
let bak =
@@ -542,6 +578,8 @@ let main o =
("--sync-dir="^o.server.sync_dir) ::
("--base-url="^o.builder.Builder.base_url) ::
("--port="^string_of_int o.server.port) ::
(match child_pid with None -> [] | Some n -> ["--child-pid="^string_of_int n])
@
(match o.server.cert with None -> [] | Some c -> ["--cert="^c])
in
Lwt.return
@@ -550,8 +588,13 @@ let main o =
Unix.execv native_server
(Array.of_list (native_server::server_args))))
else begin
Printf.printf "Starting server on port %d\n%!"
!Learnocaml_server.port;
let comment = match child_pid with
| None -> ""
| Some 0 -> "(temporary)"
| Some _pid -> "(main)"
in
Printf.printf "Starting server%s on port %d\n%!"
comment !Learnocaml_server.port;
if o.builder.Builder.base_url <> "" then
Printf.printf "Base URL: %s\n%!" o.builder.Builder.base_url;
Learnocaml_server.launch () >>= fun ret ->
@@ -560,19 +603,57 @@ let main o =
else
Lwt.return (`Success true)
in
let lwt_run_server ~child_pid build_ok =
if build_ok then
run_server ~child_pid o >>= function
| `Success true -> Lwt.return (`Code 0)
| `Success false -> Lwt.return (`Code 10)
| `Continuation f -> Lwt.return (`Continuation f)
else
Lwt.return (`Code 1)
in
(* NOTE: the code below handles "learn-ocaml build serve --serve-during-build"
by relying on Lwt_unix.fork; and to stay on the safe side, we make sure
that this fork is triggered before the first Lwt_main.run command. *)
let ret =
Lwt_main.run
(grade o >>= function
| Some i -> Lwt.return (`Code i)
| None ->
generate o >>= fun success ->
if success then
run_server o >>= function
| `Success true -> Lwt.return (`Code 0)
| `Success false -> Lwt.return (`Code 10)
| `Continuation f -> Lwt.return (`Continuation f)
else
Lwt.return (`Code 1))
if o.serve_during_build then begin
if not (List.mem Build o.commands && List.mem Serve o.commands) then
(Printf.eprintf
"Error: option `--serve-during-build` requires both commands `build serve`.\n%!";
exit 1)
else if o.server.Server.replace then
(Printf.eprintf
"Error: option `--replace` is incompatible with option `--serve-during-build`.\n%!";
exit 10)
else if Learnocaml_server.check_running () <> None then
(Printf.eprintf
"Error: another server is already running on port %d \
(consider using option `--replace` instead of `--serve-during-build`)\n%!"
!Learnocaml_server.port;
exit 10);
match Lwt_unix.fork () with
| 0 ->
if Sys.file_exists o.app_dir then
Lwt_main.run (lwt_run_server ~child_pid:(Some 0) true)
else
(Printf.eprintf
"Info: no existing app-dir in '%s', \
will be available at next run (skipping temporary server start).\n%!" o.app_dir;
`Code 0)
| child_pid ->
at_exit (kill_once child_pid);
Lwt_main.run
(grade o >>= function
| Some i -> Lwt.return (`Code i)
| None ->
generate ~check_port:false o >>= lwt_run_server ~child_pid:(Some child_pid))
end
else
Lwt_main.run
(grade o >>= function
| Some i -> Lwt.return (`Code i)
| None ->
generate o >>= lwt_run_server ~child_pid:None)
in
match ret with
| `Code n -> exit n
@@ -627,7 +708,7 @@ let main_info =
~version:Learnocaml_api.version
"learn-ocaml"

let main_term = Term.(const main $ Args.term)
let main_term = Term.(const main $ Args.term child_pid)

let () =
match
20 changes: 13 additions & 7 deletions src/main/learnocaml_server_args.ml
Original file line number Diff line number Diff line change
@@ -20,9 +20,10 @@ module type S = sig
port: int;
cert: string option;
replace: bool;
child_pid: int option;
}

val term: string Cmdliner.Term.t -> string Cmdliner.Term.t -> t Cmdliner.Term.t
val term: string Cmdliner.Term.t -> string Cmdliner.Term.t -> int option Cmdliner.Term.t -> t Cmdliner.Term.t
end

module Args (SN : Section_name) = struct
@@ -54,19 +55,24 @@ module Args (SN : Section_name) = struct

let replace =
value & flag &
info ["replace"] ~doc:
"Replace a previously running instance of the server on the same port."
info ["replace"] ~env:(Cmd.Env.info "LEARNOCAML_REPLACE") ~doc:
"Replace a previously running instance of the server on the same port. \
Use this to reduce server downtime when updating the content \
of an instance: the running server will only be stopped once the \
new one is ready. If running in a Docker context, you may want to \
have a look at the flag $(b,--serve-during-build) instead."

type t = {
sync_dir: string;
base_url: string;
port: int;
cert: string option;
replace: bool;
child_pid: int option;
}

let term app_dir base_url =
let apply app_dir sync_dir base_url port cert replace =
let term app_dir base_url child_pid =
let apply app_dir sync_dir base_url port cert replace child_pid =
Learnocaml_store.static_dir := app_dir;
Learnocaml_store.sync_dir := sync_dir;
let port = match port, cert with
@@ -80,10 +86,10 @@ module Args (SN : Section_name) = struct
| None -> None);
Learnocaml_server.port := port;
Learnocaml_server.base_url := base_url;
{ sync_dir; base_url; port; cert; replace }
{ sync_dir; base_url; port; cert; replace; child_pid }
in
(* warning: if you add any options here, remember to pass them through when
calling the native server from learn-ocaml main *)
Term.(const apply $ app_dir $ sync_dir $ base_url $ port $ cert $ replace)
Term.(const apply $ app_dir $ sync_dir $ base_url $ port $ cert $ replace $ child_pid)

end
3 changes: 2 additions & 1 deletion src/main/learnocaml_server_args.mli
Original file line number Diff line number Diff line change
@@ -17,9 +17,10 @@ module type S = sig
port: int;
cert: string option;
replace: bool;
child_pid: int option;
}

val term: string Cmdliner.Term.t -> string Cmdliner.Term.t -> t Cmdliner.Term.t
val term: string Cmdliner.Term.t -> string Cmdliner.Term.t -> int option Cmdliner.Term.t -> t Cmdliner.Term.t
end

module Args : functor (_ : Section_name) -> S
77 changes: 65 additions & 12 deletions src/main/learnocaml_server_main.ml
Original file line number Diff line number Diff line change
@@ -25,22 +25,63 @@ let signal_waiter =
let _ = Lwt_unix.on_signal Sys.sigterm handler in
waiter

let kill_once pid =
let already = ref false in
fun () ->
if !already then () else
(already := true;
Unix.kill pid Sys.sigint;
Printf.eprintf "Waiting for child process %d to terminate... %!" pid;
ignore (Unix.waitpid [] pid);
prerr_endline "ok ")

let main o =
let open Server_args in
Printf.printf "Learnocaml server v.%s starting on port %d\n%!"
Learnocaml_api.version o.port;
let check_comment =
match o.child_pid with
| Some n when n < 0 ->
(Printf.eprintf "Error: incorrect value for option `--child-pid=%d`\n%!" n;
exit 10)
| None -> ""
| Some n ->
if o.replace then
(Printf.eprintf "Error: option `--replace` is incompatible with option `--child-pid`\n%!";
exit 10);
if n = 0 then
"(temporary)"
else
"(main)"
in
(* Note: "int_child_pid > 0 then at_exit (kill_once int_child_pid);" is
unneeded as "learn-ocaml serve" already made sure the child terminated. *)
Printf.printf "Learnocaml server%s v%s starting on port %d\n%!"
check_comment Learnocaml_api.version o.port;
if o.base_url <> "" then
Printf.printf "Base URL: %s\n%!" o.base_url;
let () =
match Learnocaml_server.check_running (), o.replace with
| None, _ -> ()
| Some _, false ->
Printf.eprintf "Error: another server is already running on port %d \
(consider using option `--replace`)\n%!"
!Learnocaml_server.port;
exit 10
| Some pid, true ->
Learnocaml_server.kill_running pid
match Learnocaml_server.check_running (), o.replace, o.child_pid with
| None, _, _ -> ()
| Some _, false, None ->
Printf.eprintf "Error: another server is already running on port %d \
(consider using option `--replace`)\n%!"
!Learnocaml_server.port;
exit 10
| Some _, false, Some 0 ->
Printf.eprintf "Warning(child): another server is running on port %d \
(skipping temporary server start)\n%!"
!Learnocaml_server.port;
exit 0
| Some pid, false, Some pid' ->
if pid = pid' then
kill_once pid' ()
else
(Printf.eprintf "Error: another server (pid %d) is already running on port %d \
(while expecting `--child-pid=%d`)\n%!"
pid !Learnocaml_server.port pid';
kill_once pid' ();
exit 10)
| Some pid, true, _ ->
Learnocaml_server.kill_running pid
in
let rec run () =
let minimum_duration = 15. in
@@ -98,6 +139,18 @@ let base_url =
Mandatory for '$(b,learn-ocaml build)' if the site is not hosted in path '/', \
which typically occurs for static deployment."

let child_pid =
let open Arg in
value & opt (some int) None &
info ["child-pid"] ~docv:"CHILD_PID" ~doc:
"For internal purposes."
(* This flag is used by learn-ocaml's exec call to tell learn-ocaml-server
about the pid of the child process created by 'Lwt_unix.fork' when using
the CLI option '--serve-during-build'. If 'CHILD_PID' gets the value 0,
it means the current instance is the temporary server (child process).
If 'CHILD_PID' is ommitted or has the value None, it means no fork occurred
and the server should check no concurrent server is running on this port. *)

let exits =
let open Cmd.Exit in
[ info ~doc:"Default exit." ok
@@ -116,7 +169,7 @@ let main_info =
"learn-ocaml-server"


let main_term = Term.(const main $ Server_args.term app_dir base_url)
let main_term = Term.(const main $ Server_args.term app_dir base_url child_pid)

let () =
match
6 changes: 3 additions & 3 deletions src/server/learnocaml_server.ml
Original file line number Diff line number Diff line change
@@ -733,9 +733,9 @@ let check_running () =
let kill_running pid =
let timeout = 15 in
Unix.kill pid Sys.sigint;
Printf.eprintf "Waiting for process %d to terminate... %2d%!" pid timeout;
Printf.eprintf "Waiting for process %d to terminate... %2d %!" pid timeout;
let rec aux tout =
Printf.eprintf "\027[2D%2d" tout;
Printf.eprintf "\027[3D%2d %!" tout;
if Printf.ksprintf Sys.command "lsof -ti tcp:%d -p %d >/dev/null" !port pid
= 0
then
@@ -746,4 +746,4 @@ let kill_running pid =
aux (tout - 1))
in
aux timeout;
prerr_endline "\027[2Dok"
prerr_endline "\027[3Dok "

0 comments on commit dc6f569

Please sign in to comment.