ddb-table
Version:
Strongly typed library for querying and modeling DynamoDB documents.
190 lines (189 loc) • 73.8 kB
JavaScript
"use strict";
exports.__esModule = true;
exports.default = void 0;
// Docs:
// https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/Expressions.OperatorsAndFunctions.html
const OppsiteOperandMatch = {
AND: /[ )]OR[ (]/i,
OR: /[ )]AND[ (]/i
};
class ConditionExpression {
static getAttributeTypeString(type) {
if (typeof type === 'string') {
return type;
}
switch (type) {
case String:
return 'S';
case Number:
return 'N';
case ArrayBuffer:
case Buffer:
return 'B';
case Boolean:
return 'BOOL';
case null:
return 'NULL';
case Array:
return 'L';
case Object:
return 'M';
default:
if (Array.isArray(type) && type.length === 1) {
switch (type[0]) {
case String:
return 'SS';
case Number:
return 'NS';
case Buffer:
case ArrayBuffer:
return 'BS';
}
}
throw new TypeError(`Unknown attribute type: ${type}`);
}
}
constructor(names, values, init) {
this.names = void 0;
this.values = void 0;
this.expression = void 0;
this.names = names;
this.values = values;
if (init) {
if (typeof init === 'string') {
this.expression = init.trim().replace(/\s+/g, ' ').replace(/\(\s+/g, '(').replace(/\s+\)/g, ')') || '';
} else {
this.expression = init.expression;
}
} else {
this.expression = '';
}
}
serialize() {
return this.expression || undefined;
}
name(...path) {
return this.names.add(...path);
}
value(key, value) {
return this.values.add(key, value);
}
eq(path, value) {
return this.comparator('=', path, value);
}
nq(path, value) {
return this.comparator('<>', path, value);
}
lt(path, value) {
return this.comparator('<', path, value);
}
lte(path, value) {
return this.comparator('<=', path, value);
}
gt(path, value) {
return this.comparator('>', path, value);
}
gte(path, value) {
return this.comparator('>=', path, value);
}
comparator(comparator, path, value) {
if (!Array.isArray(path)) {
path = [path];
}
const pathName = this.names.add(...path);
const valueStr = this.calcValueStr(path, pathName, value);
return this.append('AND', `${pathName} ${comparator} ${valueStr}`);
}
between(path, from, to) {
if (!Array.isArray(path)) {
path = [path];
}
const pathName = this.names.add(...path);
const fromStr = this.calcValueStr(path, pathName, from);
const toStr = this.calcValueStr(path, pathName, to);
return this.append('AND', `${pathName} BETWEEN ${fromStr} AND ${toStr}`);
}
attributeExists(path) {
if (!Array.isArray(path)) {
path = [path];
}
return this.fn('attribute_exists', this.names.add(...path));
}
attributeNotExists(path) {
if (!Array.isArray(path)) {
path = [path];
}
return this.fn('attribute_not_exists', this.names.add(...path));
}
attributeType(path, type) {
if (!Array.isArray(path)) {
path = [path];
}
return this.fn('attribute_type', this.names.add(...path), this.values.add(String(path[path.length - 1]), ConditionExpression.getAttributeTypeString(type)));
}
beginsWith(path, substr) {
if (!Array.isArray(path)) {
path = [path];
}
return this.fn('begins_with', this.names.add(...path), this.values.add(String(path[path.length - 1]), substr));
}
contains(path,
// eslint-disable-next-line @typescript-eslint/no-explicit-any
operand) {
if (!Array.isArray(path)) {
path = [path];
}
return this.fn('contains', this.names.add(...path), typeof operand === 'function' ? operand(new ConditionExpression(this.names, this.values)) : this.values.add(String(path[path.length - 1]), operand));
}
size(path) {
if (!Array.isArray(path)) {
path = [path];
}
return this.fn('size', this.names.add(...path));
}
fn(fnName, ...args) {
return this.append('AND', `${fnName}(${args.join(', ')})`);
}
and(fn) {
return this.append('AND', fn(new ConditionExpression(this.names, this.values)));
}
or(fn) {
return this.append('OR', fn(new ConditionExpression(this.names, this.values)));
}
not(fn) {
return this.append('AND', fn(new ConditionExpression(this.names, this.values)), 'NOT');
}
append(operand, item, prefix) {
const condition = typeof item === 'string' ? new ConditionExpression(this.names, this.values, item) : item;
const conditionStr = condition.serialize();
if (conditionStr) {
if (this.expression) {
if (this.expression.match(OppsiteOperandMatch[operand]) && !this.expression.match(/^\(.*\)$/)) {
this.expression = `(${this.expression})`;
}
this.expression += ` ${operand} `;
}
if (prefix) {
this.expression += `${prefix} `;
}
if (conditionStr.match(OppsiteOperandMatch[operand]) && !conditionStr.match(/^\(.*\)$/)) {
this.expression += `(${conditionStr})`;
} else {
this.expression += conditionStr;
}
}
return this;
}
calcValueStr(path, pathName, value) {
if (typeof value !== 'function') {
return this.values.add(String(path[path.length - 1]), value);
}
const res = value(pathName, new ConditionExpression(this.names, this.values));
if (typeof res === 'string') {
return res;
}
return res.serialize();
}
}
exports.default = ConditionExpression;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["OppsiteOperandMatch","AND","OR","ConditionExpression","getAttributeTypeString","type","String","Number","ArrayBuffer","Buffer","Boolean","Array","Object","isArray","length","TypeError","constructor","names","values","init","expression","trim","replace","serialize","undefined","name","path","add","value","key","eq","comparator","nq","lt","lte","gt","gte","pathName","valueStr","calcValueStr","append","between","from","to","fromStr","toStr","attributeExists","fn","attributeNotExists","attributeType","beginsWith","substr","contains","operand","size","fnName","args","join","and","or","not","item","prefix","condition","conditionStr","match","res","exports","default"],"sources":["../../src/expressions/ConditionExpression.ts"],"sourcesContent":["import ExpressionAttributeNames from './ExpressionAttributeNames';\nimport ExpressionAttributeValues from './ExpressionAttributeValues';\n\n// Docs:\n// https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/Expressions.OperatorsAndFunctions.html\n\nexport type ConditionGenerator<T> = (\n  condition: ConditionExpression<T>,\n) => ConditionExpression<T> | string;\n\nexport type ValueFn<T, V> =\n  | V\n  | ((\n      path: string,\n      condition: ConditionExpression<T>,\n    ) => ConditionExpression<T> | string);\n\nexport type Comparator = '=' | '<>' | '<' | '<=' | '>' | '>=';\n\nexport type FunctionName =\n  | 'attribute_exists'\n  | 'attribute_not_exists'\n  | 'attribute_type'\n  | 'begins_with'\n  | 'contains'\n  | 'size';\n\nexport type AttributeTypeString =\n  | 'S'\n  | 'SS'\n  | 'N'\n  | 'NS'\n  | 'B'\n  | 'BS'\n  | 'BOOL'\n  | 'NULL'\n  | 'L'\n  | 'M';\n\nexport type AttributeType =\n  | AttributeTypeString\n  | StringConstructor\n  | [StringConstructor]\n  | NumberConstructor\n  | [NumberConstructor]\n  | ArrayBufferConstructor\n  | typeof Buffer\n  | [ArrayBufferConstructor | typeof Buffer]\n  | BooleanConstructor\n  | null\n  | ArrayConstructor\n  | ObjectConstructor;\n\nconst OppsiteOperandMatch = {\n  AND: /[ )]OR[ (]/i,\n  OR: /[ )]AND[ (]/i,\n};\n\nexport type ContainsOperand<T, V> =\n  | ([NonNullable<V>] extends [string]\n      ? NonNullable<V>\n      : [NonNullable<V>] extends [Set<infer SV>]\n      ? SV\n      : never)\n  | ((exp: ConditionExpression<T>) => string);\n\nexport default class ConditionExpression<T> {\n  private names: ExpressionAttributeNames<T>;\n  private values: ExpressionAttributeValues;\n\n  private expression: string;\n\n  public static getAttributeTypeString(\n    type: AttributeType,\n  ): AttributeTypeString {\n    if (typeof type === 'string') {\n      return type;\n    }\n\n    switch (type) {\n      case String:\n        return 'S';\n\n      case Number:\n        return 'N';\n\n      case ArrayBuffer:\n      case Buffer:\n        return 'B';\n\n      case Boolean:\n        return 'BOOL';\n\n      case null:\n        return 'NULL';\n\n      case Array:\n        return 'L';\n\n      case Object:\n        return 'M';\n\n      default:\n        if (Array.isArray(type) && type.length === 1) {\n          switch (type[0]) {\n            case String:\n              return 'SS';\n\n            case Number:\n              return 'NS';\n\n            case Buffer:\n            case ArrayBuffer:\n              return 'BS';\n          }\n        }\n\n        throw new TypeError(`Unknown attribute type: ${type}`);\n    }\n  }\n\n  public constructor(\n    names: ExpressionAttributeNames<T>,\n    values: ExpressionAttributeValues,\n    init?: string | ConditionExpression<T>,\n  ) {\n    this.names = names;\n    this.values = values;\n\n    if (init) {\n      if (typeof init === 'string') {\n        this.expression =\n          init\n            .trim()\n            .replace(/\\s+/g, ' ')\n            .replace(/\\(\\s+/g, '(')\n            .replace(/\\s+\\)/g, ')') || '';\n      } else {\n        this.expression = init.expression;\n      }\n    } else {\n      this.expression = '';\n    }\n  }\n\n  public serialize(): string | undefined {\n    return this.expression || undefined;\n  }\n\n  public name<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    ...path: [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]]\n  ): string {\n    return this.names.add(...path);\n  }\n\n  public value(key: string, value: unknown): string {\n    return this.values.add(key, value);\n  }\n\n  public eq<K1 extends keyof T>(\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public eq<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], value: ValueFn<T, T[K1][K2][K3]>): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], value: ValueFn<T, T[K1][K2][K3][K4]>): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], value: ValueFn<T, T[K1][K2][K3][K4][K5]>): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public eq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    return this.comparator(\n      '=',\n      path as [K1, K2, K3, K4, K5, K6, K7, K8],\n      value as ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    );\n  }\n\n  public nq<K1 extends keyof T>(\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public nq<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], value: ValueFn<T, T[K1][K2][K3]>): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], value: ValueFn<T, T[K1][K2][K3][K4]>): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], value: ValueFn<T, T[K1][K2][K3][K4][K5]>): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public nq<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    return this.comparator(\n      '<>',\n      path as [K1, K2, K3, K4, K5, K6, K7, K8],\n      value as ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    );\n  }\n\n  public lt<K1 extends keyof T>(\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public lt<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], value: ValueFn<T, T[K1][K2][K3]>): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], value: ValueFn<T, T[K1][K2][K3][K4]>): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], value: ValueFn<T, T[K1][K2][K3][K4][K5]>): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public lt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    return this.comparator(\n      '<',\n      path as [K1, K2, K3, K4, K5, K6, K7, K8],\n      value as ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    );\n  }\n\n  public lte<K1 extends keyof T>(\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public lte<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], value: ValueFn<T, T[K1][K2][K3]>): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], value: ValueFn<T, T[K1][K2][K3][K4]>): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], value: ValueFn<T, T[K1][K2][K3][K4][K5]>): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public lte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    return this.comparator(\n      '<=',\n      path as [K1, K2, K3, K4, K5, K6, K7, K8],\n      value as ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    );\n  }\n\n  public gt<K1 extends keyof T>(\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public gt<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], value: ValueFn<T, T[K1][K2][K3]>): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], value: ValueFn<T, T[K1][K2][K3][K4]>): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], value: ValueFn<T, T[K1][K2][K3][K4][K5]>): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public gt<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    return this.comparator(\n      '>',\n      path as [K1, K2, K3, K4, K5, K6, K7, K8],\n      value as ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    );\n  }\n\n  public gte<K1 extends keyof T>(\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public gte<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], value: ValueFn<T, T[K1][K2][K3]>): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], value: ValueFn<T, T[K1][K2][K3][K4]>): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], value: ValueFn<T, T[K1][K2][K3][K4][K5]>): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public gte<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    return this.comparator(\n      '>=',\n      path as [K1, K2, K3, K4, K5, K6, K7, K8],\n      value as ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    );\n  }\n\n  public comparator<K1 extends keyof T>(\n    comparator: Comparator,\n    path: K1 | [K1],\n    value: ValueFn<T, T[K1]>,\n  ): this;\n  public comparator<K1 extends keyof T, K2 extends keyof T[K1]>(\n    comparator: Comparator,\n    path: [K1, K2],\n    value: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(\n    comparator: Comparator,\n    path: [K1, K2, K3],\n    value: ValueFn<T, T[K1][K2][K3]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(\n    comparator: Comparator,\n    path: [K1, K2, K3, K4],\n    value: ValueFn<T, T[K1][K2][K3][K4]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(\n    comparator: Comparator,\n    path: [K1, K2, K3, K4, K5],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    comparator: Comparator,\n    path: [K1, K2, K3, K4, K5, K6],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    comparator: Comparator,\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    comparator: Comparator,\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    value: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public comparator<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    comparator: Comparator,\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    value: ValueFn<T, unknown>,\n  ): this {\n    if (!Array.isArray(path)) {\n      path = [path];\n    }\n\n    const pathName = this.names.add(...path);\n    const valueStr = this.calcValueStr(path, pathName, value);\n\n    return this.append('AND', `${pathName} ${comparator} ${valueStr}`);\n  }\n\n  public between<K1 extends keyof T>(\n    path: K1 | [K1],\n    from: ValueFn<T, T[K1]>,\n    to: ValueFn<T, T[K1]>,\n  ): this;\n  public between<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    from: ValueFn<T, T[K1][K2]>,\n    to: ValueFn<T, T[K1][K2]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(\n    path: [K1, K2, K3],\n    from: ValueFn<T, T[K1][K2][K3]>,\n    to: ValueFn<T, T[K1][K2][K3]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(\n    path: [K1, K2, K3, K4],\n    from: ValueFn<T, T[K1][K2][K3][K4]>,\n    to: ValueFn<T, T[K1][K2][K3][K4]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(\n    path: [K1, K2, K3, K4, K5],\n    from: ValueFn<T, T[K1][K2][K3][K4][K5]>,\n    to: ValueFn<T, T[K1][K2][K3][K4][K5]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(\n    path: [K1, K2, K3, K4, K5, K6],\n    from: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n    to: ValueFn<T, T[K1][K2][K3][K4][K5][K6]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7],\n    from: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n    to: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: [K1, K2, K3, K4, K5, K6, K7, K8],\n    from: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n    to: ValueFn<T, T[K1][K2][K3][K4][K5][K6][K7][K8]>,\n  ): this;\n  public between<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n    from: ValueFn<T, unknown>,\n    to: ValueFn<T, unknown>,\n  ): this {\n    if (!Array.isArray(path)) {\n      path = [path];\n    }\n\n    const pathName = this.names.add(...path);\n    const fromStr = this.calcValueStr(path, pathName, from);\n    const toStr = this.calcValueStr(path, pathName, to);\n\n    return this.append('AND', `${pathName} BETWEEN ${fromStr} AND ${toStr}`);\n  }\n\n  public attributeExists<K1 extends keyof T>(path: K1 | [K1]): this;\n  public attributeExists<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n  ): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3]): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4]): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5]): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(path: [K1, K2, K3, K4, K5, K6]): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(path: [K1, K2, K3, K4, K5, K6, K7]): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(path: [K1, K2, K3, K4, K5, K6, K7, K8]): this;\n  public attributeExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n  ): this {\n    if (!Array.isArray(path)) {\n      path = [path];\n    }\n\n    return this.fn('attribute_exists', this.names.add(...path));\n  }\n\n  public attributeNotExists<K1 extends keyof T>(path: K1 | [K1]): this;\n  public attributeNotExists<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n  ): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3]): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4]): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5]): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(path: [K1, K2, K3, K4, K5, K6]): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(path: [K1, K2, K3, K4, K5, K6, K7]): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(path: [K1, K2, K3, K4, K5, K6, K7, K8]): this;\n  public attributeNotExists<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n    K8 extends keyof T[K1][K2][K3][K4][K5][K6][K7],\n  >(\n    path: K1 | [K1, K2?, K3?, K4?, K5?, K6?, K7?, K8?, ...(string | number)[]],\n  ): this {\n    if (!Array.isArray(path)) {\n      path = [path];\n    }\n\n    return this.fn('attribute_not_exists', this.names.add(...path));\n  }\n\n  public attributeType<K1 extends keyof T>(\n    path: K1 | [K1],\n    type: AttributeType,\n  ): this;\n  public attributeType<K1 extends keyof T, K2 extends keyof T[K1]>(\n    path: [K1, K2],\n    type: AttributeType,\n  ): this;\n  public attributeType<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n  >(path: [K1, K2, K3], type: AttributeType): this;\n  public attributeType<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n  >(path: [K1, K2, K3, K4], type: AttributeType): this;\n  public attributeType<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n  >(path: [K1, K2, K3, K4, K5], type: AttributeType): this;\n  public attributeType<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n  >(path: [K1, K2, K3, K4, K5, K6], type: AttributeType): this;\n  public attributeType<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2],\n    K4 extends keyof T[K1][K2][K3],\n    K5 extends keyof T[K1][K2][K3][K4],\n    K6 extends keyof T[K1][K2][K3][K4][K5],\n    K7 extends keyof T[K1][K2][K3][K4][K5][K6],\n  >(path: [K1, K2, K3, K4, K5, K6, K7], type: AttributeType): this;\n  public attributeType<\n    K1 extends keyof T,\n    K2 extends keyof T[K1],\n    K3 extends keyof T[K1][K2