123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225 |
- "use strict"
- var os = require('os'),
- f = require('util').format;
- /**
- * Emit event if it exists
- * @method
- */
- function emitSDAMEvent(self, event, description) {
- if(self.listeners(event).length > 0) {
- self.emit(event, description);
- }
- }
- // Get package.json variable
- var driverVersion = require(__dirname + '/../../package.json').version;
- var nodejsversion = f('Node.js %s, %s', process.version, os.endianness());
- var type = os.type();
- var name = process.platform;
- var architecture = process.arch;
- var release = os.release();
- function createClientInfo(options) {
- // Build default client information
- var clientInfo = options.clientInfo ? clone(options.clientInfo) : {
- driver: {
- name: "nodejs-core",
- version: driverVersion
- },
- os: {
- type: type,
- name: name,
- architecture: architecture,
- version: release
- }
- }
- // Is platform specified
- if(clientInfo.platform && clientInfo.platform.indexOf('mongodb-core') == -1) {
- clientInfo.platform = f('%s, mongodb-core: %s', clientInfo.platform, driverVersion);
- } else if(!clientInfo.platform){
- clientInfo.platform = nodejsversion;
- }
- // Do we have an application specific string
- if(options.appname) {
- // Cut at 128 bytes
- var buffer = new Buffer(options.appname);
- // Return the truncated appname
- var appname = buffer.length > 128 ? buffer.slice(0, 128).toString('utf8') : options.appname;
- // Add to the clientInfo
- clientInfo.application = { name: appname };
- }
- return clientInfo;
- }
- function clone(object) {
- return JSON.parse(JSON.stringify(object));
- }
- var getPreviousDescription = function(self) {
- if(!self.s.serverDescription) {
- self.s.serverDescription = {
- address: self.name,
- arbiters: [], hosts: [], passives: [], type: 'Unknown'
- }
- }
- return self.s.serverDescription;
- }
- var emitServerDescriptionChanged = function(self, description) {
- if(self.listeners('serverDescriptionChanged').length > 0) {
- // Emit the server description changed events
- self.emit('serverDescriptionChanged', {
- topologyId: self.s.topologyId != -1 ? self.s.topologyId : self.id, address: self.name,
- previousDescription: getPreviousDescription(self),
- newDescription: description
- });
- self.s.serverDescription = description;
- }
- }
- var getPreviousTopologyDescription = function(self) {
- if(!self.s.topologyDescription) {
- self.s.topologyDescription = {
- topologyType: 'Unknown',
- servers: [{
- address: self.name, arbiters: [], hosts: [], passives: [], type: 'Unknown'
- }]
- }
- }
- return self.s.topologyDescription;
- }
- var emitTopologyDescriptionChanged = function(self, description) {
- if(self.listeners('topologyDescriptionChanged').length > 0) {
- // Emit the server description changed events
- self.emit('topologyDescriptionChanged', {
- topologyId: self.s.topologyId != -1 ? self.s.topologyId : self.id, address: self.name,
- previousDescription: getPreviousTopologyDescription(self),
- newDescription: description
- });
- self.s.serverDescription = description;
- }
- }
- var changedIsMaster = function(self, currentIsmaster, ismaster) {
- var currentType = getTopologyType(self, currentIsmaster);
- var newType = getTopologyType(self, ismaster);
- if(newType != currentType) return true;
- return false;
- }
- var getTopologyType = function(self, ismaster) {
- if(!ismaster) {
- ismaster = self.ismaster;
- }
- if(!ismaster) return 'Unknown';
- if(ismaster.ismaster && !ismaster.hosts) return 'Standalone';
- if(ismaster.ismaster && ismaster.msg == 'isdbgrid') return 'Mongos';
- if(ismaster.ismaster) return 'RSPrimary';
- if(ismaster.secondary) return 'RSSecondary';
- if(ismaster.arbiterOnly) return 'RSArbiter';
- return 'Unknown';
- }
- var inquireServerState = function(self) {
- return function(callback) {
- if(self.s.state == 'destroyed') return;
- // Record response time
- var start = new Date().getTime();
- // emitSDAMEvent
- emitSDAMEvent(self, 'serverHeartbeatStarted', { connectionId: self.name });
- // Attempt to execute ismaster command
- self.command('admin.$cmd', { ismaster:true }, { monitoring:true }, function(err, r) {
- if(!err) {
- // Legacy event sender
- self.emit('ismaster', r, self);
- // Calculate latencyMS
- var latencyMS = new Date().getTime() - start;
- // Server heart beat event
- emitSDAMEvent(self, 'serverHeartbeatSucceeded', { durationMS: latencyMS, reply: r.result, connectionId: self.name });
- // Did the server change
- if(changedIsMaster(self, self.s.ismaster, r.result)) {
- // Emit server description changed if something listening
- emitServerDescriptionChanged(self, {
- address: self.name, arbiters: [], hosts: [], passives: [], type: !self.s.inTopology ? 'Standalone' : getTopologyType(self)
- });
- }
- // Updat ismaster view
- self.s.ismaster = r.result;
- // Set server response time
- self.s.isMasterLatencyMS = latencyMS;
- } else {
- emitSDAMEvent(self, 'serverHeartbeatFailed', { durationMS: latencyMS, failure: err, connectionId: self.name });
- }
- // Peforming an ismaster monitoring callback operation
- if(typeof callback == 'function') {
- return callback(err, r);
- }
- // Perform another sweep
- self.s.inquireServerStateTimeout = setTimeout(inquireServerState(self), self.s.haInterval);
- });
- };
- }
- // Object.assign method or polyfille
- var assign = Object.assign ? Object.assign : function assign(target) {
- if (target === undefined || target === null) {
- throw new TypeError('Cannot convert first argument to object');
- }
- var to = Object(target);
- for (var i = 1; i < arguments.length; i++) {
- var nextSource = arguments[i];
- if (nextSource === undefined || nextSource === null) {
- continue;
- }
- var keysArray = Object.keys(Object(nextSource));
- for (var nextIndex = 0, len = keysArray.length; nextIndex < len; nextIndex++) {
- var nextKey = keysArray[nextIndex];
- var desc = Object.getOwnPropertyDescriptor(nextSource, nextKey);
- if (desc !== undefined && desc.enumerable) {
- to[nextKey] = nextSource[nextKey];
- }
- }
- }
- return to;
- }
- //
- // Clone the options
- var cloneOptions = function(options) {
- var opts = {};
- for(var name in options) {
- opts[name] = options[name];
- }
- return opts;
- }
- module.exports.inquireServerState = inquireServerState
- module.exports.getTopologyType = getTopologyType;
- module.exports.emitServerDescriptionChanged = emitServerDescriptionChanged;
- module.exports.emitTopologyDescriptionChanged = emitTopologyDescriptionChanged;
- module.exports.cloneOptions = cloneOptions;
- module.exports.assign = assign;
- module.exports.createClientInfo = createClientInfo;
- module.exports.clone = clone;
|