// Copyright (c) FIRST and other WPILib contributors. // Open Source Software; you can modify and/or share it under the terms of // the WPILib BSD license file in the root directory of this project. #include "wpi/commands2/ParallelDeadlineGroup.hpp" #include #include #include #include "CommandTestBase.hpp" #include "CompositionTestBase.hpp" #include "wpi/commands2/InstantCommand.hpp" #include "wpi/commands2/WaitUntilCommand.hpp" using namespace wpi::cmd; class ParallelDeadlineGroupTest : public CommandTestBase {}; TEST_F(ParallelDeadlineGroupTest, DeadlineGroupSchedule) { CommandScheduler scheduler = GetScheduler(); std::unique_ptr command1Holder = std::make_unique(); std::unique_ptr command2Holder = std::make_unique(); std::unique_ptr command3Holder = std::make_unique(); MockCommand* command1 = command1Holder.get(); MockCommand* command2 = command2Holder.get(); MockCommand* command3 = command3Holder.get(); ParallelDeadlineGroup group( std::move(command1Holder), make_vector>(std::move(command2Holder), std::move(command3Holder))); EXPECT_CALL(*command1, Initialize()); EXPECT_CALL(*command1, Execute()).Times(2); EXPECT_CALL(*command1, End(false)); EXPECT_CALL(*command2, Initialize()); EXPECT_CALL(*command2, Execute()).Times(1); EXPECT_CALL(*command2, End(false)); EXPECT_CALL(*command3, Initialize()); EXPECT_CALL(*command3, Execute()).Times(2); EXPECT_CALL(*command3, End(true)); scheduler.Schedule(&group); command2->SetFinished(true); scheduler.Run(); command1->SetFinished(true); scheduler.Run(); EXPECT_FALSE(scheduler.IsScheduled(&group)); } TEST_F(ParallelDeadlineGroupTest, SequentialGroupInterrupt) { CommandScheduler scheduler = GetScheduler(); TestSubsystem subsystem; std::unique_ptr command1Holder = std::make_unique(); std::unique_ptr command2Holder = std::make_unique(); std::unique_ptr command3Holder = std::make_unique(); MockCommand* command1 = command1Holder.get(); MockCommand* command2 = command2Holder.get(); MockCommand* command3 = command3Holder.get(); ParallelDeadlineGroup group( std::move(command1Holder), make_vector>(std::move(command2Holder), std::move(command3Holder))); EXPECT_CALL(*command1, Initialize()); EXPECT_CALL(*command1, Execute()).Times(1); EXPECT_CALL(*command1, End(true)); EXPECT_CALL(*command2, Initialize()); EXPECT_CALL(*command2, Execute()).Times(1); EXPECT_CALL(*command2, End(true)); EXPECT_CALL(*command3, Initialize()); EXPECT_CALL(*command3, Execute()).Times(1); EXPECT_CALL(*command3, End(true)); scheduler.Schedule(&group); scheduler.Run(); scheduler.Cancel(&group); scheduler.Run(); EXPECT_FALSE(scheduler.IsScheduled(&group)); } TEST_F(ParallelDeadlineGroupTest, DeadlineGroupNotScheduledCancel) { CommandScheduler scheduler = GetScheduler(); auto group = cmd::Deadline(cmd::None(), cmd::None()); EXPECT_NO_FATAL_FAILURE(scheduler.Cancel(group)); } TEST_F(ParallelDeadlineGroupTest, ParallelDeadlineCopy) { CommandScheduler scheduler = GetScheduler(); bool finished = false; auto command = cmd::WaitUntil([&finished] { return finished; }); auto group = cmd::Deadline(std::move(command)); scheduler.Schedule(group); scheduler.Run(); EXPECT_TRUE(scheduler.IsScheduled(group)); finished = true; scheduler.Run(); EXPECT_FALSE(scheduler.IsScheduled(group)); } TEST_F(ParallelDeadlineGroupTest, ParallelDeadlineRequirement) { CommandScheduler scheduler = GetScheduler(); TestSubsystem requirement1; TestSubsystem requirement2; TestSubsystem requirement3; TestSubsystem requirement4; auto command1 = cmd::RunOnce([] {}, {&requirement1, &requirement2}); auto command2 = cmd::RunOnce([] {}, {&requirement3}); auto command3 = cmd::RunOnce([] {}, {&requirement3, &requirement4}); auto group = cmd::Deadline(std::move(command1), std::move(command2)); scheduler.Schedule(group); scheduler.Schedule(command3); EXPECT_TRUE(scheduler.IsScheduled(command3)); EXPECT_FALSE(scheduler.IsScheduled(group)); } class TestableDeadlineCommand : public ParallelDeadlineGroup { static ParallelDeadlineGroup ToCommand( std::vector>&& commands) { std::vector> vec; std::unique_ptr deadline = std::move(commands[0]); for (unsigned int i = 1; i < commands.size(); i++) { vec.emplace_back(std::move(commands[i])); } return ParallelDeadlineGroup(std::move(deadline), std::move(vec)); } public: explicit TestableDeadlineCommand( std::vector>&& commands) : ParallelDeadlineGroup(ToCommand(std::move(commands))) {} }; INSTANTIATE_MULTI_COMMAND_COMPOSITION_TEST_SUITE(ParallelDeadlineGroupTest, TestableDeadlineCommand);