Test your API against OpenApi v3 documentation
Inspired by Apivore
Add this line to your application's Gemfile:
gem 'openapi_rspec'
And then execute:
$ bundle
Or install it yourself as:
$ gem install openapi_rspec
Add spec/openapi_helper.rb
and set OpenapiRspec.config.app
:
# spec/openapi_helper.rb
require "rails_helper"
OpenapiRspec.config.app = Rails.application # or any other Rack app, thanks to rack-test gem
Then configure path to your documentation. You can use documentation defined as:
- static file with
.yaml
/.yml
or.json
extension - uri to OpenAPI schema in your application
# spec/openapi_helper.rb
#...
# static file
API_V1 = OpenapiRspec.api("./spec/data/openapi.yml")
# application path
API_V2 = OpenapiRspec.api_by_path("/openapi.json")
RSpec.describe "API v1" do
subject { API_V1 }
it "is valid OpenAPI spec" do
expect(subject).to validate_documentation
end
end
You can validate documentation against additional custom schemata, for example, to enforce organization documentation standards:
# spec/openapi_helper.rb
#...
API_V1 = OpenapiRspec.api("./spec/data/openapi.yml", additional_schemas: ["./spec/data/acme_schema.yml"])
# openapi_v1_spec.rb
RSpec.describe "API v1" do
subject { API_V1 }
it "is valid OpenAPI and ACME spec" do
expect(subject).to validate_documentation
end
end
General example:
require "openapi_rspec"
RSpec.describe "API v1 /pets" do
subject { API_V1 }
get "/pets" do
headers { { "X-Client-Device" => "ios" } }
query { { tags: ["lucky"] } }
validate_code(200) do |validator|
result = JSON.parse(validator.response.body)
expect(result.first["name"]).to eq("Lucky")
end
end
post "/pets" do
params { { name: "Lucky" } }
validate_code(201)
end
get "/pets/{id}" do
let(:id) { 23 }
validate_code(200)
context "when pet not found" do
let(:id) { "bad_id" }
validate_code(404)
end
end
To validate response use:
get
,post
,put
,patch
,delete
andhead
methods to describe operation with the pathvalidate_code
method with passed expected code
# ...
get "/pets" do
validate_code(200)
end
# ...
To set request body (as form data) use params
helper method and provide a Hash
:
# ...
post "/pets" do
params { { name: "Lucky" } }
validate_code(201)
end
# ...
To set raw request body use the params
helper method and provide a String
:
# ...
post "/pets" do
params { JSON.dump(name: "Lucky") }
validate_code(201)
end
# ...
To set path parameter use let
helper method:
# ...
get "/pets/{id}" do
let(:id) { 23 }
validate_code(200)
end
# ...
To set query parameters use query
helper method:
# ...
get "/pets" do
query { { name: "lucky" } }
validate_code(200)
end
# ...
To set header parameters use headers
helper method:
# ...
get "/pets" do
headers { { "X-User-Token" => "bad_token" } }
validate_code(401)
end
# ...
You can access response to add custom validation like this:
# ...
get "/pets" do
validate_code(200) do |validator|
result = JSON.parse(validator.response.body)
expect(result).not_to be_empty
end
end
# ...
To prefix each request with "/some_path"
use :api_base_path
option:
# spec/openapi_helper.rb
#...
API_V1 = OpenapiRspec.api("./spec/data/openapi.yml", api_base_path: "/some_path")
To validate this we will use a small hack:
# spec/openapi_helper.rb
# ...
API_V1_DOC = OpenapiRspec.api("./openapi/openapi.yml", api_base_path: "/api/v1")
RSpec.configure do |config|
config.after(:suite) do
unvalidated_requests = API_V1_DOC.unvalidated_requests
if unvalidated_requests.any? && ENV["TEST_COVERAGE"]
raise unvalidated_requests.map { |path| "Path #{path.join(' ')} not tested" }.join("\n")
end
end
end
Then run your specs:
$ TEST_COVERAGE=1 rspec
This will raise an error if any of the documented paths are not validated.
Bug reports and pull requests are welcome on GitHub at https://github.com/medsolutions/openapi_rspec.
The gem is available as open source under the terms of the MIT License.