From 02576a62471a36696dab8ee3334202d2bb58c135 Mon Sep 17 00:00:00 2001 From: Ruy Adorno Date: Tue, 30 May 2023 16:28:49 -0400 Subject: [PATCH] fix: skip config validation when using connector This changeset fixes a problem in the original custom `connector` factory method implementation that required the `server` config property to be defined even though its value is ignored. Removing the validation when `config.options.connector` is defined fixes the problem. Ref: https://github.com/tediousjs/tedious/pull/1540 Fixes: https://github.com/tediousjs/tedious/issues/1541 Signed-off-by: Ruy Adorno --- src/connection.ts | 38 +++++++++++++++++++++++++++-------- test/unit/custom-connector.js | 37 ++++++++++++++++++++++++++++++++-- 2 files changed, 65 insertions(+), 10 deletions(-) diff --git a/src/connection.ts b/src/connection.ts index 4241efc85..ac236182e 100644 --- a/src/connection.ts +++ b/src/connection.ts @@ -1052,7 +1052,7 @@ class Connection extends EventEmitter { throw new TypeError('The "config" argument is required and must be of type Object.'); } - if (typeof config.server !== 'string') { + if (typeof config.server !== 'string' && !config.options!.connector) { throw new TypeError('The "config.server" property is required and must be of type string.'); } @@ -1343,8 +1343,15 @@ class Connection extends EventEmitter { if (typeof config.options.connector !== 'function') { throw new TypeError('The "config.options.connector" property must be a function.'); } + if (config.server) { + throw new Error('Server and connector are mutually exclusive, but ' + config.server + ' and a connector function were provided'); + } + if (config.options.port) { + throw new Error('Port and connector are mutually exclusive, but ' + config.options.port + ' and a connector function were provided'); + } this.config.options.connector = config.options.connector; + this.config.options.port = undefined; } if (config.options.cryptoCredentialsDetails !== undefined) { @@ -1900,7 +1907,10 @@ class Connection extends EventEmitter { initialiseConnection() { const signal = this.createConnectTimer(); - if (this.config.options.port) { + if (this.config.options.connector) { + // port and multiSubnetFailover are not used when using a custom connector + return this.connectOnPort(0, false, signal, this.config.options.connector); + } else if (this.config.options.port) { return this.connectOnPort(this.config.options.port, this.config.options.multiSubnetFailover, signal, this.config.options.connector); } else { return instanceLookup({ @@ -1995,7 +2005,11 @@ class Connection extends EventEmitter { this.socket = socket; this.closed = false; - this.debug.log('connected to ' + this.config.server + ':' + this.config.options.port); + + const connectedMsg = customConnector ? + 'connected via custom connector' : + 'connected to ' + this.config.server + ':' + this.config.options.port; + this.debug.log(connectedMsg); this.sendPreLogin(); this.transitionTo(this.STATE.SENT_PRELOGIN); @@ -2073,7 +2087,8 @@ class Connection extends EventEmitter { */ connectTimeout() { const message = `Failed to connect to ${this.config.server}${this.config.options.port ? `:${this.config.options.port}` : `\\${this.config.options.instanceName}`} in ${this.config.options.connectTimeout}ms`; - this.debug.log(message); + const customConnectorMessage = `Failed to connect using custom connector in ${this.config.options.connectTimeout}ms`; + this.debug.log(this.config.options.connector ? customConnectorMessage : message); this.emit('connect', new ConnectionError(message, 'ETIMEOUT')); this.connectTimer = undefined; this.dispatchEvent('connectTimeout'); @@ -2202,7 +2217,8 @@ class Connection extends EventEmitter { socketError(error: Error) { if (this.state === this.STATE.CONNECTING || this.state === this.STATE.SENT_TLSSSLNEGOTIATION) { const message = `Failed to connect to ${this.config.server}:${this.config.options.port} - ${error.message}`; - this.debug.log(message); + const customConnectorMessage = `Failed to connect using custom connector - ${error.message}`; + this.debug.log(this.config.options.connector ? customConnectorMessage : message); this.emit('connect', new ConnectionError(message, 'ESOCKET')); } else { const message = `Connection lost - ${error.message}`; @@ -2228,15 +2244,21 @@ class Connection extends EventEmitter { * @private */ socketClose() { - this.debug.log('connection to ' + this.config.server + ':' + this.config.options.port + ' closed'); + const message = 'connection to ' + this.config.server + ':' + this.config.options.port + ' closed'; + const customConnectorMessage = 'connection closed'; + this.debug.log(this.config.options.connector ? customConnectorMessage : message); if (this.state === this.STATE.REROUTING) { - this.debug.log('Rerouting to ' + this.routingData!.server + ':' + this.routingData!.port); + const message = 'Rerouting to ' + this.routingData!.server + ':' + this.routingData!.port; + const customConnectorMessage = 'Rerouting'; + this.debug.log(this.config.options.connector ? customConnectorMessage : message); this.dispatchEvent('reconnect'); } else if (this.state === this.STATE.TRANSIENT_FAILURE_RETRY) { const server = this.routingData ? this.routingData.server : this.config.server; const port = this.routingData ? this.routingData.port : this.config.options.port; - this.debug.log('Retry after transient failure connecting to ' + server + ':' + port); + const message = 'Retry after transient failure connecting to ' + server + ':' + port; + const customConnectorMessage = 'Retry after transient failure connecting'; + this.debug.log(this.config.options.connector ? customConnectorMessage : message); this.dispatchEvent('retry'); } else { diff --git a/test/unit/custom-connector.js b/test/unit/custom-connector.js index 36e49d19d..cbe1b26c8 100644 --- a/test/unit/custom-connector.js +++ b/test/unit/custom-connector.js @@ -28,7 +28,6 @@ describe('custom connector', function() { const host = server.address().address; const port = server.address().port; const connection = new Connection({ - server: host, options: { connector: async () => { customConnectorCalled = true; @@ -37,7 +36,6 @@ describe('custom connector', function() { port, }); }, - port }, }); @@ -59,4 +57,39 @@ describe('custom connector', function() { connection.connect(); }); + + it('should only accept functions', function(done) { + assert.throws(() => { + new Connection({ + options: { + connector: 'foo', + }, + }); + }, Error, 'The "config.options.connector" property must be a function.'); + done(); + }); + + it('should not allow setting both server and connector options', function(done) { + assert.throws(() => { + new Connection({ + server: '0.0.0.0', + options: { + connector: async () => {}, + }, + }); + }, Error, 'Server and connector are mutually exclusive, but 0.0.0.0 and a connector function were provided'); + done(); + }); + + it('should not allow setting both port and connector options', function(done) { + assert.throws(() => { + new Connection({ + options: { + connector: async () => {}, + port: 8080, + }, + }); + }, Error, 'Port and connector are mutually exclusive, but 8080 and a connector function were provided'); + done(); + }); });