Skip to content

Commit

Permalink
Add TransportAmr capability and RMF workcell
Browse files Browse the repository at this point in the history
Signed-off-by: Luca Della Vedova <[email protected]>
  • Loading branch information
luca-della-vedova committed Dec 11, 2024
1 parent f5cd789 commit 89203ff
Show file tree
Hide file tree
Showing 8 changed files with 791 additions and 4 deletions.
9 changes: 9 additions & 0 deletions nexus_capabilities/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,11 @@ set(dep_pkgs
nexus_common
nexus_endpoints
nexus_orchestrator_msgs
nlohmann_json
rclcpp
rclcpp_lifecycle
rmf_dispenser_msgs
rmf_task_msgs
yaml-cpp
)
foreach(pkg ${dep_pkgs})
Expand Down Expand Up @@ -126,6 +129,8 @@ add_library(nexus_builtin_capabilities SHARED
src/capabilities/gripper_control.cpp
src/capabilities/plan_motion_capability.cpp
src/capabilities/plan_motion.cpp
src/capabilities/transport_amr.cpp
src/capabilities/transport_amr_capability.cpp
)

target_compile_options(nexus_builtin_capabilities PUBLIC INTERFACE cxx_std_17)
Expand All @@ -135,9 +140,13 @@ target_link_libraries(nexus_builtin_capabilities
${PROJECT_NAME}
nexus_common::nexus_common
nexus_endpoints::nexus_endpoints
nlohmann_json::nlohmann_json
pluginlib::pluginlib
tf2::tf2
tf2_ros::tf2_ros
${rmf_dispenser_msgs_TARGETS}
${rmf_task_msgs_TARGETS}
yaml-cpp
)

install(
Expand Down
4 changes: 4 additions & 0 deletions nexus_capabilities/src/capabilities/plugins.xml
Original file line number Diff line number Diff line change
Expand Up @@ -18,4 +18,8 @@
<class type="nexus::capabilities::PlanMotionCapability" base_class_type="nexus::Capability">
<description>PlanMotion capability.</description>
</class>

<class type="nexus::capabilities::TransportAmrCapability" base_class_type="nexus::Capability">
<description>TransportAmr capability.</description>
</class>
</library>
336 changes: 336 additions & 0 deletions nexus_capabilities/src/capabilities/transport_amr.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,336 @@
/*
* Copyright (C) 2023 Johnson & Johnson
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*/

#include "transport_amr.hpp"

#include <nlohmann/json.hpp>
#include <yaml-cpp/yaml.h>

namespace nexus::capabilities {

BT::NodeStatus DispatchRmfRequest::onStart()
{
const auto destinations = this->getInput<std::deque<AmrDestination>>("destinations");
if (!destinations)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [destinations] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}

const auto transient_qos =
rclcpp::SystemDefaultsQoS().transient_local().keep_last(10).reliable();
this->_api_request_pub = this->_node->create_publisher<ApiRequest>("/task_api_requests", transient_qos);
// TODO(luca) publish request here with unique UUID
this->_api_response_sub = this->_node->create_subscription<ApiResponse>("/task_api_responses", transient_qos,
[&](ApiResponse::UniquePtr msg)
{
this->api_response_cb(*msg);
});
this->submit_itinerary(*destinations);
return BT::NodeStatus::RUNNING;
}

void DispatchRmfRequest::submit_itinerary(const std::deque<AmrDestination>& destinations)
{
nlohmann::json j;
j["type"] = "dispatch_task_request";
nlohmann::json r;
r["unix_millis_request_time"] = 0;
r["unix_millis_earliest_start_time"] = 0;
r["requester"] = this->_node->get_name();
r["category"] = "compose";
nlohmann::json d;
d["category"] = "multi_delivery";
d["phases"] = nlohmann::json::array();
nlohmann::json activity;
activity["category"] = "sequence";
activity["description"]["activities"] = nlohmann::json::array();
for (const auto& destination : destinations)
{
nlohmann::json a;
a["category"] = destination.action;
nlohmann::json p;
// TODO(luca) exception safety for wrong types? Checking for pickup only since we don't do ingestors yet?
p["place"] = destination.workcell;
// TODO(luca) We should assign a handler that is related to the workcell.
// For now the assumption is that a location has only one handler
p["handler"] = destination.workcell;
p["payload"] = nlohmann::json::array();
a["description"] = p;
activity["description"]["activities"].push_back(a);
}
nlohmann::json act_obj;
act_obj["activity"] = activity;
d["phases"].push_back(act_obj);
r["description"] = d;
j["request"] = r;
ApiRequest msg;
msg.json_msg = j.dump();
// Time since epoch as a unique id
auto now = std::chrono::steady_clock::now().time_since_epoch();
msg.request_id = std::chrono::duration_cast<std::chrono::nanoseconds>(now).count();
this->requested_id = msg.request_id;
this->_api_request_pub->publish(msg);
}

void DispatchRmfRequest::api_response_cb(const ApiResponse& msg)
{
// Receive response, populate hashmaps
if (msg.type != msg.TYPE_RESPONDING)
{
return;
}
if (msg.request_id != this->requested_id)
{
return;
}
auto j = nlohmann::json::parse(msg.json_msg, nullptr, false);
if (j.is_discarded())
{
RCLCPP_ERROR(this->_node->get_logger(), "Invalid json in api response");
return;
}
// TODO(luca) exception safety for missing fields, return FAILURE if
// submission failed
if (j["success"] == false)
{
RCLCPP_ERROR(this->_node->get_logger(), "Task submission failed");
return;
}
// Task cancellations don't have a state field
if (!j.contains("state"))
return;
this->rmf_task_id = j["state"]["booking"]["id"];
}

BT::NodeStatus DispatchRmfRequest::onRunning()
{
if (rmf_task_id.has_value())
{
this->setOutput("rmf_task_id", *rmf_task_id);
return BT::NodeStatus::SUCCESS;
}
return BT::NodeStatus::RUNNING;
}

BT::NodeStatus ExtractDestinations::tick()
{
const auto ctx = this->_ctx_mgr->current_context();
const auto& task = ctx->task;
std::deque<AmrDestination> destinations;
for (const auto& node : task.data)
{
if (node["type"] && node["destination"])
{
auto type = node["type"].as<std::string>();
auto destination = node["destination"].as<std::string>();
destinations.push_back(AmrDestination {type, destination});
}
else
{
RCLCPP_ERROR(
this->_node->get_logger(),
"Order element did not contain \"type\" and \"destination\" fields");
return BT::NodeStatus::FAILURE;
}
}
this->setOutput("destinations", destinations);
return BT::NodeStatus::SUCCESS;
}

BT::NodeStatus UnpackDestinationData::tick()
{
const auto destination = this->getInput<AmrDestination>("destination");
if (!destination)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [destination] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}
this->setOutput("workcell", destination->workcell);
this->setOutput("type", destination->action);
return BT::NodeStatus::SUCCESS;
}

BT::NodeStatus SignalAmr::tick()
{
const auto workcell = this->getInput<std::string>("workcell");
if (!workcell)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [workcell] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}
const auto rmf_task_id = this->getInput<std::string>("rmf_task_id");
if (!rmf_task_id)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [rmf_task_id] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}
this->_dispenser_result_pub = this->_node->create_publisher<DispenserResult>("/dispenser_results", 10);

DispenserResult msg;
msg.request_guid = *rmf_task_id;
msg.source_guid = *workcell;
msg.status = DispenserResult::SUCCESS;
this->_dispenser_result_pub->publish(msg);
return BT::NodeStatus::SUCCESS;
}

BT::NodeStatus LoopDestinations::tick()
{
if (!this->_initialized)
{
auto queue = this->getInput<std::deque<AmrDestination>>("queue");
if (!queue)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [queue] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}
this->_queue = *queue;
this->_initialized = true;
}

if (this->_queue.size() == 0)
{
return BT::NodeStatus::SUCCESS;
}

this->setStatus(BT::NodeStatus::RUNNING);

while (this->_queue.size() > 0)
{
RCLCPP_INFO(
this->_node->get_logger(), "%s: [queue] has %u items",
this->name().c_str(), this->_queue.size());
auto current_value = this->_queue.front();
this->setOutput("value", current_value);

this->setStatus(BT::NodeStatus::RUNNING);
auto child_state = this->child_node_->executeTick();
switch (child_state)
{
case BT::NodeStatus::SUCCESS:
this->haltChild();
this->_queue.pop_front();
break;
case BT::NodeStatus::RUNNING:
return BT::NodeStatus::RUNNING;
case BT::NodeStatus::FAILURE:
this->haltChild();
return BT::NodeStatus::FAILURE;
case BT::NodeStatus::IDLE:
throw BT::LogicError("A child node must never return IDLE");
}
}

return BT::NodeStatus::SUCCESS;
}

BT::NodeStatus WaitForAmr::onStart()
{
const auto workcell = this->getInput<std::string>("workcell");
if (!workcell)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [workcell] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}
const auto rmf_task_id = this->getInput<std::string>("rmf_task_id");
if (!rmf_task_id)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: [rmf_task_id] port is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}

this->_workcell = *workcell;
this->_rmf_task_id = *rmf_task_id;

this->_dispenser_request_sub = this->_node->create_subscription<DispenserRequest>("/dispenser_requests", 10,
[&](DispenserRequest::UniquePtr msg)
{
this->dispenser_request_cb(*msg);
});
return BT::NodeStatus::RUNNING;
}

void WaitForAmr::dispenser_request_cb(const DispenserRequest& msg)
{
if (msg.request_guid == this->_rmf_task_id &&
msg.target_guid == this->_workcell)
{
this->_amr_ready = true;
}
}

BT::NodeStatus WaitForAmr::onRunning()
{
RCLCPP_INFO(
this->_node->get_logger(), "%s: Waiting for AMR to reach %s",
this->name().c_str(), this->_workcell.c_str());
if (this->_amr_ready)
{
this->_amr_ready = false;
return BT::NodeStatus::SUCCESS;
}
return BT::NodeStatus::RUNNING;
}

BT::NodeStatus SendSignal::tick()
{
const auto ctx = this->_ctx_mgr->current_context();
auto signal = this->getInput<std::string>("signal");
if (!signal)
{
RCLCPP_ERROR(
this->_node->get_logger(), "%s: port [signal] is required",
this->name().c_str());
return BT::NodeStatus::FAILURE;
}

this->_signal_client = std::make_unique<common::SyncServiceClient<endpoints::
SignalWorkcellService::ServiceType>>(
this->_node, endpoints::SignalWorkcellService::service_name("system_orchestrator"));

auto req =
std::make_shared<endpoints::SignalWorkcellService::ServiceType::Request>();
req->task_id = ctx->task.id;
req->signal = *signal;
auto resp = this->_signal_client->send_request(req);
if (!resp->success)
{
// TODO(luca) implement a queueing mechanism if the request fails?
RCLCPP_WARN(
this->_node->get_logger(),
"%s: System orchestrator is not able to accept [%s] signal now. Skipping signaling, error: [%s]",
this->_node->get_name(), signal->c_str(), resp->message.c_str());
}
return BT::NodeStatus::SUCCESS;
}

}
Loading

0 comments on commit 89203ff

Please sign in to comment.