loredata 0.2.0 → 0.4.1
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/README.md +126 -65
- package/data/avengers/addresses.json +27 -0
- package/data/avengers/characters.json +305 -0
- package/data/avengers/meta.json +13 -0
- package/data/better-call-saul/addresses.json +37 -0
- package/data/better-call-saul/characters.json +268 -0
- package/data/better-call-saul/meta.json +13 -0
- package/data/big-bang-theory/addresses.json +31 -0
- package/data/big-bang-theory/characters.json +198 -0
- package/data/big-bang-theory/meta.json +13 -0
- package/data/breaking-bad/characters.json +249 -148
- package/data/breaking-bad/meta.json +8 -1
- package/data/fast-and-furious/addresses.json +1 -2
- package/data/fast-and-furious/characters.json +226 -133
- package/data/fast-and-furious/meta.json +13 -1
- package/data/friends/characters.json +402 -281
- package/data/friends/meta.json +8 -1
- package/data/game-of-thrones/addresses.json +6 -22
- package/data/game-of-thrones/characters.json +507 -299
- package/data/game-of-thrones/meta.json +8 -1
- package/data/guardians-of-the-galaxy/addresses.json +17 -0
- package/data/guardians-of-the-galaxy/characters.json +226 -0
- package/data/guardians-of-the-galaxy/meta.json +13 -0
- package/data/harry-potter/addresses.json +6 -11
- package/data/harry-potter/characters.json +239 -147
- package/data/harry-potter/meta.json +8 -1
- package/data/house-md/characters.json +184 -52
- package/data/house-md/meta.json +8 -1
- package/data/lost/characters.json +420 -299
- package/data/lost/meta.json +8 -1
- package/data/matrix/characters.json +144 -34
- package/data/matrix/meta.json +13 -1
- package/data/peaky-blinders/characters.json +191 -37
- package/data/peaky-blinders/meta.json +8 -1
- package/data/prison-break/addresses.json +24 -0
- package/data/prison-break/characters.json +188 -0
- package/data/prison-break/meta.json +13 -0
- package/data/sherlock/characters.json +369 -263
- package/data/sherlock/meta.json +8 -1
- package/data/simpsons/addresses.json +21 -0
- package/data/simpsons/characters.json +383 -0
- package/data/simpsons/meta.json +13 -0
- package/data/sopranos/addresses.json +37 -0
- package/data/sopranos/characters.json +292 -0
- package/data/sopranos/meta.json +13 -0
- package/data/south-park/addresses.json +31 -0
- package/data/south-park/characters.json +372 -0
- package/data/south-park/meta.json +13 -0
- package/data/spider-man/addresses.json +31 -0
- package/data/spider-man/characters.json +219 -0
- package/data/spider-man/meta.json +13 -0
- package/data/star-wars/addresses.json +19 -0
- package/data/star-wars/characters.json +254 -0
- package/data/star-wars/meta.json +13 -0
- package/data/stranger-things/addresses.json +31 -0
- package/data/stranger-things/characters.json +226 -0
- package/data/stranger-things/meta.json +13 -0
- package/data/supernatural/addresses.json +25 -0
- package/data/supernatural/characters.json +190 -0
- package/data/supernatural/meta.json +13 -0
- package/data/the-crown/addresses.json +27 -0
- package/data/the-crown/characters.json +275 -0
- package/data/the-crown/meta.json +13 -0
- package/data/the-office/characters.json +478 -283
- package/data/the-office/meta.json +8 -1
- package/data/the-walking-dead/addresses.json +25 -0
- package/data/the-walking-dead/characters.json +186 -0
- package/data/the-walking-dead/meta.json +13 -0
- package/data/the-witcher/addresses.json +22 -0
- package/data/the-witcher/characters.json +272 -0
- package/data/the-witcher/meta.json +13 -0
- package/data/vikings/addresses.json +26 -0
- package/data/vikings/characters.json +240 -0
- package/data/vikings/meta.json +13 -0
- package/data/westworld/addresses.json +19 -0
- package/data/westworld/characters.json +161 -0
- package/data/westworld/meta.json +13 -0
- package/data/x-men/addresses.json +24 -0
- package/data/x-men/characters.json +242 -0
- package/data/x-men/meta.json +18 -0
- package/dist/browser.cjs +18 -72
- package/dist/browser.cjs.map +1 -1
- package/dist/browser.d.cts +3 -3
- package/dist/browser.d.ts +3 -3
- package/dist/browser.js +18 -72
- package/dist/browser.js.map +1 -1
- package/dist/cli/cli.cjs +41 -73
- package/dist/cli/cli.cjs.map +1 -1
- package/dist/cli/cli.js +41 -73
- package/dist/cli/cli.js.map +1 -1
- package/dist/index.cjs +41 -72
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +5 -4
- package/dist/index.d.ts +5 -4
- package/dist/index.js +41 -72
- package/dist/index.js.map +1 -1
- package/dist/{universe-store-Del1UHgn.d.cts → universe-store-CWhCn3U6.d.cts} +21 -11
- package/dist/{universe-store-Del1UHgn.d.ts → universe-store-CWhCn3U6.d.ts} +21 -11
- package/package.json +85 -81
- package/data/breaking-bad/domains.json +0 -6
- package/data/fast-and-furious/domains.json +0 -6
- package/data/friends/domains.json +0 -6
- package/data/game-of-thrones/domains.json +0 -6
- package/data/harry-potter/domains.json +0 -6
- package/data/house-md/domains.json +0 -6
- package/data/lost/domains.json +0 -6
- package/data/matrix/domains.json +0 -6
- package/data/peaky-blinders/domains.json +0 -6
- package/data/sherlock/domains.json +0 -6
- package/data/the-office/domains.json +0 -6
package/dist/cli/cli.js
CHANGED
|
@@ -26,7 +26,6 @@ var UniverseLoader = class {
|
|
|
26
26
|
const meta = JSON.parse(readFileSync(join(universeDir, "meta.json"), "utf-8"));
|
|
27
27
|
const characters = JSON.parse(readFileSync(join(universeDir, "characters.json"), "utf-8"));
|
|
28
28
|
const addresses = JSON.parse(readFileSync(join(universeDir, "addresses.json"), "utf-8"));
|
|
29
|
-
const domains = JSON.parse(readFileSync(join(universeDir, "domains.json"), "utf-8"));
|
|
30
29
|
const universeData = {
|
|
31
30
|
id: meta.id,
|
|
32
31
|
name: meta.name,
|
|
@@ -34,7 +33,13 @@ var UniverseLoader = class {
|
|
|
34
33
|
description: meta.description,
|
|
35
34
|
characters,
|
|
36
35
|
addresses,
|
|
37
|
-
|
|
36
|
+
...meta.tmdbId !== void 0 && { tmdbId: meta.tmdbId },
|
|
37
|
+
...meta.mediaType !== void 0 && { mediaType: meta.mediaType },
|
|
38
|
+
...meta.rating !== void 0 && { rating: meta.rating },
|
|
39
|
+
...meta.year !== void 0 && { year: meta.year },
|
|
40
|
+
...meta.networks !== void 0 && { networks: meta.networks },
|
|
41
|
+
...meta.posterPath !== void 0 && { posterPath: meta.posterPath },
|
|
42
|
+
...meta.backdropPath !== void 0 && { backdropPath: meta.backdropPath }
|
|
38
43
|
};
|
|
39
44
|
this.cache.set(universeId, universeData);
|
|
40
45
|
return universeData;
|
|
@@ -43,14 +48,7 @@ var UniverseLoader = class {
|
|
|
43
48
|
const ids = this.listAvailable();
|
|
44
49
|
const result = ids.map((id) => {
|
|
45
50
|
const universeDir = join(dataDir, id);
|
|
46
|
-
|
|
47
|
-
const entry = {
|
|
48
|
-
id: meta.id,
|
|
49
|
-
name: meta.name,
|
|
50
|
-
genre: meta.genre,
|
|
51
|
-
description: meta.description
|
|
52
|
-
};
|
|
53
|
-
return entry;
|
|
51
|
+
return JSON.parse(readFileSync(join(universeDir, "meta.json"), "utf-8"));
|
|
54
52
|
});
|
|
55
53
|
return result;
|
|
56
54
|
}
|
|
@@ -70,17 +68,41 @@ var UniverseLoader = class {
|
|
|
70
68
|
}
|
|
71
69
|
static getAllLocations() {
|
|
72
70
|
const ids = this.listAvailable();
|
|
73
|
-
const
|
|
71
|
+
const seen = /* @__PURE__ */ new Map();
|
|
74
72
|
for (const id of ids) {
|
|
75
73
|
const universe = this.load(id);
|
|
76
74
|
for (const address of universe.addresses) {
|
|
77
|
-
if (address.city) {
|
|
78
|
-
|
|
75
|
+
if (address.city && !seen.has(`city:${address.city}`)) {
|
|
76
|
+
seen.set(`city:${address.city}`, { name: address.city, type: "city" });
|
|
77
|
+
}
|
|
78
|
+
if (address.state && !seen.has(`state:${address.state}`)) {
|
|
79
|
+
seen.set(`state:${address.state}`, { name: address.state, type: "state" });
|
|
80
|
+
}
|
|
81
|
+
if (address.country && !seen.has(`country:${address.country}`)) {
|
|
82
|
+
seen.set(`country:${address.country}`, { name: address.country, type: "country" });
|
|
79
83
|
}
|
|
80
84
|
}
|
|
81
85
|
}
|
|
82
|
-
|
|
83
|
-
|
|
86
|
+
return Array.from(seen.values()).sort((a, b) => a.name.localeCompare(b.name));
|
|
87
|
+
}
|
|
88
|
+
static buildCharacterIndex() {
|
|
89
|
+
const ids = this.listAvailable();
|
|
90
|
+
const entries = [];
|
|
91
|
+
for (const id of ids) {
|
|
92
|
+
const universe = this.load(id);
|
|
93
|
+
for (const character of universe.characters) {
|
|
94
|
+
entries.push({
|
|
95
|
+
characterId: character.id,
|
|
96
|
+
universeId: universe.id,
|
|
97
|
+
universeName: universe.name,
|
|
98
|
+
firstName: character.firstName ?? "",
|
|
99
|
+
lastName: character.lastName ?? "",
|
|
100
|
+
interests: character.interests,
|
|
101
|
+
profession: character.profession
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
return entries;
|
|
84
106
|
}
|
|
85
107
|
static listAvailable() {
|
|
86
108
|
const entries = readdirSync(dataDir, { withFileTypes: true });
|
|
@@ -120,55 +142,12 @@ function pickRandom(array, rng) {
|
|
|
120
142
|
|
|
121
143
|
// src/generators/email.ts
|
|
122
144
|
var EmailGenerator = class {
|
|
123
|
-
static
|
|
124
|
-
const domain = pickRandom(
|
|
125
|
-
const first = firstName.toLowerCase();
|
|
126
|
-
const last = lastName.toLowerCase();
|
|
127
|
-
return `${first}.${last}@${domain}`;
|
|
128
|
-
}
|
|
129
|
-
static generateFromUsername(username, domains, rng) {
|
|
130
|
-
const domain = pickRandom(domains.emailDomains, rng);
|
|
145
|
+
static generateFromUsername(username, emailDomains, rng) {
|
|
146
|
+
const domain = pickRandom(emailDomains, rng);
|
|
131
147
|
return `${username}@${domain}`;
|
|
132
148
|
}
|
|
133
149
|
};
|
|
134
150
|
|
|
135
|
-
// src/generators/phone.ts
|
|
136
|
-
var PhoneGenerator = class {
|
|
137
|
-
static generate(domains, rng) {
|
|
138
|
-
if (domains.phoneEasterEggs && domains.phoneEasterEggs.length > 0 && rng() < 0.2) {
|
|
139
|
-
return pickRandom(domains.phoneEasterEggs, rng);
|
|
140
|
-
}
|
|
141
|
-
const prefix = pickRandom(domains.phonePrefixes, rng);
|
|
142
|
-
const suffix = this.randomDigits(7, rng);
|
|
143
|
-
return `${prefix}-${suffix.slice(0, 3)}-${suffix.slice(3)}`;
|
|
144
|
-
}
|
|
145
|
-
static randomDigits(count, rng) {
|
|
146
|
-
let result = "";
|
|
147
|
-
for (let i = 0; i < count; i++) {
|
|
148
|
-
result += Math.floor(rng() * 10).toString();
|
|
149
|
-
}
|
|
150
|
-
return result;
|
|
151
|
-
}
|
|
152
|
-
};
|
|
153
|
-
|
|
154
|
-
// src/generators/password.ts
|
|
155
|
-
var PasswordGenerator = class {
|
|
156
|
-
static generate(domains, rng) {
|
|
157
|
-
if (domains.passwordEasterEggs && domains.passwordEasterEggs.length > 0 && rng() < 0.3) {
|
|
158
|
-
return pickRandom(domains.passwordEasterEggs, rng);
|
|
159
|
-
}
|
|
160
|
-
return this.randomPassword(12, rng);
|
|
161
|
-
}
|
|
162
|
-
static randomPassword(length, rng) {
|
|
163
|
-
const chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$";
|
|
164
|
-
let result = "";
|
|
165
|
-
for (let i = 0; i < length; i++) {
|
|
166
|
-
result += chars[Math.floor(rng() * chars.length)];
|
|
167
|
-
}
|
|
168
|
-
return result;
|
|
169
|
-
}
|
|
170
|
-
};
|
|
171
|
-
|
|
172
151
|
// src/engine/person-factory.ts
|
|
173
152
|
var PersonFactory = class {
|
|
174
153
|
static build(universe, seed) {
|
|
@@ -191,11 +170,8 @@ var PersonFactory = class {
|
|
|
191
170
|
}
|
|
192
171
|
const address = character.address ?? universe.addresses[0];
|
|
193
172
|
const username = character.usernames[0];
|
|
194
|
-
const domains = character.emailDomain ? { ...universe.domains, emailDomains: [character.emailDomain] } : universe.domains;
|
|
195
173
|
const rng = createRng(0);
|
|
196
|
-
const email = EmailGenerator.generateFromUsername(username,
|
|
197
|
-
const phone = PhoneGenerator.generate(universe.domains, rng);
|
|
198
|
-
const password = PasswordGenerator.generate(universe.domains, rng);
|
|
174
|
+
const email = EmailGenerator.generateFromUsername(username, character.emailDomains, rng);
|
|
199
175
|
const quote = character.quotes[0];
|
|
200
176
|
const person = {
|
|
201
177
|
id: crypto.randomUUID(),
|
|
@@ -204,8 +180,6 @@ var PersonFactory = class {
|
|
|
204
180
|
lastName: character.lastName,
|
|
205
181
|
username,
|
|
206
182
|
email,
|
|
207
|
-
password,
|
|
208
|
-
phone,
|
|
209
183
|
address: {
|
|
210
184
|
street: address.street,
|
|
211
185
|
city: address.city,
|
|
@@ -226,10 +200,7 @@ var PersonFactory = class {
|
|
|
226
200
|
static buildFromCharacter(character, universe, rng) {
|
|
227
201
|
const address = character.address ?? pickRandom(universe.addresses, rng);
|
|
228
202
|
const username = pickRandom(character.usernames, rng);
|
|
229
|
-
const
|
|
230
|
-
const email = EmailGenerator.generateFromUsername(username, domains, rng);
|
|
231
|
-
const phone = PhoneGenerator.generate(universe.domains, rng);
|
|
232
|
-
const password = PasswordGenerator.generate(universe.domains, rng);
|
|
203
|
+
const email = EmailGenerator.generateFromUsername(username, character.emailDomains, rng);
|
|
233
204
|
const quote = pickRandom(character.quotes, rng);
|
|
234
205
|
const person = {
|
|
235
206
|
id: crypto.randomUUID(),
|
|
@@ -238,8 +209,6 @@ var PersonFactory = class {
|
|
|
238
209
|
lastName: character.lastName,
|
|
239
210
|
username,
|
|
240
211
|
email,
|
|
241
|
-
password,
|
|
242
|
-
phone,
|
|
243
212
|
address: {
|
|
244
213
|
street: address.street,
|
|
245
214
|
city: address.city,
|
|
@@ -430,7 +399,6 @@ program.command("person").description("Generate a single persona").option("-u, -
|
|
|
430
399
|
`
|
|
431
400
|
Name: ${result.firstName} ${result.lastName} (@${result.username})
|
|
432
401
|
Email: ${result.email}
|
|
433
|
-
Phone: ${result.phone}
|
|
434
402
|
Address: ${[result.address.street, result.address.city, result.address.country].filter(Boolean).join(", ")}
|
|
435
403
|
Profession: ${result.profession}
|
|
436
404
|
Interests: ${result.interests.join(", ")}
|
package/dist/cli/cli.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/cli/cli.ts","../../src/universes/loader.ts","../../src/generators/random.ts","../../src/generators/email.ts","../../src/generators/phone.ts","../../src/generators/password.ts","../../src/engine/person-factory.ts","../../src/engine/persona-builder.ts","../../src/engine/universe-store.ts","../../src/index.ts"],"sourcesContent":["#!/usr/bin/env node\nimport { Command } from 'commander';\n\nimport { universes, UniverseLoader, UniverseStore } from '@/index';\n\nimport type { CharacterQuery } from '@/types';\n\nconst program = new Command();\n\nprogram.name('loredata').description('Pop culture fake data generator').version('0.1.0');\n\nprogram\n\t.command('person')\n\t.description('Generate a single persona')\n\t.option('-u, --universe <id>', 'Universe ID (e.g. breaking-bad). Omit for all universes.')\n\t.option('-s, --seed <n>', 'Seed for deterministic generation')\n\t.option('-f, --format <fmt>', 'Output format: pretty | json', 'pretty')\n\t.option('--interests <list>', 'Comma-separated interests filter (e.g. chemistry,cooking)')\n\t.option('--interests-mode <mode>', 'Match mode: and | or (default: or)', 'or')\n\t.option('--name <query>', 'Search by character name')\n\t.action(\n\t\t(options: { universe?: string; seed?: string; format: string; interests?: string; interestsMode: string; name?: string }) => {\n\t\t\tconst seed = options.seed ? parseInt(options.seed, 10) : undefined;\n\t\t\tconst universeIds = options.universe ? [options.universe] : UniverseLoader.listAvailable();\n\t\t\tconst universesData = universeIds.map((id) => UniverseLoader.load(id));\n\t\t\tconst storeInstance = new UniverseStore(universesData);\n\n\t\t\tconst query: CharacterQuery = {\n\t\t\t\tuniverses: options.universe ? [options.universe] : [],\n\t\t\t\tinterests: options.interests ? options.interests.split(',').map((s) => s.trim()) : [],\n\t\t\t\tinterestsMode: options.interestsMode === 'and' ? 'and' : 'or',\n\t\t\t\tname: options.name\n\t\t\t};\n\n\t\t\tconst result = storeInstance.generatePersonas(query, 1, seed)[0];\n\n\t\t\tif (!result) {\n\t\t\t\tconsole.error('No characters match the given filters.');\n\t\t\t\tprocess.exit(1);\n\t\t\t}\n\n\t\t\tif (options.format === 'json') {\n\t\t\t\tconsole.log(JSON.stringify(result, null, 2));\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconsole.log(\n\t\t\t\t`\nName: ${result.firstName} ${result.lastName} (@${result.username})\nEmail: ${result.email}\nPhone: ${result.phone}\nAddress: ${[result.address.street, result.address.city, result.address.country].filter(Boolean).join(', ')}\nProfession: ${result.profession}\nInterests: ${result.interests.join(', ')}\nQuote: ${result.quote}\nUniverse: ${result.universe}\n `.trim()\n\t\t\t);\n\t\t}\n\t);\n\nprogram\n\t.command('group')\n\t.description('Generate multiple personas')\n\t.option('-u, --universe <id>', 'Universe ID (e.g. got). Omit for all universes.')\n\t.option('-n, --size <n>', 'Number of personas', '3')\n\t.option('-s, --seed <n>', 'Seed for deterministic generation')\n\t.option('-f, --format <fmt>', 'Output format: pretty | json', 'pretty')\n\t.option('--interests <list>', 'Comma-separated interests filter (e.g. chemistry,cooking)')\n\t.option('--interests-mode <mode>', 'Match mode: and | or (default: or)', 'or')\n\t.option('--name <query>', 'Search by character name')\n\t.action(\n\t\t(options: {\n\t\t\tuniverse?: string;\n\t\t\tsize: string;\n\t\t\tseed?: string;\n\t\t\tformat: string;\n\t\t\tinterests?: string;\n\t\t\tinterestsMode: string;\n\t\t\tname?: string;\n\t\t}) => {\n\t\t\tconst size = parseInt(options.size, 10);\n\t\t\tconst seed = options.seed ? parseInt(options.seed, 10) : undefined;\n\t\t\tconst universeIds = options.universe ? [options.universe] : UniverseLoader.listAvailable();\n\t\t\tconst universesData = universeIds.map((id) => UniverseLoader.load(id));\n\t\t\tconst storeInstance = new UniverseStore(universesData);\n\n\t\t\tconst query: CharacterQuery = {\n\t\t\t\tuniverses: options.universe ? [options.universe] : [],\n\t\t\t\tinterests: options.interests ? options.interests.split(',').map((s) => s.trim()) : [],\n\t\t\t\tinterestsMode: options.interestsMode === 'and' ? 'and' : 'or',\n\t\t\t\tname: options.name\n\t\t\t};\n\n\t\t\tconst result = storeInstance.generatePersonas(query, size, seed);\n\n\t\t\tif (result.length === 0) {\n\t\t\t\tconsole.error('No characters match the given filters.');\n\t\t\t\tprocess.exit(1);\n\t\t\t}\n\n\t\t\tif (options.format === 'json') {\n\t\t\t\tconsole.log(JSON.stringify(result, null, 2));\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor (const p of result) {\n\t\t\t\tconsole.log(`${p.firstName} ${p.lastName} <${p.email}> — ${p.profession}`);\n\t\t\t}\n\t\t}\n\t);\n\nprogram\n\t.command('universes')\n\t.description('List available universes')\n\t.option('-f, --format <fmt>', 'Output format: pretty | json', 'pretty')\n\t.action((options: { format: string }) => {\n\t\tconst list = universes();\n\n\t\tif (options.format === 'json') {\n\t\t\tconsole.log(JSON.stringify(list, null, 2));\n\n\t\t\treturn;\n\t\t}\n\n\t\tfor (const u of list) {\n\t\t\tconsole.log(` ${u.id.padEnd(20)} ${u.name}`);\n\t\t}\n\t});\n\nprogram.parse();\n","import { readFileSync, readdirSync } from 'fs';\nimport { createRequire } from 'module';\nimport { dirname, join } from 'path';\n\nimport type { UniverseData, UniverseMeta, CharacterData, AddressData, DomainsData } from '@/types';\n\nconst require = createRequire(import.meta.url);\nconst packageRoot = dirname(require.resolve('loredata/package.json'));\nconst dataDir = join(packageRoot, 'data');\n\nexport class UniverseLoader {\n\tprivate static cache = new Map<string, UniverseData>();\n\n\tstatic register(data: UniverseData): void {\n\t\tthis.cache.set(data.id, data);\n\t}\n\n\tstatic load(universeId: string): UniverseData {\n\t\tif (process.env.NODE_ENV !== 'development') {\n\t\t\tconst cached = this.cache.get(universeId);\n\n\t\t\tif (cached) {\n\t\t\t\treturn cached;\n\t\t\t}\n\t\t}\n\n\t\tconst universeDir = join(dataDir, universeId);\n\t\tconst meta = JSON.parse(readFileSync(join(universeDir, 'meta.json'), 'utf-8')) as {\n\t\t\tid: string;\n\t\t\tname: string;\n\t\t\tgenre: string[];\n\t\t\tdescription: string;\n\t\t};\n\n\t\tconst characters = JSON.parse(readFileSync(join(universeDir, 'characters.json'), 'utf-8')) as CharacterData[];\n\n\t\tconst addresses = JSON.parse(readFileSync(join(universeDir, 'addresses.json'), 'utf-8')) as AddressData[];\n\n\t\tconst domains = JSON.parse(readFileSync(join(universeDir, 'domains.json'), 'utf-8')) as DomainsData;\n\n\t\tconst universeData: UniverseData = {\n\t\t\tid: meta.id,\n\t\t\tname: meta.name,\n\t\t\tgenre: meta.genre,\n\t\t\tdescription: meta.description,\n\t\t\tcharacters,\n\t\t\taddresses,\n\t\t\tdomains\n\t\t};\n\n\t\tthis.cache.set(universeId, universeData);\n\n\t\treturn universeData;\n\t}\n\n\tstatic getManifest(): UniverseMeta[] {\n\t\tconst ids = this.listAvailable();\n\n\t\tconst result = ids.map((id) => {\n\t\t\tconst universeDir = join(dataDir, id);\n\t\t\tconst meta = JSON.parse(readFileSync(join(universeDir, 'meta.json'), 'utf-8')) as UniverseMeta;\n\n\t\t\tconst entry: UniverseMeta = {\n\t\t\t\tid: meta.id,\n\t\t\t\tname: meta.name,\n\t\t\t\tgenre: meta.genre,\n\t\t\t\tdescription: meta.description\n\t\t\t};\n\n\t\t\treturn entry;\n\t\t});\n\n\t\treturn result;\n\t}\n\n\tstatic getAllInterests(): string[] {\n\t\tconst ids = this.listAvailable();\n\t\tconst interestSet = new Set<string>();\n\n\t\tfor (const id of ids) {\n\t\t\tconst universe = this.load(id);\n\n\t\t\tfor (const character of universe.characters) {\n\t\t\t\tfor (const interest of character.interests) {\n\t\t\t\t\tinterestSet.add(interest);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst result = Array.from(interestSet).sort();\n\n\t\treturn result;\n\t}\n\n\tstatic getAllLocations(): string[] {\n\t\tconst ids = this.listAvailable();\n\t\tconst citySet = new Set<string>();\n\n\t\tfor (const id of ids) {\n\t\t\tconst universe = this.load(id);\n\n\t\t\tfor (const address of universe.addresses) {\n\t\t\t\tif (address.city) {\n\t\t\t\t\tcitySet.add(address.city);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst result = Array.from(citySet).sort();\n\n\t\treturn result;\n\t}\n\n\tstatic listAvailable(): string[] {\n\t\tconst entries = readdirSync(dataDir, { withFileTypes: true });\n\n\t\tconst ids = entries\n\t\t\t.filter((entry) => {\n\t\t\t\tif (!entry.isDirectory()) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\ttry {\n\t\t\t\t\treadFileSync(join(dataDir, entry.name, 'meta.json'), 'utf-8');\n\n\t\t\t\t\treturn true;\n\t\t\t\t} catch {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\t\t\t})\n\t\t\t.map((entry) => entry.name);\n\n\t\treturn ids;\n\t}\n}\n","/**\n * Seeded pseudo-random number generator (mulberry32).\n * Returns a function that produces values in [0, 1).\n */\nexport function createRng(seed?: number): () => number {\n\tif (seed === undefined) {\n\t\treturn Math.random;\n\t}\n\n\tlet s = seed;\n\n\treturn function rng(): number {\n\t\ts += 0x6d2b79f5;\n\t\tlet z = s;\n\t\tz = Math.imul(z ^ (z >>> 15), z | 1);\n\t\tz ^= z + Math.imul(z ^ (z >>> 7), z | 61);\n\n\t\treturn ((z ^ (z >>> 14)) >>> 0) / 4294967296;\n\t};\n}\n\nexport function pickRandom<T>(array: T[], rng: () => number): T {\n\tconst index = Math.floor(rng() * array.length);\n\n\treturn array[index];\n}\n","import { pickRandom } from './random';\n\nimport type { DomainsData } from '@/types';\n\nexport class EmailGenerator {\n\tstatic generate(firstName: string, lastName: string, domains: DomainsData, rng: () => number): string {\n\t\tconst domain = pickRandom(domains.emailDomains, rng);\n\t\tconst first = firstName.toLowerCase();\n\t\tconst last = lastName.toLowerCase();\n\n\t\treturn `${first}.${last}@${domain}`;\n\t}\n\n\tstatic generateFromUsername(username: string, domains: DomainsData, rng: () => number): string {\n\t\tconst domain = pickRandom(domains.emailDomains, rng);\n\n\t\treturn `${username}@${domain}`;\n\t}\n}\n","import { pickRandom } from './random';\n\nimport type { DomainsData } from '@/types';\n\nexport class PhoneGenerator {\n\tstatic generate(domains: DomainsData, rng: () => number): string {\n\t\tif (domains.phoneEasterEggs && domains.phoneEasterEggs.length > 0 && rng() < 0.2) {\n\t\t\treturn pickRandom(domains.phoneEasterEggs, rng);\n\t\t}\n\n\t\tconst prefix = pickRandom(domains.phonePrefixes, rng);\n\t\tconst suffix = this.randomDigits(7, rng);\n\n\t\treturn `${prefix}-${suffix.slice(0, 3)}-${suffix.slice(3)}`;\n\t}\n\n\tprivate static randomDigits(count: number, rng: () => number): string {\n\t\tlet result = '';\n\n\t\tfor (let i = 0; i < count; i++) {\n\t\t\tresult += Math.floor(rng() * 10).toString();\n\t\t}\n\n\t\treturn result;\n\t}\n}\n","import { pickRandom } from './random';\n\nimport type { DomainsData } from '@/types';\n\nexport class PasswordGenerator {\n\tstatic generate(domains: DomainsData, rng: () => number): string {\n\t\tif (domains.passwordEasterEggs && domains.passwordEasterEggs.length > 0 && rng() < 0.3) {\n\t\t\treturn pickRandom(domains.passwordEasterEggs, rng);\n\t\t}\n\n\t\treturn this.randomPassword(12, rng);\n\t}\n\n\tprivate static randomPassword(length: number, rng: () => number): string {\n\t\tconst chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$';\n\t\tlet result = '';\n\n\t\tfor (let i = 0; i < length; i++) {\n\t\t\tresult += chars[Math.floor(rng() * chars.length)];\n\t\t}\n\n\t\treturn result;\n\t}\n}\n","import { createRng, pickRandom, EmailGenerator, PhoneGenerator, PasswordGenerator } from '@/generators';\n\nimport type { Person, UniverseData, CharacterData } from '@/types';\n\nexport class PersonFactory {\n\tstatic build(universe: UniverseData, seed?: number): Person {\n\t\tconst rng = createRng(seed);\n\n\t\tconst character = pickRandom(universe.characters, rng);\n\n\t\treturn this.buildFromCharacter(character, universe, rng);\n\t}\n\n\tstatic buildByCharacterId(characterId: string, universe: UniverseData, seed?: number): Person {\n\t\tconst character = universe.characters.find((c) => c.id === characterId);\n\n\t\tif (!character) {\n\t\t\tthrow new Error(`Character \"${characterId}\" not found in universe \"${universe.id}\"`);\n\t\t}\n\n\t\tconst rng = createRng(seed);\n\n\t\treturn this.buildFromCharacter(character, universe, rng);\n\t}\n\n\tstatic buildCanonical(characterId: string, universe: UniverseData): Person {\n\t\tconst character = universe.characters.find((c) => c.id === characterId);\n\n\t\tif (!character) {\n\t\t\tthrow new Error(`Character \"${characterId}\" not found in universe \"${universe.id}\"`);\n\t\t}\n\n\t\tconst address = character.address ?? universe.addresses[0];\n\t\tconst username = character.usernames[0];\n\t\tconst domains = character.emailDomain ? { ...universe.domains, emailDomains: [character.emailDomain] } : universe.domains;\n\t\tconst rng = createRng(0);\n\t\tconst email = EmailGenerator.generateFromUsername(username, domains, rng);\n\t\tconst phone = PhoneGenerator.generate(universe.domains, rng);\n\t\tconst password = PasswordGenerator.generate(universe.domains, rng);\n\t\tconst quote = character.quotes[0];\n\n\t\tconst person: Person = {\n\t\t\tid: crypto.randomUUID(),\n\t\t\tcharacterId: character.id,\n\t\t\tfirstName: character.firstName,\n\t\t\tlastName: character.lastName,\n\t\t\tusername,\n\t\t\temail,\n\t\t\tpassword,\n\t\t\tphone,\n\t\t\taddress: {\n\t\t\t\tstreet: address.street,\n\t\t\t\tcity: address.city,\n\t\t\t\tstate: address.state ?? '',\n\t\t\t\tzip: address.zip ?? '',\n\t\t\t\tcountry: address.country\n\t\t\t},\n\t\t\tprofession: character.profession,\n\t\t\tinterests: character.interests,\n\t\t\tquote,\n\t\t\tuniverse: universe.id,\n\t\t\tuniverseName: universe.name,\n\t\t\t...(character.symbol ? { symbol: character.symbol } : {}),\n\t\t\t...(character.color ? { color: character.color } : {})\n\t\t};\n\n\t\treturn person;\n\t}\n\n\tprivate static buildFromCharacter(character: CharacterData, universe: UniverseData, rng: () => number): Person {\n\t\tconst address = character.address ?? pickRandom(universe.addresses, rng);\n\n\t\tconst username = pickRandom(character.usernames, rng);\n\n\t\tconst domains = character.emailDomain ? { ...universe.domains, emailDomains: [character.emailDomain] } : universe.domains;\n\n\t\tconst email = EmailGenerator.generateFromUsername(username, domains, rng);\n\t\tconst phone = PhoneGenerator.generate(universe.domains, rng);\n\t\tconst password = PasswordGenerator.generate(universe.domains, rng);\n\t\tconst quote = pickRandom(character.quotes, rng);\n\n\t\tconst person: Person = {\n\t\t\tid: crypto.randomUUID(),\n\t\t\tcharacterId: character.id,\n\t\t\tfirstName: character.firstName,\n\t\t\tlastName: character.lastName,\n\t\t\tusername,\n\t\t\temail,\n\t\t\tpassword,\n\t\t\tphone,\n\t\t\taddress: {\n\t\t\t\tstreet: address.street,\n\t\t\t\tcity: address.city,\n\t\t\t\tstate: address.state ?? '',\n\t\t\t\tzip: address.zip ?? '',\n\t\t\t\tcountry: address.country\n\t\t\t},\n\t\t\tprofession: character.profession,\n\t\t\tinterests: character.interests,\n\t\t\tquote,\n\t\t\tuniverse: universe.id,\n\t\t\tuniverseName: universe.name,\n\t\t\t...(character.symbol ? { symbol: character.symbol } : {}),\n\t\t\t...(character.color ? { color: character.color } : {})\n\t\t};\n\n\t\treturn person;\n\t}\n}\n","import { UniverseLoader } from '@/universes';\n\nimport { PersonFactory } from './person-factory';\n\nimport type { Person, PersonOptions, GroupOptions, UniverseMeta, UniverseData } from '@/types';\n\nexport class PersonaBuilder {\n\tstatic buildPersonFromData(universe: UniverseData, options: Omit<PersonOptions, 'universe'>): Person {\n\t\treturn PersonFactory.build(universe, options.seed);\n\t}\n\n\tstatic buildPerson(options: PersonOptions): Person {\n\t\tconst universe = UniverseLoader.load(options.universe);\n\n\t\treturn PersonFactory.build(universe, options.seed);\n\t}\n\n\tstatic buildGroup(options: GroupOptions): Person[] {\n\t\tconst persons: Person[] = [];\n\n\t\tfor (let i = 0; i < options.size; i++) {\n\t\t\tconst seed = options.seed !== undefined ? options.seed + i : undefined;\n\n\t\t\tpersons.push(this.buildPerson({ universe: options.universe, seed }));\n\t\t}\n\n\t\treturn persons;\n\t}\n\n\tstatic listUniverses(): UniverseMeta[] {\n\t\tconst ids = UniverseLoader.listAvailable();\n\n\t\tconst result: UniverseMeta[] = ids.map((id) => {\n\t\t\tconst data = UniverseLoader.load(id);\n\n\t\t\tconst meta: UniverseMeta = {\n\t\t\t\tid,\n\t\t\t\tname: data.name,\n\t\t\t\tgenre: data.genre,\n\t\t\t\tdescription: data.description\n\t\t\t};\n\n\t\t\treturn meta;\n\t\t});\n\n\t\treturn result;\n\t}\n}\n","import { createRng } from '@/generators';\n\nimport { PersonFactory } from './person-factory';\n\nimport type { UniverseData, UniverseMeta, Person, CharacterEntry, CharacterQuery } from '@/types';\n\nexport class UniverseStore {\n\tprivate universes: UniverseData[];\n\tprivate characterIndex: CharacterEntry[];\n\n\tconstructor(universes: UniverseData[]) {\n\t\tthis.universes = universes;\n\t\tthis.characterIndex = this.buildIndex(universes);\n\t}\n\n\tget index(): CharacterEntry[] {\n\t\treturn this.characterIndex;\n\t}\n\n\tgetInterests(universeIds?: string[]): string[] {\n\t\tconst entries =\n\t\t\tuniverseIds && universeIds.length > 0\n\t\t\t\t? this.characterIndex.filter((e) => universeIds.includes(e.universeId))\n\t\t\t\t: this.characterIndex;\n\n\t\tconst interestSet = new Set<string>();\n\n\t\tfor (const entry of entries) {\n\t\t\tfor (const interest of entry.interests) {\n\t\t\t\tinterestSet.add(interest);\n\t\t\t}\n\t\t}\n\n\t\tconst result = Array.from(interestSet).sort();\n\n\t\treturn result;\n\t}\n\n\tgetUniverses(): UniverseMeta[] {\n\t\tconst result = this.universes.map((u) => {\n\t\t\tconst meta: UniverseMeta = {\n\t\t\t\tid: u.id,\n\t\t\t\tname: u.name,\n\t\t\t\tgenre: u.genre,\n\t\t\t\tdescription: u.description\n\t\t\t};\n\n\t\t\treturn meta;\n\t\t});\n\n\t\treturn result;\n\t}\n\n\tfindCharacters(query: CharacterQuery): CharacterEntry[] {\n\t\tlet entries = this.characterIndex;\n\n\t\tif (query.universes && query.universes.length > 0) {\n\t\t\tentries = entries.filter((e) => query.universes!.includes(e.universeId));\n\t\t}\n\n\t\tif (query.name && query.name.trim().length > 0) {\n\t\t\tconst needle = query.name.toLowerCase();\n\n\t\t\tentries = entries.filter((e) => {\n\t\t\t\tconst fullName = `${e.firstName} ${e.lastName}`.toLowerCase();\n\n\t\t\t\treturn fullName.includes(needle);\n\t\t\t});\n\t\t}\n\n\t\tif (query.interests && query.interests.length > 0) {\n\t\t\tconst mode = query.interestsMode ?? 'or';\n\n\t\t\tif (mode === 'and') {\n\t\t\t\tentries = entries.filter((e) => query.interests!.every((interest) => e.interests.includes(interest)));\n\t\t\t} else {\n\t\t\t\tentries = entries.filter((e) => query.interests!.some((interest) => e.interests.includes(interest)));\n\t\t\t}\n\t\t}\n\n\t\treturn entries;\n\t}\n\n\tpersonByCharacterId(characterId: string, seed?: number): Person {\n\t\tconst entry = this.characterIndex.find((e) => e.characterId === characterId);\n\n\t\tif (!entry) {\n\t\t\tthrow new Error(`Character \"${characterId}\" not found in store`);\n\t\t}\n\n\t\tconst universe = this.universes.find((u) => u.id === entry.universeId);\n\n\t\tif (!universe) {\n\t\t\tthrow new Error(`Universe \"${entry.universeId}\" not found in store`);\n\t\t}\n\n\t\treturn PersonFactory.buildByCharacterId(characterId, universe, seed);\n\t}\n\n\tgeneratePersonas(query: CharacterQuery, count: number, seed?: number): Person[] {\n\t\tconst matching = this.findCharacters(query);\n\n\t\tif (matching.length === 0) {\n\t\t\treturn [];\n\t\t}\n\n\t\tconst rng = createRng(seed);\n\t\tconst actualCount = Math.min(count, matching.length);\n\n\t\tconst shuffled = [...matching].sort(() => rng() - 0.5);\n\t\tconst selected = shuffled.slice(0, actualCount);\n\t\tconst personas: Person[] = [];\n\n\t\tfor (let i = 0; i < selected.length; i++) {\n\t\t\tconst entry = selected[i];\n\t\t\tconst universe = this.universes.find((u) => u.id === entry.universeId)!;\n\t\t\tconst personSeed = seed !== undefined ? seed + i : undefined;\n\n\t\t\tpersonas.push(PersonFactory.buildByCharacterId(entry.characterId, universe, personSeed));\n\t\t}\n\n\t\treturn personas;\n\t}\n\n\tprivate buildIndex(universes: UniverseData[]): CharacterEntry[] {\n\t\tconst index: CharacterEntry[] = [];\n\n\t\tfor (const universe of universes) {\n\t\t\tfor (const character of universe.characters) {\n\t\t\t\tconst entry: CharacterEntry = {\n\t\t\t\t\tcharacterId: character.id,\n\t\t\t\t\tuniverseId: universe.id,\n\t\t\t\t\tuniverseName: universe.name,\n\t\t\t\t\tfirstName: character.firstName,\n\t\t\t\t\tlastName: character.lastName,\n\t\t\t\t\tinterests: character.interests,\n\t\t\t\t\tprofession: character.profession\n\t\t\t\t};\n\n\t\t\t\tindex.push(entry);\n\t\t\t}\n\t\t}\n\n\t\treturn index;\n\t}\n}\n","import { PersonaBuilder } from '@/engine';\n\nimport type { Person, UniverseMeta, PersonOptions, GroupOptions, UniverseData } from '@/types';\n\nexport function person(options: PersonOptions): Person {\n\treturn PersonaBuilder.buildPerson(options);\n}\n\nexport function personFromData(universe: UniverseData, seed?: number): Person {\n\treturn PersonaBuilder.buildPersonFromData(universe, { seed });\n}\n\nexport function group(options: GroupOptions): Person[] {\n\treturn PersonaBuilder.buildGroup(options);\n}\n\nexport function universes(): UniverseMeta[] {\n\treturn PersonaBuilder.listUniverses();\n}\n\nexport { UniverseLoader } from '@/universes';\nexport { UniverseStore, PersonFactory } from '@/engine';\n\nexport type { Person, UniverseMeta, PersonOptions, GroupOptions } from '@/types';\nexport type { Address, UniverseData, CharacterData, AddressData, DomainsData } from '@/types';\nexport type { CharacterEntry, CharacterQuery } from '@/types';\n"],"mappings":";;;AACA,SAAS,eAAe;;;ACDxB,SAAS,cAAc,mBAAmB;AAC1C,SAAS,qBAAqB;AAC9B,SAAS,SAAS,YAAY;AAI9B,IAAMA,WAAU,cAAc,YAAY,GAAG;AAC7C,IAAM,cAAc,QAAQA,SAAQ,QAAQ,uBAAuB,CAAC;AACpE,IAAM,UAAU,KAAK,aAAa,MAAM;AAEjC,IAAM,iBAAN,MAAqB;AAAA,EAC3B,OAAe,QAAQ,oBAAI,IAA0B;AAAA,EAErD,OAAO,SAAS,MAA0B;AACzC,SAAK,MAAM,IAAI,KAAK,IAAI,IAAI;AAAA,EAC7B;AAAA,EAEA,OAAO,KAAK,YAAkC;AAC7C,QAAI,QAAQ,IAAI,aAAa,eAAe;AAC3C,YAAM,SAAS,KAAK,MAAM,IAAI,UAAU;AAExC,UAAI,QAAQ;AACX,eAAO;AAAA,MACR;AAAA,IACD;AAEA,UAAM,cAAc,KAAK,SAAS,UAAU;AAC5C,UAAM,OAAO,KAAK,MAAM,aAAa,KAAK,aAAa,WAAW,GAAG,OAAO,CAAC;AAO7E,UAAM,aAAa,KAAK,MAAM,aAAa,KAAK,aAAa,iBAAiB,GAAG,OAAO,CAAC;AAEzF,UAAM,YAAY,KAAK,MAAM,aAAa,KAAK,aAAa,gBAAgB,GAAG,OAAO,CAAC;AAEvF,UAAM,UAAU,KAAK,MAAM,aAAa,KAAK,aAAa,cAAc,GAAG,OAAO,CAAC;AAEnF,UAAM,eAA6B;AAAA,MAClC,IAAI,KAAK;AAAA,MACT,MAAM,KAAK;AAAA,MACX,OAAO,KAAK;AAAA,MACZ,aAAa,KAAK;AAAA,MAClB;AAAA,MACA;AAAA,MACA;AAAA,IACD;AAEA,SAAK,MAAM,IAAI,YAAY,YAAY;AAEvC,WAAO;AAAA,EACR;AAAA,EAEA,OAAO,cAA8B;AACpC,UAAM,MAAM,KAAK,cAAc;AAE/B,UAAM,SAAS,IAAI,IAAI,CAAC,OAAO;AAC9B,YAAM,cAAc,KAAK,SAAS,EAAE;AACpC,YAAM,OAAO,KAAK,MAAM,aAAa,KAAK,aAAa,WAAW,GAAG,OAAO,CAAC;AAE7E,YAAM,QAAsB;AAAA,QAC3B,IAAI,KAAK;AAAA,QACT,MAAM,KAAK;AAAA,QACX,OAAO,KAAK;AAAA,QACZ,aAAa,KAAK;AAAA,MACnB;AAEA,aAAO;AAAA,IACR,CAAC;AAED,WAAO;AAAA,EACR;AAAA,EAEA,OAAO,kBAA4B;AAClC,UAAM,MAAM,KAAK,cAAc;AAC/B,UAAM,cAAc,oBAAI,IAAY;AAEpC,eAAW,MAAM,KAAK;AACrB,YAAM,WAAW,KAAK,KAAK,EAAE;AAE7B,iBAAW,aAAa,SAAS,YAAY;AAC5C,mBAAW,YAAY,UAAU,WAAW;AAC3C,sBAAY,IAAI,QAAQ;AAAA,QACzB;AAAA,MACD;AAAA,IACD;AAEA,UAAM,SAAS,MAAM,KAAK,WAAW,EAAE,KAAK;AAE5C,WAAO;AAAA,EACR;AAAA,EAEA,OAAO,kBAA4B;AAClC,UAAM,MAAM,KAAK,cAAc;AAC/B,UAAM,UAAU,oBAAI,IAAY;AAEhC,eAAW,MAAM,KAAK;AACrB,YAAM,WAAW,KAAK,KAAK,EAAE;AAE7B,iBAAW,WAAW,SAAS,WAAW;AACzC,YAAI,QAAQ,MAAM;AACjB,kBAAQ,IAAI,QAAQ,IAAI;AAAA,QACzB;AAAA,MACD;AAAA,IACD;AAEA,UAAM,SAAS,MAAM,KAAK,OAAO,EAAE,KAAK;AAExC,WAAO;AAAA,EACR;AAAA,EAEA,OAAO,gBAA0B;AAChC,UAAM,UAAU,YAAY,SAAS,EAAE,eAAe,KAAK,CAAC;AAE5D,UAAM,MAAM,QACV,OAAO,CAAC,UAAU;AAClB,UAAI,CAAC,MAAM,YAAY,GAAG;AACzB,eAAO;AAAA,MACR;AAEA,UAAI;AACH,qBAAa,KAAK,SAAS,MAAM,MAAM,WAAW,GAAG,OAAO;AAE5D,eAAO;AAAA,MACR,QAAQ;AACP,eAAO;AAAA,MACR;AAAA,IACD,CAAC,EACA,IAAI,CAAC,UAAU,MAAM,IAAI;AAE3B,WAAO;AAAA,EACR;AACD;;;AClIO,SAAS,UAAU,MAA6B;AACtD,MAAI,SAAS,QAAW;AACvB,WAAO,KAAK;AAAA,EACb;AAEA,MAAI,IAAI;AAER,SAAO,SAAS,MAAc;AAC7B,SAAK;AACL,QAAI,IAAI;AACR,QAAI,KAAK,KAAK,IAAK,MAAM,IAAK,IAAI,CAAC;AACnC,SAAK,IAAI,KAAK,KAAK,IAAK,MAAM,GAAI,IAAI,EAAE;AAExC,aAAS,IAAK,MAAM,QAAS,KAAK;AAAA,EACnC;AACD;AAEO,SAAS,WAAc,OAAY,KAAsB;AAC/D,QAAM,QAAQ,KAAK,MAAM,IAAI,IAAI,MAAM,MAAM;AAE7C,SAAO,MAAM,KAAK;AACnB;;;ACrBO,IAAM,iBAAN,MAAqB;AAAA,EAC3B,OAAO,SAAS,WAAmB,UAAkB,SAAsB,KAA2B;AACrG,UAAM,SAAS,WAAW,QAAQ,cAAc,GAAG;AACnD,UAAM,QAAQ,UAAU,YAAY;AACpC,UAAM,OAAO,SAAS,YAAY;AAElC,WAAO,GAAG,KAAK,IAAI,IAAI,IAAI,MAAM;AAAA,EAClC;AAAA,EAEA,OAAO,qBAAqB,UAAkB,SAAsB,KAA2B;AAC9F,UAAM,SAAS,WAAW,QAAQ,cAAc,GAAG;AAEnD,WAAO,GAAG,QAAQ,IAAI,MAAM;AAAA,EAC7B;AACD;;;ACdO,IAAM,iBAAN,MAAqB;AAAA,EAC3B,OAAO,SAAS,SAAsB,KAA2B;AAChE,QAAI,QAAQ,mBAAmB,QAAQ,gBAAgB,SAAS,KAAK,IAAI,IAAI,KAAK;AACjF,aAAO,WAAW,QAAQ,iBAAiB,GAAG;AAAA,IAC/C;AAEA,UAAM,SAAS,WAAW,QAAQ,eAAe,GAAG;AACpD,UAAM,SAAS,KAAK,aAAa,GAAG,GAAG;AAEvC,WAAO,GAAG,MAAM,IAAI,OAAO,MAAM,GAAG,CAAC,CAAC,IAAI,OAAO,MAAM,CAAC,CAAC;AAAA,EAC1D;AAAA,EAEA,OAAe,aAAa,OAAe,KAA2B;AACrE,QAAI,SAAS;AAEb,aAAS,IAAI,GAAG,IAAI,OAAO,KAAK;AAC/B,gBAAU,KAAK,MAAM,IAAI,IAAI,EAAE,EAAE,SAAS;AAAA,IAC3C;AAEA,WAAO;AAAA,EACR;AACD;;;ACrBO,IAAM,oBAAN,MAAwB;AAAA,EAC9B,OAAO,SAAS,SAAsB,KAA2B;AAChE,QAAI,QAAQ,sBAAsB,QAAQ,mBAAmB,SAAS,KAAK,IAAI,IAAI,KAAK;AACvF,aAAO,WAAW,QAAQ,oBAAoB,GAAG;AAAA,IAClD;AAEA,WAAO,KAAK,eAAe,IAAI,GAAG;AAAA,EACnC;AAAA,EAEA,OAAe,eAAe,QAAgB,KAA2B;AACxE,UAAM,QAAQ;AACd,QAAI,SAAS;AAEb,aAAS,IAAI,GAAG,IAAI,QAAQ,KAAK;AAChC,gBAAU,MAAM,KAAK,MAAM,IAAI,IAAI,MAAM,MAAM,CAAC;AAAA,IACjD;AAEA,WAAO;AAAA,EACR;AACD;;;ACnBO,IAAM,gBAAN,MAAoB;AAAA,EAC1B,OAAO,MAAM,UAAwB,MAAuB;AAC3D,UAAM,MAAM,UAAU,IAAI;AAE1B,UAAM,YAAY,WAAW,SAAS,YAAY,GAAG;AAErD,WAAO,KAAK,mBAAmB,WAAW,UAAU,GAAG;AAAA,EACxD;AAAA,EAEA,OAAO,mBAAmB,aAAqB,UAAwB,MAAuB;AAC7F,UAAM,YAAY,SAAS,WAAW,KAAK,CAAC,MAAM,EAAE,OAAO,WAAW;AAEtE,QAAI,CAAC,WAAW;AACf,YAAM,IAAI,MAAM,cAAc,WAAW,4BAA4B,SAAS,EAAE,GAAG;AAAA,IACpF;AAEA,UAAM,MAAM,UAAU,IAAI;AAE1B,WAAO,KAAK,mBAAmB,WAAW,UAAU,GAAG;AAAA,EACxD;AAAA,EAEA,OAAO,eAAe,aAAqB,UAAgC;AAC1E,UAAM,YAAY,SAAS,WAAW,KAAK,CAAC,MAAM,EAAE,OAAO,WAAW;AAEtE,QAAI,CAAC,WAAW;AACf,YAAM,IAAI,MAAM,cAAc,WAAW,4BAA4B,SAAS,EAAE,GAAG;AAAA,IACpF;AAEA,UAAM,UAAU,UAAU,WAAW,SAAS,UAAU,CAAC;AACzD,UAAM,WAAW,UAAU,UAAU,CAAC;AACtC,UAAM,UAAU,UAAU,cAAc,EAAE,GAAG,SAAS,SAAS,cAAc,CAAC,UAAU,WAAW,EAAE,IAAI,SAAS;AAClH,UAAM,MAAM,UAAU,CAAC;AACvB,UAAM,QAAQ,eAAe,qBAAqB,UAAU,SAAS,GAAG;AACxE,UAAM,QAAQ,eAAe,SAAS,SAAS,SAAS,GAAG;AAC3D,UAAM,WAAW,kBAAkB,SAAS,SAAS,SAAS,GAAG;AACjE,UAAM,QAAQ,UAAU,OAAO,CAAC;AAEhC,UAAM,SAAiB;AAAA,MACtB,IAAI,OAAO,WAAW;AAAA,MACtB,aAAa,UAAU;AAAA,MACvB,WAAW,UAAU;AAAA,MACrB,UAAU,UAAU;AAAA,MACpB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA,SAAS;AAAA,QACR,QAAQ,QAAQ;AAAA,QAChB,MAAM,QAAQ;AAAA,QACd,OAAO,QAAQ,SAAS;AAAA,QACxB,KAAK,QAAQ,OAAO;AAAA,QACpB,SAAS,QAAQ;AAAA,MAClB;AAAA,MACA,YAAY,UAAU;AAAA,MACtB,WAAW,UAAU;AAAA,MACrB;AAAA,MACA,UAAU,SAAS;AAAA,MACnB,cAAc,SAAS;AAAA,MACvB,GAAI,UAAU,SAAS,EAAE,QAAQ,UAAU,OAAO,IAAI,CAAC;AAAA,MACvD,GAAI,UAAU,QAAQ,EAAE,OAAO,UAAU,MAAM,IAAI,CAAC;AAAA,IACrD;AAEA,WAAO;AAAA,EACR;AAAA,EAEA,OAAe,mBAAmB,WAA0B,UAAwB,KAA2B;AAC9G,UAAM,UAAU,UAAU,WAAW,WAAW,SAAS,WAAW,GAAG;AAEvE,UAAM,WAAW,WAAW,UAAU,WAAW,GAAG;AAEpD,UAAM,UAAU,UAAU,cAAc,EAAE,GAAG,SAAS,SAAS,cAAc,CAAC,UAAU,WAAW,EAAE,IAAI,SAAS;AAElH,UAAM,QAAQ,eAAe,qBAAqB,UAAU,SAAS,GAAG;AACxE,UAAM,QAAQ,eAAe,SAAS,SAAS,SAAS,GAAG;AAC3D,UAAM,WAAW,kBAAkB,SAAS,SAAS,SAAS,GAAG;AACjE,UAAM,QAAQ,WAAW,UAAU,QAAQ,GAAG;AAE9C,UAAM,SAAiB;AAAA,MACtB,IAAI,OAAO,WAAW;AAAA,MACtB,aAAa,UAAU;AAAA,MACvB,WAAW,UAAU;AAAA,MACrB,UAAU,UAAU;AAAA,MACpB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA,SAAS;AAAA,QACR,QAAQ,QAAQ;AAAA,QAChB,MAAM,QAAQ;AAAA,QACd,OAAO,QAAQ,SAAS;AAAA,QACxB,KAAK,QAAQ,OAAO;AAAA,QACpB,SAAS,QAAQ;AAAA,MAClB;AAAA,MACA,YAAY,UAAU;AAAA,MACtB,WAAW,UAAU;AAAA,MACrB;AAAA,MACA,UAAU,SAAS;AAAA,MACnB,cAAc,SAAS;AAAA,MACvB,GAAI,UAAU,SAAS,EAAE,QAAQ,UAAU,OAAO,IAAI,CAAC;AAAA,MACvD,GAAI,UAAU,QAAQ,EAAE,OAAO,UAAU,MAAM,IAAI,CAAC;AAAA,IACrD;AAEA,WAAO;AAAA,EACR;AACD;;;ACtGO,IAAM,iBAAN,MAAqB;AAAA,EAC3B,OAAO,oBAAoB,UAAwB,SAAkD;AACpG,WAAO,cAAc,MAAM,UAAU,QAAQ,IAAI;AAAA,EAClD;AAAA,EAEA,OAAO,YAAY,SAAgC;AAClD,UAAM,WAAW,eAAe,KAAK,QAAQ,QAAQ;AAErD,WAAO,cAAc,MAAM,UAAU,QAAQ,IAAI;AAAA,EAClD;AAAA,EAEA,OAAO,WAAW,SAAiC;AAClD,UAAM,UAAoB,CAAC;AAE3B,aAAS,IAAI,GAAG,IAAI,QAAQ,MAAM,KAAK;AACtC,YAAM,OAAO,QAAQ,SAAS,SAAY,QAAQ,OAAO,IAAI;AAE7D,cAAQ,KAAK,KAAK,YAAY,EAAE,UAAU,QAAQ,UAAU,KAAK,CAAC,CAAC;AAAA,IACpE;AAEA,WAAO;AAAA,EACR;AAAA,EAEA,OAAO,gBAAgC;AACtC,UAAM,MAAM,eAAe,cAAc;AAEzC,UAAM,SAAyB,IAAI,IAAI,CAAC,OAAO;AAC9C,YAAM,OAAO,eAAe,KAAK,EAAE;AAEnC,YAAM,OAAqB;AAAA,QAC1B;AAAA,QACA,MAAM,KAAK;AAAA,QACX,OAAO,KAAK;AAAA,QACZ,aAAa,KAAK;AAAA,MACnB;AAEA,aAAO;AAAA,IACR,CAAC;AAED,WAAO;AAAA,EACR;AACD;;;ACzCO,IAAM,gBAAN,MAAoB;AAAA,EAClB;AAAA,EACA;AAAA,EAER,YAAYC,YAA2B;AACtC,SAAK,YAAYA;AACjB,SAAK,iBAAiB,KAAK,WAAWA,UAAS;AAAA,EAChD;AAAA,EAEA,IAAI,QAA0B;AAC7B,WAAO,KAAK;AAAA,EACb;AAAA,EAEA,aAAa,aAAkC;AAC9C,UAAM,UACL,eAAe,YAAY,SAAS,IACjC,KAAK,eAAe,OAAO,CAAC,MAAM,YAAY,SAAS,EAAE,UAAU,CAAC,IACpE,KAAK;AAET,UAAM,cAAc,oBAAI,IAAY;AAEpC,eAAW,SAAS,SAAS;AAC5B,iBAAW,YAAY,MAAM,WAAW;AACvC,oBAAY,IAAI,QAAQ;AAAA,MACzB;AAAA,IACD;AAEA,UAAM,SAAS,MAAM,KAAK,WAAW,EAAE,KAAK;AAE5C,WAAO;AAAA,EACR;AAAA,EAEA,eAA+B;AAC9B,UAAM,SAAS,KAAK,UAAU,IAAI,CAAC,MAAM;AACxC,YAAM,OAAqB;AAAA,QAC1B,IAAI,EAAE;AAAA,QACN,MAAM,EAAE;AAAA,QACR,OAAO,EAAE;AAAA,QACT,aAAa,EAAE;AAAA,MAChB;AAEA,aAAO;AAAA,IACR,CAAC;AAED,WAAO;AAAA,EACR;AAAA,EAEA,eAAe,OAAyC;AACvD,QAAI,UAAU,KAAK;AAEnB,QAAI,MAAM,aAAa,MAAM,UAAU,SAAS,GAAG;AAClD,gBAAU,QAAQ,OAAO,CAAC,MAAM,MAAM,UAAW,SAAS,EAAE,UAAU,CAAC;AAAA,IACxE;AAEA,QAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,EAAE,SAAS,GAAG;AAC/C,YAAM,SAAS,MAAM,KAAK,YAAY;AAEtC,gBAAU,QAAQ,OAAO,CAAC,MAAM;AAC/B,cAAM,WAAW,GAAG,EAAE,SAAS,IAAI,EAAE,QAAQ,GAAG,YAAY;AAE5D,eAAO,SAAS,SAAS,MAAM;AAAA,MAChC,CAAC;AAAA,IACF;AAEA,QAAI,MAAM,aAAa,MAAM,UAAU,SAAS,GAAG;AAClD,YAAM,OAAO,MAAM,iBAAiB;AAEpC,UAAI,SAAS,OAAO;AACnB,kBAAU,QAAQ,OAAO,CAAC,MAAM,MAAM,UAAW,MAAM,CAAC,aAAa,EAAE,UAAU,SAAS,QAAQ,CAAC,CAAC;AAAA,MACrG,OAAO;AACN,kBAAU,QAAQ,OAAO,CAAC,MAAM,MAAM,UAAW,KAAK,CAAC,aAAa,EAAE,UAAU,SAAS,QAAQ,CAAC,CAAC;AAAA,MACpG;AAAA,IACD;AAEA,WAAO;AAAA,EACR;AAAA,EAEA,oBAAoB,aAAqB,MAAuB;AAC/D,UAAM,QAAQ,KAAK,eAAe,KAAK,CAAC,MAAM,EAAE,gBAAgB,WAAW;AAE3E,QAAI,CAAC,OAAO;AACX,YAAM,IAAI,MAAM,cAAc,WAAW,sBAAsB;AAAA,IAChE;AAEA,UAAM,WAAW,KAAK,UAAU,KAAK,CAAC,MAAM,EAAE,OAAO,MAAM,UAAU;AAErE,QAAI,CAAC,UAAU;AACd,YAAM,IAAI,MAAM,aAAa,MAAM,UAAU,sBAAsB;AAAA,IACpE;AAEA,WAAO,cAAc,mBAAmB,aAAa,UAAU,IAAI;AAAA,EACpE;AAAA,EAEA,iBAAiB,OAAuB,OAAe,MAAyB;AAC/E,UAAM,WAAW,KAAK,eAAe,KAAK;AAE1C,QAAI,SAAS,WAAW,GAAG;AAC1B,aAAO,CAAC;AAAA,IACT;AAEA,UAAM,MAAM,UAAU,IAAI;AAC1B,UAAM,cAAc,KAAK,IAAI,OAAO,SAAS,MAAM;AAEnD,UAAM,WAAW,CAAC,GAAG,QAAQ,EAAE,KAAK,MAAM,IAAI,IAAI,GAAG;AACrD,UAAM,WAAW,SAAS,MAAM,GAAG,WAAW;AAC9C,UAAM,WAAqB,CAAC;AAE5B,aAAS,IAAI,GAAG,IAAI,SAAS,QAAQ,KAAK;AACzC,YAAM,QAAQ,SAAS,CAAC;AACxB,YAAM,WAAW,KAAK,UAAU,KAAK,CAAC,MAAM,EAAE,OAAO,MAAM,UAAU;AACrE,YAAM,aAAa,SAAS,SAAY,OAAO,IAAI;AAEnD,eAAS,KAAK,cAAc,mBAAmB,MAAM,aAAa,UAAU,UAAU,CAAC;AAAA,IACxF;AAEA,WAAO;AAAA,EACR;AAAA,EAEQ,WAAWA,YAA6C;AAC/D,UAAM,QAA0B,CAAC;AAEjC,eAAW,YAAYA,YAAW;AACjC,iBAAW,aAAa,SAAS,YAAY;AAC5C,cAAM,QAAwB;AAAA,UAC7B,aAAa,UAAU;AAAA,UACvB,YAAY,SAAS;AAAA,UACrB,cAAc,SAAS;AAAA,UACvB,WAAW,UAAU;AAAA,UACrB,UAAU,UAAU;AAAA,UACpB,WAAW,UAAU;AAAA,UACrB,YAAY,UAAU;AAAA,QACvB;AAEA,cAAM,KAAK,KAAK;AAAA,MACjB;AAAA,IACD;AAEA,WAAO;AAAA,EACR;AACD;;;ACjIO,SAAS,YAA4B;AAC3C,SAAO,eAAe,cAAc;AACrC;;;ATXA,IAAM,UAAU,IAAI,QAAQ;AAE5B,QAAQ,KAAK,UAAU,EAAE,YAAY,iCAAiC,EAAE,QAAQ,OAAO;AAEvF,QACE,QAAQ,QAAQ,EAChB,YAAY,2BAA2B,EACvC,OAAO,uBAAuB,0DAA0D,EACxF,OAAO,kBAAkB,mCAAmC,EAC5D,OAAO,sBAAsB,gCAAgC,QAAQ,EACrE,OAAO,sBAAsB,2DAA2D,EACxF,OAAO,2BAA2B,sCAAsC,IAAI,EAC5E,OAAO,kBAAkB,0BAA0B,EACnD;AAAA,EACA,CAAC,YAA4H;AAC5H,UAAM,OAAO,QAAQ,OAAO,SAAS,QAAQ,MAAM,EAAE,IAAI;AACzD,UAAM,cAAc,QAAQ,WAAW,CAAC,QAAQ,QAAQ,IAAI,eAAe,cAAc;AACzF,UAAM,gBAAgB,YAAY,IAAI,CAAC,OAAO,eAAe,KAAK,EAAE,CAAC;AACrE,UAAM,gBAAgB,IAAI,cAAc,aAAa;AAErD,UAAM,QAAwB;AAAA,MAC7B,WAAW,QAAQ,WAAW,CAAC,QAAQ,QAAQ,IAAI,CAAC;AAAA,MACpD,WAAW,QAAQ,YAAY,QAAQ,UAAU,MAAM,GAAG,EAAE,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,IAAI,CAAC;AAAA,MACpF,eAAe,QAAQ,kBAAkB,QAAQ,QAAQ;AAAA,MACzD,MAAM,QAAQ;AAAA,IACf;AAEA,UAAM,SAAS,cAAc,iBAAiB,OAAO,GAAG,IAAI,EAAE,CAAC;AAE/D,QAAI,CAAC,QAAQ;AACZ,cAAQ,MAAM,wCAAwC;AACtD,cAAQ,KAAK,CAAC;AAAA,IACf;AAEA,QAAI,QAAQ,WAAW,QAAQ;AAC9B,cAAQ,IAAI,KAAK,UAAU,QAAQ,MAAM,CAAC,CAAC;AAE3C;AAAA,IACD;AAEA,YAAQ;AAAA,MACP;AAAA,cACU,OAAO,SAAS,IAAI,OAAO,QAAQ,MAAM,OAAO,QAAQ;AAAA,cACxD,OAAO,KAAK;AAAA,cACZ,OAAO,KAAK;AAAA,cACZ,CAAC,OAAO,QAAQ,QAAQ,OAAO,QAAQ,MAAM,OAAO,QAAQ,OAAO,EAAE,OAAO,OAAO,EAAE,KAAK,IAAI,CAAC;AAAA,cAC/F,OAAO,UAAU;AAAA,cACjB,OAAO,UAAU,KAAK,IAAI,CAAC;AAAA,cAC3B,OAAO,KAAK;AAAA,cACZ,OAAO,QAAQ;AAAA,MACvB,KAAK;AAAA,IACR;AAAA,EACD;AACD;AAED,QACE,QAAQ,OAAO,EACf,YAAY,4BAA4B,EACxC,OAAO,uBAAuB,iDAAiD,EAC/E,OAAO,kBAAkB,sBAAsB,GAAG,EAClD,OAAO,kBAAkB,mCAAmC,EAC5D,OAAO,sBAAsB,gCAAgC,QAAQ,EACrE,OAAO,sBAAsB,2DAA2D,EACxF,OAAO,2BAA2B,sCAAsC,IAAI,EAC5E,OAAO,kBAAkB,0BAA0B,EACnD;AAAA,EACA,CAAC,YAQK;AACL,UAAM,OAAO,SAAS,QAAQ,MAAM,EAAE;AACtC,UAAM,OAAO,QAAQ,OAAO,SAAS,QAAQ,MAAM,EAAE,IAAI;AACzD,UAAM,cAAc,QAAQ,WAAW,CAAC,QAAQ,QAAQ,IAAI,eAAe,cAAc;AACzF,UAAM,gBAAgB,YAAY,IAAI,CAAC,OAAO,eAAe,KAAK,EAAE,CAAC;AACrE,UAAM,gBAAgB,IAAI,cAAc,aAAa;AAErD,UAAM,QAAwB;AAAA,MAC7B,WAAW,QAAQ,WAAW,CAAC,QAAQ,QAAQ,IAAI,CAAC;AAAA,MACpD,WAAW,QAAQ,YAAY,QAAQ,UAAU,MAAM,GAAG,EAAE,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,IAAI,CAAC;AAAA,MACpF,eAAe,QAAQ,kBAAkB,QAAQ,QAAQ;AAAA,MACzD,MAAM,QAAQ;AAAA,IACf;AAEA,UAAM,SAAS,cAAc,iBAAiB,OAAO,MAAM,IAAI;AAE/D,QAAI,OAAO,WAAW,GAAG;AACxB,cAAQ,MAAM,wCAAwC;AACtD,cAAQ,KAAK,CAAC;AAAA,IACf;AAEA,QAAI,QAAQ,WAAW,QAAQ;AAC9B,cAAQ,IAAI,KAAK,UAAU,QAAQ,MAAM,CAAC,CAAC;AAE3C;AAAA,IACD;AAEA,eAAW,KAAK,QAAQ;AACvB,cAAQ,IAAI,GAAG,EAAE,SAAS,IAAI,EAAE,QAAQ,KAAK,EAAE,KAAK,YAAO,EAAE,UAAU,EAAE;AAAA,IAC1E;AAAA,EACD;AACD;AAED,QACE,QAAQ,WAAW,EACnB,YAAY,0BAA0B,EACtC,OAAO,sBAAsB,gCAAgC,QAAQ,EACrE,OAAO,CAAC,YAAgC;AACxC,QAAM,OAAO,UAAU;AAEvB,MAAI,QAAQ,WAAW,QAAQ;AAC9B,YAAQ,IAAI,KAAK,UAAU,MAAM,MAAM,CAAC,CAAC;AAEzC;AAAA,EACD;AAEA,aAAW,KAAK,MAAM;AACrB,YAAQ,IAAI,KAAK,EAAE,GAAG,OAAO,EAAE,CAAC,IAAI,EAAE,IAAI,EAAE;AAAA,EAC7C;AACD,CAAC;AAEF,QAAQ,MAAM;","names":["require","universes"]}
|
|
1
|
+
{"version":3,"sources":["../../src/cli/cli.ts","../../src/universes/loader.ts","../../src/generators/random.ts","../../src/generators/email.ts","../../src/engine/person-factory.ts","../../src/engine/persona-builder.ts","../../src/engine/universe-store.ts","../../src/index.ts"],"sourcesContent":["#!/usr/bin/env node\nimport { Command } from 'commander';\n\nimport { universes, UniverseLoader, UniverseStore } from '@/index';\n\nimport type { CharacterQuery } from '@/types';\n\nconst program = new Command();\n\nprogram.name('loredata').description('Pop culture fake data generator').version('0.1.0');\n\nprogram\n\t.command('person')\n\t.description('Generate a single persona')\n\t.option('-u, --universe <id>', 'Universe ID (e.g. breaking-bad). Omit for all universes.')\n\t.option('-s, --seed <n>', 'Seed for deterministic generation')\n\t.option('-f, --format <fmt>', 'Output format: pretty | json', 'pretty')\n\t.option('--interests <list>', 'Comma-separated interests filter (e.g. chemistry,cooking)')\n\t.option('--interests-mode <mode>', 'Match mode: and | or (default: or)', 'or')\n\t.option('--name <query>', 'Search by character name')\n\t.action(\n\t\t(options: { universe?: string; seed?: string; format: string; interests?: string; interestsMode: string; name?: string }) => {\n\t\t\tconst seed = options.seed ? parseInt(options.seed, 10) : undefined;\n\t\t\tconst universeIds = options.universe ? [options.universe] : UniverseLoader.listAvailable();\n\t\t\tconst universesData = universeIds.map((id) => UniverseLoader.load(id));\n\t\t\tconst storeInstance = new UniverseStore(universesData);\n\n\t\t\tconst query: CharacterQuery = {\n\t\t\t\tuniverses: options.universe ? [options.universe] : [],\n\t\t\t\tinterests: options.interests ? options.interests.split(',').map((s) => s.trim()) : [],\n\t\t\t\tinterestsMode: options.interestsMode === 'and' ? 'and' : 'or',\n\t\t\t\tname: options.name\n\t\t\t};\n\n\t\t\tconst result = storeInstance.generatePersonas(query, 1, seed)[0];\n\n\t\t\tif (!result) {\n\t\t\t\tconsole.error('No characters match the given filters.');\n\t\t\t\tprocess.exit(1);\n\t\t\t}\n\n\t\t\tif (options.format === 'json') {\n\t\t\t\tconsole.log(JSON.stringify(result, null, 2));\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconsole.log(\n\t\t\t\t`\nName: ${result.firstName} ${result.lastName} (@${result.username})\nEmail: ${result.email}\nAddress: ${[result.address.street, result.address.city, result.address.country].filter(Boolean).join(', ')}\nProfession: ${result.profession}\nInterests: ${result.interests.join(', ')}\nQuote: ${result.quote}\nUniverse: ${result.universe}\n `.trim()\n\t\t\t);\n\t\t}\n\t);\n\nprogram\n\t.command('group')\n\t.description('Generate multiple personas')\n\t.option('-u, --universe <id>', 'Universe ID (e.g. got). Omit for all universes.')\n\t.option('-n, --size <n>', 'Number of personas', '3')\n\t.option('-s, --seed <n>', 'Seed for deterministic generation')\n\t.option('-f, --format <fmt>', 'Output format: pretty | json', 'pretty')\n\t.option('--interests <list>', 'Comma-separated interests filter (e.g. chemistry,cooking)')\n\t.option('--interests-mode <mode>', 'Match mode: and | or (default: or)', 'or')\n\t.option('--name <query>', 'Search by character name')\n\t.action(\n\t\t(options: {\n\t\t\tuniverse?: string;\n\t\t\tsize: string;\n\t\t\tseed?: string;\n\t\t\tformat: string;\n\t\t\tinterests?: string;\n\t\t\tinterestsMode: string;\n\t\t\tname?: string;\n\t\t}) => {\n\t\t\tconst size = parseInt(options.size, 10);\n\t\t\tconst seed = options.seed ? parseInt(options.seed, 10) : undefined;\n\t\t\tconst universeIds = options.universe ? [options.universe] : UniverseLoader.listAvailable();\n\t\t\tconst universesData = universeIds.map((id) => UniverseLoader.load(id));\n\t\t\tconst storeInstance = new UniverseStore(universesData);\n\n\t\t\tconst query: CharacterQuery = {\n\t\t\t\tuniverses: options.universe ? [options.universe] : [],\n\t\t\t\tinterests: options.interests ? options.interests.split(',').map((s) => s.trim()) : [],\n\t\t\t\tinterestsMode: options.interestsMode === 'and' ? 'and' : 'or',\n\t\t\t\tname: options.name\n\t\t\t};\n\n\t\t\tconst result = storeInstance.generatePersonas(query, size, seed);\n\n\t\t\tif (result.length === 0) {\n\t\t\t\tconsole.error('No characters match the given filters.');\n\t\t\t\tprocess.exit(1);\n\t\t\t}\n\n\t\t\tif (options.format === 'json') {\n\t\t\t\tconsole.log(JSON.stringify(result, null, 2));\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor (const p of result) {\n\t\t\t\tconsole.log(`${p.firstName} ${p.lastName} <${p.email}> — ${p.profession}`);\n\t\t\t}\n\t\t}\n\t);\n\nprogram\n\t.command('universes')\n\t.description('List available universes')\n\t.option('-f, --format <fmt>', 'Output format: pretty | json', 'pretty')\n\t.action((options: { format: string }) => {\n\t\tconst list = universes();\n\n\t\tif (options.format === 'json') {\n\t\t\tconsole.log(JSON.stringify(list, null, 2));\n\n\t\t\treturn;\n\t\t}\n\n\t\tfor (const u of list) {\n\t\t\tconsole.log(` ${u.id.padEnd(20)} ${u.name}`);\n\t\t}\n\t});\n\nprogram.parse();\n","import { readFileSync, readdirSync } from 'fs';\nimport { createRequire } from 'module';\nimport { dirname, join } from 'path';\n\nimport type { UniverseData, UniverseMeta, CharacterData, AddressData, CharacterEntry, LocationEntry } from '@/types';\n\nconst require = createRequire(import.meta.url);\nconst packageRoot = dirname(require.resolve('loredata/package.json'));\nconst dataDir = join(packageRoot, 'data');\n\nexport class UniverseLoader {\n\tprivate static cache = new Map<string, UniverseData>();\n\n\tstatic register(data: UniverseData): void {\n\t\tthis.cache.set(data.id, data);\n\t}\n\n\tstatic load(universeId: string): UniverseData {\n\t\tif (process.env.NODE_ENV !== 'development') {\n\t\t\tconst cached = this.cache.get(universeId);\n\n\t\t\tif (cached) {\n\t\t\t\treturn cached;\n\t\t\t}\n\t\t}\n\n\t\tconst universeDir = join(dataDir, universeId);\n\t\tconst meta = JSON.parse(readFileSync(join(universeDir, 'meta.json'), 'utf-8')) as UniverseMeta;\n\n\t\tconst characters = JSON.parse(readFileSync(join(universeDir, 'characters.json'), 'utf-8')) as CharacterData[];\n\n\t\tconst addresses = JSON.parse(readFileSync(join(universeDir, 'addresses.json'), 'utf-8')) as AddressData[];\n\n\t\tconst universeData: UniverseData = {\n\t\t\tid: meta.id,\n\t\t\tname: meta.name,\n\t\t\tgenre: meta.genre,\n\t\t\tdescription: meta.description,\n\t\t\tcharacters,\n\t\t\taddresses,\n\t\t\t...(meta.tmdbId !== undefined && { tmdbId: meta.tmdbId }),\n\t\t\t...(meta.mediaType !== undefined && { mediaType: meta.mediaType }),\n\t\t\t...(meta.rating !== undefined && { rating: meta.rating }),\n\t\t\t...(meta.year !== undefined && { year: meta.year }),\n\t\t\t...(meta.networks !== undefined && { networks: meta.networks }),\n\t\t\t...(meta.posterPath !== undefined && { posterPath: meta.posterPath }),\n\t\t\t...(meta.backdropPath !== undefined && { backdropPath: meta.backdropPath })\n\t\t};\n\n\t\tthis.cache.set(universeId, universeData);\n\n\t\treturn universeData;\n\t}\n\n\tstatic getManifest(): UniverseMeta[] {\n\t\tconst ids = this.listAvailable();\n\n\t\tconst result = ids.map((id) => {\n\t\t\tconst universeDir = join(dataDir, id);\n\n\t\t\treturn JSON.parse(readFileSync(join(universeDir, 'meta.json'), 'utf-8')) as UniverseMeta;\n\t\t});\n\n\t\treturn result;\n\t}\n\n\tstatic getAllInterests(): string[] {\n\t\tconst ids = this.listAvailable();\n\t\tconst interestSet = new Set<string>();\n\n\t\tfor (const id of ids) {\n\t\t\tconst universe = this.load(id);\n\n\t\t\tfor (const character of universe.characters) {\n\t\t\t\tfor (const interest of character.interests) {\n\t\t\t\t\tinterestSet.add(interest);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst result = Array.from(interestSet).sort();\n\n\t\treturn result;\n\t}\n\n\tstatic getAllLocations(): LocationEntry[] {\n\t\tconst ids = this.listAvailable();\n\t\tconst seen = new Map<string, LocationEntry>();\n\n\t\tfor (const id of ids) {\n\t\t\tconst universe = this.load(id);\n\n\t\t\tfor (const address of universe.addresses) {\n\t\t\t\tif (address.city && !seen.has(`city:${address.city}`)) {\n\t\t\t\t\tseen.set(`city:${address.city}`, { name: address.city, type: 'city' });\n\t\t\t\t}\n\n\t\t\t\tif (address.state && !seen.has(`state:${address.state}`)) {\n\t\t\t\t\tseen.set(`state:${address.state}`, { name: address.state, type: 'state' });\n\t\t\t\t}\n\n\t\t\t\tif (address.country && !seen.has(`country:${address.country}`)) {\n\t\t\t\t\tseen.set(`country:${address.country}`, { name: address.country, type: 'country' });\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn Array.from(seen.values()).sort((a, b) => a.name.localeCompare(b.name));\n\t}\n\n\tstatic buildCharacterIndex(): CharacterEntry[] {\n\t\tconst ids = this.listAvailable();\n\t\tconst entries: CharacterEntry[] = [];\n\n\t\tfor (const id of ids) {\n\t\t\tconst universe = this.load(id);\n\n\t\t\tfor (const character of universe.characters) {\n\t\t\t\tentries.push({\n\t\t\t\t\tcharacterId: character.id,\n\t\t\t\t\tuniverseId: universe.id,\n\t\t\t\t\tuniverseName: universe.name,\n\t\t\t\t\tfirstName: character.firstName ?? '',\n\t\t\t\t\tlastName: character.lastName ?? '',\n\t\t\t\t\tinterests: character.interests,\n\t\t\t\t\tprofession: character.profession\n\t\t\t\t});\n\t\t\t}\n\t\t}\n\n\t\treturn entries;\n\t}\n\n\tstatic listAvailable(): string[] {\n\t\tconst entries = readdirSync(dataDir, { withFileTypes: true });\n\n\t\tconst ids = entries\n\t\t\t.filter((entry) => {\n\t\t\t\tif (!entry.isDirectory()) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\ttry {\n\t\t\t\t\treadFileSync(join(dataDir, entry.name, 'meta.json'), 'utf-8');\n\n\t\t\t\t\treturn true;\n\t\t\t\t} catch {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\t\t\t})\n\t\t\t.map((entry) => entry.name);\n\n\t\treturn ids;\n\t}\n}\n","/**\n * Seeded pseudo-random number generator (mulberry32).\n * Returns a function that produces values in [0, 1).\n */\nexport function createRng(seed?: number): () => number {\n\tif (seed === undefined) {\n\t\treturn Math.random;\n\t}\n\n\tlet s = seed;\n\n\treturn function rng(): number {\n\t\ts += 0x6d2b79f5;\n\t\tlet z = s;\n\t\tz = Math.imul(z ^ (z >>> 15), z | 1);\n\t\tz ^= z + Math.imul(z ^ (z >>> 7), z | 61);\n\n\t\treturn ((z ^ (z >>> 14)) >>> 0) / 4294967296;\n\t};\n}\n\nexport function pickRandom<T>(array: T[], rng: () => number): T {\n\tconst index = Math.floor(rng() * array.length);\n\n\treturn array[index];\n}\n","import { pickRandom } from './random';\n\nexport class EmailGenerator {\n\tstatic generateFromUsername(username: string, emailDomains: string[], rng: () => number): string {\n\t\tconst domain = pickRandom(emailDomains, rng);\n\n\t\treturn `${username}@${domain}`;\n\t}\n}\n","import { createRng, pickRandom, EmailGenerator } from '@/generators';\n\nimport type { Person, UniverseData, CharacterData } from '@/types';\n\nexport class PersonFactory {\n\tstatic build(universe: UniverseData, seed?: number): Person {\n\t\tconst rng = createRng(seed);\n\n\t\tconst character = pickRandom(universe.characters, rng);\n\n\t\treturn this.buildFromCharacter(character, universe, rng);\n\t}\n\n\tstatic buildByCharacterId(characterId: string, universe: UniverseData, seed?: number): Person {\n\t\tconst character = universe.characters.find((c) => c.id === characterId);\n\n\t\tif (!character) {\n\t\t\tthrow new Error(`Character \"${characterId}\" not found in universe \"${universe.id}\"`);\n\t\t}\n\n\t\tconst rng = createRng(seed);\n\n\t\treturn this.buildFromCharacter(character, universe, rng);\n\t}\n\n\tstatic buildCanonical(characterId: string, universe: UniverseData): Person {\n\t\tconst character = universe.characters.find((c) => c.id === characterId);\n\n\t\tif (!character) {\n\t\t\tthrow new Error(`Character \"${characterId}\" not found in universe \"${universe.id}\"`);\n\t\t}\n\n\t\tconst address = character.address ?? universe.addresses[0];\n\t\tconst username = character.usernames[0];\n\t\tconst rng = createRng(0);\n\t\tconst email = EmailGenerator.generateFromUsername(username, character.emailDomains, rng);\n\t\tconst quote = character.quotes[0];\n\n\t\tconst person: Person = {\n\t\t\tid: crypto.randomUUID(),\n\t\t\tcharacterId: character.id,\n\t\t\tfirstName: character.firstName,\n\t\t\tlastName: character.lastName,\n\t\t\tusername,\n\t\t\temail,\n\t\t\taddress: {\n\t\t\t\tstreet: address.street,\n\t\t\t\tcity: address.city,\n\t\t\t\tstate: address.state ?? '',\n\t\t\t\tzip: address.zip ?? '',\n\t\t\t\tcountry: address.country\n\t\t\t},\n\t\t\tprofession: character.profession,\n\t\t\tinterests: character.interests,\n\t\t\tquote,\n\t\t\tuniverse: universe.id,\n\t\t\tuniverseName: universe.name,\n\t\t\t...(character.symbol ? { symbol: character.symbol } : {}),\n\t\t\t...(character.color ? { color: character.color } : {})\n\t\t};\n\n\t\treturn person;\n\t}\n\n\tprivate static buildFromCharacter(character: CharacterData, universe: UniverseData, rng: () => number): Person {\n\t\tconst address = character.address ?? pickRandom(universe.addresses, rng);\n\t\tconst username = pickRandom(character.usernames, rng);\n\t\tconst email = EmailGenerator.generateFromUsername(username, character.emailDomains, rng);\n\t\tconst quote = pickRandom(character.quotes, rng);\n\n\t\tconst person: Person = {\n\t\t\tid: crypto.randomUUID(),\n\t\t\tcharacterId: character.id,\n\t\t\tfirstName: character.firstName,\n\t\t\tlastName: character.lastName,\n\t\t\tusername,\n\t\t\temail,\n\t\t\taddress: {\n\t\t\t\tstreet: address.street,\n\t\t\t\tcity: address.city,\n\t\t\t\tstate: address.state ?? '',\n\t\t\t\tzip: address.zip ?? '',\n\t\t\t\tcountry: address.country\n\t\t\t},\n\t\t\tprofession: character.profession,\n\t\t\tinterests: character.interests,\n\t\t\tquote,\n\t\t\tuniverse: universe.id,\n\t\t\tuniverseName: universe.name,\n\t\t\t...(character.symbol ? { symbol: character.symbol } : {}),\n\t\t\t...(character.color ? { color: character.color } : {})\n\t\t};\n\n\t\treturn person;\n\t}\n}\n","import { UniverseLoader } from '@/universes';\n\nimport { PersonFactory } from './person-factory';\n\nimport type { Person, PersonOptions, GroupOptions, UniverseMeta, UniverseData } from '@/types';\n\nexport class PersonaBuilder {\n\tstatic buildPersonFromData(universe: UniverseData, options: Omit<PersonOptions, 'universe'>): Person {\n\t\treturn PersonFactory.build(universe, options.seed);\n\t}\n\n\tstatic buildPerson(options: PersonOptions): Person {\n\t\tconst universe = UniverseLoader.load(options.universe);\n\n\t\treturn PersonFactory.build(universe, options.seed);\n\t}\n\n\tstatic buildGroup(options: GroupOptions): Person[] {\n\t\tconst persons: Person[] = [];\n\n\t\tfor (let i = 0; i < options.size; i++) {\n\t\t\tconst seed = options.seed !== undefined ? options.seed + i : undefined;\n\n\t\t\tpersons.push(this.buildPerson({ universe: options.universe, seed }));\n\t\t}\n\n\t\treturn persons;\n\t}\n\n\tstatic listUniverses(): UniverseMeta[] {\n\t\tconst ids = UniverseLoader.listAvailable();\n\n\t\tconst result: UniverseMeta[] = ids.map((id) => {\n\t\t\tconst data = UniverseLoader.load(id);\n\n\t\t\tconst meta: UniverseMeta = {\n\t\t\t\tid,\n\t\t\t\tname: data.name,\n\t\t\t\tgenre: data.genre,\n\t\t\t\tdescription: data.description\n\t\t\t};\n\n\t\t\treturn meta;\n\t\t});\n\n\t\treturn result;\n\t}\n}\n","import { createRng } from '@/generators';\n\nimport { PersonFactory } from './person-factory';\n\nimport type { UniverseData, UniverseMeta, Person, CharacterEntry, CharacterQuery } from '@/types';\n\nexport class UniverseStore {\n\tprivate universes: UniverseData[];\n\tprivate characterIndex: CharacterEntry[];\n\n\tconstructor(universes: UniverseData[]) {\n\t\tthis.universes = universes;\n\t\tthis.characterIndex = this.buildIndex(universes);\n\t}\n\n\tget index(): CharacterEntry[] {\n\t\treturn this.characterIndex;\n\t}\n\n\tgetInterests(universeIds?: string[]): string[] {\n\t\tconst entries =\n\t\t\tuniverseIds && universeIds.length > 0\n\t\t\t\t? this.characterIndex.filter((e) => universeIds.includes(e.universeId))\n\t\t\t\t: this.characterIndex;\n\n\t\tconst interestSet = new Set<string>();\n\n\t\tfor (const entry of entries) {\n\t\t\tfor (const interest of entry.interests) {\n\t\t\t\tinterestSet.add(interest);\n\t\t\t}\n\t\t}\n\n\t\tconst result = Array.from(interestSet).sort();\n\n\t\treturn result;\n\t}\n\n\tgetUniverses(): UniverseMeta[] {\n\t\tconst result = this.universes.map((u) => {\n\t\t\tconst meta: UniverseMeta = {\n\t\t\t\tid: u.id,\n\t\t\t\tname: u.name,\n\t\t\t\tgenre: u.genre,\n\t\t\t\tdescription: u.description\n\t\t\t};\n\n\t\t\treturn meta;\n\t\t});\n\n\t\treturn result;\n\t}\n\n\tfindCharacters(query: CharacterQuery): CharacterEntry[] {\n\t\tlet entries = this.characterIndex;\n\n\t\tif (query.universes && query.universes.length > 0) {\n\t\t\tentries = entries.filter((e) => query.universes!.includes(e.universeId));\n\t\t}\n\n\t\tif (query.name && query.name.trim().length > 0) {\n\t\t\tconst needle = query.name.toLowerCase();\n\n\t\t\tentries = entries.filter((e) => {\n\t\t\t\tconst fullName = `${e.firstName} ${e.lastName}`.toLowerCase();\n\n\t\t\t\treturn fullName.includes(needle);\n\t\t\t});\n\t\t}\n\n\t\tif (query.interests && query.interests.length > 0) {\n\t\t\tconst mode = query.interestsMode ?? 'or';\n\n\t\t\tif (mode === 'and') {\n\t\t\t\tentries = entries.filter((e) => query.interests!.every((interest) => e.interests.includes(interest)));\n\t\t\t} else {\n\t\t\t\tentries = entries.filter((e) => query.interests!.some((interest) => e.interests.includes(interest)));\n\t\t\t}\n\t\t}\n\n\t\treturn entries;\n\t}\n\n\tpersonByCharacterId(characterId: string, seed?: number): Person {\n\t\tconst entry = this.characterIndex.find((e) => e.characterId === characterId);\n\n\t\tif (!entry) {\n\t\t\tthrow new Error(`Character \"${characterId}\" not found in store`);\n\t\t}\n\n\t\tconst universe = this.universes.find((u) => u.id === entry.universeId);\n\n\t\tif (!universe) {\n\t\t\tthrow new Error(`Universe \"${entry.universeId}\" not found in store`);\n\t\t}\n\n\t\treturn PersonFactory.buildByCharacterId(characterId, universe, seed);\n\t}\n\n\tgeneratePersonas(query: CharacterQuery, count: number, seed?: number): Person[] {\n\t\tconst matching = this.findCharacters(query);\n\n\t\tif (matching.length === 0) {\n\t\t\treturn [];\n\t\t}\n\n\t\tconst rng = createRng(seed);\n\t\tconst actualCount = Math.min(count, matching.length);\n\n\t\tconst shuffled = [...matching].sort(() => rng() - 0.5);\n\t\tconst selected = shuffled.slice(0, actualCount);\n\t\tconst personas: Person[] = [];\n\n\t\tfor (let i = 0; i < selected.length; i++) {\n\t\t\tconst entry = selected[i];\n\t\t\tconst universe = this.universes.find((u) => u.id === entry.universeId)!;\n\t\t\tconst personSeed = seed !== undefined ? seed + i : undefined;\n\n\t\t\tpersonas.push(PersonFactory.buildByCharacterId(entry.characterId, universe, personSeed));\n\t\t}\n\n\t\treturn personas;\n\t}\n\n\tprivate buildIndex(universes: UniverseData[]): CharacterEntry[] {\n\t\tconst index: CharacterEntry[] = [];\n\n\t\tfor (const universe of universes) {\n\t\t\tfor (const character of universe.characters) {\n\t\t\t\tconst entry: CharacterEntry = {\n\t\t\t\t\tcharacterId: character.id,\n\t\t\t\t\tuniverseId: universe.id,\n\t\t\t\t\tuniverseName: universe.name,\n\t\t\t\t\tfirstName: character.firstName,\n\t\t\t\t\tlastName: character.lastName,\n\t\t\t\t\tinterests: character.interests,\n\t\t\t\t\tprofession: character.profession\n\t\t\t\t};\n\n\t\t\t\tindex.push(entry);\n\t\t\t}\n\t\t}\n\n\t\treturn index;\n\t}\n}\n","import { PersonaBuilder } from '@/engine';\n\nimport type { Person, UniverseMeta, PersonOptions, GroupOptions, UniverseData } from '@/types';\n\nexport function person(options: PersonOptions): Person {\n\treturn PersonaBuilder.buildPerson(options);\n}\n\nexport function personFromData(universe: UniverseData, seed?: number): Person {\n\treturn PersonaBuilder.buildPersonFromData(universe, { seed });\n}\n\nexport function group(options: GroupOptions): Person[] {\n\treturn PersonaBuilder.buildGroup(options);\n}\n\nexport function universes(): UniverseMeta[] {\n\treturn PersonaBuilder.listUniverses();\n}\n\nexport { UniverseLoader } from '@/universes';\nexport { UniverseStore, PersonFactory } from '@/engine';\n\nexport type { Person, UniverseMeta, PersonOptions, GroupOptions } from '@/types';\nexport type { Address, UniverseData, CharacterData, AddressData } from '@/types';\nexport type { CharacterEntry, CharacterQuery, LocationEntry, LocationType } from '@/types';\n"],"mappings":"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","names":["require","universes"]}
|
package/dist/index.cjs
CHANGED
|
@@ -54,7 +54,6 @@ var UniverseLoader = class {
|
|
|
54
54
|
const meta = JSON.parse((0, import_fs.readFileSync)((0, import_path.join)(universeDir, "meta.json"), "utf-8"));
|
|
55
55
|
const characters = JSON.parse((0, import_fs.readFileSync)((0, import_path.join)(universeDir, "characters.json"), "utf-8"));
|
|
56
56
|
const addresses = JSON.parse((0, import_fs.readFileSync)((0, import_path.join)(universeDir, "addresses.json"), "utf-8"));
|
|
57
|
-
const domains = JSON.parse((0, import_fs.readFileSync)((0, import_path.join)(universeDir, "domains.json"), "utf-8"));
|
|
58
57
|
const universeData = {
|
|
59
58
|
id: meta.id,
|
|
60
59
|
name: meta.name,
|
|
@@ -62,7 +61,13 @@ var UniverseLoader = class {
|
|
|
62
61
|
description: meta.description,
|
|
63
62
|
characters,
|
|
64
63
|
addresses,
|
|
65
|
-
|
|
64
|
+
...meta.tmdbId !== void 0 && { tmdbId: meta.tmdbId },
|
|
65
|
+
...meta.mediaType !== void 0 && { mediaType: meta.mediaType },
|
|
66
|
+
...meta.rating !== void 0 && { rating: meta.rating },
|
|
67
|
+
...meta.year !== void 0 && { year: meta.year },
|
|
68
|
+
...meta.networks !== void 0 && { networks: meta.networks },
|
|
69
|
+
...meta.posterPath !== void 0 && { posterPath: meta.posterPath },
|
|
70
|
+
...meta.backdropPath !== void 0 && { backdropPath: meta.backdropPath }
|
|
66
71
|
};
|
|
67
72
|
this.cache.set(universeId, universeData);
|
|
68
73
|
return universeData;
|
|
@@ -71,14 +76,7 @@ var UniverseLoader = class {
|
|
|
71
76
|
const ids = this.listAvailable();
|
|
72
77
|
const result = ids.map((id) => {
|
|
73
78
|
const universeDir = (0, import_path.join)(dataDir, id);
|
|
74
|
-
|
|
75
|
-
const entry = {
|
|
76
|
-
id: meta.id,
|
|
77
|
-
name: meta.name,
|
|
78
|
-
genre: meta.genre,
|
|
79
|
-
description: meta.description
|
|
80
|
-
};
|
|
81
|
-
return entry;
|
|
79
|
+
return JSON.parse((0, import_fs.readFileSync)((0, import_path.join)(universeDir, "meta.json"), "utf-8"));
|
|
82
80
|
});
|
|
83
81
|
return result;
|
|
84
82
|
}
|
|
@@ -98,17 +96,41 @@ var UniverseLoader = class {
|
|
|
98
96
|
}
|
|
99
97
|
static getAllLocations() {
|
|
100
98
|
const ids = this.listAvailable();
|
|
101
|
-
const
|
|
99
|
+
const seen = /* @__PURE__ */ new Map();
|
|
102
100
|
for (const id of ids) {
|
|
103
101
|
const universe = this.load(id);
|
|
104
102
|
for (const address of universe.addresses) {
|
|
105
|
-
if (address.city) {
|
|
106
|
-
|
|
103
|
+
if (address.city && !seen.has(`city:${address.city}`)) {
|
|
104
|
+
seen.set(`city:${address.city}`, { name: address.city, type: "city" });
|
|
105
|
+
}
|
|
106
|
+
if (address.state && !seen.has(`state:${address.state}`)) {
|
|
107
|
+
seen.set(`state:${address.state}`, { name: address.state, type: "state" });
|
|
108
|
+
}
|
|
109
|
+
if (address.country && !seen.has(`country:${address.country}`)) {
|
|
110
|
+
seen.set(`country:${address.country}`, { name: address.country, type: "country" });
|
|
107
111
|
}
|
|
108
112
|
}
|
|
109
113
|
}
|
|
110
|
-
|
|
111
|
-
|
|
114
|
+
return Array.from(seen.values()).sort((a, b) => a.name.localeCompare(b.name));
|
|
115
|
+
}
|
|
116
|
+
static buildCharacterIndex() {
|
|
117
|
+
const ids = this.listAvailable();
|
|
118
|
+
const entries = [];
|
|
119
|
+
for (const id of ids) {
|
|
120
|
+
const universe = this.load(id);
|
|
121
|
+
for (const character of universe.characters) {
|
|
122
|
+
entries.push({
|
|
123
|
+
characterId: character.id,
|
|
124
|
+
universeId: universe.id,
|
|
125
|
+
universeName: universe.name,
|
|
126
|
+
firstName: character.firstName ?? "",
|
|
127
|
+
lastName: character.lastName ?? "",
|
|
128
|
+
interests: character.interests,
|
|
129
|
+
profession: character.profession
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
return entries;
|
|
112
134
|
}
|
|
113
135
|
static listAvailable() {
|
|
114
136
|
const entries = (0, import_fs.readdirSync)(dataDir, { withFileTypes: true });
|
|
@@ -148,55 +170,12 @@ function pickRandom(array, rng) {
|
|
|
148
170
|
|
|
149
171
|
// src/generators/email.ts
|
|
150
172
|
var EmailGenerator = class {
|
|
151
|
-
static
|
|
152
|
-
const domain = pickRandom(
|
|
153
|
-
const first = firstName.toLowerCase();
|
|
154
|
-
const last = lastName.toLowerCase();
|
|
155
|
-
return `${first}.${last}@${domain}`;
|
|
156
|
-
}
|
|
157
|
-
static generateFromUsername(username, domains, rng) {
|
|
158
|
-
const domain = pickRandom(domains.emailDomains, rng);
|
|
173
|
+
static generateFromUsername(username, emailDomains, rng) {
|
|
174
|
+
const domain = pickRandom(emailDomains, rng);
|
|
159
175
|
return `${username}@${domain}`;
|
|
160
176
|
}
|
|
161
177
|
};
|
|
162
178
|
|
|
163
|
-
// src/generators/phone.ts
|
|
164
|
-
var PhoneGenerator = class {
|
|
165
|
-
static generate(domains, rng) {
|
|
166
|
-
if (domains.phoneEasterEggs && domains.phoneEasterEggs.length > 0 && rng() < 0.2) {
|
|
167
|
-
return pickRandom(domains.phoneEasterEggs, rng);
|
|
168
|
-
}
|
|
169
|
-
const prefix = pickRandom(domains.phonePrefixes, rng);
|
|
170
|
-
const suffix = this.randomDigits(7, rng);
|
|
171
|
-
return `${prefix}-${suffix.slice(0, 3)}-${suffix.slice(3)}`;
|
|
172
|
-
}
|
|
173
|
-
static randomDigits(count, rng) {
|
|
174
|
-
let result = "";
|
|
175
|
-
for (let i = 0; i < count; i++) {
|
|
176
|
-
result += Math.floor(rng() * 10).toString();
|
|
177
|
-
}
|
|
178
|
-
return result;
|
|
179
|
-
}
|
|
180
|
-
};
|
|
181
|
-
|
|
182
|
-
// src/generators/password.ts
|
|
183
|
-
var PasswordGenerator = class {
|
|
184
|
-
static generate(domains, rng) {
|
|
185
|
-
if (domains.passwordEasterEggs && domains.passwordEasterEggs.length > 0 && rng() < 0.3) {
|
|
186
|
-
return pickRandom(domains.passwordEasterEggs, rng);
|
|
187
|
-
}
|
|
188
|
-
return this.randomPassword(12, rng);
|
|
189
|
-
}
|
|
190
|
-
static randomPassword(length, rng) {
|
|
191
|
-
const chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$";
|
|
192
|
-
let result = "";
|
|
193
|
-
for (let i = 0; i < length; i++) {
|
|
194
|
-
result += chars[Math.floor(rng() * chars.length)];
|
|
195
|
-
}
|
|
196
|
-
return result;
|
|
197
|
-
}
|
|
198
|
-
};
|
|
199
|
-
|
|
200
179
|
// src/engine/person-factory.ts
|
|
201
180
|
var PersonFactory = class {
|
|
202
181
|
static build(universe, seed) {
|
|
@@ -219,11 +198,8 @@ var PersonFactory = class {
|
|
|
219
198
|
}
|
|
220
199
|
const address = character.address ?? universe.addresses[0];
|
|
221
200
|
const username = character.usernames[0];
|
|
222
|
-
const domains = character.emailDomain ? { ...universe.domains, emailDomains: [character.emailDomain] } : universe.domains;
|
|
223
201
|
const rng = createRng(0);
|
|
224
|
-
const email = EmailGenerator.generateFromUsername(username,
|
|
225
|
-
const phone = PhoneGenerator.generate(universe.domains, rng);
|
|
226
|
-
const password = PasswordGenerator.generate(universe.domains, rng);
|
|
202
|
+
const email = EmailGenerator.generateFromUsername(username, character.emailDomains, rng);
|
|
227
203
|
const quote = character.quotes[0];
|
|
228
204
|
const person2 = {
|
|
229
205
|
id: crypto.randomUUID(),
|
|
@@ -232,8 +208,6 @@ var PersonFactory = class {
|
|
|
232
208
|
lastName: character.lastName,
|
|
233
209
|
username,
|
|
234
210
|
email,
|
|
235
|
-
password,
|
|
236
|
-
phone,
|
|
237
211
|
address: {
|
|
238
212
|
street: address.street,
|
|
239
213
|
city: address.city,
|
|
@@ -254,10 +228,7 @@ var PersonFactory = class {
|
|
|
254
228
|
static buildFromCharacter(character, universe, rng) {
|
|
255
229
|
const address = character.address ?? pickRandom(universe.addresses, rng);
|
|
256
230
|
const username = pickRandom(character.usernames, rng);
|
|
257
|
-
const
|
|
258
|
-
const email = EmailGenerator.generateFromUsername(username, domains, rng);
|
|
259
|
-
const phone = PhoneGenerator.generate(universe.domains, rng);
|
|
260
|
-
const password = PasswordGenerator.generate(universe.domains, rng);
|
|
231
|
+
const email = EmailGenerator.generateFromUsername(username, character.emailDomains, rng);
|
|
261
232
|
const quote = pickRandom(character.quotes, rng);
|
|
262
233
|
const person2 = {
|
|
263
234
|
id: crypto.randomUUID(),
|
|
@@ -266,8 +237,6 @@ var PersonFactory = class {
|
|
|
266
237
|
lastName: character.lastName,
|
|
267
238
|
username,
|
|
268
239
|
email,
|
|
269
|
-
password,
|
|
270
|
-
phone,
|
|
271
240
|
address: {
|
|
272
241
|
street: address.street,
|
|
273
242
|
city: address.city,
|