Skip to content

Commit 0b4c854

Browse files
davidebrianinathanalderson
authored andcommitted
Feat: add --check option in Mix tasks to compare generated spec w/ file (open-api-spex#618)
Add a --check option to run Mix tasks and compare the generated spec with a previously generated file. This is useful for scenarios where a CI check is desirable to catch unwanted drifts from a validated OpenAPI spec: e.g. the OpenAPI spec is committed and reviewed, and should not change without additional review. Fixes open-api-spex#617 Signed-off-by: Davide Briani <davide@briani.dev>
1 parent 9c5b82e commit 0b4c854

File tree

3 files changed

+30
-5
lines changed

3 files changed

+30
-5
lines changed

lib/mix/tasks/openapi.spec.json.ex

+3
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ defmodule Mix.Tasks.Openapi.Spec.Json do
77
88
$ mix openapi.spec.json --spec PhoenixAppWeb.ApiSpec apispec.json
99
$ mix openapi.spec.json --spec PhoenixAppWeb.ApiSpec --pretty=true
10+
$ mix openapi.spec.json --spec PhoenixAppWeb.ApiSpec --check=true
1011
$ mix openapi.spec.json --spec PhoenixAppWeb.ApiSpec --start-app=false
1112
$ mix openapi.spec.json --spec PhoenixAppWeb.ApiSpec --vendor-extensions=false
1213
@@ -16,6 +17,8 @@ defmodule Mix.Tasks.Openapi.Spec.Json do
1617
1718
* `--pretty` - Whether to prettify the generated JSON (defaults to false)
1819
20+
* `--check` - Whether to only compare the generated JSON with the spec file (defaults to false)
21+
1922
* `--start-app` - Whether need to start application before generate schema (defaults to true)
2023
2124
* `--vendor-extensions` - Whether to include open_api_spex OpenAPI vendor extensions

lib/mix/tasks/openapi.spec.yaml.ex

+3
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,16 @@ defmodule Mix.Tasks.Openapi.Spec.Yaml do
66
## Examples
77
88
$ mix openapi.spec.yaml --spec PhoenixAppWeb.ApiSpec apispec.yaml
9+
$ mix openapi.spec.yaml --spec PhoenixAppWeb.ApiSpec --check=true
910
$ mix openapi.spec.yaml --spec PhoenixAppWeb.ApiSpec --start-app=false
1011
$ mix openapi.spec.yaml --spec PhoenixAppWeb.ApiSpec --vendor-extensions=false
1112
1213
## Command line options
1314
1415
* `--spec` - The ApiSpec module from which to generate the OpenAPI YAML file
1516
17+
* `--check` - Whether to only compare the generated YAML with the spec file (defaults to false)
18+
1619
* `--start-app` - Whether to start the application before generating the schema (defaults to true)
1720
1821
* `--vendor-extensions` - Whether to include open_api_spex OpenAPI vendor extensions

lib/open_api_spex/export_spec.ex

+24-5
Original file line numberDiff line numberDiff line change
@@ -7,16 +7,27 @@ defmodule OpenApiSpex.ExportSpec do
77
defmodule Options do
88
@moduledoc false
99

10-
defstruct filename: nil, spec: nil, pretty: false, vendor_extensions: true, quiet: false
10+
defstruct filename: nil,
11+
spec: nil,
12+
pretty: false,
13+
check: false,
14+
vendor_extensions: true,
15+
quiet: false
1116
end
1217

1318
def call(argv, encode_spec, default_filename) do
1419
opts = parse_options(argv, default_filename)
1520

16-
opts
17-
|> generate_spec()
18-
|> encode_spec.(opts)
19-
|> write_spec(opts)
21+
encoded_spec =
22+
opts
23+
|> generate_spec()
24+
|> encode_spec.(opts)
25+
26+
if opts.check do
27+
check_spec(encoded_spec, opts)
28+
else
29+
write_spec(encoded_spec, opts)
30+
end
2031
end
2132

2233
defp generate_spec(%{spec: spec, vendor_extensions: vendor_extensions}) do
@@ -45,6 +56,7 @@ defmodule OpenApiSpex.ExportSpec do
4556
spec: :string,
4657
endpoint: :string,
4758
pretty: :boolean,
59+
check: :boolean,
4860
vendor_extensions: :boolean,
4961
quiet: :boolean
5062
]
@@ -56,11 +68,18 @@ defmodule OpenApiSpex.ExportSpec do
5668
filename: args |> List.first() || default_filename,
5769
spec: find_spec(opts),
5870
pretty: Keyword.get(opts, :pretty, false),
71+
check: Keyword.get(opts, :check, false),
5972
vendor_extensions: Keyword.get(opts, :vendor_extensions, true),
6073
quiet: Keyword.get(opts, :quiet, false)
6174
}
6275
end
6376

77+
defp check_spec(content, opts) do
78+
unless content == File.read!(opts.filename) do
79+
Mix.raise("The OpenAPI spec file does not match the generated spec:\n\n#{content}")
80+
end
81+
end
82+
6483
defp write_spec(content, opts) do
6584
case Path.dirname(opts.filename) do
6685
"." -> true

0 commit comments

Comments
 (0)