Files
UnrealEngineUWP/Engine/Source/Programs/Horde/HordeServer/Controllers/ProjectsController.cs
Ben Marsh 01b9fec4f3 Horde: Restore endpoint to allow retrieving project logos.
[CL 16358816 by Ben Marsh in ue5-main branch]
2021-05-17 16:43:09 -04:00

281 lines
9.3 KiB
C#

// Copyright Epic Games, Inc. All Rights Reserved.
using HordeServer.Api;
using HordeServer.Models;
using HordeServer.Services;
using HordeServer.Utilities;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using MongoDB.Bson;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Threading.Tasks;
using ProjectId = HordeServer.Utilities.StringId<HordeServer.Models.IProject>;
namespace HordeServer.Controllers
{
/// <summary>
/// Controller for the /api/v1/projects endpoint
/// </summary>
[ApiController]
[Authorize]
[Route("[controller]")]
public class ProjectsController : ControllerBase
{
/// <summary>
/// Singleton instance of the ACL service
/// </summary>
private readonly AclService AclService;
/// <summary>
/// Singleton instance of the project service
/// </summary>
private readonly ProjectService ProjectService;
/// <summary>
/// Singleton instance of the stream service
/// </summary>
private readonly StreamService StreamService;
/// <summary>
/// Singleton instance of the job service
/// </summary>
private readonly JobService JobService;
/// <summary>
/// Singleton instance of the schedule service
/// </summary>
private readonly ScheduleService ScheduleService;
/// <summary>
/// Constructor
/// </summary>
/// <param name="AclService">The ACL service</param>
/// <param name="ProjectService">The project service</param>
/// <param name="StreamService">The stream service</param>
/// <param name="JobService">The job service</param>
/// <param name="ScheduleService">The schedule service</param>
public ProjectsController(AclService AclService, ProjectService ProjectService, StreamService StreamService, JobService JobService, ScheduleService ScheduleService)
{
this.AclService = AclService;
this.ProjectService = ProjectService;
this.StreamService = StreamService;
this.JobService = JobService;
this.ScheduleService = ScheduleService;
}
/// <summary>
/// Creates a new project
/// </summary>
/// <param name="Create">Parameters for the new project.</param>
/// <returns>Http result code</returns>
[HttpPost]
[Route("/api/v1/projects")]
public async Task<ActionResult<CreateProjectResponse>> CreateProjectAsync([FromBody] CreateProjectRequest Create)
{
if(!await AclService.AuthorizeAsync(AclAction.CreateProject, User, null))
{
return Forbid();
}
IProject? NewProject = await ProjectService.TryCreateProjectAsync(ProjectId.Sanitize(Create.Name), Create.Name, Create.Order, Create.Categories?.ConvertAll(x => new StreamCategory(x)), Create.Properties);
if (NewProject == null)
{
return BadRequest("Project already exists");
}
else
{
return new CreateProjectResponse(NewProject.Id.ToString());
}
}
/// <summary>
/// Query all the projects
/// </summary>
/// <param name="IncludeStreams">Whether to include streams in the response</param>
/// <param name="IncludeCategories">Whether to include categories in the response</param>
/// <param name="Filter">Filter for the properties to return</param>
/// <returns>Information about all the projects</returns>
[HttpGet]
[Route("/api/v1/projects")]
[ProducesResponseType(typeof(List<GetProjectResponse>), 200)]
public async Task<ActionResult<List<object>>> GetProjectsAsync([FromQuery(Name = "Streams")] bool IncludeStreams = false, [FromQuery(Name = "Categories")] bool IncludeCategories = false, [FromQuery] PropertyFilter? Filter = null)
{
List<IProject> Projects = await ProjectService.GetProjectsAsync();
ProjectPermissionsCache PermissionsCache = new ProjectPermissionsCache();
List<IStream>? Streams = null;
if (IncludeStreams || IncludeCategories)
{
Streams = await StreamService.GetStreamsAsync();
}
List<object> Responses = new List<object>();
foreach (IProject Project in Projects)
{
if (await ProjectService.AuthorizeAsync(Project, AclAction.ViewProject, User, PermissionsCache))
{
bool bIncludeAcl = await ProjectService.AuthorizeAsync(Project, AclAction.ViewPermissions, User, PermissionsCache);
Responses.Add(Project.ToResponse(IncludeStreams, IncludeCategories, Streams, bIncludeAcl).ApplyFilter(Filter));
}
}
return Responses;
}
/// <summary>
/// Retrieve information about a specific project
/// </summary>
/// <param name="ProjectId">Id of the project to get information about</param>
/// <param name="Filter">Filter for the properties to return</param>
/// <returns>Information about the requested project</returns>
[HttpGet]
[Route("/api/v1/projects/{ProjectId}")]
[ProducesResponseType(typeof(List<GetProjectResponse>), 200)]
public async Task<ActionResult<object>> GetProjectAsync(string ProjectId, [FromQuery] PropertyFilter? Filter = null)
{
ProjectId ProjectIdValue = new ProjectId(ProjectId);
IProject? Project = await ProjectService.GetProjectAsync(ProjectIdValue);
if (Project == null)
{
return NotFound();
}
ProjectPermissionsCache Cache = new ProjectPermissionsCache();
if (!await ProjectService.AuthorizeAsync(Project, AclAction.ViewProject, User, Cache))
{
return Forbid();
}
bool bIncludeStreams = PropertyFilter.Includes(Filter, nameof(GetProjectResponse.Streams));
bool bIncludeCategories = PropertyFilter.Includes(Filter, nameof(GetProjectResponse.Categories));
List<IStream>? VisibleStreams = null;
if (bIncludeStreams || bIncludeCategories)
{
VisibleStreams = new List<IStream>();
List<IStream> Streams = await StreamService.GetStreamsAsync(Project.Id);
foreach (IStream Stream in Streams)
{
if (await StreamService.AuthorizeAsync(Stream, AclAction.ViewStream, User, Cache))
{
VisibleStreams.Add(Stream);
}
}
}
bool bIncludeAcl = await ProjectService.AuthorizeAsync(Project, AclAction.ViewPermissions, User, Cache);
return Project.ToResponse(bIncludeStreams, bIncludeCategories, VisibleStreams, bIncludeAcl).ApplyFilter(Filter);
}
/// <summary>
/// Retrieve information about a specific project
/// </summary>
/// <param name="ProjectId">Id of the project to get information about</param>
/// <returns>Information about the requested project</returns>
[HttpGet]
[Route("/api/v1/projects/{ProjectId}/logo")]
public async Task<ActionResult<object>> GetProjectLogoAsync(string ProjectId)
{
ProjectId ProjectIdValue = new ProjectId(ProjectId);
IProject? Project = await ProjectService.GetProjectAsync(ProjectIdValue);
if (Project == null)
{
return NotFound();
}
if (!await ProjectService.AuthorizeAsync(Project, AclAction.ViewProject, User, null))
{
return Forbid();
}
IProjectLogo? ProjectLogo = await ProjectService.Collection.GetLogoAsync(ProjectIdValue);
if (ProjectLogo == null)
{
return NotFound();
}
return new FileContentResult(ProjectLogo.Data, ProjectLogo.MimeType);
}
/// <summary>
/// Update a project's properties.
/// </summary>
/// <param name="ProjectId">Id of the project to update</param>
/// <param name="Update">Items on the project to update</param>
/// <returns>Http result code</returns>
[HttpPut]
[Route("/api/v1/projects/{ProjectId}")]
public async Task<ActionResult> UpdateProjectAsync(string ProjectId, [FromBody] UpdateProjectRequest Update)
{
ProjectId ProjectIdValue = new ProjectId(ProjectId);
for (; ; )
{
IProject? Project = await ProjectService.GetProjectAsync(ProjectIdValue);
if (Project == null)
{
if (!await AclService.AuthorizeAsync(AclAction.CreateProject, User, null))
{
return Forbid();
}
if (Update.Name == null)
{
return BadRequest("Project does not exist; 'Name' must be specified on update request to create it");
}
IProject? NewProject = await ProjectService.TryCreateProjectAsync(ProjectIdValue, Update.Name, Update.Order, Update.Categories?.ConvertAll(x => new StreamCategory(x)), Update.Properties);
if (NewProject != null)
{
return Ok();
}
}
else
{
ProjectPermissionsCache PermissionsCache = new ProjectPermissionsCache();
if (!await ProjectService.AuthorizeAsync(Project, AclAction.UpdateProject, User, PermissionsCache))
{
return Forbid();
}
if (Update.Acl != null && !await ProjectService.AuthorizeAsync(Project, AclAction.ChangePermissions, User, PermissionsCache))
{
return Forbid();
}
List<StreamCategory>? NewCategories = Update.Categories?.ConvertAll(x => new StreamCategory(x));
await ProjectService.UpdateProjectAsync(ProjectIdValue, Update.Name, Update.Order, NewCategories, Update.Properties, Acl.Merge(Project.Acl, Update.Acl));
return new OkResult();
}
}
}
/// <summary>
/// Delete a project
/// </summary>
/// <param name="ProjectId">Id of the project to update</param>
/// <returns>Http result code</returns>
[HttpDelete]
[Route("/api/v1/projects/{ProjectId}")]
public async Task<ActionResult> DeleteProjectAsync(string ProjectId)
{
ProjectId ProjectIdValue = new ProjectId(ProjectId);
IProject? Project = await ProjectService.GetProjectAsync(ProjectIdValue);
if (Project == null)
{
return NotFound();
}
if (!await ProjectService.AuthorizeAsync(Project, AclAction.DeleteProject, User, null))
{
return Forbid();
}
await ProjectService.DeleteProjectAsync(ProjectIdValue);
return new OkResult();
}
}
}