APIs

Show:
  1. "use strict";
  2.  
  3. /**
  4. * @module opcua.server
  5. */
  6.  
  7. const _ = require("underscore");
  8. const crypto = require("crypto");
  9.  
  10. const NodeId = require("node-opcua-nodeid").NodeId;
  11. const sameNodeId = require("node-opcua-nodeid").sameNodeId;
  12.  
  13. const NodeIdType = require("node-opcua-nodeid").NodeIdType;
  14. const ServerSidePublishEngine = require("./server_publish_engine").ServerSidePublishEngine;
  15. const StatusCodes = require("node-opcua-status-code").StatusCodes;
  16. const NodeClass = require("node-opcua-data-model").NodeClass;
  17. const QualifiedName = require("node-opcua-data-model").QualifiedName;
  18.  
  19. const DataValue = require("node-opcua-data-value").DataValue;
  20. const VariableIds = require("node-opcua-constants").VariableIds;
  21.  
  22. const ec = require("node-opcua-basic-types");
  23. const assert = require("node-opcua-assert").assert;
  24. const util = require("util");
  25. const EventEmitter = require("events").EventEmitter;
  26. const eoan = require("node-opcua-address-space");
  27. const makeNodeId = require("node-opcua-nodeid").makeNodeId;
  28. const ObjectIds = require("node-opcua-constants").ObjectIds;
  29.  
  30. const WatchDog = require("node-opcua-utils").WatchDog;
  31. const theWatchDog = new WatchDog();
  32.  
  33. const ContinuationPointManager = require("./continuation_point_manager").ContinuationPointManager;
  34.  
  35. const utils = require("node-opcua-utils");
  36. const debugLog = require("node-opcua-debug").make_debugLog(__filename);
  37. const doDebug = require("node-opcua-debug").checkDebugFlag(__filename);
  38.  
  39. /**
  40. *
  41. * A Server session object.
  42. *
  43. * **from OPCUA Spec 1.02:**
  44. *
  45. * * Sessions are created to be independent of the underlying communications connection. Therefore, if a communication
  46. * connection fails, the Session is not immediately affected. The exact mechanism to recover from an underlying
  47. * communication connection error depends on the SecureChannel mapping as described in Part 6.
  48. *
  49. * * Sessions are terminated by the Server automatically if the Client fails to issue a Service request on the Session
  50. * within the timeout period negotiated by the Server in the CreateSession Service response. This protects the Server
  51. * against Client failures and against situations where a failed underlying connection cannot be re-established.
  52. *
  53. * * Clients shall be prepared to submit requests in a timely manner to prevent the Session from closing automatically.
  54. *
  55. * * Clients may explicitly terminate Sessions using the CloseSession Service.
  56. *
  57. * * When a Session is terminated, all outstanding requests on the Session are aborted and Bad_SessionClosed StatusCodes
  58. * are returned to the Client. In addition, the Server deletes the entry for the Client from its
  59. * SessionDiagnosticsArray Variable and notifies any other Clients who were subscribed to this entry.
  60. *
  61. * @class ServerSession
  62. *
  63. * @param parent {SessionEngine}
  64. * @param sessionTimeout {Number}
  65. * @private
  66. * @constructor
  67. */
  68. function ServerSession(parent, sessionTimeout) {
  69.  
  70. const session = this;
  71.  
  72. session.parent = parent; // SessionEngine
  73.  
  74. EventEmitter.apply(session, arguments);
  75.  
  76. ServerSession.registry.register(session);
  77.  
  78. assert(_.isFinite(sessionTimeout));
  79. assert(sessionTimeout >= 0, " sessionTimeout");
  80. session.sessionTimeout = sessionTimeout;
  81.  
  82. const authenticationTokenBuf = crypto.randomBytes(16);
  83. session.authenticationToken = new NodeId(NodeIdType.BYTESTRING, authenticationTokenBuf);
  84.  
  85. // the sessionId
  86. const ownNamespaceIndex = 1; // addressSpace.getOwnNamespace().index;
  87. session.nodeId = new NodeId(NodeIdType.GUID, ec.randomGuid(), ownNamespaceIndex);
  88.  
  89. assert(session.authenticationToken instanceof NodeId);
  90. assert(session.nodeId instanceof NodeId);
  91.  
  92. session._cumulatedSubscriptionCount = 0;
  93.  
  94. session.publishEngine = new ServerSidePublishEngine({
  95. maxPublishRequestInQueue: ServerSession.maxPublishRequestInQueue
  96. });
  97.  
  98. session.publishEngine.setMaxListeners(100);
  99.  
  100. theWatchDog.addSubscriber(session, session.sessionTimeout);
  101.  
  102. session.__status = "new";
  103.  
  104. /**
  105. * the continuation point manager for this session
  106. * @property continuationPointManager
  107. * @type {ContinuationPointManager}
  108. */
  109. session.continuationPointManager = new ContinuationPointManager();
  110.  
  111. /**
  112. * @property creationDate
  113. * @type {Date}
  114. */
  115. session.creationDate = new Date();
  116.  
  117.  
  118. session._registeredNodesCounter = 0;
  119. session._registeredNodes = {};
  120. session._registeredNodesInv = {};
  121. }
  122.  
  123. util.inherits(ServerSession, EventEmitter);
  124.  
  125. const ObjectRegistry = require("node-opcua-object-registry").ObjectRegistry;
  126. ServerSession.registry = new ObjectRegistry();
  127.  
  128. ServerSession.prototype.dispose = function() {
  129.  
  130. debugLog("ServerSession#dispose()");
  131.  
  132. const self = this;
  133.  
  134. assert(!self.sessionObject," sessionObject has not been cleared !");
  135.  
  136. self.parent = null;
  137. self.authenticationToken = null;
  138.  
  139. if (self.publishEngine) {
  140. self.publishEngine.dispose();
  141. self.publishEngine = null;
  142. }
  143.  
  144. self._sessionDiagnostics = null;
  145.  
  146. self._registeredNodesCounter = 0;
  147. self._registeredNodes = null;
  148. self._registeredNodesInv = null;
  149. self.continuationPointManager = null;
  150. self.removeAllListeners();
  151. self.__status = "disposed";
  152.  
  153. ServerSession.registry.unregister(self);
  154.  
  155. };
  156. ServerSession.maxPublishRequestInQueue = 100;
  157.  
  158.  
  159. Object.defineProperty(ServerSession.prototype, "clientConnectionTime", {
  160. get: function () {
  161. const self = this;
  162. return self.creationDate;
  163. }
  164. });
  165.  
  166. Object.defineProperty(ServerSession.prototype, "clientLastContactTime", {
  167. get: function () {
  168. const self = this;
  169. return self._watchDogData.last_seen;
  170. }
  171. });
  172.  
  173. Object.defineProperty(ServerSession.prototype, "status", {
  174. get: function () {
  175. return this.__status;
  176. },
  177. set: function (value) {
  178.  
  179. if (value === "active") {
  180. assert(this.__value !== "active");
  181. this._createSessionObjectInAddressSpace();
  182. }
  183. this.__status = value;
  184. }
  185. });
  186.  
  187. ServerSession.prototype.__defineGetter__("addressSpace", function () {
  188. return this.parent ? this.parent.addressSpace : null;
  189. });
  190.  
  191. ServerSession.prototype.__defineGetter__("currentPublishRequestInQueue", function () {
  192. const self = this;
  193. return self.publishEngine ? self.publishEngine.pendingPublishRequestCount : 0;
  194. });
  195.  
  196. //xx ServerSession.prototype.__defineGetter__("serverDiagnostics")
  197.  
  198. const ServiceCounter = require("node-opcua-common").ServiceCounter;
  199. const SessionDiagnostics = require("node-opcua-common").SessionDiagnostics;
  200. const SubscriptionDiagnostics = require("node-opcua-common").SubscriptionDiagnostics;
  201.  
  202.  
  203. ServerSession.prototype.updateClientLastContactTime = function (currentTime) {
  204. const session = this;
  205. if (session._sessionDiagnostics && session._sessionDiagnostics.clientLastContactTime) {
  206. currentTime = currentTime || new Date();
  207. // do not record all ticks as this may be overwhelming,
  208. if (currentTime.getTime() - 250 >= session._sessionDiagnostics.clientLastContactTime.getTime()) {
  209. session._sessionDiagnostics.clientLastContactTime = currentTime;
  210. }
  211. }
  212. };
  213.  
  214.  
  215. /**
  216. * @method onClientSeen
  217. * required for watch dog
  218. * @param currentTime {DateTime}
  219. * @private
  220. */
  221. ServerSession.prototype.onClientSeen = function (currentTime) {
  222. const session = this;
  223.  
  224. session.updateClientLastContactTime(currentTime);
  225.  
  226. if (session._sessionDiagnostics) {
  227. // see https://opcfoundation-onlineapplications.org/mantis/view.php?id=4111
  228. assert(session._sessionDiagnostics.hasOwnProperty("currentMonitoredItemsCount"));
  229. assert(session._sessionDiagnostics.hasOwnProperty("currentSubscriptionsCount"));
  230. assert(session._sessionDiagnostics.hasOwnProperty("currentPublishRequestsInQueue"));
  231.  
  232. // note : https://opcfoundation-onlineapplications.org/mantis/view.php?id=4111
  233. // sessionDiagnostics extension object uses a different spelling
  234. // here with an S !!!!
  235. session._sessionDiagnostics.currentMonitoredItemsCount = session.currentMonitoredItemCount;
  236. session._sessionDiagnostics.currentSubscriptionsCount = session.currentSubscriptionCount;
  237. session._sessionDiagnostics.currentPublishRequestsInQueue = session.currentPublishRequestInQueue;
  238. }
  239. };
  240.  
  241. ServerSession.prototype.incrementTotalRequestCount = function () {
  242.  
  243. const session = this;
  244. if (session._sessionDiagnostics && session._sessionDiagnostics.totalRequestCount) {
  245. session._sessionDiagnostics.totalRequestCount.totalCount += 1;
  246. }
  247. };
  248.  
  249. const lowerFirstLetter = require("node-opcua-utils").lowerFirstLetter;
  250.  
  251. ServerSession.prototype.incrementRequestTotalCounter = function (counterName) {
  252. const session = this;
  253. if (session._sessionDiagnostics) {
  254. const propName = lowerFirstLetter(counterName + "Count");
  255. if (!session._sessionDiagnostics.hasOwnProperty(propName)) {
  256. console.log(" cannot find", propName);
  257. //xx return;
  258. }
  259. // console.log(self._sessionDiagnostics.toString());
  260. session._sessionDiagnostics[propName].totalCount = session._sessionDiagnostics[propName].totalCount + 1;
  261. }
  262. };
  263. ServerSession.prototype.incrementRequestErrorCounter = function (counterName) {
  264. const session = this;
  265. if (session._sessionDiagnostics) {
  266. const propName = lowerFirstLetter(counterName + "Count");
  267. if (!session._sessionDiagnostics.hasOwnProperty(propName)) {
  268. console.log(" cannot find", propName);
  269. //xx return;
  270. }
  271. session._sessionDiagnostics[propName].errorCount += 1;
  272. }
  273. };
  274.  
  275. /**
  276. * return rootFolder.objects.server.serverDiagnostics.sessionsDiagnosticsSummary
  277. * @returns {UAObject}
  278. */
  279. ServerSession.prototype.getSessionDiagnosticsArray = function () {
  280. const self = this;
  281. return self.addressSpace.rootFolder.objects.server.serverDiagnostics.sessionsDiagnosticsSummary.sessionDiagnosticsArray
  282. };
  283.  
  284. ServerSession.prototype._createSessionObjectInAddressSpace = function () {
  285.  
  286. const session = this;
  287. if (session.sessionObject) {
  288. return;
  289. }
  290. assert(!session.sessionObject, "ServerSession#_createSessionObjectInAddressSpace already called ?");
  291.  
  292. session.sessionObject = null;
  293. if (!session.addressSpace) {
  294. debugLog("ServerSession#_createSessionObjectInAddressSpace : no addressSpace");
  295. return; // no addressSpace
  296. }
  297. const root = session.addressSpace.findNode(makeNodeId(ObjectIds.RootFolder));
  298. assert(root, "expecting a root object");
  299.  
  300. if (!root.objects) {
  301. debugLog("ServerSession#_createSessionObjectInAddressSpace : no object folder");
  302. return false;
  303. }
  304. if (!root.objects.server) {
  305. debugLog("ServerSession#_createSessionObjectInAddressSpace : no server object");
  306. return false;
  307. }
  308.  
  309. // self.addressSpace.findNode(makeNodeId(ObjectIds.Server_ServerDiagnostics));
  310. const serverDiagnosticsNode = root.objects.server.serverDiagnostics;
  311.  
  312. if (!serverDiagnosticsNode || !serverDiagnosticsNode.sessionsDiagnosticsSummary) {
  313. debugLog("ServerSession#_createSessionObjectInAddressSpace : no serverDiagnostics.sessionsDiagnosticsSummary");
  314. return false;
  315. }
  316.  
  317. const sessionDiagnosticsDataType = session.addressSpace.findDataType("SessionDiagnosticsDataType");
  318.  
  319. const sessionDiagnosticsObjectType = session.addressSpace.findObjectType("SessionDiagnosticsObjectType");
  320. const sessionDiagnosticsVariableType = session.addressSpace.findVariableType("SessionDiagnosticsVariableType");
  321.  
  322. const references = [];
  323. if (sessionDiagnosticsObjectType) {
  324. references.push({referenceType: "HasTypeDefinition", isForward: true, nodeId: sessionDiagnosticsObjectType});
  325. }
  326.  
  327. const namespace = session.addressSpace.getOwnNamespace();
  328. session.sessionObject = namespace.createNode({
  329. nodeId: session.nodeId,
  330. nodeClass: NodeClass.Object,
  331. browseName: session.sessionName || "Session-" + session.nodeId.toString(),
  332. componentOf: serverDiagnosticsNode.sessionsDiagnosticsSummary,
  333. typeDefinition: sessionDiagnosticsObjectType,
  334. references: references
  335. });
  336.  
  337. if (sessionDiagnosticsDataType && sessionDiagnosticsVariableType) {
  338.  
  339. // the extension object
  340. session._sessionDiagnostics = session.addressSpace.constructExtensionObject(sessionDiagnosticsDataType, {});
  341. session._sessionDiagnostics.session = session;
  342.  
  343. // install property getter on property that are unlikely to change
  344. if (session.parent.clientDescription) {
  345. session._sessionDiagnostics.clientDescription = session.parent.clientDescription;
  346. }
  347.  
  348. Object.defineProperty(session._sessionDiagnostics, "clientConnectionTime", {
  349. get: function () {
  350. return this.session.clientConnectionTime;
  351. }
  352. });
  353.  
  354. Object.defineProperty(session._sessionDiagnostics, "actualSessionTimeout", {
  355. get: function () {
  356. return this.session.sessionTimeout;
  357. }
  358. });
  359.  
  360. Object.defineProperty(session._sessionDiagnostics, "sessionId", {
  361. get: function () {
  362. return this.session.nodeId;
  363. }
  364. });
  365.  
  366. Object.defineProperty(session._sessionDiagnostics, "sessionName", {
  367. get: function () {
  368. assert(_.isString(session.sessionName));
  369. return this.session.sessionName.toString();
  370. }
  371. });
  372.  
  373. session.sessionDiagnostics = sessionDiagnosticsVariableType.instantiate({
  374. browseName: new QualifiedName({ name: "SessionDiagnostics", namespaceIndex: 0}),
  375. componentOf: session.sessionObject,
  376. extensionObject: session._sessionDiagnostics,
  377. minimumSamplingInterval: 2000 // 2 seconds
  378. });
  379.  
  380. session._sessionDiagnostics = session.sessionDiagnostics.$extensionObject;
  381. assert(session._sessionDiagnostics.session === session);
  382.  
  383. const sessionDiagnosticsArray = session.getSessionDiagnosticsArray();
  384.  
  385. // add sessionDiagnostics into sessionDiagnoticsArray
  386. eoan.addElement(session._sessionDiagnostics, sessionDiagnosticsArray);
  387.  
  388. }
  389.  
  390. const subscriptionDiagnosticsArrayType = session.addressSpace.findVariableType("SubscriptionDiagnosticsArrayType");
  391. assert(subscriptionDiagnosticsArrayType.nodeId.toString() === "ns=0;i=2171");
  392.  
  393. session.subscriptionDiagnosticsArray=
  394. eoan.createExtObjArrayNode(session.sessionObject, {
  395. browseName: { namespaceIndex: 0,name:"SubscriptionDiagnosticsArray" },
  396. complexVariableType: "SubscriptionDiagnosticsArrayType",
  397. variableType: "SubscriptionDiagnosticsType",
  398. indexPropertyName: "subscriptionId",
  399. minimumSamplingInterval: 2000 // 2 seconds
  400. });
  401.  
  402. return session.sessionObject;
  403. };
  404.  
  405. ServerSession.prototype._removeSessionObjectFromAddressSpace = function () {
  406.  
  407. const session = this;
  408.  
  409. // todo : dump session statistics in a file or somewhere for deeper diagnostic analysis on closed session
  410.  
  411. if (!session.addressSpace) {
  412. return;
  413. }
  414. if (session.sessionDiagnostics) {
  415.  
  416. const sessionDiagnosticsArray = session.getSessionDiagnosticsArray();
  417. eoan.removeElement(sessionDiagnosticsArray, session.sessionDiagnostics.$extensionObject);
  418.  
  419. session.addressSpace.deleteNode(session.sessionDiagnostics);
  420.  
  421. assert(session._sessionDiagnostics.session === session);
  422. session._sessionDiagnostics.session = null;
  423.  
  424. session._sessionDiagnostics = null;
  425. session.sessionDiagnostics = null;
  426.  
  427. }
  428. if (session.sessionObject) {
  429. session.addressSpace.deleteNode(session.sessionObject);
  430. session.sessionObject = null;
  431. }
  432. };
  433.  
  434. const Subscription = require("./server_subscription").Subscription;
  435.  
  436.  
  437.  
  438.  
  439. ServerSession.prototype._getSubscriptionDiagnosticsArray = function () {
  440.  
  441. const session = this;
  442. if (!session.addressSpace) {
  443. if (doDebug) {
  444. console.warn("ServerSession#_getSubscriptionDiagnosticsArray : no addressSpace");
  445. }
  446. return null; // no addressSpace
  447. }
  448.  
  449. const subscriptionDiagnosticsArray = session.subscriptionDiagnosticsArray;
  450. if (!subscriptionDiagnosticsArray) {
  451. return null; // no subscriptionDiagnosticsArray
  452. }
  453. assert(subscriptionDiagnosticsArray.browseName.toString() === "SubscriptionDiagnosticsArray");
  454. return subscriptionDiagnosticsArray;
  455. };
  456.  
  457. ServerSession.prototype._exposeSubscriptionDiagnostics = function (subscription) {
  458. const session = this;
  459. debugLog("ServerSession#_exposeSubscriptionDiagnostics");
  460. assert(subscription.$session === session);
  461. const subscriptionDiagnosticsArray = session._getSubscriptionDiagnosticsArray();
  462. const subscriptionDiagnostics = subscription.subscriptionDiagnostics;
  463. assert(subscriptionDiagnostics.$subscription == subscription);
  464.  
  465. if (subscriptionDiagnostics && subscriptionDiagnosticsArray) {
  466. //xx console.log("GGGGGGGGGGGGGGGG => ServerSession Exposing subscription diagnostics =>",subscription.id,"on session", session.nodeId.toString());
  467. eoan.addElement(subscriptionDiagnostics, subscriptionDiagnosticsArray);
  468. }
  469. };
  470.  
  471. function compareSessionId(sessionDiagnostics1, sessionDiagnostics2) {
  472. return sessionDiagnostics1.sessionId.toString() == sessionDiagnostics2.sessionId.toString();
  473. }
  474.  
  475. ServerSession.prototype._unexposeSubscriptionDiagnostics = function (subscription) {
  476.  
  477. const session = this;
  478. const subscriptionDiagnosticsArray = session._getSubscriptionDiagnosticsArray();
  479. const subscriptionDiagnostics = subscription.subscriptionDiagnostics;
  480. assert(subscriptionDiagnostics instanceof SubscriptionDiagnostics);
  481. if (subscriptionDiagnostics && subscriptionDiagnosticsArray) {
  482. //xx console.log("GGGGGGGGGGGGGGGG => ServerSession **Unexposing** subscription diagnostics =>",subscription.id,"on session", session.nodeId.toString());
  483. eoan.removeElement(subscriptionDiagnosticsArray, subscriptionDiagnostics);
  484. }
  485. debugLog("ServerSession#_unexposeSubscriptionDiagnostics");
  486. };
  487.  
  488.  
  489. ServerSession.prototype.assignSubscription = function (subscription) {
  490. const session = this;
  491. assert(!subscription.$session);
  492. assert(session.nodeId instanceof NodeId);
  493.  
  494. subscription.$session = session;
  495.  
  496. subscription.sessionId = session.nodeId;
  497.  
  498. session._cumulatedSubscriptionCount += 1;
  499.  
  500. // Notify the owner that a new subscription has been created
  501. // @event new_subscription
  502. // @param {Subscription} subscription
  503. session.emit("new_subscription", subscription);
  504.  
  505. // add subscription diagnostics to SubscriptionDiagnosticsArray
  506. session._exposeSubscriptionDiagnostics(subscription);
  507.  
  508. subscription.once("terminated", function () {
  509. //Xx session._unexposeSubscriptionDiagnostics(subscription);
  510. // Notify the owner that a new subscription has been terminated
  511. // @event subscription_terminated
  512. // @param {Subscription} subscription
  513. session.emit("subscription_terminated", subscription);
  514. });
  515.  
  516.  
  517. };
  518. ServerSession.prototype.createSubscription = function(parameters) {
  519. const session = this;
  520. const subscription = session.parent._createSubscriptionOnSession(session,parameters);
  521. session.assignSubscription(subscription);
  522. assert(subscription.$session === session);
  523. assert(subscription.sessionId instanceof NodeId);
  524. assert(sameNodeId(subscription.sessionId,session.nodeId));
  525. return subscription;
  526. };
  527.  
  528.  
  529. /**
  530. * number of active subscriptions
  531. * @property currentSubscriptionCount
  532. * @type {Number}
  533. */
  534. ServerSession.prototype.__defineGetter__("currentSubscriptionCount", function () {
  535. const self = this;
  536. return self.publishEngine ? self.publishEngine.subscriptionCount : 0;
  537. });
  538.  
  539. /**
  540. * number of subscriptions ever created since this object is live
  541. * @property cumulatedSubscriptionCount
  542. * @type {Number}
  543. */
  544. ServerSession.prototype.__defineGetter__("cumulatedSubscriptionCount", function () {
  545. return this._cumulatedSubscriptionCount;
  546. });
  547.  
  548. /**
  549. * number of monitored items
  550. * @property currentMonitoredItemCount
  551. * @type {Number}
  552. */
  553. ServerSession.prototype.__defineGetter__("currentMonitoredItemCount", function () {
  554. const self = this;
  555. return self.publishEngine ? self.publishEngine.currentMonitoredItemCount : 0;
  556. });
  557.  
  558.  
  559. const SubscriptionState = require("./server_subscription").SubscriptionState;
  560. /**
  561. * retrieve an existing subscription by subscriptionId
  562. * @method getSubscription
  563. * @param subscriptionId {Number}
  564. * @return {Subscription}
  565. */
  566. ServerSession.prototype.getSubscription = function (subscriptionId) {
  567. const subscription = this.publishEngine.getSubscriptionById(subscriptionId);
  568. if (subscription && subscription.state === SubscriptionState.CLOSED) {
  569. // subscription is CLOSED but has not been notified yet
  570. // it should be considered as excluded
  571. return null;
  572. }
  573. assert(!subscription || subscription.state !== SubscriptionState.CLOSED, "CLOSED subscription shall not be managed by publish engine anymore");
  574. return subscription;
  575. };
  576.  
  577. /**
  578. * @method deleteSubscription
  579. * @param subscriptionId {Number}
  580. * @return {StatusCode}
  581. */
  582. ServerSession.prototype.deleteSubscription = function (subscriptionId) {
  583.  
  584. const session = this;
  585. const subscription = session.getSubscription(subscriptionId);
  586. if (!subscription) {
  587. return StatusCodes.BadSubscriptionIdInvalid;
  588. }
  589.  
  590. //xx this.publishEngine.remove_subscription(subscription);
  591. subscription.terminate();
  592.  
  593. if (session.currentSubscriptionCount === 0) {
  594.  
  595. const local_publishEngine = session.publishEngine;
  596. local_publishEngine.cancelPendingPublishRequest();
  597. }
  598. return StatusCodes.Good;
  599. };
  600.  
  601. ServerSession.prototype._deleteSubscriptions = function () {
  602.  
  603. const session = this;
  604. assert(session.publishEngine);
  605.  
  606. const subscriptions = session.publishEngine.subscriptions;
  607.  
  608. subscriptions.forEach(function (subscription) {
  609. session.deleteSubscription(subscription.id);
  610. });
  611.  
  612. };
  613.  
  614. /**
  615. * close a ServerSession, this will also delete the subscriptions if the flag is set.
  616. *
  617. * Spec extract:
  618. *
  619. * If a Client invokes the CloseSession Service then all Subscriptions associated with the Session are also deleted
  620. * if the deleteSubscriptions flag is set to TRUE. If a Server terminates a Session for any other reason,
  621. * Subscriptions associated with the Session, are not deleted. Each Subscription has its own lifetime to protect
  622. * against data loss in the case of a Session termination. In these cases, the Subscription can be reassigned to
  623. * another Client before its lifetime expires.
  624. *
  625. * @method close
  626. * @param {Boolean} deleteSubscriptions : should we delete subscription ?
  627. * @param {String} [reason = "CloseSession"] the reason for closing the session (shall be "Timeout", "Terminated" or "CloseSession")
  628. *
  629. */
  630. ServerSession.prototype.close = function (deleteSubscriptions, reason) {
  631.  
  632. //xx console.log("xxxxxxxxxxxxxxxxxxxxxxxxxxx => ServerSession.close() ");
  633. const session = this;
  634.  
  635. if (session.publishEngine) {
  636. session.publishEngine.onSessionClose();
  637. }
  638.  
  639. theWatchDog.removeSubscriber(session);
  640. // --------------- delete associated subscriptions ---------------------
  641.  
  642.  
  643. if (!deleteSubscriptions && session.currentSubscriptionCount !== 0) {
  644.  
  645. // I don't know what to do yet if deleteSubscriptions is false
  646. console.log("TO DO : Closing session without deleting subscription not yet implemented");
  647. //to do: Put subscriptions in safe place for future transfer if any
  648.  
  649. }
  650.  
  651. session._deleteSubscriptions();
  652.  
  653. assert(session.currentSubscriptionCount === 0);
  654.  
  655.  
  656. // Post-Conditions
  657. assert(session.currentSubscriptionCount === 0);
  658.  
  659. session.status = "closed";
  660. /**
  661. * @event session_closed
  662. * @param deleteSubscriptions {Boolean}
  663. * @param reason {String}
  664. */
  665. session.emit("session_closed", session, deleteSubscriptions, reason);
  666.  
  667.  
  668. // ---------------- shut down publish engine
  669. if (session.publishEngine) {
  670.  
  671. // remove subscription
  672. session.publishEngine.shutdown();
  673.  
  674. assert(session.publishEngine.subscriptionCount === 0);
  675. session.publishEngine.dispose();
  676. session.publishEngine = null;
  677. }
  678.  
  679. session._removeSessionObjectFromAddressSpace();
  680.  
  681. assert(!session.sessionDiagnostics, "ServerSession#_removeSessionObjectFromAddressSpace must be called");
  682. assert(!session.sessionObject, "ServerSession#_removeSessionObjectFromAddressSpace must be called");
  683.  
  684. };
  685.  
  686. /**
  687. * @method watchdogReset
  688. * used as a callback for the Watchdog
  689. * @private
  690. */
  691. ServerSession.prototype.watchdogReset = function () {
  692. const self = this;
  693. // the server session has expired and must be removed from the server
  694. self.emit("timeout");
  695. };
  696.  
  697. const registeredNodeNameSpace = 9999 ;
  698.  
  699. ServerSession.prototype.registerNode = function(nodeId) {
  700.  
  701. assert(nodeId instanceof NodeId);
  702. const session = this;
  703.  
  704. if (nodeId.namespace === 0 && nodeId.identifierType.value === NodeIdType.NUMERIC.value) {
  705. return nodeId;
  706. }
  707.  
  708. const key = nodeId.toString();
  709.  
  710. const registeredNode = session._registeredNodes[key];
  711. if (registeredNode) {
  712. // already registered
  713. return registeredNode;
  714. }
  715.  
  716. const node = session.addressSpace.findNode(nodeId);
  717. if (!node) {
  718. return nodeId;
  719. }
  720.  
  721. session._registeredNodesCounter +=1;
  722.  
  723. const aliasNodeId = makeNodeId(session._registeredNodesCounter,registeredNodeNameSpace);
  724. session._registeredNodes[key] = aliasNodeId;
  725. session._registeredNodesInv[aliasNodeId.toString()] = node;
  726. return aliasNodeId;
  727. };
  728.  
  729. ServerSession.prototype.unRegisterNode = function(aliasNodeId) {
  730.  
  731. assert(aliasNodeId instanceof NodeId);
  732. if (aliasNodeId.namespace !== registeredNodeNameSpace) {
  733. return aliasNodeId; // not a registered Node
  734. }
  735. const session = this;
  736.  
  737.  
  738. const node = session._registeredNodesInv[aliasNodeId.toString()];
  739. if (!node) {
  740. return ;
  741. }
  742. session._registeredNodesInv[aliasNodeId.toString()] = null;
  743. session._registeredNodes[node.nodeId.toString()] = null;
  744.  
  745. };
  746.  
  747. ServerSession.prototype.resolveRegisteredNode = function(aliasNodeId) {
  748.  
  749. const session = this;
  750. if (aliasNodeId.namespace !== registeredNodeNameSpace) {
  751. return aliasNodeId; // not a registered Node
  752. }
  753. const node = session._registeredNodesInv[aliasNodeId.toString()];
  754. if (!node) {
  755. return aliasNodeId;
  756. }
  757. return node.nodeId;
  758. };
  759.  
  760. /**
  761. * true if the underlying channel has been closed or aborted...
  762. */
  763. ServerSession.prototype.__defineGetter__("aborted", function () {
  764. const session = this;
  765. if (!session.channel) {
  766. return true;
  767. }
  768. return session.channel.aborted;
  769. });
  770.  
  771. function on_channel_abort()
  772. {
  773. const session = this;
  774. debugLog("ON CHANNEL ABORT ON SESSION!!!");
  775. /**
  776. * @event channel_aborted
  777. */
  778. session.emit("channel_aborted");
  779. }
  780.  
  781. ServerSession.prototype._attach_channel = function(channel) {
  782. const session = this;
  783. assert(session.nonce && session.nonce instanceof Buffer);
  784. session.channel = channel;
  785. session.secureChannelId = channel.secureChannelId;
  786. const key = session.authenticationToken.toString("hex");
  787. assert(!channel.sessionTokens.hasOwnProperty(key), "channel has already a session");
  788. channel.sessionTokens[key] = session;
  789.  
  790. // when channel is aborting
  791. session.channel_abort_event_handler = on_channel_abort.bind(session);
  792. channel.on("abort", session.channel_abort_event_handler);
  793.  
  794. };
  795.  
  796. ServerSession.prototype._detach_channel = function() {
  797.  
  798. const session = this;
  799. const channel = session.channel;
  800. assert(channel,"expecting a valid channel");
  801. assert(session.nonce && session.nonce instanceof Buffer);
  802. assert(session.authenticationToken);
  803. const key = session.authenticationToken.toString("hex");
  804. assert(channel.sessionTokens.hasOwnProperty(key));
  805. assert(session.channel);
  806. assert(_.isFunction(session.channel_abort_event_handler));
  807. channel.removeListener("abort", session.channel_abort_event_handler);
  808.  
  809. delete channel.sessionTokens[key];
  810. session.channel = null;
  811. session.secureChannelId = null;
  812. };
  813.  
  814. exports.ServerSession = ServerSession;
  815.