mirror of
https://github.com/chylex/Minecraft-Phantom-Panel.git
synced 2026-01-14 05:50:30 +01:00
Compare commits
5 Commits
27e70d47c3
...
wip-update
| Author | SHA1 | Date | |
|---|---|---|---|
|
65699a0331
|
|||
|
ccf3ecb180
|
|||
|
366735d351
|
|||
|
ea66f9f056
|
|||
|
9a69a6b2bb
|
@@ -3,11 +3,11 @@
|
|||||||
"isRoot": true,
|
"isRoot": true,
|
||||||
"tools": {
|
"tools": {
|
||||||
"dotnet-ef": {
|
"dotnet-ef": {
|
||||||
"version": "9.0.9",
|
"version": "10.0.1",
|
||||||
"commands": [
|
"commands": [
|
||||||
"dotnet-ef"
|
"dotnet-ef"
|
||||||
],
|
],
|
||||||
"rollForward": false
|
"rollForward": false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,4 +15,30 @@ public sealed partial record Agent(
|
|||||||
) {
|
) {
|
||||||
[MemoryPackIgnore]
|
[MemoryPackIgnore]
|
||||||
public RamAllocationUnits? AvailableMemory => RuntimeInfo.MaxMemory - Stats?.RunningInstanceMemory;
|
public RamAllocationUnits? AvailableMemory => RuntimeInfo.MaxMemory - Stats?.RunningInstanceMemory;
|
||||||
|
|
||||||
|
[MemoryPackable(GenerateType.VersionTolerant)]
|
||||||
|
public sealed partial record Update(
|
||||||
|
[property: MemoryPackOrder(0)] Optional<AgentConfiguration> Configuration,
|
||||||
|
[property: MemoryPackOrder(1)] Optional<ImmutableArray<byte>> ConnectionKey,
|
||||||
|
[property: MemoryPackOrder(2)] Optional<AgentRuntimeInfo> RuntimeInfo,
|
||||||
|
[property: MemoryPackOrder(3)] OptionalNullable<AgentStats> Stats,
|
||||||
|
[property: MemoryPackOrder(4)] Optional<IAgentConnectionStatus> ConnectionStatus
|
||||||
|
) {
|
||||||
|
public Update Merge(Update newer) => new (
|
||||||
|
newer.Configuration.Or(Configuration),
|
||||||
|
newer.ConnectionKey.Or(ConnectionKey),
|
||||||
|
newer.RuntimeInfo.Or(RuntimeInfo),
|
||||||
|
newer.Stats.Or(Stats),
|
||||||
|
newer.ConnectionStatus.Or(ConnectionStatus)
|
||||||
|
);
|
||||||
|
|
||||||
|
public Agent Apply(Agent target) => new (
|
||||||
|
target.AgentGuid,
|
||||||
|
Configuration.Or(target.Configuration),
|
||||||
|
ConnectionKey.Or(target.ConnectionKey),
|
||||||
|
RuntimeInfo.Or(target.RuntimeInfo),
|
||||||
|
Stats.Or(target.Stats),
|
||||||
|
ConnectionStatus.Or(target.ConnectionStatus)
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
5
Common/Phantom.Common.Data.Web/RemoteDictionary.cs
Normal file
5
Common/Phantom.Common.Data.Web/RemoteDictionary.cs
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
namespace Phantom.Common.Data.Web;
|
||||||
|
|
||||||
|
public class RemoteDictionary<K, V> {
|
||||||
|
|
||||||
|
}
|
||||||
@@ -3,8 +3,47 @@
|
|||||||
namespace Phantom.Common.Data;
|
namespace Phantom.Common.Data;
|
||||||
|
|
||||||
[MemoryPackable]
|
[MemoryPackable]
|
||||||
public readonly partial record struct Optional<T>(T? Value) {
|
public readonly partial struct Optional<T> {
|
||||||
public static implicit operator Optional<T>(T? value) {
|
[MemoryPackOrder(0)]
|
||||||
|
public bool HasValue { get; }
|
||||||
|
|
||||||
|
[MemoryPackOrder(1)]
|
||||||
|
public T Value {
|
||||||
|
get {
|
||||||
|
if (HasValue) {
|
||||||
|
return field!;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new InvalidOperationException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[MemoryPackIgnore]
|
||||||
|
public T? ValueOrDefault => HasValue ? Value : default;
|
||||||
|
|
||||||
|
public Optional() : this(hasValue: false, value: default) {}
|
||||||
|
public Optional(T value) : this(hasValue: true, value) {}
|
||||||
|
|
||||||
|
[MemoryPackConstructor]
|
||||||
|
private Optional(bool hasValue, T? value) {
|
||||||
|
this.HasValue = hasValue;
|
||||||
|
this.Value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Optional<R> Map<R>(Func<T, R> func) {
|
||||||
|
return HasValue ? new Optional<R>(func(Value)) : new Optional<R>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public T Or(T fallbackValue) {
|
||||||
|
return HasValue ? Value : fallbackValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Optional<T> Or(Optional<T> fallbackOptional) {
|
||||||
|
return HasValue ? Value : fallbackOptional;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator Optional<T>(T value) {
|
||||||
return new Optional<T>(value);
|
return new Optional<T>(value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
42
Common/Phantom.Common.Data/OptionalNullable.cs
Normal file
42
Common/Phantom.Common.Data/OptionalNullable.cs
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
using MemoryPack;
|
||||||
|
|
||||||
|
namespace Phantom.Common.Data;
|
||||||
|
|
||||||
|
[MemoryPackable]
|
||||||
|
public readonly partial struct OptionalNullable<T> {
|
||||||
|
[MemoryPackOrder(0)]
|
||||||
|
public bool HasValue { get; }
|
||||||
|
|
||||||
|
[MemoryPackOrder(1)]
|
||||||
|
public T? Value {
|
||||||
|
get {
|
||||||
|
if (HasValue) {
|
||||||
|
return field;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new InvalidOperationException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public OptionalNullable() : this(hasValue: false, value: default) {}
|
||||||
|
public OptionalNullable(T? value) : this(hasValue: true, value) {}
|
||||||
|
|
||||||
|
[MemoryPackConstructor]
|
||||||
|
private OptionalNullable(bool hasValue, T? value) {
|
||||||
|
this.HasValue = hasValue;
|
||||||
|
this.Value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public T? Or(T? fallbackValue) {
|
||||||
|
return HasValue ? Value : fallbackValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
public OptionalNullable<T> Or(OptionalNullable<T> fallbackOptional) {
|
||||||
|
return HasValue ? Value : fallbackOptional;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator OptionalNullable<T>(T? value) {
|
||||||
|
return new OptionalNullable<T>(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
using System.Collections.Immutable;
|
||||||
|
using MemoryPack;
|
||||||
|
using Phantom.Common.Data.Web.Agent;
|
||||||
|
|
||||||
|
namespace Phantom.Common.Messages.Web.ToWeb;
|
||||||
|
|
||||||
|
[MemoryPackable(GenerateType.VersionTolerant)]
|
||||||
|
public sealed partial record RefreshAgentsMessage2(
|
||||||
|
[property: MemoryPackOrder(0)] ImmutableArray<RefreshAgentsMessage2.IItemAction> Actions
|
||||||
|
) : IMessageToWeb {
|
||||||
|
[MemoryPackable]
|
||||||
|
[MemoryPackUnion(tag: 0, typeof(RemoveItem))]
|
||||||
|
[MemoryPackUnion(tag: 1, typeof(SetItem))]
|
||||||
|
[MemoryPackUnion(tag: 2, typeof(UpdateItem))]
|
||||||
|
public partial interface IItemAction;
|
||||||
|
|
||||||
|
[MemoryPackable]
|
||||||
|
public sealed partial record RemoveItem(Guid AgentGuid) : IItemAction;
|
||||||
|
|
||||||
|
[MemoryPackable]
|
||||||
|
public sealed partial record SetItem(Agent Agent) : IItemAction;
|
||||||
|
|
||||||
|
[MemoryPackable]
|
||||||
|
public sealed partial record UpdateItem(Guid AgentGuid, Agent.Update Update) : IItemAction;
|
||||||
|
}
|
||||||
@@ -42,6 +42,7 @@ public static class WebMessageRegistries {
|
|||||||
ToController.Add<GetEventLogMessage, Result<ImmutableArray<EventLogItem>, UserActionFailure>>();
|
ToController.Add<GetEventLogMessage, Result<ImmutableArray<EventLogItem>, UserActionFailure>>();
|
||||||
|
|
||||||
ToWeb.Add<RefreshAgentsMessage>();
|
ToWeb.Add<RefreshAgentsMessage>();
|
||||||
|
ToWeb.Add<RefreshAgentsMessage2>();
|
||||||
ToWeb.Add<RefreshInstancesMessage>();
|
ToWeb.Add<RefreshInstancesMessage>();
|
||||||
ToWeb.Add<InstanceOutputMessage>();
|
ToWeb.Add<InstanceOutputMessage>();
|
||||||
ToWeb.Add<RefreshUserSessionMessage>();
|
ToWeb.Add<RefreshUserSessionMessage>();
|
||||||
|
|||||||
@@ -11,7 +11,6 @@
|
|||||||
<PrivateAssets>all</PrivateAssets>
|
<PrivateAssets>all</PrivateAssets>
|
||||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
<PackageReference Include="System.Linq.Async" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using Phantom.Common.Data.Web.Users;
|
using Phantom.Common.Data.Web.Users;
|
||||||
using Phantom.Controller.Database;
|
using Phantom.Controller.Database;
|
||||||
using Phantom.Controller.Database.Entities;
|
using Phantom.Controller.Database.Entities;
|
||||||
@@ -9,7 +10,7 @@ namespace Phantom.Controller.Services.Users.Sessions;
|
|||||||
sealed class AuthenticatedUserCache {
|
sealed class AuthenticatedUserCache {
|
||||||
private readonly ConcurrentDictionary<Guid, AuthenticatedUserInfo> authenticatedUsersByGuid = new ();
|
private readonly ConcurrentDictionary<Guid, AuthenticatedUserInfo> authenticatedUsersByGuid = new ();
|
||||||
|
|
||||||
public bool TryGet(Guid userGuid, out AuthenticatedUserInfo? userInfo) {
|
public bool TryGet(Guid userGuid, [NotNullWhen(true)] out AuthenticatedUserInfo? userInfo) {
|
||||||
return authenticatedUsersByGuid.TryGetValue(userGuid, out userInfo);
|
return authenticatedUsersByGuid.TryGetValue(userGuid, out userInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -77,8 +77,13 @@ sealed class UserLoginManager {
|
|||||||
return userGuid != null && authenticatedUserCache.TryGet(userGuid.Value, out var userInfo) ? new LoggedInUser(userInfo) : default;
|
return userGuid != null && authenticatedUserCache.TryGet(userGuid.Value, out var userInfo) ? new LoggedInUser(userInfo) : default;
|
||||||
}
|
}
|
||||||
|
|
||||||
public AuthenticatedUserInfo? GetAuthenticatedUser(Guid userGuid, ImmutableArray<byte> authToken) {
|
public Optional<AuthenticatedUserInfo> GetAuthenticatedUser(Guid userGuid, ImmutableArray<byte> authToken) {
|
||||||
return authenticatedUserCache.TryGet(userGuid, out var userInfo) && GetSessionBucket(authToken).Contains(userGuid, authToken) ? userInfo : null;
|
if (authenticatedUserCache.TryGet(userGuid, out var userInfo) && GetSessionBucket(authToken).Contains(userGuid, authToken)) {
|
||||||
|
return userInfo;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return default;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private sealed class UserSessionBucket {
|
private sealed class UserSessionBucket {
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
<Project>
|
<Project>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>net9.0</TargetFramework>
|
<TargetFramework>net10.0</TargetFramework>
|
||||||
<LangVersion>13</LangVersion>
|
<LangVersion>14</LangVersion>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# +---------------+
|
# +---------------+
|
||||||
# | Prepare build |
|
# | Prepare build |
|
||||||
# +---------------+
|
# +---------------+
|
||||||
FROM --platform=$BUILDPLATFORM mcr.microsoft.com/dotnet/nightly/sdk:9.0 AS phantom-builder
|
FROM --platform=$BUILDPLATFORM mcr.microsoft.com/dotnet/nightly/sdk:10.0 AS phantom-builder
|
||||||
ARG TARGETARCH
|
ARG TARGETARCH
|
||||||
|
|
||||||
ADD . /app
|
ADD . /app
|
||||||
@@ -19,7 +19,7 @@ RUN find .artifacts/publish/*/* -maxdepth 0 -execdir mv '{}' 'release' \;
|
|||||||
# +---------------------+
|
# +---------------------+
|
||||||
# | Phantom Agent image |
|
# | Phantom Agent image |
|
||||||
# +---------------------+
|
# +---------------------+
|
||||||
FROM mcr.microsoft.com/dotnet/nightly/runtime:9.0 AS phantom-agent
|
FROM mcr.microsoft.com/dotnet/nightly/runtime:10.0 AS phantom-agent
|
||||||
|
|
||||||
RUN mkdir /data && chmod 777 /data
|
RUN mkdir /data && chmod 777 /data
|
||||||
WORKDIR /data
|
WORKDIR /data
|
||||||
@@ -46,7 +46,7 @@ ENTRYPOINT ["dotnet", "/app/Phantom.Agent.dll"]
|
|||||||
# +--------------------------+
|
# +--------------------------+
|
||||||
# | Phantom Controller image |
|
# | Phantom Controller image |
|
||||||
# +--------------------------+
|
# +--------------------------+
|
||||||
FROM mcr.microsoft.com/dotnet/nightly/runtime:9.0 AS phantom-controller
|
FROM mcr.microsoft.com/dotnet/nightly/runtime:10.0 AS phantom-controller
|
||||||
|
|
||||||
RUN mkdir /data && chmod 777 /data
|
RUN mkdir /data && chmod 777 /data
|
||||||
WORKDIR /data
|
WORKDIR /data
|
||||||
|
|||||||
@@ -1,15 +1,14 @@
|
|||||||
<Project>
|
<Project>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Update="Microsoft.AspNetCore.Components.Authorization" Version="9.0.9" />
|
<PackageReference Update="Microsoft.AspNetCore.Components.Authorization" Version="10.0.1" />
|
||||||
<PackageReference Update="Microsoft.AspNetCore.Components.Web" Version="9.0.9" />
|
<PackageReference Update="Microsoft.AspNetCore.Components.Web" Version="10.0.1" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Update="Microsoft.EntityFrameworkCore.Relational" Version="9.0.9" />
|
<PackageReference Update="Microsoft.EntityFrameworkCore.Tools" Version="10.0.1" />
|
||||||
<PackageReference Update="Microsoft.EntityFrameworkCore.Tools" Version="9.0.9" />
|
<PackageReference Update="Microsoft.EntityFrameworkCore.Relational" Version="10.0.1" />
|
||||||
<PackageReference Update="Npgsql.EntityFrameworkCore.PostgreSQL" Version="9.0.4" />
|
<PackageReference Update="Npgsql.EntityFrameworkCore.PostgreSQL" Version="10.0.0" />
|
||||||
<PackageReference Update="System.Linq.Async" Version="6.0.3" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
@@ -23,7 +22,7 @@
|
|||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Update="Serilog" Version="4.3.0" />
|
<PackageReference Update="Serilog" Version="4.3.0" />
|
||||||
<PackageReference Update="Serilog.AspNetCore" Version="9.0.0" />
|
<PackageReference Update="Serilog.AspNetCore" Version="10.0.0" />
|
||||||
<PackageReference Update="Serilog.Sinks.Async" Version="2.1.0" />
|
<PackageReference Update="Serilog.Sinks.Async" Version="2.1.0" />
|
||||||
<PackageReference Update="Serilog.Sinks.Console" Version="6.0.0" />
|
<PackageReference Update="Serilog.Sinks.Console" Version="6.0.0" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|||||||
@@ -2,17 +2,9 @@
|
|||||||
|
|
||||||
namespace Phantom.Utils.Processes;
|
namespace Phantom.Utils.Processes;
|
||||||
|
|
||||||
public sealed class OneShotProcess {
|
public sealed class OneShotProcess(ILogger logger, ProcessConfigurator configurator) {
|
||||||
private readonly ILogger logger;
|
|
||||||
private readonly ProcessConfigurator configurator;
|
|
||||||
|
|
||||||
public event EventHandler<Process.Output>? OutputReceived;
|
public event EventHandler<Process.Output>? OutputReceived;
|
||||||
|
|
||||||
public OneShotProcess(ILogger logger, ProcessConfigurator configurator) {
|
|
||||||
this.logger = logger;
|
|
||||||
this.configurator = configurator;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<bool> Run(CancellationToken cancellationToken) {
|
public async Task<bool> Run(CancellationToken cancellationToken) {
|
||||||
using var process = configurator.CreateProcess();
|
using var process = configurator.CreateProcess();
|
||||||
process.OutputReceived += OutputReceived;
|
process.OutputReceived += OutputReceived;
|
||||||
|
|||||||
@@ -31,6 +31,12 @@ public sealed class ProcessConfigurator {
|
|||||||
set => startInfo.UseShellExecute = value;
|
set => startInfo.UseShellExecute = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ProcessConfigurator() {
|
||||||
|
if (OperatingSystem.IsWindows()) {
|
||||||
|
startInfo.CreateNewProcessGroup = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public Process CreateProcess() {
|
public Process CreateProcess() {
|
||||||
return new Process(new System.Diagnostics.Process { StartInfo = startInfo });
|
return new Process(new System.Diagnostics.Process { StartInfo = startInfo });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -87,12 +87,7 @@ public sealed class CustomAuthenticationStateProvider : ServerAuthenticationStat
|
|||||||
}
|
}
|
||||||
|
|
||||||
var session = await controllerConnection.Send<GetAuthenticatedUser, Optional<AuthenticatedUserInfo>>(new GetAuthenticatedUser(userGuid, authToken), TimeSpan.FromSeconds(30), cancellationToken);
|
var session = await controllerConnection.Send<GetAuthenticatedUser, Optional<AuthenticatedUserInfo>>(new GetAuthenticatedUser(userGuid, authToken), TimeSpan.FromSeconds(30), cancellationToken);
|
||||||
if (session.Value is {} userInfo) {
|
return session.HasValue ? new AuthenticatedUser(session.Value, authToken) : null;
|
||||||
return new AuthenticatedUser(userInfo, authToken);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SetLoadedSession(AuthenticatedUser authenticatedUser) {
|
private void SetLoadedSession(AuthenticatedUser authenticatedUser) {
|
||||||
|
|||||||
@@ -19,10 +19,12 @@ public sealed class UserLoginManager(Navigation navigation, UserSessionBrowserSt
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (result.Value is not var (userInfo, authToken)) {
|
if (!result.HasValue) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var (userInfo, authToken) = result.Value;
|
||||||
|
|
||||||
Logger.Information("Successfully logged in {Username}.", username);
|
Logger.Information("Successfully logged in {Username}.", username);
|
||||||
|
|
||||||
authenticationStateProvider.SetUnloadedSession();
|
authenticationStateProvider.SetUnloadedSession();
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
@using Phantom.Web.Services
|
@using Phantom.Web.Errors
|
||||||
|
@using Phantom.Web.Services
|
||||||
@inject Navigation Navigation
|
@inject Navigation Navigation
|
||||||
|
|
||||||
<CascadingAuthenticationState>
|
<CascadingAuthenticationState>
|
||||||
<Router AppAssembly="@typeof(App).Assembly">
|
<Router AppAssembly="@typeof(App).Assembly" NotFoundPage="typeof(NotFound)">
|
||||||
<Found Context="routeData">
|
<Found Context="routeData">
|
||||||
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)">
|
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)">
|
||||||
<NotAuthorized>
|
<NotAuthorized>
|
||||||
@@ -17,11 +18,5 @@
|
|||||||
</AuthorizeRouteView>
|
</AuthorizeRouteView>
|
||||||
<FocusOnNavigate RouteData="@routeData" Selector="h1" />
|
<FocusOnNavigate RouteData="@routeData" Selector="h1" />
|
||||||
</Found>
|
</Found>
|
||||||
<NotFound>
|
|
||||||
<LayoutView Layout="@typeof(MainLayout)">
|
|
||||||
<h1>Not Found</h1>
|
|
||||||
<p role="alert">Sorry, there's nothing at this address.</p>
|
|
||||||
</LayoutView>
|
|
||||||
</NotFound>
|
|
||||||
</Router>
|
</Router>
|
||||||
</CascadingAuthenticationState>
|
</CascadingAuthenticationState>
|
||||||
|
|||||||
5
Web/Phantom.Web/Errors/NotFound.razor
Normal file
5
Web/Phantom.Web/Errors/NotFound.razor
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
@page "/error/404"
|
||||||
|
@layout MainLayout
|
||||||
|
|
||||||
|
<h1>Not Found</h1>
|
||||||
|
<p role="alert">Sorry, there's nothing at this address.</p>
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"sdk": {
|
"sdk": {
|
||||||
"version": "9.0.0",
|
"version": "10.0.0",
|
||||||
"rollForward": "latestMinor",
|
"rollForward": "latestMinor",
|
||||||
"allowPrerelease": true
|
"allowPrerelease": true
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user