AddResourceUseCase.cs 1.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243
  1. using RackPeek.Domain.Helpers;
  2. using RackPeek.Domain.Persistence;
  3. using RackPeek.Domain.Resources;
  4. namespace RackPeek.Domain.UseCases;
  5. public interface IAddResourceUseCase<T> : IResourceUseCase<T>
  6. where T : Resource
  7. {
  8. Task ExecuteAsync(string name, List<string>? runsOn = null);
  9. }
  10. public class AddResourceUseCase<T>(IResourceCollection repo) : IAddResourceUseCase<T> where T : Resource
  11. {
  12. public async Task ExecuteAsync(string name, List<string>? runsOn = null)
  13. {
  14. name = Normalize.HardwareName(name);
  15. ThrowIfInvalid.ResourceName(name);
  16. Resource? existingResource = await repo.GetByNameAsync(name);
  17. if (existingResource != null)
  18. throw new ConflictException($"Resource '{name}' ({existingResource.Kind}) already exists.");
  19. if (runsOn != null)
  20. foreach (var parent in runsOn)
  21. {
  22. var normalizedParent = Normalize.HardwareName(parent);
  23. ThrowIfInvalid.ResourceName(normalizedParent);
  24. Resource? parentResource = await repo.GetByNameAsync(normalizedParent);
  25. if (parentResource == null) throw new NotFoundException($"Resource '{normalizedParent}' not found.");
  26. if (!Resource.CanRunOn<T>(parentResource))
  27. throw new InvalidOperationException(
  28. $" {Resource.GetKind<T>()} cannot run on {parentResource.Kind} '{normalizedParent}'.");
  29. }
  30. T resource = Activator.CreateInstance<T>();
  31. resource.Name = name;
  32. resource.RunsOn = runsOn ?? new List<string>();
  33. await repo.AddAsync(resource);
  34. }
  35. }