431 lines
13 KiB
C#
431 lines
13 KiB
C#
using Microsoft.EntityFrameworkCore;
|
|
using Moonlight.App.Database;
|
|
using Moonlight.App.Database.Entities;
|
|
using Moonlight.App.Exceptions;
|
|
using Moonlight.App.Helpers;
|
|
using Moonlight.App.Helpers.Files;
|
|
using Moonlight.App.Models.Misc;
|
|
using Moonlight.App.Models.Wings;
|
|
using Moonlight.App.Models.Wings.Requests;
|
|
using Moonlight.App.Models.Wings.Resources;
|
|
using Moonlight.App.Repositories;
|
|
using Moonlight.App.Repositories.Servers;
|
|
using Moonlight.App.Services.LogServices;
|
|
using FileAccess = Moonlight.App.Helpers.Files.FileAccess;
|
|
|
|
namespace Moonlight.App.Services;
|
|
|
|
public class ServerService
|
|
{
|
|
private readonly ServerRepository ServerRepository;
|
|
private readonly UserRepository UserRepository;
|
|
private readonly ImageRepository ImageRepository;
|
|
private readonly NodeRepository NodeRepository;
|
|
private readonly WingsApiHelper WingsApiHelper;
|
|
private readonly MessageService MessageService;
|
|
private readonly UserService UserService;
|
|
private readonly ConfigService ConfigService;
|
|
private readonly WingsJwtHelper WingsJwtHelper;
|
|
private readonly SecurityLogService SecurityLogService;
|
|
private readonly AuditLogService AuditLogService;
|
|
private readonly ErrorLogService ErrorLogService;
|
|
private readonly NodeService NodeService;
|
|
|
|
public ServerService(
|
|
ServerRepository serverRepository,
|
|
WingsApiHelper wingsApiHelper,
|
|
UserRepository userRepository,
|
|
ImageRepository imageRepository,
|
|
NodeRepository nodeRepository,
|
|
MessageService messageService,
|
|
UserService userService,
|
|
ConfigService configService,
|
|
WingsJwtHelper wingsJwtHelper,
|
|
SecurityLogService securityLogService,
|
|
AuditLogService auditLogService,
|
|
ErrorLogService errorLogService,
|
|
NodeService nodeService)
|
|
{
|
|
ServerRepository = serverRepository;
|
|
WingsApiHelper = wingsApiHelper;
|
|
UserRepository = userRepository;
|
|
ImageRepository = imageRepository;
|
|
NodeRepository = nodeRepository;
|
|
MessageService = messageService;
|
|
UserService = userService;
|
|
ConfigService = configService;
|
|
WingsJwtHelper = wingsJwtHelper;
|
|
SecurityLogService = securityLogService;
|
|
AuditLogService = auditLogService;
|
|
ErrorLogService = errorLogService;
|
|
NodeService = nodeService;
|
|
}
|
|
|
|
private Server EnsureNodeData(Server s)
|
|
{
|
|
if (s.Node == null) // Ensure node data is available
|
|
{
|
|
return ServerRepository
|
|
.Get()
|
|
.Include(x => x.Node)
|
|
.First(x => x.Id == s.Id);
|
|
}
|
|
else
|
|
return s;
|
|
}
|
|
|
|
public async Task<ServerDetails> GetDetails(Server s)
|
|
{
|
|
Server server = EnsureNodeData(s);
|
|
|
|
return await WingsApiHelper.Get<ServerDetails>(
|
|
server.Node,
|
|
$"api/servers/{server.Uuid}"
|
|
);
|
|
}
|
|
|
|
public async Task SetPowerState(Server s, PowerSignal signal)
|
|
{
|
|
Server server = EnsureNodeData(s);
|
|
|
|
var rawSignal = signal.ToString().ToLower();
|
|
|
|
await WingsApiHelper.Post(server.Node, $"api/servers/{server.Uuid}/power", new ServerPower()
|
|
{
|
|
Action = rawSignal
|
|
});
|
|
|
|
await AuditLogService.Log(AuditLogType.ChangePowerState, x =>
|
|
{
|
|
x.Add<Server>(server.Uuid);
|
|
x.Add<PowerSignal>(rawSignal);
|
|
});
|
|
}
|
|
|
|
public async Task<ServerBackup> CreateBackup(Server server)
|
|
{
|
|
var serverData = ServerRepository // Ensure data
|
|
.Get()
|
|
.Include(x => x.Node)
|
|
.Include(x => x.Backups)
|
|
.First(x => x.Id == server.Id);
|
|
|
|
var backup = new ServerBackup()
|
|
{
|
|
Name = $"Created at {DateTime.Now.ToShortDateString()} {DateTime.Now.ToShortTimeString()}",
|
|
Uuid = Guid.NewGuid(),
|
|
CreatedAt = DateTime.Now,
|
|
Created = false
|
|
};
|
|
|
|
serverData.Backups.Add(backup);
|
|
ServerRepository.Update(serverData);
|
|
|
|
await WingsApiHelper.Post(serverData.Node, $"api/servers/{serverData.Uuid}/backup", new CreateBackup()
|
|
{
|
|
Adapter = "wings",
|
|
Uuid = backup.Uuid,
|
|
Ignore = ""
|
|
});
|
|
|
|
await AuditLogService.Log(AuditLogType.CreateBackup,
|
|
x =>
|
|
{
|
|
x.Add<Server>(server.Uuid);
|
|
x.Add<ServerBackup>(backup.Uuid);
|
|
});
|
|
|
|
return backup;
|
|
}
|
|
|
|
public Task<ServerBackup[]> GetBackups(Server server, bool forceReload = false)
|
|
{
|
|
if (forceReload) //TODO: Find an alternative to avoid cache and the creation of a new db context
|
|
{
|
|
var serverData = new ServerRepository(new DataContext(ConfigService))
|
|
.Get()
|
|
.Include(x => x.Backups)
|
|
.First(x => x.Id == server.Id);
|
|
|
|
return Task.FromResult(serverData.Backups.ToArray());
|
|
}
|
|
else
|
|
{
|
|
var serverData = ServerRepository
|
|
.Get()
|
|
.Include(x => x.Backups)
|
|
.First(x => x.Id == server.Id);
|
|
|
|
return Task.FromResult(serverData.Backups.ToArray());
|
|
}
|
|
}
|
|
|
|
public async Task RestoreBackup(Server s, ServerBackup serverBackup)
|
|
{
|
|
Server server = EnsureNodeData(s);
|
|
|
|
await WingsApiHelper.Post(server.Node, $"api/servers/{server.Uuid}/backup/{serverBackup.Uuid}/restore",
|
|
new RestoreBackup()
|
|
{
|
|
Adapter = "wings"
|
|
});
|
|
|
|
await AuditLogService.Log(AuditLogType.RestoreBackup,
|
|
x =>
|
|
{
|
|
x.Add<Server>(server.Uuid);
|
|
x.Add<ServerBackup>(serverBackup.Uuid);
|
|
});
|
|
}
|
|
|
|
public async Task DeleteBackup(Server server, ServerBackup serverBackup)
|
|
{
|
|
var serverData = ServerRepository
|
|
.Get()
|
|
.Include(x => x.Node)
|
|
.Include(x => x.Backups)
|
|
.First(x => x.Id == server.Id);
|
|
|
|
await WingsApiHelper.Delete(serverData.Node, $"api/servers/{serverData.Uuid}/backup/{serverBackup.Uuid}",
|
|
null);
|
|
|
|
var backup = serverData.Backups.First(x => x.Uuid == serverBackup.Uuid);
|
|
serverData.Backups.Remove(backup);
|
|
|
|
ServerRepository.Update(serverData);
|
|
|
|
await MessageService.Emit("wings.backups.delete", backup);
|
|
|
|
await AuditLogService.Log(AuditLogType.DeleteBackup,
|
|
x =>
|
|
{
|
|
x.Add<Server>(server.Uuid);
|
|
x.Add<ServerBackup>(backup.Uuid);
|
|
}
|
|
);
|
|
}
|
|
|
|
public async Task<string> DownloadBackup(Server s, ServerBackup serverBackup)
|
|
{
|
|
Server server = EnsureNodeData(s);
|
|
|
|
var token = WingsJwtHelper.Generate(server.Node.Token, claims =>
|
|
{
|
|
claims.Add("server_uuid", server.Uuid.ToString());
|
|
claims.Add("backup_uuid", serverBackup.Uuid.ToString());
|
|
});
|
|
|
|
await AuditLogService.Log(AuditLogType.DownloadBackup,
|
|
x =>
|
|
{
|
|
x.Add<Server>(server.Uuid);
|
|
x.Add<ServerBackup>(serverBackup.Uuid);
|
|
});
|
|
|
|
if (server.Node.Ssl)
|
|
return $"https://{server.Node.Fqdn}:{server.Node.HttpPort}/download/backup?token={token}";
|
|
else
|
|
return $"http://{server.Node.Fqdn}:{server.Node.HttpPort}/download/backup?token={token}";
|
|
}
|
|
|
|
public Task<FileAccess> CreateFileAccess(Server s, User user) // We need the user to create the launch url
|
|
{
|
|
Server server = EnsureNodeData(s);
|
|
|
|
return Task.FromResult(
|
|
(FileAccess)new WingsFileAccess(
|
|
WingsApiHelper,
|
|
WingsJwtHelper,
|
|
server,
|
|
ConfigService,
|
|
user
|
|
)
|
|
);
|
|
}
|
|
|
|
public async Task<Server> Create(string name, int cpu, long memory, long disk, User u, Image i, Node? n = null,
|
|
Action<Server>? modifyDetails = null, int allocations = 1)
|
|
{
|
|
var user = UserRepository
|
|
.Get()
|
|
.First(x => x.Id == u.Id);
|
|
|
|
var image = ImageRepository
|
|
.Get()
|
|
.Include(x => x.Variables)
|
|
.Include(x => x.DockerImages)
|
|
.First(x => x.Id == i.Id);
|
|
|
|
Node node;
|
|
|
|
if (n == null)
|
|
{
|
|
node = NodeRepository
|
|
.Get()
|
|
.Include(x => x.Allocations)
|
|
.First(); //TODO: Add smart deploy maybe
|
|
}
|
|
else
|
|
{
|
|
node = NodeRepository
|
|
.Get()
|
|
.Include(x => x.Allocations)
|
|
.First(x => x.Id == n.Id);
|
|
}
|
|
|
|
NodeAllocation[] freeAllocations;
|
|
|
|
try
|
|
{
|
|
freeAllocations = node.Allocations
|
|
.Where(a => !ServerRepository.Get()
|
|
.SelectMany(s => s.Allocations)
|
|
.Any(b => b.Id == a.Id))
|
|
.Take(allocations).ToArray();
|
|
}
|
|
catch (Exception)
|
|
{
|
|
throw new DisplayException("No allocation found");
|
|
}
|
|
|
|
if (!freeAllocations.Any())
|
|
throw new DisplayException("No allocation found");
|
|
|
|
if (freeAllocations.Length != allocations)
|
|
throw new DisplayException("Not enough allocations found");
|
|
|
|
var server = new Server()
|
|
{
|
|
Cpu = cpu,
|
|
Memory = memory,
|
|
Disk = disk,
|
|
Name = name,
|
|
Image = image,
|
|
Owner = user,
|
|
Node = node,
|
|
Uuid = Guid.NewGuid(),
|
|
MainAllocation = freeAllocations.First(),
|
|
Allocations = freeAllocations.ToList(),
|
|
Backups = new(),
|
|
OverrideStartup = "",
|
|
DockerImageIndex = image.DockerImages.FindIndex(x => x.Default)
|
|
};
|
|
|
|
foreach (var imageVariable in image.Variables)
|
|
{
|
|
server.Variables.Add(new()
|
|
{
|
|
Key = imageVariable.Key,
|
|
Value = imageVariable.DefaultValue
|
|
});
|
|
}
|
|
|
|
if (modifyDetails != null)
|
|
modifyDetails.Invoke(server);
|
|
|
|
var newServerData = ServerRepository.Add(server);
|
|
|
|
try
|
|
{
|
|
await WingsApiHelper.Post(node, $"api/servers", new CreateServer()
|
|
{
|
|
Uuid = newServerData.Uuid,
|
|
StartOnCompletion = false
|
|
});
|
|
|
|
await AuditLogService.Log(AuditLogType.CreateServer, x => { x.Add<Server>(newServerData.Uuid); });
|
|
|
|
return newServerData;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
await ErrorLogService.Log(e, x =>
|
|
{
|
|
x.Add<Server>(newServerData.Uuid);
|
|
x.Add<Node>(node.Id);
|
|
});
|
|
|
|
ServerRepository.Delete(newServerData); //TODO Remove unsinged table stuff
|
|
|
|
throw new DisplayException("Error creating server on wings");
|
|
}
|
|
}
|
|
|
|
public async Task Reinstall(Server s)
|
|
{
|
|
Server server = EnsureNodeData(s);
|
|
|
|
await WingsApiHelper.Post(server.Node, $"api/servers/{server.Uuid}/reinstall", null);
|
|
|
|
await AuditLogService.Log(AuditLogType.ReinstallServer, x => { x.Add<Server>(server.Uuid); });
|
|
}
|
|
|
|
public async Task<Server> SftpServerLogin(int serverId, int id, string password)
|
|
{
|
|
var server = ServerRepository.Get().FirstOrDefault(x => x.Id == serverId);
|
|
|
|
if (server == null)
|
|
{
|
|
await SecurityLogService.LogSystem(SecurityLogType.SftpBruteForce, x => { x.Add<int>(id); });
|
|
throw new Exception("Server not found");
|
|
}
|
|
|
|
var user = await UserService.SftpLogin(id, password);
|
|
|
|
if (server.Owner.Id == user.Id)
|
|
{
|
|
return server;
|
|
}
|
|
else
|
|
{
|
|
//TODO: Decide if logging
|
|
throw new Exception("User and owner id do not match");
|
|
}
|
|
}
|
|
|
|
public async Task Sync(Server s)
|
|
{
|
|
var server = EnsureNodeData(s);
|
|
|
|
await WingsApiHelper.Post(server.Node, $"api/servers/{server.Uuid}/sync", null);
|
|
}
|
|
|
|
public async Task Delete(Server s)
|
|
{
|
|
var server = EnsureNodeData(s);
|
|
|
|
var backups = await GetBackups(server);
|
|
|
|
foreach (var backup in backups)
|
|
{
|
|
try
|
|
{
|
|
await DeleteBackup(server, backup);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
// ignored
|
|
}
|
|
}
|
|
|
|
await WingsApiHelper.Delete(server.Node, $"api/servers/{server.Uuid}", null);
|
|
|
|
//TODO: Fix empty data models
|
|
|
|
server.Allocations = new();
|
|
server.MainAllocation = null;
|
|
server.Variables = new();
|
|
server.Backups = new();
|
|
|
|
ServerRepository.Update(server);
|
|
|
|
ServerRepository.Delete(server);
|
|
}
|
|
|
|
public async Task<bool> IsHostUp(Server s)
|
|
{
|
|
var server = EnsureNodeData(s);
|
|
|
|
return await NodeService.IsHostUp(server.Node);
|
|
}
|
|
} |