UNPKG

angular-odata

Version:

Client side OData typescript library for Angular

316 lines 50.8 kB
import { DEFAULT_VERSION, PARAM_SEPARATOR, QUERY_SEPARATOR, VALUE_SEPARATOR, } from '../constants'; import { ODataHelper } from '../helper'; import { ODataStructuredType } from '../schema'; import { PathSegment, } from '../types'; import { Objects, Strings, Types } from '../utils'; import { ODataPathSegments, ODataPathSegmentsHandler } from './path'; import { isQueryCustomType, ODataQueryOptions, ODataQueryOptionsHandler, } from './query'; export class ODataResource { // VARIABLES api; pathSegments; queryOptions; constructor(api, { segments, query, } = {}) { this.api = api; this.pathSegments = segments ?? new ODataPathSegments(); this.queryOptions = query ?? new ODataQueryOptions(); } /** * @returns string The outgoing type of the resource */ outgoingType() { return this.pathSegments.last()?.outgoingType(); } /** * @returns string The incoming type of the resource */ incomingType() { return this.pathSegments.last()?.incomingType(); } /** * @returns string The binding type of the resource */ bindingType() { return this.pathSegments.last()?.bindingType(); } /** * @returns string All covered types of the resource */ types() { return this.pathSegments.types(); } callable() { const type = this.outgoingType() ?? this.incomingType(); return type !== undefined ? this.api.callable(type) : undefined; } enumType() { const type = this.outgoingType() ?? this.incomingType(); return type !== undefined ? this.api.enumType(type) : undefined; } structuredType() { const type = this.outgoingType() ?? this.incomingType(); return type !== undefined ? this.api.structuredType(type) : undefined; } /** * @returns boolean The resource has key ? */ hasKey() { return Boolean(this.pathSegments.last({ key: true })?.hasKey()); } hasEntityKey() { return Boolean(this.pathSegments.get(PathSegment.entitySet)?.hasKey()); } clearKey() { return this.pathSegments.last({ key: true })?.clearKey(); } asModel(entity, { reset, annots, ModelType, } = {}) { reset ??= annots !== undefined; let resource = this; const type = annots?.type ?? this.incomingType(); if (type === undefined) throw Error(`No type for model`); if (ModelType === undefined) ModelType = this.api.modelForType(type); let entitySet = annots?.entitySet; if (entitySet !== undefined) { resource = this.api.entitySet(entitySet).entity(entity); resource.query((q) => q.restore(this.queryOptions.toQueryArguments())); } return new ModelType(entity, { resource, annots, reset }); } asCollection(entities, { reset, annots, CollectionType, } = {}) { reset ??= annots !== undefined; let resource = this; const type = annots?.type ?? this.incomingType(); if (type === undefined) throw Error(`No type for collection`); if (CollectionType === undefined) CollectionType = this.api.collectionForType(type); let entitySet = annots?.entitySet; if (entitySet !== undefined) { resource = this.api.entitySet(entitySet); resource.query((q) => q.restore(this.queryOptions.toQueryArguments())); } return new CollectionType(entities, { resource, annots, reset }); } //#endregion isTypeOf(other) { const thisStructured = this.structuredType(); const otherStructured = other.structuredType(); return (thisStructured !== undefined && otherStructured !== undefined && thisStructured.isTypeOf(otherStructured)); } isSubtypeOf(other) { const thisStructured = this.structuredType(); const otherStructured = other.structuredType(); return (thisStructured !== undefined && otherStructured !== undefined && thisStructured.isSubtypeOf(otherStructured)); } isSupertypeOf(other) { const thisStructured = this.structuredType(); const otherStructured = other.structuredType(); return (thisStructured !== undefined && otherStructured !== undefined && thisStructured.isSupertypeOf(otherStructured)); } isEqualTo(other, test) { const [selfPath, selfParams] = this.pathAndParams(); const [otherPath, otherParams] = other.pathAndParams(); return test === 'path' ? otherPath === selfPath : test === 'params' ? Types.isEqual(selfParams, otherParams) : otherPath === selfPath && Types.isEqual(selfParams, otherParams); } pathAndParams({ escape, ...options } = { escape: false, }) { const type = this.outgoingType(); const parser = type !== undefined ? this.api.parserForType(type) : undefined; const [spath, sparams] = this.pathSegments.pathAndParams({ escape, parser, options, }); const [, qparams] = this.queryOptions.pathAndParams({ escape, parser, options, }); return [spath, { ...sparams, ...qparams }]; } endpointUrl({ escape = false, params = true, ...options } = {}) { let [path, qparams] = this.pathAndParams({ escape, ...options }); if (params && !Types.isEmpty(qparams)) { path = `${path}${QUERY_SEPARATOR}${Object.entries(qparams) .map((e) => `${e[0]}${VALUE_SEPARATOR}${e[1]}`) .join(PARAM_SEPARATOR)}`; } return `${this.api.serviceRootUrl}${path}`; } toString({ escape, ...options } = { escape: false, }) { let [path, params] = this.pathAndParams({ escape, ...options }); let queryString = Object.entries(params) .map((e) => `${e[0]}${VALUE_SEPARATOR}${e[1]}`) .join(PARAM_SEPARATOR); return queryString ? `${path}${QUERY_SEPARATOR}${queryString}` : path; } clone() { const Ctor = this.constructor; return new Ctor(this.api, { segments: this.cloneSegments(), query: this.cloneQuery(), }); } __parser(value, options, resourceType, bindingType) { const dataType = options !== undefined && Types.isPlainObject(value) ? ODataHelper[options.version ?? DEFAULT_VERSION].type(value) : undefined; if (dataType !== undefined) { // Parser from data type return this.api.parserForType(dataType); } else if (resourceType !== undefined) { // Parser from resource type return this.api.parserForType(resourceType, bindingType); } return undefined; } deserialize(value, options) { const resourceType = this.incomingType(); const bindingType = this.bindingType(); const _d = (value, options) => { const parser = this.__parser(value, options, resourceType, bindingType); return parser !== undefined ? parser.deserialize(value, options) : value; }; return Array.isArray(value) ? value.map((v) => _d(v, options)) : _d(value, options); } serialize(value, options) { const resourceType = this.outgoingType(); const bindingType = this.bindingType(); const _s = (value, options) => { const parser = this.__parser(value, options, resourceType, bindingType); return parser !== undefined ? parser.serialize(value, options) : value; }; return Array.isArray(value) ? value.map((v) => _s(v, options)) : _s(value, options); } encode(value, options) { const resourceType = this.outgoingType(); const bindingType = this.bindingType(); const _e = (value, options) => { const parser = this.__parser(value, options, resourceType, bindingType); return parser !== undefined ? parser.encode(value, options) : value; }; return Array.isArray(value) ? value.map((v) => _e(v, options)) : _e(value, options); } toJson() { return { segments: this.pathSegments.toJson(), options: this.queryOptions.toJson(), }; } cloneSegments() { return this.pathSegments.clone(); } //#region Query Options clearQuery() { this.queryOptions.clear(); return this; } cloneQuery() { return this.queryOptions.clone(); } /** * Handle the path segments of the resource * Create an object handler for mutate the path segments of the resource * @param f Function context for handle the segments * @returns ODataActionResource */ segment(f) { const type = this.outgoingType(); f(new ODataPathSegmentsHandler(this.pathSegments), type !== undefined ? this.api.structuredType(type) : undefined); return this; } /** * Handle the query options of the resource * Create an object handler for mutate the query options of the resource * @param f Function context for handle the query options */ query(f) { const type = this.outgoingType(); f(new ODataQueryOptionsHandler(this.queryOptions), type !== undefined ? this.api.structuredType(type) : undefined); return this; } transform(opts, { type, fields, } = {}) { if (type === undefined) { type = Strings.uniqueId({ prefix: 'Transformation', suffix: 'Type' }); } // Resolve Structured Type let structuredType = this.api.findStructuredType(type); if (structuredType === undefined) { // Resolve Schema let schema = this.api.findSchema(type); if (schema === undefined) { const namespace = type.substring(0, type.lastIndexOf('.')) ?? this.api.name; schema = this.api.createSchema({ namespace }); } const name = type.substring(type.lastIndexOf('.')); structuredType = schema.createStructuredType({ name, fields }); } // Segments const segments = this.cloneSegments(); segments.last()?.incomingType(structuredType.type()); // Query const query = this.cloneQuery(); const handler = new ODataQueryOptionsHandler(query); handler.apply(opts); const Ctor = this.constructor; return new Ctor(this.api, { segments, query, }); } static resolveKey(value, schema) { if (isQueryCustomType(value)) { return value; } else if (Types.isPlainObject(value)) { return schema instanceof ODataStructuredType ? schema.resolveKey(value) : Objects.resolveKey(value); } return value; } resolveKey(value) { const type = this.outgoingType(); const structured = type !== undefined ? this.api.structuredType(type) : undefined; return ODataResource.resolveKey(value, structured); } //#endregion get(options = {}) { return this.api.request('GET', this, options); } post(body, options = {}) { return this.api.request('POST', this, { body, ...options }); } put(body, options = {}) { return this.api.request('PUT', this, { body, ...options }); } patch(body, options = {}) { return this.api.request('PATCH', this, { body, ...options }); } delete(options = {}) { return this.api.request('DELETE', this, options); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource.js","sourceRoot":"","sources":["../../../../../projects/angular-odata/src/lib/resources/resource.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable } from 'rxjs';\nimport { ODataApi } from '../api';\nimport {\n  DEFAULT_VERSION,\n  PARAM_SEPARATOR,\n  QUERY_SEPARATOR,\n  VALUE_SEPARATOR,\n} from '../constants';\nimport { ODataHelper } from '../helper';\nimport { ModelInterface, ODataCollection, ODataModel } from '../models';\nimport { ODataStructuredType } from '../schema';\nimport {\n  ParserOptions,\n  Parser,\n  QueryOption,\n  PathSegment,\n  StructuredTypeFieldConfig,\n} from '../types';\nimport { Objects, Strings, Types } from '../utils';\nimport { ODataPathSegments, ODataPathSegmentsHandler } from './path';\nimport {\n  isQueryCustomType,\n  ODataQueryOptions,\n  ODataQueryOptionsHandler,\n} from './query';\nimport {\n  ApplyExpression,\n  ApplyExpressionBuilder,\n  QueryCustomType,\n} from './query';\nimport { ODataOptions } from './types';\nimport {\n  ODataEntitiesAnnotations,\n  ODataEntityAnnotations,\n} from '../annotations';\n\nexport type EntityKey<T> =\n  | {\n      readonly [P in keyof T]?: T[P];\n    }\n  | QueryCustomType\n  | string\n  | number;\n\nexport class ODataResource<T> {\n  // VARIABLES\n  public api: ODataApi;\n  protected pathSegments: ODataPathSegments;\n  protected queryOptions: ODataQueryOptions<T>;\n  constructor(\n    api: ODataApi,\n    {\n      segments,\n      query,\n    }: {\n      segments?: ODataPathSegments;\n      query?: ODataQueryOptions<T>;\n    } = {},\n  ) {\n    this.api = api;\n    this.pathSegments = segments ?? new ODataPathSegments();\n    this.queryOptions = query ?? new ODataQueryOptions();\n  }\n\n  /**\n   * @returns string The outgoing type of the resource\n   */\n  outgoingType() {\n    return this.pathSegments.last()?.outgoingType();\n  }\n\n  /**\n   * @returns string The incoming type of the resource\n   */\n  incomingType() {\n    return this.pathSegments.last()?.incomingType();\n  }\n\n  /**\n   * @returns string The binding type of the resource\n   */\n  bindingType() {\n    return this.pathSegments.last()?.bindingType();\n  }\n\n  /**\n   * @returns string All covered types of the resource\n   */\n  types(): string[] {\n    return this.pathSegments.types();\n  }\n\n  callable() {\n    const type = this.outgoingType() ?? this.incomingType();\n    return type !== undefined ? this.api.callable<T>(type) : undefined;\n  }\n\n  enumType() {\n    const type = this.outgoingType() ?? this.incomingType();\n    return type !== undefined ? this.api.enumType<T>(type) : undefined;\n  }\n\n  structuredType() {\n    const type = this.outgoingType() ?? this.incomingType();\n    return type !== undefined ? this.api.structuredType<T>(type) : undefined;\n  }\n\n  /**\n   * @returns boolean The resource has key ?\n   */\n  hasKey() {\n    return Boolean(this.pathSegments.last({ key: true })?.hasKey());\n  }\n\n  hasEntityKey() {\n    return Boolean(this.pathSegments.get(PathSegment.entitySet)?.hasKey());\n  }\n\n  clearKey() {\n    return this.pathSegments.last({ key: true })?.clearKey();\n  }\n\n  //#region Models\n  asModel(\n    entity?: Partial<T> | { [name: string]: any },\n  ): ODataModel<T> & ModelInterface<T>;\n  asModel(\n    entity: Partial<T> | { [name: string]: any },\n    {\n      reset,\n      annots,\n      ModelType,\n    }: {\n      reset?: boolean;\n      annots?: ODataEntityAnnotations<T>;\n      ModelType?: typeof ODataModel;\n    },\n  ): ODataModel<T> & ModelInterface<T>;\n  asModel<M extends ODataModel<T>>(\n    entity?: Partial<T> | { [name: string]: any },\n  ): M;\n  asModel<M extends ODataModel<T>>(\n    entity: Partial<T> | { [name: string]: any },\n    {\n      reset,\n      annots,\n      ModelType,\n    }: {\n      reset?: boolean;\n      annots?: ODataEntityAnnotations<T>;\n      ModelType?: typeof ODataModel;\n    },\n  ): M;\n  asModel(\n    entity?: Partial<T> | { [name: string]: any },\n    {\n      reset,\n      annots,\n      ModelType,\n    }: {\n      reset?: boolean;\n      annots?: ODataEntityAnnotations<T>;\n      ModelType?: typeof ODataModel;\n    } = {},\n  ) {\n    reset ??= annots !== undefined;\n    let resource: ODataResource<T> = this as ODataResource<T>;\n    const type = annots?.type ?? this.incomingType();\n    if (type === undefined) throw Error(`No type for model`);\n    if (ModelType === undefined) ModelType = this.api.modelForType(type);\n    let entitySet = annots?.entitySet;\n    if (entitySet !== undefined) {\n      resource = this.api.entitySet<T>(entitySet).entity(entity as Partial<T>);\n      resource.query((q) => q.restore(this.queryOptions.toQueryArguments()));\n    }\n    return new ModelType(entity, { resource, annots, reset });\n  }\n\n  asCollection(\n    entities?: Partial<T>[] | { [name: string]: any }[],\n  ): ODataCollection<T, ODataModel<T> & ModelInterface<T>>;\n  asCollection(\n    entities: Partial<T>[] | { [name: string]: any }[],\n    {\n      reset,\n      annots,\n      CollectionType,\n    }: {\n      reset?: boolean;\n      annots?: ODataEntitiesAnnotations<T>;\n      CollectionType?: typeof ODataCollection;\n    },\n  ): ODataCollection<T, ODataModel<T> & ModelInterface<T>>;\n  asCollection<M extends ODataModel<T>, C extends ODataCollection<T, M>>(\n    entities?: Partial<T>[] | { [name: string]: any }[],\n  ): C;\n  asCollection<M extends ODataModel<T>, C extends ODataCollection<T, M>>(\n    entities: Partial<T>[] | { [name: string]: any }[],\n    {\n      reset,\n      annots,\n      CollectionType,\n    }: {\n      reset?: boolean;\n      annots?: ODataEntitiesAnnotations<T>;\n      CollectionType?: typeof ODataCollection;\n    },\n  ): C;\n  asCollection(\n    entities?: Partial<T>[] | { [name: string]: any }[],\n    {\n      reset,\n      annots,\n      CollectionType,\n    }: {\n      reset?: boolean;\n      annots?: ODataEntitiesAnnotations<T>;\n      CollectionType?: typeof ODataCollection;\n    } = {},\n  ) {\n    reset ??= annots !== undefined;\n    let resource: ODataResource<T> = this as ODataResource<T>;\n    const type = annots?.type ?? this.incomingType();\n    if (type === undefined) throw Error(`No type for collection`);\n    if (CollectionType === undefined)\n      CollectionType = this.api.collectionForType(type);\n    let entitySet = annots?.entitySet;\n    if (entitySet !== undefined) {\n      resource = this.api.entitySet<T>(entitySet);\n      resource.query((q) => q.restore(this.queryOptions.toQueryArguments()));\n    }\n    return new CollectionType(entities, { resource, annots, reset });\n  }\n  //#endregion\n\n  isTypeOf(other: ODataResource<any>) {\n    const thisStructured = this.structuredType();\n    const otherStructured = other.structuredType();\n    return (\n      thisStructured !== undefined &&\n      otherStructured !== undefined &&\n      thisStructured.isTypeOf(otherStructured)\n    );\n  }\n\n  isSubtypeOf(other: ODataResource<any>) {\n    const thisStructured = this.structuredType();\n    const otherStructured = other.structuredType();\n    return (\n      thisStructured !== undefined &&\n      otherStructured !== undefined &&\n      thisStructured.isSubtypeOf(otherStructured)\n    );\n  }\n\n  isSupertypeOf(other: ODataResource<any>) {\n    const thisStructured = this.structuredType();\n    const otherStructured = other.structuredType();\n    return (\n      thisStructured !== undefined &&\n      otherStructured !== undefined &&\n      thisStructured.isSupertypeOf(otherStructured)\n    );\n  }\n\n  isEqualTo(other: ODataResource<any>, test?: 'path' | 'params') {\n    const [selfPath, selfParams] = this.pathAndParams();\n    const [otherPath, otherParams] = other.pathAndParams();\n    return test === 'path'\n      ? otherPath === selfPath\n      : test === 'params'\n        ? Types.isEqual(selfParams, otherParams)\n        : otherPath === selfPath && Types.isEqual(selfParams, otherParams);\n  }\n\n  pathAndParams(\n    { escape, ...options }: ParserOptions & { escape?: boolean } = {\n      escape: false,\n    },\n  ): [string, { [name: string]: any }] {\n    const type = this.outgoingType();\n    const parser =\n      type !== undefined ? this.api.parserForType<T>(type) : undefined;\n    const [spath, sparams] = this.pathSegments.pathAndParams({\n      escape,\n      parser,\n      options,\n    });\n    const [, qparams] = this.queryOptions.pathAndParams({\n      escape,\n      parser,\n      options,\n    });\n\n    return [spath, { ...sparams, ...qparams }];\n  }\n\n  endpointUrl({\n    escape = false,\n    params = true,\n    ...options\n  }: ParserOptions & { escape?: boolean; params?: boolean } = {}): string {\n    let [path, qparams] = this.pathAndParams({ escape, ...options });\n    if (params && !Types.isEmpty(qparams)) {\n      path = `${path}${QUERY_SEPARATOR}${Object.entries(qparams)\n        .map((e) => `${e[0]}${VALUE_SEPARATOR}${e[1]}`)\n        .join(PARAM_SEPARATOR)}`;\n    }\n    return `${this.api.serviceRootUrl}${path}`;\n  }\n\n  toString(\n    { escape, ...options }: ParserOptions & { escape?: boolean } = {\n      escape: false,\n    },\n  ): string {\n    let [path, params] = this.pathAndParams({ escape, ...options });\n    let queryString = Object.entries(params)\n      .map((e) => `${e[0]}${VALUE_SEPARATOR}${e[1]}`)\n      .join(PARAM_SEPARATOR);\n    return queryString ? `${path}${QUERY_SEPARATOR}${queryString}` : path;\n  }\n\n  clone(): ODataResource<T> {\n    const Ctor = this.constructor as typeof ODataResource;\n    return new Ctor(this.api, {\n      segments: this.cloneSegments(),\n      query: this.cloneQuery<T>(),\n    });\n  }\n\n  private __parser(\n    value: any,\n    options?: ParserOptions,\n    resourceType?: string,\n    bindingType?: string,\n  ): Parser<T> | undefined {\n    const dataType =\n      options !== undefined && Types.isPlainObject(value)\n        ? ODataHelper[options.version ?? DEFAULT_VERSION].type(value)\n        : undefined;\n    if (dataType !== undefined) {\n      // Parser from data type\n      return this.api.parserForType<T>(dataType);\n    } else if (resourceType !== undefined) {\n      // Parser from resource type\n      return this.api.parserForType<T>(resourceType, bindingType);\n    }\n    return undefined;\n  }\n\n  deserialize(value: any, options?: ParserOptions): any {\n    const resourceType = this.incomingType();\n    const bindingType = this.bindingType();\n    const _d = (value: any, options?: ParserOptions) => {\n      const parser = this.__parser(value, options, resourceType, bindingType);\n      return parser !== undefined ? parser.deserialize(value, options) : value;\n    };\n    return Array.isArray(value)\n      ? value.map((v) => _d(v, options))\n      : _d(value, options);\n  }\n\n  serialize(value: any, options?: ParserOptions): any {\n    const resourceType = this.outgoingType();\n    const bindingType = this.bindingType();\n    const _s = (value: any, options?: ParserOptions) => {\n      const parser = this.__parser(value, options, resourceType, bindingType);\n      return parser !== undefined ? parser.serialize(value, options) : value;\n    };\n    return Array.isArray(value)\n      ? value.map((v) => _s(v, options))\n      : _s(value, options);\n  }\n\n  encode(value: any, options?: ParserOptions): any {\n    const resourceType = this.outgoingType();\n    const bindingType = this.bindingType();\n    const _e = (value: any, options?: ParserOptions) => {\n      const parser = this.__parser(value, options, resourceType, bindingType);\n      return parser !== undefined ? parser.encode(value, options) : value;\n    };\n    return Array.isArray(value)\n      ? value.map((v) => _e(v, options))\n      : _e(value, options);\n  }\n\n  toJson() {\n    return {\n      segments: this.pathSegments.toJson(),\n      options: this.queryOptions.toJson(),\n    };\n  }\n\n  cloneSegments() {\n    return this.pathSegments.clone();\n  }\n\n  //#region Query Options\n  clearQuery() {\n    this.queryOptions.clear();\n    return this;\n  }\n\n  cloneQuery<P>() {\n    return this.queryOptions.clone<P>();\n  }\n\n  /**\n   * Handle the path segments of the resource\n   * Create an object handler for mutate the path segments of the resource\n   * @param f Function context for handle the segments\n   * @returns ODataActionResource\n   */\n  segment(\n    f: (q: ODataPathSegmentsHandler<T>, s?: ODataStructuredType<T>) => void,\n  ) {\n    const type = this.outgoingType();\n    f(\n      new ODataPathSegmentsHandler<T>(this.pathSegments),\n      type !== undefined ? this.api.structuredType<T>(type) : undefined,\n    );\n    return this;\n  }\n\n  /**\n   * Handle the query options of the resource\n   * Create an object handler for mutate the query options of the resource\n   * @param f Function context for handle the query options\n   */\n  query(\n    f: (q: ODataQueryOptionsHandler<T>, s?: ODataStructuredType<T>) => void,\n  ) {\n    const type = this.outgoingType();\n    f(\n      new ODataQueryOptionsHandler<T>(this.queryOptions),\n      type !== undefined ? this.api.structuredType<T>(type) : undefined,\n    );\n    return this;\n  }\n\n  transform<R>(\n    opts: (\n      builder: ApplyExpressionBuilder<T>,\n      current?: ApplyExpression<T>,\n    ) => ApplyExpression<T>,\n    {\n      type,\n      fields,\n    }: {\n      type?: string;\n      fields?: { [name: string]: StructuredTypeFieldConfig };\n    } = {},\n  ): ODataResource<R> {\n    if (type === undefined) {\n      type = Strings.uniqueId({ prefix: 'Transformation', suffix: 'Type' });\n    }\n\n    // Resolve Structured Type\n    let structuredType = this.api.findStructuredType<R>(type);\n    if (structuredType === undefined) {\n      // Resolve Schema\n      let schema = this.api.findSchema(type);\n      if (schema === undefined) {\n        const namespace =\n          type.substring(0, type.lastIndexOf('.')) ?? this.api.name!;\n        schema = this.api.createSchema({ namespace });\n      }\n      const name = type.substring(type.lastIndexOf('.'));\n      structuredType = schema.createStructuredType({ name, fields });\n    }\n\n    // Segments\n    const segments = this.cloneSegments();\n    segments.last()?.incomingType(structuredType.type());\n\n    // Query\n    const query = this.cloneQuery<any>();\n    const handler = new ODataQueryOptionsHandler<T>(query);\n    handler.apply(opts);\n\n    const Ctor = this.constructor as typeof ODataResource;\n    return new Ctor(this.api, {\n      segments,\n      query,\n    });\n  }\n\n  static resolveKey<T>(\n    value: any,\n    schema?: ODataStructuredType<T>,\n  ): EntityKey<T> | undefined {\n    if (isQueryCustomType(value)) {\n      return value;\n    } else if (Types.isPlainObject(value)) {\n      return schema instanceof ODataStructuredType\n        ? schema.resolveKey(value)\n        : Objects.resolveKey(value);\n    }\n    return value as EntityKey<T> | undefined;\n  }\n\n  protected resolveKey(value: any): EntityKey<T> | undefined {\n    const type = this.outgoingType();\n    const structured =\n      type !== undefined ? this.api.structuredType<T>(type) : undefined;\n    return ODataResource.resolveKey<T>(value, structured);\n  }\n  //#endregion\n\n  protected get(\n    options: ODataOptions & {\n      etag?: string;\n      responseType?:\n        | 'arraybuffer'\n        | 'blob'\n        | 'json'\n        | 'text'\n        | 'value'\n        | 'property'\n        | 'entity'\n        | 'entities';\n      withCount?: boolean;\n      bodyQueryOptions?: QueryOption[];\n    } = {},\n  ): Observable<any> {\n    return this.api.request<T>('GET', this, options);\n  }\n\n  protected post(\n    body: any,\n    options: ODataOptions & {\n      responseType?:\n        | 'arraybuffer'\n        | 'blob'\n        | 'json'\n        | 'text'\n        | 'value'\n        | 'property'\n        | 'entity'\n        | 'entities';\n      withCount?: boolean;\n    } = {},\n  ): Observable<any> {\n    return this.api.request<T>('POST', this, { body, ...options });\n  }\n\n  protected put(\n    body: any,\n    options: ODataOptions & {\n      etag?: string;\n      responseType?:\n        | 'arraybuffer'\n        | 'blob'\n        | 'json'\n        | 'text'\n        | 'value'\n        | 'property'\n        | 'entity'\n        | 'entities';\n      withCount?: boolean;\n    } = {},\n  ): Observable<any> {\n    return this.api.request<T>('PUT', this, { body, ...options });\n  }\n\n  protected patch(\n    body: any,\n    options: ODataOptions & {\n      etag?: string;\n      responseType?:\n        | 'arraybuffer'\n        | 'blob'\n        | 'json'\n        | 'text'\n        | 'value'\n     