Source: lib/action/server_goal_handle.js

  1. // Copyright (c) 2020 Matt Richard. All rights reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. 'use strict';
  15. const rclnodejs = require('bindings')('rclnodejs');
  16. const ActionInterfaces = require('./interfaces.js');
  17. const Deferred = require('./deferred.js');
  18. const { GoalEvent } = require('./response.js');
  19. /**
  20. * @class - Goal handle for working with Action Servers.
  21. * @hideconstructor
  22. */
  23. class ServerGoalHandle {
  24. constructor(actionServer, goalInfo, goalRequest) {
  25. this._actionServer = actionServer;
  26. this._goalInfo = goalInfo;
  27. this._goalRequest = goalRequest;
  28. this._cancelRequested = false;
  29. this._deferred = new Deferred();
  30. this._handle = rclnodejs.actionAcceptNewGoal(
  31. this._actionServer.handle,
  32. this._goalInfo.serialize()
  33. );
  34. }
  35. /**
  36. * Gets the goal request.
  37. */
  38. get request() {
  39. return this._goalRequest;
  40. }
  41. /**
  42. * Gets the goal Id.
  43. */
  44. get goalId() {
  45. return this._goalInfo.goal_id;
  46. }
  47. /**
  48. * Gets if the goal handle is active.
  49. */
  50. get isActive() {
  51. if (this._destroyed) {
  52. return false;
  53. }
  54. return rclnodejs.actionGoalHandleIsActive(this._handle);
  55. }
  56. /**
  57. * Gets if cancellation was requested.
  58. */
  59. get isCancelRequested() {
  60. return this.status === ActionInterfaces.GoalStatus.STATUS_CANCELING;
  61. }
  62. /**
  63. * Gets the status of the goal.
  64. */
  65. get status() {
  66. if (this._destroyed) {
  67. return ActionInterfaces.GoalStatus.STATUS_UNKNOWN;
  68. }
  69. return rclnodejs.actionGoalHandleGetStatus(this._handle);
  70. }
  71. /**
  72. * Updates the goal handle with the execute status and begins exection.
  73. * @param {function} callback - An optional callback to use instead of the one provided to the action server.
  74. * @returns {undefined}
  75. */
  76. execute(callback) {
  77. if (!this.isCancelRequested) {
  78. this._updateState(GoalEvent.EXECUTE);
  79. }
  80. this._actionServer.notifyExecute(this, callback);
  81. }
  82. /**
  83. * Sends feedback back to the client.
  84. * @param {object} feedback - The feedback to send back.
  85. * @returns {undefined}
  86. */
  87. publishFeedback(feedback) {
  88. // Ignore for already destroyed goal handles
  89. if (this._destroyed) {
  90. return;
  91. }
  92. let feedbackMessage =
  93. new this._actionServer.typeClass.impl.FeedbackMessage();
  94. feedbackMessage['goal_id'] = this.goalId;
  95. feedbackMessage.feedback = feedback;
  96. rclnodejs.actionPublishFeedback(
  97. this._actionServer.handle,
  98. feedbackMessage.serialize()
  99. );
  100. }
  101. /**
  102. * Updates the goal handle with the succeed status.
  103. * @returns {undefined}
  104. */
  105. succeed() {
  106. this._updateState(GoalEvent.SUCCEED);
  107. }
  108. /**
  109. * Updates the goal handle with the abort status.
  110. * @returns {undefined}
  111. */
  112. abort() {
  113. this._updateState(GoalEvent.ABORT);
  114. }
  115. /**
  116. * Updates the goal handle with the canceled status.
  117. * @returns {undefined}
  118. */
  119. canceled() {
  120. this._updateState(GoalEvent.CANCELED);
  121. }
  122. /**
  123. * Marks the goal handle as destroyed. Any further updates to the handle will be ignored.
  124. * @ignore
  125. * @returns {undefined}
  126. */
  127. destroy() {
  128. if (this._destroyed) {
  129. return;
  130. }
  131. this._destroyed = true;
  132. }
  133. _updateState(event) {
  134. // Ignore updates for already destroyed goal handles
  135. if (this._destroyed) {
  136. return;
  137. }
  138. // Update state
  139. rclnodejs.actionUpdateGoalState(this._handle, event);
  140. // Publish state change
  141. rclnodejs.actionPublishStatus(this._actionServer.handle);
  142. // If it's a terminal state, then also notify the action server
  143. if (!rclnodejs.actionGoalHandleIsActive(this._handle)) {
  144. rclnodejs.actionNotifyGoalDone(this._actionServer.handle);
  145. }
  146. }
  147. }
  148. module.exports = ServerGoalHandle;