2019-08-25 23:55:59 -04:00
|
|
|
/*----------------------------------------------------------------------------*/
|
2020-02-08 10:26:06 -08:00
|
|
|
/* Copyright (c) 2019-2020 FIRST. All Rights Reserved. */
|
2019-08-25 23:55:59 -04:00
|
|
|
/* Open Source Software - may be modified and shared by FRC teams. The code */
|
|
|
|
|
/* must be accompanied by the FIRST BSD license file in the root directory of */
|
|
|
|
|
/* the project. */
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
|
|
#include "CommandTestBase.h"
|
|
|
|
|
#include "frc2/command/InstantCommand.h"
|
|
|
|
|
#include "frc2/command/ParallelRaceGroup.h"
|
2019-10-18 07:55:14 -07:00
|
|
|
#include "frc2/command/SequentialCommandGroup.h"
|
2019-08-25 23:55:59 -04:00
|
|
|
#include "frc2/command/WaitUntilCommand.h"
|
|
|
|
|
|
|
|
|
|
using namespace frc2;
|
|
|
|
|
class ParallelRaceGroupTest : public CommandTestBase {};
|
|
|
|
|
|
|
|
|
|
TEST_F(ParallelRaceGroupTest, ParallelRaceScheduleTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<MockCommand> command1Holder = std::make_unique<MockCommand>();
|
|
|
|
|
std::unique_ptr<MockCommand> command2Holder = std::make_unique<MockCommand>();
|
|
|
|
|
std::unique_ptr<MockCommand> command3Holder = std::make_unique<MockCommand>();
|
|
|
|
|
|
|
|
|
|
MockCommand* command1 = command1Holder.get();
|
|
|
|
|
MockCommand* command2 = command2Holder.get();
|
|
|
|
|
MockCommand* command3 = command3Holder.get();
|
|
|
|
|
|
|
|
|
|
ParallelRaceGroup group{tcb::make_vector<std::unique_ptr<Command>>(
|
|
|
|
|
std::move(command1Holder), std::move(command2Holder),
|
|
|
|
|
std::move(command3Holder))};
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(*command1, Initialize());
|
|
|
|
|
EXPECT_CALL(*command1, Execute()).Times(2);
|
|
|
|
|
EXPECT_CALL(*command1, End(true));
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(*command2, Initialize());
|
|
|
|
|
EXPECT_CALL(*command2, Execute()).Times(2);
|
|
|
|
|
EXPECT_CALL(*command2, End(false));
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(*command3, Initialize());
|
|
|
|
|
EXPECT_CALL(*command3, Execute()).Times(2);
|
|
|
|
|
EXPECT_CALL(*command3, End(true));
|
|
|
|
|
|
|
|
|
|
scheduler.Schedule(&group);
|
|
|
|
|
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
command2->SetFinished(true);
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
|
|
|
|
|
EXPECT_FALSE(scheduler.IsScheduled(&group));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(ParallelRaceGroupTest, ParallelRaceInterruptTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<MockCommand> command1Holder = std::make_unique<MockCommand>();
|
|
|
|
|
std::unique_ptr<MockCommand> command2Holder = std::make_unique<MockCommand>();
|
|
|
|
|
std::unique_ptr<MockCommand> command3Holder = std::make_unique<MockCommand>();
|
|
|
|
|
|
|
|
|
|
MockCommand* command1 = command1Holder.get();
|
|
|
|
|
MockCommand* command2 = command2Holder.get();
|
|
|
|
|
MockCommand* command3 = command3Holder.get();
|
|
|
|
|
|
|
|
|
|
ParallelRaceGroup group{tcb::make_vector<std::unique_ptr<Command>>(
|
|
|
|
|
std::move(command1Holder), 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(ParallelRaceGroupTest, ParallelRaceNotScheduledCancelTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
ParallelRaceGroup group{InstantCommand(), InstantCommand()};
|
|
|
|
|
|
|
|
|
|
EXPECT_NO_FATAL_FAILURE(scheduler.Cancel(&group));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(ParallelRaceGroupTest, ParallelRaceCopyTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
bool finished = false;
|
|
|
|
|
|
|
|
|
|
WaitUntilCommand command([&finished] { return finished; });
|
|
|
|
|
|
|
|
|
|
ParallelRaceGroup group(command);
|
|
|
|
|
scheduler.Schedule(&group);
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
EXPECT_TRUE(scheduler.IsScheduled(&group));
|
|
|
|
|
finished = true;
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
EXPECT_FALSE(scheduler.IsScheduled(&group));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(ParallelRaceGroupTest, RaceGroupRequirementTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
TestSubsystem requirement1;
|
|
|
|
|
TestSubsystem requirement2;
|
|
|
|
|
TestSubsystem requirement3;
|
|
|
|
|
TestSubsystem requirement4;
|
|
|
|
|
|
|
|
|
|
InstantCommand command1([] {}, {&requirement1, &requirement2});
|
|
|
|
|
InstantCommand command2([] {}, {&requirement3});
|
|
|
|
|
InstantCommand command3([] {}, {&requirement3, &requirement4});
|
|
|
|
|
|
|
|
|
|
ParallelRaceGroup group(std::move(command1), std::move(command2));
|
|
|
|
|
|
|
|
|
|
scheduler.Schedule(&group);
|
|
|
|
|
scheduler.Schedule(&command3);
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(scheduler.IsScheduled(&command3));
|
|
|
|
|
EXPECT_FALSE(scheduler.IsScheduled(&group));
|
|
|
|
|
}
|
2019-10-18 07:55:14 -07:00
|
|
|
|
|
|
|
|
TEST_F(ParallelRaceGroupTest, ParallelRaceOnlyCallsEndOnceTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
bool finished1 = false;
|
|
|
|
|
bool finished2 = false;
|
|
|
|
|
bool finished3 = false;
|
|
|
|
|
|
|
|
|
|
WaitUntilCommand command1([&finished1] { return finished1; });
|
|
|
|
|
WaitUntilCommand command2([&finished2] { return finished2; });
|
|
|
|
|
WaitUntilCommand command3([&finished3] { return finished3; });
|
|
|
|
|
|
|
|
|
|
SequentialCommandGroup group1(command1, command2);
|
|
|
|
|
ParallelRaceGroup group2(std::move(group1), command3);
|
|
|
|
|
|
|
|
|
|
scheduler.Schedule(&group2);
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
EXPECT_TRUE(scheduler.IsScheduled(&group2));
|
|
|
|
|
finished1 = true;
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
finished2 = true;
|
|
|
|
|
EXPECT_NO_FATAL_FAILURE(scheduler.Run());
|
|
|
|
|
EXPECT_FALSE(scheduler.IsScheduled(&group2));
|
|
|
|
|
}
|
2020-02-08 10:26:06 -08:00
|
|
|
|
|
|
|
|
TEST_F(ParallelRaceGroupTest, ParallelRaceScheduleTwiceTest) {
|
|
|
|
|
CommandScheduler scheduler = GetScheduler();
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<MockCommand> command1Holder = std::make_unique<MockCommand>();
|
|
|
|
|
std::unique_ptr<MockCommand> command2Holder = std::make_unique<MockCommand>();
|
|
|
|
|
std::unique_ptr<MockCommand> command3Holder = std::make_unique<MockCommand>();
|
|
|
|
|
|
|
|
|
|
MockCommand* command1 = command1Holder.get();
|
|
|
|
|
MockCommand* command2 = command2Holder.get();
|
|
|
|
|
MockCommand* command3 = command3Holder.get();
|
|
|
|
|
|
|
|
|
|
ParallelRaceGroup group{tcb::make_vector<std::unique_ptr<Command>>(
|
|
|
|
|
std::move(command1Holder), std::move(command2Holder),
|
|
|
|
|
std::move(command3Holder))};
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(*command1, Initialize()).Times(2);
|
|
|
|
|
EXPECT_CALL(*command1, Execute()).Times(5);
|
|
|
|
|
EXPECT_CALL(*command1, End(true)).Times(2);
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(*command2, Initialize()).Times(2);
|
|
|
|
|
EXPECT_CALL(*command2, Execute()).Times(5);
|
|
|
|
|
EXPECT_CALL(*command2, End(false)).Times(2);
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(*command3, Initialize()).Times(2);
|
|
|
|
|
EXPECT_CALL(*command3, Execute()).Times(5);
|
|
|
|
|
EXPECT_CALL(*command3, End(true)).Times(2);
|
|
|
|
|
|
|
|
|
|
scheduler.Schedule(&group);
|
|
|
|
|
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
command2->SetFinished(true);
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
|
|
|
|
|
EXPECT_FALSE(scheduler.IsScheduled(&group));
|
|
|
|
|
|
|
|
|
|
command2->SetFinished(false);
|
|
|
|
|
|
|
|
|
|
scheduler.Schedule(&group);
|
|
|
|
|
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
EXPECT_TRUE(scheduler.IsScheduled(&group));
|
|
|
|
|
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
EXPECT_TRUE(scheduler.IsScheduled(&group));
|
|
|
|
|
|
|
|
|
|
command2->SetFinished(true);
|
|
|
|
|
scheduler.Run();
|
|
|
|
|
|
|
|
|
|
EXPECT_FALSE(scheduler.IsScheduled(&group));
|
|
|
|
|
}
|