? my $ctx = $main::context; ? $_mt->wrapper_file("wrapper.mt", "Configure", "Using Mruby")->(sub {
mruby is a lightweight implementation of the Ruby programming language. With H2O, users can implement their own request handling logic using mruby, either to generate responses or to fix-up the request / response.
The interface between the mruby program and the H2O server is based on Rack interface specification. Below is a simple configuration that returns hello world.
= $ctx->{example}->('Hello-world in mruby', <<'EOT') paths: "/": mruby.handler: | Proc.new do |env| [200, {'content-type' => 'text/plain'}, ["Hello world\n"]] end EOT ?>It should be noted that as of H2O version 1.7.0, there are limitations when compared to ordinary web application server with support for Rack such as Unicorn:
In addition to the Rack interface specification, H2O recognizes status code 399
which can be used to delegate request to the next handler.
The feature can be used to implement access control and response header modifiers.
By using the 399
status code, it is possible to implement access control using mruby.
The example below restricts access to requests from 192.168.
private address.
Support for Basic Authentication is also provided by an mruby script.
When enabled using the reproxy
directive, it is possible to delegate the request from the mruby handler to any other handler.
= $ctx->{example}->('Rewriting URL with delegation', <<'EOT') paths: "/": mruby.handler: | lambda do |env| if /\/user\/([^\/]+)/.match(env["PATH_INFO"]) return [307, {"x-reproxy-url" => "/user.php?user=#{$1}"}, []] end return [399, {}, []] end EOT ?>
When the mruby handler returns status code 399
, H2O delegates the request to the next handler while preserving the headers emitted by the handler.
The feature can be used to add extra headers to the response.
For example, the following example sets cache-control
header for requests against .css
and .js
files.
Or in the example below, the handler triggers HTTP/2 server push with the use of Link: rel=preload
headers, and then requests a FastCGI application to process the request.
Starting from version 1.7, a HTTP client API is provided. HTTP requests issued through the API will be handled asynchronously; the client does not block the event loop of the HTTP server.
= $ctx->{example}->('Mruby handler returning the response of http://example.com', <<'EOT') paths: "/": mruby.handler: | Proc.new do |env| req = http_request("http://example.com") status, headers, body = req.join [status, headers, body] end EOT ?>
http_request
is the method that issues a HTTP request.
The method takes two arguments.
First argument is the target URI.
Second argument is an optional hash; method
(defaults to GET
), header
, body
attributes are recognized.
The method returns a promise object.
When #join
method of the promise is invoked, a three-argument array containing the status code, response headers, and the body is returned.
The response body is also a promise.
Applications can choose from three ways when dealing with the body: a) call #each
method to receive the contents, b) call #join
to retrieve the body as a string, c) return the object as the response body of the mruby handler.
The header and the body object passed to http_request
should conform to the requirements laid out by the Rack specification for request header and request body.
The response header and the response body object returned by the #join
method of the promise returned by http_request
conforms to the requirements of the Rack specification.
Since the API provides an asynchronous HTTP client, it is possible to effectively issue multiple HTTP requests concurrently and merge them into a single response.
When HTTPS is used, servers are verified using the properties of proxy.ssl.cafile
and proxy.ssl.verify-peer
specified at the global level.
In version 2.3, it is possible from mruby to set and log an arbitrary-named variable that is associated to a HTTP request.
A HTTP response header that starts with x-fallthru-set-
is handled specially by the H2O server. Instead of sending the header downstream, the server accepts the value as a request environment variable, taking the suffix of the header name as the name of the variable.
This example shows how to read request data, parse json and then log data from mruby.
= $ctx->{example}->('Logging the content of a POST request via request environment variable', <<'EOT') paths: "/": mruby.handler: | Proc.new do |env| input = env["rack.input"] ? env["rack.input"].read : '{"default": "true"}' parsed_json = JSON.parse(input) parsed_json["time"] = Time.now.to_i logdata = parsed_json.to_s [204, {"x-fallthru-set-POSTDATA" => logdata}, []] end access-log: path: /path/to/access-log.json escape: json format: '{"POST": %{POSTDATA}e}' EOT ?> ? })