Skip to content

Commit d36121a

Browse files
committed
PrimaryClient: Add methods to unlock protective stop and stop the program
1 parent ce6aec2 commit d36121a

File tree

4 files changed

+198
-3
lines changed

4 files changed

+198
-3
lines changed

include/ur_client_library/exceptions.h

+18
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@
3232
#include <chrono>
3333
#include <stdexcept>
3434
#include <sstream>
35+
#include <string>
3536
#include "ur/version_information.h"
3637

3738
#ifdef _WIN32
@@ -203,5 +204,22 @@ class MissingArgument : public UrException
203204
return text_.c_str();
204205
}
205206
};
207+
208+
class InvalidStateForCommand : public UrException
209+
{
210+
private:
211+
std::string text_;
212+
213+
public:
214+
explicit InvalidStateForCommand() = delete;
215+
explicit InvalidStateForCommand(std::string text) : std::runtime_error(text)
216+
{
217+
text_ = text;
218+
}
219+
virtual const char* what() const noexcept override
220+
{
221+
return text_.c_str();
222+
}
223+
};
206224
} // namespace urcl
207225
#endif // ifndef UR_CLIENT_LIBRARY_EXCEPTIONS_H_INCLUDED

include/ur_client_library/primary/primary_client.h

+61
Original file line numberDiff line numberDiff line change
@@ -128,6 +128,39 @@ class PrimaryClient
128128
void commandBrakeRelease(const bool validate = true,
129129
const std::chrono::milliseconds timeout = std::chrono::seconds(30));
130130

131+
/*!
132+
* \brief Commands the robot to unlock the protective stop
133+
*
134+
* \param validate If true, the function will block until the protective stop is released or the
135+
* timeout passed by.
136+
* \param timeout The maximum time to wait for the robot to confirm it is no longer protective
137+
* stopped.
138+
*
139+
* \throws urcl::UrException if the command cannot be sent to the robot.
140+
* \throws urcl::TimeoutException if the robot doesn't unlock the protective stop within the
141+
* given timeout.
142+
*/
143+
void commandUnlockProtectiveStop(const bool validate = true,
144+
const std::chrono::milliseconds timeout = std::chrono::milliseconds(5000));
145+
146+
/*!
147+
* /brief Stop execution of a running or paused program
148+
*
149+
* Blocks until the robot has come to a standstill (while following
150+
* the program path) and stopped program execution.
151+
*
152+
* \param already_stopped_ok If true, the function will return true if the robot is already
153+
* stopped.
154+
* \param validate If true, the function will block until the robot has stopped or the timeout
155+
* passed by.
156+
* \param timeout The maximum time to wait for the robot to stop the program.
157+
*
158+
* \throws urcl::UrException if the command cannot be sent to the robot.
159+
* \throws urcl::TimeoutException if the robot doesn't stop the program within the given timeout.
160+
*/
161+
void commandStop(const bool already_stopped_ok = false, const bool validate = true,
162+
const std::chrono::milliseconds timeout = std::chrono::seconds(2));
163+
131164
/*!
132165
* \brief Get the latest robot mode.
133166
*
@@ -144,6 +177,34 @@ class PrimaryClient
144177
return static_cast<RobotMode>(consumer_->getRobotModeData()->robot_mode_);
145178
}
146179

180+
/*!
181+
* \brief Get the latest robot mode data.
182+
*
183+
* The robot's mode data will be updated in the background. This will always show the latest received
184+
* state independent of the time that has passed since receiving it. The return value of this
185+
* will be a nullptr if no data has been received yet.
186+
*/
187+
std::shared_ptr<RobotModeData> getRobotModeData()
188+
{
189+
return consumer_->getRobotModeData();
190+
}
191+
192+
/*!
193+
* \brief Query if the robot is protective stopped.
194+
*
195+
* The robot's protective_stop state will be updated in the background. This will always show the latest received
196+
* state independent of the time that has passed since receiving it.
197+
*/
198+
bool isRobotProtectiveStopped()
199+
{
200+
std::shared_ptr<RobotModeData> robot_mode_data = consumer_->getRobotModeData();
201+
if (robot_mode_data == nullptr)
202+
{
203+
return false;
204+
}
205+
return robot_mode_data->is_protective_stopped_;
206+
}
207+
147208
private:
148209
/*!
149210
* \brief Reconnects the primary stream used to send program to the robot.

src/primary/primary_client.cpp

+65-2
Original file line numberDiff line numberDiff line change
@@ -193,7 +193,7 @@ void PrimaryClient::commandPowerOff(const bool validate, const std::chrono::mill
193193
{
194194
waitFor([this]() { return getRobotMode() == RobotMode::POWER_OFF; }, timeout);
195195
}
196-
catch (const std::exception&)
196+
catch (const TimeoutException&)
197197
{
198198
throw TimeoutException("Robot did not power off within the given timeout", timeout);
199199
}
@@ -212,12 +212,75 @@ void PrimaryClient::commandBrakeRelease(const bool validate, const std::chrono::
212212
{
213213
waitFor([this]() { return getRobotMode() == RobotMode::RUNNING; }, timeout);
214214
}
215-
catch (const std::exception&)
215+
catch (const TimeoutException&)
216216
{
217217
throw TimeoutException("Robot did not release the brakes within the given timeout", timeout);
218218
}
219219
}
220220
}
221221

222+
void PrimaryClient::commandUnlockProtectiveStop(const bool validate, const std::chrono::milliseconds timeout)
223+
{
224+
if (!sendScript("set unlock protective stop"))
225+
{
226+
throw UrException("Failed to send unlock protective stop command to robot");
227+
}
228+
if (validate)
229+
{
230+
try
231+
{
232+
waitFor([this]() { return consumer_->getRobotModeData()->is_protective_stopped_ == false; }, timeout);
233+
}
234+
catch (const TimeoutException&)
235+
{
236+
throw TimeoutException("Robot did not unlock the protective stop within the given timeout", timeout);
237+
}
238+
}
239+
}
240+
241+
void PrimaryClient::commandStop(const bool already_stopped_ok, const bool validate,
242+
const std::chrono::milliseconds timeout)
243+
{
244+
std::shared_ptr<RobotModeData> robot_mode_data = consumer_->getRobotModeData();
245+
if (robot_mode_data == nullptr)
246+
{
247+
throw UrException("Stopping a program while robot state is unknown. This should not happen");
248+
}
249+
250+
if (!(robot_mode_data->is_program_running_ || robot_mode_data->is_program_paused_))
251+
{
252+
if (already_stopped_ok)
253+
{
254+
URCL_LOG_DEBUG("Program halt requested, but program is already stopped, skipping.");
255+
return;
256+
}
257+
else
258+
{
259+
throw InvalidStateForCommand("Cannot halt program execution, as no program is running or paused on the robot.");
260+
}
261+
}
262+
263+
if (!sendScript("stop program"))
264+
{
265+
throw UrException("Failed to send the command `stop program` to robot");
266+
}
267+
if (validate)
268+
{
269+
try
270+
{
271+
waitFor(
272+
[this]() {
273+
return !consumer_->getRobotModeData()->is_program_running_ &&
274+
!consumer_->getRobotModeData()->is_program_paused_;
275+
},
276+
timeout);
277+
}
278+
catch (const TimeoutException&)
279+
{
280+
throw TimeoutException("Robot did not stop the program within the given timeout", timeout);
281+
}
282+
}
283+
}
284+
222285
} // namespace primary_interface
223286
} // namespace urcl

tests/test_primary_client.cpp

+54-1
Original file line numberDiff line numberDiff line change
@@ -110,12 +110,65 @@ TEST_F(PrimaryClientTest, test_power_cycle_commands)
110110
EXPECT_NO_THROW(waitFor([this]() { return client_->getRobotMode() == RobotMode::RUNNING; }, timeout));
111111
}
112112

113+
TEST_F(PrimaryClientTest, test_unlock_protective_stop)
114+
{
115+
EXPECT_NO_THROW(client_->start());
116+
EXPECT_NO_THROW(client_->commandBrakeRelease(true, std::chrono::milliseconds(20000)));
117+
client_->sendScript("protective_stop()");
118+
EXPECT_NO_THROW(waitFor([this]() { return client_->isRobotProtectiveStopped(); }, std::chrono::milliseconds(1000)));
119+
// This will not happen immediately
120+
EXPECT_THROW(client_->commandUnlockProtectiveStop(true, std::chrono::milliseconds(1)), TimeoutException);
121+
122+
// It is not allowed to unlock the protective stop immediately
123+
std::this_thread::sleep_for(std::chrono::seconds(5));
124+
EXPECT_NO_THROW(client_->commandUnlockProtectiveStop());
125+
}
126+
113127
TEST_F(PrimaryClientTest, test_uninitialized_primary_client)
114128
{
115129
// The client is not started yet, so the robot mode should be UNKNOWN
116130
ASSERT_EQ(client_->getRobotMode(), RobotMode::UNKNOWN);
117131
}
118132

133+
TEST_F(PrimaryClientTest, test_stop_command)
134+
{
135+
// Without started communication the latest robot mode data is a nullptr
136+
EXPECT_THROW(client_->commandStop(), UrException);
137+
138+
EXPECT_NO_THROW(client_->start());
139+
EXPECT_NO_THROW(client_->commandPowerOff());
140+
EXPECT_NO_THROW(client_->commandBrakeRelease());
141+
142+
const std::string script_code = "def test_fun():\n"
143+
" while True:\n"
144+
" textmsg(\"still running\")\n"
145+
" sleep(1.0)\n"
146+
" sync()\n"
147+
" end\n"
148+
"end";
149+
150+
EXPECT_TRUE(client_->sendScript(script_code));
151+
waitFor([this]() { return client_->getRobotModeData()->is_program_running_; }, std::chrono::seconds(5));
152+
153+
EXPECT_NO_THROW(client_->commandStop());
154+
EXPECT_FALSE(client_->getRobotModeData()->is_program_running_);
155+
156+
// Without a program running it should throw an exception
157+
EXPECT_THROW(client_->commandStop(), InvalidStateForCommand);
158+
159+
// We can specifically allow to be stopped already
160+
EXPECT_NO_THROW(client_->commandStop(true));
161+
162+
EXPECT_TRUE(client_->sendScript(script_code));
163+
waitFor([this]() { return client_->getRobotModeData()->is_program_running_; }, std::chrono::seconds(5));
164+
EXPECT_THROW(client_->commandStop(true, true, std::chrono::milliseconds(1)), TimeoutException);
165+
EXPECT_NO_THROW(waitFor(
166+
[this]() {
167+
return !client_->getRobotModeData()->is_program_running_ && !client_->getRobotModeData()->is_program_paused_;
168+
},
169+
std::chrono::seconds(5)));
170+
}
171+
119172
int main(int argc, char* argv[])
120173
{
121174
::testing::InitGoogleTest(&argc, argv);
@@ -129,4 +182,4 @@ int main(int argc, char* argv[])
129182
}
130183

131184
return RUN_ALL_TESTS();
132-
}
185+
}

0 commit comments

Comments
 (0)