AddResourceUseCase.cs 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445
  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. var existingResource = await repo.GetByNameAsync(name);
  17. if (existingResource != null)
  18. throw new ConflictException($"Resource '{name}' ({existingResource.Kind}) already exists.");
  19. if (runsOn != null)
  20. {
  21. foreach (var parent in runsOn) {
  22. var normalizedParent = Normalize.HardwareName(parent);
  23. ThrowIfInvalid.ResourceName(normalizedParent);
  24. var 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. }
  31. var resource = Activator.CreateInstance<T>();
  32. resource.Name = name;
  33. resource.RunsOn = runsOn ?? new List<string>();
  34. await repo.AddAsync(resource);
  35. }
  36. }