Quando pensamos em API RESTful no C#, a primeira opção que vem a mente é o ASP.NET Web API. Ele é um ótimo framework, utilizado em vários sistemas, com uma abundante documentação e material de apoio. Mas ele não é a nossa única opção para a criação de API Rest.
Em artigos passados, conhecemos o ServiceStack, mas outra opção que está crescendo é o NancyFX.
Inspirado no framework Sinatra do Ruby, o Nancy é um framework de criação de serviços HTTP, leve e direto ao ponto. Tem por objetivo não atrapalhar o desenvolvedor. Ele obtém isso através de uma série de configurações padrões e convenções. Assim, com o Nancy é possível criar um site em minutos.
Claro que todas as suas convenções e configurações podem ser alteradas, caso este seja o desejo do desenvolvedor.
Para este artigo, mostrarei como é possível criar uma API simples, rapidamente utilizando o Nancy.
Criando a aplicação
Neste artigo estou utilizando o .NET Core, desta forma, mostrarei como criar a aplicação através do terminal. Inicialmente é necessário criar uma aplicação web vazia:
dotnet new web -n NancyAPI
No projeto criado, adicione a referência do Nancy:
dotnet add package Nancy --version 2.0.0-clinteastwood
Agora é necessário dizer ao ASP.NET que iremos utilizar o Nancy, para isso, faremos uso do OWIN.
O OWIN significa “Open Web Interface for .Net. Ele é um conjunto de padrões voltados para o .NET, que visa facilitar e encorajar a implementação de projetos que tentam desacoplar a aplicação do servidor. Na prática ele define um middleware ao pipeline de execução de uma aplicação ASP.NET.
Por se tratar de um middleware, deve ser configurado no método Configure
da classe Startup
:
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
app.UseOwin(o => o.UseNancy());
}
Com isso, o Nancy já está ativo na nossa aplicação.
Módulos
No Nancy os endpoints devem ser definidos em módulos. Um módulo nada mais é que uma classe que herde a classe NancyModule
. Esta classe pode ser definida em qualquer ponto do projeto, pois o Nancy fará uma busca pelos módulos quando um endpoint for invocado. O único ponto importante é que esta classe seja definida como pública (public
) para que possa ser localizada.
Por exemplo:
using Nancy;
namespace NancyAPI.Module
{
public class HomeModule : NancyModule
{
public HomeModule()
{
Get("/", _ => "Hello World from Nancy!");
}
}
}
Note que a rota está sendo definida como o primeiro parâmetro do método Get
. Como é possível supor, esta rota será invocada quando for solicitada através de uma requisição GET
.
O Nancy define métodos para os principais verbos do HTTP:
-
Delete
=DELETE
; -
Get
=GET
; -
Head
=HEAD
; -
Options
=OPTIONS
; -
Post
=POST
; -
Put
=PUT
; -
Patch
=PATCH
.
No segundo parâmetro do método deve ser passado a referência de uma função ( uma Function
), que no exemplo acima, se utiliza lambda para definir a função. Este é o padrão recomendado pelo Nancy.
Ao executar a aplicação e acessar o endpoint definido, teremos o resultado:
Simples, não é?
Definindo a API RESTful
Para exemplificar uma API REST, vamos definir um modelo:
namespace NancyAPI.Models
{
public class Pessoa
{
public int Id { get; set; }
public string Nome { get; set; }
}
}
E um repositório:
using System.Collections.Generic;
using NancyAPI.Models;
using System.Linq;
namespace NancyAPI.Repositories
{
public class PessoaRepository
{
private static Dictionary<int, Pessoa> pessoas = new Dictionary<int, Pessoa>();
public List<Pessoa> GetAll(){
return pessoas.Values.ToList();
}
public Pessoa Get(int id){
return pessoas.GetValueOrDefault(id);
}
public void Add(Pessoa pessoa){
pessoas.Add(pessoa.Id, pessoa);
}
public void Edit(Pessoa pessoa){
pessoas.Remove(pessoa.Id);
pessoas.Add(pessoa.Id, pessoa);
}
public void Delete(int id){
pessoas.Remove(id);
}
}
}
Agora só falta o nosso endpoint:
using Nancy;
using Nancy.ModelBinding;
using NancyAPI.Models;
using NancyAPI.Repositories;
namespace NancyAPI.Module
{
public class PessoaModule: NancyModule
{
public readonly PessoaRepository repository;
public PessoaModule()
{
repository = new PessoaRepository();
Get("/pessoa/", _ => repository.GetAll());
Get("/pessoa/{id}", args => repository.Get(args.id));
Post("/pessoa/", args => {
var pessoa = this.Bind<Pessoa>();
repository.Add(pessoa);
return pessoa;
});
Put("/pessoa/{id}", args => {
var pessoa = this.Bind<Pessoa>();
pessoa.Id = args.id;
repository.Edit(pessoa);
return pessoa;
});
}
}
}
Neste endpoint é possível notar que os argumentos da URL são obtidos através do parâmetro do lambda:
Get("/pessoa/{id}", args => repository.Get(args.id));
Este parâmetro é um objeto dinâmico. Assim, o Nancy tentará obter o valor do argumento com base na propriedade informada.
Já os dados passados na solicitação, podem ser obtidos pelo Bind
:
var pessoa = this.Bind<Pessoa>();
O Nancy irá converter automaticamente os dados obtidos para o tipo de objeto indicado.
Para testar os endpoints, podemos utilizar o Postman:
POST
GET
PUT
GET ID
Conclusão
Graças a sua estrutura simples, porém sofisticada, o Nancy tem atraído cada vez mais a atenção dos desenvolvedores. Se tornando uma ótima alternativa ao ASP.NET Web API.
Caso necessite criar uma API REST, não esqueça de dar uma boa olhada nele.
Curso C# (C Sharp) - APIs REST com ASP.NET Web API
Conhecer o cursoVocê pode ver o projeto apresentado neste artigo no meu GitHub.