RestClientSourceGenerator 1.0.0-alpha.11
dotnet add package RestClientSourceGenerator --version 1.0.0-alpha.11
NuGet\Install-Package RestClientSourceGenerator -Version 1.0.0-alpha.11
<PackageReference Include="RestClientSourceGenerator" Version="1.0.0-alpha.11" />
paket add RestClientSourceGenerator --version 1.0.0-alpha.11
#r "nuget: RestClientSourceGenerator, 1.0.0-alpha.11"
// Install RestClientSourceGenerator as a Cake Addin #addin nuget:?package=RestClientSourceGenerator&version=1.0.0-alpha.11&prerelease // Install RestClientSourceGenerator as a Cake Tool #tool nuget:?package=RestClientSourceGenerator&version=1.0.0-alpha.11&prerelease
RestClientGenerator
A .Net source generator for generating REST clients from interface definitions
Installation
RestClientGenerator is available on Nuget
Defining a client
A client is defined by creating an interface and decorating it with the HttpClientContractAttribute
class. This allows you to define the base route and content type to be used by default by all methods on the interface. These can be also be set on a per method basis and so do not have to be configured at this point.
A call to a REST endpoint is defined by adding a method to the interface. The method determines the parameters and return type that will be used to send data to and receive data from the REST endpoint.
Certain special parameters are supported to allow interception of the http request and response at points in the call (More on that later).
How the method is translated into the appropriate REST call is determined by the attributes used to decorate the method.
Methods can be defined as synchronous or asynchronous.
public interface IClient
{
[Get("api/v1/wiget/{id}")]
Task<HttpResponseMessage> GetWidgetAsync(string id);
}
To access the clients you must create a partial class that inherits from RestClientContext
, and decorate it with RestClientAttribute
attributes for each interface definition:
[RestClient(typeof(IClient))]
public partial class ClientContext
: RestClientContext
{
public override T Deserialize<T>(string content)
{
return new JsonContext().Deserialize<T>(content);
}
public override string Serialize<T>(T obj)
{
return new JsonContext().Serialize<T>(obj);
}
}
The clients can then be used in code as normal:
public class Program
{
public static async Task Main(string[] args)
{
var context = new ClientContext();
context.Options = new RestClientOptions()
{
BaseUrl = "http://localhost",
HttpClient = new HttpClient()
};
var client = context.GetClient();
var response = await client.GetWidgetAsync("id");
response.EnsureSuccessCode();
}
}
Method Parameters and Return Types
Return Types
As mentioned earlier methods can be synchronous or asynchronous so supported return types are:
void
- No return value.T
- Any type that the content can be deserialised to.HttpResponseMessage
- The full HttpResponseMessage.Task
- A task that upon completion will have no return value.Task<T>
- A task that upon completion will return any type the content can be deserailised to.Task<HttpResponseMessage>
- A task that upon completion will return the full HttpResponseMethod.Task<Stream>
- A task that upon completion will return the content property as a stream.
Return values can also be decorated with attributes to control their behaviour:
FromJsonAttribute can be used to return a property from json response content.
FromModelAttribute can be used to return a property from response content deserialised as a specific model.
Parameters
The methods parameters can be used to define content, query parameters, or headers, and attributes are used to determine their use.
SendAsContentAttribute
- Specifies that a parameter is to be used as the requests content.SendAsQueryAttribute
- Specifies that a parameter is to be used as a query parameter.SendAsHeaderAttribute
- Specfies that a parameter is to be used as a request header.SendAsFormUrlAttribute
- Specifies that a paremeter is to be used as a form url content property.
public interface ICustomerClient
{
[Get("api/customer/update")]
Task<HttpResponseMessage> UpdateCustomerAsync(
[SendAsHeader]string id,
[SendAsContent]CustomerModel customer);
}
Special Parameters
Special parameter types are also supported:
A parameter of type
Action<HttpRequestMessage>
orFunc<HttpRequestMessage, Task>
will be called just before the request is sent.A parameter of type
Action<HttpResponseMessage>
orFunc<HttpResponseMessage, Task>
will be called just after the response has been received.A parameter of type
Func<HttpResponseMessage, *ReturnType*>
will be called just before the method returns. The method will then return the response from the lambda allowing it to decode the response.
public interface ICustomerClient
{
[Get("api/customer/{id}")]
Task<CustomerModel> GetCustomerAsync(
string id,
Func<HttpResponseMessage, CustomerMode> responseFunc);
}
Attributes
Attributes are used to define a methods behaviours:
RestClientGenerator provides a set of attributes that can be use to define the HttpMethod
type:
PostAttribute
- Specifies that the request has a POST method.GetAttribute
- Specifies that the request has a GET method.PutAttribute
- Specifies that the request has a PUT method.PatchAttribute
- Specifies that the request has a PATCH method.DeleteAttribute
- Specifies that the request has a DELETE method.
AddHeaderAttribute
can be applied to a method or an interface to add a header value to the request or all requests:
AddFormUrlEncodedPropertyAttribute
can be used to add a form url property to a request content.AddAuthorizationHeaderAttribute
can be applied to a method or an interface to add an authorization header to the request or all requests.
Http Response Processors
For clients that wish to create a more complex response model there is the HttpResponseProcessor
type. These types allow the response to be intercepted and converted into much more complex results.
public interface IResult<T>
{
int StatusCode { get; }
T Result { get; }
string ErrorMessage { get; }
}
public class Result<T>
: IResult<T>
{
public int StatusCode { get; set; }
public T Result { get; set; }
public string ErrorMessage { get; set; }
}
public class ResultProcessor<T>
: HttpResponseProcessor<IResult<T>>
{
public override async Task<IResult<T>> ProcessResponseAsync(HttpResponseMessage httpResponse)
{
var result = new Result<T>;
result.StatusCode = (int)httpResponse.StatusCode;
if (httpResponse.IsSuccessStatusCode == true)
{
var content = await response.Content.ReadAsStringAsync();
if (string.IsNullOrEmpty(content) == false)
{
result.Result = JsonConvert.DeserializeObject<T>(content);
}
}
else
{
result.ErrorMessage = "";
}
return result;
}
}
[HttpClientContract(Route = "api/customers", ContentType = "aplication/json")]
public interface ICustomerClient
{
[Get("")]
[HttpResponseProcessor(typeof(ResultProcessor<IEnumerable<CustomerModel>>))]
Task<IResult<IEnumerable<CustomerModel>>> GetCustomersAsync();
}
Example synchronous client
Create an interface and decorate with attributes to define how the client should interact with the service:
[HttpClientContract(Route = "api/customers", ContentType = "aplication/json")]
public interface ICustomerClient
{
[Post("")]
CreateCustomerResponseModel CreateCustomer(CreateCustomerModel customer);
[Get("")]
IEnumerable<CustomerModel> GetCustomers();
[Get("{name}")]
CustomerModel GetCustomerByName(string name);
}
Create a client context class to access the clients:
[RestClient(typeof(ICustomerClient))]
public partial ClientContext
: RestClientContext
{
public override T Deserialize<T>(string content)
{
return new JsonContext().Deserialize<T>(content);
}
public override string Serialize<T>(T obj)
{
return new JsonContext().Serialize<T>(obj);
}
}
Once the interface is defined a proxy can be generated and then used to call the service:
var clientContext = new ClientContext();
context.Options = new RestClientOptions()
{
BaseUrl = "http://localhost",
HttpClient = new HttpClient()
};
var customerClient = context.GetCustomerClient();
var response = customerClient.CreateCustomer(
new CreateCustomerModel()
{
Name = "Customer",
Address = "Somewhere",
PhoneNumber = "123456789"
});
Example asynchronous client
Create an interface and decorate with attributes to define how the client should interact with the service:
[HttpClientContract(Route = "api/customers", ContentType = "aplication/json")]
public interface ICustomerClient
{
[Post("")]
Task<CreateCustomerResponseModel> CreateCustomerAsync(CreateCustomerModel customer);
[Get("")]
Task<IEnumerable<CustomerModel>> GetCustomersAsync();
[Get("{name}")]
Task<CustomerModel> GetCustomerByName(string name);
}
Create a client context class to access the clients:
[RestClient(typeof(ICustomerClient))]
public partial ClientContext
: RestClientContext
{
public override T Deserialize<T>(string content)
{
return new JsonContext().Deserialize<T>(content);
}
public override string Serialize<T>(T obj)
{
return new JsonContext().Serialize<T>(obj);
}
}
Again, once the interface is defined a proxy can be generated and then used to call the service:
var clientContext = new ClientContext();
context.Options = new RestClientOptions()
{
BaseUrl = "http://localhost",
HttpClient = new HttpClient()
};
var customerClient = context.GetCustomerClient();
var response = await customerClient.CreateCustomerAsync(
new CreateCustomerModel()
{
Name = "Customer",
Address = "Somewhere",
PhoneNumber = "123456789"
});
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. |
.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
- No dependencies.
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 |
---|---|---|
1.0.0-alpha.11 | 56 | 9/30/2024 |
1.0.0-alpha.10 | 242 | 4/5/2024 |
1.0.0-alpha.9 | 92 | 1/25/2024 |
1.0.0-alpha.8 | 56 | 1/25/2024 |
1.0.0-alpha.7 | 60 | 1/25/2024 |
1.0.0-alpha.6 | 57 | 1/24/2024 |
1.0.0-alpha.5 | 57 | 1/19/2024 |
1.0.0-alpha.4 | 54 | 1/18/2024 |
1.0.0-alpha.3 | 63 | 1/17/2024 |
1.0.0-alpha.2 | 60 | 1/17/2024 |
1.0.0-alpha.1 | 63 | 1/16/2024 |