UNPKG

comic-vine-sdk

Version:

A JS/TS client for the Comic Vine API

68 lines 12.5 kB
import { HttpClientFactory } from './http-client/index.js'; import { loadOptions } from './options/index.js'; import { ResourceFactory } from './resources/index.js'; import * as resources from './resources/resource-list.js'; function classNameToPropertyName(className) { return className.charAt(0).toLowerCase() + className.slice(1); } export class ComicVine { constructor(key, options) { this.resourceCache = new Map(); const _options = loadOptions(options); const httpClient = HttpClientFactory.createClient(); const urlBuilder = HttpClientFactory.createUrlBuilder(key, _options.baseUrl); this.resourceFactory = new ResourceFactory(httpClient, urlBuilder); // Discover available resources dynamically this.resourceNames = Object.keys(resources); // Return a proxy that provides lazy loading with full type safety return new Proxy(this, { get(target, prop) { if (typeof prop === 'string' && target.isResourceProperty(prop)) { return target.getResource(prop); } return Reflect.get(target, prop); }, }); } isResourceProperty(prop) { // Check if this property corresponds to a known resource const className = prop.charAt(0).toUpperCase() + prop.slice(1); return this.resourceNames.includes(className); } getResource(propertyName) { // Lazy loading: create resource only when first accessed if (!this.resourceCache.has(propertyName)) { const className = propertyName.charAt(0).toUpperCase() + propertyName.slice(1); try { const resource = this.resourceFactory.create(className); this.resourceCache.set(propertyName, resource); } catch (error) { throw new Error(`Failed to create resource '${className}': ${error}`); } } return this.resourceCache.get(propertyName); } getAvailableResources() { return this.resourceNames.map((name) => classNameToPropertyName(name)); } hasResource(resourceName) { return this.isResourceProperty(resourceName); } getResourceByName(resourceName) { if (!this.isResourceProperty(resourceName)) { return undefined; } return this.getResource(resourceName); } isResourceLoaded(resourceName) { return this.resourceCache.has(resourceName); } getCacheStats() { const total = this.resourceNames.length; const loaded = this.resourceCache.size; const loadedResources = Array.from(this.resourceCache.keys()); return { total, loaded, loadedResources }; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comic-vine.js","sourceRoot":"","sources":["../../src/comic-vine.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClientFactory } from './http-client/index.js';\nimport { userOptions, loadOptions } from './options/index.js';\nimport { ResourceFactory } from './resources/index.js';\nimport * as resources from './resources/resource-list.js';\n\nfunction classNameToPropertyName(className: string): string {\n  return className.charAt(0).toLowerCase() + className.slice(1);\n}\n\ntype ResourceInstance = ReturnType<ResourceFactory['create']>;\n\n// Create resource property type mapping dynamically\ntype ResourcePropertyMap = {\n  [K in keyof typeof resources as Uncapitalize<K>]: InstanceType<\n    (typeof resources)[K]\n  >;\n};\n\nexport class ComicVine implements ResourcePropertyMap {\n  private resourceFactory: ResourceFactory;\n  private resourceCache = new Map<string, ResourceInstance>();\n  private resourceNames: string[];\n\n  // TypeScript property declarations for static typing (will be provided by Proxy)\n  declare readonly character: ResourcePropertyMap['character'];\n  declare readonly concept: ResourcePropertyMap['concept'];\n  declare readonly episode: ResourcePropertyMap['episode'];\n  declare readonly issue: ResourcePropertyMap['issue'];\n  declare readonly location: ResourcePropertyMap['location'];\n  declare readonly movie: ResourcePropertyMap['movie'];\n  declare readonly origin: ResourcePropertyMap['origin'];\n  declare readonly person: ResourcePropertyMap['person'];\n  declare readonly power: ResourcePropertyMap['power'];\n  declare readonly promo: ResourcePropertyMap['promo'];\n  declare readonly publisher: ResourcePropertyMap['publisher'];\n  declare readonly series: ResourcePropertyMap['series'];\n  declare readonly storyArc: ResourcePropertyMap['storyArc'];\n  declare readonly team: ResourcePropertyMap['team'];\n  declare readonly thing: ResourcePropertyMap['thing'];\n  declare readonly video: ResourcePropertyMap['video'];\n  declare readonly videoCategory: ResourcePropertyMap['videoCategory'];\n  declare readonly videoType: ResourcePropertyMap['videoType'];\n  declare readonly volume: ResourcePropertyMap['volume'];\n\n  constructor(key: string, options?: userOptions) {\n    const _options = loadOptions(options);\n    const httpClient = HttpClientFactory.createClient();\n    const urlBuilder = HttpClientFactory.createUrlBuilder(\n      key,\n      _options.baseUrl,\n    );\n    this.resourceFactory = new ResourceFactory(httpClient, urlBuilder);\n\n    // Discover available resources dynamically\n    this.resourceNames = Object.keys(resources);\n\n    // Return a proxy that provides lazy loading with full type safety\n    return new Proxy(this, {\n      get(target, prop: string | symbol) {\n        if (typeof prop === 'string' && target.isResourceProperty(prop)) {\n          return target.getResource(prop);\n        }\n        return Reflect.get(target, prop);\n      },\n    }) as ComicVine;\n  }\n\n  private isResourceProperty(prop: string): boolean {\n    // Check if this property corresponds to a known resource\n    const className = prop.charAt(0).toUpperCase() + prop.slice(1);\n    return this.resourceNames.includes(className);\n  }\n\n  private getResource(propertyName: string): ResourceInstance {\n    // Lazy loading: create resource only when first accessed\n    if (!this.resourceCache.has(propertyName)) {\n      const className =\n        propertyName.charAt(0).toUpperCase() + propertyName.slice(1);\n      try {\n        const resource = this.resourceFactory.create(\n          className as keyof typeof resources,\n        );\n        this.resourceCache.set(propertyName, resource);\n      } catch (error) {\n        throw new Error(`Failed to create resource '${className}': ${error}`);\n      }\n    }\n    return this.resourceCache.get(propertyName)!;\n  }\n\n  getAvailableResources(): string[] {\n    return this.resourceNames.map((name) => classNameToPropertyName(name));\n  }\n\n  hasResource(resourceName: string): boolean {\n    return this.isResourceProperty(resourceName);\n  }\n\n  getResourceByName(resourceName: string): ResourceInstance | undefined {\n    if (!this.isResourceProperty(resourceName)) {\n      return undefined;\n    }\n    return this.getResource(resourceName);\n  }\n\n  isResourceLoaded(resourceName: string): boolean {\n    return this.resourceCache.has(resourceName);\n  }\n\n  getCacheStats(): {\n    total: number;\n    loaded: number;\n    loadedResources: string[];\n  } {\n    const total = this.resourceNames.length;\n    const loaded = this.resourceCache.size;\n    const loadedResources = Array.from(this.resourceCache.keys());\n    return { total, loaded, loadedResources };\n  }\n}\n"]}