mirror of
				https://scm.univ-tours.fr/22107988t/rappaurio-sae501_502.git
				synced 2025-10-25 01:25:01 +02:00 
			
		
		
		
	Started with database connection
This commit is contained in:
		
							
								
								
									
										941
									
								
								app/node_modules/mysql2/lib/connection.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										941
									
								
								app/node_modules/mysql2/lib/connection.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,941 @@ | ||||
| // This file was modified by Oracle on June 1, 2021. | ||||
| // The changes involve new logic to handle an additional ERR Packet sent by | ||||
| // the MySQL server when the connection is closed unexpectedly. | ||||
| // Modifications copyright (c) 2021, Oracle and/or its affiliates. | ||||
|  | ||||
| // This file was modified by Oracle on June 17, 2021. | ||||
| // The changes involve logic to ensure the socket connection is closed when | ||||
| // there is a fatal error. | ||||
| // Modifications copyright (c) 2021, Oracle and/or its affiliates. | ||||
|  | ||||
| // This file was modified by Oracle on September 21, 2021. | ||||
| // The changes involve passing additional authentication factor passwords | ||||
| // to the ChangeUser Command instance. | ||||
| // Modifications copyright (c) 2021, Oracle and/or its affiliates. | ||||
|  | ||||
| 'use strict'; | ||||
|  | ||||
| const Net = require('net'); | ||||
| const Tls = require('tls'); | ||||
| const Timers = require('timers'); | ||||
| const EventEmitter = require('events').EventEmitter; | ||||
| const Readable = require('stream').Readable; | ||||
| const Queue = require('denque'); | ||||
| const SqlString = require('sqlstring'); | ||||
| const LRU = require('lru-cache').default; | ||||
|  | ||||
| const PacketParser = require('./packet_parser.js'); | ||||
| const Packets = require('./packets/index.js'); | ||||
| const Commands = require('./commands/index.js'); | ||||
| const ConnectionConfig = require('./connection_config.js'); | ||||
| const CharsetToEncoding = require('./constants/charset_encodings.js'); | ||||
|  | ||||
| let _connectionId = 0; | ||||
|  | ||||
| let convertNamedPlaceholders = null; | ||||
|  | ||||
| class Connection extends EventEmitter { | ||||
|   constructor(opts) { | ||||
|     super(); | ||||
|     this.config = opts.config; | ||||
|     // TODO: fill defaults | ||||
|     // if no params, connect to /var/lib/mysql/mysql.sock ( /tmp/mysql.sock on OSX ) | ||||
|     // if host is given, connect to host:3306 | ||||
|     // TODO: use `/usr/local/mysql/bin/mysql_config --socket` output? as default socketPath | ||||
|     // if there is no host/port and no socketPath parameters? | ||||
|     if (!opts.config.stream) { | ||||
|       if (opts.config.socketPath) { | ||||
|         this.stream = Net.connect(opts.config.socketPath); | ||||
|       } else { | ||||
|         this.stream = Net.connect( | ||||
|           opts.config.port, | ||||
|           opts.config.host | ||||
|         ); | ||||
|  | ||||
|         // Optionally enable keep-alive on the socket. | ||||
|         if (this.config.enableKeepAlive) { | ||||
|           this.stream.on('connect', () => { | ||||
|             this.stream.setKeepAlive(true, this.config.keepAliveInitialDelay); | ||||
|           }); | ||||
|         } | ||||
|  | ||||
|         // Enable TCP_NODELAY flag. This is needed so that the network packets | ||||
|         // are sent immediately to the server | ||||
|         this.stream.setNoDelay(true); | ||||
|       } | ||||
|       // if stream is a function, treat it as "stream agent / factory" | ||||
|     } else if (typeof opts.config.stream === 'function')  { | ||||
|       this.stream = opts.config.stream(opts); | ||||
|     } else { | ||||
|       this.stream = opts.config.stream; | ||||
|     } | ||||
|  | ||||
|     this._internalId = _connectionId++; | ||||
|     this._commands = new Queue(); | ||||
|     this._command = null; | ||||
|     this._paused = false; | ||||
|     this._paused_packets = new Queue(); | ||||
|     this._statements = new LRU({ | ||||
|       max: this.config.maxPreparedStatements, | ||||
|       dispose: function(statement) { | ||||
|         statement.close(); | ||||
|       } | ||||
|     }); | ||||
|     this.serverCapabilityFlags = 0; | ||||
|     this.authorized = false; | ||||
|     this.sequenceId = 0; | ||||
|     this.compressedSequenceId = 0; | ||||
|     this.threadId = null; | ||||
|     this._handshakePacket = null; | ||||
|     this._fatalError = null; | ||||
|     this._protocolError = null; | ||||
|     this._outOfOrderPackets = []; | ||||
|     this.clientEncoding = CharsetToEncoding[this.config.charsetNumber]; | ||||
|     this.stream.on('error', this._handleNetworkError.bind(this)); | ||||
|     // see https://gist.github.com/khoomeister/4985691#use-that-instead-of-bind | ||||
|     this.packetParser = new PacketParser(p => { | ||||
|       this.handlePacket(p); | ||||
|     }); | ||||
|     this.stream.on('data', data => { | ||||
|       if (this.connectTimeout) { | ||||
|         Timers.clearTimeout(this.connectTimeout); | ||||
|         this.connectTimeout = null; | ||||
|       } | ||||
|       this.packetParser.execute(data); | ||||
|     }); | ||||
|     this.stream.on('end', () => { | ||||
|       // emit the end event so that the pooled connection can close the connection | ||||
|       this.emit('end'); | ||||
|     }); | ||||
|     this.stream.on('close', () => { | ||||
|       // we need to set this flag everywhere where we want connection to close | ||||
|       if (this._closing) { | ||||
|         return; | ||||
|       } | ||||
|       if (!this._protocolError) { | ||||
|         // no particular error message before disconnect | ||||
|         this._protocolError = new Error( | ||||
|           'Connection lost: The server closed the connection.' | ||||
|         ); | ||||
|         this._protocolError.fatal = true; | ||||
|         this._protocolError.code = 'PROTOCOL_CONNECTION_LOST'; | ||||
|       } | ||||
|       this._notifyError(this._protocolError); | ||||
|     }); | ||||
|     let handshakeCommand; | ||||
|     if (!this.config.isServer) { | ||||
|       handshakeCommand = new Commands.ClientHandshake(this.config.clientFlags); | ||||
|       handshakeCommand.on('end', () => { | ||||
|         // this happens when handshake finishes early either because there was | ||||
|         // some fatal error or the server sent an error packet instead of | ||||
|         // an hello packet (for example, 'Too many connections' error) | ||||
|         if (!handshakeCommand.handshake || this._fatalError || this._protocolError) { | ||||
|           return; | ||||
|         } | ||||
|         this._handshakePacket = handshakeCommand.handshake; | ||||
|         this.threadId = handshakeCommand.handshake.connectionId; | ||||
|         this.emit('connect', handshakeCommand.handshake); | ||||
|       }); | ||||
|       handshakeCommand.on('error', err => { | ||||
|         this._closing = true; | ||||
|         this._notifyError(err); | ||||
|       }); | ||||
|       this.addCommand(handshakeCommand); | ||||
|     } | ||||
|     // in case there was no initial handshake but we need to read sting, assume it utf-8 | ||||
|     // most common example: "Too many connections" error ( packet is sent immediately on connection attempt, we don't know server encoding yet) | ||||
|     // will be overwritten with actual encoding value as soon as server handshake packet is received | ||||
|     this.serverEncoding = 'utf8'; | ||||
|     if (this.config.connectTimeout) { | ||||
|       const timeoutHandler = this._handleTimeoutError.bind(this); | ||||
|       this.connectTimeout = Timers.setTimeout( | ||||
|         timeoutHandler, | ||||
|         this.config.connectTimeout | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   promise(promiseImpl) { | ||||
|     const PromiseConnection = require('../promise').PromiseConnection; | ||||
|     return new PromiseConnection(this, promiseImpl); | ||||
|   } | ||||
|  | ||||
|   _addCommandClosedState(cmd) { | ||||
|     const err = new Error( | ||||
|       "Can't add new command when connection is in closed state" | ||||
|     ); | ||||
|     err.fatal = true; | ||||
|     if (cmd.onResult) { | ||||
|       cmd.onResult(err); | ||||
|     } else { | ||||
|       this.emit('error', err); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   _handleFatalError(err) { | ||||
|     err.fatal = true; | ||||
|     // stop receiving packets | ||||
|     this.stream.removeAllListeners('data'); | ||||
|     this.addCommand = this._addCommandClosedState; | ||||
|     this.write = () => { | ||||
|       this.emit('error', new Error("Can't write in closed state")); | ||||
|     }; | ||||
|     this._notifyError(err); | ||||
|     this._fatalError = err; | ||||
|   } | ||||
|  | ||||
|   _handleNetworkError(err) { | ||||
|     if (this.connectTimeout) { | ||||
|       Timers.clearTimeout(this.connectTimeout); | ||||
|       this.connectTimeout = null; | ||||
|     } | ||||
|     // Do not throw an error when a connection ends with a RST,ACK packet | ||||
|     if (err.code === 'ECONNRESET' && this._closing) { | ||||
|       return; | ||||
|     } | ||||
|     this._handleFatalError(err); | ||||
|   } | ||||
|  | ||||
|   _handleTimeoutError() { | ||||
|     if (this.connectTimeout) { | ||||
|       Timers.clearTimeout(this.connectTimeout); | ||||
|       this.connectTimeout = null; | ||||
|     } | ||||
|     this.stream.destroy && this.stream.destroy(); | ||||
|     const err = new Error('connect ETIMEDOUT'); | ||||
|     err.errorno = 'ETIMEDOUT'; | ||||
|     err.code = 'ETIMEDOUT'; | ||||
|     err.syscall = 'connect'; | ||||
|     this._handleNetworkError(err); | ||||
|   } | ||||
|  | ||||
|   // notify all commands in the queue and bubble error as connection "error" | ||||
|   // called on stream error or unexpected termination | ||||
|   _notifyError(err) { | ||||
|     if (this.connectTimeout) { | ||||
|       Timers.clearTimeout(this.connectTimeout); | ||||
|       this.connectTimeout = null; | ||||
|     } | ||||
|     // prevent from emitting 'PROTOCOL_CONNECTION_LOST' after EPIPE or ECONNRESET | ||||
|     if (this._fatalError) { | ||||
|       return; | ||||
|     } | ||||
|     let command; | ||||
|     // if there is no active command, notify connection | ||||
|     // if there are commands and all of them have callbacks, pass error via callback | ||||
|     let bubbleErrorToConnection = !this._command; | ||||
|     if (this._command && this._command.onResult) { | ||||
|       this._command.onResult(err); | ||||
|       this._command = null; | ||||
|       // connection handshake is special because we allow it to be implicit | ||||
|       // if error happened during handshake, but there are others commands in queue | ||||
|       // then bubble error to other commands and not to connection | ||||
|     } else if ( | ||||
|       !( | ||||
|         this._command && | ||||
|         this._command.constructor === Commands.ClientHandshake && | ||||
|         this._commands.length > 0 | ||||
|       ) | ||||
|     ) { | ||||
|       bubbleErrorToConnection = true; | ||||
|     } | ||||
|     while ((command = this._commands.shift())) { | ||||
|       if (command.onResult) { | ||||
|         command.onResult(err); | ||||
|       } else { | ||||
|         bubbleErrorToConnection = true; | ||||
|       } | ||||
|     } | ||||
|     // notify connection if some comands in the queue did not have callbacks | ||||
|     // or if this is pool connection ( so it can be removed from pool ) | ||||
|     if (bubbleErrorToConnection || this._pool) { | ||||
|       this.emit('error', err); | ||||
|     } | ||||
|     // close connection after emitting the event in case of a fatal error | ||||
|     if (err.fatal) { | ||||
|       this.close(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   write(buffer) { | ||||
|     const result = this.stream.write(buffer, err => { | ||||
|       if (err) { | ||||
|         this._handleNetworkError(err); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     if (!result) { | ||||
|       this.stream.emit('pause'); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // http://dev.mysql.com/doc/internals/en/sequence-id.html | ||||
|   // | ||||
|   // The sequence-id is incremented with each packet and may wrap around. | ||||
|   // It starts at 0 and is reset to 0 when a new command | ||||
|   // begins in the Command Phase. | ||||
|   // http://dev.mysql.com/doc/internals/en/example-several-mysql-packets.html | ||||
|   _resetSequenceId() { | ||||
|     this.sequenceId = 0; | ||||
|     this.compressedSequenceId = 0; | ||||
|   } | ||||
|  | ||||
|   _bumpCompressedSequenceId(numPackets) { | ||||
|     this.compressedSequenceId += numPackets; | ||||
|     this.compressedSequenceId %= 256; | ||||
|   } | ||||
|  | ||||
|   _bumpSequenceId(numPackets) { | ||||
|     this.sequenceId += numPackets; | ||||
|     this.sequenceId %= 256; | ||||
|   } | ||||
|  | ||||
|   writePacket(packet) { | ||||
|     const MAX_PACKET_LENGTH = 16777215; | ||||
|     const length = packet.length(); | ||||
|     let chunk, offset, header; | ||||
|     if (length < MAX_PACKET_LENGTH) { | ||||
|       packet.writeHeader(this.sequenceId); | ||||
|       if (this.config.debug) { | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.log( | ||||
|           `${this._internalId} ${this.connectionId} <== ${this._command._commandName}#${this._command.stateName()}(${[this.sequenceId, packet._name, packet.length()].join(',')})` | ||||
|         ); | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.log( | ||||
|           `${this._internalId} ${this.connectionId} <== ${packet.buffer.toString('hex')}` | ||||
|         ); | ||||
|       } | ||||
|       this._bumpSequenceId(1); | ||||
|       this.write(packet.buffer); | ||||
|     } else { | ||||
|       if (this.config.debug) { | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.log( | ||||
|           `${this._internalId} ${this.connectionId} <== Writing large packet, raw content not written:` | ||||
|         ); | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.log( | ||||
|           `${this._internalId} ${this.connectionId} <== ${this._command._commandName}#${this._command.stateName()}(${[this.sequenceId, packet._name, packet.length()].join(',')})` | ||||
|         ); | ||||
|       } | ||||
|       for (offset = 4; offset < 4 + length; offset += MAX_PACKET_LENGTH) { | ||||
|         chunk = packet.buffer.slice(offset, offset + MAX_PACKET_LENGTH); | ||||
|         if (chunk.length === MAX_PACKET_LENGTH) { | ||||
|           header = Buffer.from([0xff, 0xff, 0xff, this.sequenceId]); | ||||
|         } else { | ||||
|           header = Buffer.from([ | ||||
|             chunk.length & 0xff, | ||||
|             (chunk.length >> 8) & 0xff, | ||||
|             (chunk.length >> 16) & 0xff, | ||||
|             this.sequenceId | ||||
|           ]); | ||||
|         } | ||||
|         this._bumpSequenceId(1); | ||||
|         this.write(header); | ||||
|         this.write(chunk); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // 0.11+ environment | ||||
|   startTLS(onSecure) { | ||||
|     if (this.config.debug) { | ||||
|       // eslint-disable-next-line no-console | ||||
|       console.log('Upgrading connection to TLS'); | ||||
|     } | ||||
|     const secureContext = Tls.createSecureContext({ | ||||
|       ca: this.config.ssl.ca, | ||||
|       cert: this.config.ssl.cert, | ||||
|       ciphers: this.config.ssl.ciphers, | ||||
|       key: this.config.ssl.key, | ||||
|       passphrase: this.config.ssl.passphrase, | ||||
|       minVersion: this.config.ssl.minVersion, | ||||
|       maxVersion: this.config.ssl.maxVersion | ||||
|     }); | ||||
|     const rejectUnauthorized = this.config.ssl.rejectUnauthorized; | ||||
|     const verifyIdentity = this.config.ssl.verifyIdentity; | ||||
|     const servername = this.config.host; | ||||
|  | ||||
|     let secureEstablished = false; | ||||
|     this.stream.removeAllListeners('data'); | ||||
|     const secureSocket = Tls.connect({ | ||||
|       rejectUnauthorized, | ||||
|       requestCert: rejectUnauthorized, | ||||
|       secureContext, | ||||
|       isServer: false, | ||||
|       socket: this.stream, | ||||
|       servername | ||||
|     }, () => { | ||||
|       secureEstablished = true; | ||||
|       if (rejectUnauthorized) { | ||||
|         if (typeof servername === 'string' && verifyIdentity) { | ||||
|           const cert = secureSocket.getPeerCertificate(true); | ||||
|           const serverIdentityCheckError = Tls.checkServerIdentity(servername, cert); | ||||
|           if (serverIdentityCheckError) { | ||||
|             onSecure(serverIdentityCheckError); | ||||
|             return; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       onSecure(); | ||||
|     }); | ||||
|     // error handler for secure socket | ||||
|     secureSocket.on('error', err => { | ||||
|       if (secureEstablished) { | ||||
|         this._handleNetworkError(err); | ||||
|       } else { | ||||
|         onSecure(err); | ||||
|       } | ||||
|     }); | ||||
|     secureSocket.on('data', data => { | ||||
|       this.packetParser.execute(data); | ||||
|     }); | ||||
|     this.write = buffer => secureSocket.write(buffer); | ||||
|   } | ||||
|  | ||||
|   protocolError(message, code) { | ||||
|     // Starting with MySQL 8.0.24, if the client closes the connection | ||||
|     // unexpectedly, the server will send a last ERR Packet, which we can | ||||
|     // safely ignore. | ||||
|     // https://dev.mysql.com/worklog/task/?id=12999 | ||||
|     if (this._closing) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     const err = new Error(message); | ||||
|     err.fatal = true; | ||||
|     err.code = code || 'PROTOCOL_ERROR'; | ||||
|     this.emit('error', err); | ||||
|   } | ||||
|    | ||||
|   get fatalError() { | ||||
|     return this._fatalError; | ||||
|   } | ||||
|  | ||||
|   handlePacket(packet) { | ||||
|     if (this._paused) { | ||||
|       this._paused_packets.push(packet); | ||||
|       return; | ||||
|     } | ||||
|     if (this.config.debug) { | ||||
|       if (packet) { | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.log( | ||||
|           ` raw: ${packet.buffer | ||||
|             .slice(packet.offset, packet.offset + packet.length()) | ||||
|             .toString('hex')}` | ||||
|         ); | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.trace(); | ||||
|         const commandName = this._command | ||||
|           ? this._command._commandName | ||||
|           : '(no command)'; | ||||
|         const stateName = this._command | ||||
|           ? this._command.stateName() | ||||
|           : '(no command)'; | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.log( | ||||
|           `${this._internalId} ${this.connectionId} ==> ${commandName}#${stateName}(${[packet.sequenceId, packet.type(), packet.length()].join(',')})` | ||||
|         ); | ||||
|       } | ||||
|     } | ||||
|     if (!this._command) { | ||||
|       const marker = packet.peekByte(); | ||||
|       // If it's an Err Packet, we should use it. | ||||
|       if (marker === 0xff) { | ||||
|         const error = Packets.Error.fromPacket(packet); | ||||
|         this.protocolError(error.message, error.code); | ||||
|       } else { | ||||
|         // Otherwise, it means it's some other unexpected packet. | ||||
|         this.protocolError( | ||||
|           'Unexpected packet while no commands in the queue', | ||||
|           'PROTOCOL_UNEXPECTED_PACKET' | ||||
|         ); | ||||
|       } | ||||
|       this.close(); | ||||
|       return; | ||||
|     } | ||||
|     if (packet) { | ||||
|       // Note: when server closes connection due to inactivity, Err packet ER_CLIENT_INTERACTION_TIMEOUT from MySQL 8.0.24, sequenceId will be 0 | ||||
|       if (this.sequenceId !== packet.sequenceId) { | ||||
|         const err = new Error( | ||||
|           `Warning: got packets out of order. Expected ${this.sequenceId} but received ${packet.sequenceId}` | ||||
|         ); | ||||
|         err.expected = this.sequenceId; | ||||
|         err.received = packet.sequenceId; | ||||
|         this.emit('warn', err); // REVIEW | ||||
|         // eslint-disable-next-line no-console | ||||
|         console.error(err.message); | ||||
|       } | ||||
|       this._bumpSequenceId(packet.numPackets); | ||||
|     } | ||||
|     try { | ||||
|       if (this._fatalError) { | ||||
|         // skip remaining packets after client is in the error state | ||||
|         return; | ||||
|       } | ||||
|       const done = this._command.execute(packet, this); | ||||
|       if (done) { | ||||
|         this._command = this._commands.shift(); | ||||
|         if (this._command) { | ||||
|           this.sequenceId = 0; | ||||
|           this.compressedSequenceId = 0; | ||||
|           this.handlePacket(); | ||||
|         } | ||||
|       } | ||||
|     } catch (err) { | ||||
|       this._handleFatalError(err); | ||||
|       this.stream.destroy(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   addCommand(cmd) { | ||||
|     // this.compressedSequenceId = 0; | ||||
|     // this.sequenceId = 0; | ||||
|     if (this.config.debug) { | ||||
|       const commandName = cmd.constructor.name; | ||||
|       // eslint-disable-next-line no-console | ||||
|       console.log(`Add command: ${commandName}`); | ||||
|       cmd._commandName = commandName; | ||||
|     } | ||||
|     if (!this._command) { | ||||
|       this._command = cmd; | ||||
|       this.handlePacket(); | ||||
|     } else { | ||||
|       this._commands.push(cmd); | ||||
|     } | ||||
|     return cmd; | ||||
|   } | ||||
|  | ||||
|   format(sql, values) { | ||||
|     if (typeof this.config.queryFormat === 'function') { | ||||
|       return this.config.queryFormat.call( | ||||
|         this, | ||||
|         sql, | ||||
|         values, | ||||
|         this.config.timezone | ||||
|       ); | ||||
|     } | ||||
|     const opts = { | ||||
|       sql: sql, | ||||
|       values: values | ||||
|     }; | ||||
|     this._resolveNamedPlaceholders(opts); | ||||
|     return SqlString.format( | ||||
|       opts.sql, | ||||
|       opts.values, | ||||
|       this.config.stringifyObjects, | ||||
|       this.config.timezone | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   escape(value) { | ||||
|     return SqlString.escape(value, false, this.config.timezone); | ||||
|   } | ||||
|  | ||||
|   escapeId(value) { | ||||
|     return SqlString.escapeId(value, false); | ||||
|   } | ||||
|  | ||||
|   raw(sql) { | ||||
|     return SqlString.raw(sql); | ||||
|   } | ||||
|  | ||||
|   _resolveNamedPlaceholders(options) { | ||||
|     let unnamed; | ||||
|     if (this.config.namedPlaceholders || options.namedPlaceholders) { | ||||
|       if (Array.isArray(options.values)) { | ||||
|         // if an array is provided as the values, assume the conversion is not necessary. | ||||
|         // this allows the usage of unnamed placeholders even if the namedPlaceholders flag is enabled. | ||||
|         return | ||||
|       } | ||||
|       if (convertNamedPlaceholders === null) { | ||||
|         convertNamedPlaceholders = require('named-placeholders')(); | ||||
|       } | ||||
|       unnamed = convertNamedPlaceholders(options.sql, options.values); | ||||
|       options.sql = unnamed[0]; | ||||
|       options.values = unnamed[1]; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   query(sql, values, cb) { | ||||
|     let cmdQuery; | ||||
|     if (sql.constructor === Commands.Query) { | ||||
|       cmdQuery = sql; | ||||
|     } else { | ||||
|       cmdQuery = Connection.createQuery(sql, values, cb, this.config); | ||||
|     } | ||||
|     this._resolveNamedPlaceholders(cmdQuery); | ||||
|     const rawSql = this.format(cmdQuery.sql, cmdQuery.values !== undefined ? cmdQuery.values : []); | ||||
|     cmdQuery.sql = rawSql; | ||||
|     return this.addCommand(cmdQuery); | ||||
|   } | ||||
|  | ||||
|   pause() { | ||||
|     this._paused = true; | ||||
|     this.stream.pause(); | ||||
|   } | ||||
|  | ||||
|   resume() { | ||||
|     let packet; | ||||
|     this._paused = false; | ||||
|     while ((packet = this._paused_packets.shift())) { | ||||
|       this.handlePacket(packet); | ||||
|       // don't resume if packet handler paused connection | ||||
|       if (this._paused) { | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|     this.stream.resume(); | ||||
|   } | ||||
|  | ||||
|   // TODO: named placeholders support | ||||
|   prepare(options, cb) { | ||||
|     if (typeof options === 'string') { | ||||
|       options = { sql: options }; | ||||
|     } | ||||
|     return this.addCommand(new Commands.Prepare(options, cb)); | ||||
|   } | ||||
|  | ||||
|   unprepare(sql) { | ||||
|     let options = {}; | ||||
|     if (typeof sql === 'object') { | ||||
|       options = sql; | ||||
|     } else { | ||||
|       options.sql = sql; | ||||
|     } | ||||
|     const key = Connection.statementKey(options); | ||||
|     const stmt = this._statements.get(key); | ||||
|     if (stmt) { | ||||
|       this._statements.delete(key); | ||||
|       stmt.close(); | ||||
|     } | ||||
|     return stmt; | ||||
|   } | ||||
|  | ||||
|   execute(sql, values, cb) { | ||||
|     let options = { | ||||
|       infileStreamFactory: this.config.infileStreamFactory | ||||
|     }; | ||||
|     if (typeof sql === 'object') { | ||||
|       // execute(options, cb) | ||||
|       options = { | ||||
|         ...options, | ||||
|         ...sql | ||||
|       }; | ||||
|       if (typeof values === 'function') { | ||||
|         cb = values; | ||||
|       } else { | ||||
|         options.values = options.values || values; | ||||
|       } | ||||
|     } else if (typeof values === 'function') { | ||||
|       // execute(sql, cb) | ||||
|       cb = values; | ||||
|       options.sql = sql; | ||||
|       options.values = undefined; | ||||
|     } else { | ||||
|       // execute(sql, values, cb) | ||||
|       options.sql = sql; | ||||
|       options.values = values; | ||||
|     } | ||||
|     this._resolveNamedPlaceholders(options); | ||||
|     // check for values containing undefined | ||||
|     if (options.values) { | ||||
|       //If namedPlaceholder is not enabled and object is passed as bind parameters | ||||
|       if (!Array.isArray(options.values)) { | ||||
|         throw new TypeError( | ||||
|           'Bind parameters must be array if namedPlaceholders parameter is not enabled' | ||||
|         ); | ||||
|       } | ||||
|       options.values.forEach(val => { | ||||
|         //If namedPlaceholder is not enabled and object is passed as bind parameters | ||||
|         if (!Array.isArray(options.values)) { | ||||
|           throw new TypeError( | ||||
|             'Bind parameters must be array if namedPlaceholders parameter is not enabled' | ||||
|           ); | ||||
|         } | ||||
|         if (val === undefined) { | ||||
|           throw new TypeError( | ||||
|             'Bind parameters must not contain undefined. To pass SQL NULL specify JS null' | ||||
|           ); | ||||
|         } | ||||
|         if (typeof val === 'function') { | ||||
|           throw new TypeError( | ||||
|             'Bind parameters must not contain function(s). To pass the body of a function as a string call .toString() first' | ||||
|           ); | ||||
|         } | ||||
|       }); | ||||
|     } | ||||
|     const executeCommand = new Commands.Execute(options, cb); | ||||
|     const prepareCommand = new Commands.Prepare(options, (err, stmt) => { | ||||
|       if (err) { | ||||
|         // skip execute command if prepare failed, we have main | ||||
|         // combined callback here | ||||
|         executeCommand.start = function() { | ||||
|           return null; | ||||
|         }; | ||||
|         if (cb) { | ||||
|           cb(err); | ||||
|         } else { | ||||
|           executeCommand.emit('error', err); | ||||
|         } | ||||
|         executeCommand.emit('end'); | ||||
|         return; | ||||
|       } | ||||
|       executeCommand.statement = stmt; | ||||
|     }); | ||||
|     this.addCommand(prepareCommand); | ||||
|     this.addCommand(executeCommand); | ||||
|     return executeCommand; | ||||
|   } | ||||
|  | ||||
|   changeUser(options, callback) { | ||||
|     if (!callback && typeof options === 'function') { | ||||
|       callback = options; | ||||
|       options = {}; | ||||
|     } | ||||
|     const charsetNumber = options.charset | ||||
|       ? ConnectionConfig.getCharsetNumber(options.charset) | ||||
|       : this.config.charsetNumber; | ||||
|     return this.addCommand( | ||||
|       new Commands.ChangeUser( | ||||
|         { | ||||
|           user: options.user || this.config.user, | ||||
|           // for the purpose of multi-factor authentication, or not, the main | ||||
|           // password (used for the 1st authentication factor) can also be | ||||
|           // provided via the "password1" option | ||||
|           password: options.password || options.password1 || this.config.password || this.config.password1, | ||||
|           password2: options.password2 || this.config.password2, | ||||
|           password3: options.password3 || this.config.password3, | ||||
|           passwordSha1: options.passwordSha1 || this.config.passwordSha1, | ||||
|           database: options.database || this.config.database, | ||||
|           timeout: options.timeout, | ||||
|           charsetNumber: charsetNumber, | ||||
|           currentConfig: this.config | ||||
|         }, | ||||
|         err => { | ||||
|           if (err) { | ||||
|             err.fatal = true; | ||||
|           } | ||||
|           if (callback) { | ||||
|             callback(err); | ||||
|           } | ||||
|         } | ||||
|       ) | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   // transaction helpers | ||||
|   beginTransaction(cb) { | ||||
|     return this.query('START TRANSACTION', cb); | ||||
|   } | ||||
|  | ||||
|   commit(cb) { | ||||
|     return this.query('COMMIT', cb); | ||||
|   } | ||||
|  | ||||
|   rollback(cb) { | ||||
|     return this.query('ROLLBACK', cb); | ||||
|   } | ||||
|  | ||||
|   ping(cb) { | ||||
|     return this.addCommand(new Commands.Ping(cb)); | ||||
|   } | ||||
|  | ||||
|   _registerSlave(opts, cb) { | ||||
|     return this.addCommand(new Commands.RegisterSlave(opts, cb)); | ||||
|   } | ||||
|  | ||||
|   _binlogDump(opts, cb) { | ||||
|     return this.addCommand(new Commands.BinlogDump(opts, cb)); | ||||
|   } | ||||
|  | ||||
|   // currently just alias to close | ||||
|   destroy() { | ||||
|     this.close(); | ||||
|   } | ||||
|  | ||||
|   close() { | ||||
|     if (this.connectTimeout) { | ||||
|       Timers.clearTimeout(this.connectTimeout); | ||||
|       this.connectTimeout = null; | ||||
|     } | ||||
|     this._closing = true; | ||||
|     this.stream.end(); | ||||
|     this.addCommand = this._addCommandClosedState; | ||||
|   } | ||||
|  | ||||
|   createBinlogStream(opts) { | ||||
|     // TODO: create proper stream class | ||||
|     // TODO: use through2 | ||||
|     let test = 1; | ||||
|     const stream = new Readable({ objectMode: true }); | ||||
|     stream._read = function() { | ||||
|       return { | ||||
|         data: test++ | ||||
|       }; | ||||
|     }; | ||||
|     this._registerSlave(opts, () => { | ||||
|       const dumpCmd = this._binlogDump(opts); | ||||
|       dumpCmd.on('event', ev => { | ||||
|         stream.push(ev); | ||||
|       }); | ||||
|       dumpCmd.on('eof', () => { | ||||
|         stream.push(null); | ||||
|         // if non-blocking, then close stream to prevent errors | ||||
|         if (opts.flags && opts.flags & 0x01) { | ||||
|           this.close(); | ||||
|         } | ||||
|       }); | ||||
|       // TODO: pipe errors as well | ||||
|     }); | ||||
|     return stream; | ||||
|   } | ||||
|  | ||||
|   connect(cb) { | ||||
|     if (!cb) { | ||||
|       return; | ||||
|     } | ||||
|     if (this._fatalError || this._protocolError) { | ||||
|       return cb(this._fatalError || this._protocolError); | ||||
|     } | ||||
|     if (this._handshakePacket) { | ||||
|       return cb(null, this); | ||||
|     } | ||||
|     let connectCalled = 0; | ||||
|     function callbackOnce(isErrorHandler) { | ||||
|       return function(param) { | ||||
|         if (!connectCalled) { | ||||
|           if (isErrorHandler) { | ||||
|             cb(param); | ||||
|           } else { | ||||
|             cb(null, param); | ||||
|           } | ||||
|         } | ||||
|         connectCalled = 1; | ||||
|       }; | ||||
|     } | ||||
|     this.once('error', callbackOnce(true)); | ||||
|     this.once('connect', callbackOnce(false)); | ||||
|   } | ||||
|  | ||||
|   // =================================== | ||||
|   // outgoing server connection methods | ||||
|   // =================================== | ||||
|   writeColumns(columns) { | ||||
|     this.writePacket(Packets.ResultSetHeader.toPacket(columns.length)); | ||||
|     columns.forEach(column => { | ||||
|       this.writePacket( | ||||
|         Packets.ColumnDefinition.toPacket(column, this.serverConfig.encoding) | ||||
|       ); | ||||
|     }); | ||||
|     this.writeEof(); | ||||
|   } | ||||
|  | ||||
|   // row is array of columns, not hash | ||||
|   writeTextRow(column) { | ||||
|     this.writePacket( | ||||
|       Packets.TextRow.toPacket(column, this.serverConfig.encoding) | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   writeBinaryRow(column) { | ||||
|     this.writePacket( | ||||
|       Packets.BinaryRow.toPacket(column, this.serverConfig.encoding) | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   writeTextResult(rows, columns, binary=false) { | ||||
|     this.writeColumns(columns); | ||||
|     rows.forEach(row => { | ||||
|       const arrayRow = new Array(columns.length); | ||||
|       columns.forEach(column => { | ||||
|         arrayRow.push(row[column.name]); | ||||
|       }); | ||||
|       if(binary) { | ||||
|         this.writeBinaryRow(arrayRow); | ||||
|       } | ||||
|       else this.writeTextRow(arrayRow); | ||||
|     }); | ||||
|     this.writeEof(); | ||||
|   } | ||||
|  | ||||
|   writeEof(warnings, statusFlags) { | ||||
|     this.writePacket(Packets.EOF.toPacket(warnings, statusFlags)); | ||||
|   } | ||||
|  | ||||
|   writeOk(args) { | ||||
|     if (!args) { | ||||
|       args = { affectedRows: 0 }; | ||||
|     } | ||||
|     this.writePacket(Packets.OK.toPacket(args, this.serverConfig.encoding)); | ||||
|   } | ||||
|  | ||||
|   writeError(args) { | ||||
|     // if we want to send error before initial hello was sent, use default encoding | ||||
|     const encoding = this.serverConfig ? this.serverConfig.encoding : 'cesu8'; | ||||
|     this.writePacket(Packets.Error.toPacket(args, encoding)); | ||||
|   } | ||||
|  | ||||
|   serverHandshake(args) { | ||||
|     this.serverConfig = args; | ||||
|     this.serverConfig.encoding = | ||||
|       CharsetToEncoding[this.serverConfig.characterSet]; | ||||
|     return this.addCommand(new Commands.ServerHandshake(args)); | ||||
|   } | ||||
|  | ||||
|   // =============================================================== | ||||
|   end(callback) { | ||||
|     if (this.config.isServer) { | ||||
|       this._closing = true; | ||||
|       const quitCmd = new EventEmitter(); | ||||
|       setImmediate(() => { | ||||
|         this.stream.end(); | ||||
|         quitCmd.emit('end'); | ||||
|       }); | ||||
|       return quitCmd; | ||||
|     } | ||||
|     // trigger error if more commands enqueued after end command | ||||
|     const quitCmd = this.addCommand(new Commands.Quit(callback)); | ||||
|     this.addCommand = this._addCommandClosedState; | ||||
|     return quitCmd; | ||||
|   } | ||||
|  | ||||
|   static createQuery(sql, values, cb, config) { | ||||
|     let options = { | ||||
|       rowsAsArray: config.rowsAsArray, | ||||
|       infileStreamFactory: config.infileStreamFactory | ||||
|     }; | ||||
|     if (typeof sql === 'object') { | ||||
|       // query(options, cb) | ||||
|       options = { | ||||
|         ...options, | ||||
|         ...sql | ||||
|       }; | ||||
|       if (typeof values === 'function') { | ||||
|         cb = values; | ||||
|       } else if (values !== undefined) { | ||||
|         options.values = values; | ||||
|       } | ||||
|     } else if (typeof values === 'function') { | ||||
|       // query(sql, cb) | ||||
|       cb = values; | ||||
|       options.sql = sql; | ||||
|       options.values = undefined; | ||||
|     } else { | ||||
|       // query(sql, values, cb) | ||||
|       options.sql = sql; | ||||
|       options.values = values; | ||||
|     } | ||||
|     return new Commands.Query(options, cb); | ||||
|   } | ||||
|  | ||||
|   static statementKey(options) { | ||||
|     return ( | ||||
|       `${typeof options.nestTables}/${options.nestTables}/${options.rowsAsArray}${options.sql}` | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| module.exports = Connection; | ||||
		Reference in New Issue
	
	Block a user