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.
 
 
 
 
 

173 lines
6.2 KiB

// ==========================================================================
// Squidex Headless CMS
// ==========================================================================
// Copyright (c) Squidex UG (haftungsbeschraenkt)
// All rights reserved. Licensed under the MIT license.
// ==========================================================================
using System.ComponentModel.DataAnnotations;
using System.Text.Json.Serialization;
using Elasticsearch.Net;
using Squidex.Domain.Apps.Core.HandleRules;
using Squidex.Domain.Apps.Core.Rules.Deprecated;
using Squidex.Flows;
using Squidex.Infrastructure.Json;
using Squidex.Infrastructure.Reflection;
using Squidex.Infrastructure.Validation;
namespace Squidex.Extensions.Actions.ElasticSearch;
[FlowStep(
Title = "ElasticSearch",
IconImage = "<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 29 28'><path d='M13.427 17.436H4.163C3.827 16.354 3.636 15.2 3.636 14s.182-2.355.527-3.436h15.245c1.891 0 3.418 1.545 3.418 3.445a3.421 3.421 0 0 1-3.418 3.427h-5.982zm-.436 1.146H4.6a11.508 11.508 0 0 0 4.2 4.982 11.443 11.443 0 0 0 15.827-3.209 5.793 5.793 0 0 0-4.173-1.773H12.99zm7.464-9.164a5.794 5.794 0 0 0 4.173-1.773 11.45 11.45 0 0 0-9.536-5.1c-2.327 0-4.491.7-6.3 1.891a11.554 11.554 0 0 0-4.2 4.982h15.864z'/></svg>",
IconColor = "#1e5470",
Display = "Populate ElasticSearch index",
Description = "Populate a full text search index in ElasticSearch.",
ReadMore = "https://www.elastic.co/")]
#pragma warning disable CS0618 // Type or member is obsolete
public sealed record ElasticSearchFlowStep : FlowStep, IConvertibleToAction
#pragma warning restore CS0618 // Type or member is obsolete
{
[AbsoluteUrl]
[LocalizedRequired]
[Display(Name = "Server Url", Description = "The url to the instance or cluster.")]
[Editor(FlowStepEditor.Url)]
public Uri Host { get; set; }
[LocalizedRequired]
[Display(Name = "Index Name", Description = "The name of the index.")]
[Editor(FlowStepEditor.Text)]
[Expression]
public string IndexName { get; set; }
[Display(Name = "Username", Description = "The optional username.")]
[Editor(FlowStepEditor.Text)]
public string? Username { get; set; }
[Display(Name = "Password", Description = "The optional password.")]
[Editor(FlowStepEditor.Text)]
public string? Password { get; set; }
[Display(Name = "Document", Description = "The optional custom document.")]
[Editor(FlowStepEditor.TextArea)]
[Expression(ExpressionFallback.Event)]
public string? Document { get; set; }
[Display(Name = "Deletion", Description = "The condition when to delete the document.")]
[Editor(FlowStepEditor.Text)]
public string? Delete { get; set; }
private static readonly ClientPool<(Uri Host, string? Username, string? Password), ElasticLowLevelClient> Clients = new (key =>
{
var config = new ConnectionConfiguration(key.Host);
if (!string.IsNullOrEmpty(key.Username) && !string.IsNullOrWhiteSpace(key.Password))
{
config = config.BasicAuthentication(key.Username, key.Password);
}
return new ElasticLowLevelClient(config);
});
public override ValueTask PrepareAsync(FlowExecutionContext executionContext,
CancellationToken ct)
{
var @event = ((FlowEventContext)executionContext.Context).Event;
if (@event.ShouldDelete(executionContext, Delete))
{
Document = null;
return default;
}
ElasticSearchContent content;
try
{
content = executionContext.DeserializeJson<ElasticSearchContent>(Document!);
content.ContentId = @event.GetOrCreateId().Id;
}
catch (Exception ex)
{
content = new ElasticSearchContent
{
More = new Dictionary<string, object>
{
["error"] = $"Invalid JSON: {ex.Message}",
},
ContentId = @event.GetOrCreateId().Id,
};
}
Document = executionContext.SerializeJson(content);
return default;
}
public override async ValueTask<FlowStepResult> ExecuteAsync(FlowExecutionContext executionContext,
CancellationToken ct)
{
var @event = ((FlowEventContext)executionContext.Context).Event;
var (id, isGenerated) = @event.GetOrCreateId();
if (isGenerated && Document == null)
{
executionContext.LogSkipped("Can only delete content for static identities.");
return Next();
}
if (executionContext.IsSimulation)
{
executionContext.LogSkipSimulation();
return Next();
}
try
{
void HandleResult(StringResponse response, string message)
{
if (response.OriginalException != null)
{
executionContext.Log("Failed with error", response.OriginalException.Message);
throw response.OriginalException;
}
executionContext.Log(message, response.Body);
}
var client = await Clients.GetClientAsync((Host, Username, Password));
if (Document != null)
{
var response = await client.IndexAsync<StringResponse>(IndexName, id, Document, ctx: ct);
HandleResult(response, $"Document with ID '{id}' upserted");
}
else
{
var response = await client.DeleteAsync<StringResponse>(IndexName, id, ctx: ct);
HandleResult(response, $"Document with ID '{id}' deleted");
}
return Next();
}
catch (ElasticsearchClientException ex)
{
executionContext.Log("Failed with error", ex.Message);
throw;
}
}
#pragma warning disable CS0618 // Type or member is obsolete
public RuleAction ToAction()
{
return SimpleMapper.Map(this, new ElasticSearchAction());
}
#pragma warning restore CS0618 // Type or member is obsolete
private sealed class ElasticSearchContent
{
public string ContentId { get; set; }
[JsonExtensionData]
public Dictionary<string, object> More { get; set; } = [];
}
}