Skip to content

Commit e650678

Browse files
authored
Add Kroger provider (#710)
* Add Kroger provider * Fix sub key mapping * Make some scopes optional
1 parent f1e2f76 commit e650678

File tree

10 files changed

+360
-0
lines changed

10 files changed

+360
-0
lines changed

AspNet.Security.OAuth.Providers.sln

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -285,6 +285,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AspNet.Security.OAuth.Snapc
285285
EndProject
286286
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AspNet.Security.OAuth.Feishu", "src\AspNet.Security.OAuth.Feishu\AspNet.Security.OAuth.Feishu.csproj", "{B8F9B052-84BF-436C-B22B-CEBD5EB1F8E3}"
287287
EndProject
288+
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AspNet.Security.OAuth.Kroger", "src\AspNet.Security.OAuth.Kroger\AspNet.Security.OAuth.Kroger.csproj", "{8C7A98A6-5F61-492B-980D-0A9F5F9F5C73}"
289+
EndProject
288290
Global
289291
GlobalSection(SolutionConfigurationPlatforms) = preSolution
290292
Debug|Any CPU = Debug|Any CPU
@@ -651,6 +653,10 @@ Global
651653
{B8F9B052-84BF-436C-B22B-CEBD5EB1F8E3}.Debug|Any CPU.Build.0 = Debug|Any CPU
652654
{B8F9B052-84BF-436C-B22B-CEBD5EB1F8E3}.Release|Any CPU.ActiveCfg = Release|Any CPU
653655
{B8F9B052-84BF-436C-B22B-CEBD5EB1F8E3}.Release|Any CPU.Build.0 = Release|Any CPU
656+
{8C7A98A6-5F61-492B-980D-0A9F5F9F5C73}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
657+
{8C7A98A6-5F61-492B-980D-0A9F5F9F5C73}.Debug|Any CPU.Build.0 = Debug|Any CPU
658+
{8C7A98A6-5F61-492B-980D-0A9F5F9F5C73}.Release|Any CPU.ActiveCfg = Release|Any CPU
659+
{8C7A98A6-5F61-492B-980D-0A9F5F9F5C73}.Release|Any CPU.Build.0 = Release|Any CPU
654660
EndGlobalSection
655661
GlobalSection(SolutionProperties) = preSolution
656662
HideSolutionNode = FALSE
@@ -752,6 +758,7 @@ Global
752758
{23E576EB-6514-4617-8F04-FE7D5540136D} = {C1352FD3-AE8B-43EE-B45B-F6E0B3FBAC6D}
753759
{ECD22287-9B9F-489A-84A7-E66D65A39D73} = {C1352FD3-AE8B-43EE-B45B-F6E0B3FBAC6D}
754760
{B8F9B052-84BF-436C-B22B-CEBD5EB1F8E3} = {C1352FD3-AE8B-43EE-B45B-F6E0B3FBAC6D}
761+
{8C7A98A6-5F61-492B-980D-0A9F5F9F5C73} = {C1352FD3-AE8B-43EE-B45B-F6E0B3FBAC6D}
755762
EndGlobalSection
756763
GlobalSection(ExtensibilityGlobals) = postSolution
757764
SolutionGuid = {C7B54DE2-6407-4802-AD9C-CE54BF414C8C}

README.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,7 @@ If a provider you're looking for does not exist, consider making a PR to add one
157157
| KakaoTalk | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.KakaoTalk?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.KakaoTalk/ "Download AspNet.Security.OAuth.KakaoTalk from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.KakaoTalk?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.KakaoTalk "Download AspNet.Security.OAuth.KakaoTalk from MyGet.org") | [Documentation](https://developers.kakao.com/docs/latest/en/kakaologin/common "KakaoTalk developer documentation") |
158158
| Keycloak | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.Keycloak?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.Keycloak/ "Download AspNet.Security.OAuth.Keycloak from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.Keycloak?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.Keycloak "Download AspNet.Security.OAuth.Keycloak from MyGet.org") | [Documentation](https://www.keycloak.org/docs/latest/authorization_services/#_service_overview "Keycloak developer documentation") |
159159
| Kloudless | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.Kloudless?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.Kloudless/ "Download AspNet.Security.OAuth.Kloudless from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.Kloudless?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.Kloudless "Download AspNet.Security.OAuth.Kloudless from MyGet.org") | [Documentation](https://developers.kloudless.com/docs/v1/authentication "Kloudless developer documentation") |
160+
| Kroger | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.Kroger?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.Kroger/ "Download AspNet.Security.OAuth.Kroger from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.Kroger?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.Kroger "Download AspNet.Security.OAuth.Kroger from MyGet.org") | [Documentation](https://developer.kroger.com/reference/#section/Authentication "Kroger developer documentation") |
160161
| Lichess | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.Lichess?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.Lichess/ "Download AspNet.Security.OAuth.Lichess from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.Lichess?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.Lichess "Download AspNet.Security.OAuth.Lichess from MyGet.org") | [Documentation](https://lichess.org/api#section/Authentication "Lichess developer documentation") |
161162
| Line | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.Line?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.Line/ "Download AspNet.Security.OAuth.Line from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.Line?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.Line "Download AspNet.Security.OAuth.Line from MyGet.org") | [Documentation](https://developers.line.biz/en/docs/line-login/integrate-line-login "Line developer documentation") |
162163
| LinkedIn | [![NuGet](https://buildstats.info/nuget/AspNet.Security.OAuth.LinkedIn?includePreReleases=false)](https://www.nuget.org/packages/AspNet.Security.OAuth.LinkedIn/ "Download AspNet.Security.OAuth.LinkedIn from NuGet.org") | [![MyGet](https://buildstats.info/myget/aspnet-contrib/AspNet.Security.OAuth.LinkedIn?includePreReleases=true)](https://www.myget.org/feed/aspnet-contrib/package/nuget/AspNet.Security.OAuth.LinkedIn "Download AspNet.Security.OAuth.LinkedIn from MyGet.org") | [Documentation](https://docs.microsoft.com/en-us/linkedin/shared/authentication/authentication "LinkedIn developer documentation") |

docs/kroger.md

Lines changed: 25 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
# Integrating the Kroger Provider
2+
3+
## Example
4+
5+
```csharp
6+
services.AddAuthentication(options => /* Auth configuration */)
7+
.AddKroger(options =>
8+
{
9+
options.ClientId = "my-client-id";
10+
options.ClientSecret = "my-client-secret";
11+
12+
// Optionally request other permissions
13+
// See https://developer.kroger.com/reference/#section/Authentication for details.
14+
options.Scope.Add("product.compact");
15+
options.Scope.Add("cart.basic:write");
16+
});
17+
```
18+
19+
## Required Additional Settings
20+
21+
_None._
22+
23+
## Optional Settings
24+
25+
_None._
Lines changed: 20 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<PropertyGroup>
4+
<DisablePackageBaselineValidation>true</DisablePackageBaselineValidation>
5+
<PackageValidationBaselineVersion>6.0.9</PackageValidationBaselineVersion>
6+
<TargetFrameworks>$(DefaultNetCoreTargetFramework)</TargetFrameworks>
7+
</PropertyGroup>
8+
9+
<PropertyGroup>
10+
<Description>ASP.NET Core security middleware enabling Kroger authentication.</Description>
11+
<Authors>Luke Moore</Authors>
12+
<PackageTags>aspnetcore;authentication;kroger;oauth;security</PackageTags>
13+
</PropertyGroup>
14+
15+
<ItemGroup>
16+
<FrameworkReference Include="Microsoft.AspNetCore.App" />
17+
<PackageReference Include="JetBrains.Annotations" PrivateAssets="All" />
18+
</ItemGroup>
19+
20+
</Project>
Lines changed: 48 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
/*
2+
* Licensed under the Apache License, Version 2.0 (http://www.apache.org/licenses/LICENSE-2.0)
3+
* See https://github.com/aspnet-contrib/AspNet.Security.OAuth.Providers
4+
* for more information concerning the license and the contributors participating to this project.
5+
*/
6+
7+
namespace AspNet.Security.OAuth.Kroger;
8+
9+
/// <summary>
10+
/// Default values used by the Kroger authentication middleware.
11+
/// </summary>
12+
public static class KrogerAuthenticationDefaults
13+
{
14+
/// <summary>
15+
/// Default value for <see cref="AuthenticationScheme.Name"/>.
16+
/// </summary>
17+
public const string AuthenticationScheme = "Kroger";
18+
19+
/// <summary>
20+
/// Default value for <see cref="AuthenticationScheme.DisplayName"/>.
21+
/// </summary>
22+
public static readonly string DisplayName = "Kroger";
23+
24+
/// <summary>
25+
/// Default value for <see cref="AuthenticationSchemeOptions.ClaimsIssuer"/>.
26+
/// </summary>
27+
public static readonly string Issuer = "Kroger";
28+
29+
/// <summary>
30+
/// Default value for <see cref="RemoteAuthenticationOptions.CallbackPath"/>.
31+
/// </summary>
32+
public static readonly string CallbackPath = "/signin-kroger";
33+
34+
/// <summary>
35+
/// Default value for <see cref="OAuthOptions.AuthorizationEndpoint"/>.
36+
/// </summary>
37+
public static readonly string AuthorizationEndpoint = "https://api.kroger.com/v1/connect/oauth2/authorize";
38+
39+
/// <summary>
40+
/// Default value for <see cref="OAuthOptions.TokenEndpoint"/>.
41+
/// </summary>
42+
public static readonly string TokenEndpoint = "https://api.kroger.com/v1/connect/oauth2/token";
43+
44+
/// <summary>
45+
/// Default value for <see cref="OAuthOptions.UserInformationEndpoint"/>.
46+
/// </summary>
47+
public static readonly string UserInformationEndpoint = "https://api.kroger.com/v1/identity/profile";
48+
}
Lines changed: 74 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
/*
2+
* Licensed under the Apache License, Version 2.0 (http://www.apache.org/licenses/LICENSE-2.0)
3+
* See https://github.com/aspnet-contrib/AspNet.Security.OAuth.Providers
4+
* for more information concerning the license and the contributors participating to this project.
5+
*/
6+
7+
using AspNet.Security.OAuth.Kroger;
8+
9+
namespace Microsoft.Extensions.DependencyInjection;
10+
11+
/// <summary>
12+
/// Extension methods to add Kroger authentication capabilities to an HTTP application pipeline.
13+
/// </summary>
14+
public static class KrogerAuthenticationExtensions
15+
{
16+
/// <summary>
17+
/// Adds <see cref="KrogerAuthenticationHandler"/> to the specified
18+
/// <see cref="AuthenticationBuilder"/>, which enables Kroger authentication capabilities.
19+
/// </summary>
20+
/// <param name="builder">The authentication builder.</param>
21+
/// <returns>The <see cref="AuthenticationBuilder"/>.</returns>
22+
public static AuthenticationBuilder AddKroger([NotNull] this AuthenticationBuilder builder)
23+
{
24+
return builder.AddKroger(KrogerAuthenticationDefaults.AuthenticationScheme, options => { });
25+
}
26+
27+
/// <summary>
28+
/// Adds <see cref="KrogerAuthenticationHandler"/> to the specified
29+
/// <see cref="AuthenticationBuilder"/>, which enables Kroger authentication capabilities.
30+
/// </summary>
31+
/// <param name="builder">The authentication builder.</param>
32+
/// <param name="configuration">The delegate used to configure the Kroger options.</param>
33+
/// <returns>The <see cref="AuthenticationBuilder"/>.</returns>
34+
public static AuthenticationBuilder AddKroger(
35+
[NotNull] this AuthenticationBuilder builder,
36+
[NotNull] Action<KrogerAuthenticationOptions> configuration)
37+
{
38+
return builder.AddKroger(KrogerAuthenticationDefaults.AuthenticationScheme, configuration);
39+
}
40+
41+
/// <summary>
42+
/// Adds <see cref="KrogerAuthenticationHandler"/> to the specified
43+
/// <see cref="AuthenticationBuilder"/>, which enables Kroger authentication capabilities.
44+
/// </summary>
45+
/// <param name="builder">The authentication builder.</param>
46+
/// <param name="scheme">The authentication scheme associated with this instance.</param>
47+
/// <param name="configuration">The delegate used to configure the Kroger options.</param>
48+
/// <returns>The <see cref="AuthenticationBuilder"/>.</returns>
49+
public static AuthenticationBuilder AddKroger(
50+
[NotNull] this AuthenticationBuilder builder,
51+
[NotNull] string scheme,
52+
[NotNull] Action<KrogerAuthenticationOptions> configuration)
53+
{
54+
return builder.AddKroger(scheme, KrogerAuthenticationDefaults.DisplayName, configuration);
55+
}
56+
57+
/// <summary>
58+
/// Adds <see cref="KrogerAuthenticationHandler"/> to the specified
59+
/// <see cref="AuthenticationBuilder"/>, which enables Kroger authentication capabilities.
60+
/// </summary>
61+
/// <param name="builder">The authentication builder.</param>
62+
/// <param name="scheme">The authentication scheme associated with this instance.</param>
63+
/// <param name="caption">The optional display name associated with this instance.</param>
64+
/// <param name="configuration">The delegate used to configure the Kroger options.</param>
65+
/// <returns>The <see cref="AuthenticationBuilder"/>.</returns>
66+
public static AuthenticationBuilder AddKroger(
67+
[NotNull] this AuthenticationBuilder builder,
68+
[NotNull] string scheme,
69+
[CanBeNull] string caption,
70+
[NotNull] Action<KrogerAuthenticationOptions> configuration)
71+
{
72+
return builder.AddOAuth<KrogerAuthenticationOptions, KrogerAuthenticationHandler>(scheme, caption, configuration);
73+
}
74+
}
Lines changed: 82 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
/*
2+
* Licensed under the Apache License, Version 2.0 (http://www.apache.org/licenses/LICENSE-2.0)
3+
* See https://github.com/aspnet-contrib/AspNet.Security.OAuth.Providers
4+
* for more information concerning the license and the contributors participating to this project.
5+
*/
6+
7+
using System.Net.Http.Headers;
8+
using System.Security.Claims;
9+
using System.Text.Encodings.Web;
10+
using System.Text.Json;
11+
using Microsoft.Extensions.Logging;
12+
using Microsoft.Extensions.Options;
13+
14+
namespace AspNet.Security.OAuth.Kroger;
15+
16+
/// <summary>
17+
/// Defines a handler for authentication using Kroger.
18+
/// </summary>
19+
public partial class KrogerAuthenticationHandler : OAuthHandler<KrogerAuthenticationOptions>
20+
{
21+
/// <summary>
22+
/// Initializes a new instance of the <see cref="KrogerAuthenticationHandler"/> class.
23+
/// </summary>
24+
/// <param name="options">The authentication options.</param>
25+
/// <param name="logger">The logger to use.</param>
26+
/// <param name="encoder">The URL encoder to use.</param>
27+
/// <param name="clock">The system clock to use.</param>
28+
public KrogerAuthenticationHandler(
29+
IOptionsMonitor<KrogerAuthenticationOptions> options,
30+
ILoggerFactory logger,
31+
UrlEncoder encoder,
32+
ISystemClock clock)
33+
: base(options, logger, encoder, clock)
34+
{
35+
}
36+
37+
/// <inheritdoc />
38+
protected override async Task<AuthenticationTicket> CreateTicketAsync(
39+
[NotNull] ClaimsIdentity identity,
40+
[NotNull] AuthenticationProperties properties,
41+
[NotNull] OAuthTokenResponse tokens)
42+
{
43+
using var request = new HttpRequestMessage(HttpMethod.Get, Options.UserInformationEndpoint);
44+
request.Headers.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
45+
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", tokens.AccessToken);
46+
47+
using var response = await Backchannel.SendAsync(request, HttpCompletionOption.ResponseHeadersRead, Context.RequestAborted);
48+
if (!response.IsSuccessStatusCode)
49+
{
50+
await Log.UserProfileErrorAsync(Logger, response, Context.RequestAborted);
51+
throw new HttpRequestException("An error occurred while retrieving the user profile.");
52+
}
53+
54+
using var payload = JsonDocument.Parse(await response.Content.ReadAsStringAsync(Context.RequestAborted));
55+
56+
var principal = new ClaimsPrincipal(identity);
57+
var context = new OAuthCreatingTicketContext(principal, properties, Context, Scheme, Options, Backchannel, tokens, payload.RootElement);
58+
context.RunClaimActions();
59+
60+
await Events.CreatingTicket(context);
61+
return new AuthenticationTicket(context.Principal!, context.Properties, Scheme.Name);
62+
}
63+
64+
private static partial class Log
65+
{
66+
internal static async Task UserProfileErrorAsync(ILogger logger, HttpResponseMessage response, CancellationToken cancellationToken)
67+
{
68+
UserProfileError(
69+
logger,
70+
response.StatusCode,
71+
response.Headers.ToString(),
72+
await response.Content.ReadAsStringAsync(cancellationToken));
73+
}
74+
75+
[LoggerMessage(1, LogLevel.Error, "An error occurred while retrieving the user profile: the remote server returned a {Status} response with the following payload: {Headers} {Body}.")]
76+
private static partial void UserProfileError(
77+
ILogger logger,
78+
System.Net.HttpStatusCode status,
79+
string headers,
80+
string body);
81+
}
82+
}
Lines changed: 30 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
/*
2+
* Licensed under the Apache License, Version 2.0 (http://www.apache.org/licenses/LICENSE-2.0)
3+
* See https://github.com/aspnet-contrib/AspNet.Security.OAuth.Providers
4+
* for more information concerning the license and the contributors participating to this project.
5+
*/
6+
7+
using System.Security.Claims;
8+
9+
namespace AspNet.Security.OAuth.Kroger;
10+
11+
/// <summary>
12+
/// Defines a set of options used by <see cref="KrogerAuthenticationHandler"/>.
13+
/// </summary>
14+
public class KrogerAuthenticationOptions : OAuthOptions
15+
{
16+
public KrogerAuthenticationOptions()
17+
{
18+
ClaimsIssuer = KrogerAuthenticationDefaults.Issuer;
19+
20+
CallbackPath = KrogerAuthenticationDefaults.CallbackPath;
21+
22+
AuthorizationEndpoint = KrogerAuthenticationDefaults.AuthorizationEndpoint;
23+
TokenEndpoint = KrogerAuthenticationDefaults.TokenEndpoint;
24+
UserInformationEndpoint = KrogerAuthenticationDefaults.UserInformationEndpoint;
25+
26+
Scope.Add("profile.compact");
27+
28+
ClaimActions.MapJsonSubKey(ClaimTypes.NameIdentifier, "data", "id");
29+
}
30+
}
Lines changed: 36 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
/*
2+
* Licensed under the Apache License, Version 2.0 (http://www.apache.org/licenses/LICENSE-2.0)
3+
* See https://github.com/aspnet-contrib/AspNet.Security.OAuth.Providers
4+
* for more information concerning the license and the contributors participating to this project.
5+
*/
6+
7+
namespace AspNet.Security.OAuth.Kroger;
8+
9+
public class KrogerTests : OAuthTests<KrogerAuthenticationOptions>
10+
{
11+
public KrogerTests(ITestOutputHelper outputHelper)
12+
{
13+
OutputHelper = outputHelper;
14+
}
15+
16+
public override string DefaultScheme => KrogerAuthenticationDefaults.AuthenticationScheme;
17+
18+
protected internal override void RegisterAuthentication(AuthenticationBuilder builder)
19+
{
20+
builder.AddKroger(options => ConfigureDefaults(builder, options));
21+
}
22+
23+
[Theory]
24+
[InlineData(ClaimTypes.NameIdentifier, "53990804-cfd1-43f3-8256-bdc9817a4fd0")]
25+
public async Task Can_Sign_In_Using_Kroger(string claimType, string claimValue)
26+
{
27+
// Arrange
28+
using var server = CreateTestServer();
29+
30+
// Act
31+
var claims = await AuthenticateUserAsync(server);
32+
33+
// Assert
34+
AssertClaim(claims, claimType, claimValue);
35+
}
36+
}
Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
{
2+
"$schema": "https://raw.githubusercontent.com/justeat/httpclient-interception/master/src/HttpClientInterception/Bundles/http-request-bundle-schema.json",
3+
"items": [
4+
{
5+
"comment": "https://developer.kroger.com/reference#operation/accessToken",
6+
"uri": "https://api.kroger.com/v1/connect/oauth2/token",
7+
"method": "POST",
8+
"contentFormat": "json",
9+
"contentJson": {
10+
"access_token": "secret-access-token",
11+
"token_type": "bearer",
12+
"expires_in": "1800"
13+
}
14+
},
15+
{
16+
"comment": "https://developer.kroger.com/reference#tag/Identity",
17+
"uri": "https://api.kroger.com/v1/identity/profile",
18+
"contentFormat": "json",
19+
"contentJson": {
20+
"data": {
21+
"id": "53990804-cfd1-43f3-8256-bdc9817a4fd0"
22+
},
23+
"meta": {
24+
"pagination": {
25+
"total": 0,
26+
"start": 0,
27+
"limit": 0
28+
},
29+
"warnings": [
30+
"string"
31+
]
32+
}
33+
}
34+
}
35+
]
36+
}
37+

0 commit comments

Comments
 (0)