Skip to content

Rack middlewares to create OpenAPI based HTTP APIs

License

Notifications You must be signed in to change notification settings

gjkliewer/openapi_first

 
 

Repository files navigation

OpenapiFirst

Join the chat at https://gitter.im/openapi_first/community

OpenapiFirst helps to implement HTTP APIs based on an OpenApi API description. The idea is that you create an API description first, then add code that returns data and implements your business logic and be done.

Start with writing an OpenAPI file that describes the API, which you are about to implement. Use a validator to make sure the file is valid.

You can use OpenapiFirst via its Rack middlewares or in standalone mode.

Alternatives

This gem is inspired by committee (Ruby) and connexion (Python).

Here's a comparison between committee and openapi_first.

Rack middlewares

OpenapiFirst consists of these Rack middlewares:

  • OpenapiFirst::RequestValidation – Validates the request against the API description and returns 400 if the request is invalid.
  • OpenapiFirst::ResponseValidation Validates the response and raises an exception if the response body is invalid.
  • OpenapiFirst::Router – This internal middleware is added automatically before request/response validation. Finds the OpenAPI operation for the current request or returns 404 if no operation was found. This can be customized by adding it yourself.

And these Rack apps:

  • OpenapiFirst::Responder calls the handler found for the operation, sets the correct content-type and serializes the response body to json if needed.
  • OpenapiFirst::RackResponder calls the handler found for the operation as a normal Rack application (call(env)) and returns the result as is.

OpenapiFirst::RequestValidation

This middleware returns a 400 status code with a body that describes the error if the request is not valid.

use OpenapiFirst::RequestValidatio, spec: 'openapi.yaml'

Options and defaults

Name Possible values Description Default
spec: The path to the spec file or spec loaded via OpenapiFirst.load
raise_error: false, true If set to true the middleware raises OpenapiFirst::RequestInvalidError instead of returning 4xx. false (don't raise an exception)

The error responses conform with JSON:API.

Here's an example response body for a missing query parameter "search":

http-status: 400
content-type: "application/vnd.api+json"

{
  "errors": [
    {
      "title": "is missing",
      "source": {
        "parameter": "search"
      }
    }
  ]
}

This middleware adds env[OpenapiFirst::INBOX] which holds the (filtered) path and query parameters and the parsed request body.

Parameter validation

The middleware filteres all top-level query parameters and paths parameters and tries to convert numeric values. Meaning, if you have an :something_id path with type: integer, it will try convert the value to an integer.

It just works with a parameter with name: filter[age].

OpenapiFirst also supports type: array for query parameters and will convert items just as described above. style and explode attributes are not supported for query parameters. It will always act as if style: form and explode: false were used for query parameters.

Conversion is currently done only for path and query parameters, but not for the request body. OpenapiFirst currently does not convert date, date-time or time formats.

If you want to forbid nested query parameters you will need to use additionalProperties: false in your query parameter JSON schema.

OpenapiFirst always treats query parameters like style: deepObject, but it just works with nested objects (filter[foo][bar]=baz) (see this discussion).

Request body validation

The middleware will return a status 415 if the requests content type does not match or 400 if the request body is invalid. This will also add the parsed request body to env[OpenapiFirst::REQUEST_BODY].

Header, Cookie, Path parameter validation

tbd.

readOnly / writeOnly properties

Request validation fails if request includes a property with readOnly: true.

Response validation fails if response body includes a property with writeOnly: true.

OpenapiFirst::ResponseValidation

This middleware is especially useful when testing. It always raises an error if the response is not valid.

use OpenapiFirst::ResponseValidation, spec: 'openapi.yaml' if ENV['RACK_ENV'] == 'test'

Options

Name Possible values Description Default
spec: The path to the spec file or spec loaded via OpenapiFirst.load

OpenapiFirst::Router

This middleware first always used automatically, but you can add it to the top of your middleware stack if you want to change configuration.

use OpenapiFirst::Router, spec: './openapi/openapi.yaml'

This middleware adds env[OpenapiFirst::OPERATION] which holds an Operation object that responds to #operation_id, #path (and #[] to access raw fields).

Options and defaults

Name Possible values Description Default
spec: The path to the spec file or spec loaded via OpenapiFirst.load
                                                                                              |                                    |

| raise_error: | false, true | If set to true the middleware raises OpenapiFirst::NotFoundError when a path or method was not found in the API description. This is useful during testing to spot an incomplete API description. | false (don't raise an exception) | | not_found: | :continue, :halt | If set to :continue the middleware will not return 404 (405, 415), but just pass handling the request to the next middleware or application in the Rack stack. If combined with raise_error: true raise_error gets preference and an exception is raised. | :halt (return 4xx response) |

OpenapiFirst::RackResponder

This Rack endpoint maps the HTTP request to a method call based on the operationId in your API description and calls it as a normal Rack application. It does not not serialize objects as JSON or adds a content-type.

run OpenapiFirst::RackResponder

Options

Name Description
namespace: Optional. A class or module where to find the handler method.
resolver: Optional. An object that responds to #call(operation) and returns a handler. By default this is an instance of DefaultOperationResolver

OpenapiFirst::Responder

This Rack endpoint maps the HTTP request to a method call based on the operationId in your API description and calls it. Responder also adds a content-type to the response.

run OpenapiFirst::Responder

Options

Name Description
namespace: Optional. A class or module where to find the handler method.
resolver: Optional. An object that responds to #call(operation) and returns a handler. By default this is an instance of DefaultOperationResolver

OpenapiFirst::DefaultOperationResolver

This is the default way to look up a handler method for an operation. Handlers are always looked up in a namespace module that needs to be specified.

It works like this:

  • An operationId "create_pet" or "createPet" or "create pet" calls MyApi.create_pet(params, response)
  • "some_things.create" calls: MyApi::SomeThings.create(params, response)
  • "pets#create" instantiates the class once (MyApi::Pets::Create.new) and calls it on every request(instance.call(params, response)`).

Handlers

These handler methods are called with two arguments:

  • params - Holds the parsed request body, filtered query params and path parameters (same as env[OpenapiFirst::INBOX])
  • res - Holds a Rack::Response that you can modify if needed

You can call params.env to access the Rack env (just like in Hanami actions)

There are two ways to set the response body:

  • Calling res.write "things" (see Rack::Response)
  • Returning a value which will get converted to JSON

Standalone usage

Instead of composing these middlewares yourself you can use OpenapiFirst.app.

module Pets
  def self.find_pet(params, res)
    {
      id: params[:id],
      name: 'Oscar'
    }
  end
end

# In config.ru:
require 'openapi_first'
run OpenapiFirst.app(
  './openapi/openapi.yaml',
  namespace: Pets,
  response_validation: ENV['RACK_ENV'] == 'test',
  router_raise_error:  ENV['RACK_ENV'] == 'test'
)

The above will use the mentioned Rack middlewares to:

  • Validate the request and respond with 400 if the request does not match with your API description
  • Map the request to a method call Pets.find_pet based on the operationId in the API description
  • Set the response content type according to your spec (here with the default status code 200)

Options and defaults

Name Possible values Description Default
spec_path A filepath to an OpenAPI definition file.
namespace: A class or module where to find the handler methods.
response_validation: true, false If set to true it raises an exception if the response is invalid. This is useful during testing. false
router_raise_error: true, false If set to true it raises an exception (subclass of OpenapiFirst::Error when a request path/method is not specified. This is useful during testing. false
request_validation_raise_error: true, false If set to true it raises an exception (subclass of OpenapiFirst::Error when a request is not valid. false
resolver: Option to customize finding the handler method for an operation. See OpenapiFirst::Responder for details.

Handler functions (find_pet) are called with two arguments:

  • params - Holds the parsed request body, filtered query params and path parameters
  • res - Holds a Rack::Response that you can modify if needed If you want to access to plain Rack env you can call params.env.

If your API description does not contain all endpoints

run OpenapiFirst.middleware('./openapi/openapi.yaml', namespace: Pets)

Here all requests that are not part of the API description will be passed to the next app.

Try it out

See examples.

Installation

Add this line to your application's Gemfile:

gem 'openapi_first'

OpenapiFirst uses multi_json.

Manual response validation

Instead of using the ResponseValidation middleware you can validate the response in your test manually via rack-test and ResponseValidator.

# In your test (rspec example):
require 'openapi_first'
validator = OpenapiFirst::ResponseValidator.new('petstore.yaml')

# This will raise an exception if it found an error
validator.validate(last_request, last_response)

Handling only certain paths

You can filter the URIs that should be handled by passing only to OpenapiFirst.load:

spec = OpenapiFirst.load('./openapi/openapi.yaml', only: { |path| path.starts_with? '/pets' })
run OpenapiFirst.app(spec, namespace: Pets)

Coverage

(This is a bit experimental. Please try it out and give feedback.)

OpenapiFirst::Coverage helps you make sure, that you have called all endpoints of your OAS file when running tests via rack-test.

# In your test (rspec example):
require 'openapi_first/coverage'

describe MyApp do
  include Rack::Test::Methods

  before(:all) do
    @app_wrapper = OpenapiFirst::Coverage.new(MyApp, 'petstore.yaml')
  end

  after(:all) do
    message = "The following paths have not been called yet: #{@app_wrapper.to_be_called}"
    expect(@app_wrapper.to_be_called).to be_empty
  end

  # Overwrite `#app` to make rack-test call the wrapped app
  def app
    @app_wrapper
  end

  it 'does things' do
    get '/i/my/stuff'
    # …
  end
end

Mocking

Out of scope. Use Prism or fakeit.

Development

Run bin/setup to install dependencies.

Run bundle exec rspec to run the tests.

See bundle exec rake -T for rubygems related tasks.

Benchmarks

Results

Run benchmarks

cd benchmarks
bundle
bundle exec ruby benchmarks.rb

Contributing

If you have a question or an idea or found a bug don't hesitate to create an issue on GitHub or reach out via chat.

Pull requests are very welcome as well, of course. Feel free to create a "draft" pull request early on, even if your change is still work in progress. 🤗

About

Rack middlewares to create OpenAPI based HTTP APIs

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Ruby 99.5%
  • Other 0.5%