-
Notifications
You must be signed in to change notification settings - Fork 11
This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add an Opium example once more? #148
Comments
Hello! I don't use Opium these days so I have fallen behind on updating examples related to it after making some breaking changes within this library. That said, here's an example that should help you get started! This takes a couple of different handlers from the example shown in Opium's readme, but with open Opium
module Person = struct
type t = { name : string; age : int }
let yojson_of_t t = `Assoc [ ("name", `String t.name); ("age", `Int t.age) ]
let t_of_yojson yojson =
match yojson with
| `Assoc [ ("name", `String name); ("age", `Int age) ] -> { name; age }
| _ -> failwith "invalid person json"
end
let print_param_handler name _req =
Response.of_plain_text (Printf.sprintf "Hello, %s" name) |> Lwt.return
let print_person_handler name age _req =
{ Person.name; age } |> Person.yojson_of_t |> Response.of_json |> Lwt.return
let not_found _req = Response.of_plain_text ~status:`Not_found "" |> Lwt.return
let routed_handler routes =
let router = Routes.one_of routes in
fun req ->
match Routes.match' router ~target:req.Request.target with
| Routes.NoMatch -> not_found req
| FullMatch r -> r req
| MatchWithTrailingSlash r ->
(* This branch indicates that incoming request's path finishes with a
trailing slash. If you app needs to distinguish trailing slashes
(/a/b vs /a/b/), this is where you'd write something to handle the
use-case *)
r req
let () =
App.empty
|> App.all "**" (fun req ->
routed_handler
Routes.
[
route (s "hello" / str /? nil) print_param_handler;
route (s "person" / str / int /? nil) print_person_handler;
]
req)
|> App.run_command |
Thanks a lot, your example really helped! What library do you use to build your HTTP layer? I really like your library so I'd be interested to hear your opinion :) Usually, my routes define acting on "resources", receiving one or many verbs. Since type 'a resource = { get : 'a Routes.route; post : 'a Routes.route }
let hello_resource =
{ get = Routes.(route (s "hello" / str /? nil) print_param_handler)
; post = Routes.(route (s "hello" / str /? nil) post_stuff)
}
;;
let () =
App.empty
|> App.get "**" (fun req ->
routed_handler
Routes.
[ hello_resource.get
; route (s "person" / str / int /? nil) print_person_handler
]
req)
|> App.post "**" (fun req -> routed_handler [ hello_resource.post ] req)
|> App.run_command
;; Please let me know if you can think of a better way :) Have a great day! |
My preference is to use async when writing concurrent applications. I've been working on https://github.com/anuragsoni/shuttle/tree/main/http which provides a http codec implementation (only HTTP/1.1 for now) and my current preference is to use this when I need to write http services/clients. There are some docs hosted online (https://anuragsoni.github.io/shuttle/shuttle_http/index.html for an intro to writing servers) and https://anuragsoni.github.io/shuttle/shuttle_http/Shuttle_http/Client/index.html has some docs for using clients.
This is my preferred approach as well. I handle this slightly different from your example though, and implement the method handling within a handler like so: let validate_methods methods handler request =
match methods with
| [] -> handler request
| xs ->
if List.mem request.Request.meth xs then handler request
else Lwt.return (Response.make ~status:`Method_not_allowed ())
let print_param_handler name =
validate_methods [ `GET ] (fun _req ->
(* Dispatch different actions for HTTP verbs if needed *)
Response.of_plain_text (Printf.sprintf "Hello, %s" name) |> Lwt.return) I like this as it makes it very clear to me that a handler was invoked because a route "matched", but if it has a http verb that shouldn't be used we exit early and return a HTTP 405. Another potential option to organize this logic could be to create one router per http verb, and dispatch requests like so: module MethodMap = Map.Make (struct
type t = Method.t
let compare a b = compare a b
end)
let routed_handler' routes req =
match MethodMap.find_opt req.Request.meth routes with
| None -> Lwt.return (Response.make ~status:`Method_not_allowed ())
| Some router -> (
match Routes.match' router ~target:req.Request.target with
| Routes.NoMatch -> not_found req
| FullMatch r -> r req
| MatchWithTrailingSlash r ->
(* This branch indicates that incoming request's path finishes with a
trailing slash. If you app needs to distinguish trailing slashes
(/a/b vs /a/b/), this is where you'd write something to handle the
use-case *)
r req) I tend to not prefer this as it leads to some duplication of route definitions if the same path sequence needs to work with multiple http verbs. |
I'm going to migrate this issue to a discussion as I don't think I'll add any opium related samples directly within this repo. |
This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
Hello,
I see that you provided an example of an integration with Opium a while ago, but then removed id.
I've been having issues trying to make my demo work so I'd appreciate if you could provide a little example once more.
Thanks :)
The text was updated successfully, but these errors were encountered: