@ogc/postgres-tmp
Version:
Fastest full featured PostgreSQL client for Node.js
234 lines (206 loc) • 5.48 kB
JavaScript
import { EventEmitter } from 'node:events'
import { Buffer } from 'node:buffer'
const Crypto = globalThis.crypto
let ids = 1
const tasks = new Set()
const v4Seg = '(?:[0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])'
const v4Str = `(${v4Seg}[.]){3}${v4Seg}`
const IPv4Reg = new RegExp(`^${v4Str}$`)
const v6Seg = '(?:[0-9a-fA-F]{1,4})'
const IPv6Reg = new RegExp(
'^(' +
`(?:${v6Seg}:){7}(?:${v6Seg}|:)|` +
`(?:${v6Seg}:){6}(?:${v4Str}|:${v6Seg}|:)|` +
`(?:${v6Seg}:){5}(?::${v4Str}|(:${v6Seg}){1,2}|:)|` +
`(?:${v6Seg}:){4}(?:(:${v6Seg}){0,1}:${v4Str}|(:${v6Seg}){1,3}|:)|` +
`(?:${v6Seg}:){3}(?:(:${v6Seg}){0,2}:${v4Str}|(:${v6Seg}){1,4}|:)|` +
`(?:${v6Seg}:){2}(?:(:${v6Seg}){0,3}:${v4Str}|(:${v6Seg}){1,5}|:)|` +
`(?:${v6Seg}:){1}(?:(:${v6Seg}){0,4}:${v4Str}|(:${v6Seg}){1,6}|:)|` +
`(?::((?::${v6Seg}){0,5}:${v4Str}|(?::${v6Seg}){1,7}|:))` +
')(%[0-9a-zA-Z-.:]{1,})?$'
)
const textEncoder = new TextEncoder()
export const crypto = {
randomBytes: l => Crypto.getRandomValues(Buffer.alloc(l)),
pbkdf2Sync: async(password, salt, iterations, keylen) =>
Crypto.subtle.deriveBits(
{
name: 'PBKDF2',
hash: 'SHA-256',
salt,
iterations
},
await Crypto.subtle.importKey(
'raw',
textEncoder.encode(password),
'PBKDF2',
false,
['deriveBits']
),
keylen * 8,
['deriveBits']
),
createHash: type => ({
update: x => ({
digest: encoding => {
if (!(x instanceof Uint8Array)) {
x = textEncoder.encode(x)
}
let prom
if (type === 'sha256') {
prom = Crypto.subtle.digest('SHA-256', x)
} else if (type === 'md5') {
prom = Crypto.subtle.digest('md5', x)
} else {
throw Error('createHash only supports sha256 or md5 in this environment, not ${type}.')
}
if (encoding === 'hex') {
return prom.then((arrayBuf) => Buffer.from(arrayBuf).toString('hex'))
} else if (encoding) {
throw Error(`createHash only supports hex encoding or unencoded in this environment, not ${encoding}`)
} else {
return prom
}
}
})
}),
createHmac: (type, key) => ({
update: x => ({
digest: async() =>
Buffer.from(
await Crypto.subtle.sign(
'HMAC',
await Crypto.subtle.importKey('raw', key, { name: 'HMAC', hash: 'SHA-256' }, false, ['sign']),
textEncoder.encode(x)
)
)
})
})
}
export const performance = globalThis.performance
export const process = {
env: {}
}
export const os = {
userInfo() {
return { username: 'postgres' }
}
}
export const fs = {
readFile() {
throw new Error('Reading files not supported on CloudFlare')
}
}
export const net = {
isIP: (x) => IPv4Reg.test(x) ? 4 : IPv6Reg.test(x) ? 6 : 0,
Socket
}
export { setImmediate, clearImmediate }
export const tls = {
connect({ socket: tcp, servername }) {
tcp.writer.releaseLock()
tcp.reader.releaseLock()
tcp.readyState = 'upgrading'
tcp.raw = tcp.raw.startTls({ servername })
tcp.raw.closed.then(
() => tcp.emit('close'),
(e) => tcp.emit('error', e)
)
tcp.writer = tcp.raw.writable.getWriter()
tcp.reader = tcp.raw.readable.getReader()
tcp.writer.ready.then(() => {
tcp.read()
tcp.readyState = 'upgrade'
})
return tcp
}
}
function Socket() {
const tcp = Object.assign(new EventEmitter(), {
readyState: 'open',
raw: null,
writer: null,
reader: null,
connect,
write,
end,
destroy,
read
})
return tcp
async function connect(port, host) {
try {
tcp.readyState = 'opening'
const { connect } = await import('cloudflare:sockets')
tcp.raw = connect(host + ':' + port, tcp.ssl ? { secureTransport: 'starttls' } : {})
tcp.raw.closed.then(
() => {
tcp.readyState !== 'upgrade'
? close()
: ((tcp.readyState = 'open'), tcp.emit('secureConnect'))
},
(e) => tcp.emit('error', e)
)
tcp.writer = tcp.raw.writable.getWriter()
tcp.reader = tcp.raw.readable.getReader()
tcp.ssl ? readFirst() : read()
tcp.writer.ready.then(() => {
tcp.readyState = 'open'
tcp.emit('connect')
})
} catch (err) {
error(err)
}
}
function close() {
if (tcp.readyState === 'closed')
return
tcp.readyState = 'closed'
tcp.emit('close')
}
function write(data, cb) {
tcp.writer.write(data).then(cb, error)
return true
}
function end(data) {
return data
? tcp.write(data, () => tcp.raw.close())
: tcp.raw.close()
}
function destroy() {
tcp.destroyed = true
tcp.end()
}
async function read() {
try {
let done
, value
while (({ done, value } = await tcp.reader.read(), !done))
tcp.emit('data', Buffer.from(value))
} catch (err) {
error(err)
}
}
async function readFirst() {
const { value } = await tcp.reader.read()
tcp.emit('data', Buffer.from(value))
}
function error(err) {
tcp.emit('error', err)
tcp.emit('close')
}
}
function setImmediate(fn) {
const id = ids++
tasks.add(id)
queueMicrotask(() => {
if (tasks.has(id)) {
fn()
tasks.delete(id)
}
})
return id
}
function clearImmediate(id) {
tasks.delete(id)
}