Source: lib/time.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 Duration = require('./duration.js');
  17. const ClockType = require('./clock_type.js');
  18. const S_TO_NS = 10n ** 9n;
  19. /**
  20. * @class - Class representing a Time in ROS
  21. */
  22. class Time {
  23. /**
  24. * Create a Time.
  25. * @param {bigint} [seconds=0] - The second part of the time.
  26. * @param {bigint} [nanoseconds=0] - The nanosecond part of the time.
  27. * @param {ClockType} [clockType=Clock.ClockType.SYSTEM_TIME] - The clock type.
  28. */
  29. constructor(
  30. seconds = 0n,
  31. nanoseconds = 0n,
  32. clockType = ClockType.SYSTEM_TIME
  33. ) {
  34. if (typeof seconds !== 'bigint') {
  35. throw new TypeError('Invalid argument of seconds');
  36. }
  37. if (typeof nanoseconds !== 'bigint') {
  38. throw new TypeError('Invalid argument of nanoseconds');
  39. }
  40. if (typeof clockType !== 'number') {
  41. throw new TypeError('Invalid argument of clockType');
  42. }
  43. if (seconds < 0n) {
  44. throw new RangeError('seconds value must not be negative');
  45. }
  46. if (nanoseconds < 0n) {
  47. throw new RangeError('nanoseconds value must not be negative');
  48. }
  49. const total = seconds * S_TO_NS + nanoseconds;
  50. if (total >= 2n ** 63n) {
  51. throw new RangeError(
  52. 'Total nanoseconds value is too large to store in C time point.'
  53. );
  54. }
  55. this._nanoseconds = total;
  56. this._handle = rclnodejs.createTimePoint(this._nanoseconds, clockType);
  57. this._clockType = clockType;
  58. }
  59. /**
  60. * Get the the clock type of the Time object.
  61. * @name Time#get:clockType
  62. * @function
  63. * @return {ClockType} - The clock type.
  64. */
  65. get clockType() {
  66. return this._clockType;
  67. }
  68. /**
  69. * Get the nanosecond part of the time.
  70. * @name Time#get:nanoseconds
  71. * @function
  72. * @return {bigint} - value in nanosecond.
  73. */
  74. get nanoseconds() {
  75. return rclnodejs.getNanoseconds(this._handle);
  76. }
  77. /**
  78. * Get the time as separate seconds and nanoseconds component.
  79. * @name Time#get:secondsAndNanoseconds
  80. * @function
  81. * @return {object} - object with properties seconds and nanoseconds.
  82. */
  83. get secondsAndNanoseconds() {
  84. const nanoseconds = this._nanoseconds;
  85. return {
  86. seconds: nanoseconds / S_TO_NS,
  87. nanoseconds: nanoseconds % S_TO_NS,
  88. };
  89. }
  90. /**
  91. * Add a duration to this time object.
  92. * @param {Duration} other - The Duration object to be added.
  93. * @return {Time} Return the result of a new Time object.
  94. */
  95. add(other) {
  96. if (other instanceof Duration) {
  97. return new Time(
  98. 0n,
  99. this._nanoseconds + other.nanoseconds,
  100. this._clockType
  101. );
  102. }
  103. throw new TypeError('Invalid argument');
  104. }
  105. /**
  106. * Subtract a duration/time to this time object.
  107. * @param {Duration|Time} other - The time to be subtracted.
  108. * @return {Duration|Time} Return the result.
  109. */
  110. sub(other) {
  111. if (other instanceof Time) {
  112. if (other._clockType !== this._clockType) {
  113. throw new TypeError("Can't subtract times with different clock types");
  114. }
  115. return new Duration(0n, this._nanoseconds - other._nanoseconds);
  116. } else if (other instanceof Duration) {
  117. return new Time(
  118. 0n,
  119. this._nanoseconds - other._nanoseconds,
  120. this._clockType
  121. );
  122. }
  123. throw new TypeError('Invalid argument');
  124. }
  125. /**
  126. * Determine whether two Time objects are equal.
  127. * @param {Time} other - The time object to be compared.
  128. * @return {boolean} Return true if they are equal.
  129. */
  130. eq(other) {
  131. if (other instanceof Time) {
  132. if (other._clockType !== this._clockType) {
  133. throw new TypeError("Can't compare times with different clock types");
  134. }
  135. return this._nanoseconds === other.nanoseconds;
  136. }
  137. throw new TypeError('Invalid argument');
  138. }
  139. /**
  140. * Determine whether two Time objects are not equal.
  141. * @param {Time} other - The time object to be compared.
  142. * @return {boolean} Return true if they are not equal.
  143. */
  144. ne(other) {
  145. if (other instanceof Time) {
  146. if (other._clockType !== this._clockType) {
  147. throw new TypeError("Can't compare times with different clock types");
  148. }
  149. return this._nanoseconds !== other.nanoseconds;
  150. }
  151. }
  152. /**
  153. * Determine whether the time is less than another one.
  154. * @param {Time} other - The time object to be compared.
  155. * @return {boolean} Return true if it's less than other.
  156. */
  157. lt(other) {
  158. if (other instanceof Time) {
  159. if (other._clockType !== this._clockType) {
  160. throw new TypeError("Can't compare times with different clock types");
  161. }
  162. return this._nanoseconds < other.nanoseconds;
  163. }
  164. throw new TypeError('Invalid argument');
  165. }
  166. /**
  167. * Determine whether the time is less than or equal with another one.
  168. * @param {Time} other - The time object to be compared.
  169. * @return {boolean} Return true if it's less than or equal with other.
  170. */
  171. lte(other) {
  172. if (other instanceof Time) {
  173. if (other._clockType !== this._clockType) {
  174. throw new TypeError("Can't compare times with different clock types");
  175. }
  176. return this._nanoseconds <= other.nanoseconds;
  177. }
  178. throw new TypeError('Invalid argument');
  179. }
  180. /**
  181. * Determine whether the time is greater than another one.
  182. * @param {Time} other - The time object to be compared.
  183. * @return {boolean} Return true if it's greater than other.
  184. */
  185. gt(other) {
  186. if (other instanceof Time) {
  187. if (other._clockType !== this._clockType) {
  188. throw new TypeError("Can't compare times with different clock types");
  189. }
  190. return this._nanoseconds > other.nanoseconds;
  191. }
  192. throw new TypeError('Invalid argument');
  193. }
  194. /**
  195. * Determine whether the time is greater than or equal with another one.
  196. * @param {Time} other - The time object to be compared.
  197. * @return {boolean} Return true if it's greater than or equal with other.
  198. */
  199. gte(other) {
  200. if (other instanceof Time) {
  201. if (other._clockType !== this._clockType) {
  202. throw new TypeError("Can't compare times with different clock types");
  203. }
  204. return this._nanoseconds >= other.nanoseconds;
  205. }
  206. throw new TypeError('Invalid argument');
  207. }
  208. /**
  209. * Create a builtin_interfaces.msg.Time message
  210. *
  211. * @return {builtin_interfaces.msg.Time} - The new Time message.
  212. */
  213. toMsg() {
  214. const secondsAndNanoseconds = this.secondsAndNanoseconds;
  215. return {
  216. sec: secondsAndNanoseconds.seconds,
  217. nanosec: secondsAndNanoseconds.nanoseconds,
  218. };
  219. }
  220. /**
  221. * Create a Time object from a message of builtin_interfaces/msg/Time
  222. * @param {object} msg - The builtin_interfaces.msg.Time message to be
  223. * created from.
  224. * @param {ClockType} [clockType=Clock.ClockType.ROS_TIME] - The type of the time object.
  225. * @return {Time} Return the created Time object.
  226. */
  227. static fromMsg(msg, clockType = ClockType.ROS_TIME) {
  228. return new Time(BigInt(msg.sec), BigInt(msg.nanosec), clockType);
  229. }
  230. }
  231. module.exports = Time;