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