mirror of
https://github.com/Smaug123/WoofWare.Myriad
synced 2025-10-05 20:18:43 +00:00
Compare commits
18 Commits
WoofWare.M
...
WoofWare.M
Author | SHA1 | Date | |
---|---|---|---|
|
4236b26189 | ||
|
4fe4e3f277 | ||
|
9473a080ff | ||
|
e6867572b7 | ||
|
5a92d86ad1 | ||
|
fbfd7131f3 | ||
|
ca72b07c33 | ||
|
cdaa46fe00 | ||
|
c70c68b15b | ||
|
17cbaf4a85 | ||
|
e48c5209c7 | ||
|
cc8e3205b1 | ||
|
34cb74dc7b | ||
|
099d14b0b1 | ||
|
96908a5fa6 | ||
|
3e39e187df | ||
|
9f4245341c | ||
|
de58f5ed1f |
@@ -3,13 +3,13 @@
|
|||||||
"isRoot": true,
|
"isRoot": true,
|
||||||
"tools": {
|
"tools": {
|
||||||
"fantomas": {
|
"fantomas": {
|
||||||
"version": "7.0.1",
|
"version": "7.0.2",
|
||||||
"commands": [
|
"commands": [
|
||||||
"fantomas"
|
"fantomas"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"fsharp-analyzers": {
|
"fsharp-analyzers": {
|
||||||
"version": "0.30.0",
|
"version": "0.31.0",
|
||||||
"commands": [
|
"commands": [
|
||||||
"fsharp-analyzers"
|
"fsharp-analyzers"
|
||||||
]
|
]
|
||||||
|
4
.github/workflows/dotnet.yaml
vendored
4
.github/workflows/dotnet.yaml
vendored
@@ -254,7 +254,7 @@ jobs:
|
|||||||
name: nuget-package-attribute
|
name: nuget-package-attribute
|
||||||
path: packed
|
path: packed
|
||||||
- name: Attest Build Provenance
|
- name: Attest Build Provenance
|
||||||
uses: actions/attest-build-provenance@c074443f1aee8d4aeeae555aebba3282517141b2 # v2.2.3
|
uses: actions/attest-build-provenance@e8998f949152b193b063cb0ec769d69d929409be # v2.4.0
|
||||||
with:
|
with:
|
||||||
subject-path: "packed/*.nupkg"
|
subject-path: "packed/*.nupkg"
|
||||||
|
|
||||||
@@ -273,7 +273,7 @@ jobs:
|
|||||||
name: nuget-package-plugin
|
name: nuget-package-plugin
|
||||||
path: packed
|
path: packed
|
||||||
- name: Attest Build Provenance
|
- name: Attest Build Provenance
|
||||||
uses: actions/attest-build-provenance@c074443f1aee8d4aeeae555aebba3282517141b2 # v2.2.3
|
uses: actions/attest-build-provenance@e8998f949152b193b063cb0ec769d69d929409be # v2.4.0
|
||||||
with:
|
with:
|
||||||
subject-path: "packed/*.nupkg"
|
subject-path: "packed/*.nupkg"
|
||||||
|
|
||||||
|
3
.github/workflows/flake_update.yaml
vendored
3
.github/workflows/flake_update.yaml
vendored
@@ -41,11 +41,12 @@ jobs:
|
|||||||
private-key: ${{ secrets.APP_PRIVATE_KEY }}
|
private-key: ${{ secrets.APP_PRIVATE_KEY }}
|
||||||
|
|
||||||
- name: Raise pull request
|
- name: Raise pull request
|
||||||
uses: Smaug123/commit-action@cc25e6d80a796c49669dda4a0aa36c54c573983d
|
uses: Smaug123/commit-action@d34807f26cb52c7a05bbd80efe9f964cdf29bc87
|
||||||
id: cpr
|
id: cpr
|
||||||
with:
|
with:
|
||||||
bearer-token: ${{ steps.generate-token.outputs.token }}
|
bearer-token: ${{ steps.generate-token.outputs.token }}
|
||||||
pr-title: "Upgrade Nix flake and deps"
|
pr-title: "Upgrade Nix flake and deps"
|
||||||
|
branch-name: "auto-pr"
|
||||||
|
|
||||||
- name: Enable Pull Request Automerge
|
- name: Enable Pull Request Automerge
|
||||||
if: ${{ steps.cpr.outputs.pull-request-number }}
|
if: ${{ steps.cpr.outputs.pull-request-number }}
|
||||||
|
@@ -1,5 +1,10 @@
|
|||||||
Notable changes are recorded here.
|
Notable changes are recorded here.
|
||||||
|
|
||||||
|
# WoofWare.Myriad.Plugins 8.0.3
|
||||||
|
|
||||||
|
The RestEase-style HTTP client generator now automatically adds the `application/json` content type header to requests which are POSTing a body that is known to be JSON-serialised.
|
||||||
|
You can override this by setting the `[<RestEase.Header ("Content-Type", "desired content type")>]` header manually on any affected member.
|
||||||
|
|
||||||
# WoofWare.Myriad.Plugins 7.0.1
|
# WoofWare.Myriad.Plugins 7.0.1
|
||||||
|
|
||||||
All generators should now be compatible with `<Nullable>enable</Nullable>`.
|
All generators should now be compatible with `<Nullable>enable</Nullable>`.
|
||||||
|
@@ -375,7 +375,9 @@ module PureGymApi =
|
|||||||
match node with
|
match node with
|
||||||
| None -> "null"
|
| None -> "null"
|
||||||
| Some node -> node.ToJsonString ()
|
| Some node -> node.ToJsonString ()
|
||||||
)
|
),
|
||||||
|
null,
|
||||||
|
"application/json"
|
||||||
)
|
)
|
||||||
|
|
||||||
do httpMessage.Content <- queryParams
|
do httpMessage.Content <- queryParams
|
||||||
@@ -667,7 +669,9 @@ module PureGymApi =
|
|||||||
|
|
||||||
let queryParams =
|
let queryParams =
|
||||||
new System.Net.Http.StringContent (
|
new System.Net.Http.StringContent (
|
||||||
user |> PureGym.Member.toJsonNode |> (fun node -> node.ToJsonString ())
|
user |> PureGym.Member.toJsonNode |> (fun node -> node.ToJsonString ()),
|
||||||
|
null,
|
||||||
|
"application/json"
|
||||||
)
|
)
|
||||||
|
|
||||||
do httpMessage.Content <- queryParams
|
do httpMessage.Content <- queryParams
|
||||||
@@ -710,7 +714,9 @@ module PureGymApi =
|
|||||||
)
|
)
|
||||||
| field -> field)
|
| field -> field)
|
||||||
)
|
)
|
||||||
|> (fun node -> node.ToJsonString ())
|
|> (fun node -> node.ToJsonString ()),
|
||||||
|
null,
|
||||||
|
"application/json"
|
||||||
)
|
)
|
||||||
|
|
||||||
do httpMessage.Content <- queryParams
|
do httpMessage.Content <- queryParams
|
||||||
@@ -753,7 +759,9 @@ module PureGymApi =
|
|||||||
)
|
)
|
||||||
| field -> field)
|
| field -> field)
|
||||||
)
|
)
|
||||||
|> (fun node -> node.ToJsonString ())
|
|> (fun node -> node.ToJsonString ()),
|
||||||
|
null,
|
||||||
|
"application/json"
|
||||||
)
|
)
|
||||||
|
|
||||||
do httpMessage.Content <- queryParams
|
do httpMessage.Content <- queryParams
|
||||||
@@ -1801,3 +1809,174 @@ module ApiWithHeaders2 =
|
|||||||
}
|
}
|
||||||
|> (fun a -> Async.StartAsTask (a, ?cancellationToken = ct))
|
|> (fun a -> Async.StartAsTask (a, ?cancellationToken = ct))
|
||||||
}
|
}
|
||||||
|
namespace PureGym
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Threading
|
||||||
|
open System.Threading.Tasks
|
||||||
|
open System.IO
|
||||||
|
open System.Net
|
||||||
|
open System.Net.Http
|
||||||
|
open RestEase
|
||||||
|
|
||||||
|
/// Module for constructing a REST client.
|
||||||
|
[<CompilationRepresentation(CompilationRepresentationFlags.ModuleSuffix) ; RequireQualifiedAccess>]
|
||||||
|
module ClientWithJsonBody =
|
||||||
|
/// Create a REST client.
|
||||||
|
let make (client : System.Net.Http.HttpClient) : IClientWithJsonBody =
|
||||||
|
{ new IClientWithJsonBody with
|
||||||
|
member _.GetPathParam (parameter : string, mem : PureGym.Member, ct : CancellationToken option) =
|
||||||
|
async {
|
||||||
|
let! ct = Async.CancellationToken
|
||||||
|
|
||||||
|
let uri =
|
||||||
|
System.Uri (
|
||||||
|
(match client.BaseAddress with
|
||||||
|
| null ->
|
||||||
|
raise (
|
||||||
|
System.ArgumentNullException (
|
||||||
|
nameof (client.BaseAddress),
|
||||||
|
"No base address was supplied on the type, and no BaseAddress was on the HttpClient."
|
||||||
|
)
|
||||||
|
)
|
||||||
|
| v -> v),
|
||||||
|
System.Uri (
|
||||||
|
"endpoint/{param}"
|
||||||
|
.Replace ("{param}", parameter.ToString () |> System.Uri.EscapeDataString),
|
||||||
|
System.UriKind.Relative
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
let httpMessage =
|
||||||
|
new System.Net.Http.HttpRequestMessage (
|
||||||
|
Method = System.Net.Http.HttpMethod.Post,
|
||||||
|
RequestUri = uri
|
||||||
|
)
|
||||||
|
|
||||||
|
let queryParams =
|
||||||
|
new System.Net.Http.StringContent (
|
||||||
|
mem |> PureGym.Member.toJsonNode |> (fun node -> node.ToJsonString ()),
|
||||||
|
null,
|
||||||
|
"application/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
do httpMessage.Content <- queryParams
|
||||||
|
let! response = client.SendAsync (httpMessage, ct) |> Async.AwaitTask
|
||||||
|
let response = response.EnsureSuccessStatusCode ()
|
||||||
|
let! responseString = response.Content.ReadAsStringAsync ct |> Async.AwaitTask
|
||||||
|
return responseString
|
||||||
|
}
|
||||||
|
|> (fun a -> Async.StartAsTask (a, ?cancellationToken = ct))
|
||||||
|
}
|
||||||
|
namespace PureGym
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Threading
|
||||||
|
open System.Threading.Tasks
|
||||||
|
open System.IO
|
||||||
|
open System.Net
|
||||||
|
open System.Net.Http
|
||||||
|
open RestEase
|
||||||
|
|
||||||
|
/// Module for constructing a REST client.
|
||||||
|
[<CompilationRepresentation(CompilationRepresentationFlags.ModuleSuffix) ; RequireQualifiedAccess>]
|
||||||
|
module ClientWithJsonBodyOverridden =
|
||||||
|
/// Create a REST client.
|
||||||
|
let make (client : System.Net.Http.HttpClient) : IClientWithJsonBodyOverridden =
|
||||||
|
{ new IClientWithJsonBodyOverridden with
|
||||||
|
member _.GetPathParam (parameter : string, mem : PureGym.Member, ct : CancellationToken option) =
|
||||||
|
async {
|
||||||
|
let! ct = Async.CancellationToken
|
||||||
|
|
||||||
|
let uri =
|
||||||
|
System.Uri (
|
||||||
|
(match client.BaseAddress with
|
||||||
|
| null ->
|
||||||
|
raise (
|
||||||
|
System.ArgumentNullException (
|
||||||
|
nameof (client.BaseAddress),
|
||||||
|
"No base address was supplied on the type, and no BaseAddress was on the HttpClient."
|
||||||
|
)
|
||||||
|
)
|
||||||
|
| v -> v),
|
||||||
|
System.Uri (
|
||||||
|
"endpoint/{param}"
|
||||||
|
.Replace ("{param}", parameter.ToString () |> System.Uri.EscapeDataString),
|
||||||
|
System.UriKind.Relative
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
let httpMessage =
|
||||||
|
new System.Net.Http.HttpRequestMessage (
|
||||||
|
Method = System.Net.Http.HttpMethod.Post,
|
||||||
|
RequestUri = uri
|
||||||
|
)
|
||||||
|
|
||||||
|
let queryParams =
|
||||||
|
new System.Net.Http.StringContent (
|
||||||
|
mem |> PureGym.Member.toJsonNode |> (fun node -> node.ToJsonString ()),
|
||||||
|
null,
|
||||||
|
"application/ecmascript"
|
||||||
|
)
|
||||||
|
|
||||||
|
do httpMessage.Content <- queryParams
|
||||||
|
let! response = client.SendAsync (httpMessage, ct) |> Async.AwaitTask
|
||||||
|
let response = response.EnsureSuccessStatusCode ()
|
||||||
|
let! responseString = response.Content.ReadAsStringAsync ct |> Async.AwaitTask
|
||||||
|
return responseString
|
||||||
|
}
|
||||||
|
|> (fun a -> Async.StartAsTask (a, ?cancellationToken = ct))
|
||||||
|
}
|
||||||
|
namespace PureGym
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Threading
|
||||||
|
open System.Threading.Tasks
|
||||||
|
open System.IO
|
||||||
|
open System.Net
|
||||||
|
open System.Net.Http
|
||||||
|
open RestEase
|
||||||
|
|
||||||
|
/// Module for constructing a REST client.
|
||||||
|
[<CompilationRepresentation(CompilationRepresentationFlags.ModuleSuffix) ; RequireQualifiedAccess>]
|
||||||
|
module ClientWithStringBody =
|
||||||
|
/// Create a REST client.
|
||||||
|
let make (client : System.Net.Http.HttpClient) : IClientWithStringBody =
|
||||||
|
{ new IClientWithStringBody with
|
||||||
|
member _.GetPathParam (parameter : string, mem : string, ct : CancellationToken option) =
|
||||||
|
async {
|
||||||
|
let! ct = Async.CancellationToken
|
||||||
|
|
||||||
|
let uri =
|
||||||
|
System.Uri (
|
||||||
|
(match client.BaseAddress with
|
||||||
|
| null ->
|
||||||
|
raise (
|
||||||
|
System.ArgumentNullException (
|
||||||
|
nameof (client.BaseAddress),
|
||||||
|
"No base address was supplied on the type, and no BaseAddress was on the HttpClient."
|
||||||
|
)
|
||||||
|
)
|
||||||
|
| v -> v),
|
||||||
|
System.Uri (
|
||||||
|
"endpoint/{param}"
|
||||||
|
.Replace ("{param}", parameter.ToString () |> System.Uri.EscapeDataString),
|
||||||
|
System.UriKind.Relative
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
let httpMessage =
|
||||||
|
new System.Net.Http.HttpRequestMessage (
|
||||||
|
Method = System.Net.Http.HttpMethod.Post,
|
||||||
|
RequestUri = uri
|
||||||
|
)
|
||||||
|
|
||||||
|
let queryParams = new System.Net.Http.StringContent (mem)
|
||||||
|
do httpMessage.Content <- queryParams
|
||||||
|
let! response = client.SendAsync (httpMessage, ct) |> Async.AwaitTask
|
||||||
|
let response = response.EnsureSuccessStatusCode ()
|
||||||
|
let! responseString = response.Content.ReadAsStringAsync ct |> Async.AwaitTask
|
||||||
|
return responseString
|
||||||
|
}
|
||||||
|
|> (fun a -> Async.StartAsTask (a, ?cancellationToken = ct))
|
||||||
|
}
|
||||||
|
@@ -203,3 +203,35 @@ type IApiWithHeaders2 =
|
|||||||
[<Get "endpoint/{param}">]
|
[<Get "endpoint/{param}">]
|
||||||
abstract GetPathParam :
|
abstract GetPathParam :
|
||||||
[<WoofWare.Myriad.Plugins.RestEase.Path "param">] parameter : string * ?ct : CancellationToken -> Task<string>
|
[<WoofWare.Myriad.Plugins.RestEase.Path "param">] parameter : string * ?ct : CancellationToken -> Task<string>
|
||||||
|
|
||||||
|
[<WoofWare.Myriad.Plugins.HttpClient>]
|
||||||
|
type IClientWithJsonBody =
|
||||||
|
// As a POST request of a JSON-serialised body, we automatically set Content-Type: application/json.
|
||||||
|
[<Post "endpoint/{param}">]
|
||||||
|
abstract GetPathParam :
|
||||||
|
[<RestEase.Path "param">] parameter : string *
|
||||||
|
[<WoofWare.Myriad.Plugins.RestEase.Body>] mem : PureGym.Member *
|
||||||
|
?ct : CancellationToken ->
|
||||||
|
Task<string>
|
||||||
|
|
||||||
|
[<WoofWare.Myriad.Plugins.HttpClient>]
|
||||||
|
type IClientWithJsonBodyOverridden =
|
||||||
|
// As a POST request of a JSON-serialised body, we *would* automatically set Content-Type: application/json,
|
||||||
|
// but this method has overridden it.
|
||||||
|
[<Post "endpoint/{param}">]
|
||||||
|
[<Header("Content-Type", "application/ecmascript")>]
|
||||||
|
abstract GetPathParam :
|
||||||
|
[<RestEase.Path "param">] parameter : string *
|
||||||
|
[<WoofWare.Myriad.Plugins.RestEase.Body>] mem : PureGym.Member *
|
||||||
|
?ct : CancellationToken ->
|
||||||
|
Task<string>
|
||||||
|
|
||||||
|
[<WoofWare.Myriad.Plugins.HttpClient>]
|
||||||
|
type IClientWithStringBody =
|
||||||
|
// As a POST request of a bare string body, we don't override the Content-Type.
|
||||||
|
[<Post "endpoint/{param}">]
|
||||||
|
abstract GetPathParam :
|
||||||
|
[<RestEase.Path "param">] parameter : string *
|
||||||
|
[<WoofWare.Myriad.Plugins.RestEase.Body>] mem : string *
|
||||||
|
?ct : CancellationToken ->
|
||||||
|
Task<string>
|
||||||
|
@@ -647,3 +647,7 @@ I'm hopefully going to get round to writing a more powerful source generation sy
|
|||||||
You should probably add these files to your [fantomasignore](https://github.com/fsprojects/fantomas/blob/a999b77ca5a024fbc3409955faac797e29b39d27/docs/docs/end-users/IgnoreFiles.md)
|
You should probably add these files to your [fantomasignore](https://github.com/fsprojects/fantomas/blob/a999b77ca5a024fbc3409955faac797e29b39d27/docs/docs/end-users/IgnoreFiles.md)
|
||||||
if you use Fantomas to format your repo;
|
if you use Fantomas to format your repo;
|
||||||
the alternative is to manually reformat every time Myriad changes the generated files.
|
the alternative is to manually reformat every time Myriad changes the generated files.
|
||||||
|
|
||||||
|
# Licence
|
||||||
|
|
||||||
|
The code is MIT-licenced, except for the Swagger API examples in WoofWare.Myriad.Plugins.Test, which are [CC-BY 4.0](https://creativecommons.org/licenses/by/4.0/), copyright 2023 by the OpenAPI Initiative, and obtained from https://learn.openapis.org/examples/ with no changes made.
|
||||||
|
@@ -17,8 +17,8 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="ApiSurface" Version="4.1.20" />
|
<PackageReference Include="ApiSurface" Version="4.1.21" />
|
||||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0"/>
|
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.14.1"/>
|
||||||
<PackageReference Include="NUnit" Version="4.3.2"/>
|
<PackageReference Include="NUnit" Version="4.3.2"/>
|
||||||
<PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/>
|
<PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
22
WoofWare.Myriad.Plugins.Test/Assembly.fs
Normal file
22
WoofWare.Myriad.Plugins.Test/Assembly.fs
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
namespace WoofWare.Myriad.Plugins.Test
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.IO
|
||||||
|
open System.Reflection
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module Assembly =
|
||||||
|
|
||||||
|
let getEmbeddedResource (assembly : Assembly) (name : string) : string =
|
||||||
|
let names = assembly.GetManifestResourceNames ()
|
||||||
|
|
||||||
|
let names =
|
||||||
|
names |> Seq.filter (fun s -> s.EndsWith (name, StringComparison.Ordinal))
|
||||||
|
|
||||||
|
use s =
|
||||||
|
names
|
||||||
|
|> Seq.exactlyOne
|
||||||
|
|> assembly.GetManifestResourceStream
|
||||||
|
|> fun s -> new StreamReader (s)
|
||||||
|
|
||||||
|
s.ReadToEnd ()
|
@@ -3,10 +3,12 @@ namespace WoofWare.Myriad.Plugins.Test
|
|||||||
open System
|
open System
|
||||||
open System.Net
|
open System.Net
|
||||||
open System.Net.Http
|
open System.Net.Http
|
||||||
|
open System.Text.Json.Nodes
|
||||||
open System.Threading
|
open System.Threading
|
||||||
open NUnit.Framework
|
open NUnit.Framework
|
||||||
open FsUnitTyped
|
open FsUnitTyped
|
||||||
open PureGym
|
open PureGym
|
||||||
|
open WoofWare.Expect
|
||||||
|
|
||||||
[<TestFixture>]
|
[<TestFixture>]
|
||||||
module TestVariableHeader =
|
module TestVariableHeader =
|
||||||
@@ -50,15 +52,17 @@ module TestVariableHeader =
|
|||||||
someHeaderCount.Value |> shouldEqual 10
|
someHeaderCount.Value |> shouldEqual 10
|
||||||
someOtherHeaderCount.Value |> shouldEqual -100
|
someOtherHeaderCount.Value |> shouldEqual -100
|
||||||
|
|
||||||
api.GetPathParam("param").Result.Split "\n"
|
expect {
|
||||||
|> Array.sort
|
snapshotJson
|
||||||
|> shouldEqual
|
@"[
|
||||||
[|
|
""Authorization: -99"",
|
||||||
"Authorization: -99"
|
""Header-Name: Header-Value"",
|
||||||
"Header-Name: Header-Value"
|
""Something-Else: val"",
|
||||||
"Something-Else: val"
|
""X-Foo: 11""
|
||||||
"X-Foo: 11"
|
]"
|
||||||
|]
|
|
||||||
|
return api.GetPathParam("param").Result.Split "\n" |> Array.sort
|
||||||
|
}
|
||||||
|
|
||||||
someHeaderCount.Value |> shouldEqual 11
|
someHeaderCount.Value |> shouldEqual 11
|
||||||
someOtherHeaderCount.Value |> shouldEqual -99
|
someOtherHeaderCount.Value |> shouldEqual -99
|
||||||
@@ -102,25 +106,156 @@ module TestVariableHeader =
|
|||||||
someHeaderCount.Value |> shouldEqual 10
|
someHeaderCount.Value |> shouldEqual 10
|
||||||
someOtherHeaderCount.Value |> shouldEqual -100
|
someOtherHeaderCount.Value |> shouldEqual -100
|
||||||
|
|
||||||
api.GetPathParam("param").Result.Split "\n"
|
expect {
|
||||||
|> Array.sort
|
snapshotJson
|
||||||
|> shouldEqual
|
@"[
|
||||||
[|
|
""Authorization: -99"",
|
||||||
"Authorization: -99"
|
""Header-Name: Header-Value"",
|
||||||
"Header-Name: Header-Value"
|
""Something-Else: val"",
|
||||||
"Something-Else: val"
|
""X-Foo: 11""
|
||||||
"X-Foo: 11"
|
]"
|
||||||
|]
|
|
||||||
|
|
||||||
api.GetPathParam("param").Result.Split "\n"
|
return api.GetPathParam("param").Result.Split "\n" |> Array.sort
|
||||||
|> Array.sort
|
}
|
||||||
|> shouldEqual
|
|
||||||
[|
|
expect {
|
||||||
"Authorization: -98"
|
snapshotJson
|
||||||
"Header-Name: Header-Value"
|
@"[
|
||||||
"Something-Else: val"
|
""Authorization: -98"",
|
||||||
"X-Foo: 12"
|
""Header-Name: Header-Value"",
|
||||||
|]
|
""Something-Else: val"",
|
||||||
|
""X-Foo: 12""
|
||||||
|
]"
|
||||||
|
|
||||||
|
return api.GetPathParam("param").Result.Split "\n" |> Array.sort
|
||||||
|
}
|
||||||
|
|
||||||
someHeaderCount.Value |> shouldEqual 12
|
someHeaderCount.Value |> shouldEqual 12
|
||||||
someOtherHeaderCount.Value |> shouldEqual -98
|
someOtherHeaderCount.Value |> shouldEqual -98
|
||||||
|
|
||||||
|
let pureGymMember =
|
||||||
|
{
|
||||||
|
Id = 3
|
||||||
|
CompoundMemberId = "compound"
|
||||||
|
FirstName = "Patrick"
|
||||||
|
LastName = "Stevens"
|
||||||
|
HomeGymId = 1223
|
||||||
|
HomeGymName = "Arnie's Temple o' Gainz"
|
||||||
|
EmailAddress = "patrick@home"
|
||||||
|
GymAccessPin = "1234"
|
||||||
|
DateOfBirth = DateOnly (1992, 03, 04)
|
||||||
|
MobileNumber = "number"
|
||||||
|
Postcode = "postcode"
|
||||||
|
MembershipName = "member"
|
||||||
|
MembershipLevel = 9999
|
||||||
|
SuspendedReason = -1
|
||||||
|
MemberStatus = 100
|
||||||
|
}
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
let ``Content-Type header is automatically inserted`` () =
|
||||||
|
let proc (message : HttpRequestMessage) : HttpResponseMessage Async =
|
||||||
|
async {
|
||||||
|
message.Method |> shouldEqual HttpMethod.Post
|
||||||
|
|
||||||
|
message.RequestUri.ToString ()
|
||||||
|
|> shouldEqual "https://example.com/endpoint/param"
|
||||||
|
|
||||||
|
let headers =
|
||||||
|
[
|
||||||
|
for h in message.Content.Headers do
|
||||||
|
yield $"%s{h.Key}: %s{Seq.exactlyOne h.Value}"
|
||||||
|
]
|
||||||
|
|> String.concat "\n"
|
||||||
|
|
||||||
|
let! ct = Async.CancellationToken
|
||||||
|
let! content = message.Content.ReadAsStringAsync ct |> Async.AwaitTask
|
||||||
|
content |> JsonNode.Parse |> Member.jsonParse |> shouldEqual pureGymMember
|
||||||
|
|
||||||
|
let content = new StringContent (headers)
|
||||||
|
let resp = new HttpResponseMessage (HttpStatusCode.OK)
|
||||||
|
resp.Content <- content
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
|
use client = HttpClientMock.make (Uri "https://example.com") proc
|
||||||
|
|
||||||
|
let api = ClientWithJsonBody.make client
|
||||||
|
let result = api.GetPathParam ("param", pureGymMember) |> _.Result
|
||||||
|
|
||||||
|
expect {
|
||||||
|
snapshot @"Content-Type: application/json; charset=utf-8"
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
let ``Content-Type header is respected if overridden`` () =
|
||||||
|
let proc (message : HttpRequestMessage) : HttpResponseMessage Async =
|
||||||
|
async {
|
||||||
|
message.Method |> shouldEqual HttpMethod.Post
|
||||||
|
|
||||||
|
message.RequestUri.ToString ()
|
||||||
|
|> shouldEqual "https://example.com/endpoint/param"
|
||||||
|
|
||||||
|
let headers =
|
||||||
|
[
|
||||||
|
for h in message.Content.Headers do
|
||||||
|
yield $"%s{h.Key}: %s{Seq.exactlyOne h.Value}"
|
||||||
|
]
|
||||||
|
|> String.concat "\n"
|
||||||
|
|
||||||
|
let! ct = Async.CancellationToken
|
||||||
|
let! content = message.Content.ReadAsStringAsync ct |> Async.AwaitTask
|
||||||
|
content |> JsonNode.Parse |> Member.jsonParse |> shouldEqual pureGymMember
|
||||||
|
|
||||||
|
let content = new StringContent (headers)
|
||||||
|
let resp = new HttpResponseMessage (HttpStatusCode.OK)
|
||||||
|
resp.Content <- content
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
|
use client = HttpClientMock.make (Uri "https://example.com") proc
|
||||||
|
|
||||||
|
let api = ClientWithJsonBodyOverridden.make client
|
||||||
|
let result = api.GetPathParam ("param", pureGymMember) |> _.Result
|
||||||
|
|
||||||
|
expect {
|
||||||
|
snapshot @"Content-Type: application/ecmascript; charset=utf-8"
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
let ``Content-Type header is the default for strings`` () =
|
||||||
|
let proc (message : HttpRequestMessage) : HttpResponseMessage Async =
|
||||||
|
async {
|
||||||
|
message.Method |> shouldEqual HttpMethod.Post
|
||||||
|
|
||||||
|
message.RequestUri.ToString ()
|
||||||
|
|> shouldEqual "https://example.com/endpoint/param"
|
||||||
|
|
||||||
|
let headers =
|
||||||
|
[
|
||||||
|
for h in message.Content.Headers do
|
||||||
|
yield $"%s{h.Key}: %s{Seq.exactlyOne h.Value}"
|
||||||
|
]
|
||||||
|
|> String.concat "\n"
|
||||||
|
|
||||||
|
let! ct = Async.CancellationToken
|
||||||
|
let! content = message.Content.ReadAsStringAsync ct |> Async.AwaitTask
|
||||||
|
content |> shouldEqual "hello!"
|
||||||
|
|
||||||
|
let content = new StringContent (headers)
|
||||||
|
let resp = new HttpResponseMessage (HttpStatusCode.OK)
|
||||||
|
resp.Content <- content
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
|
use client = HttpClientMock.make (Uri "https://example.com") proc
|
||||||
|
|
||||||
|
let api = ClientWithStringBody.make client
|
||||||
|
let result = api.GetPathParam ("param", "hello!") |> _.Result
|
||||||
|
|
||||||
|
expect {
|
||||||
|
snapshot @"Content-Type: text/plain; charset=utf-8"
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
3503
WoofWare.Myriad.Plugins.Test/TestSwagger/TestOpenApi3Parse.fs
Normal file
3503
WoofWare.Myriad.Plugins.Test/TestSwagger/TestOpenApi3Parse.fs
Normal file
File diff suppressed because it is too large
Load Diff
@@ -3,7 +3,7 @@ namespace WoofWare.Myriad.Plugins.Test
|
|||||||
open System.Text.Json.Nodes
|
open System.Text.Json.Nodes
|
||||||
open NUnit.Framework
|
open NUnit.Framework
|
||||||
open FsUnitTyped
|
open FsUnitTyped
|
||||||
open WoofWare.Myriad.Plugins
|
open WoofWare.Myriad.Plugins.SwaggerV2
|
||||||
|
|
||||||
[<TestFixture>]
|
[<TestFixture>]
|
||||||
module TestSwaggerParse =
|
module TestSwaggerParse =
|
||||||
|
192
WoofWare.Myriad.Plugins.Test/TestSwagger/api-with-examples.json
Normal file
192
WoofWare.Myriad.Plugins.Test/TestSwagger/api-with-examples.json
Normal file
@@ -0,0 +1,192 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.0.0",
|
||||||
|
"info": {
|
||||||
|
"title": "Simple API overview",
|
||||||
|
"version": "2.0.0"
|
||||||
|
},
|
||||||
|
"paths": {
|
||||||
|
"/": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "listVersionsv2",
|
||||||
|
"summary": "List API versions",
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "200 response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"examples": {
|
||||||
|
"foo": {
|
||||||
|
"value": {
|
||||||
|
"versions": [
|
||||||
|
{
|
||||||
|
"status": "CURRENT",
|
||||||
|
"updated": "2011-01-21T11:33:21Z",
|
||||||
|
"id": "v2.0",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"href": "http://127.0.0.1:8774/v2/",
|
||||||
|
"rel": "self"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"status": "EXPERIMENTAL",
|
||||||
|
"updated": "2013-07-23T11:33:21Z",
|
||||||
|
"id": "v3.0",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"href": "http://127.0.0.1:8774/v3/",
|
||||||
|
"rel": "self"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"300": {
|
||||||
|
"description": "300 response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"examples": {
|
||||||
|
"foo": {
|
||||||
|
"value": {
|
||||||
|
"versions": [
|
||||||
|
{
|
||||||
|
"status": "CURRENT",
|
||||||
|
"updated": "2011-01-21T11:33:21Z",
|
||||||
|
"id": "v2.0",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"href": "http://127.0.0.1:8774/v2/",
|
||||||
|
"rel": "self"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"status": "EXPERIMENTAL",
|
||||||
|
"updated": "2013-07-23T11:33:21Z",
|
||||||
|
"id": "v3.0",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"href": "http://127.0.0.1:8774/v3/",
|
||||||
|
"rel": "self"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/v2": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "getVersionDetailsv2",
|
||||||
|
"summary": "Show API version details",
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "200 response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"examples": {
|
||||||
|
"foo": {
|
||||||
|
"value": {
|
||||||
|
"version": {
|
||||||
|
"status": "CURRENT",
|
||||||
|
"updated": "2011-01-21T11:33:21Z",
|
||||||
|
"media-types": [
|
||||||
|
{
|
||||||
|
"base": "application/xml",
|
||||||
|
"type": "application/vnd.openstack.compute+xml;version=2"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"base": "application/json",
|
||||||
|
"type": "application/vnd.openstack.compute+json;version=2"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"id": "v2.0",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"href": "http://127.0.0.1:8774/v2/",
|
||||||
|
"rel": "self"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"href": "http://docs.openstack.org/api/openstack-compute/2/os-compute-devguide-2.pdf",
|
||||||
|
"type": "application/pdf",
|
||||||
|
"rel": "describedby"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"href": "http://docs.openstack.org/api/openstack-compute/2/wadl/os-compute-2.wadl",
|
||||||
|
"type": "application/vnd.sun.wadl+xml",
|
||||||
|
"rel": "describedby"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"href": "http://docs.openstack.org/api/openstack-compute/2/wadl/os-compute-2.wadl",
|
||||||
|
"type": "application/vnd.sun.wadl+xml",
|
||||||
|
"rel": "describedby"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"203": {
|
||||||
|
"description": "203 response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"examples": {
|
||||||
|
"foo": {
|
||||||
|
"value": {
|
||||||
|
"version": {
|
||||||
|
"status": "CURRENT",
|
||||||
|
"updated": "2011-01-21T11:33:21Z",
|
||||||
|
"media-types": [
|
||||||
|
{
|
||||||
|
"base": "application/xml",
|
||||||
|
"type": "application/vnd.openstack.compute+xml;version=2"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"base": "application/json",
|
||||||
|
"type": "application/vnd.openstack.compute+json;version=2"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"id": "v2.0",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"href": "http://23.253.228.211:8774/v2/",
|
||||||
|
"rel": "self"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"href": "http://docs.openstack.org/api/openstack-compute/2/os-compute-devguide-2.pdf",
|
||||||
|
"type": "application/pdf",
|
||||||
|
"rel": "describedby"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"href": "http://docs.openstack.org/api/openstack-compute/2/wadl/os-compute-2.wadl",
|
||||||
|
"type": "application/vnd.sun.wadl+xml",
|
||||||
|
"rel": "describedby"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,82 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.0.0",
|
||||||
|
"info": {
|
||||||
|
"title": "Callback Example",
|
||||||
|
"version": "1.0.0"
|
||||||
|
},
|
||||||
|
"paths": {
|
||||||
|
"/streams": {
|
||||||
|
"post": {
|
||||||
|
"description": "subscribes a client to receive out-of-band data",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "callbackUrl",
|
||||||
|
"in": "query",
|
||||||
|
"required": true,
|
||||||
|
"description": "the location where data will be sent. Must be network accessible\nby the source server\n",
|
||||||
|
"schema": {
|
||||||
|
"type": "string",
|
||||||
|
"format": "uri",
|
||||||
|
"example": "https://tonys-server.com"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"201": {
|
||||||
|
"description": "subscription successfully created",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"description": "subscription information",
|
||||||
|
"required": ["subscriptionId"],
|
||||||
|
"properties": {
|
||||||
|
"subscriptionId": {
|
||||||
|
"description": "this unique identifier allows management of the subscription",
|
||||||
|
"type": "string",
|
||||||
|
"example": "2531329f-fb09-4ef7-887e-84e648214436"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"callbacks": {
|
||||||
|
"onData": {
|
||||||
|
"{$request.query.callbackUrl}/data": {
|
||||||
|
"post": {
|
||||||
|
"requestBody": {
|
||||||
|
"description": "subscription payload",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"timestamp": {
|
||||||
|
"type": "string",
|
||||||
|
"format": "date-time"
|
||||||
|
},
|
||||||
|
"userData": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"202": {
|
||||||
|
"description": "Your server implementation should return this HTTP status code\nif the data was received successfully\n"
|
||||||
|
},
|
||||||
|
"204": {
|
||||||
|
"description": "Your server should return this HTTP status code if no longer interested\nin further updates\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
319
WoofWare.Myriad.Plugins.Test/TestSwagger/link-example.json
Normal file
319
WoofWare.Myriad.Plugins.Test/TestSwagger/link-example.json
Normal file
@@ -0,0 +1,319 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.0.0",
|
||||||
|
"info": {
|
||||||
|
"title": "Link Example",
|
||||||
|
"version": "1.0.0"
|
||||||
|
},
|
||||||
|
"paths": {
|
||||||
|
"/2.0/users/{username}": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "getUserByName",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "username",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "The User",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/user"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"links": {
|
||||||
|
"userRepositories": {
|
||||||
|
"$ref": "#/components/links/UserRepositories"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/2.0/repositories/{username}": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "getRepositoriesByOwner",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "username",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "repositories owned by the supplied user",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/repository"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"links": {
|
||||||
|
"userRepository": {
|
||||||
|
"$ref": "#/components/links/UserRepository"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/2.0/repositories/{username}/{slug}": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "getRepository",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "username",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "slug",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "The repository",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/repository"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"links": {
|
||||||
|
"repositoryPullRequests": {
|
||||||
|
"$ref": "#/components/links/RepositoryPullRequests"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/2.0/repositories/{username}/{slug}/pullrequests": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "getPullRequestsByRepository",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "username",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "slug",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "state",
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["open", "merged", "declined"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "an array of pull request objects",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/pullrequest"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/2.0/repositories/{username}/{slug}/pullrequests/{pid}": {
|
||||||
|
"get": {
|
||||||
|
"operationId": "getPullRequestsById",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "username",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "slug",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "pid",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "a pull request object",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/pullrequest"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"links": {
|
||||||
|
"pullRequestMerge": {
|
||||||
|
"$ref": "#/components/links/PullRequestMerge"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/2.0/repositories/{username}/{slug}/pullrequests/{pid}/merge": {
|
||||||
|
"post": {
|
||||||
|
"operationId": "mergePullRequest",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "username",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "slug",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "pid",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"204": {
|
||||||
|
"description": "the PR was successfully merged"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"links": {
|
||||||
|
"UserRepositories": {
|
||||||
|
"operationId": "getRepositoriesByOwner",
|
||||||
|
"parameters": {
|
||||||
|
"username": "$response.body#/username"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"UserRepository": {
|
||||||
|
"operationId": "getRepository",
|
||||||
|
"parameters": {
|
||||||
|
"username": "$response.body#/owner/username",
|
||||||
|
"slug": "$response.body#/slug"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"RepositoryPullRequests": {
|
||||||
|
"operationId": "getPullRequestsByRepository",
|
||||||
|
"parameters": {
|
||||||
|
"username": "$response.body#/owner/username",
|
||||||
|
"slug": "$response.body#/slug"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"PullRequestMerge": {
|
||||||
|
"operationId": "mergePullRequest",
|
||||||
|
"parameters": {
|
||||||
|
"username": "$response.body#/author/username",
|
||||||
|
"slug": "$response.body#/repository/slug",
|
||||||
|
"pid": "$response.body#/id"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"schemas": {
|
||||||
|
"user": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"username": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"uuid": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"slug": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"owner": {
|
||||||
|
"$ref": "#/components/schemas/user"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"pullrequest": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"id": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"title": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"$ref": "#/components/schemas/repository"
|
||||||
|
},
|
||||||
|
"author": {
|
||||||
|
"$ref": "#/components/schemas/user"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,28 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.1.0",
|
||||||
|
"info": {
|
||||||
|
"title": "Non-oAuth Scopes example",
|
||||||
|
"version": "1.0.0"
|
||||||
|
},
|
||||||
|
"paths": {
|
||||||
|
"/users": {
|
||||||
|
"get": {
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"bearerAuth": ["read:users", "public"]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"securitySchemes": {
|
||||||
|
"bearerAuth": {
|
||||||
|
"type": "http",
|
||||||
|
"scheme": "bearer",
|
||||||
|
"bearerFormat": "jwt",
|
||||||
|
"description": "note: non-oauth scopes are not defined at the securityScheme level"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
235
WoofWare.Myriad.Plugins.Test/TestSwagger/petstore-expanded.json
Normal file
235
WoofWare.Myriad.Plugins.Test/TestSwagger/petstore-expanded.json
Normal file
@@ -0,0 +1,235 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.0.0",
|
||||||
|
"info": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"title": "Swagger Petstore",
|
||||||
|
"description": "A sample API that uses a petstore as an example to demonstrate features in the OpenAPI 3.0 specification",
|
||||||
|
"termsOfService": "http://swagger.io/terms/",
|
||||||
|
"contact": {
|
||||||
|
"name": "Swagger API Team",
|
||||||
|
"email": "apiteam@swagger.io",
|
||||||
|
"url": "http://swagger.io"
|
||||||
|
},
|
||||||
|
"license": {
|
||||||
|
"name": "Apache 2.0",
|
||||||
|
"url": "https://www.apache.org/licenses/LICENSE-2.0.html"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"servers": [
|
||||||
|
{
|
||||||
|
"url": "https://petstore.swagger.io/v2"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"paths": {
|
||||||
|
"/pets": {
|
||||||
|
"get": {
|
||||||
|
"description": "Returns all pets from the system that the user has access to\nNam sed condimentum est. Maecenas tempor sagittis sapien, nec rhoncus sem sagittis sit amet. Aenean at gravida augue, ac iaculis sem. Curabitur odio lorem, ornare eget elementum nec, cursus id lectus. Duis mi turpis, pulvinar ac eros ac, tincidunt varius justo. In hac habitasse platea dictumst. Integer at adipiscing ante, a sagittis ligula. Aenean pharetra tempor ante molestie imperdiet. Vivamus id aliquam diam. Cras quis velit non tortor eleifend sagittis. Praesent at enim pharetra urna volutpat venenatis eget eget mauris. In eleifend fermentum facilisis. Praesent enim enim, gravida ac sodales sed, placerat id erat. Suspendisse lacus dolor, consectetur non augue vel, vehicula interdum libero. Morbi euismod sagittis libero sed lacinia.\n\nSed tempus felis lobortis leo pulvinar rutrum. Nam mattis velit nisl, eu condimentum ligula luctus nec. Phasellus semper velit eget aliquet faucibus. In a mattis elit. Phasellus vel urna viverra, condimentum lorem id, rhoncus nibh. Ut pellentesque posuere elementum. Sed a varius odio. Morbi rhoncus ligula libero, vel eleifend nunc tristique vitae. Fusce et sem dui. Aenean nec scelerisque tortor. Fusce malesuada accumsan magna vel tempus. Quisque mollis felis eu dolor tristique, sit amet auctor felis gravida. Sed libero lorem, molestie sed nisl in, accumsan tempor nisi. Fusce sollicitudin massa ut lacinia mattis. Sed vel eleifend lorem. Pellentesque vitae felis pretium, pulvinar elit eu, euismod sapien.\n",
|
||||||
|
"operationId": "findPets",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "tags",
|
||||||
|
"in": "query",
|
||||||
|
"description": "tags to filter by",
|
||||||
|
"required": false,
|
||||||
|
"style": "form",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "limit",
|
||||||
|
"in": "query",
|
||||||
|
"description": "maximum number of results to return",
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int32"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "pet response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"post": {
|
||||||
|
"description": "Creates a new pet in the store. Duplicates are allowed",
|
||||||
|
"operationId": "addPet",
|
||||||
|
"requestBody": {
|
||||||
|
"description": "Pet to add to the store",
|
||||||
|
"required": true,
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/NewPet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "pet response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/pets/{id}": {
|
||||||
|
"get": {
|
||||||
|
"description": "Returns a user based on a single ID, if the user does not have access to the pet",
|
||||||
|
"operationId": "find pet by id",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "id",
|
||||||
|
"in": "path",
|
||||||
|
"description": "ID of pet to fetch",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int64"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "pet response",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"delete": {
|
||||||
|
"description": "deletes a single pet based on the ID supplied",
|
||||||
|
"operationId": "deletePet",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "id",
|
||||||
|
"in": "path",
|
||||||
|
"description": "ID of pet to delete",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int64"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"204": {
|
||||||
|
"description": "pet deleted"
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"schemas": {
|
||||||
|
"Pet": {
|
||||||
|
"allOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/components/schemas/NewPet"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"required": ["id"],
|
||||||
|
"properties": {
|
||||||
|
"id": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int64"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"NewPet": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["name"],
|
||||||
|
"properties": {
|
||||||
|
"name": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"tag": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Error": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["code", "message"],
|
||||||
|
"properties": {
|
||||||
|
"code": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int32"
|
||||||
|
},
|
||||||
|
"message": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
177
WoofWare.Myriad.Plugins.Test/TestSwagger/petstore.json
Normal file
177
WoofWare.Myriad.Plugins.Test/TestSwagger/petstore.json
Normal file
@@ -0,0 +1,177 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.0.0",
|
||||||
|
"info": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"title": "Swagger Petstore",
|
||||||
|
"license": {
|
||||||
|
"name": "MIT"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"servers": [
|
||||||
|
{
|
||||||
|
"url": "http://petstore.swagger.io/v1"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"paths": {
|
||||||
|
"/pets": {
|
||||||
|
"get": {
|
||||||
|
"summary": "List all pets",
|
||||||
|
"operationId": "listPets",
|
||||||
|
"tags": ["pets"],
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "limit",
|
||||||
|
"in": "query",
|
||||||
|
"description": "How many items to return at one time (max 100)",
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "integer",
|
||||||
|
"maximum": 100,
|
||||||
|
"format": "int32"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "A paged array of pets",
|
||||||
|
"headers": {
|
||||||
|
"x-next": {
|
||||||
|
"description": "A link to the next page of responses",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Pets"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"post": {
|
||||||
|
"summary": "Create a pet",
|
||||||
|
"operationId": "createPets",
|
||||||
|
"tags": ["pets"],
|
||||||
|
"requestBody": {
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": true
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"201": {
|
||||||
|
"description": "Null response"
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/pets/{petId}": {
|
||||||
|
"get": {
|
||||||
|
"summary": "Info for a specific pet",
|
||||||
|
"operationId": "showPetById",
|
||||||
|
"tags": ["pets"],
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "petId",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"description": "The id of the pet to retrieve",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "Expected response to a valid request",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"default": {
|
||||||
|
"description": "unexpected error",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"schemas": {
|
||||||
|
"Pet": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["id", "name"],
|
||||||
|
"properties": {
|
||||||
|
"id": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int64"
|
||||||
|
},
|
||||||
|
"name": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"tag": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Pets": {
|
||||||
|
"type": "array",
|
||||||
|
"maxItems": 100,
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Error": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["code", "message"],
|
||||||
|
"properties": {
|
||||||
|
"code": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int32"
|
||||||
|
},
|
||||||
|
"message": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
261
WoofWare.Myriad.Plugins.Test/TestSwagger/tictactoe.json
Normal file
261
WoofWare.Myriad.Plugins.Test/TestSwagger/tictactoe.json
Normal file
@@ -0,0 +1,261 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.1.0",
|
||||||
|
"info": {
|
||||||
|
"title": "Tic Tac Toe",
|
||||||
|
"description": "This API allows writing down marks on a Tic Tac Toe board\nand requesting the state of the board or of individual squares.\n",
|
||||||
|
"version": "1.0.0"
|
||||||
|
},
|
||||||
|
"tags": [
|
||||||
|
{
|
||||||
|
"name": "Gameplay"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"paths": {
|
||||||
|
"/board": {
|
||||||
|
"get": {
|
||||||
|
"summary": "Get the whole board",
|
||||||
|
"description": "Retrieves the current state of the board and the winner.",
|
||||||
|
"tags": ["Gameplay"],
|
||||||
|
"operationId": "get-board",
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "OK",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/status"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"defaultApiKey": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"app2AppOauth": ["board:read"]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/board/{row}/{column}": {
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"$ref": "#/components/parameters/rowParam"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"$ref": "#/components/parameters/columnParam"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"get": {
|
||||||
|
"summary": "Get a single board square",
|
||||||
|
"description": "Retrieves the requested square.",
|
||||||
|
"tags": ["Gameplay"],
|
||||||
|
"operationId": "get-square",
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "OK",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/mark"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"400": {
|
||||||
|
"description": "The provided parameters are incorrect",
|
||||||
|
"content": {
|
||||||
|
"text/html": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/errorMessage"
|
||||||
|
},
|
||||||
|
"example": "Illegal coordinates"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"bearerHttpAuthentication": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"user2AppOauth": ["board:read"]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"put": {
|
||||||
|
"summary": "Set a single board square",
|
||||||
|
"description": "Places a mark on the board and retrieves the whole board and the winner (if any).",
|
||||||
|
"tags": ["Gameplay"],
|
||||||
|
"operationId": "put-square",
|
||||||
|
"requestBody": {
|
||||||
|
"required": true,
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/mark"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "OK",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/status"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"400": {
|
||||||
|
"description": "The provided parameters are incorrect",
|
||||||
|
"content": {
|
||||||
|
"text/html": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/errorMessage"
|
||||||
|
},
|
||||||
|
"examples": {
|
||||||
|
"illegalCoordinates": {
|
||||||
|
"value": "Illegal coordinates."
|
||||||
|
},
|
||||||
|
"notEmpty": {
|
||||||
|
"value": "Square is not empty."
|
||||||
|
},
|
||||||
|
"invalidMark": {
|
||||||
|
"value": "Invalid Mark (X or O)."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"bearerHttpAuthentication": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"user2AppOauth": ["board:write"]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"parameters": {
|
||||||
|
"rowParam": {
|
||||||
|
"description": "Board row (vertical coordinate)",
|
||||||
|
"name": "row",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/coordinate"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"columnParam": {
|
||||||
|
"description": "Board column (horizontal coordinate)",
|
||||||
|
"name": "column",
|
||||||
|
"in": "path",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/coordinate"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"schemas": {
|
||||||
|
"errorMessage": {
|
||||||
|
"type": "string",
|
||||||
|
"maxLength": 256,
|
||||||
|
"description": "A text message describing an error"
|
||||||
|
},
|
||||||
|
"coordinate": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 1,
|
||||||
|
"maximum": 3,
|
||||||
|
"example": 1
|
||||||
|
},
|
||||||
|
"mark": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": [".", "X", "O"],
|
||||||
|
"description": "Possible values for a board square. `.` means empty square.",
|
||||||
|
"example": "."
|
||||||
|
},
|
||||||
|
"board": {
|
||||||
|
"type": "array",
|
||||||
|
"maxItems": 3,
|
||||||
|
"minItems": 3,
|
||||||
|
"items": {
|
||||||
|
"type": "array",
|
||||||
|
"maxItems": 3,
|
||||||
|
"minItems": 3,
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/mark"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"winner": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": [".", "X", "O"],
|
||||||
|
"description": "Winner of the game. `.` means nobody has won yet.",
|
||||||
|
"example": "."
|
||||||
|
},
|
||||||
|
"status": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"winner": {
|
||||||
|
"$ref": "#/components/schemas/winner"
|
||||||
|
},
|
||||||
|
"board": {
|
||||||
|
"$ref": "#/components/schemas/board"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"securitySchemes": {
|
||||||
|
"defaultApiKey": {
|
||||||
|
"description": "API key provided in console",
|
||||||
|
"type": "apiKey",
|
||||||
|
"name": "api-key",
|
||||||
|
"in": "header"
|
||||||
|
},
|
||||||
|
"basicHttpAuthentication": {
|
||||||
|
"description": "Basic HTTP Authentication",
|
||||||
|
"type": "http",
|
||||||
|
"scheme": "Basic"
|
||||||
|
},
|
||||||
|
"bearerHttpAuthentication": {
|
||||||
|
"description": "Bearer token using a JWT",
|
||||||
|
"type": "http",
|
||||||
|
"scheme": "Bearer",
|
||||||
|
"bearerFormat": "JWT"
|
||||||
|
},
|
||||||
|
"app2AppOauth": {
|
||||||
|
"type": "oauth2",
|
||||||
|
"flows": {
|
||||||
|
"clientCredentials": {
|
||||||
|
"tokenUrl": "https://learn.openapis.org/oauth/2.0/token",
|
||||||
|
"scopes": {
|
||||||
|
"board:read": "Read the board"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"user2AppOauth": {
|
||||||
|
"type": "oauth2",
|
||||||
|
"flows": {
|
||||||
|
"authorizationCode": {
|
||||||
|
"authorizationUrl": "https://learn.openapis.org/oauth/2.0/auth",
|
||||||
|
"tokenUrl": "https://learn.openapis.org/oauth/2.0/token",
|
||||||
|
"scopes": {
|
||||||
|
"board:read": "Read the board",
|
||||||
|
"board:write": "Write to the board"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
241
WoofWare.Myriad.Plugins.Test/TestSwagger/uspto.json
Normal file
241
WoofWare.Myriad.Plugins.Test/TestSwagger/uspto.json
Normal file
@@ -0,0 +1,241 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.0.1",
|
||||||
|
"servers": [
|
||||||
|
{
|
||||||
|
"url": "{scheme}://developer.uspto.gov/ds-api",
|
||||||
|
"variables": {
|
||||||
|
"scheme": {
|
||||||
|
"description": "The Data Set API is accessible via https and http",
|
||||||
|
"enum": ["https", "http"],
|
||||||
|
"default": "https"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"info": {
|
||||||
|
"description": "The Data Set API (DSAPI) allows the public users to discover and search USPTO exported data sets. This is a generic API that allows USPTO users to make any CSV based data files searchable through API. With the help of GET call, it returns the list of data fields that are searchable. With the help of POST call, data can be fetched based on the filters on the field names. Please note that POST call is used to search the actual data. The reason for the POST call is that it allows users to specify any complex search criteria without worry about the GET size limitations as well as encoding of the input parameters.",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"title": "USPTO Data Set API",
|
||||||
|
"contact": {
|
||||||
|
"name": "Open Data Portal",
|
||||||
|
"url": "https://developer.uspto.gov",
|
||||||
|
"email": "developer@uspto.gov"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"tags": [
|
||||||
|
{
|
||||||
|
"name": "metadata",
|
||||||
|
"description": "Find out about the data sets"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "search",
|
||||||
|
"description": "Search a data set"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"paths": {
|
||||||
|
"/": {
|
||||||
|
"get": {
|
||||||
|
"tags": ["metadata"],
|
||||||
|
"operationId": "list-data-sets",
|
||||||
|
"summary": "List available data sets",
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "Returns a list of data sets",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/dataSetList"
|
||||||
|
},
|
||||||
|
"example": {
|
||||||
|
"total": 2,
|
||||||
|
"apis": [
|
||||||
|
{
|
||||||
|
"apiKey": "oa_citations",
|
||||||
|
"apiVersionNumber": "v1",
|
||||||
|
"apiUrl": "https://developer.uspto.gov/ds-api/oa_citations/v1/fields",
|
||||||
|
"apiDocumentationUrl": "https://developer.uspto.gov/ds-api-docs/index.html?url=https://developer.uspto.gov/ds-api/swagger/docs/oa_citations.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"apiKey": "cancer_moonshot",
|
||||||
|
"apiVersionNumber": "v1",
|
||||||
|
"apiUrl": "https://developer.uspto.gov/ds-api/cancer_moonshot/v1/fields",
|
||||||
|
"apiDocumentationUrl": "https://developer.uspto.gov/ds-api-docs/index.html?url=https://developer.uspto.gov/ds-api/swagger/docs/cancer_moonshot.json"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/{dataset}/{version}/fields": {
|
||||||
|
"get": {
|
||||||
|
"tags": ["metadata"],
|
||||||
|
"summary": "Provides the general information about the API and the list of fields that can be used to query the dataset.",
|
||||||
|
"description": "This GET API returns the list of all the searchable field names that are in the oa_citations. Please see the 'fields' attribute which returns an array of field names. Each field or a combination of fields can be searched using the syntax options shown below.",
|
||||||
|
"operationId": "list-searchable-fields",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "dataset",
|
||||||
|
"in": "path",
|
||||||
|
"description": "Name of the dataset.",
|
||||||
|
"required": true,
|
||||||
|
"example": "oa_citations",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "version",
|
||||||
|
"in": "path",
|
||||||
|
"description": "Version of the dataset.",
|
||||||
|
"required": true,
|
||||||
|
"example": "v1",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "The dataset API for the given version is found and it is accessible to consume.",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"404": {
|
||||||
|
"description": "The combination of dataset name and version is not found in the system or it is not published yet to be consumed by public.",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"/{dataset}/{version}/records": {
|
||||||
|
"post": {
|
||||||
|
"tags": ["search"],
|
||||||
|
"summary": "Provides search capability for the data set with the given search criteria.",
|
||||||
|
"description": "This API is based on Solr/Lucene Search. The data is indexed using SOLR. This GET API returns the list of all the searchable field names that are in the Solr Index. Please see the 'fields' attribute which returns an array of field names. Each field or a combination of fields can be searched using the Solr/Lucene Syntax. Please refer https://lucene.apache.org/core/3_6_2/queryparsersyntax.html#Overview for the query syntax. List of field names that are searchable can be determined using above GET api.",
|
||||||
|
"operationId": "perform-search",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "version",
|
||||||
|
"in": "path",
|
||||||
|
"description": "Version of the dataset.",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "v1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "dataset",
|
||||||
|
"in": "path",
|
||||||
|
"description": "Name of the dataset. In this case, the default value is oa_citations",
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "oa_citations"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "successful operation",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": {
|
||||||
|
"type": "object"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"404": {
|
||||||
|
"description": "No matching record found for the given criteria."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"requestBody": {
|
||||||
|
"content": {
|
||||||
|
"application/x-www-form-urlencoded": {
|
||||||
|
"schema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"criteria": {
|
||||||
|
"description": "Uses Lucene Query Syntax in the format of propertyName:value, propertyName:[num1 TO num2] and date range format: propertyName:[yyyyMMdd TO yyyyMMdd]. In the response please see the 'docs' element which has the list of record objects. Each record structure would consist of all the fields and their corresponding values.",
|
||||||
|
"type": "string",
|
||||||
|
"default": "*:*"
|
||||||
|
},
|
||||||
|
"start": {
|
||||||
|
"description": "Starting record number. Default value is 0.",
|
||||||
|
"type": "integer",
|
||||||
|
"default": 0
|
||||||
|
},
|
||||||
|
"rows": {
|
||||||
|
"description": "Specify number of rows to be returned. If you run the search with default values, in the response you will see 'numFound' attribute which will tell the number of records available in the dataset.",
|
||||||
|
"type": "integer",
|
||||||
|
"default": 100
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["criteria"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"schemas": {
|
||||||
|
"dataSetList": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"total": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"apis": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"apiKey": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "To be used as a dataset parameter value"
|
||||||
|
},
|
||||||
|
"apiVersionNumber": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "To be used as a version parameter value"
|
||||||
|
},
|
||||||
|
"apiUrl": {
|
||||||
|
"type": "string",
|
||||||
|
"format": "uriref",
|
||||||
|
"description": "The URL describing the dataset's fields"
|
||||||
|
},
|
||||||
|
"apiDocumentationUrl": {
|
||||||
|
"type": "string",
|
||||||
|
"format": "uriref",
|
||||||
|
"description": "A URL to the API console for each API"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,47 @@
|
|||||||
|
{
|
||||||
|
"openapi": "3.1.0",
|
||||||
|
"info": {
|
||||||
|
"title": "Webhook Example",
|
||||||
|
"version": "1.0.0"
|
||||||
|
},
|
||||||
|
"webhooks": {
|
||||||
|
"newPet": {
|
||||||
|
"post": {
|
||||||
|
"requestBody": {
|
||||||
|
"description": "Information about a new pet in the system",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/Pet"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "Return a 200 status to indicate that the data was received successfully"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"schemas": {
|
||||||
|
"Pet": {
|
||||||
|
"required": ["id", "name"],
|
||||||
|
"properties": {
|
||||||
|
"id": {
|
||||||
|
"type": "integer",
|
||||||
|
"format": "int64"
|
||||||
|
},
|
||||||
|
"name": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"tag": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -13,6 +13,7 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<Compile Include="Assembly.fs" />
|
||||||
<Compile Include="HttpClient.fs"/>
|
<Compile Include="HttpClient.fs"/>
|
||||||
<Compile Include="PureGymDtos.fs"/>
|
<Compile Include="PureGymDtos.fs"/>
|
||||||
<Compile Include="TestJsonParse\TestJsonParse.fs" />
|
<Compile Include="TestJsonParse\TestJsonParse.fs" />
|
||||||
@@ -36,18 +37,29 @@
|
|||||||
<Compile Include="TestCataGenerator\TestMyList2.fs" />
|
<Compile Include="TestCataGenerator\TestMyList2.fs" />
|
||||||
<Compile Include="TestArgParser\TestArgParser.fs" />
|
<Compile Include="TestArgParser\TestArgParser.fs" />
|
||||||
<Compile Include="TestSwagger\TestSwaggerParse.fs" />
|
<Compile Include="TestSwagger\TestSwaggerParse.fs" />
|
||||||
|
<Compile Include="TestSwagger\TestOpenApi3Parse.fs" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\api-with-examples.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\callback-example.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\link-example.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\non-oauth-scopes.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\petstore.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\petstore-expanded.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\tictactoe.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\uspto.json" />
|
||||||
|
<EmbeddedResource Include="TestSwagger\webhook-example.json" />
|
||||||
<Compile Include="TestRemoveOptions.fs"/>
|
<Compile Include="TestRemoveOptions.fs"/>
|
||||||
<Compile Include="TestSurface.fs"/>
|
<Compile Include="TestSurface.fs"/>
|
||||||
<None Include="../.github/workflows/dotnet.yaml" />
|
<None Include="../.github/workflows/dotnet.yaml" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="ApiSurface" Version="4.1.20"/>
|
<PackageReference Include="ApiSurface" Version="4.1.21"/>
|
||||||
<PackageReference Include="FsCheck" Version="3.2.0"/>
|
<PackageReference Include="FsCheck" Version="3.3.0"/>
|
||||||
<PackageReference Include="FsUnit" Version="7.0.1"/>
|
<PackageReference Include="FsUnit" Version="7.0.1"/>
|
||||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0"/>
|
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.14.1"/>
|
||||||
<PackageReference Include="NUnit" Version="4.3.2"/>
|
<PackageReference Include="NUnit" Version="4.3.2"/>
|
||||||
<PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/>
|
<PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/>
|
||||||
|
<PackageReference Include="WoofWare.Expect" Version="0.4.2" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
@@ -1,201 +0,0 @@
|
|||||||
Apache License
|
|
||||||
Version 2.0, January 2004
|
|
||||||
http://www.apache.org/licenses/
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
|
||||||
|
|
||||||
1. Definitions.
|
|
||||||
|
|
||||||
"License" shall mean the terms and conditions for use, reproduction,
|
|
||||||
and distribution as defined by Sections 1 through 9 of this document.
|
|
||||||
|
|
||||||
"Licensor" shall mean the copyright owner or entity authorized by
|
|
||||||
the copyright owner that is granting the License.
|
|
||||||
|
|
||||||
"Legal Entity" shall mean the union of the acting entity and all
|
|
||||||
other entities that control, are controlled by, or are under common
|
|
||||||
control with that entity. For the purposes of this definition,
|
|
||||||
"control" means (i) the power, direct or indirect, to cause the
|
|
||||||
direction or management of such entity, whether by contract or
|
|
||||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
|
||||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
|
||||||
|
|
||||||
"You" (or "Your") shall mean an individual or Legal Entity
|
|
||||||
exercising permissions granted by this License.
|
|
||||||
|
|
||||||
"Source" form shall mean the preferred form for making modifications,
|
|
||||||
including but not limited to software source code, documentation
|
|
||||||
source, and configuration files.
|
|
||||||
|
|
||||||
"Object" form shall mean any form resulting from mechanical
|
|
||||||
transformation or translation of a Source form, including but
|
|
||||||
not limited to compiled object code, generated documentation,
|
|
||||||
and conversions to other media types.
|
|
||||||
|
|
||||||
"Work" shall mean the work of authorship, whether in Source or
|
|
||||||
Object form, made available under the License, as indicated by a
|
|
||||||
copyright notice that is included in or attached to the work
|
|
||||||
(an example is provided in the Appendix below).
|
|
||||||
|
|
||||||
"Derivative Works" shall mean any work, whether in Source or Object
|
|
||||||
form, that is based on (or derived from) the Work and for which the
|
|
||||||
editorial revisions, annotations, elaborations, or other modifications
|
|
||||||
represent, as a whole, an original work of authorship. For the purposes
|
|
||||||
of this License, Derivative Works shall not include works that remain
|
|
||||||
separable from, or merely link (or bind by name) to the interfaces of,
|
|
||||||
the Work and Derivative Works thereof.
|
|
||||||
|
|
||||||
"Contribution" shall mean any work of authorship, including
|
|
||||||
the original version of the Work and any modifications or additions
|
|
||||||
to that Work or Derivative Works thereof, that is intentionally
|
|
||||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
|
||||||
or by an individual or Legal Entity authorized to submit on behalf of
|
|
||||||
the copyright owner. For the purposes of this definition, "submitted"
|
|
||||||
means any form of electronic, verbal, or written communication sent
|
|
||||||
to the Licensor or its representatives, including but not limited to
|
|
||||||
communication on electronic mailing lists, source code control systems,
|
|
||||||
and issue tracking systems that are managed by, or on behalf of, the
|
|
||||||
Licensor for the purpose of discussing and improving the Work, but
|
|
||||||
excluding communication that is conspicuously marked or otherwise
|
|
||||||
designated in writing by the copyright owner as "Not a Contribution."
|
|
||||||
|
|
||||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
|
||||||
on behalf of whom a Contribution has been received by Licensor and
|
|
||||||
subsequently incorporated within the Work.
|
|
||||||
|
|
||||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
copyright license to reproduce, prepare Derivative Works of,
|
|
||||||
publicly display, publicly perform, sublicense, and distribute the
|
|
||||||
Work and such Derivative Works in Source or Object form.
|
|
||||||
|
|
||||||
3. Grant of Patent License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
(except as stated in this section) patent license to make, have made,
|
|
||||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
|
||||||
where such license applies only to those patent claims licensable
|
|
||||||
by such Contributor that are necessarily infringed by their
|
|
||||||
Contribution(s) alone or by combination of their Contribution(s)
|
|
||||||
with the Work to which such Contribution(s) was submitted. If You
|
|
||||||
institute patent litigation against any entity (including a
|
|
||||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
|
||||||
or a Contribution incorporated within the Work constitutes direct
|
|
||||||
or contributory patent infringement, then any patent licenses
|
|
||||||
granted to You under this License for that Work shall terminate
|
|
||||||
as of the date such litigation is filed.
|
|
||||||
|
|
||||||
4. Redistribution. You may reproduce and distribute copies of the
|
|
||||||
Work or Derivative Works thereof in any medium, with or without
|
|
||||||
modifications, and in Source or Object form, provided that You
|
|
||||||
meet the following conditions:
|
|
||||||
|
|
||||||
(a) You must give any other recipients of the Work or
|
|
||||||
Derivative Works a copy of this License; and
|
|
||||||
|
|
||||||
(b) You must cause any modified files to carry prominent notices
|
|
||||||
stating that You changed the files; and
|
|
||||||
|
|
||||||
(c) You must retain, in the Source form of any Derivative Works
|
|
||||||
that You distribute, all copyright, patent, trademark, and
|
|
||||||
attribution notices from the Source form of the Work,
|
|
||||||
excluding those notices that do not pertain to any part of
|
|
||||||
the Derivative Works; and
|
|
||||||
|
|
||||||
(d) If the Work includes a "NOTICE" text file as part of its
|
|
||||||
distribution, then any Derivative Works that You distribute must
|
|
||||||
include a readable copy of the attribution notices contained
|
|
||||||
within such NOTICE file, excluding those notices that do not
|
|
||||||
pertain to any part of the Derivative Works, in at least one
|
|
||||||
of the following places: within a NOTICE text file distributed
|
|
||||||
as part of the Derivative Works; within the Source form or
|
|
||||||
documentation, if provided along with the Derivative Works; or,
|
|
||||||
within a display generated by the Derivative Works, if and
|
|
||||||
wherever such third-party notices normally appear. The contents
|
|
||||||
of the NOTICE file are for informational purposes only and
|
|
||||||
do not modify the License. You may add Your own attribution
|
|
||||||
notices within Derivative Works that You distribute, alongside
|
|
||||||
or as an addendum to the NOTICE text from the Work, provided
|
|
||||||
that such additional attribution notices cannot be construed
|
|
||||||
as modifying the License.
|
|
||||||
|
|
||||||
You may add Your own copyright statement to Your modifications and
|
|
||||||
may provide additional or different license terms and conditions
|
|
||||||
for use, reproduction, or distribution of Your modifications, or
|
|
||||||
for any such Derivative Works as a whole, provided Your use,
|
|
||||||
reproduction, and distribution of the Work otherwise complies with
|
|
||||||
the conditions stated in this License.
|
|
||||||
|
|
||||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
|
||||||
any Contribution intentionally submitted for inclusion in the Work
|
|
||||||
by You to the Licensor shall be under the terms and conditions of
|
|
||||||
this License, without any additional terms or conditions.
|
|
||||||
Notwithstanding the above, nothing herein shall supersede or modify
|
|
||||||
the terms of any separate license agreement you may have executed
|
|
||||||
with Licensor regarding such Contributions.
|
|
||||||
|
|
||||||
6. Trademarks. This License does not grant permission to use the trade
|
|
||||||
names, trademarks, service marks, or product names of the Licensor,
|
|
||||||
except as required for reasonable and customary use in describing the
|
|
||||||
origin of the Work and reproducing the content of the NOTICE file.
|
|
||||||
|
|
||||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
|
||||||
agreed to in writing, Licensor provides the Work (and each
|
|
||||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
||||||
implied, including, without limitation, any warranties or conditions
|
|
||||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
|
||||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
|
||||||
appropriateness of using or redistributing the Work and assume any
|
|
||||||
risks associated with Your exercise of permissions under this License.
|
|
||||||
|
|
||||||
8. Limitation of Liability. In no event and under no legal theory,
|
|
||||||
whether in tort (including negligence), contract, or otherwise,
|
|
||||||
unless required by applicable law (such as deliberate and grossly
|
|
||||||
negligent acts) or agreed to in writing, shall any Contributor be
|
|
||||||
liable to You for damages, including any direct, indirect, special,
|
|
||||||
incidental, or consequential damages of any character arising as a
|
|
||||||
result of this License or out of the use or inability to use the
|
|
||||||
Work (including but not limited to damages for loss of goodwill,
|
|
||||||
work stoppage, computer failure or malfunction, or any and all
|
|
||||||
other commercial damages or losses), even if such Contributor
|
|
||||||
has been advised of the possibility of such damages.
|
|
||||||
|
|
||||||
9. Accepting Warranty or Additional Liability. While redistributing
|
|
||||||
the Work or Derivative Works thereof, You may choose to offer,
|
|
||||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
|
||||||
or other liability obligations and/or rights consistent with this
|
|
||||||
License. However, in accepting such obligations, You may act only
|
|
||||||
on Your own behalf and on Your sole responsibility, not on behalf
|
|
||||||
of any other Contributor, and only if You agree to indemnify,
|
|
||||||
defend, and hold each Contributor harmless for any liability
|
|
||||||
incurred by, or claims asserted against, such Contributor by reason
|
|
||||||
of your accepting any such warranty or additional liability.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
APPENDIX: How to apply the Apache License to your work.
|
|
||||||
|
|
||||||
To apply the Apache License to your work, attach the following
|
|
||||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
|
||||||
replaced with your own identifying information. (Don't include
|
|
||||||
the brackets!) The text should be enclosed in the appropriate
|
|
||||||
comment syntax for the file format. We also recommend that a
|
|
||||||
file or class name and description of purpose be included on the
|
|
||||||
same "printed page" as the copyright notice for easier
|
|
||||||
identification within third-party archives.
|
|
||||||
|
|
||||||
Copyright [yyyy] [name of copyright owner]
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
6
WoofWare.Myriad.Plugins/AssemblyInfo.fs
Normal file
6
WoofWare.Myriad.Plugins/AssemblyInfo.fs
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
module internal WoofWare.Myriad.Plugins.AssemblyInfo
|
||||||
|
|
||||||
|
open System.Runtime.CompilerServices
|
||||||
|
|
||||||
|
[<assembly : InternalsVisibleTo("WoofWare.Myriad.Plugins.Test")>]
|
||||||
|
do ()
|
@@ -1174,28 +1174,30 @@ module internal CataGenerator =
|
|||||||
]
|
]
|
||||||
|> SynModuleOrNamespace.createNamespace ns
|
|> SynModuleOrNamespace.createNamespace ns
|
||||||
|
|
||||||
/// This function comes from Myriad, and is therefore derived from an Apache 2.0-licenced work.
|
/// For each namespace/module, grab the types which are defined in consecutive `and`-knots in that namespace/module,
|
||||||
/// https://github.com/MoiraeSoftware/myriad/blob/3c9818faabf9d508c10c28d5ecd26e66fafb48a1/src/Myriad.Core/Ast.fs#L160
|
/// and also return the fully-qualified namespace/module name alongside that group of types.
|
||||||
/// A copy of the Apache 2.0 licence is at ApacheLicence.txt.
|
/// A given module LongIdent may show up many times in the output: once for each recursive knot.
|
||||||
|
// Function originally inspired by https://github.com/MoiraeSoftware/myriad/blob/3c9818faabf9d508c10c28d5ecd26e66fafb48a1/src/Myriad.Core/Ast.fs#L160
|
||||||
|
// but there's really only one reasonable implementation of this type signature and semantics.
|
||||||
let groupedTypeDefns (ast : ParsedInput) : (LongIdent * SynTypeDefn list) list =
|
let groupedTypeDefns (ast : ParsedInput) : (LongIdent * SynTypeDefn list) list =
|
||||||
let rec extractTypes (moduleDecls : SynModuleDecl list) (ns : LongIdent) =
|
let rec extractTypes (decls : SynModuleDecl list) (ns : LongIdent) =
|
||||||
[
|
decls
|
||||||
for moduleDecl in moduleDecls do
|
|> List.collect (fun moduleDecl ->
|
||||||
match moduleDecl with
|
match moduleDecl with
|
||||||
| SynModuleDecl.Types (types, _) -> yield (ns, types)
|
| SynModuleDecl.Types (types, _) -> [ ns, types ]
|
||||||
| SynModuleDecl.NestedModule (SynComponentInfo (_, _, _, longId, _, _, _, _), _, decls, _, _, _) ->
|
| SynModuleDecl.NestedModule (SynComponentInfo (_, _, _, longId, _, _, _, _), _, decls, _, _, _) ->
|
||||||
let combined = longId |> List.append ns
|
let combined = longId |> List.append ns
|
||||||
yield! (extractTypes decls combined)
|
extractTypes decls combined
|
||||||
| _ -> ()
|
| _ -> []
|
||||||
]
|
)
|
||||||
|
|
||||||
[
|
match ast with
|
||||||
match ast with
|
| ParsedInput.ImplFile (ParsedImplFileInput (_, _, _, _, _, contents, _, _, _)) ->
|
||||||
| ParsedInput.ImplFile (ParsedImplFileInput (_, _, _, _, _, modules, _, _, _)) ->
|
contents
|
||||||
for SynModuleOrNamespace (namespaceId, _, _, moduleDecls, _, _, _, _, _) in modules do
|
|> List.collect (fun (SynModuleOrNamespace (namespaceId, _, _, moduleDecls, _, _, _, _, _)) ->
|
||||||
yield! extractTypes moduleDecls namespaceId
|
extractTypes moduleDecls namespaceId
|
||||||
| _ -> ()
|
)
|
||||||
]
|
| _ -> []
|
||||||
|
|
||||||
open Myriad.Core
|
open Myriad.Core
|
||||||
|
|
||||||
|
@@ -453,15 +453,20 @@ module internal HttpClientGenerator =
|
|||||||
|
|
||||||
let contentTypeHeader, memberHeaders =
|
let contentTypeHeader, memberHeaders =
|
||||||
info.Headers
|
info.Headers
|
||||||
|> List.partition (fun (headerName, headerValue) ->
|
|> List.partition (fun (headerName, _headerValue) ->
|
||||||
match headerName |> SynExpr.stripOptionalParen with
|
match headerName |> SynExpr.stripOptionalParen with
|
||||||
| SynExpr.Const (SynConst.String ("Content-Type", _, _), _) -> true
|
| SynExpr.Const (SynConst.String (s, _, _), _) ->
|
||||||
|
System.String.Equals (s, "Content-Type", System.StringComparison.OrdinalIgnoreCase)
|
||||||
| _ -> false
|
| _ -> false
|
||||||
)
|
)
|
||||||
|
|
||||||
let contentTypeHeader =
|
let contentTypeHeader =
|
||||||
match contentTypeHeader with
|
match contentTypeHeader with
|
||||||
| [] -> None
|
| [] ->
|
||||||
|
// Set application/json if we *know* we're sending JSON
|
||||||
|
match bodyParam with
|
||||||
|
| Some (BodyParamMethods.Serialise _, _) -> Some (SynExpr.CreateConst "application/json")
|
||||||
|
| _ -> None
|
||||||
| [ _, ct ] -> Some (SynExpr.stripOptionalParen ct)
|
| [ _, ct ] -> Some (SynExpr.stripOptionalParen ct)
|
||||||
| _ -> failwith "Unexpectedly got multiple Content-Type headers"
|
| _ -> failwith "Unexpectedly got multiple Content-Type headers"
|
||||||
|
|
||||||
|
66
WoofWare.Myriad.Plugins/HttpMethod.fs
Normal file
66
WoofWare.Myriad.Plugins/HttpMethod.fs
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
namespace WoofWare.Myriad.Plugins
|
||||||
|
|
||||||
|
open System
|
||||||
|
|
||||||
|
/// An HTTP method. This is System.Net.Http.HttpMethod, but
|
||||||
|
/// a proper discriminated union.
|
||||||
|
type HttpMethod =
|
||||||
|
/// HTTP Get
|
||||||
|
| Get
|
||||||
|
/// HTTP Post
|
||||||
|
| Post
|
||||||
|
/// HTTP Delete
|
||||||
|
| Delete
|
||||||
|
/// HTTP Patch
|
||||||
|
| Patch
|
||||||
|
/// HTTP Options
|
||||||
|
| Options
|
||||||
|
/// HTTP Head
|
||||||
|
| Head
|
||||||
|
/// HTTP Put
|
||||||
|
| Put
|
||||||
|
/// HTTP Trace
|
||||||
|
| Trace
|
||||||
|
|
||||||
|
/// Convert to the standard library's enum type.
|
||||||
|
member this.ToDotNet () : System.Net.Http.HttpMethod =
|
||||||
|
match this with
|
||||||
|
| HttpMethod.Get -> System.Net.Http.HttpMethod.Get
|
||||||
|
| HttpMethod.Post -> System.Net.Http.HttpMethod.Post
|
||||||
|
| HttpMethod.Delete -> System.Net.Http.HttpMethod.Delete
|
||||||
|
| HttpMethod.Patch -> System.Net.Http.HttpMethod.Patch
|
||||||
|
| HttpMethod.Options -> System.Net.Http.HttpMethod.Options
|
||||||
|
| HttpMethod.Head -> System.Net.Http.HttpMethod.Head
|
||||||
|
| HttpMethod.Put -> System.Net.Http.HttpMethod.Put
|
||||||
|
| HttpMethod.Trace -> System.Net.Http.HttpMethod.Trace
|
||||||
|
|
||||||
|
/// Human-readable string representation.
|
||||||
|
override this.ToString () : string =
|
||||||
|
match this with
|
||||||
|
| HttpMethod.Get -> "Get"
|
||||||
|
| HttpMethod.Post -> "Post"
|
||||||
|
| HttpMethod.Delete -> "Delete"
|
||||||
|
| HttpMethod.Patch -> "Patch"
|
||||||
|
| HttpMethod.Options -> "Options"
|
||||||
|
| HttpMethod.Head -> "Head"
|
||||||
|
| HttpMethod.Put -> "Put"
|
||||||
|
| HttpMethod.Trace -> "Trace"
|
||||||
|
|
||||||
|
/// Throws on invalid inputs.
|
||||||
|
static member Parse (s : string) : HttpMethod =
|
||||||
|
if String.Equals (s, "get", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Get
|
||||||
|
elif String.Equals (s, "post", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Post
|
||||||
|
elif String.Equals (s, "patch", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Patch
|
||||||
|
elif String.Equals (s, "delete", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Delete
|
||||||
|
elif String.Equals (s, "head", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Head
|
||||||
|
elif String.Equals (s, "options", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Options
|
||||||
|
elif String.Equals (s, "put", StringComparison.OrdinalIgnoreCase) then
|
||||||
|
HttpMethod.Put
|
||||||
|
else
|
||||||
|
failwith $"Unrecognised method: %s{s}"
|
48
WoofWare.Myriad.Plugins/JsonHelpers.fs
Normal file
48
WoofWare.Myriad.Plugins/JsonHelpers.fs
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
namespace WoofWare.Myriad.Plugins
|
||||||
|
|
||||||
|
open System.Text.Json.Nodes
|
||||||
|
|
||||||
|
[<AutoOpen>]
|
||||||
|
module internal JsonHelpers =
|
||||||
|
let inline asString (n : JsonNode) (key : string) : string =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
||||||
|
| s -> s.GetValue<string> ()
|
||||||
|
|
||||||
|
[<RequiresExplicitTypeArguments>]
|
||||||
|
let inline asOpt<'ret> (n : JsonNode) (key : string) : 'ret option =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> None
|
||||||
|
| s -> s.GetValue<'ret> () |> Some
|
||||||
|
|
||||||
|
let inline asObj (n : JsonNode) (key : string) : JsonObject =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
||||||
|
| o -> o.AsObject ()
|
||||||
|
|
||||||
|
let inline asObjOpt (n : JsonNode) (key : string) : JsonObject option =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> None
|
||||||
|
| o -> o.AsObject () |> Some
|
||||||
|
|
||||||
|
let inline asArr (n : JsonNode) (key : string) : JsonArray =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
||||||
|
| o -> o.AsArray ()
|
||||||
|
|
||||||
|
let inline asArrOpt (n : JsonNode) (key : string) : JsonArray option =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> None
|
||||||
|
| o -> o.AsArray () |> Some
|
||||||
|
|
||||||
|
[<RequiresExplicitTypeArguments>]
|
||||||
|
let inline asArr'<'v> (n : JsonNode) (key : string) : 'v list =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
||||||
|
| o -> o.AsArray () |> Seq.map (fun v -> v.GetValue<'v> ()) |> Seq.toList
|
||||||
|
|
||||||
|
[<RequiresExplicitTypeArguments>]
|
||||||
|
let inline asArrOpt'<'v> (n : JsonNode) (key : string) : 'v list option =
|
||||||
|
match n.[key] with
|
||||||
|
| null -> None
|
||||||
|
| o -> o.AsArray () |> Seq.map (fun v -> v.GetValue<'v> ()) |> Seq.toList |> Some
|
1332
WoofWare.Myriad.Plugins/OpenApi3.fs
Normal file
1332
WoofWare.Myriad.Plugins/OpenApi3.fs
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,84 +1,7 @@
|
|||||||
WoofWare.Myriad.Plugins.AdditionalProperties inherit obj, implements WoofWare.Myriad.Plugins.AdditionalProperties System.IEquatable, System.Collections.IStructuralEquatable - union type with 2 cases
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties+Constrained inherit WoofWare.Myriad.Plugins.AdditionalProperties
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties+Constrained.get_Item [method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties+Constrained.Item [property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties+Tags inherit obj
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties+Tags.Constrained [static field]: int = 1
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties+Tags.Never [static field]: int = 0
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.Equals [method]: (WoofWare.Myriad.Plugins.AdditionalProperties, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.get_IsConstrained [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.get_IsNever [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.get_Never [static method]: unit -> WoofWare.Myriad.Plugins.AdditionalProperties
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.get_Tag [method]: unit -> int
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.IsConstrained [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.IsNever [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.Never [static property]: [read-only] WoofWare.Myriad.Plugins.AdditionalProperties
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.NewConstrained [static method]: WoofWare.Myriad.Plugins.Definition -> WoofWare.Myriad.Plugins.AdditionalProperties
|
|
||||||
WoofWare.Myriad.Plugins.AdditionalProperties.Tag [property]: [read-only] int
|
|
||||||
WoofWare.Myriad.Plugins.ArgParserGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
WoofWare.Myriad.Plugins.ArgParserGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
||||||
WoofWare.Myriad.Plugins.ArgParserGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.ArgParserGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.ArrayTypeDefinition inherit obj, implements WoofWare.Myriad.Plugins.ArrayTypeDefinition System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.ArrayTypeDefinition..ctor [constructor]: WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.ArrayTypeDefinition.Equals [method]: (WoofWare.Myriad.Plugins.ArrayTypeDefinition, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ArrayTypeDefinition.get_Items [method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.ArrayTypeDefinition.Items [property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.ArrayTypeDefinition.Parse [static method]: System.Text.Json.Nodes.JsonNode -> WoofWare.Myriad.Plugins.ArrayTypeDefinition
|
|
||||||
WoofWare.Myriad.Plugins.CreateCatamorphismGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
WoofWare.Myriad.Plugins.CreateCatamorphismGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
||||||
WoofWare.Myriad.Plugins.CreateCatamorphismGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.CreateCatamorphismGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.Definition inherit obj, implements WoofWare.Myriad.Plugins.Definition System.IEquatable, System.Collections.IStructuralEquatable - union type with 8 cases
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Array inherit WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Array.get_Item [method]: unit -> WoofWare.Myriad.Plugins.ArrayTypeDefinition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Array.Item [property]: [read-only] WoofWare.Myriad.Plugins.ArrayTypeDefinition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Handle inherit WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Handle.get_Item [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Handle.Item [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Integer inherit WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Integer.format [property]: [read-only] string option
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Integer.get_format [method]: unit -> string option
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Object inherit WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Object.get_Item [method]: unit -> WoofWare.Myriad.Plugins.ObjectTypeDefinition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Object.Item [property]: [read-only] WoofWare.Myriad.Plugins.ObjectTypeDefinition
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags inherit obj
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.Array [static field]: int = 2
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.Boolean [static field]: int = 4
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.File [static field]: int = 7
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.Handle [static field]: int = 0
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.Integer [static field]: int = 6
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.Object [static field]: int = 1
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.String [static field]: int = 3
|
|
||||||
WoofWare.Myriad.Plugins.Definition+Tags.Unspecified [static field]: int = 5
|
|
||||||
WoofWare.Myriad.Plugins.Definition.Boolean [static property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.Equals [method]: (WoofWare.Myriad.Plugins.Definition, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.File [static property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_Boolean [static method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_File [static method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsArray [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsBoolean [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsFile [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsHandle [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsInteger [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsObject [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsString [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_IsUnspecified [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_String [static method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_Tag [method]: unit -> int
|
|
||||||
WoofWare.Myriad.Plugins.Definition.get_Unspecified [static method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsArray [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsBoolean [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsFile [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsHandle [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsInteger [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsObject [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsString [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.IsUnspecified [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.Definition.NewArray [static method]: WoofWare.Myriad.Plugins.ArrayTypeDefinition -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.NewHandle [static method]: string -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.NewInteger [static method]: string option -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.NewObject [static method]: WoofWare.Myriad.Plugins.ObjectTypeDefinition -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.String [static property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Definition.Tag [property]: [read-only] int
|
|
||||||
WoofWare.Myriad.Plugins.Definition.Unspecified [static property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.HttpClientGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
WoofWare.Myriad.Plugins.HttpClientGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
||||||
WoofWare.Myriad.Plugins.HttpClientGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.HttpClientGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.HttpMethod inherit obj, implements WoofWare.Myriad.Plugins.HttpMethod System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.Myriad.Plugins.HttpMethod System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 8 cases
|
WoofWare.Myriad.Plugins.HttpMethod inherit obj, implements WoofWare.Myriad.Plugins.HttpMethod System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.Myriad.Plugins.HttpMethod System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 8 cases
|
||||||
@@ -134,162 +57,7 @@ WoofWare.Myriad.Plugins.JsonParseGenerator inherit obj, implements Myriad.Core.I
|
|||||||
WoofWare.Myriad.Plugins.JsonParseGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.JsonParseGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.JsonSerializeGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
WoofWare.Myriad.Plugins.JsonSerializeGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
||||||
WoofWare.Myriad.Plugins.JsonSerializeGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.JsonSerializeGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.MimeType inherit obj, implements WoofWare.Myriad.Plugins.MimeType System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.Myriad.Plugins.MimeType System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 1 cases
|
|
||||||
WoofWare.Myriad.Plugins.MimeType.Equals [method]: (WoofWare.Myriad.Plugins.MimeType, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.MimeType.get_Item [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.MimeType.get_Tag [method]: unit -> int
|
|
||||||
WoofWare.Myriad.Plugins.MimeType.Item [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.MimeType.NewMimeType [static method]: string -> WoofWare.Myriad.Plugins.MimeType
|
|
||||||
WoofWare.Myriad.Plugins.MimeType.Tag [property]: [read-only] int
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition inherit obj, implements WoofWare.Myriad.Plugins.ObjectTypeDefinition System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition..ctor [constructor]: (string option, Map<string, WoofWare.Myriad.Plugins.Definition> option, Map<string, System.Text.Json.Nodes.JsonNode>, string list option, WoofWare.Myriad.Plugins.AdditionalProperties option, System.Text.Json.Nodes.JsonObject option)
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.AdditionalProperties [property]: [read-only] WoofWare.Myriad.Plugins.AdditionalProperties option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Description [property]: [read-only] string option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Equals [method]: (WoofWare.Myriad.Plugins.ObjectTypeDefinition, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Example [property]: [read-only] System.Text.Json.Nodes.JsonObject option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Extras [property]: [read-only] Map<string, System.Text.Json.Nodes.JsonNode>
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.get_AdditionalProperties [method]: unit -> WoofWare.Myriad.Plugins.AdditionalProperties option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.get_Description [method]: unit -> string option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.get_Example [method]: unit -> System.Text.Json.Nodes.JsonObject option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.get_Extras [method]: unit -> Map<string, System.Text.Json.Nodes.JsonNode>
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.get_Properties [method]: unit -> Map<string, WoofWare.Myriad.Plugins.Definition> option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.get_Required [method]: unit -> string list option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.ObjectTypeDefinition
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Properties [property]: [read-only] Map<string, WoofWare.Myriad.Plugins.Definition> option
|
|
||||||
WoofWare.Myriad.Plugins.ObjectTypeDefinition.Required [property]: [read-only] string list option
|
|
||||||
WoofWare.Myriad.Plugins.OperationId inherit obj, implements WoofWare.Myriad.Plugins.OperationId System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.Myriad.Plugins.OperationId System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 1 cases
|
|
||||||
WoofWare.Myriad.Plugins.OperationId.Equals [method]: (WoofWare.Myriad.Plugins.OperationId, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.OperationId.get_Item [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.OperationId.get_Tag [method]: unit -> int
|
|
||||||
WoofWare.Myriad.Plugins.OperationId.Item [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.OperationId.NewOperationId [static method]: string -> WoofWare.Myriad.Plugins.OperationId
|
|
||||||
WoofWare.Myriad.Plugins.OperationId.Tag [property]: [read-only] int
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn inherit obj, implements WoofWare.Myriad.Plugins.ParameterIn System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.Myriad.Plugins.ParameterIn System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 4 cases
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Path inherit WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Path.get_name [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Path.name [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Query inherit WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Query.get_name [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Query.name [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Tags inherit obj
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Tags.Body [static field]: int = 2
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Tags.Path [static field]: int = 0
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Tags.Query [static field]: int = 1
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Tags.Unrecognised [static field]: int = 3
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Unrecognised inherit WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Unrecognised.get_name [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Unrecognised.get_op [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Unrecognised.name [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn+Unrecognised.op [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.Body [static property]: [read-only] WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.Equals [method]: (WoofWare.Myriad.Plugins.ParameterIn, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.get_Body [static method]: unit -> WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.get_IsBody [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.get_IsPath [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.get_IsQuery [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.get_IsUnrecognised [method]: unit -> bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.get_Tag [method]: unit -> int
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.IsBody [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.IsPath [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.IsQuery [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.IsUnrecognised [property]: [read-only] bool
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.NewPath [static method]: string -> WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.NewQuery [static method]: string -> WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.NewUnrecognised [static method]: (string, string) -> WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.ParameterIn.Tag [property]: [read-only] int
|
|
||||||
WoofWare.Myriad.Plugins.RemoveOptionsGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
WoofWare.Myriad.Plugins.RemoveOptionsGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
||||||
WoofWare.Myriad.Plugins.RemoveOptionsGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.RemoveOptionsGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.Response inherit obj, implements WoofWare.Myriad.Plugins.Response System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.Response..ctor [constructor]: (string, WoofWare.Myriad.Plugins.Definition)
|
|
||||||
WoofWare.Myriad.Plugins.Response.Description [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.Response.Equals [method]: (WoofWare.Myriad.Plugins.Response, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Response.get_Description [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.Response.get_Schema [method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Response.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.Response
|
|
||||||
WoofWare.Myriad.Plugins.Response.Schema [property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.Scheme inherit obj, implements WoofWare.Myriad.Plugins.Scheme System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.Myriad.Plugins.Scheme System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 1 cases
|
|
||||||
WoofWare.Myriad.Plugins.Scheme.Equals [method]: (WoofWare.Myriad.Plugins.Scheme, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Scheme.get_Item [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.Scheme.get_Tag [method]: unit -> int
|
|
||||||
WoofWare.Myriad.Plugins.Scheme.Item [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.Scheme.NewScheme [static method]: string -> WoofWare.Myriad.Plugins.Scheme
|
|
||||||
WoofWare.Myriad.Plugins.Scheme.Tag [property]: [read-only] int
|
|
||||||
WoofWare.Myriad.Plugins.Swagger inherit obj, implements WoofWare.Myriad.Plugins.Swagger System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.Swagger..ctor [constructor]: (WoofWare.Myriad.Plugins.MimeType list, WoofWare.Myriad.Plugins.MimeType list, WoofWare.Myriad.Plugins.Scheme list, System.Version, WoofWare.Myriad.Plugins.SwaggerInfo, string, Map<string, Map<WoofWare.Myriad.Plugins.HttpMethod, WoofWare.Myriad.Plugins.SwaggerEndpoint>>, Map<string, WoofWare.Myriad.Plugins.Definition>, Map<string, WoofWare.Myriad.Plugins.Response>)
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.BasePath [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Consumes [property]: [read-only] WoofWare.Myriad.Plugins.MimeType list
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Definitions [property]: [read-only] Map<string, WoofWare.Myriad.Plugins.Definition>
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Equals [method]: (WoofWare.Myriad.Plugins.Swagger, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_BasePath [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Consumes [method]: unit -> WoofWare.Myriad.Plugins.MimeType list
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Definitions [method]: unit -> Map<string, WoofWare.Myriad.Plugins.Definition>
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Info [method]: unit -> WoofWare.Myriad.Plugins.SwaggerInfo
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Paths [method]: unit -> Map<string, Map<WoofWare.Myriad.Plugins.HttpMethod, WoofWare.Myriad.Plugins.SwaggerEndpoint>>
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Produces [method]: unit -> WoofWare.Myriad.Plugins.MimeType list
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Responses [method]: unit -> Map<string, WoofWare.Myriad.Plugins.Response>
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Schemes [method]: unit -> WoofWare.Myriad.Plugins.Scheme list
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.get_Swagger [method]: unit -> System.Version
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Info [property]: [read-only] WoofWare.Myriad.Plugins.SwaggerInfo
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Paths [property]: [read-only] Map<string, Map<WoofWare.Myriad.Plugins.HttpMethod, WoofWare.Myriad.Plugins.SwaggerEndpoint>>
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Produces [property]: [read-only] WoofWare.Myriad.Plugins.MimeType list
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Responses [property]: [read-only] Map<string, WoofWare.Myriad.Plugins.Response>
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Schemes [property]: [read-only] WoofWare.Myriad.Plugins.Scheme list
|
|
||||||
WoofWare.Myriad.Plugins.Swagger.Swagger [property]: [read-only] System.Version
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerClientGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
WoofWare.Myriad.Plugins.SwaggerClientGenerator inherit obj, implements Myriad.Core.IMyriadGenerator
|
||||||
WoofWare.Myriad.Plugins.SwaggerClientGenerator..ctor [constructor]: unit
|
WoofWare.Myriad.Plugins.SwaggerClientGenerator..ctor [constructor]: unit
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint inherit obj, implements WoofWare.Myriad.Plugins.SwaggerEndpoint System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint..ctor [constructor]: (WoofWare.Myriad.Plugins.MimeType list option, WoofWare.Myriad.Plugins.MimeType list option, string list, string, WoofWare.Myriad.Plugins.OperationId, WoofWare.Myriad.Plugins.SwaggerParameter list option, Map<int, WoofWare.Myriad.Plugins.Definition>)
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Consumes [property]: [read-only] WoofWare.Myriad.Plugins.MimeType list option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Equals [method]: (WoofWare.Myriad.Plugins.SwaggerEndpoint, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_Consumes [method]: unit -> WoofWare.Myriad.Plugins.MimeType list option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_OperationId [method]: unit -> WoofWare.Myriad.Plugins.OperationId
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_Parameters [method]: unit -> WoofWare.Myriad.Plugins.SwaggerParameter list option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_Produces [method]: unit -> WoofWare.Myriad.Plugins.MimeType list option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_Responses [method]: unit -> Map<int, WoofWare.Myriad.Plugins.Definition>
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_Summary [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.get_Tags [method]: unit -> string list
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.OperationId [property]: [read-only] WoofWare.Myriad.Plugins.OperationId
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Parameters [property]: [read-only] WoofWare.Myriad.Plugins.SwaggerParameter list option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.SwaggerEndpoint
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Produces [property]: [read-only] WoofWare.Myriad.Plugins.MimeType list option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Responses [property]: [read-only] Map<int, WoofWare.Myriad.Plugins.Definition>
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Summary [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerEndpoint.Tags [property]: [read-only] string list
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo inherit obj, implements WoofWare.Myriad.Plugins.SwaggerInfo System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo..ctor [constructor]: (string, string, WoofWare.Myriad.Plugins.SwaggerLicense, System.Version)
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.Description [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.Equals [method]: (WoofWare.Myriad.Plugins.SwaggerInfo, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.get_Description [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.get_License [method]: unit -> WoofWare.Myriad.Plugins.SwaggerLicense
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.get_Title [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.get_Version [method]: unit -> System.Version
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.License [property]: [read-only] WoofWare.Myriad.Plugins.SwaggerLicense
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.SwaggerInfo
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.Title [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerInfo.Version [property]: [read-only] System.Version
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense inherit obj, implements WoofWare.Myriad.Plugins.SwaggerLicense System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense..ctor [constructor]: (string, System.Uri option, string option)
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.Equals [method]: (WoofWare.Myriad.Plugins.SwaggerLicense, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.get_Identifier [method]: unit -> string option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.get_Name [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.get_Url [method]: unit -> System.Uri option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.Identifier [property]: [read-only] string option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.Name [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.SwaggerLicense
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerLicense.Url [property]: [read-only] System.Uri option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerModule inherit obj
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerModule.parse [static method]: string -> WoofWare.Myriad.Plugins.Swagger
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter inherit obj, implements WoofWare.Myriad.Plugins.SwaggerParameter System.IEquatable, System.Collections.IStructuralEquatable
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter..ctor [constructor]: (WoofWare.Myriad.Plugins.Definition, string option, WoofWare.Myriad.Plugins.ParameterIn, string, bool option)
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.Description [property]: [read-only] string option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.Equals [method]: (WoofWare.Myriad.Plugins.SwaggerParameter, System.Collections.IEqualityComparer) -> bool
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.get_Description [method]: unit -> string option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.get_In [method]: unit -> WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.get_Name [method]: unit -> string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.get_Required [method]: unit -> bool option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.get_Type [method]: unit -> WoofWare.Myriad.Plugins.Definition
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.In [property]: [read-only] WoofWare.Myriad.Plugins.ParameterIn
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.Name [property]: [read-only] string
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.Parse [static method]: System.Text.Json.Nodes.JsonObject -> WoofWare.Myriad.Plugins.SwaggerParameter
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.Required [property]: [read-only] bool option
|
|
||||||
WoofWare.Myriad.Plugins.SwaggerParameter.Type [property]: [read-only] WoofWare.Myriad.Plugins.Definition
|
|
@@ -24,10 +24,10 @@ type internal Endpoint =
|
|||||||
{
|
{
|
||||||
DocString : PreXmlDoc
|
DocString : PreXmlDoc
|
||||||
Produces : Produces
|
Produces : Produces
|
||||||
ReturnType : Definition
|
ReturnType : SwaggerV2.Definition
|
||||||
Method : WoofWare.Myriad.Plugins.HttpMethod
|
Method : WoofWare.Myriad.Plugins.HttpMethod
|
||||||
Operation : OperationId
|
Operation : SwaggerV2.OperationId
|
||||||
Parameters : SwaggerParameter list
|
Parameters : SwaggerV2.SwaggerParameter list
|
||||||
Endpoint : string
|
Endpoint : string
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -42,7 +42,7 @@ type internal TypeEntry =
|
|||||||
type internal Types =
|
type internal Types =
|
||||||
{
|
{
|
||||||
ByHandle : IReadOnlyDictionary<string, TypeEntry>
|
ByHandle : IReadOnlyDictionary<string, TypeEntry>
|
||||||
ByDefinition : IReadOnlyDictionary<Definition, TypeEntry>
|
ByDefinition : IReadOnlyDictionary<SwaggerV2.Definition, TypeEntry>
|
||||||
}
|
}
|
||||||
|
|
||||||
[<RequireQualifiedAccess>]
|
[<RequireQualifiedAccess>]
|
||||||
@@ -50,33 +50,33 @@ module internal SwaggerClientGenerator =
|
|||||||
|
|
||||||
let internal log (_ : string) = ()
|
let internal log (_ : string) = ()
|
||||||
|
|
||||||
let renderType (types : Types) (defn : Definition) : SynType option =
|
let renderType (types : Types) (defn : SwaggerV2.Definition) : SynType option =
|
||||||
match types.ByDefinition.TryGetValue defn with
|
match types.ByDefinition.TryGetValue defn with
|
||||||
| true, v -> Some v.Signature
|
| true, v -> Some v.Signature
|
||||||
| false, _ ->
|
| false, _ ->
|
||||||
|
|
||||||
match defn with
|
match defn with
|
||||||
| Definition.Handle h ->
|
| SwaggerV2.Definition.Handle h ->
|
||||||
match types.ByHandle.TryGetValue h with
|
match types.ByHandle.TryGetValue h with
|
||||||
| false, _ -> None
|
| false, _ -> None
|
||||||
| true, v -> Some v.Signature
|
| true, v -> Some v.Signature
|
||||||
| Definition.Object _ -> failwith "should not hit"
|
| SwaggerV2.Definition.Object _ -> failwith "should not hit"
|
||||||
| Definition.Array _ -> failwith "should not hit"
|
| SwaggerV2.Definition.Array _ -> failwith "should not hit"
|
||||||
| Definition.Unspecified -> failwith "should not hit"
|
| SwaggerV2.Definition.Unspecified -> failwith "should not hit"
|
||||||
| Definition.String -> SynType.string |> Some
|
| SwaggerV2.Definition.String -> SynType.string |> Some
|
||||||
| Definition.Boolean -> SynType.bool |> Some
|
| SwaggerV2.Definition.Boolean -> SynType.bool |> Some
|
||||||
| Definition.Integer _ -> SynType.int |> Some
|
| SwaggerV2.Definition.Integer _ -> SynType.int |> Some
|
||||||
| Definition.File -> SynType.createLongIdent' [ "System" ; "IO" ; "Stream" ] |> Some
|
| SwaggerV2.Definition.File -> SynType.createLongIdent' [ "System" ; "IO" ; "Stream" ] |> Some
|
||||||
|
|
||||||
/// Returns None if we lacked the information required to do this.
|
/// Returns None if we lacked the information required to do this.
|
||||||
/// bigCache is a map of e.g. {"securityDefinition": {Defn : F# type}}.
|
/// bigCache is a map of e.g. {"securityDefinition": {Defn : F# type}}.
|
||||||
let rec defnToType
|
let rec defnToType
|
||||||
(anonymousTypeCount : int ref)
|
(anonymousTypeCount : int ref)
|
||||||
(handlesMap : Dictionary<string, TypeEntry>)
|
(handlesMap : Dictionary<string, TypeEntry>)
|
||||||
(bigCache : Dictionary<string, Dictionary<Definition, TypeEntry>>)
|
(bigCache : Dictionary<string, Dictionary<SwaggerV2.Definition, TypeEntry>>)
|
||||||
(thisKey : string)
|
(thisKey : string)
|
||||||
(typeName : string option)
|
(typeName : string option)
|
||||||
(d : Definition)
|
(d : SwaggerV2.Definition)
|
||||||
: TypeEntry option
|
: TypeEntry option
|
||||||
=
|
=
|
||||||
let cache =
|
let cache =
|
||||||
@@ -111,7 +111,7 @@ module internal SwaggerClientGenerator =
|
|||||||
|
|
||||||
let result =
|
let result =
|
||||||
match d with
|
match d with
|
||||||
| Definition.Object obj ->
|
| SwaggerV2.Definition.Object obj ->
|
||||||
let requiredFields = obj.Required |> Option.defaultValue [] |> Set.ofList
|
let requiredFields = obj.Required |> Option.defaultValue [] |> Set.ofList
|
||||||
|
|
||||||
let namedProperties =
|
let namedProperties =
|
||||||
@@ -123,7 +123,7 @@ module internal SwaggerClientGenerator =
|
|||||||
// Special case for when this is a reference to this very type
|
// Special case for when this is a reference to this very type
|
||||||
let isOurself =
|
let isOurself =
|
||||||
match defn with
|
match defn with
|
||||||
| Definition.Handle h ->
|
| SwaggerV2.Definition.Handle h ->
|
||||||
match h.Split '/' with
|
match h.Split '/' with
|
||||||
| [| "#" ; location ; ty |] when location = thisKey && Some ty = typeName ->
|
| [| "#" ; location ; ty |] when location = thisKey && Some ty = typeName ->
|
||||||
SynType.named ty |> Some
|
SynType.named ty |> Some
|
||||||
@@ -198,8 +198,8 @@ module internal SwaggerClientGenerator =
|
|||||||
|> SynField.make
|
|> SynField.make
|
||||||
|> List.singleton
|
|> List.singleton
|
||||||
|> Some
|
|> Some
|
||||||
| Some AdditionalProperties.Never -> Some []
|
| Some SwaggerV2.AdditionalProperties.Never -> Some []
|
||||||
| Some (AdditionalProperties.Constrained defn) ->
|
| Some (SwaggerV2.AdditionalProperties.Constrained defn) ->
|
||||||
let defn' = defnToType anonymousTypeCount handlesMap bigCache thisKey None defn
|
let defn' = defnToType anonymousTypeCount handlesMap bigCache thisKey None defn
|
||||||
|
|
||||||
match defn' with
|
match defn' with
|
||||||
@@ -277,7 +277,7 @@ module internal SwaggerClientGenerator =
|
|||||||
|
|
||||||
defn |> Some
|
defn |> Some
|
||||||
|
|
||||||
| Definition.Array elt ->
|
| SwaggerV2.Definition.Array elt ->
|
||||||
let child = defnToType anonymousTypeCount handlesMap bigCache thisKey None elt.Items
|
let child = defnToType anonymousTypeCount handlesMap bigCache thisKey None elt.Items
|
||||||
|
|
||||||
match child with
|
match child with
|
||||||
@@ -290,37 +290,37 @@ module internal SwaggerClientGenerator =
|
|||||||
}
|
}
|
||||||
|
|
||||||
Some defn
|
Some defn
|
||||||
| Definition.String ->
|
| SwaggerV2.Definition.String ->
|
||||||
{
|
{
|
||||||
Signature = SynType.string
|
Signature = SynType.string
|
||||||
FSharpDefinition = None
|
FSharpDefinition = None
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| Definition.Boolean ->
|
| SwaggerV2.Definition.Boolean ->
|
||||||
{
|
{
|
||||||
Signature = SynType.bool
|
Signature = SynType.bool
|
||||||
FSharpDefinition = None
|
FSharpDefinition = None
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| Definition.Unspecified ->
|
| SwaggerV2.Definition.Unspecified ->
|
||||||
{
|
{
|
||||||
Signature = SynType.unit
|
Signature = SynType.unit
|
||||||
FSharpDefinition = None
|
FSharpDefinition = None
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| Definition.Integer _ ->
|
| SwaggerV2.Definition.Integer _ ->
|
||||||
{
|
{
|
||||||
Signature = SynType.createLongIdent' [ "int" ]
|
Signature = SynType.createLongIdent' [ "int" ]
|
||||||
FSharpDefinition = None
|
FSharpDefinition = None
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| Definition.File ->
|
| SwaggerV2.Definition.File ->
|
||||||
{
|
{
|
||||||
Signature = SynType.createLongIdent' [ "System" ; "IO" ; "Stream" ]
|
Signature = SynType.createLongIdent' [ "System" ; "IO" ; "Stream" ]
|
||||||
FSharpDefinition = None
|
FSharpDefinition = None
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| Definition.Handle s ->
|
| SwaggerV2.Definition.Handle s ->
|
||||||
let split = s.Split '/' |> List.ofArray
|
let split = s.Split '/' |> List.ofArray
|
||||||
|
|
||||||
match split with
|
match split with
|
||||||
@@ -382,14 +382,14 @@ module internal SwaggerClientGenerator =
|
|||||||
|> List.map (fun par ->
|
|> List.map (fun par ->
|
||||||
let inParam =
|
let inParam =
|
||||||
match par.In with
|
match par.In with
|
||||||
| ParameterIn.Unrecognised (f, name) ->
|
| SwaggerV2.ParameterIn.Unrecognised (f, name) ->
|
||||||
log
|
log
|
||||||
$"Skipping %O{ep.Operation} at %s{ep.Endpoint}: unrecognised In parameter %s{f} with name %s{name}"
|
$"Skipping %O{ep.Operation} at %s{ep.Endpoint}: unrecognised In parameter %s{f} with name %s{name}"
|
||||||
|
|
||||||
None
|
None
|
||||||
| ParameterIn.Body -> Some IsIn.Body
|
| SwaggerV2.ParameterIn.Body -> Some IsIn.Body
|
||||||
| ParameterIn.Query name -> Some (IsIn.Query name)
|
| SwaggerV2.ParameterIn.Query name -> Some (IsIn.Query name)
|
||||||
| ParameterIn.Path name -> Some (IsIn.Path name)
|
| SwaggerV2.ParameterIn.Path name -> Some (IsIn.Path name)
|
||||||
|
|
||||||
match inParam with
|
match inParam with
|
||||||
| None -> None
|
| None -> None
|
||||||
@@ -515,6 +515,200 @@ module internal SwaggerClientGenerator =
|
|||||||
open Myriad.Core
|
open Myriad.Core
|
||||||
open System.IO
|
open System.IO
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module internal SwaggerV2Generator =
|
||||||
|
let generate (pars : Map<string, string>) (contents : SwaggerV2.SwaggerV2) : Output =
|
||||||
|
let scheme =
|
||||||
|
let preferred = SwaggerV2.Scheme "https"
|
||||||
|
|
||||||
|
if List.isEmpty contents.Schemes then
|
||||||
|
failwith "no schemes specified in API spec!"
|
||||||
|
|
||||||
|
if List.contains preferred contents.Schemes then
|
||||||
|
preferred
|
||||||
|
else
|
||||||
|
List.head contents.Schemes
|
||||||
|
|
||||||
|
let clientDocstring = contents.Info.Description |> PreXmlDoc.create
|
||||||
|
|
||||||
|
let basePath = contents.BasePath
|
||||||
|
|
||||||
|
let typeDefs =
|
||||||
|
let bigCache = Dictionary<_, Dictionary<_, _>> ()
|
||||||
|
|
||||||
|
let countAll () =
|
||||||
|
(0, bigCache) ||> Seq.fold (fun count (KeyValue (_, v)) -> count + v.Count)
|
||||||
|
|
||||||
|
let byHandle = Dictionary ()
|
||||||
|
let anonymousTypeCount = ref 0
|
||||||
|
|
||||||
|
let rec go (contents : ((string * SwaggerV2.Definition) * string) list) =
|
||||||
|
let lastRound = countAll ()
|
||||||
|
|
||||||
|
contents
|
||||||
|
|> List.filter (fun ((name, defn), defnClass) ->
|
||||||
|
let doIt =
|
||||||
|
SwaggerClientGenerator.defnToType
|
||||||
|
anonymousTypeCount
|
||||||
|
byHandle
|
||||||
|
bigCache
|
||||||
|
defnClass
|
||||||
|
(Some name)
|
||||||
|
defn
|
||||||
|
|
||||||
|
match doIt with
|
||||||
|
| None -> true
|
||||||
|
| Some _ -> false
|
||||||
|
)
|
||||||
|
|> fun remaining ->
|
||||||
|
if not remaining.IsEmpty then
|
||||||
|
let currentCount = countAll ()
|
||||||
|
|
||||||
|
if currentCount = lastRound then
|
||||||
|
for (name, remaining), kind in remaining do
|
||||||
|
SwaggerClientGenerator.log $"Remaining: %s{name} (%s{kind})"
|
||||||
|
|
||||||
|
SwaggerClientGenerator.log "--------"
|
||||||
|
|
||||||
|
for KeyValue (handle, defn) in byHandle do
|
||||||
|
SwaggerClientGenerator.log $"Known: %s{handle} %O{defn}"
|
||||||
|
|
||||||
|
// TODO: ohh noooooo the Gitea spec is genuinely circular,
|
||||||
|
// it's impossible to construct a Repository type
|
||||||
|
// we're going to have to somehow detect this case and break the cycle
|
||||||
|
// by artificially making a property optional
|
||||||
|
// :sob: Gitea why are you like this
|
||||||
|
// failwith "Made no further progress rendering types"
|
||||||
|
()
|
||||||
|
else
|
||||||
|
go remaining
|
||||||
|
|
||||||
|
seq {
|
||||||
|
for defnClass in [ "definitions" ; "responses" ] do
|
||||||
|
match defnClass with
|
||||||
|
| "definitions" ->
|
||||||
|
for KeyValue (k, v) in contents.Definitions do
|
||||||
|
yield (k, v), defnClass
|
||||||
|
| "responses" ->
|
||||||
|
for KeyValue (k, v) in contents.Responses do
|
||||||
|
yield (k, v.Schema), defnClass
|
||||||
|
| _ -> failwith "oh no"
|
||||||
|
}
|
||||||
|
|> Seq.toList
|
||||||
|
|> go
|
||||||
|
|
||||||
|
let result = Dictionary ()
|
||||||
|
|
||||||
|
for KeyValue (_container, types) in bigCache do
|
||||||
|
for KeyValue (defn, rendered) in types do
|
||||||
|
result.TryAdd (defn, rendered) |> ignore<bool>
|
||||||
|
|
||||||
|
{
|
||||||
|
ByHandle = byHandle
|
||||||
|
ByDefinition = result :> IReadOnlyDictionary<_, _>
|
||||||
|
}
|
||||||
|
|
||||||
|
let summary =
|
||||||
|
contents.Paths
|
||||||
|
|> Seq.collect (fun (KeyValue (path, endpoints)) ->
|
||||||
|
endpoints
|
||||||
|
|> Seq.choose (fun (KeyValue (method, endpoint)) ->
|
||||||
|
let docstring = endpoint.Summary |> PreXmlDoc.create
|
||||||
|
|
||||||
|
let produces =
|
||||||
|
match endpoint.Produces with
|
||||||
|
| None -> Produces.Produces "json"
|
||||||
|
| Some [] -> failwith $"API specified empty Produces: %s{path} (%O{method})"
|
||||||
|
| Some [ SwaggerV2.MimeType "application/octet-stream" ] -> Produces.OctetStream
|
||||||
|
| Some [ SwaggerV2.MimeType "application/json" ] -> Produces.Produces "json"
|
||||||
|
| Some [ SwaggerV2.MimeType (StartsWith "text/" t) ] -> Produces.Produces t
|
||||||
|
| Some [ SwaggerV2.MimeType s ] ->
|
||||||
|
failwithf
|
||||||
|
$"we don't support non-JSON Produces right now, got: %s{s} (%s{path} %O{method})"
|
||||||
|
| Some (_ :: _) ->
|
||||||
|
failwith $"we don't support multiple Produces right now, at %s{path} (%O{method})"
|
||||||
|
|
||||||
|
let returnType =
|
||||||
|
endpoint.Responses
|
||||||
|
|> Seq.choose (fun (KeyValue (response, defn)) ->
|
||||||
|
if 200 <= response && response < 300 then
|
||||||
|
Some defn
|
||||||
|
else
|
||||||
|
None
|
||||||
|
)
|
||||||
|
|> Seq.toList
|
||||||
|
|
||||||
|
let returnType =
|
||||||
|
match returnType with
|
||||||
|
| [ t ] -> Some t
|
||||||
|
| [] -> failwith $"got no successful response results, %s{path} %O{method}"
|
||||||
|
| _ ->
|
||||||
|
SwaggerClientGenerator.log
|
||||||
|
$"Ignoring %s{path} %O{method} due to multiple success responses"
|
||||||
|
// can't be bothered to work out how to deal with multiple success
|
||||||
|
// results right now
|
||||||
|
None
|
||||||
|
|
||||||
|
match returnType with
|
||||||
|
| None -> None
|
||||||
|
| Some returnType ->
|
||||||
|
|
||||||
|
{
|
||||||
|
Method = method
|
||||||
|
Produces = produces
|
||||||
|
DocString = docstring
|
||||||
|
ReturnType = returnType
|
||||||
|
Operation = endpoint.OperationId
|
||||||
|
Parameters = endpoint.Parameters |> Option.defaultValue []
|
||||||
|
Endpoint = path
|
||||||
|
}
|
||||||
|
|> Some
|
||||||
|
)
|
||||||
|
|> Seq.toList
|
||||||
|
)
|
||||||
|
|> Seq.toList
|
||||||
|
|
||||||
|
let config =
|
||||||
|
let createMock =
|
||||||
|
match Map.tryFind "GENERATEMOCKVISIBILITY" pars with
|
||||||
|
| None -> None
|
||||||
|
| Some v ->
|
||||||
|
match v.ToLowerInvariant () with
|
||||||
|
| "internal" -> Some true
|
||||||
|
| "public" -> Some false
|
||||||
|
| _ ->
|
||||||
|
failwith
|
||||||
|
$"Expected GenerateMockVisibility parameter to be 'internal' or 'public', but was: '%s{v.ToLowerInvariant ()}'"
|
||||||
|
|
||||||
|
let className =
|
||||||
|
match Map.tryFind "CLASSNAME" pars with
|
||||||
|
| None -> failwith "You must supply the <ClassName /> parameter in <MyriadParams />."
|
||||||
|
| Some v -> v
|
||||||
|
|
||||||
|
{
|
||||||
|
CreateMock = createMock
|
||||||
|
ClassName = className
|
||||||
|
}
|
||||||
|
|
||||||
|
let ty =
|
||||||
|
SwaggerClientGenerator.computeType config basePath typeDefs clientDocstring summary
|
||||||
|
|
||||||
|
[
|
||||||
|
yield
|
||||||
|
SynModuleDecl.Open (
|
||||||
|
SynOpenDeclTarget.ModuleOrNamespace (
|
||||||
|
SynLongIdent.createS' [ "WoofWare" ; "Myriad" ; "Plugins" ],
|
||||||
|
range0
|
||||||
|
),
|
||||||
|
range0
|
||||||
|
)
|
||||||
|
yield SwaggerClientGenerator.instantiateRequiredTypes typeDefs
|
||||||
|
yield! ty
|
||||||
|
]
|
||||||
|
|> SynModuleOrNamespace.createNamespace [ Ident.create config.ClassName ]
|
||||||
|
|> List.singleton
|
||||||
|
|> Output.Ast
|
||||||
|
|
||||||
/// Myriad generator that stamps out an interface and class to access a Swagger-specified API.
|
/// Myriad generator that stamps out an interface and class to access a Swagger-specified API.
|
||||||
[<MyriadGenerator("swagger-client")>]
|
[<MyriadGenerator("swagger-client")>]
|
||||||
type SwaggerClientGenerator () =
|
type SwaggerClientGenerator () =
|
||||||
@@ -523,206 +717,19 @@ type SwaggerClientGenerator () =
|
|||||||
member _.ValidInputExtensions = [ ".json" ]
|
member _.ValidInputExtensions = [ ".json" ]
|
||||||
|
|
||||||
member _.Generate (context : GeneratorContext) =
|
member _.Generate (context : GeneratorContext) =
|
||||||
let contents = File.ReadAllText context.InputFilename |> Swagger.parse
|
let pars = MyriadParamParser.render context.AdditionalParameters
|
||||||
|
|
||||||
let scheme =
|
let pars =
|
||||||
let preferred = Scheme "https"
|
pars
|
||||||
|
|> Map.toSeq
|
||||||
|
|> Seq.map (fun (k, v) -> k.ToUpperInvariant (), v)
|
||||||
|
|> Map.ofSeq
|
||||||
|
|
||||||
if List.isEmpty contents.Schemes then
|
if pars.IsEmpty then
|
||||||
failwith "no schemes specified in API spec!"
|
failwith "No parameters given. You must supply the <ClassName /> parameter in <MyriadParams />."
|
||||||
|
|
||||||
if List.contains preferred contents.Schemes then
|
let contents = File.ReadAllText context.InputFilename |> SwaggerV2.parse
|
||||||
preferred
|
|
||||||
else
|
|
||||||
List.head contents.Schemes
|
|
||||||
|
|
||||||
let clientDocstring = contents.Info.Description |> PreXmlDoc.create
|
match contents with
|
||||||
|
| Ok contents -> SwaggerV2Generator.generate pars contents
|
||||||
let basePath = contents.BasePath
|
| Error node -> failwith "Input was not a Swagger 2 spec"
|
||||||
|
|
||||||
let typeDefs =
|
|
||||||
let bigCache = Dictionary<_, Dictionary<_, _>> ()
|
|
||||||
|
|
||||||
let countAll () =
|
|
||||||
(0, bigCache) ||> Seq.fold (fun count (KeyValue (_, v)) -> count + v.Count)
|
|
||||||
|
|
||||||
let byHandle = Dictionary ()
|
|
||||||
let anonymousTypeCount = ref 0
|
|
||||||
|
|
||||||
let rec go (contents : ((string * Definition) * string) list) =
|
|
||||||
let lastRound = countAll ()
|
|
||||||
|
|
||||||
contents
|
|
||||||
|> List.filter (fun ((name, defn), defnClass) ->
|
|
||||||
let doIt =
|
|
||||||
SwaggerClientGenerator.defnToType
|
|
||||||
anonymousTypeCount
|
|
||||||
byHandle
|
|
||||||
bigCache
|
|
||||||
defnClass
|
|
||||||
(Some name)
|
|
||||||
defn
|
|
||||||
|
|
||||||
match doIt with
|
|
||||||
| None -> true
|
|
||||||
| Some _ -> false
|
|
||||||
)
|
|
||||||
|> fun remaining ->
|
|
||||||
if not remaining.IsEmpty then
|
|
||||||
let currentCount = countAll ()
|
|
||||||
|
|
||||||
if currentCount = lastRound then
|
|
||||||
for (name, remaining), kind in remaining do
|
|
||||||
SwaggerClientGenerator.log $"Remaining: %s{name} (%s{kind})"
|
|
||||||
|
|
||||||
SwaggerClientGenerator.log "--------"
|
|
||||||
|
|
||||||
for KeyValue (handle, defn) in byHandle do
|
|
||||||
SwaggerClientGenerator.log $"Known: %s{handle} %O{defn}"
|
|
||||||
|
|
||||||
// TODO: ohh noooooo the Gitea spec is genuinely circular,
|
|
||||||
// it's impossible to construct a Repository type
|
|
||||||
// we're going to have to somehow detect this case and break the cycle
|
|
||||||
// by artificially making a property optional
|
|
||||||
// :sob: Gitea why are you like this
|
|
||||||
// failwith "Made no further progress rendering types"
|
|
||||||
()
|
|
||||||
else
|
|
||||||
go remaining
|
|
||||||
|
|
||||||
seq {
|
|
||||||
for defnClass in [ "definitions" ; "responses" ] do
|
|
||||||
match defnClass with
|
|
||||||
| "definitions" ->
|
|
||||||
for KeyValue (k, v) in contents.Definitions do
|
|
||||||
yield (k, v), defnClass
|
|
||||||
| "responses" ->
|
|
||||||
for KeyValue (k, v) in contents.Responses do
|
|
||||||
yield (k, v.Schema), defnClass
|
|
||||||
| _ -> failwith "oh no"
|
|
||||||
}
|
|
||||||
|> Seq.toList
|
|
||||||
|> go
|
|
||||||
|
|
||||||
let result = Dictionary ()
|
|
||||||
|
|
||||||
for KeyValue (_container, types) in bigCache do
|
|
||||||
for KeyValue (defn, rendered) in types do
|
|
||||||
result.TryAdd (defn, rendered) |> ignore<bool>
|
|
||||||
|
|
||||||
{
|
|
||||||
ByHandle = byHandle
|
|
||||||
ByDefinition = result :> IReadOnlyDictionary<_, _>
|
|
||||||
}
|
|
||||||
|
|
||||||
let summary =
|
|
||||||
contents.Paths
|
|
||||||
|> Seq.collect (fun (KeyValue (path, endpoints)) ->
|
|
||||||
endpoints
|
|
||||||
|> Seq.choose (fun (KeyValue (method, endpoint)) ->
|
|
||||||
let docstring = endpoint.Summary |> PreXmlDoc.create
|
|
||||||
|
|
||||||
let produces =
|
|
||||||
match endpoint.Produces with
|
|
||||||
| None -> Produces.Produces "json"
|
|
||||||
| Some [] -> failwith $"API specified empty Produces: %s{path} (%O{method})"
|
|
||||||
| Some [ MimeType "application/octet-stream" ] -> Produces.OctetStream
|
|
||||||
| Some [ MimeType "application/json" ] -> Produces.Produces "json"
|
|
||||||
| Some [ MimeType (StartsWith "text/" t) ] -> Produces.Produces t
|
|
||||||
| Some [ MimeType s ] ->
|
|
||||||
failwithf
|
|
||||||
$"we don't support non-JSON Produces right now, got: %s{s} (%s{path} %O{method})"
|
|
||||||
| Some (_ :: _) ->
|
|
||||||
failwith $"we don't support multiple Produces right now, at %s{path} (%O{method})"
|
|
||||||
|
|
||||||
let returnType =
|
|
||||||
endpoint.Responses
|
|
||||||
|> Seq.choose (fun (KeyValue (response, defn)) ->
|
|
||||||
if 200 <= response && response < 300 then
|
|
||||||
Some defn
|
|
||||||
else
|
|
||||||
None
|
|
||||||
)
|
|
||||||
|> Seq.toList
|
|
||||||
|
|
||||||
let returnType =
|
|
||||||
match returnType with
|
|
||||||
| [ t ] -> Some t
|
|
||||||
| [] -> failwith $"got no successful response results, %s{path} %O{method}"
|
|
||||||
| _ ->
|
|
||||||
SwaggerClientGenerator.log
|
|
||||||
$"Ignoring %s{path} %O{method} due to multiple success responses"
|
|
||||||
// can't be bothered to work out how to deal with multiple success
|
|
||||||
// results right now
|
|
||||||
None
|
|
||||||
|
|
||||||
match returnType with
|
|
||||||
| None -> None
|
|
||||||
| Some returnType ->
|
|
||||||
|
|
||||||
{
|
|
||||||
Method = method
|
|
||||||
Produces = produces
|
|
||||||
DocString = docstring
|
|
||||||
ReturnType = returnType
|
|
||||||
Operation = endpoint.OperationId
|
|
||||||
Parameters = endpoint.Parameters |> Option.defaultValue []
|
|
||||||
Endpoint = path
|
|
||||||
}
|
|
||||||
|> Some
|
|
||||||
)
|
|
||||||
|> Seq.toList
|
|
||||||
)
|
|
||||||
|> Seq.toList
|
|
||||||
|
|
||||||
let config =
|
|
||||||
let pars = MyriadParamParser.render context.AdditionalParameters
|
|
||||||
|
|
||||||
let pars =
|
|
||||||
pars
|
|
||||||
|> Map.toSeq
|
|
||||||
|> Seq.map (fun (k, v) -> k.ToUpperInvariant (), v)
|
|
||||||
|> Map.ofSeq
|
|
||||||
|
|
||||||
if pars.IsEmpty then
|
|
||||||
failwith "No parameters given. You must supply the <ClassName /> parameter in <MyriadParams />."
|
|
||||||
|
|
||||||
let createMock =
|
|
||||||
match Map.tryFind "GENERATEMOCKVISIBILITY" pars with
|
|
||||||
| None -> None
|
|
||||||
| Some v ->
|
|
||||||
match v.ToLowerInvariant () with
|
|
||||||
| "internal" -> Some true
|
|
||||||
| "public" -> Some false
|
|
||||||
| _ ->
|
|
||||||
failwith
|
|
||||||
$"Expected GenerateMockVisibility parameter to be 'internal' or 'public', but was: '%s{v.ToLowerInvariant ()}'"
|
|
||||||
|
|
||||||
let className =
|
|
||||||
match Map.tryFind "CLASSNAME" pars with
|
|
||||||
| None -> failwith "You must supply the <ClassName /> parameter in <MyriadParams />."
|
|
||||||
| Some v -> v
|
|
||||||
|
|
||||||
{
|
|
||||||
CreateMock = createMock
|
|
||||||
ClassName = className
|
|
||||||
}
|
|
||||||
|
|
||||||
let ty =
|
|
||||||
SwaggerClientGenerator.computeType config basePath typeDefs clientDocstring summary
|
|
||||||
|
|
||||||
[
|
|
||||||
yield
|
|
||||||
SynModuleDecl.Open (
|
|
||||||
SynOpenDeclTarget.ModuleOrNamespace (
|
|
||||||
SynLongIdent.createS' [ "WoofWare" ; "Myriad" ; "Plugins" ],
|
|
||||||
range0
|
|
||||||
),
|
|
||||||
range0
|
|
||||||
)
|
|
||||||
yield SwaggerClientGenerator.instantiateRequiredTypes typeDefs
|
|
||||||
yield! ty
|
|
||||||
]
|
|
||||||
|> SynModuleOrNamespace.createNamespace [ Ident.create config.ClassName ]
|
|
||||||
|> List.singleton
|
|
||||||
|> Output.Ast
|
|
||||||
|
@@ -1,53 +1,8 @@
|
|||||||
namespace WoofWare.Myriad.Plugins
|
module internal WoofWare.Myriad.Plugins.SwaggerV2
|
||||||
|
|
||||||
open System
|
open System
|
||||||
open System.Text.Json.Nodes
|
open System.Text.Json.Nodes
|
||||||
|
|
||||||
[<AutoOpen>]
|
|
||||||
module internal JsonHelpers =
|
|
||||||
let inline asString (n : JsonNode) (key : string) : string =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
|
||||||
| s -> s.GetValue<string> ()
|
|
||||||
|
|
||||||
[<RequiresExplicitTypeArguments>]
|
|
||||||
let inline asOpt<'ret> (n : JsonNode) (key : string) : 'ret option =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> None
|
|
||||||
| s -> s.GetValue<'ret> () |> Some
|
|
||||||
|
|
||||||
let inline asObj (n : JsonNode) (key : string) : JsonObject =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
|
||||||
| o -> o.AsObject ()
|
|
||||||
|
|
||||||
let inline asObjOpt (n : JsonNode) (key : string) : JsonObject option =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> None
|
|
||||||
| o -> o.AsObject () |> Some
|
|
||||||
|
|
||||||
let inline asArr (n : JsonNode) (key : string) : JsonArray =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
|
||||||
| o -> o.AsArray ()
|
|
||||||
|
|
||||||
let inline asArrOpt (n : JsonNode) (key : string) : JsonArray option =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> None
|
|
||||||
| o -> o.AsArray () |> Some
|
|
||||||
|
|
||||||
[<RequiresExplicitTypeArguments>]
|
|
||||||
let inline asArr'<'v> (n : JsonNode) (key : string) : 'v list =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> failwith $"Expected node to have a key '%s{key}', but it did not: %s{n.ToJsonString ()}"
|
|
||||||
| o -> o.AsArray () |> Seq.map (fun v -> v.GetValue<'v> ()) |> Seq.toList
|
|
||||||
|
|
||||||
[<RequiresExplicitTypeArguments>]
|
|
||||||
let inline asArrOpt'<'v> (n : JsonNode) (key : string) : 'v list option =
|
|
||||||
match n.[key] with
|
|
||||||
| null -> None
|
|
||||||
| o -> o.AsArray () |> Seq.map (fun v -> v.GetValue<'v> ()) |> Seq.toList |> Some
|
|
||||||
|
|
||||||
/// A MIME type, like "application/json"
|
/// A MIME type, like "application/json"
|
||||||
type MimeType =
|
type MimeType =
|
||||||
/// A MIME type, like "application/json"
|
/// A MIME type, like "application/json"
|
||||||
@@ -434,71 +389,8 @@ type Response =
|
|||||||
Schema = schema
|
Schema = schema
|
||||||
}
|
}
|
||||||
|
|
||||||
/// An HTTP method. This is System.Net.Http.HttpMethod, but
|
|
||||||
/// a proper discriminated union.
|
|
||||||
type HttpMethod =
|
|
||||||
/// HTTP Get
|
|
||||||
| Get
|
|
||||||
/// HTTP Post
|
|
||||||
| Post
|
|
||||||
/// HTTP Delete
|
|
||||||
| Delete
|
|
||||||
/// HTTP Patch
|
|
||||||
| Patch
|
|
||||||
/// HTTP Options
|
|
||||||
| Options
|
|
||||||
/// HTTP Head
|
|
||||||
| Head
|
|
||||||
/// HTTP Put
|
|
||||||
| Put
|
|
||||||
/// HTTP Trace
|
|
||||||
| Trace
|
|
||||||
|
|
||||||
/// Convert to the standard library's enum type.
|
|
||||||
member this.ToDotNet () : System.Net.Http.HttpMethod =
|
|
||||||
match this with
|
|
||||||
| HttpMethod.Get -> System.Net.Http.HttpMethod.Get
|
|
||||||
| HttpMethod.Post -> System.Net.Http.HttpMethod.Post
|
|
||||||
| HttpMethod.Delete -> System.Net.Http.HttpMethod.Delete
|
|
||||||
| HttpMethod.Patch -> System.Net.Http.HttpMethod.Patch
|
|
||||||
| HttpMethod.Options -> System.Net.Http.HttpMethod.Options
|
|
||||||
| HttpMethod.Head -> System.Net.Http.HttpMethod.Head
|
|
||||||
| HttpMethod.Put -> System.Net.Http.HttpMethod.Put
|
|
||||||
| HttpMethod.Trace -> System.Net.Http.HttpMethod.Trace
|
|
||||||
|
|
||||||
/// Human-readable string representation.
|
|
||||||
override this.ToString () : string =
|
|
||||||
match this with
|
|
||||||
| HttpMethod.Get -> "Get"
|
|
||||||
| HttpMethod.Post -> "Post"
|
|
||||||
| HttpMethod.Delete -> "Delete"
|
|
||||||
| HttpMethod.Patch -> "Patch"
|
|
||||||
| HttpMethod.Options -> "Options"
|
|
||||||
| HttpMethod.Head -> "Head"
|
|
||||||
| HttpMethod.Put -> "Put"
|
|
||||||
| HttpMethod.Trace -> "Trace"
|
|
||||||
|
|
||||||
/// Throws on invalid inputs.
|
|
||||||
static member Parse (s : string) : HttpMethod =
|
|
||||||
if String.Equals (s, "get", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Get
|
|
||||||
elif String.Equals (s, "post", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Post
|
|
||||||
elif String.Equals (s, "patch", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Patch
|
|
||||||
elif String.Equals (s, "delete", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Delete
|
|
||||||
elif String.Equals (s, "head", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Head
|
|
||||||
elif String.Equals (s, "options", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Options
|
|
||||||
elif String.Equals (s, "put", StringComparison.OrdinalIgnoreCase) then
|
|
||||||
HttpMethod.Put
|
|
||||||
else
|
|
||||||
failwith $"Unrecognised method: %s{s}"
|
|
||||||
|
|
||||||
/// A Swagger API specification.
|
/// A Swagger API specification.
|
||||||
type Swagger =
|
type SwaggerV2 =
|
||||||
{
|
{
|
||||||
/// Global collection of MIME types which any endpoint expects to consume its inputs in.
|
/// Global collection of MIME types which any endpoint expects to consume its inputs in.
|
||||||
/// This may be overridden on any individual endpoint by that endpoint.
|
/// This may be overridden on any individual endpoint by that endpoint.
|
||||||
@@ -526,58 +418,63 @@ type Swagger =
|
|||||||
Responses : Map<string, Response>
|
Responses : Map<string, Response>
|
||||||
}
|
}
|
||||||
|
|
||||||
[<RequireQualifiedAccess>]
|
/// Parse a JSON-schema-based specification of a Swagger 2.0 API and
|
||||||
module Swagger =
|
/// build the strongly-typed version. Throws on invalid inputs; returns Error if the input is JSON but has a Swagger
|
||||||
/// Parse a JSON-schema-based specification of a Swagger 2.0 API and
|
/// version that is not in the 2.0 series.
|
||||||
/// build the strongly-typed version. Throws on invalid inputs.
|
let parse (s : string) : Result<SwaggerV2, JsonNode> =
|
||||||
let parse (s : string) : Swagger =
|
let node = JsonNode.Parse s
|
||||||
let node = JsonNode.Parse s
|
let swagger = asString node "swagger" |> Version.Parse
|
||||||
let consumes = asArr'<string> node "consumes" |> List.map MimeType
|
|
||||||
let produces = asArr'<string> node "produces" |> List.map MimeType
|
|
||||||
let schemes = asArr'<string> node "schemes" |> List.map Scheme
|
|
||||||
let swagger = asString node "swagger" |> Version.Parse
|
|
||||||
let info = asObj node "info" |> SwaggerInfo.Parse
|
|
||||||
let basePath = asString node "basePath"
|
|
||||||
|
|
||||||
let definitions =
|
if swagger.Major <> 2 then
|
||||||
asObj node "definitions"
|
Error node
|
||||||
|> Seq.map (fun (KeyValue (key, value)) ->
|
else
|
||||||
let value = value.AsObject ()
|
|
||||||
key, Definition.Parse value
|
|
||||||
)
|
|
||||||
|> Map.ofSeq
|
|
||||||
|
|
||||||
let paths =
|
let consumes = asArr'<string> node "consumes" |> List.map MimeType
|
||||||
asObj node "paths"
|
let produces = asArr'<string> node "produces" |> List.map MimeType
|
||||||
|> Seq.map (fun (KeyValue (key, value)) ->
|
let schemes = asArr'<string> node "schemes" |> List.map Scheme
|
||||||
let contents =
|
let info = asObj node "info" |> SwaggerInfo.Parse
|
||||||
value.AsObject ()
|
let basePath = asString node "basePath"
|
||||||
|> Seq.map (fun (KeyValue (endpoint, contents)) ->
|
|
||||||
let contents = contents.AsObject ()
|
|
||||||
HttpMethod.Parse endpoint, SwaggerEndpoint.Parse contents
|
|
||||||
)
|
|
||||||
|> Map.ofSeq
|
|
||||||
|
|
||||||
key, contents
|
let definitions =
|
||||||
)
|
asObj node "definitions"
|
||||||
|> Map.ofSeq
|
|> Seq.map (fun (KeyValue (key, value)) ->
|
||||||
|
let value = value.AsObject ()
|
||||||
|
key, Definition.Parse value
|
||||||
|
)
|
||||||
|
|> Map.ofSeq
|
||||||
|
|
||||||
let responses =
|
let paths =
|
||||||
asObj node "responses"
|
asObj node "paths"
|
||||||
|> Seq.map (fun (KeyValue (key, value)) ->
|
|> Seq.map (fun (KeyValue (key, value)) ->
|
||||||
let value = value.AsObject ()
|
let contents =
|
||||||
key, Response.Parse value
|
value.AsObject ()
|
||||||
)
|
|> Seq.map (fun (KeyValue (endpoint, contents)) ->
|
||||||
|> Map.ofSeq
|
let contents = contents.AsObject ()
|
||||||
|
HttpMethod.Parse endpoint, SwaggerEndpoint.Parse contents
|
||||||
|
)
|
||||||
|
|> Map.ofSeq
|
||||||
|
|
||||||
{
|
key, contents
|
||||||
Consumes = consumes
|
)
|
||||||
Produces = produces
|
|> Map.ofSeq
|
||||||
Schemes = schemes
|
|
||||||
Swagger = swagger
|
let responses =
|
||||||
Info = info
|
asObj node "responses"
|
||||||
BasePath = basePath
|
|> Seq.map (fun (KeyValue (key, value)) ->
|
||||||
Paths = paths
|
let value = value.AsObject ()
|
||||||
Definitions = definitions
|
key, Response.Parse value
|
||||||
Responses = responses
|
)
|
||||||
}
|
|> Map.ofSeq
|
||||||
|
|
||||||
|
{
|
||||||
|
Consumes = consumes
|
||||||
|
Produces = produces
|
||||||
|
Schemes = schemes
|
||||||
|
Swagger = swagger
|
||||||
|
Info = info
|
||||||
|
BasePath = basePath
|
||||||
|
Paths = paths
|
||||||
|
Definitions = definitions
|
||||||
|
Responses = responses
|
||||||
|
}
|
||||||
|
|> Ok
|
@@ -28,6 +28,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<Compile Include="AssemblyInfo.fs" />
|
||||||
<Compile Include="List.fs"/>
|
<Compile Include="List.fs"/>
|
||||||
<Compile Include="Text.fs" />
|
<Compile Include="Text.fs" />
|
||||||
<Compile Include="Measure.fs" />
|
<Compile Include="Measure.fs" />
|
||||||
@@ -41,9 +42,11 @@
|
|||||||
<Compile Include="HttpClientGenerator.fs"/>
|
<Compile Include="HttpClientGenerator.fs"/>
|
||||||
<Compile Include="CataGenerator.fs" />
|
<Compile Include="CataGenerator.fs" />
|
||||||
<Compile Include="ArgParserGenerator.fs" />
|
<Compile Include="ArgParserGenerator.fs" />
|
||||||
<Compile Include="Swagger.fs" />
|
<Compile Include="JsonHelpers.fs" />
|
||||||
|
<Compile Include="HttpMethod.fs" />
|
||||||
|
<Compile Include="SwaggerV2.fs" />
|
||||||
|
<Compile Include="OpenApi3.fs" />
|
||||||
<Compile Include="SwaggerClientGenerator.fs" />
|
<Compile Include="SwaggerClientGenerator.fs" />
|
||||||
<None Include="ApacheLicence.txt" />
|
|
||||||
<EmbeddedResource Include="version.json"/>
|
<EmbeddedResource Include="version.json"/>
|
||||||
<EmbeddedResource Include="SurfaceBaseline.txt"/>
|
<EmbeddedResource Include="SurfaceBaseline.txt"/>
|
||||||
<None Include="..\README.md">
|
<None Include="..\README.md">
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"version": "7.0",
|
"version": "8.0",
|
||||||
"publicReleaseRefSpec": [
|
"publicReleaseRefSpec": [
|
||||||
"^refs/heads/main$"
|
"^refs/heads/main$"
|
||||||
],
|
],
|
||||||
@@ -11,4 +11,4 @@
|
|||||||
":/README.md",
|
":/README.md",
|
||||||
":/Directory.Build.props"
|
":/Directory.Build.props"
|
||||||
]
|
]
|
||||||
}
|
}
|
@@ -10,7 +10,7 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageDownload Include="G-Research.FSharp.Analyzers" Version="[0.14.0]" />
|
<PackageDownload Include="G-Research.FSharp.Analyzers" Version="[0.15.0]" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
6
flake.lock
generated
6
flake.lock
generated
@@ -20,11 +20,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1744868846,
|
"lastModified": 1749903597,
|
||||||
"narHash": "sha256-5RJTdUHDmj12Qsv7XOhuospjAjATNiTMElplWnJE9Hs=",
|
"narHash": "sha256-jp0D4vzBcRKwNZwfY4BcWHemLGUs4JrS3X9w5k/JYDA=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "ebe4301cbd8f81c4f8d3244b3632338bbeb6d49c",
|
"rev": "41da1e3ea8e23e094e5e3eeb1e6b830468a7399e",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
@@ -1,13 +1,13 @@
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
"pname": "ApiSurface",
|
"pname": "ApiSurface",
|
||||||
"version": "4.1.20",
|
"version": "4.1.21",
|
||||||
"hash": "sha256-koWgO9FC9ax+Ij56ug8kxeyknl0yhLqnNLOUdxtqqo4="
|
"hash": "sha256-v2adBYoE9NZPaQR3u2qq9r/9PxAM/wqi2Uiky0xGq+E="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "fantomas",
|
"pname": "fantomas",
|
||||||
"version": "7.0.1",
|
"version": "7.0.2",
|
||||||
"hash": "sha256-2aGD6Kjh83gmssRqqZ/Uihi7VbNqNUelX4otIfCuhTI="
|
"hash": "sha256-BAaENIm/ksTiXrUImRgKoIXTGIlgsX7ch6ayoFjhJXA="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Fantomas.Core",
|
"pname": "Fantomas.Core",
|
||||||
@@ -21,13 +21,13 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "FsCheck",
|
"pname": "FsCheck",
|
||||||
"version": "3.2.0",
|
"version": "3.3.0",
|
||||||
"hash": "sha256-ksZ4vLgWwyQOzFuK2BczdtDtWWYmedG7UBAg4pYuI8g="
|
"hash": "sha256-TFDR/uAGv4OqrMX8/reQ4faaAhH9hxTHr1T/YkNPCCU="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "fsharp-analyzers",
|
"pname": "fsharp-analyzers",
|
||||||
"version": "0.30.0",
|
"version": "0.31.0",
|
||||||
"hash": "sha256-7oaSwpHAU1opzpz4szLU/gDaJC/ww9eiFkPu0nr4Mj4="
|
"hash": "sha256-PoAvaXbXsmvVw870UsnqdD20HoBHO7u4bzoaz5DXfzM="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "FSharp.Core",
|
"pname": "FSharp.Core",
|
||||||
@@ -44,6 +44,11 @@
|
|||||||
"version": "9.0.202",
|
"version": "9.0.202",
|
||||||
"hash": "sha256-64Gub0qemmCoMa1tDus6TeTuB1+5sHfE6KD2j4o84mA="
|
"hash": "sha256-64Gub0qemmCoMa1tDus6TeTuB1+5sHfE6KD2j4o84mA="
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"pname": "FSharp.SystemTextJson",
|
||||||
|
"version": "1.4.36",
|
||||||
|
"hash": "sha256-zZEhjP0mdc5E3fBPS4/lqD7sxoaoT5SOspP546RWYdc="
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"pname": "FsUnit",
|
"pname": "FsUnit",
|
||||||
"version": "7.0.1",
|
"version": "7.0.1",
|
||||||
@@ -81,13 +86,13 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Microsoft.CodeCoverage",
|
"pname": "Microsoft.CodeCoverage",
|
||||||
"version": "17.13.0",
|
"version": "17.14.1",
|
||||||
"hash": "sha256-GKrIxeyQo5Az1mztfQgea1kGtJwonnNOrXK/0ULfu8o="
|
"hash": "sha256-f8QytG8GvRoP47rO2KEmnDLxIpyesaq26TFjDdW40Gs="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Microsoft.NET.Test.Sdk",
|
"pname": "Microsoft.NET.Test.Sdk",
|
||||||
"version": "17.13.0",
|
"version": "17.14.1",
|
||||||
"hash": "sha256-sc2wvyV8cGm1FrNP2GGHEI584RCvRPu15erYCsgw5QY="
|
"hash": "sha256-mZUzDFvFp7x1nKrcnRd0hhbNu5g8EQYt8SKnRgdhT/A="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Microsoft.NETCore.App.Host.linux-arm64",
|
"pname": "Microsoft.NETCore.App.Host.linux-arm64",
|
||||||
@@ -191,13 +196,13 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Microsoft.TestPlatform.ObjectModel",
|
"pname": "Microsoft.TestPlatform.ObjectModel",
|
||||||
"version": "17.13.0",
|
"version": "17.14.1",
|
||||||
"hash": "sha256-6S0fjfj8vA+h6dJVNwLi6oZhYDO/I/6hBZaq2VTW+Uk="
|
"hash": "sha256-QMf6O+w0IT+16Mrzo7wn+N20f3L1/mDhs/qjmEo1rYs="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Microsoft.TestPlatform.TestHost",
|
"pname": "Microsoft.TestPlatform.TestHost",
|
||||||
"version": "17.13.0",
|
"version": "17.14.1",
|
||||||
"hash": "sha256-L/CJzou7dhmShUgXq3aXL3CaLTJll17Q+JY2DBdUUpo="
|
"hash": "sha256-1cxHWcvHRD7orQ3EEEPPxVGEkTpxom1/zoICC9SInJs="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "Myriad.Core",
|
"pname": "Myriad.Core",
|
||||||
@@ -231,33 +236,33 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NuGet.Common",
|
"pname": "NuGet.Common",
|
||||||
"version": "6.13.2",
|
"version": "6.14.0",
|
||||||
"hash": "sha256-ASLa/Jigg5Eop0ZrXPl98RW2rxnJRC7pbbxhuV74hFw="
|
"hash": "sha256-jDOwt3veI1GSG8CfBnf2+dJxD3E/Nmlc+vHtD4J76Ms="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NuGet.Configuration",
|
"pname": "NuGet.Configuration",
|
||||||
"version": "6.13.2",
|
"version": "6.14.0",
|
||||||
"hash": "sha256-z8VW1YdRDanyyRTDYRvRkSv/XPR3c/hMM1y8cNNjx0Y="
|
"hash": "sha256-1PN9s6fhCw3wd2260U6hQ4vG3jIvcG8GIn1oQgxMXA0="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NuGet.Frameworks",
|
"pname": "NuGet.Frameworks",
|
||||||
"version": "6.13.2",
|
"version": "6.14.0",
|
||||||
"hash": "sha256-caDyc+WgYOo43AUTjtbP0MyvYDb6JweEKDdIul61Cac="
|
"hash": "sha256-3ViM3R1ucQMEL2hQYsivT86kI6veMQK2xDsiAcFcVQk="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NuGet.Packaging",
|
"pname": "NuGet.Packaging",
|
||||||
"version": "6.13.2",
|
"version": "6.14.0",
|
||||||
"hash": "sha256-lhO+SFwIYZ4aPHxIGm5ubkkE2a5Ve2xgtroRbNh7hpw="
|
"hash": "sha256-Yafbnxs3maj55bJ1oKQiZ0QkntFUzXdhorL94YEUOhY="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NuGet.Protocol",
|
"pname": "NuGet.Protocol",
|
||||||
"version": "6.13.2",
|
"version": "6.14.0",
|
||||||
"hash": "sha256-5lnAHHZjy7A4vgv65AeBAs64mSNpuoUjxW3HnrMpuzY="
|
"hash": "sha256-uLDKfs+QN1MdnuQtTES8qfNzzsmYKM6XB9pwJc4G+eo="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NuGet.Versioning",
|
"pname": "NuGet.Versioning",
|
||||||
"version": "6.13.2",
|
"version": "6.14.0",
|
||||||
"hash": "sha256-gmpyBpKnt+GHqgx/2uFKp+J2csbxEAy1E7WdVT117sw="
|
"hash": "sha256-DqdOJgsphKxSvqB8b60zNPCaiLfbiu3WnUJ/90feLrY="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "NUnit",
|
"pname": "NUnit",
|
||||||
@@ -309,11 +314,6 @@
|
|||||||
"version": "6.0.0",
|
"version": "6.0.0",
|
||||||
"hash": "sha256-KaMHgIRBF7Nf3VwOo+gJS1DcD+41cJDPWFh+TDQ8ee8="
|
"hash": "sha256-KaMHgIRBF7Nf3VwOo+gJS1DcD+41cJDPWFh+TDQ8ee8="
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"pname": "System.Formats.Asn1",
|
|
||||||
"version": "8.0.1",
|
|
||||||
"hash": "sha256-may/Wg+esmm1N14kQTG4ESMBi+GQKPp0ZrrBo/o6OXM="
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"pname": "System.IO.Abstractions",
|
"pname": "System.IO.Abstractions",
|
||||||
"version": "4.2.13",
|
"version": "4.2.13",
|
||||||
@@ -336,8 +336,8 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "System.Reflection.Metadata",
|
"pname": "System.Reflection.Metadata",
|
||||||
"version": "1.6.0",
|
"version": "8.0.0",
|
||||||
"hash": "sha256-JJfgaPav7UfEh4yRAQdGhLZF1brr0tUWPl6qmfNWq/E="
|
"hash": "sha256-dQGC30JauIDWNWXMrSNOJncVa1umR1sijazYwUDdSIE="
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pname": "System.Runtime",
|
"pname": "System.Runtime",
|
||||||
@@ -369,16 +369,31 @@
|
|||||||
"version": "4.5.0",
|
"version": "4.5.0",
|
||||||
"hash": "sha256-BkUYNguz0e4NJp1kkW7aJBn3dyH9STwB5N8XqnlCsmY="
|
"hash": "sha256-BkUYNguz0e4NJp1kkW7aJBn3dyH9STwB5N8XqnlCsmY="
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Text.Json",
|
||||||
|
"version": "6.0.10",
|
||||||
|
"hash": "sha256-UijYh0dxFjFinMPSTJob96oaRkNm+Wsa+7Ffg6mRnsc="
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"pname": "System.Text.Json",
|
"pname": "System.Text.Json",
|
||||||
"version": "8.0.5",
|
"version": "8.0.5",
|
||||||
"hash": "sha256-yKxo54w5odWT6nPruUVsaX53oPRe+gKzGvLnnxtwP68="
|
"hash": "sha256-yKxo54w5odWT6nPruUVsaX53oPRe+gKzGvLnnxtwP68="
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Text.Json",
|
||||||
|
"version": "9.0.0",
|
||||||
|
"hash": "sha256-aM5Dh4okLnDv940zmoFAzRmqZre83uQBtGOImJpoIqk="
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"pname": "TypeEquality",
|
"pname": "TypeEquality",
|
||||||
"version": "0.3.0",
|
"version": "0.3.0",
|
||||||
"hash": "sha256-V50xAOzzyUJrY+MYPRxtnqW5MVeATXCes89wPprv1r4="
|
"hash": "sha256-V50xAOzzyUJrY+MYPRxtnqW5MVeATXCes89wPprv1r4="
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"pname": "WoofWare.Expect",
|
||||||
|
"version": "0.4.2",
|
||||||
|
"hash": "sha256-CaVcj9Fo0VSMgfKIukM9WHGufPWHDqMO1D4VYVdJKJk="
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"pname": "WoofWare.Whippet.Fantomas",
|
"pname": "WoofWare.Whippet.Fantomas",
|
||||||
"version": "0.6.3",
|
"version": "0.6.3",
|
||||||
|
Reference in New Issue
Block a user