feat: add data shaping to cuntries controller

This commit is contained in:
cuqmbr 2022-11-03 10:50:41 +02:00
parent 4b5462e889
commit 8bf18c948e
8 changed files with 109 additions and 18 deletions

View File

@ -46,14 +46,14 @@ public class CountryManagementController : ControllerBase
} }
[HttpGet("{id}")] [HttpGet("{id}")]
public async Task<IActionResult> GetCountry(int id) public async Task<IActionResult> GetCountry(int id, [FromQuery] string? fields)
{ {
if (!await _countryManagementService.IsCountryExists(id)) if (!await _countryManagementService.IsCountryExists(id))
{ {
return NotFound(); return NotFound();
} }
var result = await _countryManagementService.GetCountry(id); var result = await _countryManagementService.GetCountry(id, fields);
if (!result.isSucceed) if (!result.isSucceed)
{ {
@ -78,7 +78,7 @@ public class CountryManagementController : ControllerBase
return BadRequest(result.message); return BadRequest(result.message);
} }
return Ok(result); return Ok(result.country);
} }
[HttpDelete("{id}")] [HttpDelete("{id}")]

View File

@ -0,0 +1,69 @@
using System.Dynamic;
using System.Reflection;
namespace Server.Helpers;
public class DataShaper<T> : IDataShaper<T>
{
public PropertyInfo[] Properties { get; set; }
public DataShaper()
{
Properties = typeof(T).GetProperties(BindingFlags.Public | BindingFlags.Instance);
}
public IEnumerable<ExpandoObject> ShapeData(IEnumerable<T> entities, string? fieldsString)
{
var requiredProperties = GetRequiredProperties(fieldsString);
return FetchData(entities, requiredProperties);
}
public ExpandoObject ShapeData(T entity, string? fieldsString)
{
var requiredProperties = GetRequiredProperties(fieldsString);
return FetchDataForEntity(entity, requiredProperties);
}
private IEnumerable<PropertyInfo> GetRequiredProperties(string? fieldsString)
{
var requiredProperties = new List<PropertyInfo>();
if (!string.IsNullOrWhiteSpace(fieldsString))
{
var fields = fieldsString.Split(',', StringSplitOptions.RemoveEmptyEntries);
foreach (var field in fields)
{
var property = Properties.FirstOrDefault(pi => pi.Name.Equals(field.Trim(), StringComparison.InvariantCultureIgnoreCase));
if (property == null)
continue;
requiredProperties.Add(property);
}
}
else
{
requiredProperties = Properties.ToList();
}
return requiredProperties;
}
private IEnumerable<ExpandoObject> FetchData(IEnumerable<T> entities, IEnumerable<PropertyInfo> requiredProperties)
{
var shapedData = new List<ExpandoObject>();
foreach (var entity in entities)
{
var shapedObject = FetchDataForEntity(entity, requiredProperties);
shapedData.Add(shapedObject);
}
return shapedData;
}
private ExpandoObject FetchDataForEntity(T entity, IEnumerable<PropertyInfo> requiredProperties)
{
var shapedObject = new ExpandoObject();
foreach (var property in requiredProperties)
{
var objectPropertyValue = property.GetValue(entity);
shapedObject.TryAdd(property.Name, objectPropertyValue);
}
return shapedObject;
}
}

View File

@ -0,0 +1,9 @@
using System.Dynamic;
namespace Server.Helpers;
public interface IDataShaper<T>
{
IEnumerable<ExpandoObject> ShapeData(IEnumerable<T> entities, string? fieldsString);
ExpandoObject ShapeData(T entity, string? fieldsString);
}

View File

@ -5,6 +5,7 @@ using Microsoft.EntityFrameworkCore;
using Microsoft.IdentityModel.Tokens; using Microsoft.IdentityModel.Tokens;
using Microsoft.OpenApi.Models; using Microsoft.OpenApi.Models;
using Newtonsoft.Json; using Newtonsoft.Json;
using Newtonsoft.Json.Serialization;
using Server.Configurations; using Server.Configurations;
using Server.Data; using Server.Data;
using Server.Helpers; using Server.Helpers;
@ -15,8 +16,11 @@ var builder = WebApplication.CreateBuilder(args);
// Add services to the container. // Add services to the container.
builder.Services.AddControllers().AddNewtonsoftJson(options => builder.Services.AddControllers().AddNewtonsoftJson(options => {
options.SerializerSettings.Formatting = Formatting.Indented); options.SerializerSettings.Formatting = Formatting.Indented;
options.SerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver();
});
// Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle // Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle
builder.Services.AddEndpointsApiExplorer(); builder.Services.AddEndpointsApiExplorer();
@ -88,6 +92,8 @@ builder.Services.AddScoped<IDateTimeService, DateTimeService>();
builder.Services.AddScoped<ISortHelper<Country>, SortHelper<Country>>(); builder.Services.AddScoped<ISortHelper<Country>, SortHelper<Country>>();
builder.Services.AddScoped<IDataShaper<Country>, DataShaper<Country>>();
// Adding DB Context with PostgreSQL // Adding DB Context with PostgreSQL
var connectionString = builder.Configuration.GetConnectionString("DefaultConnection"); var connectionString = builder.Configuration.GetConnectionString("DefaultConnection");
builder.Services.AddDbContext<ApplicationDbContext>(options => builder.Services.AddDbContext<ApplicationDbContext>(options =>

View File

@ -1,3 +1,4 @@
using System.Dynamic;
using System.Linq.Dynamic.Core; using System.Linq.Dynamic.Core;
using AutoMapper; using AutoMapper;
using AutoMapper.QueryableExtensions; using AutoMapper.QueryableExtensions;
@ -15,13 +16,16 @@ public class CountryManagementService : ICountryManagementService
private readonly ApplicationDbContext _dbContext; private readonly ApplicationDbContext _dbContext;
private readonly IMapper _mapper; private readonly IMapper _mapper;
private readonly ISortHelper<Country> _countrySortHelper; private readonly ISortHelper<Country> _countrySortHelper;
private readonly IDataShaper<Country> _countryDataShaper;
public CountryManagementService(ApplicationDbContext dbContext, public CountryManagementService(ApplicationDbContext dbContext,
IMapper mapper, ISortHelper<Country> countrySortHelper) IMapper mapper, ISortHelper<Country> countrySortHelper,
IDataShaper<Country> countryDataShaper)
{ {
_dbContext = dbContext; _dbContext = dbContext;
_mapper = mapper; _mapper = mapper;
_countrySortHelper = countrySortHelper; _countrySortHelper = countrySortHelper;
_countryDataShaper = countryDataShaper;
} }
public async Task<(bool isSucceed, string message, CountryDto country)> AddCountry(CreateCountryDto createCountryDto) public async Task<(bool isSucceed, string message, CountryDto country)> AddCountry(CreateCountryDto createCountryDto)
@ -34,7 +38,7 @@ public class CountryManagementService : ICountryManagementService
return (true, String.Empty, _mapper.Map<CountryDto>(country)); return (true, String.Empty, _mapper.Map<CountryDto>(country));
} }
public async Task<(bool isSucceed, string message, IEnumerable<CountryDto> countries, public async Task<(bool isSucceed, string message, IEnumerable<ExpandoObject> countries,
PagingMetadata<Country> pagingMetadata)> GetCountries(CountryParameters parameters) PagingMetadata<Country> pagingMetadata)> GetCountries(CountryParameters parameters)
{ {
var dbCountries = _dbContext.Countries.AsQueryable(); var dbCountries = _dbContext.Countries.AsQueryable();
@ -59,10 +63,9 @@ public class CountryManagementService : ICountryManagementService
var pagingMetadata = ApplyPaging(ref dbCountries, parameters.PageNumber, var pagingMetadata = ApplyPaging(ref dbCountries, parameters.PageNumber,
parameters.PageSize); parameters.PageSize);
var countryDtos = var shapedCountiesData = _countryDataShaper.ShapeData(dbCountries, parameters.Fields);
dbCountries.ProjectTo<CountryDto>(_mapper.ConfigurationProvider);
return (true, "", countryDtos, pagingMetadata); return (true, "", shapedCountiesData, pagingMetadata);
void SearchByAllCountryFields(ref IQueryable<Country> countries, void SearchByAllCountryFields(ref IQueryable<Country> countries,
string? search) string? search)
@ -117,21 +120,22 @@ public class CountryManagementService : ICountryManagementService
} }
} }
public async Task<(bool isSucceed, string message, CountryDto country)> GetCountry(int id) public async Task<(bool isSucceed, string message, ExpandoObject country)> GetCountry(int id, string? fields)
{ {
var dbCountry = await _dbContext.Countries.Where(c => c.Id == id) var dbCountry = await _dbContext.Countries.Where(c => c.Id == id)
.ProjectTo<CountryDto>(_mapper.ConfigurationProvider)
.FirstOrDefaultAsync(); .FirstOrDefaultAsync();
if (dbCountry == null) if (dbCountry == null)
{ {
return (false, $"Country doesn't exist", null)!; return (false, $"Country doesn't exist", null)!;
} }
var shapedCountryData = _countryDataShaper.ShapeData(dbCountry, fields);
return (true, "", dbCountry); return (true, "", shapedCountryData);
} }
public async Task<(bool isSucceed, string message, CountryDto country)> UpdateCountry(UpdateCountryDto updateCountryDto) public async Task<(bool isSucceed, string message, UpdateCountryDto country)> UpdateCountry(UpdateCountryDto updateCountryDto)
{ {
var country = _mapper.Map<Country>(updateCountryDto); var country = _mapper.Map<Country>(updateCountryDto);
_dbContext.Entry(country).State = EntityState.Modified; _dbContext.Entry(country).State = EntityState.Modified;
@ -152,7 +156,7 @@ public class CountryManagementService : ICountryManagementService
var dbCountry = await _dbContext.Countries.FirstOrDefaultAsync(c => c.Id == country.Id); var dbCountry = await _dbContext.Countries.FirstOrDefaultAsync(c => c.Id == country.Id);
return (true, String.Empty, _mapper.Map<CountryDto>(dbCountry)); return (true, String.Empty, _mapper.Map<UpdateCountryDto>(dbCountry));
} }
public async Task<(bool isSucceed, string message)> DeleteCountry(int id) public async Task<(bool isSucceed, string message)> DeleteCountry(int id)

View File

@ -1,3 +1,4 @@
using System.Dynamic;
using Server.Models; using Server.Models;
using SharedModels.DataTransferObjects; using SharedModels.DataTransferObjects;
using SharedModels.QueryStringParameters; using SharedModels.QueryStringParameters;
@ -8,10 +9,10 @@ public interface ICountryManagementService
{ {
Task<(bool isSucceed, string message, CountryDto country)> AddCountry(CreateCountryDto createCountryDto); Task<(bool isSucceed, string message, CountryDto country)> AddCountry(CreateCountryDto createCountryDto);
Task<(bool isSucceed, string message, IEnumerable<CountryDto> countries, Task<(bool isSucceed, string message, IEnumerable<ExpandoObject> countries,
PagingMetadata<Country> pagingMetadata)> GetCountries(CountryParameters parameters); PagingMetadata<Country> pagingMetadata)> GetCountries(CountryParameters parameters);
Task<(bool isSucceed, string message, CountryDto country)> GetCountry(int id); Task<(bool isSucceed, string message, ExpandoObject country)> GetCountry(int id, string? fields);
Task<(bool isSucceed, string message, CountryDto country)> UpdateCountry(UpdateCountryDto updateCountryDto); Task<(bool isSucceed, string message, UpdateCountryDto country)> UpdateCountry(UpdateCountryDto updateCountryDto);
Task<(bool isSucceed, string message)> DeleteCountry(int id); Task<(bool isSucceed, string message)> DeleteCountry(int id);
Task<bool> IsCountryExists(int id); Task<bool> IsCountryExists(int id);
} }

View File

@ -5,6 +5,7 @@ public class CountryParameters : QueryStringParameters
public CountryParameters() public CountryParameters()
{ {
Sort = "id"; Sort = "id";
Fields = "id,code,name,states";
} }
public string? Code { get; set; } public string? Code { get; set; }

View File

@ -14,4 +14,5 @@ public class QueryStringParameters
public string? Search { get; set; } public string? Search { get; set; }
public string? Sort { get; set; } public string? Sort { get; set; }
public string? Fields { get; set; }
} }