using System; using System.Collections.Generic; using System.Drawing; using System.Drawing.Imaging; using System.IO; using System.Linq; using System.Net.Http.Headers; using System.Text; using System.Text.RegularExpressions; using System.Threading.Tasks; using Manager.Interfaces.DTO; using Manager.Interfaces.Models; using Manager.Services; using ManagerService.Helpers; using ManagerService.Service.Services; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Server.IIS.Core; using Microsoft.Extensions.Logging; using Newtonsoft.Json; using NSwag.Annotations; using static System.Net.Mime.MediaTypeNames; namespace ManagerService.Controllers { [Authorize] // TODO Add ROLES (Roles = "Admin") [ApiController, Route("api/[controller]")] [OpenApiTag("Resource", Description = "Resource management")] public class ResourceController : ControllerBase { private ResourceDatabaseService _resourceService; private SectionDatabaseService _sectionService; private ConfigurationDatabaseService _configurationService; private readonly ILogger _logger; private static int MaxWidth = 1024; private static int MaxHeight = 1024; public ResourceController(ILogger logger, ResourceDatabaseService resourceService, SectionDatabaseService sectionService, ConfigurationDatabaseService configurationService) { _logger = logger; _resourceService = resourceService; _sectionService = sectionService; _configurationService = configurationService; } /// /// Get a list of all resources (summary) /// /// id instance /// types of resource [ProducesResponseType(typeof(List), 200)] [ProducesResponseType(typeof(string), 500)] [HttpGet] public ObjectResult Get([FromQuery] string instanceId, [FromQuery] List types) { try { if (instanceId == null) throw new ArgumentNullException("InstanceId needed"); List resources = new List(); if (types.Count > 0) { resources = _resourceService.GetAllByType(instanceId, types); } else { resources = _resourceService.GetAll(instanceId); } List resourceDTOs = new List(); foreach(var resource in resources) { ResourceDTO resourceDTO = new ResourceDTO(); resourceDTO = resource.ToDTO(); resourceDTOs.Add(resourceDTO); } return new OkObjectResult(resourceDTOs.OrderByDescending(r => r.dateCreation)); } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } /// /// Get a specific resource /// /// id resource [AllowAnonymous] [ProducesResponseType(typeof(ResourceDTO), 200)] [ProducesResponseType(typeof(string), 404)] [ProducesResponseType(typeof(string), 500)] [HttpGet("{id}/detail")] public ObjectResult GetDetail(string id) { try { Resource resource = _resourceService.GetById(id); if (resource == null) throw new KeyNotFoundException("This resource was not found"); ResourceDTO resourceDTO = new ResourceDTO(); resourceDTO = resource.ToDTO(); /*if (resource.Type == ResourceType.ImageUrl) { var resourceData = _resourceDataService.GetByResourceId(resource.Id); resourceDTO.data = resourceData != null ? resourceData.Data : null; }*/ // RESIZE IMAGE /*byte[] imageBytes = Convert.FromBase64String(resourceData.Data); using (MemoryStream originalImageMemoryStream = new MemoryStream(imageBytes)) { using (Image image = Image.FromStream(originalImageMemoryStream)) { var width = image.Width; var height = image.Height; if (image.Width > MaxWidth || image.Height > MaxHeight) { Size newSize = ImageResizer.ResizeKeepAspect(image.Size, MaxWidth, MaxHeight); byte[] resizedImage = ImageResizer.ResizeImage(image, newSize.Width, newSize.Height, image.Width, image.Height); resourceData.Data = Convert.ToBase64String(resizedImage); ResourceData resourceModified = _resourceDataService.Update(resourceData.Id, resourceData); } } }*/ return new OkObjectResult(resourceDTO); } catch (KeyNotFoundException ex) { return new NotFoundObjectResult(ex.Message) {}; } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } /// /// Show a specific resource (as a picture or video stream) /// /// id resource [AllowAnonymous] [ProducesResponseType(typeof(FileResult), 200)] [ProducesResponseType(typeof(string), 404)] [ProducesResponseType(typeof(string), 500)] [HttpGet("{id}")] public ActionResult Show(string id) { try { Resource resource = _resourceService.GetById(id); if (resource == null) throw new KeyNotFoundException("This resource was not found"); //var file = Convert.FromBase64String(resourceData.Data); // RESIZE IMAGE /*using (MemoryStream originalImageMemoryStream = new MemoryStream(file)) { using (Image image = Image.FromStream(originalImageMemoryStream)) { var width = image.Width; var height = image.Height; if(image.Width > MaxWidth || image.Height > MaxHeight) { Size newSize = ImageResizer.ResizeKeepAspect(image.Size, MaxWidth, MaxHeight); byte[] resizedImage = ImageResizer.ResizeImage(image, newSize.Width, newSize.Height, image.Width, image.Height); resourceData.Data = Convert.ToBase64String(resizedImage); ResourceData resourceModified = _resourceDataService.Update(resourceData.Id, resourceData); } } }*/ /*if (resource.Type == ResourceType.Image) { return new FileContentResult(file, "image/png") { FileDownloadName = resource.Label + ".png" }; } if (resource.Type == ResourceType.Video || resource.Type == ResourceType.Audio) { return new FileContentResult(file, "application/octet-stream") { FileDownloadName = resource.Type == ResourceType.Audio ? resource.Label + ".mp3" : resource.Label + ".mp4", }; } return new FileContentResult(file, "image/png");*/ return new NotFoundObjectResult("No more supported") { }; } catch (KeyNotFoundException ex) { return new NotFoundObjectResult(ex.Message) { }; } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } // OLD VERSION /// /// Upload a specific resource (picture or video) /// [ProducesResponseType(typeof(string), 200)] [ProducesResponseType(typeof(string), 404)] [ProducesResponseType(typeof(string), 500)] [HttpPost("upload"), DisableRequestSizeLimit] public IActionResult Upload([FromForm] string label, [FromForm] string type, [FromForm] string instanceId) // Create but with local //[FromBody] ResourceDetailDTO uploadResource { try { if (label == null || type == null || instanceId == null) throw new ArgumentNullException("One of resource params is null"); var resourceType = (ResourceType)Enum.Parse(typeof(ResourceType), type); List resources = new List(); foreach (var file in Request.Form.Files) { if (file.Length > 0) { var stringResult = ""; double fileSizeibMbs = (double) ((double)file.Length) / (1024*1024); if (fileSizeibMbs <= 1.5 || resourceType == ResourceType.Image) { using (var ms = new MemoryStream()) { file.CopyTo(ms); var fileBytes = ms.ToArray(); if (resourceType == ResourceType.Image) { bool isFort = instanceId == "633ee379d9405f32f166f047"; // If fort saint heribert, TODO add watermark in configuration and model if(isFort) // TODO We need to know for which purpose (mobile or tablet) { fileBytes = ImageHelper.ResizeAndAddWatermark(fileBytes, isFort, MaxWidth, MaxHeight); } } stringResult = Convert.ToBase64String(fileBytes); } } else { throw new FileLoadException(message: "Fichier inexistant ou trop volumineux (max 4Mb)"); } // Todo add some verification ? Resource resource = new Resource(); resource.Label = label; resource.Type = resourceType; resource.DateCreation = DateTime.Now; resource.InstanceId = instanceId; Resource resourceCreated = _resourceService.Create(resource); resources.Add(resourceCreated); } } return Ok(resources.Select(r => r.ToDTO())); } catch (ArgumentNullException ex) { return new BadRequestObjectResult(ex.Message) { }; } catch (FileLoadException ex) { return new BadRequestObjectResult(ex.Message) { }; } catch (InvalidOperationException ex) { return new ConflictObjectResult(ex.Message) { }; } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } /// /// Create a new resource /// /// New resource info [ProducesResponseType(typeof(ResourceDTO), 200)] [ProducesResponseType(typeof(string), 400)] [ProducesResponseType(typeof(string), 409)] [ProducesResponseType(typeof(string), 500)] [HttpPost] public ObjectResult Create([FromBody] ResourceDTO newResource) { try { if (newResource == null) throw new ArgumentNullException("Resource param is null"); // Todo add some verification ? Resource resource = new Resource(); resource.InstanceId = newResource.instanceId; resource.Label = newResource.label; resource.Type = newResource.type; resource.Url = newResource.url; resource.DateCreation = DateTime.Now; //resource.Data = newResource.data; resource.InstanceId = newResource.instanceId; Resource resourceCreated = _resourceService.Create(resource); return new OkObjectResult(resourceCreated.ToDTO()); // WITHOUT DATA } catch (ArgumentNullException ex) { return new BadRequestObjectResult(ex.Message) {}; } catch (InvalidOperationException ex) { return new ConflictObjectResult(ex.Message) {}; } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } /// /// Update a resource /// /// Resource to update [ProducesResponseType(typeof(ResourceDTO), 200)] [ProducesResponseType(typeof(string), 400)] [ProducesResponseType(typeof(string), 404)] [ProducesResponseType(typeof(string), 500)] [HttpPut] public ObjectResult Update([FromBody] ResourceDTO updatedResource) { try { if (updatedResource == null) throw new ArgumentNullException("Resource param is null"); Resource resource = _resourceService.GetById(updatedResource.id); if (resource == null) throw new KeyNotFoundException("Resource does not exist"); // Todo add some verification ? resource.InstanceId = updatedResource.instanceId; resource.Label = updatedResource.label; resource.Type = updatedResource.type; resource.Url = updatedResource.url; //resource.Data = updatedResource.data; // NOT ALLOWED Resource resourceModified = _resourceService.Update(updatedResource.id, resource); return new OkObjectResult(resourceModified.ToDTO()); } catch (ArgumentNullException ex) { return new BadRequestObjectResult(ex.Message) {}; } catch (KeyNotFoundException ex) { return new NotFoundObjectResult(ex.Message) {}; } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } /// /// Delete a resource /// /// Id of resource to delete [ProducesResponseType(typeof(string), 202)] [ProducesResponseType(typeof(string), 400)] [ProducesResponseType(typeof(string), 404)] [ProducesResponseType(typeof(string), 500)] [HttpDelete("{id}")] public ObjectResult Delete(string id) { try { if (id == null) throw new ArgumentNullException("Resource param is null"); var ressource = _resourceService.GetById(id); if (ressource == null) throw new KeyNotFoundException("Resource does not exist"); foreach (var configuration in _configurationService.GetAll(ressource.InstanceId)) { if (configuration.ImageId == id) { configuration.ImageId = null; configuration.ImageSource = null; } } // Delete all resource occurence foreach (var section in _sectionService.GetAll(ressource.InstanceId)) { if (section.ImageId == id) { section.ImageId = null; section.ImageSource = null; } switch (section.Type) { case SectionType.Map: MapDTO mapDTO = JsonConvert.DeserializeObject(section.Data); mapDTO.iconResourceId = mapDTO.iconResourceId == id ? null : mapDTO.iconResourceId; foreach (var point in mapDTO.points) { point.imageResourceId = point.imageResourceId == id ? null : point.imageResourceId; foreach (var content in point.contents) { content.resourceUrl = content.resourceId == id ? null : content.resourceUrl; content.resourceId = content.resourceId == id ? null : content.resourceId; } } foreach (var categorie in mapDTO.categories) { categorie.iconUrl = categorie.iconResourceId == id ? null : categorie.iconUrl; categorie.iconResourceId = categorie.iconResourceId == id ? null : categorie.iconResourceId; } section.Data = JsonConvert.SerializeObject(mapDTO); break; case SectionType.Slider: SliderDTO sliderDTO = JsonConvert.DeserializeObject(section.Data); List contentsToKeep = new List(); foreach (var content in sliderDTO.contents) { if (content.resourceId != id) contentsToKeep.Add(content); } sliderDTO.contents = contentsToKeep; section.Data = JsonConvert.SerializeObject(sliderDTO); break; case SectionType.Quizz: QuizzDTO quizzDTO = JsonConvert.DeserializeObject(section.Data); foreach (var question in quizzDTO.questions) { if (question.label != null) { foreach (var questionLabel in question.label) { questionLabel.resourceUrl = questionLabel.resourceId == id ? null : questionLabel.resourceUrl; questionLabel.resourceId = questionLabel.resourceId == id ? null : questionLabel.resourceId; } } question.imageBackgroundResourceUrl = question.imageBackgroundResourceId == id ? null : question.imageBackgroundResourceUrl; question.imageBackgroundResourceId = question.imageBackgroundResourceId == id ? null : question.imageBackgroundResourceId; foreach (var response in question.responses) { if (response.label != null) { foreach (var responseLabel in response.label) { responseLabel.resourceUrl = responseLabel.resourceId == id ? null : responseLabel.resourceUrl; responseLabel.resourceId = responseLabel.resourceId == id ? null : responseLabel.resourceId; } } } } if (quizzDTO.bad_level != null) { if (quizzDTO.bad_level.label != null) { foreach (var badLevelLabel in quizzDTO.bad_level.label) { badLevelLabel.resourceUrl = badLevelLabel.resourceId == id ? null : badLevelLabel.resourceUrl; badLevelLabel.resourceId = badLevelLabel.resourceId == id ? null : badLevelLabel.resourceId; } } } if (quizzDTO.medium_level != null) { if (quizzDTO.medium_level.label != null) { foreach (var medium_levelLabel in quizzDTO.medium_level.label) { medium_levelLabel.resourceUrl = medium_levelLabel.resourceId == id ? null : medium_levelLabel.resourceUrl; medium_levelLabel.resourceId = medium_levelLabel.resourceId == id ? null : medium_levelLabel.resourceId; } } } if (quizzDTO.good_level != null) { if (quizzDTO.good_level.label != null) { foreach (var gooLevelLabel in quizzDTO.good_level.label) { gooLevelLabel.resourceUrl = gooLevelLabel.resourceId == id ? null : gooLevelLabel.resourceUrl; gooLevelLabel.resourceId = gooLevelLabel.resourceId == id ? null : gooLevelLabel.resourceId; } } } if (quizzDTO.great_level != null) { if (quizzDTO.great_level.label != null) { foreach (var greatLevelLabel in quizzDTO.great_level.label) { greatLevelLabel.resourceUrl = greatLevelLabel.resourceId == id ? null : greatLevelLabel.resourceUrl; greatLevelLabel.resourceId = greatLevelLabel.resourceId == id ? null : greatLevelLabel.resourceId; } } } section.Data = JsonConvert.SerializeObject(quizzDTO); break; case SectionType.Article: ArticleDTO articleDTO = JsonConvert.DeserializeObject(section.Data); List contentsArticleToKeep = new List(); foreach (var content in articleDTO.contents) { if (content.resourceId != id) contentsArticleToKeep.Add(content); } articleDTO.contents = contentsArticleToKeep; section.Data = JsonConvert.SerializeObject(articleDTO); break; } _sectionService.Update(section.Id, section); } _resourceService.Remove(id); return new ObjectResult("The resource has been deleted") { StatusCode = 202 }; } catch (ArgumentNullException ex) { return new BadRequestObjectResult(ex.Message) { }; } catch (KeyNotFoundException ex) { return new NotFoundObjectResult(ex.Message) { }; } catch (Exception ex) { return new ObjectResult(ex.Message) { StatusCode = 500 }; } } } }