Workleap.Extensions.Http.Authentication.ClientCredentialsGrant
1.2.1-preview.4
Prefix Reserved
See the version list below for details.
dotnet add package Workleap.Extensions.Http.Authentication.ClientCredentialsGrant --version 1.2.1-preview.4
NuGet\Install-Package Workleap.Extensions.Http.Authentication.ClientCredentialsGrant -Version 1.2.1-preview.4
<PackageReference Include="Workleap.Extensions.Http.Authentication.ClientCredentialsGrant" Version="1.2.1-preview.4" />
paket add Workleap.Extensions.Http.Authentication.ClientCredentialsGrant --version 1.2.1-preview.4
#r "nuget: Workleap.Extensions.Http.Authentication.ClientCredentialsGrant, 1.2.1-preview.4"
// Install Workleap.Extensions.Http.Authentication.ClientCredentialsGrant as a Cake Addin #addin nuget:?package=Workleap.Extensions.Http.Authentication.ClientCredentialsGrant&version=1.2.1-preview.4&prerelease // Install Workleap.Extensions.Http.Authentication.ClientCredentialsGrant as a Cake Tool #tool nuget:?package=Workleap.Extensions.Http.Authentication.ClientCredentialsGrant&version=1.2.1-preview.4&prerelease
Workleap.Authentication.ClientCredentialsGrant
Description | Download link | Build status |
---|---|---|
Client-side library for any .NET application | ||
Server-side library for ASP.NET Core web applications |
This set of two libraries enables authenticated machine-to-machine HTTP communication between a .NET application and an ASP.NET Core web application. HTTP requests are authenticated with JSON web tokens (JWT) issued by an OAuth 2.0 authorization server using the client credentials grant flow.
┌────────────────────────────────┐
┌─────────►│ OAuth 2.0 authorization server │◄──────────┐
│ └────────────────────────────────┘ │
│ │
│ get token with get signing keys │
│ client credentials grant flow │ validate
│ │ token
┌─┴───────────┐ ┌───────────────┴────────┐
│ Client .NET ├──────────────────────────►│ Protected ASP.NET Core │
│ application │ authenticated HTTP call │ service │
└─────────────┘ └────────────────────────┘
The client-side library includes:
- Automatic acquisition and lifetime management of client credentials-based access tokens.
- Optimized access token caching with two layers of cache using IMemoryCache, IDistributedCache, and data protection for encryption.
- Built-in customizable retry policy for production-grade resilient HTTP requests made to the OAuth 2.0 authorization server.
- Built as an extension for the Microsoft.Extensions.Http library.
- Support for .NET Standard 2.0.
The server-side library includes:
- JWT authentication using the Microsoft.AspNetCore.Authentication.JwtBearer library.
- Default authorization policies, but you can still create your own policies.
- Non-intrusive: default policies must be explicitly used, and the default authentication scheme can be modified.
- Support for ASP.NET Core 6.0 and later.
Requirements and Considerations:
- Your OAuth 2.0 authorization server must expose its metadata at the URL
<AUTHORITY>/.well-known/openid-configuration
, as described in RFC 8414. - The client-side application uses data protection. It is important to note that your data protection configuration should support distributed workloads if you have multiple instances of a client application. Microsoft recommends using a combination of Azure Key Vault and Azure Storage to ensure that data encrypted by an instance of a client application can be read by another instance.
Getting started
Client-side library
Install the package Workleap.Extensions.Http.Authentication.ClientCredentialsGrant in your client-side application
that needs to communicate with the protected ASP.NET Core server. Then, use one of the following methods to configure an authenticated HttpClient
:
// Method 1: directly set the options values with C# code
services.AddHttpClient("MyClient").AddClientCredentialsHandler(options =>
{
options.Authority = "<oauth2_authorization_server_base_url>";
options.ClientId = "<oauth2_client_id>";
options.ClientSecret = "<oauth2_client_secret>"; // use a secret store instead of hardcoding the value
options.Scope = "<optional_requested_scope>"; // use "Scopes" for multiple values
});
// Method 2: bind the options to a configuration section
services.AddHttpClient("MyClient").AddClientCredentialsHandler(configuration.GetRequiredSection("MyConfigSection").Bind);
// Method 3: Lazily bind the options to a configuration section
services.AddHttpClient("MyClient").AddClientCredentialsHandler();
services.AddOptions<ClientCredentialsOptions>("MyClient").Bind(configuration.GetRequiredSection("MyConfigSection"));
// appsettings.json:
{
"MyConfigSection": {
"Authority": "<oauth2_authorization_server_base_url>",
"ClientId": "<oauth2_client_id>",
"ClientSecret": "<oauth2_client_secret>", // use a secret configuration provider instead of hardcoding the value
"Scope": "<optional_requested_scope>", // use "Scopes" for multiple values,
"EnforceHttps": "<boolean>", // use EnforceHttps to force all authenticated to be sent via https
}
}
// You can also use the generic HttpClient registration with any of these methods:
services.AddHttpClient<MyClient>().AddClientCredentialsHandler( /* [...] */);
Note on EnforceHttps
.
It is possible to allow http authenticated requests, however, this should be limited to exceptional scenarios.
It is strongly advised that you always use https for authenticated requests transmitted as the token sent will be in clear.
Then, instantiate the HttpClient
later on using IHttpClientFactory
or directly inject it in the constructor if you used the generic registration:
public class MyClient
{
private readonly HttpClient _httpClient;
public MyClient(IHttpClientFactory httpClientFactory)
{
this._httpClient = httpClientFactory.CreateClient("MyClient");
}
public async Task DoSomeAuthenticatedHttpCallAsync()
{
await this._httpClient.GetStringAsync("https://myservice");
}
}
This client-side library is based on Duende.AccessTokenManagement, Copyright (c) Brock Allen & Dominick Baier, licensed under the Apache License, Version 2.0.
Server-side library
Install the package Workleap.AspNetCore.Authentication.ClientCredentialsGrant in your server-side ASP.NET Core application and register the authentication services:
// Registers Microsoft's JwtBearer handler with a default "ClientCredentials" authentication scheme.
// This authentication scheme can be changed using other methods overloads.
builder.Services.AddAuthentication().AddClientCredentials();
This will automatically bind the configuration section Authentication:Schemes:ClientCredentials
(unless you've changed the authentication scheme).
For instance, the example above works well with this appsettings.json
:
{
"Authentication": {
"Schemes": {
"ClientCredentials": {
"Authority": "<oauth2_authorization_server_base_url>",
"Audience": "<audience>",
"MetadataAddress": "<oauth2_authorization_server_metadata_address>"
}
}
}
}
Next, register the authorization services:
builder.Services.AddAuthorization(options =>
{
// Change the scheme here if you registered a custom scheme in the authentication services.
// You can also add requirements to your policy, such as '.RequireClaim("name", "value", ["values"])'.
options.AddPolicy("my-policy", x => x.AddAuthenticationSchemes(ClientCredentialsDefaults.AuthenticationScheme).RequireAuthenticatedUser());
});
Finally, register the authentication and authorization middlewares in your ASP.NET Core app and decorate your endpoints with the AuthorizeAttribute
:
var app = builder.Build();
// [...]
app.UseAuthentication();
app.UseAuthorization();
// Minimal APIs
app.MapGet("/hello-world", () => "Hello World!").RequireAuthorization("my-policy");
// Controller-style
[Authorize("my-policy")]
[HttpGet("hello-world")]
public IActionResult HelloWorld() => this.Ok("Hello world");
Building, releasing and versioning
The project can be built by running Build.ps1
. It uses Microsoft.CodeAnalysis.PublicApiAnalyzers to help detect public API breaking changes. Use the built-in roslyn analyzer to ensure that public APIs are declared in PublicAPI.Shipped.txt
, and obsolete public APIs in PublicAPI.Unshipped.txt
.
A new preview NuGet package is automatically published on any new commit on the main branch. This means that by completing a pull request, you automatically get a new NuGet package.
When you are ready to officially release a stable NuGet package by following the SemVer guidelines, simply manually create a tag with the format x.y.z
. This will automatically create and publish a NuGet package for this version.
License
Copyright © 2023, Workleap. This code is licensed under the Apache License, Version 2.0. You may obtain a copy of this license at https://github.com/gsoft-inc/gsoft-license/blob/master/LICENSE.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- IdentityModel (>= 6.2.0)
- Microsoft.AspNetCore.DataProtection.Abstractions (>= 6.0.0)
- Microsoft.Extensions.Caching.Memory (>= 6.0.0)
- Microsoft.Extensions.Http (>= 6.0.0)
- Microsoft.Extensions.Http.Polly (>= 6.0.0)
- Microsoft.Extensions.Options (>= 6.0.0)
- Polly.Contrib.WaitAndRetry (>= 1.1.1)
- System.Text.Json (>= 6.0.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
2.1.1-preview.33 | 0 | 1/22/2025 |
2.1.1-preview.32 | 28 | 1/17/2025 |
2.1.1-preview.31 | 36 | 1/13/2025 |
2.1.1-preview.27 | 50 | 12/18/2024 |
2.1.1-preview.26 | 45 | 12/18/2024 |
2.1.1-preview.25 | 46 | 12/18/2024 |
2.1.1-preview.24 | 47 | 12/18/2024 |
2.1.1-preview.23 | 51 | 12/17/2024 |
2.1.1-preview.22 | 50 | 11/29/2024 |
2.1.1-preview.21 | 57 | 11/22/2024 |
2.1.1-preview.20 | 50 | 10/28/2024 |
2.1.1-preview.19 | 44 | 10/28/2024 |
2.1.1-preview.18 | 50 | 10/28/2024 |
2.1.1-preview.17 | 51 | 10/28/2024 |
2.1.1-preview.16 | 47 | 10/28/2024 |
2.1.1-preview.15 | 48 | 10/28/2024 |
2.1.1-preview.14 | 53 | 10/28/2024 |
2.1.1-preview.13 | 46 | 10/28/2024 |
2.1.1-preview.11 | 52 | 10/7/2024 |
2.1.1-preview.10 | 54 | 10/1/2024 |
2.1.1-preview.9 | 54 | 9/30/2024 |
2.1.1-preview.8 | 52 | 9/27/2024 |
2.1.1-preview.7 | 52 | 9/27/2024 |
2.1.1-preview.6 | 68 | 9/13/2024 |
2.1.1-preview.5 | 55 | 9/13/2024 |
2.1.1-preview.4 | 54 | 9/13/2024 |
2.1.1-preview.3 | 94 | 9/13/2024 |
2.1.1-preview.2 | 77 | 8/27/2024 |
2.1.1-preview.1 | 64 | 8/27/2024 |
2.1.0 | 1,940 | 8/19/2024 |
2.0.1-preview.6 | 77 | 8/19/2024 |
2.0.1-preview.5 | 77 | 8/13/2024 |
2.0.1-preview.4 | 50 | 8/5/2024 |
2.0.1-preview.3 | 47 | 8/5/2024 |
2.0.1-preview.2 | 51 | 7/23/2024 |
2.0.1-preview.1 | 50 | 7/23/2024 |
2.0.0 | 1,412 | 7/22/2024 |
1.3.1-preview.8 | 60 | 7/15/2024 |
1.3.1-preview.7 | 50 | 7/5/2024 |
1.3.1-preview.6 | 62 | 7/3/2024 |
1.3.1-preview.5 | 62 | 6/25/2024 |
1.3.1-preview.4 | 70 | 6/20/2024 |
1.3.1-preview.3 | 59 | 6/14/2024 |
1.3.1-preview.2 | 72 | 6/10/2024 |
1.3.1-preview.1 | 66 | 6/10/2024 |
1.3.0 | 47,335 | 5/22/2024 |
1.2.4-preview.2 | 71 | 5/22/2024 |
1.2.4-preview.1 | 59 | 5/21/2024 |
1.2.3 | 3,692 | 5/14/2024 |
1.2.3-preview.8 | 69 | 5/14/2024 |
1.2.3-preview.7 | 73 | 5/10/2024 |
1.2.3-preview.6 | 65 | 5/3/2024 |
1.2.3-preview.5 | 81 | 4/16/2024 |
1.2.3-preview.4 | 67 | 4/15/2024 |
1.2.3-preview.3 | 73 | 4/12/2024 |
1.2.3-preview.2 | 67 | 4/3/2024 |
1.2.3-preview.1 | 69 | 4/3/2024 |
1.2.2 | 31,309 | 3/26/2024 |
1.2.1-preview.17 | 63 | 3/26/2024 |
1.2.1-preview.16 | 63 | 3/26/2024 |
1.2.1-preview.15 | 60 | 3/8/2024 |
1.2.1-preview.14 | 66 | 3/7/2024 |
1.2.1-preview.13 | 90 | 3/4/2024 |
1.2.1-preview.12 | 73 | 3/1/2024 |
1.2.1-preview.11 | 81 | 2/16/2024 |
1.2.1-preview.10 | 79 | 2/12/2024 |
1.2.1-preview.9 | 67 | 2/6/2024 |
1.2.1-preview.8 | 69 | 2/6/2024 |
1.2.1-preview.7 | 60 | 2/2/2024 |
1.2.1-preview.6 | 71 | 2/2/2024 |
1.2.1-preview.4 | 69 | 1/22/2024 |
1.2.1-preview.3 | 62 | 1/12/2024 |
1.2.1-preview.2 | 71 | 1/12/2024 |
1.2.1-preview.1 | 84 | 12/18/2023 |
1.2.0 | 58,027 | 12/14/2023 |
1.1.4-preview.1 | 89 | 12/14/2023 |
1.1.3 | 1,931 | 12/5/2023 |
1.1.3-preview.11 | 96 | 12/4/2023 |
1.1.3-preview.10 | 87 | 11/26/2023 |
1.1.3-preview.9 | 77 | 11/20/2023 |
1.1.3-preview.8 | 80 | 11/17/2023 |
1.1.3-preview.7 | 81 | 11/13/2023 |
1.1.3-preview.6 | 80 | 11/13/2023 |
1.1.3-preview.5 | 84 | 11/7/2023 |
1.1.3-preview.4 | 90 | 10/30/2023 |
1.1.3-preview.3 | 90 | 10/20/2023 |
1.1.3-preview.2 | 91 | 10/17/2023 |
1.1.3-preview.1 | 86 | 10/13/2023 |
1.1.2 | 57,383 | 10/4/2023 |
1.1.2-preview.9 | 77 | 9/28/2023 |
1.1.2-preview.8 | 81 | 9/26/2023 |
1.1.2-preview.7 | 80 | 9/25/2023 |
1.1.2-preview.6 | 82 | 9/25/2023 |
1.1.2-preview.5 | 702 | 9/21/2023 |
1.1.2-preview.4 | 79 | 9/20/2023 |
1.1.2-preview.3 | 76 | 9/20/2023 |
1.1.2-preview.2 | 79 | 9/20/2023 |
1.1.1 | 357 | 8/11/2023 |
1.1.1-preview.5 | 96 | 8/11/2023 |