From 108d7e67bb66e737a965f3af29a1638777242398 Mon Sep 17 00:00:00 2001 From: Lucca Santangelo <luccasant95@gmail.com> Date: Sun, 29 Nov 2020 16:56:29 -0300 Subject: [PATCH] bedelia --- .../Tsi1.Api/Controllers/CourseController.cs | 43 ++++++++++++- Tsi1.Api/Tsi1.Api/Startup.cs | 10 +++ Tsi1.Api/Tsi1.Api/appsettings.json | 2 + .../Tsi1.BusinessLayer/Dtos/CloseRecordDto.cs | 12 ++++ .../Tsi1.BusinessLayer/Dtos/UserGradeDto.cs | 12 ++++ .../Helpers/ErrorMessages.cs | 1 + .../Interfaces/IBedeliaService.cs | 16 +++++ .../Interfaces/IUserService.cs | 2 + .../Services/BedeliaService.cs | 61 +++++++++++++++++++ .../Services/UserService.cs | 26 ++++++++ .../Tsi1.BusinessLayer.csproj | 1 + 11 files changed, 183 insertions(+), 3 deletions(-) create mode 100644 Tsi1.Api/Tsi1.BusinessLayer/Dtos/CloseRecordDto.cs create mode 100644 Tsi1.Api/Tsi1.BusinessLayer/Dtos/UserGradeDto.cs create mode 100644 Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IBedeliaService.cs create mode 100644 Tsi1.Api/Tsi1.BusinessLayer/Services/BedeliaService.cs diff --git a/Tsi1.Api/Tsi1.Api/Controllers/CourseController.cs b/Tsi1.Api/Tsi1.Api/Controllers/CourseController.cs index 3778365..dd4de59 100644 --- a/Tsi1.Api/Tsi1.Api/Controllers/CourseController.cs +++ b/Tsi1.Api/Tsi1.Api/Controllers/CourseController.cs @@ -16,10 +16,14 @@ namespace Tsi1.Api.Controllers public class CourseController : ControllerBase { private readonly ICourseService _courseService; + private readonly IUserService _userService; + private readonly IBedeliaService _bedeliaService; - public CourseController(ICourseService courseService) + public CourseController(ICourseService courseService, IUserService userService, IBedeliaService bedeliaService) { _courseService = courseService; + _userService = userService; + _bedeliaService = bedeliaService; } [Authorize(Roles = UserTypes.Student + ", " + UserTypes.Professor)] @@ -54,15 +58,20 @@ namespace Tsi1.Api.Controllers return Ok(result.Data); } - [Authorize(Roles = UserTypes.Student)] [HttpPost("Matriculate/{courseId}")] public async Task<IActionResult> Matriculate(int courseId) { var userId = int.Parse(HttpContext.User.Claims.FirstOrDefault(x => x.Type == "Id").Value); + var user = await _userService.GetById(userId); + + var response = await _bedeliaService.IsValidUser(user.Data.Student.IdentityCard); + if (response.HasError) + { + return BadRequest(response.Message); + } var result = await _courseService.Matriculate(userId, courseId); - if (result.HasError) { return BadRequest(result.Message); @@ -245,5 +254,33 @@ namespace Tsi1.Api.Controllers return Ok(result.Data); } + + [Authorize(Roles = UserTypes.Professor)] + [HttpPost("CloseRecord/{courseId}")] + public async Task<IActionResult> CloseRecord(int courseId) + { + var courseResult = await _courseService.GetById(courseId); + if (courseResult.HasError) + { + return BadRequest(courseResult.Message); + } + + var userResult = await _userService.GetUserGrades(courseId); + if (userResult.HasError) + { + return BadRequest(userResult.Message); + } + + var courseName = courseResult.Data.Name; + var userGrades = userResult.Data; + + var result = await _bedeliaService.CloseRecord(courseName, userGrades); + if (result.HasError) + { + return BadRequest(result.Message); + } + + return Ok(); + } } } diff --git a/Tsi1.Api/Tsi1.Api/Startup.cs b/Tsi1.Api/Tsi1.Api/Startup.cs index 90bca24..b1f0216 100644 --- a/Tsi1.Api/Tsi1.Api/Startup.cs +++ b/Tsi1.Api/Tsi1.Api/Startup.cs @@ -46,17 +46,27 @@ namespace Tsi1.Api { string postgreSqlSection; string mongoDbSection; + string bedeliaBaseUrlSection; + if (_env.IsProduction()) { postgreSqlSection = "PostgreSqlCloud"; mongoDbSection = "Tsi1DatabaseSettingsCloud"; + bedeliaBaseUrlSection = "BedeliaBaseUrlCloud"; } else { postgreSqlSection = "PostgreSql"; mongoDbSection = "Tsi1DatabaseSettings"; + bedeliaBaseUrlSection = "BedeliaBaseUrl"; } + var bedeliaBaseUrl = Configuration.GetSection(bedeliaBaseUrlSection).Value; + services.AddHttpClient<IBedeliaService, BedeliaService>(configureClient => + { + configureClient.BaseAddress = new Uri(bedeliaBaseUrl); + }); + services.AddControllers(); services.AddSignalR(); diff --git a/Tsi1.Api/Tsi1.Api/appsettings.json b/Tsi1.Api/Tsi1.Api/appsettings.json index aeffabd..d77c42f 100644 --- a/Tsi1.Api/Tsi1.Api/appsettings.json +++ b/Tsi1.Api/Tsi1.Api/appsettings.json @@ -28,6 +28,8 @@ "Host": "smtp.gmail.com", "Port": 587 }, + "BedeliaBaseUrl": "http://localhost:55438/", + "BedeliaBaseUrlCloud": "http://tsi-bedelia.web.elasticloud.uy/", "Logging": { "LogLevel": { "Default": "Information", diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Dtos/CloseRecordDto.cs b/Tsi1.Api/Tsi1.BusinessLayer/Dtos/CloseRecordDto.cs new file mode 100644 index 0000000..f0964b6 --- /dev/null +++ b/Tsi1.Api/Tsi1.BusinessLayer/Dtos/CloseRecordDto.cs @@ -0,0 +1,12 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace Tsi1.BusinessLayer.Dtos +{ + public class CloseRecordDto + { + public string CourseName { get; set; } + public List<UserGradeDto> UserGrades { get; set; } + } +} diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Dtos/UserGradeDto.cs b/Tsi1.Api/Tsi1.BusinessLayer/Dtos/UserGradeDto.cs new file mode 100644 index 0000000..8018e60 --- /dev/null +++ b/Tsi1.Api/Tsi1.BusinessLayer/Dtos/UserGradeDto.cs @@ -0,0 +1,12 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace Tsi1.BusinessLayer.Dtos +{ + public class UserGradeDto + { + public string IdentityCard { get; set; } + public int Grade { get; set; } + } +} diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Helpers/ErrorMessages.cs b/Tsi1.Api/Tsi1.BusinessLayer/Helpers/ErrorMessages.cs index b049c02..5f59d5b 100644 --- a/Tsi1.Api/Tsi1.BusinessLayer/Helpers/ErrorMessages.cs +++ b/Tsi1.Api/Tsi1.BusinessLayer/Helpers/ErrorMessages.cs @@ -34,6 +34,7 @@ namespace Tsi1.BusinessLayer.Helpers public const string CourseDoesNotExist = "El curso con id '{0}' no existe"; public const string DuplicateCourseName = "Ya existe un curso con nombre '{0}'"; public const string CourseIsTemplate = "El curso con id '{0}' es un template"; + public const string CourseHasNoStudents = "El curso con id '{0}' no tiene estudiantes"; public const string TenantDoesNotExist = "La Facultad '{0}' no existe"; public const string DuplicateTenantName = "Ya existe una Facultad con nombre '{0}'"; diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IBedeliaService.cs b/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IBedeliaService.cs new file mode 100644 index 0000000..88a6f29 --- /dev/null +++ b/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IBedeliaService.cs @@ -0,0 +1,16 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.Threading.Tasks; +using Tsi1.BusinessLayer.Dtos; +using Tsi1.BusinessLayer.Helpers; + +namespace Tsi1.BusinessLayer.Interfaces +{ + public interface IBedeliaService + { + Task<ServiceResult<bool>> IsValidUser(string identityCard); + + Task<ServiceResult<bool>> CloseRecord(string courseName, List<UserGradeDto> userGrades); + } +} diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IUserService.cs b/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IUserService.cs index 1fa9902..e3a8920 100644 --- a/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IUserService.cs +++ b/Tsi1.Api/Tsi1.BusinessLayer/Interfaces/IUserService.cs @@ -34,5 +34,7 @@ namespace Tsi1.BusinessLayer.Interfaces Task<ServiceResult<List<UserPreviewDto>>> GetProfessors(int tenantId); Task<ServiceResult<List<UserPreviewDto>>> GetAdmins(int tenantId, string userType); + + Task<ServiceResult<List<UserGradeDto>>> GetUserGrades(int courseId); } } diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Services/BedeliaService.cs b/Tsi1.Api/Tsi1.BusinessLayer/Services/BedeliaService.cs new file mode 100644 index 0000000..cd7a08a --- /dev/null +++ b/Tsi1.Api/Tsi1.BusinessLayer/Services/BedeliaService.cs @@ -0,0 +1,61 @@ +using Newtonsoft.Json; +using System; +using System.Collections.Generic; +using System.Net; +using System.Net.Http; +using System.Text; +using System.Threading.Tasks; +using Tsi1.BusinessLayer.Dtos; +using Tsi1.BusinessLayer.Helpers; +using Tsi1.BusinessLayer.Interfaces; + +namespace Tsi1.BusinessLayer.Services +{ + public class BedeliaService : IBedeliaService + { + private readonly HttpClient _httpClient; + + public BedeliaService(HttpClient httpClient) + { + _httpClient = httpClient; + } + + public async Task<ServiceResult<bool>> IsValidUser(string identityCard) + { + var result = new ServiceResult<bool>(); + + var response = await _httpClient.GetAsync($"api/Users/{identityCard}"); + if (response.StatusCode != HttpStatusCode.OK) + { + result.HasError = true; + var errorMessage = await response.Content.ReadAsStringAsync(); + result.AddMessage(errorMessage); + } + + return result; + } + + public async Task<ServiceResult<bool>> CloseRecord(string courseName, List<UserGradeDto> userGrades) + { + var result = new ServiceResult<bool>(); + var closeRecord = new CloseRecordDto() + { + CourseName = courseName, + UserGrades = userGrades + }; + + var jsonInString = JsonConvert.SerializeObject(closeRecord); + var model = new StringContent(jsonInString, Encoding.UTF8, "application/json"); + + var response = await _httpClient.PostAsync("api/Courses/closeRecord", model); + if (response.StatusCode != HttpStatusCode.OK) + { + result.HasError = true; + var errorMessage = await response.Content.ReadAsStringAsync(); + result.AddMessage(errorMessage); + } + + return result; + } + } +} diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Services/UserService.cs b/Tsi1.Api/Tsi1.BusinessLayer/Services/UserService.cs index ca09556..445386e 100644 --- a/Tsi1.Api/Tsi1.BusinessLayer/Services/UserService.cs +++ b/Tsi1.Api/Tsi1.BusinessLayer/Services/UserService.cs @@ -358,5 +358,31 @@ namespace Tsi1.BusinessLayer.Services return result; } + + public async Task<ServiceResult<List<UserGradeDto>>> GetUserGrades(int courseId) + { + var result = new ServiceResult<List<UserGradeDto>>(); + + var course = await _context.Courses + .Include(x => x.StudentCourses) + .ThenInclude(x => x.Student) + .FirstOrDefaultAsync(x => x.Id == courseId); + + if (!course.StudentCourses.Any()) + { + result.HasError = true; + result.AddMessage(string.Format(ErrorMessages.CourseHasNoStudents, courseId)); + return result; + } + + // TODO: obtain the grade from StudentCourses + result.Data = course.StudentCourses.Select(x => new UserGradeDto() + { + Grade = 10, + IdentityCard = x.Student.IdentityCard + }).ToList(); + + return result; + } } } diff --git a/Tsi1.Api/Tsi1.BusinessLayer/Tsi1.BusinessLayer.csproj b/Tsi1.Api/Tsi1.BusinessLayer/Tsi1.BusinessLayer.csproj index a18735e..d2c88df 100644 --- a/Tsi1.Api/Tsi1.BusinessLayer/Tsi1.BusinessLayer.csproj +++ b/Tsi1.Api/Tsi1.BusinessLayer/Tsi1.BusinessLayer.csproj @@ -11,6 +11,7 @@ <PackageReference Include="Microsoft.AspNetCore.Hosting.Abstractions" Version="2.2.0" /> <PackageReference Include="Microsoft.AspNetCore.Http.Features" Version="3.1.9" /> <PackageReference Include="MimeKit" Version="2.9.2" /> + <PackageReference Include="Newtonsoft.Json" Version="12.0.3" /> </ItemGroup> <ItemGroup> -- GitLab