@steipete/summarize-core 0.7.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 (182) hide show
  1. package/README.md +7 -0
  2. package/dist/esm/content/index.js +5 -0
  3. package/dist/esm/content/index.js.map +1 -0
  4. package/dist/esm/content/link-preview/client.js +28 -0
  5. package/dist/esm/content/link-preview/client.js.map +1 -0
  6. package/dist/esm/content/link-preview/content/article.js +155 -0
  7. package/dist/esm/content/link-preview/content/article.js.map +1 -0
  8. package/dist/esm/content/link-preview/content/cleaner.js +55 -0
  9. package/dist/esm/content/link-preview/content/cleaner.js.map +1 -0
  10. package/dist/esm/content/link-preview/content/constants.js +7 -0
  11. package/dist/esm/content/link-preview/content/constants.js.map +1 -0
  12. package/dist/esm/content/link-preview/content/fetcher.js +124 -0
  13. package/dist/esm/content/link-preview/content/fetcher.js.map +1 -0
  14. package/dist/esm/content/link-preview/content/firecrawl.js +86 -0
  15. package/dist/esm/content/link-preview/content/firecrawl.js.map +1 -0
  16. package/dist/esm/content/link-preview/content/html.js +162 -0
  17. package/dist/esm/content/link-preview/content/html.js.map +1 -0
  18. package/dist/esm/content/link-preview/content/index.js +345 -0
  19. package/dist/esm/content/link-preview/content/index.js.map +1 -0
  20. package/dist/esm/content/link-preview/content/jsonld.js +77 -0
  21. package/dist/esm/content/link-preview/content/jsonld.js.map +1 -0
  22. package/dist/esm/content/link-preview/content/parsers.js +77 -0
  23. package/dist/esm/content/link-preview/content/parsers.js.map +1 -0
  24. package/dist/esm/content/link-preview/content/podcast-utils.js +79 -0
  25. package/dist/esm/content/link-preview/content/podcast-utils.js.map +1 -0
  26. package/dist/esm/content/link-preview/content/readability.js +53 -0
  27. package/dist/esm/content/link-preview/content/readability.js.map +1 -0
  28. package/dist/esm/content/link-preview/content/twitter-utils.js +68 -0
  29. package/dist/esm/content/link-preview/content/twitter-utils.js.map +1 -0
  30. package/dist/esm/content/link-preview/content/types.js +4 -0
  31. package/dist/esm/content/link-preview/content/types.js.map +1 -0
  32. package/dist/esm/content/link-preview/content/utils.js +164 -0
  33. package/dist/esm/content/link-preview/content/utils.js.map +1 -0
  34. package/dist/esm/content/link-preview/content/video.js +96 -0
  35. package/dist/esm/content/link-preview/content/video.js.map +1 -0
  36. package/dist/esm/content/link-preview/content/youtube.js +82 -0
  37. package/dist/esm/content/link-preview/content/youtube.js.map +1 -0
  38. package/dist/esm/content/link-preview/deps.js +20 -0
  39. package/dist/esm/content/link-preview/deps.js.map +1 -0
  40. package/dist/esm/content/link-preview/fetch-with-timeout.js +35 -0
  41. package/dist/esm/content/link-preview/fetch-with-timeout.js.map +1 -0
  42. package/dist/esm/content/link-preview/types.js +2 -0
  43. package/dist/esm/content/link-preview/types.js.map +1 -0
  44. package/dist/esm/content/transcript/cache.js +79 -0
  45. package/dist/esm/content/transcript/cache.js.map +1 -0
  46. package/dist/esm/content/transcript/index.js +130 -0
  47. package/dist/esm/content/transcript/index.js.map +1 -0
  48. package/dist/esm/content/transcript/normalize.js +43 -0
  49. package/dist/esm/content/transcript/normalize.js.map +1 -0
  50. package/dist/esm/content/transcript/providers/generic.js +11 -0
  51. package/dist/esm/content/transcript/providers/generic.js.map +1 -0
  52. package/dist/esm/content/transcript/providers/podcast/apple-flow.js +222 -0
  53. package/dist/esm/content/transcript/providers/podcast/apple-flow.js.map +1 -0
  54. package/dist/esm/content/transcript/providers/podcast/apple.js +38 -0
  55. package/dist/esm/content/transcript/providers/podcast/apple.js.map +1 -0
  56. package/dist/esm/content/transcript/providers/podcast/constants.js +8 -0
  57. package/dist/esm/content/transcript/providers/podcast/constants.js.map +1 -0
  58. package/dist/esm/content/transcript/providers/podcast/flow-context.js +2 -0
  59. package/dist/esm/content/transcript/providers/podcast/flow-context.js.map +1 -0
  60. package/dist/esm/content/transcript/providers/podcast/itunes.js +134 -0
  61. package/dist/esm/content/transcript/providers/podcast/itunes.js.map +1 -0
  62. package/dist/esm/content/transcript/providers/podcast/json.js +34 -0
  63. package/dist/esm/content/transcript/providers/podcast/json.js.map +1 -0
  64. package/dist/esm/content/transcript/providers/podcast/media.js +345 -0
  65. package/dist/esm/content/transcript/providers/podcast/media.js.map +1 -0
  66. package/dist/esm/content/transcript/providers/podcast/results.js +28 -0
  67. package/dist/esm/content/transcript/providers/podcast/results.js.map +1 -0
  68. package/dist/esm/content/transcript/providers/podcast/rss.js +253 -0
  69. package/dist/esm/content/transcript/providers/podcast/rss.js.map +1 -0
  70. package/dist/esm/content/transcript/providers/podcast/spotify-flow.js +218 -0
  71. package/dist/esm/content/transcript/providers/podcast/spotify-flow.js.map +1 -0
  72. package/dist/esm/content/transcript/providers/podcast/spotify.js +113 -0
  73. package/dist/esm/content/transcript/providers/podcast/spotify.js.map +1 -0
  74. package/dist/esm/content/transcript/providers/podcast.js +222 -0
  75. package/dist/esm/content/transcript/providers/podcast.js.map +1 -0
  76. package/dist/esm/content/transcript/providers/youtube/api.js +257 -0
  77. package/dist/esm/content/transcript/providers/youtube/api.js.map +1 -0
  78. package/dist/esm/content/transcript/providers/youtube/apify.js +55 -0
  79. package/dist/esm/content/transcript/providers/youtube/apify.js.map +1 -0
  80. package/dist/esm/content/transcript/providers/youtube/captions.js +409 -0
  81. package/dist/esm/content/transcript/providers/youtube/captions.js.map +1 -0
  82. package/dist/esm/content/transcript/providers/youtube/yt-dlp.js +166 -0
  83. package/dist/esm/content/transcript/providers/youtube/yt-dlp.js.map +1 -0
  84. package/dist/esm/content/transcript/providers/youtube.js +173 -0
  85. package/dist/esm/content/transcript/providers/youtube.js.map +1 -0
  86. package/dist/esm/content/transcript/types.js +2 -0
  87. package/dist/esm/content/transcript/types.js.map +1 -0
  88. package/dist/esm/content/transcript/utils.js +259 -0
  89. package/dist/esm/content/transcript/utils.js.map +1 -0
  90. package/dist/esm/index.js +4 -0
  91. package/dist/esm/index.js.map +1 -0
  92. package/dist/esm/language.js +126 -0
  93. package/dist/esm/language.js.map +1 -0
  94. package/dist/esm/prompts/cli.js +20 -0
  95. package/dist/esm/prompts/cli.js.map +1 -0
  96. package/dist/esm/prompts/file.js +48 -0
  97. package/dist/esm/prompts/file.js.map +1 -0
  98. package/dist/esm/prompts/index.js +4 -0
  99. package/dist/esm/prompts/index.js.map +1 -0
  100. package/dist/esm/prompts/link-summary.js +116 -0
  101. package/dist/esm/prompts/link-summary.js.map +1 -0
  102. package/dist/esm/shared/contracts.js +2 -0
  103. package/dist/esm/shared/contracts.js.map +1 -0
  104. package/dist/esm/transcription/whisper/constants.js +8 -0
  105. package/dist/esm/transcription/whisper/constants.js.map +1 -0
  106. package/dist/esm/transcription/whisper/core.js +303 -0
  107. package/dist/esm/transcription/whisper/core.js.map +1 -0
  108. package/dist/esm/transcription/whisper/fal.js +41 -0
  109. package/dist/esm/transcription/whisper/fal.js.map +1 -0
  110. package/dist/esm/transcription/whisper/ffmpeg.js +179 -0
  111. package/dist/esm/transcription/whisper/ffmpeg.js.map +1 -0
  112. package/dist/esm/transcription/whisper/openai.js +47 -0
  113. package/dist/esm/transcription/whisper/openai.js.map +1 -0
  114. package/dist/esm/transcription/whisper/types.js +2 -0
  115. package/dist/esm/transcription/whisper/types.js.map +1 -0
  116. package/dist/esm/transcription/whisper/utils.js +63 -0
  117. package/dist/esm/transcription/whisper/utils.js.map +1 -0
  118. package/dist/esm/transcription/whisper/whisper-cpp.js +227 -0
  119. package/dist/esm/transcription/whisper/whisper-cpp.js.map +1 -0
  120. package/dist/esm/transcription/whisper.js +5 -0
  121. package/dist/esm/transcription/whisper.js.map +1 -0
  122. package/dist/types/content/index.d.ts +5 -0
  123. package/dist/types/content/link-preview/client.d.ts +18 -0
  124. package/dist/types/content/link-preview/content/article.d.ts +4 -0
  125. package/dist/types/content/link-preview/content/cleaner.d.ts +12 -0
  126. package/dist/types/content/link-preview/content/constants.d.ts +6 -0
  127. package/dist/types/content/link-preview/content/fetcher.d.ts +16 -0
  128. package/dist/types/content/link-preview/content/firecrawl.d.ts +14 -0
  129. package/dist/types/content/link-preview/content/html.d.ts +17 -0
  130. package/dist/types/content/link-preview/content/index.d.ts +4 -0
  131. package/dist/types/content/link-preview/content/jsonld.d.ts +6 -0
  132. package/dist/types/content/link-preview/content/parsers.d.ts +7 -0
  133. package/dist/types/content/link-preview/content/podcast-utils.d.ts +7 -0
  134. package/dist/types/content/link-preview/content/readability.d.ts +8 -0
  135. package/dist/types/content/link-preview/content/twitter-utils.d.ts +4 -0
  136. package/dist/types/content/link-preview/content/types.d.ts +61 -0
  137. package/dist/types/content/link-preview/content/utils.d.ts +17 -0
  138. package/dist/types/content/link-preview/content/video.d.ts +5 -0
  139. package/dist/types/content/link-preview/content/youtube.d.ts +1 -0
  140. package/dist/types/content/link-preview/deps.d.ts +167 -0
  141. package/dist/types/content/link-preview/fetch-with-timeout.d.ts +4 -0
  142. package/dist/types/content/link-preview/types.d.ts +37 -0
  143. package/dist/types/content/transcript/cache.d.ts +29 -0
  144. package/dist/types/content/transcript/index.d.ts +9 -0
  145. package/dist/types/content/transcript/normalize.d.ts +3 -0
  146. package/dist/types/content/transcript/providers/generic.d.ts +3 -0
  147. package/dist/types/content/transcript/providers/podcast/apple-flow.d.ts +4 -0
  148. package/dist/types/content/transcript/providers/podcast/apple.d.ts +6 -0
  149. package/dist/types/content/transcript/providers/podcast/constants.d.ts +7 -0
  150. package/dist/types/content/transcript/providers/podcast/flow-context.d.ts +11 -0
  151. package/dist/types/content/transcript/providers/podcast/itunes.d.ts +17 -0
  152. package/dist/types/content/transcript/providers/podcast/json.d.ts +8 -0
  153. package/dist/types/content/transcript/providers/podcast/media.d.ts +42 -0
  154. package/dist/types/content/transcript/providers/podcast/results.d.ts +10 -0
  155. package/dist/types/content/transcript/providers/podcast/rss.d.ts +22 -0
  156. package/dist/types/content/transcript/providers/podcast/spotify-flow.d.ts +3 -0
  157. package/dist/types/content/transcript/providers/podcast/spotify.d.ts +24 -0
  158. package/dist/types/content/transcript/providers/podcast.d.ts +20 -0
  159. package/dist/types/content/transcript/providers/youtube/api.d.ts +26 -0
  160. package/dist/types/content/transcript/providers/youtube/apify.d.ts +1 -0
  161. package/dist/types/content/transcript/providers/youtube/captions.d.ts +7 -0
  162. package/dist/types/content/transcript/providers/youtube/yt-dlp.d.ts +17 -0
  163. package/dist/types/content/transcript/providers/youtube.d.ts +3 -0
  164. package/dist/types/content/transcript/types.d.ts +30 -0
  165. package/dist/types/content/transcript/utils.d.ts +8 -0
  166. package/dist/types/index.d.ts +4 -0
  167. package/dist/types/language.d.ts +25 -0
  168. package/dist/types/prompts/cli.d.ts +10 -0
  169. package/dist/types/prompts/file.d.ts +17 -0
  170. package/dist/types/prompts/index.d.ts +4 -0
  171. package/dist/types/prompts/link-summary.d.ts +29 -0
  172. package/dist/types/shared/contracts.d.ts +2 -0
  173. package/dist/types/transcription/whisper/constants.d.ts +7 -0
  174. package/dist/types/transcription/whisper/core.d.ts +20 -0
  175. package/dist/types/transcription/whisper/fal.d.ts +1 -0
  176. package/dist/types/transcription/whisper/ffmpeg.d.ts +16 -0
  177. package/dist/types/transcription/whisper/openai.d.ts +2 -0
  178. package/dist/types/transcription/whisper/types.d.ts +17 -0
  179. package/dist/types/transcription/whisper/utils.d.ts +5 -0
  180. package/dist/types/transcription/whisper/whisper-cpp.d.ts +9 -0
  181. package/dist/types/transcription/whisper.d.ts +5 -0
  182. package/package.json +54 -0
@@ -0,0 +1,222 @@
1
+ import { extractAppleEpisodeTitleFromHtml, extractApplePodcastIds, extractEmbeddedJsonUrl, } from './apple.js';
2
+ import { TRANSCRIPTION_TIMEOUT_MS } from './constants.js';
3
+ import { resolveApplePodcastEpisodeFromItunesLookup } from './itunes.js';
4
+ import { buildWhisperResult, joinNotes } from './results.js';
5
+ import { decodeXmlEntities, extractEnclosureForEpisode, extractEnclosureFromFeed, tryFetchTranscriptFromFeedXml, } from './rss.js';
6
+ export async function fetchAppleTranscriptFromItunesLookup(flow) {
7
+ const appleIds = typeof flow.context.html !== 'string' ? extractApplePodcastIds(flow.context.url) : null;
8
+ if (!appleIds)
9
+ return null;
10
+ try {
11
+ const episode = await resolveApplePodcastEpisodeFromItunesLookup({
12
+ fetchImpl: flow.options.fetch,
13
+ showId: appleIds.showId,
14
+ episodeId: appleIds.episodeId,
15
+ });
16
+ if (!episode) {
17
+ throw new Error('iTunes lookup did not return an episodeUrl');
18
+ }
19
+ if (episode.feedUrl && episode.episodeTitle) {
20
+ flow.pushOnce('podcastTranscript');
21
+ const feedResponse = await flow.options.fetch(episode.feedUrl, {
22
+ redirect: 'follow',
23
+ signal: AbortSignal.timeout(TRANSCRIPTION_TIMEOUT_MS),
24
+ });
25
+ if (feedResponse.ok) {
26
+ const feedXml = await feedResponse.text();
27
+ let maybeTranscript = null;
28
+ if (/podcast:transcript/i.test(feedXml)) {
29
+ maybeTranscript = await tryFetchTranscriptFromFeedXml({
30
+ fetchImpl: flow.options.fetch,
31
+ feedXml,
32
+ episodeTitle: episode.episodeTitle,
33
+ notes: flow.notes,
34
+ });
35
+ }
36
+ if (maybeTranscript) {
37
+ flow.notes.push('Resolved Apple Podcasts episode via RSS <podcast:transcript>');
38
+ return {
39
+ text: maybeTranscript.text,
40
+ source: 'podcastTranscript',
41
+ attemptedProviders: flow.attemptedProviders,
42
+ notes: joinNotes(flow.notes),
43
+ metadata: {
44
+ provider: 'podcast',
45
+ kind: 'apple_itunes_rss_transcript',
46
+ showId: appleIds.showId,
47
+ episodeId: appleIds.episodeId,
48
+ feedUrl: episode.feedUrl,
49
+ episodeTitle: episode.episodeTitle,
50
+ transcriptUrl: maybeTranscript.transcriptUrl,
51
+ transcriptType: maybeTranscript.transcriptType,
52
+ },
53
+ };
54
+ }
55
+ }
56
+ }
57
+ const missing = flow.ensureTranscriptionProvider();
58
+ if (missing)
59
+ return missing;
60
+ flow.pushOnce('whisper');
61
+ const result = await flow.transcribe({
62
+ url: episode.episodeUrl,
63
+ filenameHint: episode.fileExtension ? `episode.${episode.fileExtension}` : 'episode.mp3',
64
+ durationSecondsHint: episode.durationSeconds,
65
+ });
66
+ if (result.text) {
67
+ flow.notes.push('Resolved Apple Podcasts episode via iTunes lookup');
68
+ }
69
+ return buildWhisperResult({
70
+ attemptedProviders: flow.attemptedProviders,
71
+ notes: flow.notes,
72
+ outcome: result,
73
+ includeProviderOnFailure: true,
74
+ metadata: {
75
+ provider: 'podcast',
76
+ kind: 'apple_itunes_episode',
77
+ showId: appleIds.showId,
78
+ episodeId: appleIds.episodeId,
79
+ episodeUrl: episode.episodeUrl,
80
+ feedUrl: episode.feedUrl,
81
+ durationSeconds: episode.durationSeconds,
82
+ },
83
+ });
84
+ }
85
+ catch (error) {
86
+ return {
87
+ text: null,
88
+ source: null,
89
+ attemptedProviders: flow.attemptedProviders,
90
+ notes: `Apple Podcasts iTunes lookup failed: ${error instanceof Error ? error.message : String(error)}`,
91
+ metadata: { provider: 'podcast', kind: 'apple_itunes_episode', showId: appleIds.showId },
92
+ };
93
+ }
94
+ }
95
+ export async function fetchAppleTranscriptFromEmbeddedHtml(flow) {
96
+ if (typeof flow.context.html !== 'string')
97
+ return null;
98
+ const appleEpisodeTitle = extractAppleEpisodeTitleFromHtml(flow.context.html);
99
+ const appleFeedUrl = extractEmbeddedJsonUrl(flow.context.html, 'feedUrl');
100
+ if (appleFeedUrl) {
101
+ try {
102
+ const feedResponse = await flow.options.fetch(appleFeedUrl, {
103
+ signal: AbortSignal.timeout(TRANSCRIPTION_TIMEOUT_MS),
104
+ });
105
+ if (!feedResponse.ok) {
106
+ throw new Error(`Feed fetch failed (${feedResponse.status})`);
107
+ }
108
+ const xml = await feedResponse.text();
109
+ let maybeTranscript = null;
110
+ if (/podcast:transcript/i.test(xml)) {
111
+ flow.pushOnce('podcastTranscript');
112
+ maybeTranscript = await tryFetchTranscriptFromFeedXml({
113
+ fetchImpl: flow.options.fetch,
114
+ feedXml: xml,
115
+ episodeTitle: appleEpisodeTitle,
116
+ notes: flow.notes,
117
+ });
118
+ }
119
+ if (maybeTranscript) {
120
+ return {
121
+ text: maybeTranscript.text,
122
+ source: 'podcastTranscript',
123
+ attemptedProviders: flow.attemptedProviders,
124
+ notes: joinNotes(flow.notes),
125
+ metadata: {
126
+ provider: 'podcast',
127
+ kind: 'apple_feed_transcript',
128
+ feedUrl: appleFeedUrl,
129
+ episodeTitle: appleEpisodeTitle,
130
+ transcriptUrl: maybeTranscript.transcriptUrl,
131
+ transcriptType: maybeTranscript.transcriptType,
132
+ },
133
+ };
134
+ }
135
+ const enclosure = appleEpisodeTitle != null
136
+ ? extractEnclosureForEpisode(xml, appleEpisodeTitle)
137
+ : extractEnclosureFromFeed(xml);
138
+ if (enclosure) {
139
+ const resolvedUrl = decodeXmlEntities(enclosure.enclosureUrl);
140
+ const durationSeconds = enclosure.durationSeconds;
141
+ const missing = flow.ensureTranscriptionProvider();
142
+ if (missing)
143
+ return missing;
144
+ flow.pushOnce('whisper');
145
+ let result;
146
+ try {
147
+ result = await flow.transcribe({
148
+ url: resolvedUrl,
149
+ filenameHint: 'episode.mp3',
150
+ durationSecondsHint: durationSeconds,
151
+ });
152
+ }
153
+ catch (error) {
154
+ return {
155
+ text: null,
156
+ source: null,
157
+ attemptedProviders: flow.attemptedProviders,
158
+ notes: error instanceof Error ? error.message : String(error),
159
+ metadata: {
160
+ provider: 'podcast',
161
+ kind: 'apple_feed_url',
162
+ feedUrl: appleFeedUrl,
163
+ episodeTitle: appleEpisodeTitle,
164
+ enclosureUrl: resolvedUrl,
165
+ durationSeconds,
166
+ },
167
+ };
168
+ }
169
+ return buildWhisperResult({
170
+ attemptedProviders: flow.attemptedProviders,
171
+ notes: flow.notes,
172
+ outcome: result,
173
+ metadata: {
174
+ provider: 'podcast',
175
+ kind: 'apple_feed_url',
176
+ feedUrl: appleFeedUrl,
177
+ episodeTitle: appleEpisodeTitle,
178
+ enclosureUrl: resolvedUrl,
179
+ durationSeconds,
180
+ },
181
+ });
182
+ }
183
+ }
184
+ catch (error) {
185
+ // Apple pages usually contain both `feedUrl` and `streamUrl`. If the feed is flaky/blocked,
186
+ // fall back to `streamUrl` instead of failing the whole provider.
187
+ flow.notes.push(`Podcast feed fetch failed: ${error instanceof Error ? error.message : String(error)}`);
188
+ }
189
+ }
190
+ const appleStreamUrl = extractEmbeddedJsonUrl(flow.context.html, 'streamUrl');
191
+ if (appleStreamUrl) {
192
+ const missing = flow.ensureTranscriptionProvider();
193
+ if (missing)
194
+ return missing;
195
+ flow.pushOnce('whisper');
196
+ let result;
197
+ try {
198
+ result = await flow.transcribe({
199
+ url: appleStreamUrl,
200
+ filenameHint: 'episode.mp3',
201
+ durationSecondsHint: null,
202
+ });
203
+ }
204
+ catch (error) {
205
+ return {
206
+ text: null,
207
+ source: null,
208
+ attemptedProviders: flow.attemptedProviders,
209
+ notes: error instanceof Error ? error.message : String(error),
210
+ metadata: { provider: 'podcast', kind: 'apple_stream_url', streamUrl: appleStreamUrl },
211
+ };
212
+ }
213
+ return buildWhisperResult({
214
+ attemptedProviders: flow.attemptedProviders,
215
+ notes: flow.notes,
216
+ outcome: result,
217
+ metadata: { provider: 'podcast', kind: 'apple_stream_url', streamUrl: appleStreamUrl },
218
+ });
219
+ }
220
+ return null;
221
+ }
222
+ //# sourceMappingURL=apple-flow.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apple-flow.js","sourceRoot":"","sources":["../../../../../../src/content/transcript/providers/podcast/apple-flow.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,38 @@
1
+ export function extractAppleEpisodeTitleFromHtml(html) {
2
+ // Apple Podcast episode pages include `apple:title` and `og:title` meta tags.
3
+ // Prefer `apple:title` (episode title only) to match RSS items.
4
+ const apple = html.match(/<meta\s+name=["']apple:title["']\s+content=["']([^"']+)["']/i)?.[1] ?? null;
5
+ const og = html.match(/<meta\s+property=["']og:title["']\s+content=["']([^"']+)["']/i)?.[1] ?? null;
6
+ const title = (apple ?? og ?? '').trim();
7
+ return title.length > 0 ? title : null;
8
+ }
9
+ export function extractEmbeddedJsonUrl(html, field) {
10
+ const pattern = new RegExp(`"${field}":"((?:\\\\.|[^"\\\\])*)"`, 'i');
11
+ const match = html.match(pattern);
12
+ if (!match?.[1])
13
+ return null;
14
+ try {
15
+ return JSON.parse(`"${match[1]}"`);
16
+ }
17
+ catch {
18
+ return null;
19
+ }
20
+ }
21
+ export function extractApplePodcastIds(url) {
22
+ try {
23
+ const parsed = new URL(url);
24
+ const host = parsed.hostname.toLowerCase().replace(/^www\./, '');
25
+ if (host !== 'podcasts.apple.com')
26
+ return null;
27
+ const showId = parsed.pathname.match(/\/id(\d+)(?:\/|$)/)?.[1] ?? null;
28
+ if (!showId)
29
+ return null;
30
+ const episodeIdRaw = parsed.searchParams.get('i');
31
+ const episodeId = episodeIdRaw && /^\d+$/.test(episodeIdRaw) ? episodeIdRaw : null;
32
+ return { showId, episodeId };
33
+ }
34
+ catch {
35
+ return null;
36
+ }
37
+ }
38
+ //# sourceMappingURL=apple.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apple.js","sourceRoot":"","sources":["../../../../../../src/content/transcript/providers/podcast/apple.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,gCAAgC,CAAC,IAAY;IAC3D,8EAA8E;IAC9E,gEAAgE;IAChE,MAAM,KAAK,GACT,IAAI,CAAC,KAAK,CAAC,8DAA8D,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,CAAA;IACzF,MAAM,EAAE,GACN,IAAI,CAAC,KAAK,CAAC,+DAA+D,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,CAAA;IAC1F,MAAM,KAAK,GAAG,CAAC,KAAK,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAA;IACxC,OAAO,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAA;AACxC,CAAC;AAED,MAAM,UAAU,sBAAsB,CAAC,IAAY,EAAE,KAAa;IAChE,MAAM,OAAO,GAAG,IAAI,MAAM,CAAC,IAAI,KAAK,2BAA2B,EAAE,GAAG,CAAC,CAAA;IACrE,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAA;IACjC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QAAE,OAAO,IAAI,CAAA;IAC5B,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,GAAG,CAAW,CAAA;IAC9C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAA;IACb,CAAC;AACH,CAAC;AAED,MAAM,UAAU,sBAAsB,CACpC,GAAW;IAEX,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,CAAA;QAC3B,MAAM,IAAI,GAAG,MAAM,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAA;QAChE,IAAI,IAAI,KAAK,oBAAoB;YAAE,OAAO,IAAI,CAAA;QAC9C,MAAM,MAAM,GAAG,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,CAAA;QACtE,IAAI,CAAC,MAAM;YAAE,OAAO,IAAI,CAAA;QACxB,MAAM,YAAY,GAAG,MAAM,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;QACjD,MAAM,SAAS,GAAG,YAAY,IAAI,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAA;QAClF,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,CAAA;IAC9B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAA;IACb,CAAC;AACH,CAAC"}
@@ -0,0 +1,8 @@
1
+ export const FEED_HINT_URL_PATTERN = /rss|feed|podcast|\.xml($|[?#])/i;
2
+ export const PODCAST_PLATFORM_HOST_PATTERN = /open\.spotify\.com|spotify\.com|podcasts\.apple\.com|overcast\.fm|pca\.st|pod\.link|castbox\.fm|player\.fm/i;
3
+ export const TRANSCRIPTION_TIMEOUT_MS = 600_000;
4
+ export const MAX_REMOTE_MEDIA_BYTES = 512 * 1024 * 1024;
5
+ export const BLOCKED_HTML_HINT_PATTERN = /access denied|attention required|captcha|recaptcha|cloudflare|forbidden|verify you are human/i;
6
+ export const ITUNES_SEARCH_URL = 'https://itunes.apple.com/search';
7
+ export const ITUNES_LOOKUP_URL = 'https://itunes.apple.com/lookup';
8
+ //# sourceMappingURL=constants.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.js","sourceRoot":"","sources":["../../../../../../src/content/transcript/providers/podcast/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,qBAAqB,GAAG,iCAAiC,CAAA;AACtE,MAAM,CAAC,MAAM,6BAA6B,GACxC,6GAA6G,CAAA;AAC/G,MAAM,CAAC,MAAM,wBAAwB,GAAG,OAAO,CAAA;AAC/C,MAAM,CAAC,MAAM,sBAAsB,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAA;AACvD,MAAM,CAAC,MAAM,yBAAyB,GACpC,+FAA+F,CAAA;AACjG,MAAM,CAAC,MAAM,iBAAiB,GAAG,iCAAiC,CAAA;AAClE,MAAM,CAAC,MAAM,iBAAiB,GAAG,iCAAiC,CAAA"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=flow-context.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"flow-context.js","sourceRoot":"","sources":["../../../../../../src/content/transcript/providers/podcast/flow-context.ts"],"names":[],"mappings":""}
@@ -0,0 +1,134 @@
1
+ import { ITUNES_LOOKUP_URL, ITUNES_SEARCH_URL, TRANSCRIPTION_TIMEOUT_MS } from './constants.js';
2
+ import { asRecordArray, getJsonArray, getRecordString } from './json.js';
3
+ import { normalizeLooseTitle } from './rss.js';
4
+ export async function resolveApplePodcastEpisodeFromItunesLookup({ fetchImpl, showId, episodeId, }) {
5
+ const query = new URLSearchParams({
6
+ id: showId,
7
+ entity: 'podcastEpisode',
8
+ limit: '200',
9
+ });
10
+ const res = await fetchImpl(`${ITUNES_LOOKUP_URL}?${query.toString()}`, {
11
+ redirect: 'follow',
12
+ signal: AbortSignal.timeout(TRANSCRIPTION_TIMEOUT_MS),
13
+ headers: { accept: 'application/json' },
14
+ });
15
+ if (!res.ok)
16
+ return null;
17
+ const payload = (await res.json());
18
+ const results = asRecordArray(getJsonArray(payload, ['results']));
19
+ const show = results.find((r) => getRecordString(r, 'wrapperType') === 'track' && getRecordString(r, 'kind') === 'podcast');
20
+ const feedUrl = typeof show?.feedUrl === 'string' && show.feedUrl.trim() ? show.feedUrl.trim() : null;
21
+ const episodes = results.filter((r) => getRecordString(r, 'wrapperType') === 'podcastEpisode');
22
+ if (episodes.length === 0)
23
+ return null;
24
+ const chosen = (() => {
25
+ if (episodeId) {
26
+ const match = episodes.find((r) => String(r.trackId ?? '') === episodeId);
27
+ if (match)
28
+ return match;
29
+ }
30
+ // No i=... in URL: pick the newest episode by release date.
31
+ const sorted = [...episodes].sort((a, b) => {
32
+ const aDate = Date.parse(String(a.releaseDate ?? ''));
33
+ const bDate = Date.parse(String(b.releaseDate ?? ''));
34
+ if (!Number.isFinite(aDate) && !Number.isFinite(bDate))
35
+ return 0;
36
+ if (!Number.isFinite(aDate))
37
+ return 1;
38
+ if (!Number.isFinite(bDate))
39
+ return -1;
40
+ return bDate - aDate;
41
+ });
42
+ return sorted[0];
43
+ })();
44
+ const episodeUrlRaw = typeof chosen.episodeUrl === 'string'
45
+ ? chosen.episodeUrl.trim()
46
+ : typeof chosen.previewUrl === 'string'
47
+ ? chosen.previewUrl.trim()
48
+ : '';
49
+ if (!episodeUrlRaw || !/^https?:\/\//i.test(episodeUrlRaw))
50
+ return null;
51
+ const fileExtension = typeof chosen.episodeFileExtension === 'string' && chosen.episodeFileExtension.trim()
52
+ ? chosen.episodeFileExtension.trim().replace(/^\./, '')
53
+ : null;
54
+ const durationSeconds = typeof chosen.trackTimeMillis === 'number' && Number.isFinite(chosen.trackTimeMillis)
55
+ ? chosen.trackTimeMillis / 1000
56
+ : null;
57
+ const episodeTitle = typeof chosen.trackName === 'string' && chosen.trackName.trim() ? chosen.trackName.trim() : null;
58
+ return { episodeUrl: episodeUrlRaw, feedUrl, fileExtension, durationSeconds, episodeTitle };
59
+ }
60
+ export async function resolvePodcastFeedUrlFromItunesSearch(fetchImpl, showTitle) {
61
+ const query = new URLSearchParams({
62
+ term: showTitle,
63
+ media: 'podcast',
64
+ entity: 'podcast',
65
+ limit: '10',
66
+ });
67
+ const res = await fetchImpl(`${ITUNES_SEARCH_URL}?${query.toString()}`, {
68
+ redirect: 'follow',
69
+ signal: AbortSignal.timeout(TRANSCRIPTION_TIMEOUT_MS),
70
+ headers: { accept: 'application/json' },
71
+ });
72
+ if (!res.ok) {
73
+ return null;
74
+ }
75
+ const payload = (await res.json());
76
+ const results = asRecordArray(getJsonArray(payload, ['results']));
77
+ if (results.length === 0)
78
+ return null;
79
+ const normalizedTarget = normalizeLooseTitle(showTitle);
80
+ const exact = results.find((r) => normalizeLooseTitle(String(r.collectionName ?? '')) === normalizedTarget);
81
+ const best = exact ?? results[0];
82
+ const feedUrl = typeof best?.feedUrl === 'string' ? best.feedUrl.trim() : '';
83
+ return feedUrl && /^https?:\/\//i.test(feedUrl) ? feedUrl : null;
84
+ }
85
+ export async function resolvePodcastEpisodeFromItunesSearch(fetchImpl, showTitle, episodeTitle) {
86
+ const query = new URLSearchParams({
87
+ term: `${showTitle} ${episodeTitle}`,
88
+ media: 'podcast',
89
+ entity: 'podcastEpisode',
90
+ limit: '25',
91
+ });
92
+ const res = await fetchImpl(`${ITUNES_SEARCH_URL}?${query.toString()}`, {
93
+ redirect: 'follow',
94
+ signal: AbortSignal.timeout(TRANSCRIPTION_TIMEOUT_MS),
95
+ headers: { accept: 'application/json' },
96
+ });
97
+ if (!res.ok)
98
+ return null;
99
+ const payload = (await res.json());
100
+ const results = asRecordArray(getJsonArray(payload, ['results']));
101
+ if (results.length === 0)
102
+ return null;
103
+ const normalizedShow = normalizeLooseTitle(showTitle);
104
+ const normalizedEpisode = normalizeLooseTitle(episodeTitle);
105
+ const candidates = results
106
+ .map((record) => {
107
+ const title = getRecordString(record, 'trackName');
108
+ const collection = getRecordString(record, 'collectionName');
109
+ const episodeUrl = getRecordString(record, 'episodeUrl');
110
+ const durationMs = record.trackTimeMillis;
111
+ const durationSeconds = typeof durationMs === 'number' && Number.isFinite(durationMs) ? durationMs / 1000 : null;
112
+ return {
113
+ title,
114
+ collection,
115
+ episodeUrl,
116
+ durationSeconds,
117
+ };
118
+ })
119
+ .filter((entry) => Boolean(entry.episodeUrl) && Boolean(entry.title));
120
+ if (candidates.length === 0)
121
+ return null;
122
+ const exact = candidates.find((entry) => normalizeLooseTitle(entry.title ?? '') === normalizedEpisode &&
123
+ normalizeLooseTitle(entry.collection ?? '') === normalizedShow);
124
+ const exactEpisode = candidates.find((entry) => normalizeLooseTitle(entry.title ?? '') === normalizedEpisode);
125
+ const best = exact ?? exactEpisode ?? candidates[0];
126
+ if (!best?.episodeUrl)
127
+ return null;
128
+ return {
129
+ episodeUrl: best.episodeUrl,
130
+ durationSeconds: best.durationSeconds,
131
+ episodeTitle: best.title ?? episodeTitle,
132
+ };
133
+ }
134
+ //# sourceMappingURL=itunes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"itunes.js","sourceRoot":"","sources":["../../../../../../src/content/transcript/providers/podcast/itunes.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,34 @@
1
+ export function isJsonRecord(value) {
2
+ return typeof value === 'object' && value !== null && !Array.isArray(value);
3
+ }
4
+ export function getJsonPath(value, path) {
5
+ let current = value;
6
+ for (const key of path) {
7
+ if (!isJsonRecord(current))
8
+ return undefined;
9
+ current = current[key];
10
+ }
11
+ return current;
12
+ }
13
+ export function getJsonString(value, path) {
14
+ const found = getJsonPath(value, path);
15
+ return typeof found === 'string' ? found : null;
16
+ }
17
+ export function getJsonNumber(value, path) {
18
+ const found = getJsonPath(value, path);
19
+ return typeof found === 'number' && Number.isFinite(found) ? found : null;
20
+ }
21
+ export function getJsonArray(value, path) {
22
+ const found = getJsonPath(value, path);
23
+ return Array.isArray(found) ? found : [];
24
+ }
25
+ export function asRecordArray(value) {
26
+ if (!Array.isArray(value))
27
+ return [];
28
+ return value.filter((v) => isJsonRecord(v));
29
+ }
30
+ export function getRecordString(record, key) {
31
+ const value = record[key];
32
+ return typeof value === 'string' ? value : null;
33
+ }
34
+ //# sourceMappingURL=json.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"json.js","sourceRoot":"","sources":["../../../../../../src/content/transcript/providers/podcast/json.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,YAAY,CAAC,KAAc;IACzC,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAA;AAC7E,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,KAAc,EAAE,IAAuB;IACjE,IAAI,OAAO,GAAY,KAAK,CAAA;IAC5B,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC;YAAE,OAAO,SAAS,CAAA;QAC5C,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,CAAA;IACxB,CAAC;IACD,OAAO,OAAO,CAAA;AAChB,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAc,EAAE,IAAuB;IACnE,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,EAAE,IAAI,CAAC,CAAA;IACtC,OAAO,OAAO,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAA;AACjD,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAc,EAAE,IAAuB;IACnE,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,EAAE,IAAI,CAAC,CAAA;IACtC,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAA;AAC3E,CAAC;AAED,MAAM,UAAU,YAAY,CAAC,KAAc,EAAE,IAAuB;IAClE,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,EAAE,IAAI,CAAC,CAAA;IACtC,OAAO,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAA;AAC1C,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAAE,OAAO,EAAE,CAAA;IACpC,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAmB,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAA;AAC9D,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,MAAkB,EAAE,GAAW;IAC7D,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,CAAA;IACzB,OAAO,OAAO,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAA;AACjD,CAAC"}