-
Notifications
You must be signed in to change notification settings - Fork 11
Home
Welcome to the AspNet-WebApi wiki!
Quote from an MDN article:
Cross-Origin Resource Sharing (CORS) is a mechanism that uses additional HTTP headers to tell a browser to let a web application running at one origin (domain) have permission to access selected resources from a server at a different origin. A web application makes a cross-origin HTTP request when it requests a resource that has a different origin (domain, protocol, and port) than its own origin.
By default, all origins are allowed. You can set up origin by uncommenting this line in web.config. To specify more than one origin split them by semicolon.
For example:
<appSettings>
<add key="allowed-origins" value="http://localhost:5000; https://api.some-domain.com"/>
</appSettings>
If you want more control you can adjust CORS-policy in App_Start/CorsConfig.cs.
More information can be found here.
Dependency Injection is a form of inversion of control that supports the dependency inversion design principle.
This project uses Autofac as an IoC Container but it can be replaced with any other if you will.
First, you should to configure the container. It can be done in RegisterServices method in App_Start/AutofacConfig.cs file.
For example:
builder.RegisterType<FileStore>().As<IFileStore>().SingleInstance();
builder.RegisterType<ImageMetadataLoader>().As<IImageMetadataLoader>();
builder.RegisterType<ImageRotationService>().AsSelf();
Second, you typically use registered types and interfaces in constructors of your controllers. One service can use another that way of course.
For more complex scenarios and examples see the official documentation.
IImageMetadataLoader MetadataLoader { get; }
public ProductsController(IImageMetadataLoader metadataLoader)
{
MetadataLoader = metadataLoader ?? throw new ArgumentNullException(nameof(metadataLoader));
}
metadataLoader
parameter will be autowired by the container.
You can read more on this in the official Autofac documentation.
When you create a web-service it is a common case to map one object to another. For example, you may want to map entity read from DB to DTO (Data Transmission Object) to return it as a response. You can write your own code to do that mapping or use a ready-made solution. Automapper is a solution to this problem.
Quote from the official site:
AutoMapper is a simple little library built to solve a deceptively complex problem - getting rid of code that mapped one object to another
The detailed information about using it in different cases you can get from an official documentation.
As an additional advantage of using Automapper, you get a segregation of mapping code from using it. Here we will focus on the most common cases and some simple examples to give you a starting point.
Automapper is already configured for you and ready to use. It is integrated with Autofac so you can inject mapper to your services or controllers, or you can use Autofac to resolve services in mapping code! Let's see a simple example of how to achieve this.
Let's say you need to return Dto.Product object from some method of your controller, but you have Model.Product in your code. First, we should create a mapping from Model.Product to Dto.Product. This can be done in Configure method in App_Start/AutomapperConfig.cs:
public static IMapper Configure(HttpConfiguration config)
{
Action<IMapperConfigurationExpression> mapperConfigurationExp = cfg =>
{
cfg.ConstructServicesUsing(GetResolver(config));
// TODO: Create mappings here
// For more information see https://github.com/drwatson1/AspNet-WebApi/wiki#automapper
// New line of code. Detailed configuration skipped for simplicity
cfg.CreateMap<Model.Product, Dto.Product>();
};
var mapperConfiguration = new MapperConfiguration(mapperConfigurationExp);
Mapper = mapperConfiguration.CreateMapper();
return Mapper;
}
Second, use it in our controller:
public class ProductsController: ApiController
{
IMapper Mapper { get; }
IProductsRepo Repo { get; }
public ProductsController(IMapper mapper, IProductsRepo repo)
{
Mapper = mapper ?? throw new ArgumentNullException(nameof(mapper));
Repo = repo ?? throw new ArgumentNullException(nameof(repo));
}
[HttpGet]
Dto.Product Get(int key)
{
return Mapper.Map<Dto.Product>(Repo.GetById(key))
}
[HttpGet]
IEnumerable<Dto.Product> Get()
{
return Repo.Get(key).Select(Mapper.Map<Dto.Product>);
}
}
Let's see a couple of examples of how to configure mappings:
cfg.CreateMap<Model.Product, Dto.Product>()
// Using a custom constructor
.ConstructUsing(src => new Dto.Product(src.Id))
// Using a custom formatting
.ForMember(x => x.Date, o => o.ResolveUsing((src, dest, destMember, context) =>
$"{src.Year}-{src.Month:D2}-{src.Day:D2}"))
// Calculated value from another fields of original object
.ForMember(x => x.Contract, o => o.ResolveUsing((src, dest, destMember, context) =>
$"{src.Contract.Number} from {src.Contract.Date}"))
.AfterMap((src, dest, ctx) =>
{
// Resolve service from DI-container
dest.SupplierName = c.Options.CreateInstance<ISuppliersRepo>().GetById(src.SupplierId);
});
Content
- Getting Started
- Using PackageReference instead of package.config
- Cross-Origin Resource Sharing (CORS)
- Dependency Injection
- Automapper
- Logging
- Cache control
- Unhandled exceptions logging
- Unhandled exceptions handling
- Content formatting
- Using environment variables in configuration options
- Documenting API
- Using EntityFramework