namespace NightLight.Core.Tests open NightLight.Core.Core open NightLight.Core.Tests.GenHelpers open NightLight.Core.Tests.TimeChangedGenerators open NightLight.Core.Tests.InteractionListGenerators open NightLight.Core.Models open FsCheck.Xunit open FsCheck.FSharp type NightLightTests() = let createFakeHomeWithNightLightAndInteract (interactions: Interaction list) = let mutable nightLightStateMachine = NightLightStateMachine() let fakeHome = FakeHome() fakeHome.OnEventPublished.Add(fun event -> match event |> nightLightStateMachine.OnEventReceived with | Ok(newState, commands) -> commands |> Seq.iter fakeHome.ProcessCommand nightLightStateMachine <- newState | Error error -> failwith $"Unexpected error {error}") fakeHome.Interact interactions fakeHome [ |])>] let ``All lights should be either off, white or yellow during the day`` (light: Light) = genInteractions light |> ensurePartOfDayIs Day |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light (function | Off -> true | On(_, color) -> color = White || color = Yellow) [ |])>] let ``All lights should be either off or red during the night`` (light: Light) = genInteractions light |> ensurePartOfDayIs Night |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light (function | Off -> true | On(_, color) -> color = Red) [ |])>] let ``All non-remotely controlled lights with power should be on`` (light: Light) = genInteractions light |> ensureLightHasPower light |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOn [ |])>] let ``All remote controlled lights with power should be on if the remote was never used`` (light: Light) = genInteractionsExcept light _.IsRemoteInteraction |> ensureLightHasPower light |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOn [ |])>] let ``After pressing 'On' on the remote, if the remote isn't used again, all remotely controlled lights with power should be on`` (light: Light) = genInteractions light |> ensureLastRemoteInteractionIs RemotePressedOnButton |> ensureLightHasPower light |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOn [ |])>] let ``After a new day starts, if the remote isn't used, all remotely controlled lights with power should be on`` (light: Light) = concatGens [ genInteractions light |> ensurePartOfDayIs Night genTimeChangedToPartOfDay Day |> Gen.map List.singleton genInteractionsExcept light _.IsRemoteInteraction ] |> ensureStartsWithTimeChanged |> ensureLightHasPower light |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOn [ |])>] let ``After pressing 'Off' on the remote, if the remote isn't used again and a new day doesn't start, all remotely controlled lights should be off`` (light: Light) = concatGens [ genInteractions light RemoteInteraction RemotePressedOffButton |> List.singleton |> Gen.constant genInteractionsExcept light (fun interaction -> interaction.IsRemoteInteraction || interaction |> isTimeChangedToPartOfDay Day) ] |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOff [ |])>] let ``After pressing 'Left' on the remote, if the remote isn't used again, all left-side remotely controlled lights with power should be on`` (light: Light) = genInteractions light |> ensureLastRemoteInteractionIs RemotePressedLeftButton |> ensureStartsWithTimeChanged |> ensureLightHasPower light |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOn [ |])>] let ``After pressing 'Left' on the remote, if the remote isn't used again and a new day doesn't start, all right-side remotely controlled lights should be off`` (light: Light) = concatGens [ genInteractions light RemoteInteraction RemotePressedLeftButton |> List.singleton |> Gen.constant genInteractionsExcept light (fun interaction -> interaction.IsRemoteInteraction || interaction |> isTimeChangedToPartOfDay Day) ] |> ensureStartsWithTimeChanged |> Arb.fromGen |> Prop.forAll <| fun interactions -> let fakeHome = createFakeHomeWithNightLightAndInteract interactions fakeHome.LightShouldHaveState light _.IsOff