2.2 FramesOptions (frames.json binding + validator)

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-06-07 14:21:24 +02:00
parent 20e6aafaa1
commit d0fa755534
4 changed files with 461 additions and 1 deletions

View File

@@ -76,7 +76,7 @@ Each type lives in `src/FrameProcessor/Domain/`. Tests in `tests/FrameProcessor.
- Bound from `appsettings.json` via `builder.Services.Configure<T>(...)`.
- Validate on startup (`ValidateOnStart` + `IValidateOptions<T>` or DataAnnotations).
### [ ] 2.2 `FramesOptions` (from `frames.json`)
### [x] 2.2 `FramesOptions` (from `frames.json`)
- Top-level `{ Frames: Frame[] }` POCO.
- Register `frames.json` as an additional config source with `reloadOnChange: true`.
- Bind via `IOptionsMonitor<FramesOptions>`.

View File

@@ -0,0 +1,186 @@
using FrameProcessor.Domain;
using Microsoft.Extensions.Options;
using SixLabors.ImageSharp;
namespace FrameProcessor.Configuration;
/// <summary>
/// Bound from <c>frames.json</c> (registered as an additional config source with
/// <c>reloadOnChange: true</c>). Holds the raw, un-typed frame entries; parsing into
/// <see cref="FrameName"/>, <see cref="MacAddress"/>, etc. happens downstream after
/// <see cref="FramesOptionsValidator"/> confirms the values are well-formed.
/// See SPEC.md §6.2.
/// </summary>
public sealed class FramesOptions
{
public List<FrameOptions> Frames { get; set; } = new();
}
public sealed class FrameOptions
{
public string? Name { get; set; }
public string? Mac { get; set; }
public ResolutionOptions? Resolution { get; set; }
public string? Orientation { get; set; }
public string? Dithering { get; set; }
public List<PaletteEntryOptions> Palette { get; set; } = new();
}
public sealed class ResolutionOptions
{
public int Width { get; set; }
public int Height { get; set; }
}
public sealed class PaletteEntryOptions
{
public string? Name { get; set; }
public string? Color { get; set; }
public string? DeviceColor { get; set; }
}
/// <summary>
/// Enforces the field-level rules from SPEC.md §6.2: name is URL-safe, MAC parseable,
/// resolution positive, orientation and dithering keys recognized, palette has at
/// least two entries with parseable hex colors, and names + MACs are unique across
/// frames.
/// </summary>
public sealed class FramesOptionsValidator : IValidateOptions<FramesOptions>
{
public ValidateOptionsResult Validate(string? name, FramesOptions options)
{
var errors = new List<string>();
for (var i = 0; i < options.Frames.Count; i++)
{
ValidateFrame(options.Frames[i], $"Frames[{i}]", errors);
}
ValidateUniqueness(options.Frames, errors);
return errors.Count == 0
? ValidateOptionsResult.Success
: ValidateOptionsResult.Fail(errors);
}
private static void ValidateFrame(FrameOptions frame, string prefix, List<string> errors)
{
if (string.IsNullOrWhiteSpace(frame.Name))
{
errors.Add($"{prefix}.Name is required.");
}
else if (!FrameName.TryParse(frame.Name, out _))
{
errors.Add($"{prefix}.Name '{frame.Name}' is not a valid frame name (RFC 3986 unreserved characters only).");
}
if (string.IsNullOrWhiteSpace(frame.Mac))
{
errors.Add($"{prefix}.Mac is required.");
}
else if (!MacAddress.TryParse(frame.Mac, out _))
{
errors.Add($"{prefix}.Mac '{frame.Mac}' is not a valid MAC address.");
}
if (frame.Resolution is null)
{
errors.Add($"{prefix}.Resolution is required.");
}
else
{
if (frame.Resolution.Width <= 0)
{
errors.Add($"{prefix}.Resolution.Width must be positive (got {frame.Resolution.Width}).");
}
if (frame.Resolution.Height <= 0)
{
errors.Add($"{prefix}.Resolution.Height must be positive (got {frame.Resolution.Height}).");
}
}
if (!IsKnownOrientation(frame.Orientation))
{
errors.Add($"{prefix}.Orientation '{frame.Orientation}' is not valid. Expected 'landscape' or 'portrait'.");
}
if (!IsKnownDithering(frame.Dithering))
{
errors.Add($"{prefix}.Dithering '{frame.Dithering}' is not valid. Expected 'floyd-steinberg', 'atkinson', 'stucki', or 'jarvis'.");
}
if (frame.Palette.Count < 2)
{
errors.Add($"{prefix}.Palette must contain at least 2 entries (found {frame.Palette.Count}).");
}
for (var j = 0; j < frame.Palette.Count; j++)
{
ValidatePaletteEntry(frame.Palette[j], $"{prefix}.Palette[{j}]", errors);
}
}
private static void ValidatePaletteEntry(PaletteEntryOptions entry, string prefix, List<string> errors)
{
if (string.IsNullOrWhiteSpace(entry.Name))
{
errors.Add($"{prefix}.Name is required.");
}
if (string.IsNullOrWhiteSpace(entry.Color) || !Color.TryParseHex(entry.Color, out _))
{
errors.Add($"{prefix}.Color '{entry.Color}' is not a valid hex color.");
}
if (string.IsNullOrWhiteSpace(entry.DeviceColor) || !Color.TryParseHex(entry.DeviceColor, out _))
{
errors.Add($"{prefix}.DeviceColor '{entry.DeviceColor}' is not a valid hex color.");
}
}
private static void ValidateUniqueness(List<FrameOptions> frames, List<string> errors)
{
var seenNames = new HashSet<string>(StringComparer.Ordinal);
var seenMacs = new HashSet<string>(StringComparer.Ordinal);
for (var i = 0; i < frames.Count; i++)
{
var frame = frames[i];
if (!string.IsNullOrWhiteSpace(frame.Name) && !seenNames.Add(frame.Name))
{
errors.Add($"Frames[{i}].Name '{frame.Name}' is a duplicate.");
}
if (!string.IsNullOrWhiteSpace(frame.Mac) && MacAddress.TryParse(frame.Mac, out var mac))
{
var canonical = mac.ToString();
if (!seenMacs.Add(canonical))
{
errors.Add($"Frames[{i}].Mac '{frame.Mac}' (canonical '{canonical}') is a duplicate.");
}
}
}
}
private static bool IsKnownOrientation(string? value) => value switch
{
"landscape" or "portrait" => true,
_ => false,
};
private static bool IsKnownDithering(string? value) => value switch
{
"floyd-steinberg" or "atkinson" or "stucki" or "jarvis" => true,
_ => false,
};
}

View File

@@ -1,7 +1,10 @@
using FrameProcessor.Configuration;
using Microsoft.Extensions.Options;
var builder = WebApplication.CreateBuilder(args);
builder.Configuration.AddJsonFile("frames.json", optional: false, reloadOnChange: true);
builder.Services.AddControllers();
builder.Services.AddOptions<MqttOptions>()
@@ -24,6 +27,11 @@ builder.Services.AddOptions<ApiKeyOptions>()
.ValidateDataAnnotations()
.ValidateOnStart();
builder.Services.AddSingleton<IValidateOptions<FramesOptions>, FramesOptionsValidator>();
builder.Services.AddOptions<FramesOptions>()
.Bind(builder.Configuration)
.ValidateOnStart();
var app = builder.Build();
app.MapControllers();

View File

@@ -0,0 +1,266 @@
using FrameProcessor.Configuration;
namespace FrameProcessor.Tests;
public class FramesOptionsValidatorTests
{
private readonly FramesOptionsValidator _validator = new();
[Fact]
public void NoFrames_IsValid()
{
var result = _validator.Validate(null, new FramesOptions());
Assert.True(result.Succeeded);
}
[Fact]
public void ValidFrame_Succeeds()
{
var options = new FramesOptions { Frames = { ValidFrame() } };
var result = _validator.Validate(null, options);
Assert.True(result.Succeeded, JoinFailures(result));
}
[Theory]
[InlineData(null)]
[InlineData("")]
[InlineData(" ")]
public void Name_MissingOrBlank_Fails(string? badName)
{
var frame = ValidFrame();
frame.Name = badName;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Name is required"));
}
[Theory]
[InlineData("living room")]
[InlineData("living/room")]
[InlineData("li:vingroom")]
public void Name_NotUrlSafe_Fails(string badName)
{
var frame = ValidFrame();
frame.Name = badName;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("is not a valid frame name"));
}
[Theory]
[InlineData(null)]
[InlineData("")]
public void Mac_Missing_Fails(string? badMac)
{
var frame = ValidFrame();
frame.Mac = badMac;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Mac is required"));
}
[Theory]
[InlineData("ZZ:ZZ:ZZ:ZZ:ZZ:ZZ")]
[InlineData("AABBCC")]
[InlineData("not-a-mac")]
public void Mac_Unparseable_Fails(string badMac)
{
var frame = ValidFrame();
frame.Mac = badMac;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("is not a valid MAC address"));
}
[Fact]
public void Resolution_Missing_Fails()
{
var frame = ValidFrame();
frame.Resolution = null;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Resolution is required"));
}
[Theory]
[InlineData(0, 1200)]
[InlineData(1600, 0)]
[InlineData(-1, 1200)]
[InlineData(1600, -1)]
public void Resolution_NonPositive_Fails(int width, int height)
{
var frame = ValidFrame();
frame.Resolution = new ResolutionOptions { Width = width, Height = height };
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("must be positive"));
}
[Theory]
[InlineData(null)]
[InlineData("")]
[InlineData("LANDSCAPE")]
[InlineData("sideways")]
public void Orientation_Invalid_Fails(string? badOrientation)
{
var frame = ValidFrame();
frame.Orientation = badOrientation;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Orientation"));
}
[Theory]
[InlineData(null)]
[InlineData("")]
[InlineData("FloydSteinberg")]
[InlineData("ordered")]
public void Dithering_Invalid_Fails(string? badDithering)
{
var frame = ValidFrame();
frame.Dithering = badDithering;
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Dithering"));
}
[Fact]
public void Palette_FewerThanTwoEntries_Fails()
{
var frame = ValidFrame();
frame.Palette = new List<PaletteEntryOptions>
{
new() { Name = "black", Color = "#000000", DeviceColor = "#000000" },
};
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("at least 2 entries"));
}
[Fact]
public void Palette_EntryWithBadHex_Fails()
{
var frame = ValidFrame();
frame.Palette[0].Color = "not-a-color";
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Color 'not-a-color' is not a valid hex color"));
}
[Fact]
public void Palette_EntryWithBadDeviceHex_Fails()
{
var frame = ValidFrame();
frame.Palette[1].DeviceColor = "zzzzzz";
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("DeviceColor 'zzzzzz' is not a valid hex color"));
}
[Fact]
public void Palette_EntryMissingName_Fails()
{
var frame = ValidFrame();
frame.Palette[0].Name = "";
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Palette[0].Name is required"));
}
[Fact]
public void DuplicateName_Fails()
{
var a = ValidFrame();
var b = ValidFrame();
b.Mac = "11:22:33:44:55:66";
var result = _validator.Validate(null, new FramesOptions { Frames = { a, b } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("Name 'living-room' is a duplicate"));
}
[Fact]
public void DuplicateMac_DetectedAfterCanonicalization()
{
var a = ValidFrame();
a.Mac = "AA:BB:CC:DD:EE:FF";
var b = ValidFrame();
b.Name = "kitchen";
b.Mac = "aabbccddeeff";
var result = _validator.Validate(null, new FramesOptions { Frames = { a, b } });
Assert.False(result.Succeeded);
Assert.Contains(result.Failures!, f => f.Contains("is a duplicate"));
}
[Fact]
public void AllErrors_AreReportedTogether()
{
var frame = new FrameOptions
{
Name = "bad name",
Mac = "not-a-mac",
Resolution = null,
Orientation = "diagonal",
Dithering = "ordered",
Palette = new List<PaletteEntryOptions>(),
};
var result = _validator.Validate(null, new FramesOptions { Frames = { frame } });
Assert.False(result.Succeeded);
var failures = result.Failures!.ToList();
Assert.Contains(failures, f => f.Contains("Name"));
Assert.Contains(failures, f => f.Contains("Mac"));
Assert.Contains(failures, f => f.Contains("Resolution is required"));
Assert.Contains(failures, f => f.Contains("Orientation"));
Assert.Contains(failures, f => f.Contains("Dithering"));
Assert.Contains(failures, f => f.Contains("at least 2 entries"));
}
private static FrameOptions ValidFrame() => new()
{
Name = "living-room",
Mac = "AA:BB:CC:DD:EE:FF",
Resolution = new ResolutionOptions { Width = 1600, Height = 1200 },
Orientation = "landscape",
Dithering = "floyd-steinberg",
Palette = new List<PaletteEntryOptions>
{
new() { Name = "black", Color = "#1F2226", DeviceColor = "#000000" },
new() { Name = "white", Color = "#B9C7C9", DeviceColor = "#FFFFFF" },
},
};
private static string JoinFailures(Microsoft.Extensions.Options.ValidateOptionsResult result)
=> result.Failures is null ? string.Empty : string.Join("; ", result.Failures);
}