let src = Logs.Src.create "builder-web" ~doc:"Builder_web" module Log = (val Logs.src_log src : Logs.LOG) open Opium open Lwt.Syntax open Lwt_result.Infix type db_error = [ Caqti_error.connect | Model.error ] let pp_error ppf = function | #Caqti_error.connect as e -> Caqti_error.pp ppf e | #Model.error as e -> Model.pp_error ppf e | `Wrong_version (application_id, version) -> if application_id = Builder_db.application_id then Format.fprintf ppf "Wrong database version: %Ld" version else Format.fprintf ppf "Wrong database application id: %ld" application_id type 'a t = { pool : (Caqti_lwt.connection, [> db_error ] as 'a) Caqti_lwt.Pool.t; datadir : Fpath.t; } let realm = "builder-web" let init ?(pool_size = 10) dbpath datadir = match Caqti_lwt.connect_pool ~max_size:pool_size (Uri.make ~scheme:"sqlite3" ~path:dbpath ~query:["create", ["false"]] ()) with | Error e -> Error e | Ok pool -> Lwt_main.run (Caqti_lwt.Pool.use (fun (module Db : Caqti_lwt.CONNECTION) -> Db.find Builder_db.get_application_id () >>= fun application_id -> Db.find Builder_db.get_version () >>= fun version -> Lwt_result.return (application_id, version)) pool) |> (function | Error e -> Error (e :> [> db_error | `Wrong_version of int32 * int64 ]) | Ok (application_id, version) -> if application_id = Builder_db.application_id && version = Builder_db.current_version then Ok { pool = (pool :> (Caqti_lwt.connection, [> db_error ]) Caqti_lwt.Pool.t); datadir; } else Error (`Wrong_version (application_id, version))) let pp_exec ppf (job, uuid, _, _, _, _, _) = Format.fprintf ppf "%s(%a)" job.Builder.name Uuidm.pp uuid let safe_seg path = if Fpath.is_seg path && not (Fpath.is_rel_seg path) then Ok (Fpath.v path) else Rresult.R.error_msgf "unsafe path %S" path (* mime lookup with orb knowledge *) let mime_lookup path = match Fpath.to_string path with | "build-environment" | "opam-switch" | "system-packages" -> "text/plain" | _ -> if Fpath.has_ext "build-hashes" path then "text/plain" else if Fpath.is_prefix Fpath.(v "bin/") path then "application/octet-stream" else Magic_mime.lookup (Fpath.to_string path) let authorized t handler = fun req -> let unauthorized = Response.of_plain_text "Forbidden!\n" ~status:`Unauthorized |> Response.add_header ("WWW-Authenticate", Auth.string_of_challenge (Basic realm)) in match Request.authorization req with | None | Some (Other _) -> Lwt.return unauthorized | Some (Basic (username, password)) -> let* user_info = Caqti_lwt.Pool.use (Model.user username) t.pool in match user_info with | Ok (Some user_info) -> if Builder_web_auth.verify_password password user_info then handler req else Lwt.return unauthorized | Ok None -> let _ : Builder_web_auth.user_info = Builder_web_auth.hash ~username ~password () in Lwt.return unauthorized | Error e -> Log.warn (fun m -> m "Error getting user: %a" pp_error e); Response.of_plain_text "Internal server error\n" ~status:`Internal_server_error |> Lwt.return let routes t = let builder _req = let* jobs = Caqti_lwt.Pool.use Model.jobs t.pool in match jobs with | Error e -> Log.warn (fun m -> m "Error getting jobs: %a" pp_error e); Response.of_plain_text ~status:`Internal_server_error "Error getting jobs" |> Lwt.return | Ok jobs -> let+ jobs = List.fold_right (fun (job_id, job_name) r -> r >>= fun acc -> Caqti_lwt.Pool.use (Model.build_meta job_id) t.pool >>= function | Some (latest_build, latest_artifact) -> Lwt_result.return ((job_name, latest_build, latest_artifact) :: acc) | None -> Log.warn (fun m -> m "Job without builds: %s" job_name); Lwt_result.return acc) jobs (Lwt_result.return []) in match jobs with | Error e -> Log.warn (fun m -> m "Error getting jobs: %a" pp_error e); Response.of_plain_text ~status:`Internal_server_error "Error getting jobs" | Ok jobs -> Views.builder jobs |> Response.of_html in let job req = let job_name = Router.param req "job" in let+ job = Caqti_lwt.Pool.use (Model.job job_name) t.pool in match job with | Error e -> Log.warn (fun m -> m "Error getting job: %a" pp_error e); Response.of_plain_text ~status:`Internal_server_error "Error getting job" | Ok builds -> Views.job job_name builds |> Response.of_html in let job_build req = let job_name = Router.param req "job" and build = Router.param req "build" in match Uuidm.of_string build with | None -> Response.of_plain_text ~status:`Bad_request "Bad request.\n" |> Lwt.return | Some uuid -> let+ build_and_artifacts = Caqti_lwt.Pool.use (Model.build uuid) t.pool >>= fun (build_id, build) -> Caqti_lwt.Pool.use (Model.build_artifacts build_id) t.pool >|= fun artifacts -> (build, artifacts) in match build_and_artifacts with | Error e -> Log.warn (fun m -> m "Error getting job build: %a" pp_error e); Response.of_plain_text ~status:`Internal_server_error "Error getting job build" | Ok (build, artifacts) -> Views.job_build job_name build artifacts |> Response.of_html in let job_build_file req = let _job_name = Router.param req "job" and build = Router.param req "build" and filepath = Router.splat req |> String.concat "/" in (* XXX: We don't check safety of [file]. This should be fine however since * we don't use [file] for the filesystem but is instead used as a key for * lookup in the data table of the 'full' file. *) match Uuidm.of_string build, Fpath.of_string filepath with | None, _ -> Log.debug (fun m -> m "bad uuid: %s" build); Response.of_plain_text ~status:`Not_found "File not found" |> Lwt.return | _, Error (`Msg e) -> Log.debug (fun m -> m "bad path: %s" e); Response.of_plain_text ~status:`Not_found "File not found" |> Lwt.return | Some build, Ok filepath -> let+ artifact = Caqti_lwt.Pool.use (Model.build_artifact build filepath) t.pool in match artifact with | Error e -> Log.warn (fun m -> m "Error getting build artifact: %a" pp_error e); Response.of_plain_text ~status:`Internal_server_error "Error getting build artifact" | Ok (data, digest) -> let body = Body.of_string data in Response.make ~body () |> Response.add_header ("Content-type", mime_lookup filepath) |> Response.set_etag (Base64.encode_string (Cstruct.to_string digest)) in let upload req = let* body = Request.to_plain_text req in match Builder.Asn.exec_of_cs (Cstruct.of_string body) with | Error (`Msg e) -> Log.warn (fun m -> m "Received bad builder ASN.1"); Log.debug (fun m -> m "Parse error: %s" e); Lwt.return (Response.of_plain_text "Bad request\n" ~status:`Bad_request) | Ok ((_, uuid, _, _, _, _, _) as exec) -> Log.info (fun m -> m "Received build %a" pp_exec exec); let* r = Caqti_lwt.Pool.use (Model.build_exists uuid) t.pool in match r with | Error e -> Log.warn (fun m -> m "Error saving build %a: %a" pp_exec exec pp_error e); Lwt.return (Response.of_plain_text "Internal server error\n" ~status:`Internal_server_error) | Ok true -> Log.warn (fun m -> m "Build with same uuid exists: %a" pp_exec exec); Lwt.return (Response.of_plain_text (Fmt.strf "Build with same uuid exists: %a\n" Uuidm.pp uuid) ~status:`Conflict) | Ok false -> let* r = Caqti_lwt.Pool.use (Model.add_build t.datadir exec) t.pool in match r with | Ok () -> Lwt.return (Response.of_plain_text "Success!\n") | Error e -> Log.warn (fun m -> m "Error saving build %a: %a" pp_exec exec pp_error e); Lwt.return (Response.of_plain_text "Internal server error\n" ~status:`Internal_server_error) in let hash req = let hash_hex = Request.query "sha256" req in match Option.map (fun h -> Hex.to_cstruct (`Hex h)) hash_hex with | None -> Log.debug (fun m -> m "sha256 query parameter not provided"); Response.of_plain_text "Bad request\n" ~status:`Bad_request |> Lwt.return | Some hash -> let+ build = Caqti_lwt.Pool.use (Model.build_hash hash) t.pool in (match build with | Error e -> Log.warn (fun m -> m "Database error: %a" pp_error e); Response.of_plain_text "Internal server error\n" ~status:`Internal_server_error | Ok None -> Log.debug (fun m -> m "Hash not found: %S" (Request.query_exn "sha256" req)); Response.of_plain_text "Artifact not found\n" ~status:`Not_found | Ok (Some (job_name, build)) -> Response.redirect_to (Fmt.strf "/job/%s/build/%a/" job_name Uuidm.pp build.Builder_db.Build.uuid)) | exception Invalid_argument _ -> Log.debug (fun m -> m "Invalid hash hex %S" (Request.query_exn "sha256" req)); Response.of_plain_text "Bad request\n" ~status:`Bad_request |> Lwt.return in [ App.get "/" builder; App.get "/job/:job/" job; App.get "/job/:job/build/:build/" job_build; App.get "/job/:job/build/:build/f/**" job_build_file; App.post "/upload" (authorized t upload); App.get "/hash" hash; ] let add_routes t (app : App.t) = List.fold_right (fun route app -> route app) (routes t) app