You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
311 lines
8.0 KiB
JavaScript
311 lines
8.0 KiB
JavaScript
2 years ago
|
/*
|
||
|
* Copyright DataStax, Inc.
|
||
|
*
|
||
|
* 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 events = require('events');
|
||
|
|
||
|
const types = require('./types');
|
||
|
const utils = require('./utils.js');
|
||
|
const FrameHeader = types.FrameHeader;
|
||
|
|
||
|
/**
|
||
|
* Contains the logic to write all the different types to the frame.
|
||
|
*/
|
||
|
class FrameWriter {
|
||
|
/**
|
||
|
* Creates a new instance of FrameWriter.
|
||
|
* @param {Number} opcode
|
||
|
*/
|
||
|
constructor(opcode) {
|
||
|
if (!opcode) {
|
||
|
throw new Error('Opcode not provided');
|
||
|
}
|
||
|
this.buffers = [];
|
||
|
this.opcode = opcode;
|
||
|
this.bodyLength = 0;
|
||
|
}
|
||
|
|
||
|
add(buf) {
|
||
|
this.buffers.push(buf);
|
||
|
this.bodyLength += buf.length;
|
||
|
}
|
||
|
|
||
|
writeShort(num) {
|
||
|
const buf = utils.allocBufferUnsafe(2);
|
||
|
buf.writeUInt16BE(num, 0);
|
||
|
this.add(buf);
|
||
|
}
|
||
|
|
||
|
writeInt(num) {
|
||
|
const buf = utils.allocBufferUnsafe(4);
|
||
|
buf.writeInt32BE(num, 0);
|
||
|
this.add(buf);
|
||
|
}
|
||
|
|
||
|
/** @param {Long} num */
|
||
|
writeLong(num) {
|
||
|
this.add(types.Long.toBuffer(num));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Writes bytes according to Cassandra <int byteLength><bytes>
|
||
|
* @param {Buffer|null|types.unset} bytes
|
||
|
*/
|
||
|
writeBytes(bytes) {
|
||
|
if (bytes === null) {
|
||
|
//Only the length buffer containing -1
|
||
|
this.writeInt(-1);
|
||
|
return;
|
||
|
}
|
||
|
if (bytes === types.unset) {
|
||
|
this.writeInt(-2);
|
||
|
return;
|
||
|
}
|
||
|
//Add the length buffer
|
||
|
this.writeInt(bytes.length);
|
||
|
//Add the actual buffer
|
||
|
this.add(bytes);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Writes a buffer according to Cassandra protocol: bytes.length (2) + bytes
|
||
|
* @param {Buffer} bytes
|
||
|
*/
|
||
|
writeShortBytes(bytes) {
|
||
|
if(bytes === null) {
|
||
|
//Only the length buffer containing -1
|
||
|
this.writeShort(-1);
|
||
|
return;
|
||
|
}
|
||
|
//Add the length buffer
|
||
|
this.writeShort(bytes.length);
|
||
|
//Add the actual buffer
|
||
|
this.add(bytes);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Writes a single byte
|
||
|
* @param {Number} num Value of the byte, a number between 0 and 255.
|
||
|
*/
|
||
|
writeByte(num) {
|
||
|
this.add(utils.allocBufferFromArray([num]));
|
||
|
}
|
||
|
|
||
|
writeString(str) {
|
||
|
if (typeof str === "undefined") {
|
||
|
throw new Error("can not write undefined");
|
||
|
}
|
||
|
const len = Buffer.byteLength(str, 'utf8');
|
||
|
const buf = utils.allocBufferUnsafe(2 + len);
|
||
|
buf.writeUInt16BE(len, 0);
|
||
|
buf.write(str, 2, buf.length-2, 'utf8');
|
||
|
this.add(buf);
|
||
|
}
|
||
|
|
||
|
writeLString(str) {
|
||
|
const len = Buffer.byteLength(str, 'utf8');
|
||
|
const buf = utils.allocBufferUnsafe(4 + len);
|
||
|
buf.writeInt32BE(len, 0);
|
||
|
buf.write(str, 4, buf.length-4, 'utf8');
|
||
|
this.add(buf);
|
||
|
}
|
||
|
|
||
|
writeStringList(values) {
|
||
|
this.writeShort(values.length);
|
||
|
values.forEach(this.writeString, this);
|
||
|
}
|
||
|
|
||
|
writeCustomPayload(payload) {
|
||
|
const keys = Object.keys(payload);
|
||
|
this.writeShort(keys.length);
|
||
|
keys.forEach(k => {
|
||
|
this.writeString(k);
|
||
|
this.writeBytes(payload[k]);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
writeStringMap(map) {
|
||
|
const keys = [];
|
||
|
for (const k in map) {
|
||
|
if (map.hasOwnProperty(k)) {
|
||
|
keys.push(k);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.writeShort(keys.length);
|
||
|
|
||
|
for(let i = 0; i < keys.length; i++) {
|
||
|
const key = keys[i];
|
||
|
this.writeString(key);
|
||
|
this.writeString(map[key]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param {Number} version
|
||
|
* @param {Number} streamId
|
||
|
* @param {Number} [flags] Header flags
|
||
|
* @returns {Buffer}
|
||
|
* @throws {TypeError}
|
||
|
*/
|
||
|
write(version, streamId, flags) {
|
||
|
const header = new FrameHeader(version, flags || 0, streamId, this.opcode, this.bodyLength);
|
||
|
const headerBuffer = header.toBuffer();
|
||
|
this.buffers.unshift(headerBuffer);
|
||
|
return Buffer.concat(this.buffers, headerBuffer.length + this.bodyLength);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Represents a queue that process one write at a time (FIFO).
|
||
|
* @extends {EventEmitter}
|
||
|
*/
|
||
|
class WriteQueue extends events.EventEmitter {
|
||
|
/**
|
||
|
* Creates a new WriteQueue instance.
|
||
|
* @param {Socket} netClient
|
||
|
* @param {Encoder} encoder
|
||
|
* @param {ClientOptions} options
|
||
|
*/
|
||
|
constructor(netClient, encoder, options) {
|
||
|
super();
|
||
|
this.netClient = netClient;
|
||
|
this.encoder = encoder;
|
||
|
this.isRunning = false;
|
||
|
/** @type {Array<{operation: OperationState, callback: Function}>} */
|
||
|
this.queue = [];
|
||
|
this.coalescingThreshold = options.socketOptions.coalescingThreshold;
|
||
|
this.error = null;
|
||
|
this.canWrite = true;
|
||
|
|
||
|
// Listen to drain event that is going to be fired once
|
||
|
// the underlying buffer is empty
|
||
|
netClient.on('drain', () => {
|
||
|
this.canWrite = true;
|
||
|
this.run();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Enqueues a new request
|
||
|
* @param {OperationState} operation
|
||
|
* @param {Function} callback The write callback.
|
||
|
*/
|
||
|
push(operation, callback) {
|
||
|
const self = this;
|
||
|
|
||
|
if (this.error) {
|
||
|
// There was a write error, there is no point in further trying to write to the socket.
|
||
|
return process.nextTick(function writePushError() {
|
||
|
callback(self.error);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
this.queue.push({ operation: operation, callback: callback});
|
||
|
this.run();
|
||
|
}
|
||
|
|
||
|
run() {
|
||
|
if (!this.isRunning && this.canWrite) {
|
||
|
this.isRunning = true;
|
||
|
// Use nextTick to allow the queue to build up on the current phase
|
||
|
process.nextTick(() => this.process());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
process() {
|
||
|
if (this.error) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const buffers = [];
|
||
|
const callbacks = [];
|
||
|
let totalLength = 0;
|
||
|
|
||
|
while (this.queue.length > 0 && totalLength < this.coalescingThreshold) {
|
||
|
const writeItem = this.queue.shift();
|
||
|
if (!writeItem.operation.canBeWritten()) {
|
||
|
// Invoke the write callback with an error that is not going to be yielded to user
|
||
|
// as the operation has timed out or was cancelled.
|
||
|
writeItem.callback(new Error('The operation was already cancelled or timeout elapsed'));
|
||
|
continue;
|
||
|
}
|
||
|
let data;
|
||
|
try {
|
||
|
data = writeItem.operation.request.write(this.encoder, writeItem.operation.streamId);
|
||
|
}
|
||
|
catch (err) {
|
||
|
writeItem.callback(err);
|
||
|
continue;
|
||
|
}
|
||
|
totalLength += data.length;
|
||
|
buffers.push(data);
|
||
|
callbacks.push(writeItem.callback);
|
||
|
}
|
||
|
|
||
|
if (totalLength === 0) {
|
||
|
this.isRunning = false;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// We have to invoke the callbacks to avoid race conditions.
|
||
|
// There is a performance benefit from executing all of them in a loop
|
||
|
for (let i = 0; i < callbacks.length; i++) {
|
||
|
callbacks[i]();
|
||
|
}
|
||
|
|
||
|
// Concatenate buffers and write it to the socket
|
||
|
// Further writes will be throttled until flushed
|
||
|
this.canWrite = this.netClient.write(Buffer.concat(buffers, totalLength), err => {
|
||
|
if (err) {
|
||
|
this.setWriteError(err);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (this.queue.length === 0 || !this.canWrite) {
|
||
|
// It will start running once we get the next message or has drained
|
||
|
this.isRunning = false;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Allow IO between writes
|
||
|
setImmediate(() => this.process());
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Emits the 'error' event and callbacks items that haven't been written and clears them from the queue.
|
||
|
* @param err
|
||
|
*/
|
||
|
setWriteError(err) {
|
||
|
err.isSocketError = true;
|
||
|
this.error = new types.DriverError('Socket was closed');
|
||
|
this.error.isSocketError = true;
|
||
|
// Use an special flag for items that haven't been written
|
||
|
this.error.requestNotWritten = true;
|
||
|
this.error.innerError = err;
|
||
|
const q = this.queue;
|
||
|
// Not more items can be added to the queue.
|
||
|
this.queue = utils.emptyArray;
|
||
|
for (let i = 0; i < q.length; i++) {
|
||
|
const item = q[i];
|
||
|
// Use the error marking that it was not written
|
||
|
item.callback(this.error);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
module.exports = { FrameWriter, WriteQueue };
|