Skip to content

Commit

Permalink
Incluse some support to NuGet in csproj, image e README<draft> to val…
Browse files Browse the repository at this point in the history
…idate on NuGet.org
  • Loading branch information
evertondavid committed Jul 22, 2024
1 parent 9a46117 commit 51d6694
Show file tree
Hide file tree
Showing 4 changed files with 361 additions and 1 deletion.
8 changes: 8 additions & 0 deletions HATEOAS/HATEOAS.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -17,12 +17,20 @@
<RepositoryType>git</RepositoryType>
<PackgeTags>hateoas, hypermedia, restful, aspnetcore</PackgeTags>
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
<IncludeSymbols>true</IncludeSymbols>
<SymbolPackageFormat>snupkg</SymbolPackageFormat>
<PackageIcon>everton-hateoas-128x128.png</PackageIcon>
<PackageLicenseExpression>MIT</PackageLicenseExpression>
<PackageReadmeFile>README.md</PackageReadmeFile>



</PropertyGroup>

<ItemGroup>
<FrameworkReference Include="Microsoft.AspNetCore.App" />
<Nome Include="docs/README.md" Pack="true" PackagePath="/"/>
<Nome Include="icons/everton-hateoas-128x128.png" Pack="true" PackagePath="/"/>
</ItemGroup>

</Project>
1 change: 0 additions & 1 deletion README.md

This file was deleted.

353 changes: 353 additions & 0 deletions docs/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,353 @@
# HATEOAS-NuGet

[![NuGet](https://img.shields.io/nuget/v/HATEOAS-NuGet.svg)](https://www.nuget.org/packages/HATEOAS-NuGet)
[![Donate with PayPal](https://img.shields.io/badge/Donate-PayPal-green.svg)](https://www.paypal.com/donate/?hosted_button_id=ZJ4NQJXEKQ63A)
[![GitHub repo](https://img.shields.io/badge/GitHub-Repository-green.svg)](https://github.com/evertondavid/HATEOAS-NuGet)
![Last Commit](https://img.shields.io/github/last-commit/evertondavid/HATEOAS-NuGet)
![Forks](https://github.com/evertondavid/HATEOAS-NuGet/actions/workflows/continous-integration-nuget.yml/badge.svg?event=fork)
![Stars](https://img.shields.io/github/stars/evertondavid/HATEOAS-NuGet)
[![.NET 8 Continuous Integration with GitHub, GitHub Actions and Nuget Packages](https://github.com/evertondavid/HATEOAS-NuGet/actions/workflows/continous-integration-nuget.yml/badge.svg)](https://github.com/evertondavid/HATEOAS-NuGet/actions/workflows/continous-integration-nuget.yml)

This is a smart library to implements HATEOAS pattern in your RESTFul API's, implemented based in [this project](https://github.com/SotirisH/HyperMedia).

> ## How to use
>### 1 - Import HATEOAS-NuGet to your projetct
#### Import with command line
```bash
Install-Package HATEOAS -Version 8.0.302.2
```

#### Import with nuget package manager

![Nuget Package Mannager](https://github.com/leandrocgsi/Erudio.HATEOAS/blob/main/images/nuget_package_mannager.png?raw=true)

>### 2 - Implements *ISupportsHyperMedia* in your exposed object.
```csharp
namespace RESTFulSampleServer.Data.VO
{
public class BookVO : ISupportsHyperMedia
{
public long? Id { get; set; }
public string Title { get; set; }
public string Author { get; set; }
public decimal Price { get; set; }
public DateTime LaunchDate { get; set; }

public List<HyperMediaLink> Links { get; set; } = new List<HyperMediaLink>();
}
}
```

>### 3 - Implements your enricher with *ContentResponseEnricher<T>*.
```csharp
namespace RESTFulSampleServer.HyperMedia.Enricher
{
public class BookEnricher : ContentResponseEnricher<BookVO>
{
protected override Task EnrichModel(BookVO content, IUrlHelper urlHelper)
{
var path = "api/book";
string link = GetLink(content.Id, urlHelper, path);

content.Links.Add(new HyperMediaLink()
{
Action = HttpActionVerb.GET,
Href = link,
Rel = RelationType.self,
Type = ResponseTypeFormat.DefaultGet
});
content.Links.Add(new HyperMediaLink()
{
Action = HttpActionVerb.POST,
Href = link,
Rel = RelationType.self,
Type = ResponseTypeFormat.DefaultPost
});
content.Links.Add(new HyperMediaLink()
{
Action = HttpActionVerb.PUT,
Href = link,
Rel = RelationType.self,
Type = ResponseTypeFormat.DefaultPut
});
content.Links.Add(new HyperMediaLink()
{
Action = HttpActionVerb.DELETE,
Href = link,
Rel = RelationType.self,
Type = "int"
});
return Task.CompletedTask;
}

private string GetLink(long id, IUrlHelper urlHelper, string path)
{
lock (this)
{
var url = new { controller = path, id };
return new StringBuilder(urlHelper.Link("DefaultApi", url)).Replace("%2F", "/").ToString();
};
}
}
}
```

>### 4 - Add annotation *[TypeFilter(typeof(HyperMediaFilter))]* to your controller methods.
```csharp
namespace RESTFulSampleServer.Controllers
{

[ApiVersion("1")]
[ApiController]
[Authorize("Bearer")]
[Route("api/[controller]/v{version:apiVersion}")]
public class BookController : ControllerBase
{

private readonly ILogger<BookController> _logger;

private IBookBusiness _bookBusiness;

public BookController(ILogger<BookController> logger, IBookBusiness bookBusiness)
{
_logger = logger;
_bookBusiness = bookBusiness;
}

[HttpGet]
[ProducesResponseType((200), Type = typeof(List<BookVO>))]
[ProducesResponseType(204)]
[ProducesResponseType(400)]
[ProducesResponseType(401)]

// Adds HyperMedia filter
[TypeFilter(typeof(HyperMediaFilter))]
public IActionResult Get()
{
return Ok(_bookBusiness.FindAll());
}

[HttpGet("{id}")]
[ProducesResponseType((200), Type = typeof(BookVO))]
[ProducesResponseType(204)]
[ProducesResponseType(400)]
[ProducesResponseType(401)]

// Adds HyperMedia filter
[TypeFilter(typeof(HyperMediaFilter))]
public IActionResult Get(long id)
{
var book = _bookBusiness.FindByID(id);
if (book == null) return NotFound();
return Ok(book);
}

[HttpPost]
[ProducesResponseType((200), Type = typeof(BookVO))]
[ProducesResponseType(400)]
[ProducesResponseType(401)]

// Adds HyperMedia filter
[TypeFilter(typeof(HyperMediaFilter))]
public IActionResult Post([FromBody] BookVO book)
{
if (book == null) return BadRequest();
return Ok(_bookBusiness.Create(book));
}

[HttpPut]
[ProducesResponseType((200), Type = typeof(BookVO))]
[ProducesResponseType(400)]
[ProducesResponseType(401)]

// Adds HyperMedia filter
[TypeFilter(typeof(HyperMediaFilter))]
public IActionResult Put([FromBody] BookVO book)
{
if (book == null) return BadRequest();
return Ok(_bookBusiness.Update(book));
}

[HttpDelete("{id}")]
[ProducesResponseType(204)]
[ProducesResponseType(400)]
[ProducesResponseType(401)]
public IActionResult Delete(long id)
{
_bookBusiness.Delete(id);
return NoContent();
}
}
}

```

>### 5 - Add *HyperMediaFilterOptions* to your Program.cs.
```csharp
var filterOptions = new HyperMediaFilterOptions();
filterOptions.ContentResponseEnricherList.Add(new PersonEnricher());
filterOptions.ContentResponseEnricherList.Add(new BookEnricher());
```

>### 6 - Add a *MapControllerRoute* to your route like was defined in your enricher.
```csharp
app.MapControllers();
* app.MapControllerRoute("DefaultApi", "{controller=values}/v{version=apiVersion}/{id?}"); *

app.Run();
```
>### 7 - Enjoy
#### Response as JSON

```json
[
{
"id": 1,
"title": "Working effectively with legacy code",
"author": "Michael C. Feathers",
"price": 49.00,
"launchDate": "2017-11-29T13:50:05.878",
"links": [
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/1",
"type": "application/json",
"action": "GET"
},
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/1",
"type": "application/json",
"action": "POST"
},
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/1",
"type": "application/json",
"action": "PUT"
},
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/1",
"type": "int",
"action": "DELETE"
}
]
},
{
"id": 2,
"title": "Design Patterns",
"author": "Ralph Johnson, Erich Gamma, John Vlissides e Richard Helm",
"price": 45.00,
"launchDate": "2017-11-29T15:15:13.636",
"links": [
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/2",
"type": "application/json",
"action": "GET"
},
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/2",
"type": "application/json",
"action": "POST"
},
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/2",
"type": "application/json",
"action": "PUT"
},
{
"rel": "self",
"href": "https://localhost:44300/api/book/v1/2",
"type": "int",
"action": "DELETE"
}
]
}
]
```

#### Response as XML

```xml
<ArrayOfBookVO xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<BookVO>
<Id>1</Id>
<Title>Working effectively with legacy code</Title>
<Author>Michael C. Feathers</Author>
<Price>49.00</Price>
<LaunchDate>2017-11-29T13:50:05.878</LaunchDate>
<Links>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/1</Href>
<Type>application/json</Type>
<Action>GET</Action>
</HyperMediaLink>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/1</Href>
<Type>application/json</Type>
<Action>POST</Action>
</HyperMediaLink>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/1</Href>
<Type>application/json</Type>
<Action>PUT</Action>
</HyperMediaLink>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/1</Href>
<Type>int</Type>
<Action>DELETE</Action>
</HyperMediaLink>
</Links>
</BookVO>
<BookVO>
<Id>2</Id>
<Title>Design Patterns</Title>
<Author>Ralph Johnson, Erich Gamma, John Vlissides e Richard Helm</Author>
<Price>45.00</Price>
<LaunchDate>2017-11-29T15:15:13.636</LaunchDate>
<Links>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/2</Href>
<Type>application/json</Type>
<Action>GET</Action>
</HyperMediaLink>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/2</Href>
<Type>application/json</Type>
<Action>POST</Action>
</HyperMediaLink>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/2</Href>
<Type>application/json</Type>
<Action>PUT</Action>
</HyperMediaLink>
<HyperMediaLink>
<Rel>self</Rel>
<Href>https://localhost:44300/api/book/v1/2</Href>
<Type>int</Type>
<Action>DELETE</Action>
</HyperMediaLink>
</Links>
</BookVO>
</ArrayOfBookVO>
```

>### Suggestions are welcome. Feel free to sugest improvements.
Binary file added icons/everton-hateoas-128x128.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.

0 comments on commit 51d6694

Please sign in to comment.