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 = new this._actionServer.typeClass.impl.FeedbackMessage();
  93. feedbackMessage['goal_id'] = this.goalId;
  94. feedbackMessage.feedback = feedback;
  95. rclnodejs.actionPublishFeedback(
  96. this._actionServer.handle,
  97. feedbackMessage.serialize()
  98. );
  99. }
  100. /**
  101. * Updates the goal handle with the succeed status.
  102. * @returns {undefined}
  103. */
  104. succeed() {
  105. this._updateState(GoalEvent.SUCCEED);
  106. }
  107. /**
  108. * Updates the goal handle with the abort status.
  109. * @returns {undefined}
  110. */
  111. abort() {
  112. this._updateState(GoalEvent.ABORT);
  113. }
  114. /**
  115. * Updates the goal handle with the canceled status.
  116. * @returns {undefined}
  117. */
  118. canceled() {
  119. this._updateState(GoalEvent.CANCELED);
  120. }
  121. /**
  122. * Marks the goal handle as destroyed. Any further updates to the handle will be ignored.
  123. * @ignore
  124. * @returns {undefined}
  125. */
  126. destroy() {
  127. if (this._destroyed) {
  128. return;
  129. }
  130. this._destroyed = true;
  131. }
  132. _updateState(event) {
  133. // Ignore updates for already destroyed goal handles
  134. if (this._destroyed) {
  135. return;
  136. }
  137. // Update state
  138. rclnodejs.actionUpdateGoalState(this._handle, event);
  139. // Publish state change
  140. rclnodejs.actionPublishStatus(this._actionServer.handle);
  141. // If it's a terminal state, then also notify the action server
  142. if (!rclnodejs.actionGoalHandleIsActive(this._handle)) {
  143. rclnodejs.actionNotifyGoalDone(this._actionServer.handle);
  144. }
  145. }
  146. }
  147. module.exports = ServerGoalHandle;