Files
Baileys/src/WAM/encode.ts
2025-05-06 12:10:19 -03:00

171 lines
5.0 KiB
TypeScript

import { BinaryInfo } from './BinaryInfo'
import {
FLAG_BYTE,
FLAG_EVENT,
FLAG_EXTENDED,
FLAG_FIELD,
FLAG_GLOBAL,
Value,
WEB_EVENTS,
WEB_GLOBALS
} from './constants'
const getHeaderBitLength = (key: number) => (key < 256 ? 2 : 3)
export const encodeWAM = (binaryInfo: BinaryInfo) => {
binaryInfo.buffer = []
encodeWAMHeader(binaryInfo)
encodeEvents(binaryInfo)
console.log(binaryInfo.buffer)
const totalSize = binaryInfo.buffer.map(a => a.length).reduce((a, b) => a + b)
const buffer = Buffer.alloc(totalSize)
let offset = 0
for (const buffer_ of binaryInfo.buffer) {
buffer_.copy(buffer, offset)
offset += buffer_.length
}
return buffer
}
function encodeWAMHeader(binaryInfo: BinaryInfo) {
const headerBuffer = Buffer.alloc(8) // starting buffer
headerBuffer.write('WAM', 0, 'utf8')
headerBuffer.writeUInt8(binaryInfo.protocolVersion, 3)
headerBuffer.writeUInt8(1, 4) //random flag
headerBuffer.writeUInt16BE(binaryInfo.sequence, 5)
headerBuffer.writeUInt8(0, 7) // regular channel
binaryInfo.buffer.push(headerBuffer)
}
function encodeGlobalAttributes(binaryInfo: BinaryInfo, globals: { [key: string]: Value }) {
for (const [key, _value] of Object.entries(globals)) {
const id = WEB_GLOBALS.find(a => a?.name === key)!.id
let value = _value
if (typeof value === 'boolean') {
value = value ? 1 : 0
}
binaryInfo.buffer.push(serializeData(id, value, FLAG_GLOBAL))
}
}
function encodeEvents(binaryInfo: BinaryInfo) {
for (const [name, { props, globals }] of binaryInfo.events.map(a => Object.entries(a)[0])) {
encodeGlobalAttributes(binaryInfo, globals)
const event = WEB_EVENTS.find(a => a.name === name)!
const props_ = Object.entries(props)
let extended = false
for (const [, value] of props_) {
extended ||= value !== null
}
const eventFlag = extended ? FLAG_EVENT : FLAG_EVENT | FLAG_EXTENDED
binaryInfo.buffer.push(serializeData(event.id, -event.weight, eventFlag))
for (let i = 0; i < props_.length; i++) {
const [key, _value] = props_[i]
const id = event.props[key][0]
extended = i < props_.length - 1
let value = _value
if (typeof value === 'boolean') {
value = value ? 1 : 0
}
const fieldFlag = extended ? FLAG_EVENT : FLAG_FIELD | FLAG_EXTENDED
binaryInfo.buffer.push(serializeData(id, value, fieldFlag))
}
}
}
function serializeData(key: number, value: Value, flag: number): Buffer {
const bufferLength = getHeaderBitLength(key)
let buffer: Buffer
let offset = 0
if (value === null) {
if (flag === FLAG_GLOBAL) {
buffer = Buffer.alloc(bufferLength)
offset = serializeHeader(buffer, offset, key, flag)
return buffer
}
} else if (typeof value === 'number' && Number.isInteger(value)) {
// is number
if (value === 0 || value === 1) {
buffer = Buffer.alloc(bufferLength)
offset = serializeHeader(buffer, offset, key, flag | ((value + 1) << 4))
return buffer
} else if (-128 <= value && value < 128) {
buffer = Buffer.alloc(bufferLength + 1)
offset = serializeHeader(buffer, offset, key, flag | (3 << 4))
buffer.writeInt8(value, offset)
return buffer
} else if (-32768 <= value && value < 32768) {
buffer = Buffer.alloc(bufferLength + 2)
offset = serializeHeader(buffer, offset, key, flag | (4 << 4))
buffer.writeInt16LE(value, offset)
return buffer
} else if (-2147483648 <= value && value < 2147483648) {
buffer = Buffer.alloc(bufferLength + 4)
offset = serializeHeader(buffer, offset, key, flag | (5 << 4))
buffer.writeInt32LE(value, offset)
return buffer
} else {
buffer = Buffer.alloc(bufferLength + 8)
offset = serializeHeader(buffer, offset, key, flag | (7 << 4))
buffer.writeDoubleLE(value, offset)
return buffer
}
} else if (typeof value === 'number') {
// is float
buffer = Buffer.alloc(bufferLength + 8)
offset = serializeHeader(buffer, offset, key, flag | (7 << 4))
buffer.writeDoubleLE(value, offset)
return buffer
} else if (typeof value === 'string') {
// is string
const utf8Bytes = Buffer.byteLength(value, 'utf8')
if (utf8Bytes < 256) {
buffer = Buffer.alloc(bufferLength + 1 + utf8Bytes)
offset = serializeHeader(buffer, offset, key, flag | (8 << 4))
buffer.writeUint8(utf8Bytes, offset++)
} else if (utf8Bytes < 65536) {
buffer = Buffer.alloc(bufferLength + 2 + utf8Bytes)
offset = serializeHeader(buffer, offset, key, flag | (9 << 4))
buffer.writeUInt16LE(utf8Bytes, offset)
offset += 2
} else {
buffer = Buffer.alloc(bufferLength + 4 + utf8Bytes)
offset = serializeHeader(buffer, offset, key, flag | (10 << 4))
buffer.writeUInt32LE(utf8Bytes, offset)
offset += 4
}
buffer.write(value, offset, 'utf8')
return buffer
}
throw 'missing'
}
function serializeHeader(buffer: Buffer, offset: number, key: number, flag: number) {
if (key < 256) {
buffer.writeUInt8(flag, offset)
offset += 1
buffer.writeUInt8(key, offset)
offset += 1
} else {
buffer.writeUInt8(flag | FLAG_BYTE, offset)
offset += 1
buffer.writeUInt16LE(key, offset)
offset += 2
}
return offset
}