Source: lib/time_source.js

  1. // Copyright (c) 2018 Intel Corporation. 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 { Clock, ROSClock } = require('./clock.js');
  17. const { ClockType } = Clock;
  18. const Node = require('./node.js');
  19. const { Parameter, ParameterType } = require('./parameter.js');
  20. const Time = require('./time.js');
  21. const USE_SIM_TIME_PARAM = 'use_sim_time';
  22. const CLOCK_TOPIC = 'clock';
  23. /**
  24. * @class - Class representing a TimeSource in ROS
  25. */
  26. class TimeSource {
  27. /**
  28. * Create a TimeSource.
  29. * @param {Node} node - The node to be attached.
  30. */
  31. constructor(node) {
  32. this._node = node;
  33. this._associatedClocks = [];
  34. this._clockSubscription = undefined;
  35. this._lastTimeSet = new Time(0, 0, ClockType.ROS_TIME);
  36. this._isRosTimeActive = false;
  37. if (this._node) {
  38. this.attachNode(this._node);
  39. }
  40. }
  41. get isRosTimeActive() {
  42. return this._isRosTimeActive;
  43. }
  44. set isRosTimeActive(enabled) {
  45. if (this.isRosTimeActive === enabled) return;
  46. this._isRosTimeActive = enabled;
  47. for (const clock in this._associatedClocks) {
  48. clock.isRosTimeActive = enabled;
  49. }
  50. if (enabled) {
  51. this._subscribeToClockTopic();
  52. } else if (this._node && this._clockSubscription) {
  53. this._node.destroySubscription(this._clockSubscription);
  54. this._node._clockSubscription = null;
  55. }
  56. }
  57. /**
  58. * Return status that whether the ROS time is active.
  59. * @name TimeSource#get:isRosTimeActive
  60. * @function
  61. * @return {boolean} Return true if the time is active, otherwise return false.
  62. */
  63. get isRosTimeActive() {
  64. return this._isRosTimeActive;
  65. }
  66. /**
  67. * Set the status of time.
  68. * @param {boolean} enabled - Set the ROS time to be active if enabled is true.
  69. * @name TimeSource#set:isRosTimeActive
  70. * @function
  71. * @return {undefined}
  72. */
  73. set isRosTimeActive(enabled) {
  74. if (this._isRosTimeActive === enabled) return;
  75. this._isRosTimeActive = enabled;
  76. this._associatedClocks.forEach(clock => {
  77. clock.isRosTimeActive = enabled;
  78. });
  79. if (enabled) {
  80. this._subscribeToClockTopic();
  81. }
  82. }
  83. /**
  84. * Attach the clock to a Node object.
  85. * @param {Node} node - The node to be attached.
  86. * @return {undefined}
  87. */
  88. attachNode(node) {
  89. if (!node instanceof rclnodejs.ShadowNode) {
  90. throw new TypeError('Invalid argument, must be type of Node');
  91. }
  92. if (this._node) {
  93. this.detachNode();
  94. }
  95. this._node = node;
  96. if (!node.hasParameter(USE_SIM_TIME_PARAM)) {
  97. node.declareParameter(
  98. new Parameter(USE_SIM_TIME_PARAM, ParameterType.PARAMETER_BOOL, false)
  99. );
  100. }
  101. const useSimTimeParam = node.getParameter(USE_SIM_TIME_PARAM);
  102. if (useSimTimeParam.type !== ParameterType.PARAMETER_NOT_SET) {
  103. if (useSimTimeParam.type === ParameterType.PARAMETER_BOOL) {
  104. this._isRosTimeActive = useSimTimeParam.value;
  105. } else {
  106. node
  107. .getLogger()
  108. .error(
  109. `Invalid type for parameter ${USE_SIM_TIME_PARAM} ${useSimTimeParam.type} should be bool`
  110. );
  111. }
  112. } else {
  113. node
  114. .getLogger()
  115. .debug(
  116. `${USE_SIM_TIME_PARAM}' parameter not set, using wall time by default`
  117. );
  118. }
  119. if (this.isRosTimeActive) {
  120. this._subscribeToClockTopic();
  121. }
  122. node.addOnSetParametersCallback(this.onParameterEvent.bind(this));
  123. }
  124. /**
  125. * Detach the node which the clock have attached.
  126. * @return {undefined}
  127. */
  128. detachNode() {
  129. if (this._clockSubscription) {
  130. if (!this._node) {
  131. throw new Error(
  132. 'Unable to destroy previously created clock subscription'
  133. );
  134. }
  135. this._node.destroySubscription(this._clockSubscription);
  136. }
  137. this._clockSubscription = undefined;
  138. this._node = undefined;
  139. }
  140. /**
  141. * Attach the clock to a TimeSource object.
  142. * @param {Clock} clock - The node to be attached.
  143. * @return {undefined}
  144. */
  145. attachClock(clock) {
  146. if (!(clock instanceof ROSClock)) {
  147. throw new TypeError('Only clocks with type ROS_TIME can be attached.');
  148. }
  149. clock.rosTimeOverride = this._lastTimeSet;
  150. clock.isRosTimeActive = this._isRosTimeActive;
  151. this._associatedClocks.push(clock);
  152. }
  153. _clockCallback(msg) {
  154. this._lastTimeSet = Time.fromMsg(msg);
  155. this._associatedClocks.forEach(clock => {
  156. clock.rosTimeOverride = this._lastTimeSet;
  157. });
  158. }
  159. _subscribeToClockTopic() {
  160. if (!this._clockSubscription && this._node) {
  161. this._clockSubscription = this._node.createSubscription(
  162. 'builtin_interfaces/msg/Time',
  163. CLOCK_TOPIC,
  164. this._clockCallback.bind(this)
  165. );
  166. }
  167. }
  168. onParameterEvent(parameters = []) {
  169. for (const parameter of parameters) {
  170. if (parameter.name === USE_SIM_TIME_PARAM) {
  171. if (parameter.type === ParameterType.PARAMETER_BOOL) {
  172. this.isRosTimeActive = parameter.value;
  173. } else if (this._node) {
  174. this._node
  175. .getLogger()
  176. .error(
  177. `${USE_SIM_TIME_PARAM} parameter set to something besides a bool`
  178. );
  179. }
  180. break;
  181. }
  182. }
  183. return {
  184. successful: true,
  185. reason: '',
  186. };
  187. }
  188. }
  189. module.exports = TimeSource;