Skip to content

Commit dc4d9f8

Browse files
committed
fix(Dashboard): Function create vs update
- An `IsNew` variable was added to the state, which tracks whether the function is new or existing. This change prevents unnecessary API calls, reducing 404 errors and improving efficiency. Signed-off-by: Jean-Baptiste Bianchi <[email protected]>
1 parent 0584a8a commit dc4d9f8

File tree

4 files changed

+78
-38
lines changed

4 files changed

+78
-38
lines changed

src/dashboard/Synapse.Dashboard/Pages/Functions/Create/State.cs

+7-2
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@
1818
namespace Synapse.Dashboard.Pages.Functions.Create;
1919

2020
/// <summary>
21-
/// The <see cref="State{TState}"/> of the workflow editor
21+
/// The <see cref="State{TState}"/> of the function editor
2222
/// </summary>
2323
[Feature]
2424
public record CreateFunctionViewState
@@ -40,7 +40,12 @@ public record CreateFunctionViewState
4040
public SemVersion Version { get; set; } = new SemVersion(1, 0, 0);
4141

4242
/// <summary>
43-
/// Gets/sets the definition of the workflow to create
43+
/// Gets/sets a boolean determining if the function is new or if it's an update
44+
/// </summary>
45+
public bool IsNew { get; set; } = true;
46+
47+
/// <summary>
48+
/// Gets/sets the definition of the function to create
4449
/// </summary>
4550
public TaskDefinition? Function { get; set; } = null;
4651

src/dashboard/Synapse.Dashboard/Pages/Functions/Create/Store.cs

+52-30
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
using Semver;
1717
using ServerlessWorkflow.Sdk.Models;
1818
using Synapse.Api.Client.Services;
19+
using Synapse.Dashboard.Pages.Workflows.Create;
1920
using Synapse.Resources;
2021

2122
namespace Synapse.Dashboard.Pages.Functions.Create;
@@ -189,7 +190,6 @@ MonacoInterop monacoInterop
189190
#endregion
190191

191192
#region Setters
192-
193193
/// <summary>
194194
/// Sets the state's <see cref="CreateFunctionViewState.Name"/>
195195
/// </summary>
@@ -215,6 +215,18 @@ public void SetChosenName(string? name)
215215
});
216216
}
217217

218+
/// <summary>
219+
/// Sets the state's <see cref="CreateFunctionViewState.IsNew"/>
220+
/// </summary>
221+
/// <param name="isNew">The new <see cref="CreateFunctionViewState.IsNew"/> value</param>
222+
public void SetIsNew(bool isNew)
223+
{
224+
this.Reduce(state => state with
225+
{
226+
IsNew = isNew
227+
});
228+
}
229+
218230
/// <summary>
219231
/// Sets the state's <see cref="CreateFunctionViewState" /> <see cref="ProblemDetails"/>'s related data
220232
/// </summary>
@@ -393,6 +405,7 @@ public async Task SaveCustomFunctionAsync()
393405
this.YamlSerializer.Deserialize<TaskDefinition>(functionText)!;
394406
var name = this.Get(state => state.Name) ?? this.Get(state => state.ChosenName);
395407
var version = this.Get(state => state.Version).ToString();
408+
var isNew = this.Get(state => state.IsNew);
396409
if (string.IsNullOrEmpty(name))
397410
{
398411
this.Reduce(state => state with
@@ -407,41 +420,41 @@ public async Task SaveCustomFunctionAsync()
407420
Saving = true
408421
});
409422
CustomFunction? resource = null;
410-
try
411-
{
412-
resource = await this.ApiClient.CustomFunctions.GetAsync(name);
413-
}
414-
catch
423+
if (isNew)
415424
{
416-
// Assume 404, might need actual handling
417-
}
418-
if (resource == null)
419-
{
420-
resource = await this.ApiClient.CustomFunctions.CreateAsync(new()
425+
426+
try
421427
{
422-
Metadata = new()
428+
resource = await this.ApiClient.CustomFunctions.CreateAsync(new()
423429
{
424-
Name = name
425-
},
426-
Spec = new()
427-
{
428-
Versions = [new(version, function)]
429-
}
430-
});
431-
}
432-
else
433-
{
434-
var updatedResource = resource.Clone()!;
435-
updatedResource.Spec.Versions.Add(new(version, function));
436-
var jsonPatch = JsonPatch.FromDiff(this.JsonSerializer.SerializeToElement(resource)!.Value, this.JsonSerializer.SerializeToElement(updatedResource)!.Value);
437-
var patch = this.JsonSerializer.Deserialize<Json.Patch.JsonPatch>(jsonPatch.RootElement);
438-
if (patch != null)
430+
Metadata = new()
431+
{
432+
Name = name
433+
},
434+
Spec = new()
435+
{
436+
Versions = [new(version, function)]
437+
}
438+
});
439+
this.NavigationManager.NavigateTo($"/functions/{name}");
440+
return;
441+
}
442+
catch (ProblemDetailsException ex) when (ex.Problem.Title == "Conflict" && ex.Problem.Detail != null && ex.Problem.Detail.EndsWith("already exists"))
439443
{
440-
var resourcePatch = new Patch(PatchType.JsonPatch, jsonPatch);
441-
await this.ApiClient.ManageCluster<CustomFunction>().PatchAsync(name, resourcePatch, null, this.CancellationTokenSource.Token);
444+
// the function exists, try to update it instead
442445
}
443446
}
444-
this.NavigationManager.NavigateTo($"/functions/{name}");
447+
resource = await this.ApiClient.CustomFunctions.GetAsync(name);
448+
var updatedResource = resource.Clone()!;
449+
updatedResource.Spec.Versions.Add(new(version, function));
450+
var jsonPatch = JsonPatch.FromDiff(this.JsonSerializer.SerializeToElement(resource)!.Value, this.JsonSerializer.SerializeToElement(updatedResource)!.Value);
451+
var patch = this.JsonSerializer.Deserialize<Json.Patch.JsonPatch>(jsonPatch.RootElement);
452+
if (patch != null)
453+
{
454+
var resourcePatch = new Patch(PatchType.JsonPatch, jsonPatch);
455+
await this.ApiClient.ManageCluster<CustomFunction>().PatchAsync(name, resourcePatch, null, this.CancellationTokenSource.Token);
456+
}
457+
445458
}
446459
catch (ProblemDetailsException ex)
447460
{
@@ -477,6 +490,15 @@ public async Task SaveCustomFunctionAsync()
477490
catch (Exception ex)
478491
{
479492
this.Logger.LogError("Unable to save function definition: {exception}", ex.ToString());
493+
this.Reduce(state => state with
494+
{
495+
ProblemTitle = "Error",
496+
ProblemDetail = "An error occurred while saving the function.",
497+
ProblemErrors = new Dictionary<string, string[]>()
498+
{
499+
{"Message", [ex.ToString()] }
500+
}
501+
});
480502
}
481503
finally
482504
{

src/dashboard/Synapse.Dashboard/Pages/Functions/Create/View.razor

+10-5
Original file line numberDiff line numberDiff line change
@@ -47,10 +47,10 @@ else
4747
<PreferredLanguageSelector PreferredLanguageChange="Store.ToggleTextBasedEditorLanguageAsync" />
4848
</div>
4949
<StandaloneCodeEditor @ref="Store.TextEditor"
50-
ConstructionOptions="Store.StandaloneEditorConstructionOptions"
51-
OnDidInit="Store.OnTextBasedEditorInitAsync"
52-
OnDidChangeModelContent="Store.OnDidChangeModelContent"
53-
CssClass="h-100" />
50+
ConstructionOptions="Store.StandaloneEditorConstructionOptions"
51+
OnDidInit="Store.OnTextBasedEditorInitAsync"
52+
OnDidChangeModelContent="Store.OnDidChangeModelContent"
53+
CssClass="h-100" />
5454
@if (problemDetails != null)
5555
{
5656
<div class="problems px-3">
@@ -104,6 +104,7 @@ else
104104
await base.OnInitializedAsync();
105105
BreadcrumbManager.Use(Breadcrumbs.Functions);
106106
BreadcrumbManager.Add(new("New", "/functions/new"));
107+
Store.SetIsNew(true);
107108
Store.Name.Subscribe(value => OnStateChanged(_ => name = value), token: CancellationTokenSource.Token);
108109
Store.ChosenName.Subscribe(value => OnStateChanged(_ => chosenName = value), token: CancellationTokenSource.Token);
109110
Store.Version.Subscribe(value => OnStateChanged(_ => version = value?.ToString()), token: CancellationTokenSource.Token);
@@ -115,7 +116,11 @@ else
115116
/// <inheritdoc/>
116117
protected override void OnParametersSet()
117118
{
118-
if (Name != name) Store.SetName(Name);
119+
if (Name != name)
120+
{
121+
Store.SetName(Name);
122+
Store.SetIsNew(false);
123+
}
119124
}
120125

121126
protected void SetName()

src/dashboard/Synapse.Dashboard/Pages/Workflows/Create/Store.cs

+9-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,6 @@
1919
using Synapse.Api.Client.Services;
2020
using Synapse.Resources;
2121
using System.Text.RegularExpressions;
22-
using System.Xml.Linq;
2322

2423
namespace Synapse.Dashboard.Pages.Workflows.Create;
2524

@@ -513,6 +512,15 @@ public async Task SaveWorkflowDefinitionAsync()
513512
catch (Exception ex)
514513
{
515514
this.Logger.LogError("Unable to save workflow definition: {exception}", ex.ToString());
515+
this.Reduce(state => state with
516+
{
517+
ProblemTitle = "Error",
518+
ProblemDetail = "An error occurred while saving the workflow.",
519+
ProblemErrors = new Dictionary<string, string[]>()
520+
{
521+
{"Message", [ex.ToString()] }
522+
}
523+
});
516524
}
517525
finally
518526
{

0 commit comments

Comments
 (0)