// Copyright (c) 2017 Intel Corporation. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
'use strict';
const rclnodejs = require('bindings')('rclnodejs');
const DistroUtils = require('./distro.js');
const Entity = require('./entity.js');
const debug = require('debug')('rclnodejs:service');
/**
* The response to client
*
* @property {Object} template - Get an empty response object that is being sent to client.
* @property {Service} service - The service that this response object is attaching to.
*
* @hideconstructor
*/
class Response {
constructor(service, header) {
this._service = service;
this._header = header;
this.sent = false;
}
get template() {
let response = new this._service._typeClass.Response();
return response.toPlainObject();
}
get service() {
return this._service;
}
/**
* Send response to client (the service caller)
* @param {object} response - The plain JavaScript representing the response.
Note: you can use .template to get an empty result object.
* @return {undefined}
* @see {@link Response#template}
*/
send(response) {
let responseToReturn = new this._service._typeClass.Response(response);
const rawResponse = responseToReturn.serialize();
rclnodejs.sendResponse(this._service._handle, rawResponse, this._header);
this.sent = true;
}
}
/**
* @class - Class representing a Service in ROS
* @hideconstructor
*/
class Service extends Entity {
constructor(nodeHandle, handle, serviceName, typeClass, options, callback) {
super(handle, typeClass, options);
this._nodeHandle = nodeHandle;
this._callback = callback;
}
processRequest(headerHandle, request) {
debug(`Service has received a ${this._serviceName} request from client.`);
const plainObj = request.toPlainObject(this.typedArrayEnabled);
const response = new Response(this, headerHandle);
Promise.resolve(this._callback(plainObj, response)).then((responseToReturn) => {
if (!response.sent && responseToReturn) {
responseToReturn = new this._typeClass.Response(responseToReturn);
const rawResponse = responseToReturn.serialize();
rclnodejs.sendResponse(this._handle, rawResponse, headerHandle);
}
debug(
`Service has processed the ${this._serviceName} request and sent the response.`
);
});
}
static createService(nodeHandle, serviceName, typeClass, options, callback) {
let type = typeClass.type();
let handle = rclnodejs.createService(
nodeHandle,
serviceName,
type.interfaceName,
type.pkgName,
options.qos
);
return new Service(
nodeHandle,
handle,
serviceName,
typeClass,
options,
callback
);
}
/**
* @type {string}
*/
get serviceName() {
return rclnodejs.getServiceServiceName(this._handle);
}
/**
* Configure introspection.
* @param {Clock} clock - Clock to use for service event timestamps
* @param {QoS} qos - QoSProfile for the service event publisher
* @param {ServiceIntrospectionState} introspectionState - State to set introspection to
*/
configureIntrospection(clock, qos, introspectionState) {
if (DistroUtils.getDistroId() <= DistroUtils.getDistroId('humble')) {
console.warn(
'Service introspection is not supported by this versionof ROS 2'
);
return;
}
let type = this.typeClass.type();
rclnodejs.configureServiceIntrospection(
this.handle,
this._nodeHandle,
clock.handle,
type.interfaceName,
type.pkgName,
qos,
introspectionState,
true
);
}
}
module.exports = Service;