@skillhq/concierge 1.5.0

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.
Files changed (138) hide show
  1. package/README.md +91 -0
  2. package/dist/cli/program.d.ts +3 -0
  3. package/dist/cli/program.d.ts.map +1 -0
  4. package/dist/cli/program.js +46 -0
  5. package/dist/cli/program.js.map +1 -0
  6. package/dist/cli/shared.d.ts +18 -0
  7. package/dist/cli/shared.d.ts.map +1 -0
  8. package/dist/cli/shared.js +2 -0
  9. package/dist/cli/shared.js.map +1 -0
  10. package/dist/cli.d.ts +3 -0
  11. package/dist/cli.d.ts.map +1 -0
  12. package/dist/cli.js +5 -0
  13. package/dist/cli.js.map +1 -0
  14. package/dist/commands/call.d.ts +7 -0
  15. package/dist/commands/call.d.ts.map +1 -0
  16. package/dist/commands/call.js +409 -0
  17. package/dist/commands/call.js.map +1 -0
  18. package/dist/commands/config.d.ts +4 -0
  19. package/dist/commands/config.d.ts.map +1 -0
  20. package/dist/commands/config.js +120 -0
  21. package/dist/commands/config.js.map +1 -0
  22. package/dist/commands/find-contact.d.ts +4 -0
  23. package/dist/commands/find-contact.d.ts.map +1 -0
  24. package/dist/commands/find-contact.js +57 -0
  25. package/dist/commands/find-contact.js.map +1 -0
  26. package/dist/commands/server.d.ts +7 -0
  27. package/dist/commands/server.d.ts.map +1 -0
  28. package/dist/commands/server.js +212 -0
  29. package/dist/commands/server.js.map +1 -0
  30. package/dist/index.d.ts +4 -0
  31. package/dist/index.d.ts.map +1 -0
  32. package/dist/index.js +3 -0
  33. package/dist/index.js.map +1 -0
  34. package/dist/lib/call/audio/mulaw.d.ts +35 -0
  35. package/dist/lib/call/audio/mulaw.d.ts.map +1 -0
  36. package/dist/lib/call/audio/mulaw.js +109 -0
  37. package/dist/lib/call/audio/mulaw.js.map +1 -0
  38. package/dist/lib/call/audio/pcm-utils.d.ts +62 -0
  39. package/dist/lib/call/audio/pcm-utils.d.ts.map +1 -0
  40. package/dist/lib/call/audio/pcm-utils.js +149 -0
  41. package/dist/lib/call/audio/pcm-utils.js.map +1 -0
  42. package/dist/lib/call/audio/resample.d.ts +34 -0
  43. package/dist/lib/call/audio/resample.d.ts.map +1 -0
  44. package/dist/lib/call/audio/resample.js +97 -0
  45. package/dist/lib/call/audio/resample.js.map +1 -0
  46. package/dist/lib/call/audio/streaming-decoder.d.ts +45 -0
  47. package/dist/lib/call/audio/streaming-decoder.d.ts.map +1 -0
  48. package/dist/lib/call/audio/streaming-decoder.js +110 -0
  49. package/dist/lib/call/audio/streaming-decoder.js.map +1 -0
  50. package/dist/lib/call/call-server.d.ts +110 -0
  51. package/dist/lib/call/call-server.d.ts.map +1 -0
  52. package/dist/lib/call/call-server.js +681 -0
  53. package/dist/lib/call/call-server.js.map +1 -0
  54. package/dist/lib/call/call-session.d.ts +133 -0
  55. package/dist/lib/call/call-session.d.ts.map +1 -0
  56. package/dist/lib/call/call-session.js +890 -0
  57. package/dist/lib/call/call-session.js.map +1 -0
  58. package/dist/lib/call/call-types.d.ts +133 -0
  59. package/dist/lib/call/call-types.d.ts.map +1 -0
  60. package/dist/lib/call/call-types.js +16 -0
  61. package/dist/lib/call/call-types.js.map +1 -0
  62. package/dist/lib/call/conversation-ai.d.ts +56 -0
  63. package/dist/lib/call/conversation-ai.d.ts.map +1 -0
  64. package/dist/lib/call/conversation-ai.js +276 -0
  65. package/dist/lib/call/conversation-ai.js.map +1 -0
  66. package/dist/lib/call/eval/codec-test.d.ts +45 -0
  67. package/dist/lib/call/eval/codec-test.d.ts.map +1 -0
  68. package/dist/lib/call/eval/codec-test.js +169 -0
  69. package/dist/lib/call/eval/codec-test.js.map +1 -0
  70. package/dist/lib/call/eval/conversation-scripts.d.ts +55 -0
  71. package/dist/lib/call/eval/conversation-scripts.d.ts.map +1 -0
  72. package/dist/lib/call/eval/conversation-scripts.js +359 -0
  73. package/dist/lib/call/eval/conversation-scripts.js.map +1 -0
  74. package/dist/lib/call/eval/eval-runner.d.ts +64 -0
  75. package/dist/lib/call/eval/eval-runner.d.ts.map +1 -0
  76. package/dist/lib/call/eval/eval-runner.js +369 -0
  77. package/dist/lib/call/eval/eval-runner.js.map +1 -0
  78. package/dist/lib/call/eval/index.d.ts +9 -0
  79. package/dist/lib/call/eval/index.d.ts.map +1 -0
  80. package/dist/lib/call/eval/index.js +9 -0
  81. package/dist/lib/call/eval/index.js.map +1 -0
  82. package/dist/lib/call/eval/integration-test-suite.d.ts +71 -0
  83. package/dist/lib/call/eval/integration-test-suite.d.ts.map +1 -0
  84. package/dist/lib/call/eval/integration-test-suite.js +519 -0
  85. package/dist/lib/call/eval/integration-test-suite.js.map +1 -0
  86. package/dist/lib/call/eval/turn-taking-test.d.ts +84 -0
  87. package/dist/lib/call/eval/turn-taking-test.d.ts.map +1 -0
  88. package/dist/lib/call/eval/turn-taking-test.js +260 -0
  89. package/dist/lib/call/eval/turn-taking-test.js.map +1 -0
  90. package/dist/lib/call/index.d.ts +12 -0
  91. package/dist/lib/call/index.d.ts.map +1 -0
  92. package/dist/lib/call/index.js +17 -0
  93. package/dist/lib/call/index.js.map +1 -0
  94. package/dist/lib/call/providers/deepgram.d.ts +81 -0
  95. package/dist/lib/call/providers/deepgram.d.ts.map +1 -0
  96. package/dist/lib/call/providers/deepgram.js +279 -0
  97. package/dist/lib/call/providers/deepgram.js.map +1 -0
  98. package/dist/lib/call/providers/elevenlabs.d.ts +78 -0
  99. package/dist/lib/call/providers/elevenlabs.d.ts.map +1 -0
  100. package/dist/lib/call/providers/elevenlabs.js +272 -0
  101. package/dist/lib/call/providers/elevenlabs.js.map +1 -0
  102. package/dist/lib/call/providers/local-deps.d.ts +18 -0
  103. package/dist/lib/call/providers/local-deps.d.ts.map +1 -0
  104. package/dist/lib/call/providers/local-deps.js +114 -0
  105. package/dist/lib/call/providers/local-deps.js.map +1 -0
  106. package/dist/lib/call/providers/twilio.d.ts +53 -0
  107. package/dist/lib/call/providers/twilio.d.ts.map +1 -0
  108. package/dist/lib/call/providers/twilio.js +173 -0
  109. package/dist/lib/call/providers/twilio.js.map +1 -0
  110. package/dist/lib/concierge-client-types.d.ts +68 -0
  111. package/dist/lib/concierge-client-types.d.ts.map +1 -0
  112. package/dist/lib/concierge-client-types.js +2 -0
  113. package/dist/lib/concierge-client-types.js.map +1 -0
  114. package/dist/lib/concierge-client.d.ts +29 -0
  115. package/dist/lib/concierge-client.d.ts.map +1 -0
  116. package/dist/lib/concierge-client.js +534 -0
  117. package/dist/lib/concierge-client.js.map +1 -0
  118. package/dist/lib/config.d.ts +9 -0
  119. package/dist/lib/config.d.ts.map +1 -0
  120. package/dist/lib/config.js +66 -0
  121. package/dist/lib/config.js.map +1 -0
  122. package/dist/lib/output.d.ts +7 -0
  123. package/dist/lib/output.d.ts.map +1 -0
  124. package/dist/lib/output.js +114 -0
  125. package/dist/lib/output.js.map +1 -0
  126. package/dist/lib/utils/contact-extractor.d.ts +12 -0
  127. package/dist/lib/utils/contact-extractor.d.ts.map +1 -0
  128. package/dist/lib/utils/contact-extractor.js +159 -0
  129. package/dist/lib/utils/contact-extractor.js.map +1 -0
  130. package/dist/lib/utils/formatters.d.ts +15 -0
  131. package/dist/lib/utils/formatters.d.ts.map +1 -0
  132. package/dist/lib/utils/formatters.js +107 -0
  133. package/dist/lib/utils/formatters.js.map +1 -0
  134. package/dist/lib/utils/url-parser.d.ts +11 -0
  135. package/dist/lib/utils/url-parser.d.ts.map +1 -0
  136. package/dist/lib/utils/url-parser.js +103 -0
  137. package/dist/lib/utils/url-parser.js.map +1 -0
  138. package/package.json +67 -0
@@ -0,0 +1,534 @@
1
+ import { loadConfig } from './config.js';
2
+ import { extractContacts } from './utils/contact-extractor.js';
3
+ import { cleanPropertyName, emptyContacts, formatGoogleMapsUrl, mergeContacts } from './utils/formatters.js';
4
+ import { parseListingUrl } from './utils/url-parser.js';
5
+ export class ConciergeClient {
6
+ config;
7
+ constructor(config) {
8
+ const loadedConfig = loadConfig();
9
+ this.config = { ...loadedConfig, ...config };
10
+ }
11
+ get timeout() {
12
+ return this.config.timeoutMs ?? 30000;
13
+ }
14
+ async fetchWithTimeout(url, options = {}) {
15
+ const controller = new AbortController();
16
+ const timeout = options.timeout ?? this.timeout;
17
+ const timeoutId = setTimeout(() => controller.abort(), timeout);
18
+ try {
19
+ const response = await fetch(url, {
20
+ signal: controller.signal,
21
+ headers: {
22
+ 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36',
23
+ Accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8',
24
+ 'Accept-Language': 'en-US,en;q=0.5',
25
+ ...options.headers,
26
+ },
27
+ });
28
+ clearTimeout(timeoutId);
29
+ if (!response.ok) {
30
+ return {
31
+ success: false,
32
+ error: `HTTP ${response.status}: ${response.statusText}`,
33
+ };
34
+ }
35
+ const html = await response.text();
36
+ return { success: true, data: html };
37
+ }
38
+ catch (error) {
39
+ clearTimeout(timeoutId);
40
+ if (error instanceof Error) {
41
+ if (error.name === 'AbortError') {
42
+ return { success: false, error: `Request timed out after ${timeout}ms` };
43
+ }
44
+ return { success: false, error: error.message };
45
+ }
46
+ return { success: false, error: 'Unknown fetch error' };
47
+ }
48
+ }
49
+ async fetchJson(url, options = {}) {
50
+ const controller = new AbortController();
51
+ const timeout = options.timeout ?? this.timeout;
52
+ const timeoutId = setTimeout(() => controller.abort(), timeout);
53
+ try {
54
+ const response = await fetch(url, {
55
+ signal: controller.signal,
56
+ headers: {
57
+ Accept: 'application/json',
58
+ ...options.headers,
59
+ },
60
+ });
61
+ clearTimeout(timeoutId);
62
+ if (!response.ok) {
63
+ return {
64
+ success: false,
65
+ error: `HTTP ${response.status}: ${response.statusText}`,
66
+ };
67
+ }
68
+ const json = (await response.json());
69
+ return { success: true, data: json };
70
+ }
71
+ catch (error) {
72
+ clearTimeout(timeoutId);
73
+ if (error instanceof Error) {
74
+ if (error.name === 'AbortError') {
75
+ return { success: false, error: `Request timed out after ${timeout}ms` };
76
+ }
77
+ return { success: false, error: error.message };
78
+ }
79
+ return { success: false, error: 'Unknown fetch error' };
80
+ }
81
+ }
82
+ log(message, verbose) {
83
+ if (verbose) {
84
+ console.log(`[concierge] ${message}`);
85
+ }
86
+ }
87
+ async findContacts(url, options = {}) {
88
+ const { html, verbose = false } = options;
89
+ // Step 1: Parse URL to detect platform
90
+ const parsed = parseListingUrl(url);
91
+ this.log(`Detected platform: ${parsed.platform}`, verbose);
92
+ // Step 2: Extract property info from listing
93
+ let property;
94
+ switch (parsed.platform) {
95
+ case 'airbnb':
96
+ property = this.parseAirbnbHtml(html || '', parsed.url);
97
+ break;
98
+ case 'booking':
99
+ property = this.parseBookingHtml(html || '', parsed.url);
100
+ break;
101
+ case 'vrbo':
102
+ property = this.parseVrboHtml(html || '', parsed.url);
103
+ break;
104
+ case 'expedia':
105
+ property = this.parseExpediaHtml(html || '', parsed.url);
106
+ break;
107
+ default:
108
+ return {
109
+ success: false,
110
+ error: 'Unsupported platform. Supported: Airbnb, Booking.com, VRBO, Expedia',
111
+ };
112
+ }
113
+ this.log(`Property: ${property.name}`, verbose);
114
+ this.log(`Location: ${this.formatLocation(property.location)}`, verbose);
115
+ // Step 3: Collect contact information from various sources
116
+ let contacts = emptyContacts();
117
+ const sources = [];
118
+ // Add listing as a source
119
+ sources.push({
120
+ type: 'listing',
121
+ url: property.listingUrl,
122
+ confidence: 'high',
123
+ });
124
+ // Step 3a: Try Google Places API if configured
125
+ if (this.config.googlePlacesApiKey) {
126
+ this.log('Searching Google Places...', verbose);
127
+ const searchQuery = this.buildSearchQuery(property);
128
+ const placesResult = await this.searchGooglePlaces(searchQuery);
129
+ if (placesResult.success) {
130
+ contacts = mergeContacts(contacts, placesResult.data.contacts);
131
+ sources.push(...placesResult.data.sources);
132
+ this.log(`Found via Google Places: ${JSON.stringify(placesResult.data.contacts)}`, verbose);
133
+ }
134
+ else {
135
+ this.log(`Google Places search failed: ${placesResult.error}`, verbose);
136
+ }
137
+ }
138
+ // Step 3b: If we have a website, scrape it for contacts
139
+ if (contacts.website) {
140
+ this.log(`Scraping website: ${contacts.website}`, verbose);
141
+ const websiteResult = await this.scrapeContactPage(contacts.website);
142
+ if (websiteResult.success) {
143
+ contacts = mergeContacts(contacts, websiteResult.data.contacts);
144
+ sources.push(...websiteResult.data.sources);
145
+ this.log(`Found via website: ${JSON.stringify(websiteResult.data.contacts)}`, verbose);
146
+ }
147
+ }
148
+ // Step 3c: If we found an Instagram handle, look it up
149
+ if (contacts.instagram) {
150
+ this.log(`Looking up Instagram: ${contacts.instagram}`, verbose);
151
+ const igResult = await this.lookupInstagramProfile(contacts.instagram);
152
+ if (igResult.success) {
153
+ // Only merge new info, keep the handle
154
+ const igContacts = igResult.data.contacts;
155
+ if (igContacts.website && !contacts.website) {
156
+ contacts.website = igContacts.website;
157
+ }
158
+ if (igContacts.email?.length) {
159
+ contacts.email = [...new Set([...(contacts.email || []), ...igContacts.email])];
160
+ }
161
+ if (igContacts.phone?.length) {
162
+ contacts.phone = [...new Set([...(contacts.phone || []), ...igContacts.phone])];
163
+ }
164
+ sources.push(...igResult.data.sources);
165
+ }
166
+ }
167
+ // Build the dossier
168
+ const dossier = {
169
+ property,
170
+ contacts,
171
+ sources: this.deduplicateSources(sources),
172
+ searchedAt: new Date().toISOString(),
173
+ };
174
+ return { success: true, data: dossier };
175
+ }
176
+ // Platform parsers
177
+ parseAirbnbHtml(html, url) {
178
+ const property = {
179
+ platform: 'airbnb',
180
+ name: 'Airbnb Listing',
181
+ location: {},
182
+ listingUrl: url,
183
+ };
184
+ const idMatch = url.match(/\/rooms\/(\d+)/);
185
+ if (idMatch) {
186
+ property.name = `Airbnb Listing #${idMatch[1]}`;
187
+ }
188
+ if (!html)
189
+ return property;
190
+ // Extract from og:title
191
+ const ogTitleMatch = html.match(/<meta[^>]+property=["']og:title["'][^>]+content=["']([^"']+)["']/i);
192
+ if (ogTitleMatch) {
193
+ const title = ogTitleMatch[1].trim();
194
+ if (title && title.length < 200) {
195
+ property.name = cleanPropertyName(title);
196
+ }
197
+ }
198
+ // Extract JSON-LD
199
+ this.parseJsonLd(html, property);
200
+ // Extract host
201
+ const hostMatch = html.match(/Hosted by ([^<"\n]+)/i);
202
+ if (hostMatch) {
203
+ property.hostName = hostMatch[1].trim();
204
+ }
205
+ return property;
206
+ }
207
+ parseBookingHtml(html, url) {
208
+ const property = {
209
+ platform: 'booking',
210
+ name: 'Booking.com Listing',
211
+ location: {},
212
+ listingUrl: url,
213
+ };
214
+ const urlMatch = url.match(/\/hotel\/[a-z]{2}\/([^/?]+)/i);
215
+ if (urlMatch) {
216
+ property.name = urlMatch[1]
217
+ .replace(/\.html?$/i, '')
218
+ .replace(/-/g, ' ')
219
+ .replace(/\b\w/g, (c) => c.toUpperCase())
220
+ .trim();
221
+ }
222
+ if (!html)
223
+ return property;
224
+ const ogTitleMatch = html.match(/<meta[^>]+property=["']og:title["'][^>]+content=["']([^"']+)["']/i);
225
+ if (ogTitleMatch) {
226
+ const title = ogTitleMatch[1].replace(/\s*[-·|]\s*Booking\.com.*$/i, '').trim();
227
+ if (title && title.length < 200) {
228
+ property.name = cleanPropertyName(title);
229
+ }
230
+ }
231
+ this.parseJsonLd(html, property);
232
+ return property;
233
+ }
234
+ parseVrboHtml(html, url) {
235
+ const property = {
236
+ platform: 'vrbo',
237
+ name: 'VRBO Listing',
238
+ location: {},
239
+ listingUrl: url,
240
+ };
241
+ const idMatch = url.match(/vrbo\.com\/(\d+)/i);
242
+ if (idMatch) {
243
+ property.name = `VRBO Listing #${idMatch[1]}`;
244
+ }
245
+ if (!html)
246
+ return property;
247
+ const ogTitleMatch = html.match(/<meta[^>]+property=["']og:title["'][^>]+content=["']([^"']+)["']/i);
248
+ if (ogTitleMatch) {
249
+ const title = ogTitleMatch[1].trim();
250
+ if (title && title.length < 200) {
251
+ property.name = cleanPropertyName(title);
252
+ }
253
+ }
254
+ this.parseJsonLd(html, property);
255
+ const hostMatch = html.match(/Hosted by ([^<"\n]+)/i);
256
+ if (hostMatch) {
257
+ property.hostName = hostMatch[1].trim();
258
+ }
259
+ return property;
260
+ }
261
+ parseExpediaHtml(html, url) {
262
+ const property = {
263
+ platform: 'expedia',
264
+ name: 'Expedia Listing',
265
+ location: {},
266
+ listingUrl: url,
267
+ };
268
+ const slugMatch = url.match(/\/([^/]+)\.h\d+\./i);
269
+ if (slugMatch) {
270
+ property.name = slugMatch[1]
271
+ .replace(/-/g, ' ')
272
+ .replace(/\b\w/g, (c) => c.toUpperCase())
273
+ .trim();
274
+ }
275
+ if (!html)
276
+ return property;
277
+ const ogTitleMatch = html.match(/<meta[^>]+property=["']og:title["'][^>]+content=["']([^"']+)["']/i);
278
+ if (ogTitleMatch) {
279
+ const title = ogTitleMatch[1].replace(/\s*[-·|]\s*Expedia.*$/i, '').trim();
280
+ if (title && title.length < 200) {
281
+ property.name = cleanPropertyName(title);
282
+ }
283
+ }
284
+ this.parseJsonLd(html, property);
285
+ return property;
286
+ }
287
+ parseJsonLd(html, property) {
288
+ const jsonLdMatch = html.match(/<script[^>]+type=["']application\/ld\+json["'][^>]*>([\s\S]*?)<\/script>/gi);
289
+ if (!jsonLdMatch)
290
+ return;
291
+ for (const match of jsonLdMatch) {
292
+ const jsonContent = match.replace(/<script[^>]*>/i, '').replace(/<\/script>/i, '');
293
+ try {
294
+ const data = JSON.parse(jsonContent);
295
+ if (data.name) {
296
+ property.name = cleanPropertyName(data.name);
297
+ }
298
+ if (data.address) {
299
+ if (typeof data.address === 'string') {
300
+ property.location.address = data.address;
301
+ }
302
+ else {
303
+ property.location.city = data.address.addressLocality;
304
+ property.location.region = data.address.addressRegion;
305
+ property.location.country = data.address.addressCountry?.name || data.address.addressCountry;
306
+ }
307
+ }
308
+ if (data.geo) {
309
+ property.location.coordinates = {
310
+ lat: Number.parseFloat(data.geo.latitude),
311
+ lng: Number.parseFloat(data.geo.longitude),
312
+ };
313
+ }
314
+ }
315
+ catch {
316
+ // JSON parse failed, continue
317
+ }
318
+ }
319
+ }
320
+ // Contact discovery methods
321
+ async searchGooglePlaces(query) {
322
+ const apiKey = this.config.googlePlacesApiKey;
323
+ if (!apiKey) {
324
+ return { success: false, error: 'Google Places API key not configured' };
325
+ }
326
+ try {
327
+ const searchUrl = new URL('https://maps.googleapis.com/maps/api/place/findplacefromtext/json');
328
+ searchUrl.searchParams.set('input', query);
329
+ searchUrl.searchParams.set('inputtype', 'textquery');
330
+ searchUrl.searchParams.set('fields', 'place_id,name,formatted_address,geometry');
331
+ searchUrl.searchParams.set('key', apiKey);
332
+ const searchResult = await this.fetchJson(searchUrl.toString());
333
+ if (!searchResult.success)
334
+ return searchResult;
335
+ if (searchResult.data.status !== 'OK' || !searchResult.data.candidates?.length) {
336
+ return { success: false, error: searchResult.data.error_message || 'No places found' };
337
+ }
338
+ const place = searchResult.data.candidates[0];
339
+ const detailsUrl = new URL('https://maps.googleapis.com/maps/api/place/details/json');
340
+ detailsUrl.searchParams.set('place_id', place.place_id);
341
+ detailsUrl.searchParams.set('fields', 'name,formatted_phone_number,international_phone_number,website,url');
342
+ detailsUrl.searchParams.set('key', apiKey);
343
+ const detailsResult = await this.fetchJson(detailsUrl.toString());
344
+ if (!detailsResult.success)
345
+ return detailsResult;
346
+ if (detailsResult.data.status !== 'OK' || !detailsResult.data.result) {
347
+ return { success: false, error: detailsResult.data.error_message || 'Failed to get place details' };
348
+ }
349
+ const details = detailsResult.data.result;
350
+ const contacts = { phone: [], email: [] };
351
+ const phone = details.international_phone_number || details.formatted_phone_number;
352
+ if (phone)
353
+ contacts.phone = [phone];
354
+ if (details.website)
355
+ contacts.website = details.website;
356
+ if (details.url) {
357
+ contacts.googleMapsUrl = details.url;
358
+ }
359
+ else if (place.geometry?.location) {
360
+ contacts.googleMapsUrl = formatGoogleMapsUrl(`${place.geometry.location.lat},${place.geometry.location.lng}`);
361
+ }
362
+ return {
363
+ success: true,
364
+ data: {
365
+ contacts,
366
+ sources: [
367
+ {
368
+ type: 'google-places',
369
+ url: contacts.googleMapsUrl,
370
+ confidence: 'high',
371
+ note: `Matched: ${details.name}`,
372
+ },
373
+ ],
374
+ },
375
+ };
376
+ }
377
+ catch (error) {
378
+ return { success: false, error: error instanceof Error ? error.message : 'Google Places API error' };
379
+ }
380
+ }
381
+ async scrapeWebsiteContacts(url) {
382
+ const htmlResult = await this.fetchWithTimeout(url);
383
+ if (!htmlResult.success)
384
+ return htmlResult;
385
+ const extracted = extractContacts(htmlResult.data);
386
+ const contacts = {
387
+ phone: extracted.phones,
388
+ email: extracted.emails,
389
+ whatsapp: extracted.whatsapp[0],
390
+ instagram: extracted.instagram[0],
391
+ facebook: extracted.facebook[0],
392
+ };
393
+ const sources = [];
394
+ if (extracted.emails.length > 0 || extracted.phones.length > 0) {
395
+ sources.push({ type: 'website', url, confidence: 'medium' });
396
+ }
397
+ return { success: true, data: { contacts, sources } };
398
+ }
399
+ async scrapeContactPage(baseUrl) {
400
+ const contactPaths = ['/contact', '/contact-us', '/contacto', '/about'];
401
+ let baseUrlParsed;
402
+ try {
403
+ baseUrlParsed = new URL(baseUrl);
404
+ }
405
+ catch {
406
+ return { success: false, error: 'Invalid base URL' };
407
+ }
408
+ const allContacts = { phone: [], email: [] };
409
+ const allSources = [];
410
+ // Scrape main page first
411
+ const mainResult = await this.scrapeWebsiteContacts(baseUrl);
412
+ if (mainResult.success) {
413
+ this.mergePartialContacts(allContacts, mainResult.data.contacts);
414
+ allSources.push(...mainResult.data.sources);
415
+ }
416
+ // Try contact pages
417
+ for (const path of contactPaths) {
418
+ const contactUrl = new URL(path, baseUrlParsed).toString();
419
+ const result = await this.scrapeWebsiteContacts(contactUrl);
420
+ if (result.success && this.hasAnyPartialContact(result.data.contacts)) {
421
+ this.mergePartialContacts(allContacts, result.data.contacts);
422
+ for (const source of result.data.sources) {
423
+ if (source.type === 'website') {
424
+ source.url = contactUrl;
425
+ source.note = 'Contact page';
426
+ }
427
+ allSources.push(source);
428
+ }
429
+ if ((allContacts.email?.length ?? 0) > 0 && (allContacts.phone?.length ?? 0) > 0) {
430
+ break;
431
+ }
432
+ }
433
+ }
434
+ if (allContacts.phone)
435
+ allContacts.phone = [...new Set(allContacts.phone)];
436
+ if (allContacts.email)
437
+ allContacts.email = [...new Set(allContacts.email)];
438
+ return { success: true, data: { contacts: allContacts, sources: this.deduplicateSources(allSources) } };
439
+ }
440
+ async lookupInstagramProfile(handle) {
441
+ const cleanHandle = handle.replace(/^@/, '').toLowerCase();
442
+ const profileUrl = `https://www.instagram.com/${cleanHandle}/`;
443
+ const htmlResult = await this.fetchWithTimeout(profileUrl);
444
+ if (!htmlResult.success) {
445
+ return {
446
+ success: true,
447
+ data: {
448
+ contacts: { instagram: `@${cleanHandle}` },
449
+ sources: [
450
+ {
451
+ type: 'instagram',
452
+ url: profileUrl,
453
+ confidence: 'low',
454
+ note: 'Profile exists but details require browser access',
455
+ },
456
+ ],
457
+ },
458
+ };
459
+ }
460
+ const html = htmlResult.data;
461
+ const contacts = { instagram: `@${cleanHandle}` };
462
+ const extracted = extractContacts(html);
463
+ if (extracted.emails.length > 0)
464
+ contacts.email = extracted.emails;
465
+ if (extracted.phones.length > 0)
466
+ contacts.phone = extracted.phones;
467
+ const websiteMatch = html.match(/"external_url"\s*:\s*"([^"]+)"/i);
468
+ if (websiteMatch) {
469
+ let website = websiteMatch[1];
470
+ website = website.replace(/\\u([0-9a-fA-F]{4})/g, (_, code) => String.fromCharCode(Number.parseInt(code, 16)));
471
+ if (website && !website.includes('instagram.com')) {
472
+ contacts.website = website;
473
+ }
474
+ }
475
+ return {
476
+ success: true,
477
+ data: {
478
+ contacts,
479
+ sources: [
480
+ {
481
+ type: 'instagram',
482
+ url: profileUrl,
483
+ confidence: contacts.email || contacts.phone ? 'medium' : 'low',
484
+ },
485
+ ],
486
+ },
487
+ };
488
+ }
489
+ // Helper methods
490
+ formatLocation(location) {
491
+ return [location.city, location.region, location.country].filter(Boolean).join(', ') || 'Unknown';
492
+ }
493
+ buildSearchQuery(property) {
494
+ const parts = [property.name];
495
+ if (property.location.city)
496
+ parts.push(property.location.city);
497
+ if (property.location.country)
498
+ parts.push(property.location.country);
499
+ parts.push(property.platform === 'airbnb' || property.platform === 'vrbo' ? 'vacation rental' : 'hotel');
500
+ return parts.join(' ');
501
+ }
502
+ mergePartialContacts(target, source) {
503
+ if (source.phone)
504
+ target.phone = [...(target.phone || []), ...source.phone];
505
+ if (source.email)
506
+ target.email = [...(target.email || []), ...source.email];
507
+ if (source.whatsapp && !target.whatsapp)
508
+ target.whatsapp = source.whatsapp;
509
+ if (source.instagram && !target.instagram)
510
+ target.instagram = source.instagram;
511
+ if (source.facebook && !target.facebook)
512
+ target.facebook = source.facebook;
513
+ if (source.website && !target.website)
514
+ target.website = source.website;
515
+ }
516
+ hasAnyPartialContact(contacts) {
517
+ return ((contacts.phone?.length ?? 0) > 0 ||
518
+ (contacts.email?.length ?? 0) > 0 ||
519
+ !!contacts.whatsapp ||
520
+ !!contacts.instagram ||
521
+ !!contacts.facebook);
522
+ }
523
+ deduplicateSources(sources) {
524
+ const seen = new Set();
525
+ return sources.filter((source) => {
526
+ const key = `${source.type}-${source.url || ''}`;
527
+ if (seen.has(key))
528
+ return false;
529
+ seen.add(key);
530
+ return true;
531
+ });
532
+ }
533
+ }
534
+ //# sourceMappingURL=concierge-client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"concierge-client.js","sourceRoot":"","sources":["../../src/lib/concierge-client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { ConciergeConfig } from './concierge-client-types.js';
2
+ export declare function getConfigPath(): string;
3
+ export declare function loadConfig(): ConciergeConfig;
4
+ export declare function saveConfig(config: ConciergeConfig): void;
5
+ export declare function getConfigValue<K extends keyof ConciergeConfig>(key: K): ConciergeConfig[K] | undefined;
6
+ export declare function setConfigValue<K extends keyof ConciergeConfig>(key: K, value: ConciergeConfig[K]): void;
7
+ export declare function deleteConfigValue<K extends keyof ConciergeConfig>(key: K): void;
8
+ export declare function ensureConfigDir(): void;
9
+ //# sourceMappingURL=config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../src/lib/config.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAwBnE,wBAAgB,aAAa,IAAI,MAAM,CAEtC;AAED,wBAAgB,UAAU,IAAI,eAAe,CAW5C;AAED,wBAAgB,UAAU,CAAC,MAAM,EAAE,eAAe,GAAG,IAAI,CASxD;AAED,wBAAgB,cAAc,CAAC,CAAC,SAAS,MAAM,eAAe,EAAE,GAAG,EAAE,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,GAAG,SAAS,CAGtG;AAED,wBAAgB,cAAc,CAAC,CAAC,SAAS,MAAM,eAAe,EAAE,GAAG,EAAE,CAAC,EAAE,KAAK,EAAE,eAAe,CAAC,CAAC,CAAC,GAAG,IAAI,CAIvG;AAED,wBAAgB,iBAAiB,CAAC,CAAC,SAAS,MAAM,eAAe,EAAE,GAAG,EAAE,CAAC,GAAG,IAAI,CAI/E;AAED,wBAAgB,eAAe,IAAI,IAAI,CAItC"}
@@ -0,0 +1,66 @@
1
+ import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync } from 'node:fs';
2
+ import { homedir } from 'node:os';
3
+ import { join } from 'node:path';
4
+ import JSON5 from 'json5';
5
+ const OLD_CONFIG_DIR = join(homedir(), '.config', 'travel-concierge');
6
+ const CONFIG_DIR = join(homedir(), '.config', 'concierge');
7
+ // Migration: move old config dir to new location
8
+ function migrateConfigIfNeeded() {
9
+ if (existsSync(OLD_CONFIG_DIR) && !existsSync(CONFIG_DIR)) {
10
+ renameSync(OLD_CONFIG_DIR, CONFIG_DIR);
11
+ console.log('Migrated config from ~/.config/travel-concierge to ~/.config/concierge');
12
+ }
13
+ }
14
+ // Run migration at module load time
15
+ migrateConfigIfNeeded();
16
+ const CONFIG_FILE = join(CONFIG_DIR, 'config.json5');
17
+ const DEFAULT_CONFIG = {
18
+ timeoutMs: 30000,
19
+ callServerPort: 3000,
20
+ elevenLabsVoiceId: 'EXAVITQu4vr4xnSDxMaL', // Rachel voice
21
+ };
22
+ export function getConfigPath() {
23
+ return CONFIG_FILE;
24
+ }
25
+ export function loadConfig() {
26
+ try {
27
+ if (existsSync(CONFIG_FILE)) {
28
+ const content = readFileSync(CONFIG_FILE, 'utf-8');
29
+ const parsed = JSON5.parse(content);
30
+ return { ...DEFAULT_CONFIG, ...parsed };
31
+ }
32
+ }
33
+ catch {
34
+ // Ignore parse errors, use defaults
35
+ }
36
+ return { ...DEFAULT_CONFIG };
37
+ }
38
+ export function saveConfig(config) {
39
+ // Ensure directory exists
40
+ if (!existsSync(CONFIG_DIR)) {
41
+ mkdirSync(CONFIG_DIR, { recursive: true });
42
+ }
43
+ // Pretty print with comments
44
+ const content = JSON5.stringify(config, null, 2);
45
+ writeFileSync(CONFIG_FILE, content, 'utf-8');
46
+ }
47
+ export function getConfigValue(key) {
48
+ const config = loadConfig();
49
+ return config[key];
50
+ }
51
+ export function setConfigValue(key, value) {
52
+ const config = loadConfig();
53
+ config[key] = value;
54
+ saveConfig(config);
55
+ }
56
+ export function deleteConfigValue(key) {
57
+ const config = loadConfig();
58
+ delete config[key];
59
+ saveConfig(config);
60
+ }
61
+ export function ensureConfigDir() {
62
+ if (!existsSync(CONFIG_DIR)) {
63
+ mkdirSync(CONFIG_DIR, { recursive: true });
64
+ }
65
+ }
66
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../../src/lib/config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import type { CliContext } from '../cli/shared.js';
2
+ import type { ContactDossier } from './concierge-client-types.js';
3
+ export declare function formatDossier(dossier: ContactDossier, ctx: CliContext): string;
4
+ export declare function formatError(error: string, ctx: CliContext): string;
5
+ export declare function formatInfo(message: string, ctx: CliContext): string;
6
+ export declare function formatVerbose(message: string, ctx: CliContext): string;
7
+ //# sourceMappingURL=output.d.ts.map