Source: lib/subscription.js

  1. // Copyright (c) 2017 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 Entity = require('./entity.js');
  17. const debug = require('debug')('rclnodejs:subscription');
  18. /**
  19. * @class - Class representing a ROS 2 Subscription
  20. * @hideconstructor
  21. * Includes support for content-filtering topics beginning with the
  22. * ROS Humble release. To learn more about content-filtering
  23. * @see {@link Node#options}
  24. * @see {@link Node#createSubscription}
  25. * @see {@link https://www.omg.org/spec/DDS/1.4/PDF|DDS 1.4 specification, Annex B}
  26. */
  27. class Subscription extends Entity {
  28. constructor(handle, typeClass, topic, options, callback) {
  29. super(handle, typeClass, options);
  30. this._topic = topic;
  31. this._callback = callback;
  32. this._isRaw = options.isRaw || false;
  33. }
  34. processResponse(msg) {
  35. debug(`Message of topic ${this._topic} received.`);
  36. if (this._isRaw) {
  37. this._callback(msg);
  38. } else {
  39. this._callback(msg.toPlainObject(this.typedArrayEnabled));
  40. }
  41. }
  42. static createSubscription(nodeHandle, typeClass, topic, options, callback) {
  43. let type = typeClass.type();
  44. // convert contentFilter.parameters to a string[]
  45. if (options.contentFilter && options.contentFilter.parameters) {
  46. options.contentFilter.parameters = options.contentFilter.parameters.map(
  47. (param) => param.toString()
  48. );
  49. }
  50. let handle = rclnodejs.createSubscription(
  51. nodeHandle,
  52. type.pkgName,
  53. type.subFolder,
  54. type.interfaceName,
  55. topic,
  56. options
  57. );
  58. return new Subscription(handle, typeClass, topic, options, callback);
  59. }
  60. /**
  61. * @type {string}
  62. */
  63. get topic() {
  64. return this._topic;
  65. }
  66. /**
  67. * @type {boolean}
  68. */
  69. get isRaw() {
  70. return this._isRaw;
  71. }
  72. /**
  73. * Test if the RMW supports content-filtered topics and that this subscription
  74. * has an active wellformed content-filter.
  75. * @returns {boolean} True if content-filtering will be applied; otherwise false.
  76. */
  77. hasContentFilter() {
  78. return rclnodejs.hasContentFilter(this.handle);
  79. }
  80. /**
  81. * If the RMW supports content-filtered topics set this subscription's content-filter.
  82. * @param {object} contentFilter - The content-filter description.
  83. * @param {string} contentFilter.expression - Specifies the criteria to select messages of interest.
  84. * It is similar to the WHERE part of an SQL clause. Clear the current contentFilter if
  85. * the expression is undefined or an empty string.
  86. * @param {object[]} [contentFilter.parameters=undefined] - Array of objects that give values to
  87. * the ‘parameters’ (i.e., "%n" tokens) in the filter_expression. The number of supplied parameters must
  88. * fit with the requested values in the filter_expression (i.e., the number of %n tokens). default: undefined.
  89. * @returns {boolean} - True if successful; false otherwise
  90. * @see {@link https://www.omg.org/spec/DDS/1.4/PDF|DDS 1.4 specification, Annex B}
  91. */
  92. setContentFilter(contentFilter) {
  93. return contentFilter?.expression
  94. ? rclnodejs.setContentFilter(this.handle, contentFilter)
  95. : this.clearContentFilter();
  96. }
  97. /**
  98. * Clear the current content-filter. No filtering is to be applied.
  99. * @returns {boolean} - True if successful; false otherwise
  100. */
  101. clearContentFilter() {
  102. return this.hasContentFilter()
  103. ? rclnodejs.clearContentFilter(this.handle)
  104. : true;
  105. }
  106. }
  107. module.exports = Subscription;