Headless CMS and Content Managment Hub
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

406 lines
15 KiB

// ==========================================================================
// Squidex Headless CMS
// ==========================================================================
// Copyright (c) Squidex UG (haftungsbeschraenkt)
// All rights reserved. Licensed under the MIT license.
// ==========================================================================
using Microsoft.AspNetCore.Mvc;
using Microsoft.Net.Http.Headers;
using NSwag.Annotations;
using Squidex.Areas.Api.Controllers.Schemas.Models;
using Squidex.Domain.Apps.Core.GenerateFilters;
using Squidex.Domain.Apps.Core.Scripting;
using Squidex.Domain.Apps.Entities;
using Squidex.Domain.Apps.Entities.Apps;
using Squidex.Domain.Apps.Entities.Schemas;
using Squidex.Domain.Apps.Entities.Schemas.Commands;
using Squidex.Infrastructure;
using Squidex.Infrastructure.Commands;
using Squidex.Infrastructure.Queries;
using Squidex.Shared;
using Squidex.Web;
namespace Squidex.Areas.Api.Controllers.Schemas
{
/// <summary>
/// Manages and retrieves information about schemas.
/// </summary>
[ApiExplorerSettings(GroupName = nameof(Schemas))]
public sealed class SchemasController : ApiController
{
private readonly IAppProvider appProvider;
public SchemasController(ICommandBus commandBus, IAppProvider appProvider)
: base(commandBus)
{
this.appProvider = appProvider;
}
/// <summary>
/// Get schemas.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <returns>
/// 200 => Schemas returned.
/// 404 => App not found.
/// </returns>
[HttpGet]
[Route("apps/{app}/schemas/")]
[ProducesResponseType(typeof(SchemasDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasRead)]
[ApiCosts(0)]
public async Task<IActionResult> GetSchemas(string app)
{
var schemas = await appProvider.GetSchemasAsync(AppId, HttpContext.RequestAborted);
var response = Deferred.Response(() =>
{
return SchemasDto.FromDomain(schemas, Resources);
});
Response.Headers[HeaderNames.ETag] = schemas.ToEtag();
return Ok(response);
}
/// <summary>
/// Get a schema by name.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema to retrieve.</param>
/// <returns>
/// 200 => Schema found.
/// 404 => Schema or app not found.
/// </returns>
[HttpGet]
[Route("apps/{app}/schemas/{schema}/")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasRead)]
[ApiCosts(0)]
public IActionResult GetSchema(string app, string schema)
{
var response = Deferred.Response(() =>
{
return SchemaDto.FromDomain(Schema, Resources);
});
Response.Headers[HeaderNames.ETag] = Schema.ToEtag();
return Ok(response);
}
/// <summary>
/// Create a new schema.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="request">The schema object that needs to be added to the app.</param>
/// <returns>
/// 201 => Schema created.
/// 400 => Schema request not valid.
/// 409 => Schema name already in use.
/// </returns>
[HttpPost]
[Route("apps/{app}/schemas/")]
[ProducesResponseType(typeof(SchemaDto), 201)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasCreate)]
[ApiCosts(1)]
public async Task<IActionResult> PostSchema(string app, [FromBody] CreateSchemaDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return CreatedAtAction(nameof(GetSchema), new { app, schema = request.Name }, response);
}
/// <summary>
/// Update a schema.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema.</param>
/// <param name="request">The schema object that needs to updated.</param>
/// <returns>
/// 200 => Schema updated.
/// 400 => Schema request not valid.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasUpdate)]
[ApiCosts(1)]
public async Task<IActionResult> PutSchema(string app, string schema, [FromBody] UpdateSchemaDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Synchronize a schema.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema.</param>
/// <param name="request">The schema object that needs to updated.</param>
/// <returns>
/// 200 => Schema updated.
/// 400 => Schema request not valid.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/sync")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasUpdate)]
[ApiCosts(1)]
public async Task<IActionResult> PutSchemaSync(string app, string schema, [FromBody] SynchronizeSchemaDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Update a schema category.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema.</param>
/// <param name="request">The schema object that needs to updated.</param>
/// <returns>
/// 200 => Schema updated.
/// 400 => Schema request not valid.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/category")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasUpdate)]
[ApiCosts(1)]
public async Task<IActionResult> PutCategory(string app, string schema, [FromBody] ChangeCategoryDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Update the preview urls.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema.</param>
/// <param name="request">The preview urls for the schema.</param>
/// <returns>
/// 200 => Schema updated.
/// 400 => Schema request not valid.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/preview-urls")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasUpdate)]
[ApiCosts(1)]
public async Task<IActionResult> PutPreviewUrls(string app, string schema, [FromBody] ConfigurePreviewUrlsDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Update the scripts.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema.</param>
/// <param name="request">The schema scripts object that needs to updated.</param>
/// <returns>
/// 200 => Schema updated.
/// 400 => Schema request not valid.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/scripts/")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasScripts)]
[ApiCosts(1)]
public async Task<IActionResult> PutScripts(string app, string schema, [FromBody] SchemaScriptsDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Update the rules.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema.</param>
/// <param name="request">The schema rules object that needs to updated.</param>
/// <returns>
/// 200 => Schema updated.
/// 400 => Schema request not valid.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/rules/")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasUpdate)]
[ApiCosts(1)]
public async Task<IActionResult> PutRules(string app, string schema, [FromBody] ConfigureFieldRulesDto request)
{
var command = request.ToCommand();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Publish a schema.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema to publish.</param>
/// <returns>
/// 200 => Schema published.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/publish/")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasPublish)]
[ApiCosts(1)]
public async Task<IActionResult> PublishSchema(string app, string schema)
{
var command = new PublishSchema();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Unpublish a schema.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema to unpublish.</param>
/// <returns>
/// 200 => Schema unpublished.
/// 404 => Schema or app not found.
/// </returns>
[HttpPut]
[Route("apps/{app}/schemas/{schema}/unpublish/")]
[ProducesResponseType(typeof(SchemaDto), StatusCodes.Status200OK)]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasPublish)]
[ApiCosts(1)]
public async Task<IActionResult> UnpublishSchema(string app, string schema)
{
var command = new UnpublishSchema();
var response = await InvokeCommandAsync(command);
return Ok(response);
}
/// <summary>
/// Delete a schema.
/// </summary>
/// <param name="app">The name of the app.</param>
/// <param name="schema">The name of the schema to delete.</param>
/// <returns>
/// 204 => Schema deleted.
/// 404 => Schema or app not found.
/// </returns>
[HttpDelete]
[Route("apps/{app}/schemas/{schema}/")]
[ApiPermissionOrAnonymous(PermissionIds.AppSchemasDelete)]
[ApiCosts(1)]
public async Task<IActionResult> DeleteSchema(string app, string schema)
{
var command = new DeleteSchema();
await CommandBus.PublishAsync(command, HttpContext.RequestAborted);
return NoContent();
}
[HttpGet]
[Route("apps/{app}/schemas/{schema}/completion")]
[ApiPermissionOrAnonymous]
[ApiCosts(1)]
[OpenApiIgnore]
public async Task<IActionResult> GetScriptCompletion(string app, string schema,
[FromServices] ScriptingCompleter completer)
{
var completion = completer.ContentScript(await BuildModel());
return Ok(completion);
}
[HttpGet]
[Route("apps/{app}/schemas/{schema}/completion/triggers")]
[ApiPermissionOrAnonymous]
[ApiCosts(1)]
[OpenApiIgnore]
public async Task<IActionResult> GetScriptTriggerCompletion(string app, string schema,
[FromServices] ScriptingCompleter completer)
{
var completion = completer.ContentTrigger(await BuildModel());
return Ok(completion);
}
[HttpGet]
[Route("apps/{app}/schemas/{schema}/filters")]
[ApiPermissionOrAnonymous]
[ApiCosts(1)]
[OpenApiIgnore]
public async Task<IActionResult> GetFilters(string app, string schema)
{
var components = await appProvider.GetComponentsAsync(Schema, HttpContext.RequestAborted);
var filters = ContentQueryModel.Build(Schema.SchemaDef, App.PartitionResolver(), components).Flatten();
return Ok(filters);
}
private async Task<FilterSchema> BuildModel()
{
var components = await appProvider.GetComponentsAsync(Schema, HttpContext.RequestAborted);
return Schema.SchemaDef.BuildDataSchema(App.PartitionResolver(), components);
}
private Task<ISchemaEntity?> GetSchemaAsync(string schema)
{
if (Guid.TryParse(schema, out var guid))
{
var schemaId = DomainId.Create(guid);
return appProvider.GetSchemaAsync(AppId, schemaId, ct: HttpContext.RequestAborted);
}
else
{
return appProvider.GetSchemaAsync(AppId, schema, ct: HttpContext.RequestAborted);
}
}
private async Task<SchemaDto> InvokeCommandAsync(ICommand command)
{
var context = await CommandBus.PublishAsync(command, HttpContext.RequestAborted);
var result = context.Result<ISchemaEntity>();
var response = SchemaDto.FromDomain(result, Resources);
return response;
}
}
}