@jonaskello-forks/amqp-client
Version:
AMQP 0-9-1 client, both for browsers (WebSocket) and node (TCP Socket)
137 lines (126 loc) • 5.03 kB
text/typescript
import { AMQPBaseClient } from './amqp-base-client.js'
import { AMQPError } from './amqp-error.js'
import { AMQPView } from './amqp-view.js'
import { Buffer } from 'buffer'
import * as net from 'net'
import * as tls from 'tls'
/**
* AMQP 0-9-1 client over TCP socket.
*/
export class AMQPClient extends AMQPBaseClient {
socket?: net.Socket
readonly tls : boolean
readonly host : string
readonly port : number
private readonly insecure : boolean
private framePos: number
private frameSize: number
private frameBuffer: Buffer
/**
* @param url - uri to the server, example: amqp://user:passwd@localhost:5672/vhost
*/
constructor(url: string) {
const u = new URL(url)
const vhost = decodeURIComponent(u.pathname.slice(1)) || "/"
const username = u.username || "guest"
const password = u.password || "guest"
const name = u.searchParams.get("name") || ""
const frameMax = parseInt(u.searchParams.get("frameMax") || "4096")
const heartbeat = parseInt(u.searchParams.get("heartbeat") || "0")
const platform = `${process.release.name} ${process.version} ${process.platform} ${process.arch}`
super(vhost, username, password, name, platform, frameMax, heartbeat)
this.tls = u.protocol === "amqps:"
this.host = u.hostname || "localhost"
this.port = parseInt(u.port) || (this.tls ? 5671 : 5672)
this.insecure = u.searchParams.get("insecure") !== undefined
this.framePos = 0
this.frameSize = 0
this.frameBuffer = Buffer.allocUnsafe(frameMax)
}
override connect(): Promise<AMQPBaseClient> {
const socket = this.connectSocket()
Object.defineProperty(this, 'socket', {
value: socket,
enumerable: false // hide it from console.log etc.
})
return new Promise((resolve, reject) => {
socket.on('error', (err) => reject(new AMQPError(err.message, this)))
this.connectPromise = [resolve, reject]
})
}
private connectSocket() {
const options = {
host: this.host,
port: this.port,
servername: this.host,
rejectUnauthorized: !this.insecure
}
const sendStart = () => this.send(new Uint8Array([65, 77, 81, 80, 0, 0, 9, 1]))
const conn = this.tls ? tls.connect(options, sendStart) : net.connect(options, sendStart)
conn.on('data', this.onRead.bind(this))
return conn
}
private onRead(buf: Buffer): boolean {
const bufLen = buf.length
// A socket read can contain 0 or more frames, so find frame boundries
let bufPos = 0
while (bufPos < bufLen) {
// read frame size of next frame
if (this.frameSize === 0) {
// first 7 bytes of a frame was split over two reads, this reads the second part
if (this.framePos !== 0) {
const copied = buf.copy(this.frameBuffer, this.framePos, bufPos, bufPos + 7 - this.framePos)
if (copied === 0) throw `Copied 0 bytes framePos=${this.framePos} bufPos=${bufPos} bytesWritten=${bufLen}`
this.frameSize = this.frameBuffer.readInt32BE(bufPos + 3) + 8
this.framePos += copied
bufPos += copied
continue
}
// frame header is split over reads, copy to frameBuffer
if (bufPos + 3 + 4 > bufLen) {
const copied = buf.copy(this.frameBuffer, this.framePos, bufPos, bufLen)
if (copied === 0) throw `Copied 0 bytes framePos=${this.framePos} bufPos=${bufPos} bytesWritten=${bufLen}`
this.framePos += copied
break
}
this.frameSize = buf.readInt32BE(bufPos + 3) + 8
// avoid copying if the whole frame is in the read buffer
if (bufLen - bufPos >= this.frameSize) {
const view = new AMQPView(buf.buffer, buf.byteOffset + bufPos, this.frameSize)
this.parseFrames(view)
bufPos += this.frameSize
this.frameSize = 0
continue
}
}
const leftOfFrame = this.frameSize - this.framePos
const copyBytes = Math.min(leftOfFrame, bufLen - bufPos)
const copied = buf.copy(this.frameBuffer, this.framePos, bufPos, bufPos + copyBytes)
if (copied === 0) throw `Copied 0 bytes, please report this bug, frameSize=${this.frameSize} framePos=${this.framePos} bufPos=${bufPos} copyBytes=${copyBytes} bytesWritten=${bufLen}`
this.framePos += copied
bufPos += copied
if (this.framePos === this.frameSize) {
const view = new AMQPView(this.frameBuffer.buffer, 0, this.frameSize)
this.parseFrames(view)
this.frameSize = this.framePos = 0
}
}
return true
}
/**
* @ignore
* @param bytes to send
* @return fulfilled when the data is enqueued
*/
override send(bytes: Uint8Array): Promise<void> {
return new Promise((resolve, reject) => {
if (this.socket)
this.socket.write(bytes, undefined, (err) => err ? reject(err) : resolve())
else
reject(new AMQPError("Socket not connected", this))
})
}
protected override closeSocket() {
if(this.socket) this.socket.end()
}
}