comic-vine-sdk 1.3.0 → 1.3.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +1 -5
- package/CONTRIBUTING.md +294 -0
- package/README.md +434 -12
- package/lib/cjs/comic-vine.d.ts +41 -39
- package/lib/cjs/comic-vine.js +91 -78
- package/lib/esm/comic-vine.d.ts +41 -39
- package/lib/esm/comic-vine.js +58 -78
- package/package.json +2 -2
package/lib/cjs/comic-vine.js
CHANGED
|
@@ -1,92 +1,105 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
2
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
36
|
exports.ComicVine = void 0;
|
|
4
37
|
const index_js_1 = require("./http-client/index.js");
|
|
5
38
|
const index_js_2 = require("./options/index.js");
|
|
6
39
|
const index_js_3 = require("./resources/index.js");
|
|
40
|
+
const resources = __importStar(require("./resources/resource-list.js"));
|
|
41
|
+
function classNameToPropertyName(className) {
|
|
42
|
+
return className.charAt(0).toLowerCase() + className.slice(1);
|
|
43
|
+
}
|
|
7
44
|
class ComicVine {
|
|
8
45
|
constructor(key, options) {
|
|
46
|
+
this.resourceCache = new Map();
|
|
9
47
|
const _options = (0, index_js_2.loadOptions)(options);
|
|
10
48
|
const httpClient = index_js_1.HttpClientFactory.createClient();
|
|
11
49
|
const urlBuilder = index_js_1.HttpClientFactory.createUrlBuilder(key, _options.baseUrl);
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
this.
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
this
|
|
27
|
-
|
|
28
|
-
this.
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
return this._publisher;
|
|
65
|
-
}
|
|
66
|
-
get series() {
|
|
67
|
-
return this._series;
|
|
68
|
-
}
|
|
69
|
-
get storyArc() {
|
|
70
|
-
return this._storyArc;
|
|
71
|
-
}
|
|
72
|
-
get team() {
|
|
73
|
-
return this._team;
|
|
74
|
-
}
|
|
75
|
-
get thing() {
|
|
76
|
-
return this._thing;
|
|
77
|
-
}
|
|
78
|
-
get video() {
|
|
79
|
-
return this._video;
|
|
80
|
-
}
|
|
81
|
-
get videoCategory() {
|
|
82
|
-
return this._videoCategory;
|
|
83
|
-
}
|
|
84
|
-
get videoType() {
|
|
85
|
-
return this._videoType;
|
|
86
|
-
}
|
|
87
|
-
get volume() {
|
|
88
|
-
return this._volume;
|
|
50
|
+
this.resourceFactory = new index_js_3.ResourceFactory(httpClient, urlBuilder);
|
|
51
|
+
// Discover available resources dynamically
|
|
52
|
+
this.resourceNames = Object.keys(resources);
|
|
53
|
+
// Return a proxy that provides lazy loading with full type safety
|
|
54
|
+
return new Proxy(this, {
|
|
55
|
+
get(target, prop) {
|
|
56
|
+
if (typeof prop === 'string' && target.isResourceProperty(prop)) {
|
|
57
|
+
return target.getResource(prop);
|
|
58
|
+
}
|
|
59
|
+
return Reflect.get(target, prop);
|
|
60
|
+
},
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
isResourceProperty(prop) {
|
|
64
|
+
// Check if this property corresponds to a known resource
|
|
65
|
+
const className = prop.charAt(0).toUpperCase() + prop.slice(1);
|
|
66
|
+
return this.resourceNames.includes(className);
|
|
67
|
+
}
|
|
68
|
+
getResource(propertyName) {
|
|
69
|
+
// Lazy loading: create resource only when first accessed
|
|
70
|
+
if (!this.resourceCache.has(propertyName)) {
|
|
71
|
+
const className = propertyName.charAt(0).toUpperCase() + propertyName.slice(1);
|
|
72
|
+
try {
|
|
73
|
+
const resource = this.resourceFactory.create(className);
|
|
74
|
+
this.resourceCache.set(propertyName, resource);
|
|
75
|
+
}
|
|
76
|
+
catch (error) {
|
|
77
|
+
throw new Error(`Failed to create resource '${className}': ${error}`);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
return this.resourceCache.get(propertyName);
|
|
81
|
+
}
|
|
82
|
+
getAvailableResources() {
|
|
83
|
+
return this.resourceNames.map((name) => classNameToPropertyName(name));
|
|
84
|
+
}
|
|
85
|
+
hasResource(resourceName) {
|
|
86
|
+
return this.isResourceProperty(resourceName);
|
|
87
|
+
}
|
|
88
|
+
getResourceByName(resourceName) {
|
|
89
|
+
if (!this.isResourceProperty(resourceName)) {
|
|
90
|
+
return undefined;
|
|
91
|
+
}
|
|
92
|
+
return this.getResource(resourceName);
|
|
93
|
+
}
|
|
94
|
+
isResourceLoaded(resourceName) {
|
|
95
|
+
return this.resourceCache.has(resourceName);
|
|
96
|
+
}
|
|
97
|
+
getCacheStats() {
|
|
98
|
+
const total = this.resourceNames.length;
|
|
99
|
+
const loaded = this.resourceCache.size;
|
|
100
|
+
const loadedResources = Array.from(this.resourceCache.keys());
|
|
101
|
+
return { total, loaded, loadedResources };
|
|
89
102
|
}
|
|
90
103
|
}
|
|
91
104
|
exports.ComicVine = ComicVine;
|
|
92
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
105
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comic-vine.js","sourceRoot":"","sources":["../../src/comic-vine.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,qDAA2D;AAC3D,iDAA8D;AAC9D,mDAAuD;AACvD,wEAA0D;AAE1D,SAAS,uBAAuB,CAAC,SAAiB;IAChD,OAAO,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;AAChE,CAAC;AAWD,MAAa,SAAS;IA0BpB,YAAY,GAAW,EAAE,OAAqB;QAxBtC,kBAAa,GAAG,IAAI,GAAG,EAA4B,CAAC;QAyB1D,MAAM,QAAQ,GAAG,IAAA,sBAAW,EAAC,OAAO,CAAC,CAAC;QACtC,MAAM,UAAU,GAAG,4BAAiB,CAAC,YAAY,EAAE,CAAC;QACpD,MAAM,UAAU,GAAG,4BAAiB,CAAC,gBAAgB,CACnD,GAAG,EACH,QAAQ,CAAC,OAAO,CACjB,CAAC;QACF,IAAI,CAAC,eAAe,GAAG,IAAI,0BAAe,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC;QAEnE,2CAA2C;QAC3C,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAE5C,kEAAkE;QAClE,OAAO,IAAI,KAAK,CAAC,IAAI,EAAE;YACrB,GAAG,CAAC,MAAM,EAAE,IAAqB;gBAC/B,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,MAAM,CAAC,kBAAkB,CAAC,IAAI,CAAC,EAAE,CAAC;oBAChE,OAAO,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;gBAClC,CAAC;gBACD,OAAO,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;YACnC,CAAC;SACF,CAAc,CAAC;IAClB,CAAC;IAEO,kBAAkB,CAAC,IAAY;QACrC,yDAAyD;QACzD,MAAM,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC/D,OAAO,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;IAChD,CAAC;IAEO,WAAW,CAAC,YAAoB;QACtC,yDAAyD;QACzD,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE,CAAC;YAC1C,MAAM,SAAS,GACb,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC/D,IAAI,CAAC;gBACH,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,CAC1C,SAAmC,CACpC,CAAC;gBACF,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;YACjD,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,IAAI,KAAK,CAAC,8BAA8B,SAAS,MAAM,KAAK,EAAE,CAAC,CAAC;YACxE,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,CAAE,CAAC;IAC/C,CAAC;IAED,qBAAqB;QACnB,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,uBAAuB,CAAC,IAAI,CAAC,CAAC,CAAC;IACzE,CAAC;IAED,WAAW,CAAC,YAAoB;QAC9B,OAAO,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IAC/C,CAAC;IAED,iBAAiB,CAAC,YAAoB;QACpC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,EAAE,CAAC;YAC3C,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,OAAO,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAED,gBAAgB,CAAC,YAAoB;QACnC,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;IAC9C,CAAC;IAED,aAAa;QAKX,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC;QACxC,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC;QACvC,MAAM,eAAe,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC,CAAC;QAC9D,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,eAAe,EAAE,CAAC;IAC5C,CAAC;CACF;AArGD,8BAqGC","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"]}
|
package/lib/esm/comic-vine.d.ts
CHANGED
|
@@ -1,42 +1,44 @@
|
|
|
1
1
|
import { userOptions } from './options/index.js';
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
private
|
|
10
|
-
private
|
|
11
|
-
private
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
2
|
+
import { ResourceFactory } from './resources/index.js';
|
|
3
|
+
import * as resources from './resources/resource-list.js';
|
|
4
|
+
type ResourceInstance = ReturnType<ResourceFactory['create']>;
|
|
5
|
+
type ResourcePropertyMap = {
|
|
6
|
+
[K in keyof typeof resources as Uncapitalize<K>]: InstanceType<(typeof resources)[K]>;
|
|
7
|
+
};
|
|
8
|
+
export declare class ComicVine implements ResourcePropertyMap {
|
|
9
|
+
private resourceFactory;
|
|
10
|
+
private resourceCache;
|
|
11
|
+
private resourceNames;
|
|
12
|
+
readonly character: ResourcePropertyMap['character'];
|
|
13
|
+
readonly concept: ResourcePropertyMap['concept'];
|
|
14
|
+
readonly episode: ResourcePropertyMap['episode'];
|
|
15
|
+
readonly issue: ResourcePropertyMap['issue'];
|
|
16
|
+
readonly location: ResourcePropertyMap['location'];
|
|
17
|
+
readonly movie: ResourcePropertyMap['movie'];
|
|
18
|
+
readonly origin: ResourcePropertyMap['origin'];
|
|
19
|
+
readonly person: ResourcePropertyMap['person'];
|
|
20
|
+
readonly power: ResourcePropertyMap['power'];
|
|
21
|
+
readonly promo: ResourcePropertyMap['promo'];
|
|
22
|
+
readonly publisher: ResourcePropertyMap['publisher'];
|
|
23
|
+
readonly series: ResourcePropertyMap['series'];
|
|
24
|
+
readonly storyArc: ResourcePropertyMap['storyArc'];
|
|
25
|
+
readonly team: ResourcePropertyMap['team'];
|
|
26
|
+
readonly thing: ResourcePropertyMap['thing'];
|
|
27
|
+
readonly video: ResourcePropertyMap['video'];
|
|
28
|
+
readonly videoCategory: ResourcePropertyMap['videoCategory'];
|
|
29
|
+
readonly videoType: ResourcePropertyMap['videoType'];
|
|
30
|
+
readonly volume: ResourcePropertyMap['volume'];
|
|
22
31
|
constructor(key: string, options?: userOptions);
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
get series(): import("./resources/resource-list.js").Series;
|
|
35
|
-
get storyArc(): import("./resources/resource-list.js").StoryArc;
|
|
36
|
-
get team(): import("./resources/resource-list.js").Team;
|
|
37
|
-
get thing(): import("./resources/resource-list.js").Thing;
|
|
38
|
-
get video(): import("./resources/resource-list.js").Video;
|
|
39
|
-
get videoCategory(): import("./resources/resource-list.js").VideoCategory;
|
|
40
|
-
get videoType(): import("./resources/resource-list.js").VideoType;
|
|
41
|
-
get volume(): import("./resources/resource-list.js").Volume;
|
|
32
|
+
private isResourceProperty;
|
|
33
|
+
private getResource;
|
|
34
|
+
getAvailableResources(): string[];
|
|
35
|
+
hasResource(resourceName: string): boolean;
|
|
36
|
+
getResourceByName(resourceName: string): ResourceInstance | undefined;
|
|
37
|
+
isResourceLoaded(resourceName: string): boolean;
|
|
38
|
+
getCacheStats(): {
|
|
39
|
+
total: number;
|
|
40
|
+
loaded: number;
|
|
41
|
+
loadedResources: string[];
|
|
42
|
+
};
|
|
42
43
|
}
|
|
44
|
+
export {};
|
package/lib/esm/comic-vine.js
CHANGED
|
@@ -1,88 +1,68 @@
|
|
|
1
1
|
import { HttpClientFactory } from './http-client/index.js';
|
|
2
2
|
import { loadOptions } from './options/index.js';
|
|
3
3
|
import { ResourceFactory } from './resources/index.js';
|
|
4
|
+
import * as resources from './resources/resource-list.js';
|
|
5
|
+
function classNameToPropertyName(className) {
|
|
6
|
+
return className.charAt(0).toLowerCase() + className.slice(1);
|
|
7
|
+
}
|
|
4
8
|
export class ComicVine {
|
|
5
9
|
constructor(key, options) {
|
|
10
|
+
this.resourceCache = new Map();
|
|
6
11
|
const _options = loadOptions(options);
|
|
7
12
|
const httpClient = HttpClientFactory.createClient();
|
|
8
13
|
const urlBuilder = HttpClientFactory.createUrlBuilder(key, _options.baseUrl);
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
this.
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
this
|
|
24
|
-
|
|
25
|
-
this.
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
return this._publisher;
|
|
62
|
-
}
|
|
63
|
-
get series() {
|
|
64
|
-
return this._series;
|
|
65
|
-
}
|
|
66
|
-
get storyArc() {
|
|
67
|
-
return this._storyArc;
|
|
68
|
-
}
|
|
69
|
-
get team() {
|
|
70
|
-
return this._team;
|
|
71
|
-
}
|
|
72
|
-
get thing() {
|
|
73
|
-
return this._thing;
|
|
74
|
-
}
|
|
75
|
-
get video() {
|
|
76
|
-
return this._video;
|
|
77
|
-
}
|
|
78
|
-
get videoCategory() {
|
|
79
|
-
return this._videoCategory;
|
|
80
|
-
}
|
|
81
|
-
get videoType() {
|
|
82
|
-
return this._videoType;
|
|
83
|
-
}
|
|
84
|
-
get volume() {
|
|
85
|
-
return this._volume;
|
|
14
|
+
this.resourceFactory = new ResourceFactory(httpClient, urlBuilder);
|
|
15
|
+
// Discover available resources dynamically
|
|
16
|
+
this.resourceNames = Object.keys(resources);
|
|
17
|
+
// Return a proxy that provides lazy loading with full type safety
|
|
18
|
+
return new Proxy(this, {
|
|
19
|
+
get(target, prop) {
|
|
20
|
+
if (typeof prop === 'string' && target.isResourceProperty(prop)) {
|
|
21
|
+
return target.getResource(prop);
|
|
22
|
+
}
|
|
23
|
+
return Reflect.get(target, prop);
|
|
24
|
+
},
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
isResourceProperty(prop) {
|
|
28
|
+
// Check if this property corresponds to a known resource
|
|
29
|
+
const className = prop.charAt(0).toUpperCase() + prop.slice(1);
|
|
30
|
+
return this.resourceNames.includes(className);
|
|
31
|
+
}
|
|
32
|
+
getResource(propertyName) {
|
|
33
|
+
// Lazy loading: create resource only when first accessed
|
|
34
|
+
if (!this.resourceCache.has(propertyName)) {
|
|
35
|
+
const className = propertyName.charAt(0).toUpperCase() + propertyName.slice(1);
|
|
36
|
+
try {
|
|
37
|
+
const resource = this.resourceFactory.create(className);
|
|
38
|
+
this.resourceCache.set(propertyName, resource);
|
|
39
|
+
}
|
|
40
|
+
catch (error) {
|
|
41
|
+
throw new Error(`Failed to create resource '${className}': ${error}`);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
return this.resourceCache.get(propertyName);
|
|
45
|
+
}
|
|
46
|
+
getAvailableResources() {
|
|
47
|
+
return this.resourceNames.map((name) => classNameToPropertyName(name));
|
|
48
|
+
}
|
|
49
|
+
hasResource(resourceName) {
|
|
50
|
+
return this.isResourceProperty(resourceName);
|
|
51
|
+
}
|
|
52
|
+
getResourceByName(resourceName) {
|
|
53
|
+
if (!this.isResourceProperty(resourceName)) {
|
|
54
|
+
return undefined;
|
|
55
|
+
}
|
|
56
|
+
return this.getResource(resourceName);
|
|
57
|
+
}
|
|
58
|
+
isResourceLoaded(resourceName) {
|
|
59
|
+
return this.resourceCache.has(resourceName);
|
|
60
|
+
}
|
|
61
|
+
getCacheStats() {
|
|
62
|
+
const total = this.resourceNames.length;
|
|
63
|
+
const loaded = this.resourceCache.size;
|
|
64
|
+
const loadedResources = Array.from(this.resourceCache.keys());
|
|
65
|
+
return { total, loaded, loadedResources };
|
|
86
66
|
}
|
|
87
67
|
}
|
|
88
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
68
|
+
//# 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"]}
|
package/package.json
CHANGED
|
@@ -66,12 +66,12 @@
|
|
|
66
66
|
"metadata"
|
|
67
67
|
],
|
|
68
68
|
"engines": {
|
|
69
|
-
"node": ">=
|
|
69
|
+
"node": ">= 20.0.0"
|
|
70
70
|
},
|
|
71
71
|
"main": "lib/cjs/index.js",
|
|
72
72
|
"license": "MIT",
|
|
73
73
|
"homepage": "https://github.com/AllyMurray/comic-vine#readme",
|
|
74
|
-
"version": "1.3.
|
|
74
|
+
"version": "1.3.2",
|
|
75
75
|
"bugs": {
|
|
76
76
|
"url": "https://github.com/AllyMurray/comic-vine/issues"
|
|
77
77
|
},
|