UNPKG

angular-odata

Version:

Client side OData typescript library for Angular

187 lines 30.2 kB
import { EMPTY } from 'rxjs'; import { expand, map, reduce } from 'rxjs/operators'; import { PathSegment, } from '../../types'; import { ODataResource } from '../resource'; import { ODataValueResource } from './value'; import { ODataCountResource } from './count'; export class ODataPropertyResource extends ODataResource { //#region Factory static factory(api, { path, type, segments, }) { const segment = segments.add(PathSegment.property, path); if (type !== undefined) { segment.outgoingType(type); segment.incomingType(type); } return new ODataPropertyResource(api, { segments, }); } static fromResource(resource, path) { const baseType = resource.outgoingType(); let baseSchema = baseType !== undefined ? resource.api.structuredType(baseType) : undefined; let fieldType; if (baseSchema !== undefined) { const field = baseSchema.field(path); fieldType = field?.type; baseSchema = field !== undefined ? baseSchema.findParentSchemaForField(field) : undefined; } const property = ODataPropertyResource.factory(resource.api, { path, type: fieldType, segments: resource.cloneSegments(), }); // Switch entitySet to binding type if available if (baseSchema !== undefined && baseSchema.type() !== baseType) { let entitySet = resource.api.findEntitySet(baseSchema.type()); if (entitySet !== undefined) { property.segment((s) => s.entitySet().path(entitySet.name)); } } return property; } clone() { return super.clone(); } transform(opts, { type, fields, } = {}) { return super.transform(opts, { type, fields, }); } //#endregion key(value) { const property = this.clone(); var key = this.resolveKey(value); if (key !== undefined) property.segment((s) => s.property().key(key)); return property; } keys(values) { const property = this.clone(); const types = this.pathSegments.types({ key: true }); const keys = values.map((value, index) => ODataResource.resolveKey(value, this.api.findStructuredType(types[index]))); property.segment((s) => s.keys(keys)); return property; } value() { return ODataValueResource.fromResource(this); } count() { return ODataCountResource.factory(this.api, { segments: this.cloneSegments(), query: this.cloneQuery(), }); } /* navigationProperty<N>(path: string) { let schema: ODataStructuredType<N> | undefined; if (this.schema instanceof ODataStructuredType) { const field = this.schema.field<any>(path as keyof T); schema = field !== undefined ? this.schema.findSchemaForField<N>(field) : undefined; } return ODataNavigationPropertyResource.factory<N>(this.api, { path, schema, segments: this.cloneSegments(), query: this.cloneQuery<N>(), }); } */ property(path) { return ODataPropertyResource.fromResource(this, path); } //#region Requests get(options) { return super.get(options); } fetch(options = {}) { return this.get(options); } /** * Fetch the property value * @param options Options for the request * @returns The property value */ fetchProperty(options = {}) { return this.fetch({ responseType: 'property', ...options }).pipe(map(({ property }) => property)); } /** * Fetch the entity * @param options Options for the request * @returns The entity */ fetchEntity(options = {}) { return this.fetch({ responseType: 'entity', ...options }).pipe(map(({ entity }) => entity)); } fetchModel(options = {}) { return this.fetch({ responseType: 'entity', ...options }).pipe(map(({ entity, annots }) => entity ? this.asModel(entity, { annots, ModelType: options?.ModelType }) : null)); } /** * Fetch the entities * @param options Options for the request * @returns The entities */ fetchEntities(options = {}) { return this.fetch({ responseType: 'entities', ...options }).pipe(map(({ entities }) => entities)); } fetchCollection(options = {}) { return this.fetch({ responseType: 'entities', ...options }).pipe(map(({ entities, annots }) => entities ? this.asCollection(entities, { annots, CollectionType: options?.CollectionType, }) : null)); } fetchOne(options) { let res = this.clone(); res.query((q) => q.top(1)); return res.fetch({ responseType: 'entities', ...options }).pipe(map(({ entities, annots }) => ({ entity: entities !== null && entities.length === 1 ? entities[0] : null, annots, }))); } fetchMany(top, options) { let res = this.clone(); let fetch = (opts) => { if (opts) { res.query((q) => q.paging(opts)); } return res.fetch({ responseType: 'entities', ...options }); }; return fetch({ top }).pipe(expand(({ annots }) => annots.skip || annots.skiptoken ? fetch(annots) : EMPTY), map(({ entities, annots }) => ({ entities: entities || [], annots })), reduce((acc, { entities, annots }) => ({ entities: [...(acc.entities || []), ...(entities || [])], annots: acc.annots.union(annots), }))); } /** * Fetch all entities * @param options Options for the request * @returns All entities */ fetchAll(options = {}) { let res = this.clone(); // Clean Paging res.query((q) => q.removePaging()); let fetch = (opts) => { if (opts) { res.query((q) => q.paging(opts)); } return res.fetch({ responseType: 'entities', ...options }); }; return fetch().pipe(expand(({ annots }) => annots.skip || annots.skiptoken ? fetch(annots) : EMPTY), map(({ entities, annots }) => ({ entities: entities || [], annots })), reduce((acc, { entities, annots }) => ({ entities: [...(acc.entities || []), ...(entities || [])], annots: acc.annots.union(annots), }))); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property.js","sourceRoot":"","sources":["../../../../../../projects/angular-odata/src/lib/resources/types/property.ts"],"names":[],"mappings":"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","sourcesContent":["import { EMPTY, Observable } from 'rxjs';\nimport { expand, map, reduce } from 'rxjs/operators';\nimport { ODataApi } from '../../api';\nimport type { ModelInterface, ODataCollection, ODataModel } from '../../models';\nimport {\n  PathSegment,\n  QueryOption,\n  StructuredTypeFieldConfig,\n} from '../../types';\nimport { ODataPathSegments } from '../path';\nimport { ApplyExpression, ApplyExpressionBuilder } from '../query';\nimport { ODataResource } from '../resource';\nimport { ODataEntities, ODataEntity, ODataProperty } from '../response';\nimport {\n  ODataEntitiesOptions,\n  ODataEntityOptions,\n  ODataOptions,\n  ODataPropertyOptions,\n} from './options';\nimport { ODataValueResource } from './value';\nimport { ODataCountResource } from './count';\nimport { ODataEntitiesAnnotations } from '../../annotations';\n\nexport class ODataPropertyResource<T> extends ODataResource<T> {\n  //#region Factory\n  static factory<P>(\n    api: ODataApi,\n    {\n      path,\n      type,\n      segments,\n    }: {\n      path: string;\n      type?: string;\n      segments: ODataPathSegments;\n    },\n  ) {\n    const segment = segments.add(PathSegment.property, path);\n    if (type !== undefined) {\n      segment.outgoingType(type);\n      segment.incomingType(type);\n    }\n    return new ODataPropertyResource<P>(api, {\n      segments,\n    });\n  }\n\n  static fromResource<N>(resource: ODataResource<any>, path: string) {\n    const baseType = resource.outgoingType();\n    let baseSchema =\n      baseType !== undefined\n        ? resource.api.structuredType<any>(baseType)\n        : undefined;\n    let fieldType: string | undefined;\n    if (baseSchema !== undefined) {\n      const field = baseSchema.field<N>(path);\n      fieldType = field?.type;\n      baseSchema =\n        field !== undefined\n          ? baseSchema.findParentSchemaForField<N>(field)\n          : undefined;\n    }\n\n    const property = ODataPropertyResource.factory<N>(resource.api, {\n      path,\n      type: fieldType,\n      segments: resource.cloneSegments(),\n    });\n\n    // Switch entitySet to binding type if available\n    if (baseSchema !== undefined && baseSchema.type() !== baseType) {\n      let entitySet = resource.api.findEntitySet(baseSchema.type());\n      if (entitySet !== undefined) {\n        property.segment((s) => s.entitySet().path(entitySet!.name));\n      }\n    }\n\n    return property;\n  }\n  override clone(): ODataPropertyResource<T> {\n    return super.clone() as ODataPropertyResource<T>;\n  }\n\n  override 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  ): ODataPropertyResource<R> {\n    return super.transform<R>(opts, {\n      type,\n      fields,\n    }) as ODataPropertyResource<R>;\n  }\n  //#endregion\n\n  key(value: any) {\n    const property = this.clone();\n    var key = this.resolveKey(value);\n    if (key !== undefined) property.segment((s) => s.property().key(key));\n    return property;\n  }\n\n  keys(values: any[]) {\n    const property = this.clone();\n    const types = this.pathSegments.types({ key: true });\n    const keys = values.map((value, index) =>\n      ODataResource.resolveKey(\n        value,\n        this.api.findStructuredType<T>(types[index]),\n      ),\n    );\n    property.segment((s) => s.keys(keys));\n    return property;\n  }\n\n  value() {\n    return ODataValueResource.fromResource<T>(this);\n  }\n\n  count() {\n    return ODataCountResource.factory<T>(this.api, {\n      segments: this.cloneSegments(),\n      query: this.cloneQuery<T>(),\n    });\n  }\n\n  /*\n  navigationProperty<N>(path: string) {\n    let schema: ODataStructuredType<N> | undefined;\n    if (this.schema instanceof ODataStructuredType) {\n      const field = this.schema.field<any>(path as keyof T);\n      schema =\n        field !== undefined\n          ? this.schema.findSchemaForField<N>(field)\n          : undefined;\n    }\n    return ODataNavigationPropertyResource.factory<N>(this.api, {\n      path,\n      schema,\n      segments: this.cloneSegments(),\n      query: this.cloneQuery<N>(),\n    });\n  }\n  */\n\n  property<P>(path: string) {\n    return ODataPropertyResource.fromResource<P>(this, path);\n  }\n\n  //#region Requests\n  protected override get(\n    options?: ODataEntityOptions & ODataEntitiesOptions & ODataPropertyOptions,\n  ): Observable<any> {\n    return super.get(options);\n  }\n  //#endregion\n\n  //#region Shortcuts\n  /**\n   * Fetch the property\n   * @param options Options for the request\n   * @return The entity / entities / property value\n   */\n  fetch(options?: ODataEntityOptions): Observable<ODataEntity<T>>;\n  fetch(options?: ODataEntitiesOptions): Observable<ODataEntities<T>>;\n  fetch(options?: ODataPropertyOptions): Observable<ODataProperty<T>>;\n  fetch(\n    options: ODataEntityOptions &\n      ODataEntitiesOptions &\n      ODataPropertyOptions = {},\n  ): Observable<any> {\n    return this.get(options);\n  }\n\n  /**\n   * Fetch the property value\n   * @param options Options for the request\n   * @returns The property value\n   */\n  fetchProperty(options: ODataOptions = {}): Observable<T | null> {\n    return this.fetch({ responseType: 'property', ...options }).pipe(\n      map(({ property }) => property),\n    );\n  }\n\n  /**\n   * Fetch the entity\n   * @param options Options for the request\n   * @returns The entity\n   */\n  fetchEntity(options: ODataOptions = {}): Observable<T | null> {\n    return this.fetch({ responseType: 'entity', ...options }).pipe(\n      map(({ entity }) => entity),\n    );\n  }\n\n  /**\n   * Fetch the entity and return as model\n   * @param options Options for the request\n   * @returns The model\n   */\n  fetchModel(\n    options?: ODataOptions & {\n      bodyQueryOptions?: QueryOption[];\n      ModelType?: typeof ODataModel;\n    },\n  ): Observable<(ODataModel<T> & ModelInterface<T>) | null>;\n  fetchModel<M extends ODataModel<T>>(\n    options?: ODataOptions & {\n      bodyQueryOptions?: QueryOption[];\n      ModelType?: typeof ODataModel;\n    },\n  ): Observable<M | null>;\n  fetchModel(\n    options: ODataOptions & {\n      ModelType?: typeof ODataModel;\n    } = {},\n  ) {\n    return this.fetch({ responseType: 'entity', ...options }).pipe(\n      map(({ entity, annots }) =>\n        entity\n          ? this.asModel(entity, { annots, ModelType: options?.ModelType })\n          : null,\n      ),\n    );\n  }\n\n  /**\n   * Fetch the entities\n   * @param options Options for the request\n   * @returns The entities\n   */\n  fetchEntities(\n    options: ODataOptions & { withCount?: boolean } = {},\n  ): Observable<T[] | null> {\n    return this.fetch({ responseType: 'entities', ...options }).pipe(\n      map(({ entities }) => entities),\n    );\n  }\n\n  /**\n   * Fetch the entities and return as collection\n   * @param options Options for the request\n   * @returns The collection\n   */\n  fetchCollection(\n    options?: ODataOptions & {\n      withCount?: boolean;\n      bodyQueryOptions?: QueryOption[];\n      CollectionType?: typeof ODataCollection;\n    },\n  ): Observable<ODataCollection<T, ODataModel<T> & ModelInterface<T>> | null>;\n  fetchCollection<M extends ODataModel<T>, C extends ODataCollection<T, M>>(\n    options?: ODataOptions & {\n      withCount?: boolean;\n      bodyQueryOptions?: QueryOption[];\n      CollectionType?: typeof ODataCollection;\n    },\n  ): Observable<C | null>;\n  fetchCollection(\n    options: ODataOptions & {\n      withCount?: boolean;\n      CollectionType?: typeof ODataCollection;\n    } = {},\n  ) {\n    return this.fetch({ responseType: 'entities', ...options }).pipe(\n      map(({ entities, annots }) =>\n        entities\n          ? this.asCollection(entities, {\n              annots,\n              CollectionType: options?.CollectionType,\n            })\n          : null,\n      ),\n    );\n  }\n\n  fetchOne(\n    options?: ODataOptions & {\n      withCount?: boolean;\n      bodyQueryOptions?: QueryOption[];\n    },\n  ): Observable<{ entity: T | null; annots: ODataEntitiesAnnotations<T> }> {\n    let res = this.clone();\n    res.query((q) => q.top(1));\n    return res.fetch({ responseType: 'entities', ...options }).pipe(\n      map(({ entities, annots }) => ({\n        entity: entities !== null && entities.length === 1 ? entities[0] : null,\n        annots,\n      })),\n    );\n  }\n\n  fetchMany(\n    top: number,\n    options?: ODataOptions & {\n      withCount?: boolean;\n      bodyQueryOptions?: QueryOption[];\n    },\n  ): Observable<{ entities: T[]; annots: ODataEntitiesAnnotations<T> }> {\n    let res = this.clone();\n    let fetch = (opts?: {\n      skip?: number;\n      skiptoken?: string;\n      top?: number;\n    }): Observable<ODataEntities<T>> => {\n      if (opts) {\n        res.query((q) => q.paging(opts));\n      }\n      return res.fetch({ responseType: 'entities', ...options });\n    };\n    return fetch({ top }).pipe(\n      expand(({ annots }) =>\n        annots.skip || annots.skiptoken ? fetch(annots) : EMPTY,\n      ),\n      map(({ entities, annots }) => ({ entities: entities || [], annots })),\n      reduce((acc, { entities, annots }) => ({\n        entities: [...(acc.entities || []), ...(entities || [])],\n        annots: acc.annots.union(annots),\n      })),\n    );\n  }\n\n  /**\n   * Fetch all entities\n   * @param options Options for the request\n   * @returns All entities\n   */\n  fetchAll(\n    options: ODataOptions & {\n      withCount?: boolean;\n      bodyQueryOptions?: QueryOption[];\n    } = {},\n  ): Observable<{ entities: T[]; annots: ODataEntitiesAnnotations<T> }> {\n    let res = this.clone();\n    // Clean Paging\n    res.query((q) => q.removePaging());\n    let fetch = (opts?: {\n      skip?: number;\n      skiptoken?: string;\n      top?: number;\n    }): Observable<ODataEntities<T>> => {\n      if (opts) {\n        res.query((q) => q.paging(opts));\n      }\n      return res.fetch({ responseType: 'entities', ...options });\n    };\n    return fetch().pipe(\n      expand(({ annots }) =>\n        annots.skip || annots.skiptoken ? fetch(annots) : EMPTY,\n      ),\n      map(({ entities, annots }) => ({ entities: entities || [], annots })),\n      reduce((acc, { entities, annots }) => ({\n        entities: [...(acc.entities || []), ...(entities || [])],\n        annots: acc.annots.union(annots),\n      })),\n    );\n  }\n  //#endregion\n}\n"]}