Add Json API to some endpoints #5

Merged
reynir merged 10 commits from json_responses into main 2024-12-20 11:46:47 +00:00
Showing only changes of commit 44e7cd566f - Show all commits

View file

@ -70,11 +70,22 @@ let mime_lookup path =
let string_of_html = let string_of_html =
Format.asprintf "%a" (Tyxml.Html.pp ()) Format.asprintf "%a" (Tyxml.Html.pp ())
let or_error_response r = let is_accept_json req =
match Dream.header req "Accept" with
| Some accept when String.starts_with ~prefix:"application/json" accept ->
true
| _ -> false
let or_error_response req r =
let* r = r in let* r = r in
match r with match r with
| Ok response -> Lwt.return response | Ok response -> Lwt.return response
| Error (text, status) -> Dream.respond ~status text | Error (text, status) ->
if is_accept_json req then
let json_response = Yojson.Basic.to_string (`Assoc [ "error", `String text ]) in
Review

This could certainly be made into a much more complicated object. I don't have strong opinions on that.

This could certainly be made into a much more complicated object. I don't have strong opinions on that.
Review

There are "standards" out there suggesting you use an "errors" field with an array of error objects: https://jsonapi.org/format/#error-objects

There are "standards" out there suggesting you use an "errors" field with an array of error objects: https://jsonapi.org/format/#error-objects
Review

Yes that is probably the right way to go

Yes that is probably the right way to go
Review

I merged it as is. We can revisit the errors later.

I merged it as is. We can revisit the errors later.
Dream.json ~status json_response
else
Dream.respond ~status text
let default_log_warn ~status e = let default_log_warn ~status e =
Log.warn (fun m -> m "%s: %a" (Dream.status_to_string status) pp_error e) Log.warn (fun m -> m "%s: %a" (Dream.status_to_string status) pp_error e)
@ -397,8 +408,7 @@ let routes ~datadir ~cachedir ~configdir ~expired_jobs =
~log:(fun e -> Log.warn (fun m -> m "Error getting job build: %a" pp_error e)) ~log:(fun e -> Log.warn (fun m -> m "Error getting job build: %a" pp_error e))
>>= fun (build, main_binary, artifacts, same_input_same_output, different_input_same_output, same_input_different_output, latest, next, previous) -> >>= fun (build, main_binary, artifacts, same_input_same_output, different_input_same_output, same_input_different_output, latest, next, previous) ->
let solo5_manifest = Option.bind main_binary (Model.solo5_manifest datadir) in let solo5_manifest = Option.bind main_binary (Model.solo5_manifest datadir) in
match Dream.header req "Accept" with if is_accept_json req then
| Some accept when String.starts_with ~prefix:"application/json" accept ->
let json_response = let json_response =
`Assoc [ `Assoc [
"job_name", `String job_name; "job_name", `String job_name;
@ -410,7 +420,7 @@ let routes ~datadir ~cachedir ~configdir ~expired_jobs =
] |> Yojson.Basic.to_string ] |> Yojson.Basic.to_string
in in
Dream.json ~status:`OK json_response |> Lwt_result.ok Dream.json ~status:`OK json_response |> Lwt_result.ok
| _ -> else
Views.Job_build.make Views.Job_build.make
~job_name ~job_name
~build ~build
@ -605,45 +615,44 @@ let routes ~datadir ~cachedir ~configdir ~expired_jobs =
let compare_builds req = let compare_builds req =
process_comparison req >>= fun process_comparison req >>= fun
(job_left, job_right, build_left, build_right, build_left_file_size, build_right_file_size, env_diff, pkg_diff, opam_diff) -> (job_left, job_right, build_left, build_right, build_left_file_size, build_right_file_size, env_diff, pkg_diff, opam_diff) ->
match Dream.header req "Accept" with if is_accept_json req then
| Some accept when String.starts_with ~prefix:"application/json" accept -> let file_size_json = Option.fold ~none:`Null ~some:(fun size -> `Int size) in
let file_size_json = Option.fold ~none:`Null ~some:(fun size -> `Int size) in let json_response =
let json_response = `Assoc [
`Assoc [ "left", `Assoc [
"left", `Assoc [ "job_name", `String job_left;
"job_name", `String job_left; "uuid", `String (Uuidm.to_string build_left.uuid);
"uuid", `String (Uuidm.to_string build_left.uuid); "platform", `String build_left.platform;
"platform", `String build_left.platform; "start_time", `String (Ptime.to_rfc3339 build_left.start);
"start_time", `String (Ptime.to_rfc3339 build_left.start); "finish_time", `String (Ptime.to_rfc3339 build_left.finish);
"finish_time", `String (Ptime.to_rfc3339 build_left.finish); "main_binary", `Bool (Option.is_some build_left_file_size);
"main_binary", `Bool (Option.is_some build_left_file_size); "main_binary_size", file_size_json build_left_file_size;
"main_binary_size", file_size_json build_left_file_size; ];
]; "right", `Assoc [
"right", `Assoc [ "job_name", `String job_right;
"job_name", `String job_right; "build", `String (Uuidm.to_string build_right.uuid);
"build", `String (Uuidm.to_string build_right.uuid); "platform", `String build_right.platform;
"platform", `String build_right.platform; "start_time", `String (Ptime.to_rfc3339 build_right.start);
"start_time", `String (Ptime.to_rfc3339 build_right.start); "finish_time", `String (Ptime.to_rfc3339 build_right.finish);
"finish_time", `String (Ptime.to_rfc3339 build_right.finish); "main_binary", `Bool (Option.is_some build_right_file_size);
"main_binary", `Bool (Option.is_some build_right_file_size); "main_binary_size", file_size_json build_right_file_size;
"main_binary_size", file_size_json build_right_file_size; ];
]; "env_diff", Utils.diff_map_to_json env_diff;
"env_diff", Utils.diff_map_to_json env_diff; "package_diff", Utils.diff_map_to_json pkg_diff;
"package_diff", Utils.diff_map_to_json pkg_diff; "opam_diff", Opamdiff.compare_to_json opam_diff
"opam_diff", Opamdiff.compare_to_json opam_diff ] |> Yojson.Basic.to_string
] |> Yojson.Basic.to_string in
in Dream.json ~status:`OK json_response |> Lwt_result.ok
Dream.json ~status:`OK json_response |> Lwt_result.ok else
| _ -> Views.compare_builds
Views.compare_builds ~job_left ~job_right
~job_left ~job_right ~build_left ~build_right
~build_left ~build_right ~env_diff
~env_diff ~pkg_diff
~pkg_diff ~opam_diff
~opam_diff |> string_of_html |> Dream.html |> Lwt_result.ok
|> string_of_html |> Dream.html |> Lwt_result.ok in
in
let upload_binary req = let upload_binary req =
let job = Dream.param req "job" in let job = Dream.param req "job" in
@ -683,7 +692,7 @@ let routes ~datadir ~cachedir ~configdir ~expired_jobs =
>>= fun () -> Dream.respond "" |> Lwt_result.ok >>= fun () -> Dream.respond "" |> Lwt_result.ok
in in
let w f req = or_error_response (f req) in let w f req = or_error_response req (f req) in
[ [
`Get, "/", (w (builds ~all:false ~filter_builds_later_than:expired_jobs)); `Get, "/", (w (builds ~all:false ~filter_builds_later_than:expired_jobs));