From 56d536d96ccf76e2c9ef858c1a98edeb73b6dc30 Mon Sep 17 00:00:00 2001 From: Dijana Pavlovic Date: Thu, 29 Aug 2024 17:35:55 +0200 Subject: [PATCH 01/14] Add support for SNI when connecting to the instance --- docs/index.rst | 2 +- packages/driver/src/conUtils.ts | 43 +++++++++++++++++++- packages/driver/src/rawConn.ts | 2 +- packages/driver/test/shared-client-testcases | 2 +- 4 files changed, 45 insertions(+), 4 deletions(-) diff --git a/docs/index.rst b/docs/index.rst index bd20cb928..fa3de5a68 100644 --- a/docs/index.rst +++ b/docs/index.rst @@ -138,7 +138,7 @@ app: $ npm init -y # initialize a new npm project $ npm i edgedb $ npm i -D typescript @types/node @edgedb/generate tsx - $ npx tsc --init # initialize basic a basic TypeScript project + $ npx tsc --init # initialize a basic TypeScript project Client ^^^^^^ diff --git a/packages/driver/src/conUtils.ts b/packages/driver/src/conUtils.ts index ec7822146..7e360c2ec 100644 --- a/packages/driver/src/conUtils.ts +++ b/packages/driver/src/conUtils.ts @@ -68,6 +68,7 @@ export interface NormalizedConnectConfig extends PartiallyNormalizedConfig { logging: boolean; } +// explicit config export interface ConnectConfig { dsn?: string; instanceName?: string; @@ -84,6 +85,7 @@ export interface ConnectConfig { tlsCA?: string; tlsCAFile?: string; tlsSecurity?: TlsSecurity; + tlsServerName?: string; timeout?: number; waitUntilAvailable?: Duration | number; @@ -124,6 +126,7 @@ type ConnectConfigParams = | "cloudProfile" | "tlsCAData" | "tlsSecurity" + | "tlsServerName" | "waitUntilAvailable"; export type ResolvedConnectConfigReadonly = Readonly< @@ -165,6 +168,9 @@ export class ResolvedConnectConfig { _tlsSecurity: TlsSecurity | null = null; _tlsSecuritySource: string | null = null; + _tlsServerName: string | null = null; + _tlsServerNameSource: string | null = null; + _waitUntilAvailable: number | null = null; _waitUntilAvailableSource: string | null = null; @@ -180,6 +186,7 @@ export class ResolvedConnectConfig { this.setSecretKey = this.setSecretKey.bind(this); this.setTlsCAData = this.setTlsCAData.bind(this); this.setTlsCAFile = this.setTlsCAFile.bind(this); + this.setTlsServerName = this.setTlsServerName.bind(this); this.setTlsSecurity = this.setTlsSecurity.bind(this); this.setWaitUntilAvailable = this.setWaitUntilAvailable.bind(this); } @@ -280,6 +287,10 @@ export class ResolvedConnectConfig { ); } + setTlsServerName(serverName: string | null, source: string): boolean { + return this._setParam("tlsServerName", serverName, source, validateHost); + } + setTlsSecurity(tlsSecurity: string | null, source: string): boolean { return this._setParam( "tlsSecurity", @@ -375,6 +386,10 @@ export class ResolvedConnectConfig { return this._cloudProfile ?? "default"; } + get tlsServerName(): string | undefined { + return this._tlsServerName ?? undefined; + } + get tlsSecurity(): Exclude { return this._tlsSecurity && this._tlsSecurity !== "default" ? this._tlsSecurity @@ -431,6 +446,12 @@ export class ResolvedConnectConfig { this._tlsSecurity, this._tlsSecuritySource, ); + outputLine( + "tlsServerName", + this.tlsServerName, + this._tlsServerName, + this._tlsServerNameSource, + ); outputLine( "waitUntilAvailable", this.waitUntilAvailable, @@ -542,6 +563,7 @@ async function parseConnectDsnAndArgs( cloudProfile: getEnv("EDGEDB_CLOUD_PROFILE"), tlsCA: config.tlsCA, tlsCAFile: config.tlsCAFile, + tlsServerName: config.tlsServerName, tlsSecurity: config.tlsSecurity, serverSettings: config.serverSettings, waitUntilAvailable: config.waitUntilAvailable, @@ -565,6 +587,7 @@ async function parseConnectDsnAndArgs( tlsCA: `'tlsCA' option`, tlsCAFile: `'tlsCAFile' option`, tlsSecurity: `'tlsSecurity' option`, + tlsServerName: `tlsServerName option`, serverSettings: `'serverSettings' option`, waitUntilAvailable: `'waitUntilAvailable' option`, }, @@ -602,6 +625,7 @@ async function parseConnectDsnAndArgs( secretKey: getEnv("EDGEDB_SECRET_KEY"), tlsCA: getEnv("EDGEDB_TLS_CA"), tlsCAFile: getEnv("EDGEDB_TLS_CA_FILE"), + tlsServerName: getEnv("EDGEDB_TLS_SERVER_NAME"), tlsSecurity: getEnv("EDGEDB_CLIENT_TLS_SECURITY"), waitUntilAvailable: getEnv("EDGEDB_WAIT_UNTIL_AVAILABLE"), }, @@ -619,6 +643,7 @@ async function parseConnectDsnAndArgs( secretKey: `'EDGEDB_SECRET_KEY' environment variable`, tlsCA: `'EDGEDB_TLS_CA' environment variable`, tlsCAFile: `'EDGEDB_TLS_CA_FILE' environment variable`, + tlsServerName: `EDGEDB_TLS_SERVER_NAME environment variable`, tlsSecurity: `'EDGEDB_CLIENT_TLS_SECURITY' environment variable`, waitUntilAvailable: `'EDGEDB_WAIT_UNTIL_AVAILABLE' environment variable`, }, @@ -663,6 +688,10 @@ async function parseConnectDsnAndArgs( .readFileUtf8(stashDir, "database") .then((name) => name.trim()) .catch(() => undefined), + serverUtils + .readFileUtf8(stashDir, "branch") + .then((name) => name.trim()) + .catch(() => undefined), ]); await resolveConfigOptions( @@ -710,6 +739,7 @@ interface ResolveConfigOptionsConfig { cloudProfile: string; tlsCA: string; tlsCAFile: string; + tlsServerName: string; tlsSecurity: string; serverSettings: { [key: string]: string }; waitUntilAvailable: number | string | Duration; @@ -785,6 +815,11 @@ async function resolveConfigOptions< sources.tlsCAFile!, readFile, )) || anyOptionsUsed; + anyOptionsUsed = + resolvedConfig.setTlsServerName( + config.tlsServerName ?? null, + sources.tlsServerName!, + ) || anyOptionsUsed; anyOptionsUsed = resolvedConfig.setTlsSecurity( config.tlsSecurity ?? null, @@ -880,7 +915,6 @@ async function resolveConfigOptions< } creds = await readCredentialsFile(credentialsFile, serverUtils!); } - resolvedConfig.setHost(creds.host ?? null, source); resolvedConfig.setPort(creds.port ?? null, source); if (creds.database != null) { @@ -1087,6 +1121,13 @@ async function parseDSNIntoConfig( config.setTlsCAFile(val, _source, readFile), ); + await handleDSNPart( + "tls_server_name", + null, + config._tlsServerName, + config.setTlsServerName, + ); + await handleDSNPart( "tls_security", null, diff --git a/packages/driver/src/rawConn.ts b/packages/driver/src/rawConn.ts index 2f5d95a52..c7852b383 100644 --- a/packages/driver/src/rawConn.ts +++ b/packages/driver/src/rawConn.ts @@ -59,7 +59,7 @@ function getTlsOptions(config: ResolvedConnectConfig): tls.ConnectionOptions { if (!isIPAddress) { // XXX Deno doesn't support this and that means it won't // work with EdgeDB Cloud. - tlsOptions.servername = config.address[0]; + tlsOptions.servername = config.tlsServerName || config.address[0]; } _tlsOptions.set(config, tlsOptions); diff --git a/packages/driver/test/shared-client-testcases b/packages/driver/test/shared-client-testcases index 3ba0df7fd..9bddb7beb 160000 --- a/packages/driver/test/shared-client-testcases +++ b/packages/driver/test/shared-client-testcases @@ -1 +1 @@ -Subproject commit 3ba0df7fde1f252a40476ae665018336e7900f55 +Subproject commit 9bddb7bebecf41510a31c57e1438c35b2592d35d From cd024b5203043a9f9860ab553a343841fa261abf Mon Sep 17 00:00:00 2001 From: James Clarke Date: Wed, 2 Aug 2023 12:48:49 +0100 Subject: [PATCH 02/14] Add basic support for protocol v2 --- packages/driver/src/baseConn.ts | 2 +- packages/driver/src/codecs/array.ts | 11 +- packages/driver/src/codecs/codecs.ts | 4 +- packages/driver/src/codecs/enum.ts | 9 +- packages/driver/src/codecs/ifaces.ts | 11 +- packages/driver/src/codecs/namedtuple.ts | 11 +- packages/driver/src/codecs/range.ts | 4 +- packages/driver/src/codecs/registry.ts | 198 ++++++++++++++++++++--- packages/driver/src/codecs/tuple.ts | 4 +- packages/driver/src/primitives/buffer.ts | 8 +- packages/driver/test/datetime.test.ts | 16 +- 11 files changed, 238 insertions(+), 40 deletions(-) diff --git a/packages/driver/src/baseConn.ts b/packages/driver/src/baseConn.ts index d101f96bd..a3526d53d 100644 --- a/packages/driver/src/baseConn.ts +++ b/packages/driver/src/baseConn.ts @@ -45,7 +45,7 @@ import LRU from "./primitives/lru"; import type { SerializedSessionState } from "./options"; import { Session } from "./options"; -export const PROTO_VER: ProtocolVersion = [1, 0]; +export const PROTO_VER: ProtocolVersion = [2, 0]; export const PROTO_VER_MIN: ProtocolVersion = [0, 9]; enum TransactionStatus { diff --git a/packages/driver/src/codecs/array.ts b/packages/driver/src/codecs/array.ts index d727a3da6..878396048 100644 --- a/packages/driver/src/codecs/array.ts +++ b/packages/driver/src/codecs/array.ts @@ -27,11 +27,18 @@ import { NamedTupleCodec } from "./namedtuple"; export class ArrayCodec extends Codec implements ICodec { private subCodec: ICodec; private len: number; - - constructor(tid: uuid, subCodec: ICodec, len: number) { + public typeName: string | null; + + constructor( + tid: uuid, + typeName: string | null, + subCodec: ICodec, + len: number + ) { super(tid); this.subCodec = subCodec; this.len = len; + this.typeName = typeName; } encode(buf: WriteBuffer, obj: any): void { diff --git a/packages/driver/src/codecs/codecs.ts b/packages/driver/src/codecs/codecs.ts index 33c8289b3..ecbdbe1d5 100644 --- a/packages/driver/src/codecs/codecs.ts +++ b/packages/driver/src/codecs/codecs.ts @@ -80,14 +80,14 @@ export const INVALID_CODEC = new NullCodec(INVALID_CODEC_ID); function registerScalarCodec( typename: string, - type: new (tid: uuid) => ICodec, + type: new (tid: uuid, typename: string | null) => ICodec, ): void { const id = KNOWN_TYPENAMES.get(typename); if (id == null) { throw new InternalClientError("unknown type name"); } - SCALAR_CODECS.set(id, new type(id)); + SCALAR_CODECS.set(id, new type(id, typename)); } registerScalarCodec("std::int16", Int16Codec); diff --git a/packages/driver/src/codecs/enum.ts b/packages/driver/src/codecs/enum.ts index 72faf317b..a4b70fcab 100644 --- a/packages/driver/src/codecs/enum.ts +++ b/packages/driver/src/codecs/enum.ts @@ -22,8 +22,13 @@ import { StrCodec } from "./text"; export class EnumCodec extends StrCodec implements ICodec { readonly values: string[]; - constructor(tid: uuid, derivedFromTid: uuid | null = null, values: string[]) { - super(tid, derivedFromTid); + constructor( + tid: uuid, + typeName: string | null, + derivedFromTid: uuid | null = null, + values: string[] + ) { + super(tid, typeName, derivedFromTid); this.values = values; } } diff --git a/packages/driver/src/codecs/ifaces.ts b/packages/driver/src/codecs/ifaces.ts index d1989cd3d..fa1bc0f8b 100644 --- a/packages/driver/src/codecs/ifaces.ts +++ b/packages/driver/src/codecs/ifaces.ts @@ -70,9 +70,14 @@ export abstract class ScalarCodec extends Codec { private derivedFromTid: uuid | null = null; private typeName: string | null = null; - constructor(tid: uuid, derivedFromTid: uuid | null = null) { + constructor( + tid: uuid, + typeName: string | null, + derivedFromTid: uuid | null = null, + ) { super(tid); this.derivedFromTid = derivedFromTid; + this.typeName = typeName; } /** @internal */ @@ -80,9 +85,9 @@ export abstract class ScalarCodec extends Codec { this.typeName = typeName; } - derive(tid: uuid): Codec { + derive(tid: uuid, typeName: string | null): Codec { const self = this.constructor; - return new (self as any)(tid, this.tid) as Codec; + return new (self as any)(tid, this.tid, typeName) as Codec; } getSubcodecs(): ICodec[] { diff --git a/packages/driver/src/codecs/namedtuple.ts b/packages/driver/src/codecs/namedtuple.ts index 2335754da..710d1286c 100644 --- a/packages/driver/src/codecs/namedtuple.ts +++ b/packages/driver/src/codecs/namedtuple.ts @@ -32,12 +32,19 @@ export class NamedTupleCodec extends Codec implements ICodec, IArgsCodec { private subCodecs: ICodec[]; private names: string[]; private namesSet: Set; - - constructor(tid: uuid, codecs: ICodec[], names: string[]) { + public typeName: string | null; + + constructor( + tid: uuid, + typeName: string | null, + codecs: ICodec[], + names: string[] + ) { super(tid); this.subCodecs = codecs; this.names = names; this.namesSet = new Set(names); + this.typeName = typeName; } encode(buf: WriteBuffer, object: any): void { diff --git a/packages/driver/src/codecs/range.ts b/packages/driver/src/codecs/range.ts index f95733579..db0d3c9f0 100644 --- a/packages/driver/src/codecs/range.ts +++ b/packages/driver/src/codecs/range.ts @@ -97,10 +97,12 @@ export class RangeCodec extends Codec implements ICodec { readonly tsModule = "edgedb"; private subCodec: ICodec; + readonly typeName: string | null; - constructor(tid: uuid, subCodec: ICodec) { + constructor(tid: uuid, typeName: string | null, subCodec: ICodec) { super(tid); this.subCodec = subCodec; + this.typeName = typeName; } encode(buf: WriteBuffer, obj: any) { diff --git a/packages/driver/src/codecs/registry.ts b/packages/driver/src/codecs/registry.ts index 70e2b7a8d..585d46029 100644 --- a/packages/driver/src/codecs/registry.ts +++ b/packages/driver/src/codecs/registry.ts @@ -49,6 +49,8 @@ const CTYPE_ARRAY = 6; const CTYPE_ENUM = 7; const CTYPE_INPUT_SHAPE = 8; const CTYPE_RANGE = 9; +const CTYPE_OBJECT = 10; +const CTYPE_COMPOUND = 11; const CTYPE_MULTIRANGE = 12; export interface CustomCodecSpec { @@ -82,7 +84,7 @@ export class CodecsRegistry { if (int64_bigint) { this.customScalarCodecs.set( INT64_TYPEID, - new numbers.Int64BigintCodec(INT64_TYPEID), + new numbers.Int64BigintCodec(INT64_TYPEID, "std::int64"), ); } else { this.customScalarCodecs.delete(INT64_TYPEID); @@ -91,7 +93,7 @@ export class CodecsRegistry { if (datetime_localDatetime) { this.customScalarCodecs.set( DATETIME_TYPEID, - new datecodecs.LocalDateTimeCodec(DATETIME_TYPEID), + new datecodecs.LocalDateTimeCodec(DATETIME_TYPEID, "std::datetime"), ); } else { this.customScalarCodecs.delete(DATETIME_TYPEID); @@ -100,7 +102,7 @@ export class CodecsRegistry { if (json_string) { this.customScalarCodecs.set( JSON_TYPEID, - new JSONStringCodec(JSON_TYPEID), + new JSONStringCodec(JSON_TYPEID, "std::json"), ); } else { this.customScalarCodecs.delete(JSON_TYPEID); @@ -138,7 +140,15 @@ export class CodecsRegistry { let codec: ICodec | null = null; while (frb.length) { - codec = this._buildCodec(frb, codecsList, protocolVersion); + if (versionGreaterThanOrEqual(protocolVersion, [2, 0])) { + const descLen = frb.readInt32(); + const descBuf = ReadBuffer.alloc(); + frb.sliceInto(descBuf, descLen); + codec = this._buildCodec(descBuf, codecsList, protocolVersion, true); + descBuf.finish("unexpected trailing data in type descriptor buffer"); + } else { + codec = this._buildCodec(frb, codecsList, protocolVersion, false); + } if (codec == null) { // An annotation; ignore. continue; @@ -158,6 +168,7 @@ export class CodecsRegistry { frb: ReadBuffer, cl: ICodec[], protocolVersion: ProtocolVersion, + isProtoV2: boolean, ): ICodec | null { const t = frb.readUInt8(); const tid = frb.readUUID(); @@ -170,6 +181,10 @@ export class CodecsRegistry { if (res != null) { // We have a codec for this "tid"; advance the buffer // so that we can process the next codec. + if (isProtoV2) { + frb.discard(frb.length); + return res; + } switch (t) { case CTYPE_SET: { @@ -294,6 +309,13 @@ export class CodecsRegistry { case CTYPE_SHAPE: case CTYPE_INPUT_SHAPE: { + if (t == CTYPE_SHAPE && isProtoV2) { + // @ts-expect-error + const isEphemeralFreeShape = frb.readBoolean(); + // @ts-expect-error + const objTypePos = frb.readUInt16(); + } + const els = frb.readUInt16(); const codecs: ICodec[] = new Array(els); const names: string[] = new Array(els); @@ -325,6 +347,12 @@ export class CodecsRegistry { names[i] = name; flags[i] = flag!; cards[i] = card!; + + if (t == CTYPE_SHAPE && isProtoV2) { + const sourceTypePos = frb.readUInt16(); + // @ts-expect-error + const sourceType = cl[sourceTypePos]; + } } res = @@ -347,23 +375,86 @@ export class CodecsRegistry { } case CTYPE_SCALAR: { - const pos = frb.readUInt16(); - res = cl[pos]; - if (res == null) { - throw new ProtocolError( - "could not build scalar codec: missing a codec for base scalar", - ); - } - if (!(res instanceof ScalarCodec)) { - throw new ProtocolError( - "could not build scalar codec: base scalar has a non-scalar codec", - ); + if (isProtoV2) { + const typeName = frb.readString(); + // @ts-expect-error + const isSchemaDefined = frb.readBoolean(); + + const ancestorCount = frb.readUInt16(); + const ancestors: ICodec[] = []; + for (var i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + const ancestorCodec = cl[ancestorPos]; + if (ancestorCodec == null) { + throw new ProtocolError( + "could not build scalar codec: missing a codec for base scalar", + ); + } + + if (!(ancestorCodec instanceof ScalarCodec)) { + throw new ProtocolError( + `a scalar codec expected for base scalar type, ` + + `got ${ancestorCodec}`, + ); + } + ancestors.push(ancestorCodec); + } + + if (ancestorCount == 0) { + res = this.customScalarCodecs.get(tid) ?? SCALAR_CODECS.get(tid); + if (res == null) { + if (KNOWN_TYPES.has(tid)) { + throw new InternalClientError( + `no JS codec for ${KNOWN_TYPES.get(tid)}`, + ); + } + + throw new InternalClientError( + `no JS codec for the type with ID ${tid}`, + ); + } + } else { + const baseCodec = ancestors[ancestors.length - 1]; + if (!(baseCodec instanceof ScalarCodec)) { + throw new ProtocolError( + `a scalar codec expected for base scalar type, ` + + `got ${baseCodec}`, + ); + } + res = baseCodec.derive(tid, typeName) as ICodec; + } + } else { + const pos = frb.readUInt16(); + res = cl[pos]; + if (res == null) { + throw new ProtocolError( + "could not build scalar codec: missing a codec for base scalar", + ); + } + if (!(res instanceof ScalarCodec)) { + throw new ProtocolError( + "could not build scalar codec: base scalar has a non-scalar codec", + ); + } + res = res.derive(tid, null) as ICodec; } - res = res.derive(tid) as ICodec; break; } case CTYPE_ARRAY: { + let typeName: string | null = null; + if (isProtoV2) { + typeName = frb.readString(); + // @ts-expect-error + const isSchemaDefined = frb.readBoolean(); + const ancestorCount = frb.readUInt16(); + for (var i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + // @ts-expect-error + const ancestorCodec = cl[ancestorPos]; + } + } + const pos = frb.readUInt16(); const els = frb.readUInt16(); if (els !== 1) { @@ -378,11 +469,24 @@ export class CodecsRegistry { "could not build array codec: missing subcodec", ); } - res = new ArrayCodec(tid, subCodec, dimLen); + res = new ArrayCodec(tid, typeName, subCodec, dimLen); break; } case CTYPE_TUPLE: { + let typeName: string | null = null; + if (isProtoV2) { + typeName = frb.readString(); + // @ts-expect-error + const isSchemaDefined = frb.readBoolean(); + const ancestorCount = frb.readUInt16(); + for (var i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + // @ts-expect-error + const ancestorCodec = cl[ancestorPos]; + } + } + const els = frb.readUInt16(); if (els === 0) { res = EMPTY_TUPLE_CODEC; @@ -398,12 +502,25 @@ export class CodecsRegistry { } codecs[i] = subCodec; } - res = new TupleCodec(tid, codecs); + res = new TupleCodec(tid, typeName, codecs); } break; } case CTYPE_NAMEDTUPLE: { + let typeName: string | null = null; + if (isProtoV2) { + typeName = frb.readString(); + // @ts-expect-error + const isSchemaDefined = frb.readBoolean(); + const ancestorCount = frb.readUInt16(); + for (var i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + // @ts-expect-error + const ancestorCodec = cl[ancestorPos]; + } + } + const els = frb.readUInt16(); const codecs = new Array(els); const names = new Array(els); @@ -419,11 +536,23 @@ export class CodecsRegistry { } codecs[i] = subCodec; } - res = new NamedTupleCodec(tid, codecs, names); + res = new NamedTupleCodec(tid, typeName, codecs, names); break; } case CTYPE_ENUM: { + let typeName: string | null = null; + if (isProtoV2) { + typeName = frb.readString(); + // @ts-expect-error + const isSchemaDefined = frb.readBoolean(); + const ancestorCount = frb.readUInt16(); + for (var i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + // @ts-expect-error + const ancestorCodec = cl[ancestorPos]; + } + } /* There's no way to customize ordering in JS, so we simply ignore that information and unpack enums into simple strings. @@ -433,11 +562,24 @@ export class CodecsRegistry { for (let i = 0; i < els; i++) { values.push(frb.readString()); } - res = new EnumCodec(tid, null, values); + res = new EnumCodec(tid, typeName, null, values); break; } case CTYPE_RANGE: { + let typeName: string | null = null; + if (isProtoV2) { + typeName = frb.readString(); + // @ts-expect-error + const isSchemaDefined = frb.readBoolean(); + const ancestorCount = frb.readUInt16(); + for (var i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + // @ts-expect-error + const ancestorCodec = cl[ancestorPos]; + } + } + const pos = frb.readUInt16(); const subCodec = cl[pos]; if (subCodec == null) { @@ -445,7 +587,21 @@ export class CodecsRegistry { "could not build range codec: missing subcodec", ); } - res = new RangeCodec(tid, subCodec); + res = new RangeCodec(tid, typeName, subCodec); + break; + } + + case CTYPE_OBJECT: { + // Ignore + frb.discard(frb.length); + res = NULL_CODEC; + break; + } + + case CTYPE_COMPOUND: { + // Ignore + frb.discard(frb.length); + res = NULL_CODEC; break; } diff --git a/packages/driver/src/codecs/tuple.ts b/packages/driver/src/codecs/tuple.ts index 249761c48..d7b82f709 100644 --- a/packages/driver/src/codecs/tuple.ts +++ b/packages/driver/src/codecs/tuple.ts @@ -30,10 +30,12 @@ import { export class TupleCodec extends Codec implements ICodec, IArgsCodec { private subCodecs: ICodec[]; + public typeName: string | null; - constructor(tid: uuid, codecs: ICodec[]) { + constructor(tid: uuid, typeName: string | null, codecs: ICodec[]) { super(tid); this.subCodecs = codecs; + this.typeName = typeName; } encode(buf: WriteBuffer, object: any, allowNull = false): void { diff --git a/packages/driver/src/primitives/buffer.ts b/packages/driver/src/primitives/buffer.ts index 3b86653bd..e572eb2e6 100644 --- a/packages/driver/src/primitives/buffer.ts +++ b/packages/driver/src/primitives/buffer.ts @@ -882,9 +882,9 @@ export class ReadBuffer { return this.len - this.pos; } - finish(): void { + finish(message?: string): void { if (this.len !== this.pos) { - throw new BufferError("unexpected trailing data in buffer"); + throw new BufferError(message ?? "unexpected trailing data in buffer"); } } @@ -1005,6 +1005,10 @@ export class ReadBuffer { return ret; } + readBoolean(): boolean { + return this.readUInt8() != 0; + } + readBuffer(size: number): Uint8Array { if (this.pos + size > this.len) { throw new BufferError("buffer overread"); diff --git a/packages/driver/test/datetime.test.ts b/packages/driver/test/datetime.test.ts index b267e06f9..18b46a045 100644 --- a/packages/driver/test/datetime.test.ts +++ b/packages/driver/test/datetime.test.ts @@ -16,7 +16,10 @@ import { ReadBuffer, WriteBuffer } from "../src/primitives/buffer"; // Tests adapted from https://github.com/edgedb/edgedb-rust/blob/master/edgedb-protocol/tests/datetime_chrono.rs test("datetime", () => { - const codec = new DateTimeCodec(KNOWN_TYPENAMES.get("std::datetime")!); + const codec = new DateTimeCodec( + KNOWN_TYPENAMES.get("std::datetime")!, + "std::datetime", + ); const tests: [string, string][] = [ ["252455615999999999", "+010000-01-01T00:00:00.000Z"], // maximum @@ -55,6 +58,7 @@ test("datetime", () => { test("local_datetime", () => { const codec = new LocalDateTimeCodec( KNOWN_TYPENAMES.get("cal::local_datetime")!, + "cal::local_datetime", ); const tests: [string, string, string][] = [ @@ -181,7 +185,10 @@ test("local_datetime", () => { }); test("local_time", () => { - const codec = new LocalTimeCodec(KNOWN_TYPENAMES.get("cal::local_time")!); + const codec = new LocalTimeCodec( + KNOWN_TYPENAMES.get("cal::local_time")!, + "cal::local_time", + ); const tests: [string, string][] = [ ["00:00:00.000000000", "0"], // minimum @@ -211,7 +218,10 @@ test("local_time", () => { }); test("duration", () => { - const codec = new DurationCodec(KNOWN_TYPENAMES.get("std::duration")!); + const codec = new DurationCodec( + KNOWN_TYPENAMES.get("std::duration")!, + "std::duration", + ); const tests: [string, string][] = [ ["PT0S", "0"], // "Zero" From 85177b43a08deacd32a8772fdce62aae6bd5f1f6 Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Mon, 16 Oct 2023 12:25:30 -0400 Subject: [PATCH 03/14] Fix lint issues --- packages/driver/src/codecs/registry.ts | 18 +++++++++--------- packages/driver/src/primitives/buffer.ts | 2 +- 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/packages/driver/src/codecs/registry.ts b/packages/driver/src/codecs/registry.ts index 585d46029..b7f10caba 100644 --- a/packages/driver/src/codecs/registry.ts +++ b/packages/driver/src/codecs/registry.ts @@ -309,7 +309,7 @@ export class CodecsRegistry { case CTYPE_SHAPE: case CTYPE_INPUT_SHAPE: { - if (t == CTYPE_SHAPE && isProtoV2) { + if (t === CTYPE_SHAPE && isProtoV2) { // @ts-expect-error const isEphemeralFreeShape = frb.readBoolean(); // @ts-expect-error @@ -348,7 +348,7 @@ export class CodecsRegistry { flags[i] = flag!; cards[i] = card!; - if (t == CTYPE_SHAPE && isProtoV2) { + if (t === CTYPE_SHAPE && isProtoV2) { const sourceTypePos = frb.readUInt16(); // @ts-expect-error const sourceType = cl[sourceTypePos]; @@ -382,7 +382,7 @@ export class CodecsRegistry { const ancestorCount = frb.readUInt16(); const ancestors: ICodec[] = []; - for (var i = 0; i < ancestorCount; i++) { + for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); const ancestorCodec = cl[ancestorPos]; if (ancestorCodec == null) { @@ -400,7 +400,7 @@ export class CodecsRegistry { ancestors.push(ancestorCodec); } - if (ancestorCount == 0) { + if (ancestorCount === 0) { res = this.customScalarCodecs.get(tid) ?? SCALAR_CODECS.get(tid); if (res == null) { if (KNOWN_TYPES.has(tid)) { @@ -448,7 +448,7 @@ export class CodecsRegistry { // @ts-expect-error const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); - for (var i = 0; i < ancestorCount; i++) { + for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); // @ts-expect-error const ancestorCodec = cl[ancestorPos]; @@ -480,7 +480,7 @@ export class CodecsRegistry { // @ts-expect-error const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); - for (var i = 0; i < ancestorCount; i++) { + for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); // @ts-expect-error const ancestorCodec = cl[ancestorPos]; @@ -514,7 +514,7 @@ export class CodecsRegistry { // @ts-expect-error const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); - for (var i = 0; i < ancestorCount; i++) { + for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); // @ts-expect-error const ancestorCodec = cl[ancestorPos]; @@ -547,7 +547,7 @@ export class CodecsRegistry { // @ts-expect-error const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); - for (var i = 0; i < ancestorCount; i++) { + for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); // @ts-expect-error const ancestorCodec = cl[ancestorPos]; @@ -573,7 +573,7 @@ export class CodecsRegistry { // @ts-expect-error const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); - for (var i = 0; i < ancestorCount; i++) { + for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); // @ts-expect-error const ancestorCodec = cl[ancestorPos]; diff --git a/packages/driver/src/primitives/buffer.ts b/packages/driver/src/primitives/buffer.ts index e572eb2e6..3341ea791 100644 --- a/packages/driver/src/primitives/buffer.ts +++ b/packages/driver/src/primitives/buffer.ts @@ -1006,7 +1006,7 @@ export class ReadBuffer { } readBoolean(): boolean { - return this.readUInt8() != 0; + return this.readUInt8() !== 0; } readBuffer(size: number): Uint8Array { From d1fa49d109cfdfbd233b0e90678bc4b2d4327804 Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Fri, 20 Oct 2023 11:54:08 -0400 Subject: [PATCH 04/14] Add reason message to fix Deno failures --- packages/driver/src/codecs/registry.ts | 28 +++++++++++++------------- 1 file changed, 14 insertions(+), 14 deletions(-) diff --git a/packages/driver/src/codecs/registry.ts b/packages/driver/src/codecs/registry.ts index b7f10caba..be7b19608 100644 --- a/packages/driver/src/codecs/registry.ts +++ b/packages/driver/src/codecs/registry.ts @@ -310,9 +310,9 @@ export class CodecsRegistry { case CTYPE_SHAPE: case CTYPE_INPUT_SHAPE: { if (t === CTYPE_SHAPE && isProtoV2) { - // @ts-expect-error + // @ts-expect-error reserved for future use const isEphemeralFreeShape = frb.readBoolean(); - // @ts-expect-error + // @ts-expect-error reserved for future use const objTypePos = frb.readUInt16(); } @@ -350,7 +350,7 @@ export class CodecsRegistry { if (t === CTYPE_SHAPE && isProtoV2) { const sourceTypePos = frb.readUInt16(); - // @ts-expect-error + // @ts-expect-error reserved for future use const sourceType = cl[sourceTypePos]; } } @@ -377,7 +377,7 @@ export class CodecsRegistry { case CTYPE_SCALAR: { if (isProtoV2) { const typeName = frb.readString(); - // @ts-expect-error + // @ts-expect-error reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); @@ -445,12 +445,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error + // @ts-expect-error reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error + // @ts-expect-error reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -477,12 +477,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error + // @ts-expect-error reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error + // @ts-expect-error reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -511,12 +511,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error + // @ts-expect-error reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error + // @ts-expect-error reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -544,12 +544,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error + // @ts-expect-error reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error + // @ts-expect-error reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -570,12 +570,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error + // @ts-expect-error reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error + // @ts-expect-error reserved for future use const ancestorCodec = cl[ancestorPos]; } } From f80117a4cdd0e98af6afa4808f741df9b6a55e84 Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Thu, 26 Oct 2023 10:08:57 -0400 Subject: [PATCH 05/14] Add deno config `noUnusedLocals` is false by default, so we needed to set it in config --- packages/driver/buildDeno.ts | 9 +++++++-- packages/driver/src/codecs/registry.ts | 28 +++++++++++++------------- packages/driver/src/deno.json | 5 +++++ 3 files changed, 26 insertions(+), 16 deletions(-) create mode 100644 packages/driver/src/deno.json diff --git a/packages/driver/buildDeno.ts b/packages/driver/buildDeno.ts index 38363991a..3ed4eb364 100644 --- a/packages/driver/buildDeno.ts +++ b/packages/driver/buildDeno.ts @@ -12,10 +12,15 @@ await run({ destDir: "../deno", destEntriesToClean: ["_src", "mod.ts"], sourceFilter: (path) => { - return !/cli\.mts$/.test(path); + const doesMatch = + !/cli\.mts$/.test(path) || + !/\/syntax\//.test(path) || + path.includes("deno.json"); + return doesMatch; }, pathRewriteRules: [ { match: /^src\/index.node.ts$/, replace: "mod.ts" }, + { match: /^src\/deno.json$/, replace: "deno.json" }, { match: /^src\//, replace: "_src/" }, ], injectImports: [ @@ -28,7 +33,7 @@ await run({ from: "src/globals.deno.ts", }, ], -}).then(async () => +}).then(() => run({ sourceDir: "./test", destDir: "../deno/test", diff --git a/packages/driver/src/codecs/registry.ts b/packages/driver/src/codecs/registry.ts index be7b19608..ab4b03066 100644 --- a/packages/driver/src/codecs/registry.ts +++ b/packages/driver/src/codecs/registry.ts @@ -310,9 +310,9 @@ export class CodecsRegistry { case CTYPE_SHAPE: case CTYPE_INPUT_SHAPE: { if (t === CTYPE_SHAPE && isProtoV2) { - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isEphemeralFreeShape = frb.readBoolean(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const objTypePos = frb.readUInt16(); } @@ -350,7 +350,7 @@ export class CodecsRegistry { if (t === CTYPE_SHAPE && isProtoV2) { const sourceTypePos = frb.readUInt16(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const sourceType = cl[sourceTypePos]; } } @@ -377,7 +377,7 @@ export class CodecsRegistry { case CTYPE_SCALAR: { if (isProtoV2) { const typeName = frb.readString(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); @@ -445,12 +445,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -477,12 +477,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -511,12 +511,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -544,12 +544,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const ancestorCodec = cl[ancestorPos]; } } @@ -570,12 +570,12 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error reserved for future use + // @ts-expect-error: reserved for future use const ancestorCodec = cl[ancestorPos]; } } diff --git a/packages/driver/src/deno.json b/packages/driver/src/deno.json new file mode 100644 index 000000000..650c533a2 --- /dev/null +++ b/packages/driver/src/deno.json @@ -0,0 +1,5 @@ +{ + "compilerOptions": { + "noUnusedLocals": true + } +} From 4e3c34d7e621af32da23efedcc12b93bf926e123 Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Thu, 26 Oct 2023 20:39:57 -0400 Subject: [PATCH 06/14] Add MultiRange protocol v2 support --- packages/driver/src/codecs/range.ts | 4 +++- packages/driver/src/codecs/registry.ts | 14 +++++++++++++- 2 files changed, 16 insertions(+), 2 deletions(-) diff --git a/packages/driver/src/codecs/range.ts b/packages/driver/src/codecs/range.ts index db0d3c9f0..58eac4255 100644 --- a/packages/driver/src/codecs/range.ts +++ b/packages/driver/src/codecs/range.ts @@ -127,10 +127,12 @@ export class MultiRangeCodec extends Codec implements ICodec { readonly tsModule = "edgedb"; private subCodec: ICodec; + public typeName: string | null; - constructor(tid: uuid, subCodec: ICodec) { + constructor(tid: uuid, typeName: string | null, subCodec: ICodec) { super(tid); this.subCodec = subCodec; + this.typeName = typeName; } encode(buf: WriteBuffer, obj: any): void { diff --git a/packages/driver/src/codecs/registry.ts b/packages/driver/src/codecs/registry.ts index ab4b03066..dd3741b25 100644 --- a/packages/driver/src/codecs/registry.ts +++ b/packages/driver/src/codecs/registry.ts @@ -606,6 +606,18 @@ export class CodecsRegistry { } case CTYPE_MULTIRANGE: { + let typeName: string | null = null; + if (isProtoV2) { + typeName = frb.readString(); + // @ts-expect-error: reserved for future use + const isSchemaDefined = frb.readBoolean(); + const ancestorCount = frb.readUInt16(); + for (let i = 0; i < ancestorCount; i++) { + const ancestorPos = frb.readUInt16(); + // @ts-expect-error: reserved for future use + const ancestorCodec = cl[ancestorPos]; + } + } const pos = frb.readUInt16(); const subCodec = cl[pos]; if (subCodec == null) { @@ -613,7 +625,7 @@ export class CodecsRegistry { "could not build range codec: missing subcodec", ); } - res = new MultiRangeCodec(tid, subCodec); + res = new MultiRangeCodec(tid, typeName, subCodec); break; } } From b782cbeb00ac45c59a0c6f973eb19ab6535674ab Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Mon, 30 Oct 2023 09:27:49 -0400 Subject: [PATCH 07/14] Remove debug logging --- packages/driver/buildDeno.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/driver/buildDeno.ts b/packages/driver/buildDeno.ts index 3ed4eb364..c1bb89c58 100644 --- a/packages/driver/buildDeno.ts +++ b/packages/driver/buildDeno.ts @@ -12,11 +12,11 @@ await run({ destDir: "../deno", destEntriesToClean: ["_src", "mod.ts"], sourceFilter: (path) => { - const doesMatch = + return ( !/cli\.mts$/.test(path) || !/\/syntax\//.test(path) || - path.includes("deno.json"); - return doesMatch; + path.includes("deno.json") + ); }, pathRewriteRules: [ { match: /^src\/index.node.ts$/, replace: "mod.ts" }, From 3555a55c172826d9251c6d6016067c6732d05365 Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Thu, 29 Aug 2024 20:37:20 -0400 Subject: [PATCH 08/14] Prettier --- packages/driver/src/codecs/array.ts | 2 +- packages/driver/src/codecs/enum.ts | 2 +- packages/driver/src/codecs/namedtuple.ts | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/driver/src/codecs/array.ts b/packages/driver/src/codecs/array.ts index 878396048..45473309d 100644 --- a/packages/driver/src/codecs/array.ts +++ b/packages/driver/src/codecs/array.ts @@ -33,7 +33,7 @@ export class ArrayCodec extends Codec implements ICodec { tid: uuid, typeName: string | null, subCodec: ICodec, - len: number + len: number, ) { super(tid); this.subCodec = subCodec; diff --git a/packages/driver/src/codecs/enum.ts b/packages/driver/src/codecs/enum.ts index a4b70fcab..ecd0365de 100644 --- a/packages/driver/src/codecs/enum.ts +++ b/packages/driver/src/codecs/enum.ts @@ -26,7 +26,7 @@ export class EnumCodec extends StrCodec implements ICodec { tid: uuid, typeName: string | null, derivedFromTid: uuid | null = null, - values: string[] + values: string[], ) { super(tid, typeName, derivedFromTid); this.values = values; diff --git a/packages/driver/src/codecs/namedtuple.ts b/packages/driver/src/codecs/namedtuple.ts index 710d1286c..f4a7cafdd 100644 --- a/packages/driver/src/codecs/namedtuple.ts +++ b/packages/driver/src/codecs/namedtuple.ts @@ -38,7 +38,7 @@ export class NamedTupleCodec extends Codec implements ICodec, IArgsCodec { tid: uuid, typeName: string | null, codecs: ICodec[], - names: string[] + names: string[], ) { super(tid); this.subCodecs = codecs; From e04ec46bb28e3d97704e8db73860135b3fb20b12 Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Thu, 29 Aug 2024 21:22:21 -0400 Subject: [PATCH 09/14] Use correct combinator for deno source filter --- packages/driver/buildDeno.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/driver/buildDeno.ts b/packages/driver/buildDeno.ts index c1bb89c58..785912f4e 100644 --- a/packages/driver/buildDeno.ts +++ b/packages/driver/buildDeno.ts @@ -13,9 +13,9 @@ await run({ destEntriesToClean: ["_src", "mod.ts"], sourceFilter: (path) => { return ( - !/cli\.mts$/.test(path) || - !/\/syntax\//.test(path) || - path.includes("deno.json") + !path.endsWith("cli.mts") && + !path.includes("/syntax/") && + path.endsWith("deno.json") ); }, pathRewriteRules: [ From 9bc2a8ece88f62e000e96c52bbbcaa7af97d53ec Mon Sep 17 00:00:00 2001 From: Scott Trinh Date: Thu, 29 Aug 2024 21:39:25 -0400 Subject: [PATCH 10/14] Remove extra files being filtered I don't recall what this was about, but it was causing all of the deno files to be missing. --- packages/driver/buildDeno.ts | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/packages/driver/buildDeno.ts b/packages/driver/buildDeno.ts index 785912f4e..34a9faaf0 100644 --- a/packages/driver/buildDeno.ts +++ b/packages/driver/buildDeno.ts @@ -12,11 +12,7 @@ await run({ destDir: "../deno", destEntriesToClean: ["_src", "mod.ts"], sourceFilter: (path) => { - return ( - !path.endsWith("cli.mts") && - !path.includes("/syntax/") && - path.endsWith("deno.json") - ); + return !path.endsWith("cli.mts"); }, pathRewriteRules: [ { match: /^src\/index.node.ts$/, replace: "mod.ts" }, From ddcd0ef611bf88b21058a8ceb734a8d9201e4456 Mon Sep 17 00:00:00 2001 From: Dijana Pavlovic Date: Fri, 30 Aug 2024 14:13:07 +0200 Subject: [PATCH 11/14] Update tests to be aware of tlsServerName --- packages/driver/test/connection-config.test.ts | 4 ++++ packages/driver/test/shared-client-testcases | 2 +- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/packages/driver/test/connection-config.test.ts b/packages/driver/test/connection-config.test.ts index 0213b88b2..fbff1d49a 100644 --- a/packages/driver/test/connection-config.test.ts +++ b/packages/driver/test/connection-config.test.ts @@ -192,6 +192,7 @@ interface ConnectionResult { password: string | null; tlsCAData: string | null; tlsSecurity: boolean; + tlsServerName: string | null; serverSettings: { [key: string]: string }; waitUntilAvailable: string; } @@ -254,6 +255,7 @@ async function runConnectionTest(testcase: ConnectionTestCase): Promise { user: connectionParams.user, password: connectionParams.password ?? null, secretKey: connectionParams.secretKey ?? null, + tlsServerName: connectionParams.tlsServerName ?? null, tlsCAData: connectionParams._tlsCAData, tlsSecurity: connectionParams.tlsSecurity, serverSettings: connectionParams.serverSettings, @@ -456,6 +458,7 @@ test("logging, inProject, fromProject, fromEnv", async () => { password: null, tlsCAData: null, tlsSecurity: "strict", + tlsServerName: null, serverSettings: {}, }; @@ -608,6 +611,7 @@ test("logging, inProject, fromProject, fromEnv", async () => { user: connectionParams.user, password: connectionParams.password ?? null, tlsCAData: connectionParams._tlsCAData, + tlsServerName: connectionParams.tlsServerName ?? null, tlsSecurity: connectionParams.tlsSecurity, serverSettings: connectionParams.serverSettings, }).toEqual(testcase.result); diff --git a/packages/driver/test/shared-client-testcases b/packages/driver/test/shared-client-testcases index 9bddb7beb..071f60430 160000 --- a/packages/driver/test/shared-client-testcases +++ b/packages/driver/test/shared-client-testcases @@ -1 +1 @@ -Subproject commit 9bddb7bebecf41510a31c57e1438c35b2592d35d +Subproject commit 071f60430207e331b847e1dcc10cba19c5b5f2f4 From a0d40f3d9dd885b6dca528f28f0cf5ab89d4c460 Mon Sep 17 00:00:00 2001 From: Dijana Pavlovic Date: Mon, 2 Sep 2024 14:06:39 +0200 Subject: [PATCH 12/14] Prepend unused vars w underscore, no ts errors --- packages/driver/src/codecs/registry.ts | 48 +++++++++----------------- packages/driver/tsconfig.json | 3 +- 2 files changed, 18 insertions(+), 33 deletions(-) diff --git a/packages/driver/src/codecs/registry.ts b/packages/driver/src/codecs/registry.ts index dd3741b25..490622b13 100644 --- a/packages/driver/src/codecs/registry.ts +++ b/packages/driver/src/codecs/registry.ts @@ -310,10 +310,8 @@ export class CodecsRegistry { case CTYPE_SHAPE: case CTYPE_INPUT_SHAPE: { if (t === CTYPE_SHAPE && isProtoV2) { - // @ts-expect-error: reserved for future use - const isEphemeralFreeShape = frb.readBoolean(); - // @ts-expect-error: reserved for future use - const objTypePos = frb.readUInt16(); + const _isEphemeralFreeShape = frb.readBoolean(); + const _objTypePos = frb.readUInt16(); } const els = frb.readUInt16(); @@ -350,8 +348,7 @@ export class CodecsRegistry { if (t === CTYPE_SHAPE && isProtoV2) { const sourceTypePos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const sourceType = cl[sourceTypePos]; + const _sourceType = cl[sourceTypePos]; } } @@ -377,8 +374,7 @@ export class CodecsRegistry { case CTYPE_SCALAR: { if (isProtoV2) { const typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); const ancestors: ICodec[] = []; @@ -445,13 +441,11 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const ancestorCodec = cl[ancestorPos]; + const _ancestorCodec = cl[ancestorPos]; } } @@ -477,13 +471,11 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const ancestorCodec = cl[ancestorPos]; + const _ancestorCodec = cl[ancestorPos]; } } @@ -511,13 +503,11 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const ancestorCodec = cl[ancestorPos]; + const _ancestorCodec = cl[ancestorPos]; } } @@ -544,13 +534,11 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const ancestorCodec = cl[ancestorPos]; + const _ancestorCodec = cl[ancestorPos]; } } /* There's no way to customize ordering in JS, so we @@ -570,13 +558,11 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const ancestorCodec = cl[ancestorPos]; + const _ancestorCodec = cl[ancestorPos]; } } @@ -609,13 +595,11 @@ export class CodecsRegistry { let typeName: string | null = null; if (isProtoV2) { typeName = frb.readString(); - // @ts-expect-error: reserved for future use - const isSchemaDefined = frb.readBoolean(); + const _isSchemaDefined = frb.readBoolean(); const ancestorCount = frb.readUInt16(); for (let i = 0; i < ancestorCount; i++) { const ancestorPos = frb.readUInt16(); - // @ts-expect-error: reserved for future use - const ancestorCodec = cl[ancestorPos]; + const _ancestorCodec = cl[ancestorPos]; } } const pos = frb.readUInt16(); diff --git a/packages/driver/tsconfig.json b/packages/driver/tsconfig.json index 508864bbe..fa65cc784 100644 --- a/packages/driver/tsconfig.json +++ b/packages/driver/tsconfig.json @@ -2,7 +2,8 @@ "extends": "@repo/tsconfig/base.json", "compilerOptions": { "outDir": "./dist", - "downlevelIteration": true + "downlevelIteration": true, + "noUnusedLocals": false }, "include": ["src/**/*"], "exclude": ["src/cli.mts", "src/**/*.deno.ts"] From c7f175ec4c56c17cd7beaa56b06b4c6996eb9e02 Mon Sep 17 00:00:00 2001 From: Dijana Pavlovic Date: Mon, 2 Sep 2024 22:33:34 +0200 Subject: [PATCH 13/14] Update proto ver in fetchConn --- packages/driver/src/deno.json | 2 +- packages/driver/src/fetchConn.ts | 8 +++++++- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/packages/driver/src/deno.json b/packages/driver/src/deno.json index 650c533a2..75446423b 100644 --- a/packages/driver/src/deno.json +++ b/packages/driver/src/deno.json @@ -1,5 +1,5 @@ { "compilerOptions": { - "noUnusedLocals": true + "noUnusedLocals": false } } diff --git a/packages/driver/src/fetchConn.ts b/packages/driver/src/fetchConn.ts index 7fe201b19..84ef322b4 100644 --- a/packages/driver/src/fetchConn.ts +++ b/packages/driver/src/fetchConn.ts @@ -42,6 +42,7 @@ import Event from "./primitives/event"; import { type AuthenticatedFetch, getAuthenticatedFetch } from "./utils"; const PROTO_MIME = `application/x.edgedb.v_${PROTO_VER[0]}_${PROTO_VER[1]}.binary'`; +const PROTO_MIME_RE = /application\/x\.edgedb\.v_(\d+)_(\d+)\.binary/; const STUDIO_CAPABILITIES = (RESTRICTED_CAPABILITIES | @@ -105,9 +106,14 @@ class BaseFetchConnection extends BaseRawConnection { ); } + const contentType = resp.headers.get("content-type"); + const matchProtoVer = contentType?.match(PROTO_MIME_RE); + if (matchProtoVer) { + this.protocolVersion = [+matchProtoVer[1], +matchProtoVer[2]]; + } + const respData = await resp.arrayBuffer(); const buf = new Uint8Array(respData); - try { this.buffer.feed(buf); } catch (e: any) { From 701926dd1cc869312e1ae605e21a9a950f7f56e0 Mon Sep 17 00:00:00 2001 From: Dijana Pavlovic Date: Tue, 3 Sep 2024 15:38:57 +0200 Subject: [PATCH 14/14] Add support for SNI when connecting to the instance (#1088) I will try to add these tomorrow: Documentation is missing: - SNI, - Cloud Profile, - tlsCA maybe should be documented too, - `credentials` is not documented. **More things to be done**: add SNI to credentials.json in the cli, and later read it from there: `Oh and to address this specifically, although very marginal, we may want to update the CLI to include tls_server_name in credentials.json if --tls-server-name was specified while linking an instance.` Atm CLOUD_PROFILE is taken from env var when using options. And when using env vars it is not used at all? --------- Co-authored-by: Scott Trinh --- packages/driver/src/conUtils.ts | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/packages/driver/src/conUtils.ts b/packages/driver/src/conUtils.ts index 7e360c2ec..a9e074584 100644 --- a/packages/driver/src/conUtils.ts +++ b/packages/driver/src/conUtils.ts @@ -679,7 +679,7 @@ async function parseConnectDsnAndArgs( .catch(() => null); if (instName !== null) { - const [cloudProfile, database] = await Promise.all([ + const [cloudProfile, _database, branch] = await Promise.all([ serverUtils .readFileUtf8(stashDir, "cloud-profile") .then((name) => name.trim()) @@ -694,13 +694,26 @@ async function parseConnectDsnAndArgs( .catch(() => undefined), ]); + let database = _database; + + if (database !== undefined && branch !== undefined) { + if (database !== branch) { + throw new InterfaceError( + "Both database and branch exist in the config dir and don't match.", + ); + } else { + database = undefined; + } + } + await resolveConfigOptions( resolvedConfig, - { instanceName: instName, cloudProfile, database }, + { instanceName: instName, cloudProfile, database, branch }, { instanceName: `project linked instance ('${instName}')`, cloudProfile: `project defined cloud instance ('${cloudProfile}')`, database: `project default database`, + branch: `project default branch`, }, "", serverUtils,