@vakra-dev/reader-js 0.2.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.
package/dist/index.js ADDED
@@ -0,0 +1,460 @@
1
+ // src/errors.ts
2
+ var ReaderApiError = class extends Error {
3
+ code;
4
+ httpStatus;
5
+ details;
6
+ docsUrl;
7
+ requestId;
8
+ constructor(body, httpStatus, requestId) {
9
+ super(body.message);
10
+ this.name = "ReaderApiError";
11
+ this.code = body.code;
12
+ this.httpStatus = httpStatus;
13
+ this.details = body.details;
14
+ this.docsUrl = body.docsUrl;
15
+ this.requestId = requestId;
16
+ }
17
+ };
18
+ var InvalidRequestError = class extends ReaderApiError {
19
+ constructor(body, status, requestId) {
20
+ super(body, status, requestId);
21
+ this.name = "InvalidRequestError";
22
+ }
23
+ };
24
+ var UnauthenticatedError = class extends ReaderApiError {
25
+ constructor(body, status, requestId) {
26
+ super(body, status, requestId);
27
+ this.name = "UnauthenticatedError";
28
+ }
29
+ };
30
+ var InsufficientCreditsError = class extends ReaderApiError {
31
+ required;
32
+ available;
33
+ resetAt;
34
+ constructor(body, status, requestId) {
35
+ super(body, status, requestId);
36
+ this.name = "InsufficientCreditsError";
37
+ this.required = body.details?.required;
38
+ this.available = body.details?.available;
39
+ this.resetAt = body.details?.resetAt;
40
+ }
41
+ };
42
+ var UrlBlockedError = class extends ReaderApiError {
43
+ url;
44
+ reason;
45
+ constructor(body, status, requestId) {
46
+ super(body, status, requestId);
47
+ this.name = "UrlBlockedError";
48
+ this.url = body.details?.url;
49
+ this.reason = body.details?.reason;
50
+ }
51
+ };
52
+ var NotFoundError = class extends ReaderApiError {
53
+ constructor(body, status, requestId) {
54
+ super(body, status, requestId);
55
+ this.name = "NotFoundError";
56
+ }
57
+ };
58
+ var ConflictError = class extends ReaderApiError {
59
+ constructor(body, status, requestId) {
60
+ super(body, status, requestId);
61
+ this.name = "ConflictError";
62
+ }
63
+ };
64
+ var RateLimitedError = class extends ReaderApiError {
65
+ retryAfterSeconds;
66
+ limit;
67
+ windowSeconds;
68
+ constructor(body, status, requestId) {
69
+ super(body, status, requestId);
70
+ this.name = "RateLimitedError";
71
+ this.retryAfterSeconds = body.details?.retryAfterSeconds;
72
+ this.limit = body.details?.limit;
73
+ this.windowSeconds = body.details?.windowSeconds;
74
+ }
75
+ };
76
+ var ConcurrencyLimitedError = class extends ReaderApiError {
77
+ active;
78
+ max;
79
+ constructor(body, status, requestId) {
80
+ super(body, status, requestId);
81
+ this.name = "ConcurrencyLimitedError";
82
+ this.active = body.details?.active;
83
+ this.max = body.details?.max;
84
+ }
85
+ };
86
+ var InternalServerError = class extends ReaderApiError {
87
+ constructor(body, status, requestId) {
88
+ super(body, status, requestId);
89
+ this.name = "InternalServerError";
90
+ }
91
+ };
92
+ var UpstreamUnavailableError = class extends ReaderApiError {
93
+ constructor(body, status, requestId) {
94
+ super(body, status, requestId);
95
+ this.name = "UpstreamUnavailableError";
96
+ }
97
+ };
98
+ var ScrapeTimeoutError = class extends ReaderApiError {
99
+ timeoutMs;
100
+ constructor(body, status, requestId) {
101
+ super(body, status, requestId);
102
+ this.name = "ScrapeTimeoutError";
103
+ this.timeoutMs = body.details?.timeoutMs;
104
+ }
105
+ };
106
+ function toReaderApiError(body, httpStatus, requestId) {
107
+ switch (body.code) {
108
+ case "invalid_request":
109
+ return new InvalidRequestError(body, httpStatus, requestId);
110
+ case "unauthenticated":
111
+ return new UnauthenticatedError(body, httpStatus, requestId);
112
+ case "insufficient_credits":
113
+ return new InsufficientCreditsError(body, httpStatus, requestId);
114
+ case "url_blocked":
115
+ return new UrlBlockedError(body, httpStatus, requestId);
116
+ case "not_found":
117
+ return new NotFoundError(body, httpStatus, requestId);
118
+ case "conflict":
119
+ return new ConflictError(body, httpStatus, requestId);
120
+ case "rate_limited":
121
+ return new RateLimitedError(body, httpStatus, requestId);
122
+ case "concurrency_limited":
123
+ return new ConcurrencyLimitedError(body, httpStatus, requestId);
124
+ case "internal_error":
125
+ return new InternalServerError(body, httpStatus, requestId);
126
+ case "upstream_unavailable":
127
+ return new UpstreamUnavailableError(body, httpStatus, requestId);
128
+ case "scrape_timeout":
129
+ return new ScrapeTimeoutError(body, httpStatus, requestId);
130
+ default:
131
+ return new ReaderApiError(body, httpStatus, requestId);
132
+ }
133
+ }
134
+
135
+ // src/client.ts
136
+ var DEFAULT_BASE_URL = "https://api.reader.dev";
137
+ var DEFAULT_TIMEOUT = 6e4;
138
+ var DEFAULT_MAX_RETRIES = 2;
139
+ var DEFAULT_POLL_INTERVAL = 2e3;
140
+ var DEFAULT_POLL_TIMEOUT = 3e5;
141
+ var ReaderClient = class {
142
+ apiKey;
143
+ baseUrl;
144
+ timeout;
145
+ maxRetries;
146
+ extraHeaders;
147
+ _sessions = null;
148
+ constructor(config) {
149
+ if (!config.apiKey) {
150
+ throw new Error("API key is required");
151
+ }
152
+ this.apiKey = config.apiKey;
153
+ this.baseUrl = (config.baseUrl || DEFAULT_BASE_URL).replace(/\/$/, "");
154
+ this.timeout = config.timeout || DEFAULT_TIMEOUT;
155
+ this.maxRetries = config.maxRetries ?? DEFAULT_MAX_RETRIES;
156
+ this.extraHeaders = config.headers || {};
157
+ }
158
+ /**
159
+ * Browser sessions API.
160
+ *
161
+ * @example
162
+ * ```typescript
163
+ * const session = await client.sessions.create();
164
+ * const browser = await chromium.connectOverCDP(session.wsEndpoint);
165
+ * // ... use Playwright ...
166
+ * await client.sessions.stop(session.sessionId);
167
+ * ```
168
+ */
169
+ get sessions() {
170
+ if (!this._sessions) {
171
+ this._sessions = new SessionsAPI(this.request.bind(this));
172
+ }
173
+ return this._sessions;
174
+ }
175
+ /**
176
+ * Read (scrape, batch, or crawl) one or more URLs.
177
+ *
178
+ * - Single URL → sync scrape, returns immediately with `{ kind: "scrape", data }`
179
+ * - Multiple URLs or URL + maxDepth/maxPages → async job; this method polls
180
+ * until the job terminates and returns `{ kind: "job", data }`.
181
+ */
182
+ async read(params) {
183
+ const envelope = await this.request(
184
+ "POST",
185
+ "/v1/read",
186
+ params
187
+ );
188
+ const data = envelope.data;
189
+ if (data && typeof data === "object" && "status" in data && "mode" in data && !("markdown" in data) && !("metadata" in data)) {
190
+ const jobId = String(data.id);
191
+ const job = await this.waitForJob(jobId);
192
+ return { kind: "job", data: job };
193
+ }
194
+ return { kind: "scrape", data };
195
+ }
196
+ /**
197
+ * Get job status and a single page of results.
198
+ */
199
+ async getJob(jobId, opts) {
200
+ const query = new URLSearchParams();
201
+ if (opts?.skip !== void 0) query.set("skip", String(opts.skip));
202
+ if (opts?.limit !== void 0) query.set("limit", String(opts.limit));
203
+ const qs = query.toString();
204
+ const envelope = await this.request(
205
+ "GET",
206
+ `/v1/jobs/${jobId}${qs ? `?${qs}` : ""}`
207
+ );
208
+ return {
209
+ job: envelope.data,
210
+ hasMore: envelope.pagination.hasMore,
211
+ next: envelope.pagination.next
212
+ };
213
+ }
214
+ /**
215
+ * Fetch all job result pages by following pagination.
216
+ */
217
+ async getAllJobResults(jobId) {
218
+ const pages = [];
219
+ let skip = 0;
220
+ const limit = 100;
221
+ while (true) {
222
+ const { job, hasMore } = await this.getJob(jobId, { skip, limit });
223
+ pages.push(...job.results ?? []);
224
+ if (!hasMore) break;
225
+ skip += limit;
226
+ }
227
+ return pages;
228
+ }
229
+ /**
230
+ * Cancel a job. Throws `ConflictError` if the job is already terminal.
231
+ */
232
+ async cancelJob(jobId) {
233
+ await this.request("DELETE", `/v1/jobs/${jobId}`);
234
+ }
235
+ /**
236
+ * Retry the failed URLs in a job. Throws `InvalidRequestError` if no
237
+ * failed URLs exist.
238
+ */
239
+ async retryJob(jobId) {
240
+ const envelope = await this.request("POST", `/v1/jobs/${jobId}/retry`);
241
+ return envelope.data;
242
+ }
243
+ /**
244
+ * Poll a job until it completes, fails, or is cancelled. Collects all
245
+ * paginated results when complete.
246
+ */
247
+ async waitForJob(jobId, options) {
248
+ const interval = options?.pollInterval ?? DEFAULT_POLL_INTERVAL;
249
+ const timeout = options?.timeout ?? DEFAULT_POLL_TIMEOUT;
250
+ const start = Date.now();
251
+ while (Date.now() - start < timeout) {
252
+ const { job } = await this.getJob(jobId, { limit: 1 });
253
+ if (job.status === "completed" || job.status === "failed" || job.status === "cancelled") {
254
+ if (job.status === "completed") {
255
+ job.results = await this.getAllJobResults(jobId);
256
+ }
257
+ return job;
258
+ }
259
+ await sleep(interval);
260
+ }
261
+ throw new ScrapeTimeoutError(
262
+ {
263
+ code: "scrape_timeout",
264
+ message: `Job ${jobId} polling timed out after ${timeout}ms`,
265
+ details: { timeoutMs: timeout }
266
+ },
267
+ 504
268
+ );
269
+ }
270
+ /**
271
+ * Stream job results as they arrive via polling.
272
+ *
273
+ * @example
274
+ * for await (const event of client.stream(jobId)) {
275
+ * if (event.type === "page") console.log(event.data.url);
276
+ * if (event.type === "done") break;
277
+ * }
278
+ */
279
+ async *stream(jobId, options) {
280
+ const interval = options?.pollInterval ?? DEFAULT_POLL_INTERVAL;
281
+ const timeout = options?.timeout ?? DEFAULT_POLL_TIMEOUT;
282
+ const start = Date.now();
283
+ let lastCompleted = 0;
284
+ while (Date.now() - start < timeout) {
285
+ const { job } = await this.getJob(jobId, { skip: lastCompleted, limit: 100 });
286
+ yield {
287
+ type: "progress",
288
+ completed: job.completed,
289
+ total: job.total,
290
+ status: job.status
291
+ };
292
+ for (const page of job.results ?? []) {
293
+ if (page.error) {
294
+ yield { type: "error", url: page.url, error: page.error };
295
+ } else {
296
+ yield { type: "page", data: page };
297
+ }
298
+ lastCompleted += 1;
299
+ }
300
+ if (job.status === "completed" || job.status === "failed" || job.status === "cancelled") {
301
+ yield {
302
+ type: "done",
303
+ completed: job.completed,
304
+ total: job.total,
305
+ status: job.status
306
+ };
307
+ return;
308
+ }
309
+ await sleep(interval);
310
+ }
311
+ throw new ScrapeTimeoutError(
312
+ {
313
+ code: "scrape_timeout",
314
+ message: `Job ${jobId} stream timed out`,
315
+ details: { timeoutMs: timeout }
316
+ },
317
+ 504
318
+ );
319
+ }
320
+ /**
321
+ * Get the current credit balance for this workspace.
322
+ */
323
+ async getCredits() {
324
+ const envelope = await this.request("GET", "/v1/usage/credits");
325
+ return envelope.data;
326
+ }
327
+ // --- Internal ---
328
+ async request(method, path, body) {
329
+ const url = path.startsWith("http") ? path : `${this.baseUrl}${path}`;
330
+ let lastError = null;
331
+ for (let attempt = 0; attempt <= this.maxRetries; attempt++) {
332
+ try {
333
+ const controller = new AbortController();
334
+ const timeoutId = setTimeout(() => controller.abort(), this.timeout);
335
+ const res = await fetch(url, {
336
+ method,
337
+ headers: {
338
+ "Content-Type": "application/json",
339
+ "x-api-key": this.apiKey,
340
+ ...this.extraHeaders
341
+ },
342
+ body: body ? JSON.stringify(body) : void 0,
343
+ signal: controller.signal
344
+ });
345
+ clearTimeout(timeoutId);
346
+ const requestId = res.headers.get("x-request-id") ?? void 0;
347
+ const parsed = await res.json().catch(() => null);
348
+ if (!res.ok) {
349
+ if (parsed && "error" in parsed && parsed.error) {
350
+ const err = toReaderApiError(parsed.error, res.status, requestId);
351
+ if (res.status < 500 && res.status !== 429) throw err;
352
+ if (err instanceof RateLimitedError && err.retryAfterSeconds) {
353
+ await sleep(err.retryAfterSeconds * 1e3);
354
+ }
355
+ lastError = err;
356
+ } else {
357
+ const genericErr = new ReaderApiError(
358
+ {
359
+ code: "internal_error",
360
+ message: `Request failed with status ${res.status}`
361
+ },
362
+ res.status,
363
+ requestId
364
+ );
365
+ if (res.status < 500) throw genericErr;
366
+ lastError = genericErr;
367
+ }
368
+ } else {
369
+ return parsed;
370
+ }
371
+ } catch (err) {
372
+ if (err instanceof ReaderApiError) {
373
+ if (err.httpStatus < 500 && err.httpStatus !== 429) throw err;
374
+ lastError = err;
375
+ } else if (err instanceof Error) {
376
+ if (err.name === "AbortError") {
377
+ lastError = new ReaderApiError(
378
+ { code: "scrape_timeout", message: "Request timed out" },
379
+ 504
380
+ );
381
+ } else {
382
+ lastError = err;
383
+ }
384
+ }
385
+ }
386
+ if (attempt < this.maxRetries) {
387
+ await sleep(Math.pow(2, attempt) * 1e3);
388
+ }
389
+ }
390
+ throw lastError ?? new ReaderApiError({ code: "internal_error", message: "Request failed" }, 500);
391
+ }
392
+ };
393
+ var SessionsAPI = class {
394
+ constructor(request) {
395
+ this.request = request;
396
+ }
397
+ request;
398
+ /**
399
+ * Create a browser session. Returns a CDP WebSocket URL for
400
+ * Playwright/Puppeteer connection.
401
+ */
402
+ async create(params) {
403
+ const envelope = await this.request(
404
+ "POST",
405
+ "/v1/sessions",
406
+ params ?? {}
407
+ );
408
+ return envelope.data;
409
+ }
410
+ /**
411
+ * Get session status.
412
+ */
413
+ async get(sessionId) {
414
+ const envelope = await this.request(
415
+ "GET",
416
+ `/v1/sessions/${sessionId}`
417
+ );
418
+ return envelope.data;
419
+ }
420
+ /**
421
+ * Stop a browser session.
422
+ */
423
+ async stop(sessionId) {
424
+ const envelope = await this.request(
425
+ "DELETE",
426
+ `/v1/sessions/${sessionId}`
427
+ );
428
+ return envelope.data;
429
+ }
430
+ /**
431
+ * List active sessions.
432
+ */
433
+ async list() {
434
+ const envelope = await this.request(
435
+ "GET",
436
+ "/v1/sessions"
437
+ );
438
+ return envelope.data;
439
+ }
440
+ };
441
+ function sleep(ms) {
442
+ return new Promise((resolve) => setTimeout(resolve, ms));
443
+ }
444
+ export {
445
+ ConcurrencyLimitedError,
446
+ ConflictError,
447
+ InsufficientCreditsError,
448
+ InternalServerError,
449
+ InvalidRequestError,
450
+ NotFoundError,
451
+ RateLimitedError,
452
+ ReaderApiError,
453
+ ReaderClient,
454
+ ScrapeTimeoutError,
455
+ UnauthenticatedError,
456
+ UpstreamUnavailableError,
457
+ UrlBlockedError,
458
+ toReaderApiError
459
+ };
460
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/errors.ts","../src/client.ts"],"sourcesContent":["/**\n * Typed error classes mirroring the reader-api error code catalog.\n *\n * The API returns a stable `code` field on every error response. The SDK\n * branches on that code and throws a specific subclass, so callers can\n * write:\n *\n * try {\n * await client.read({ url });\n * } catch (err) {\n * if (err instanceof InsufficientCreditsError) {\n * // err.required, err.available, err.resetAt\n * }\n * }\n *\n * There is one subclass per code in the catalog. Unknown codes fall through\n * to the base `ReaderApiError`.\n */\n\nexport type ReaderErrorCode =\n | \"invalid_request\"\n | \"unauthenticated\"\n | \"insufficient_credits\"\n | \"url_blocked\"\n | \"not_found\"\n | \"conflict\"\n | \"rate_limited\"\n | \"concurrency_limited\"\n | \"internal_error\"\n | \"upstream_unavailable\"\n | \"scrape_timeout\";\n\nexport interface ApiErrorBody {\n code: ReaderErrorCode | string;\n message: string;\n details?: Record<string, unknown>;\n docsUrl?: string;\n}\n\nexport class ReaderApiError extends Error {\n readonly code: string;\n readonly httpStatus: number;\n readonly details?: Record<string, unknown>;\n readonly docsUrl?: string;\n readonly requestId?: string;\n\n constructor(body: ApiErrorBody, httpStatus: number, requestId?: string) {\n super(body.message);\n this.name = \"ReaderApiError\";\n this.code = body.code;\n this.httpStatus = httpStatus;\n this.details = body.details;\n this.docsUrl = body.docsUrl;\n this.requestId = requestId;\n }\n}\n\nexport class InvalidRequestError extends ReaderApiError {\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"InvalidRequestError\";\n }\n}\n\nexport class UnauthenticatedError extends ReaderApiError {\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"UnauthenticatedError\";\n }\n}\n\nexport class InsufficientCreditsError extends ReaderApiError {\n readonly required?: number;\n readonly available?: number;\n readonly resetAt?: string;\n\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"InsufficientCreditsError\";\n this.required = body.details?.required as number | undefined;\n this.available = body.details?.available as number | undefined;\n this.resetAt = body.details?.resetAt as string | undefined;\n }\n}\n\nexport class UrlBlockedError extends ReaderApiError {\n readonly url?: string;\n readonly reason?: string;\n\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"UrlBlockedError\";\n this.url = body.details?.url as string | undefined;\n this.reason = body.details?.reason as string | undefined;\n }\n}\n\nexport class NotFoundError extends ReaderApiError {\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"NotFoundError\";\n }\n}\n\nexport class ConflictError extends ReaderApiError {\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"ConflictError\";\n }\n}\n\nexport class RateLimitedError extends ReaderApiError {\n readonly retryAfterSeconds?: number;\n readonly limit?: number;\n readonly windowSeconds?: number;\n\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"RateLimitedError\";\n this.retryAfterSeconds = body.details?.retryAfterSeconds as number | undefined;\n this.limit = body.details?.limit as number | undefined;\n this.windowSeconds = body.details?.windowSeconds as number | undefined;\n }\n}\n\nexport class ConcurrencyLimitedError extends ReaderApiError {\n readonly active?: number;\n readonly max?: number;\n\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"ConcurrencyLimitedError\";\n this.active = body.details?.active as number | undefined;\n this.max = body.details?.max as number | undefined;\n }\n}\n\nexport class InternalServerError extends ReaderApiError {\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"InternalServerError\";\n }\n}\n\nexport class UpstreamUnavailableError extends ReaderApiError {\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"UpstreamUnavailableError\";\n }\n}\n\nexport class ScrapeTimeoutError extends ReaderApiError {\n readonly timeoutMs?: number;\n\n constructor(body: ApiErrorBody, status: number, requestId?: string) {\n super(body, status, requestId);\n this.name = \"ScrapeTimeoutError\";\n this.timeoutMs = body.details?.timeoutMs as number | undefined;\n }\n}\n\n/**\n * Construct the right error subclass from an error response body.\n * Unknown codes fall through to the base class.\n */\nexport function toReaderApiError(\n body: ApiErrorBody,\n httpStatus: number,\n requestId?: string,\n): ReaderApiError {\n switch (body.code) {\n case \"invalid_request\":\n return new InvalidRequestError(body, httpStatus, requestId);\n case \"unauthenticated\":\n return new UnauthenticatedError(body, httpStatus, requestId);\n case \"insufficient_credits\":\n return new InsufficientCreditsError(body, httpStatus, requestId);\n case \"url_blocked\":\n return new UrlBlockedError(body, httpStatus, requestId);\n case \"not_found\":\n return new NotFoundError(body, httpStatus, requestId);\n case \"conflict\":\n return new ConflictError(body, httpStatus, requestId);\n case \"rate_limited\":\n return new RateLimitedError(body, httpStatus, requestId);\n case \"concurrency_limited\":\n return new ConcurrencyLimitedError(body, httpStatus, requestId);\n case \"internal_error\":\n return new InternalServerError(body, httpStatus, requestId);\n case \"upstream_unavailable\":\n return new UpstreamUnavailableError(body, httpStatus, requestId);\n case \"scrape_timeout\":\n return new ScrapeTimeoutError(body, httpStatus, requestId);\n default:\n return new ReaderApiError(body, httpStatus, requestId);\n }\n}\n","/**\n * Reader SDK Client\n *\n * @example\n * import { ReaderClient } from \"@vakra-dev/reader-js\";\n *\n * const client = new ReaderClient({ apiKey: \"rdr_your_key\" });\n *\n * // Synchronous scrape (single URL)\n * const result = await client.read({ url: \"https://example.com\" });\n * if (result.kind === \"scrape\") {\n * console.log(result.data.markdown);\n * }\n *\n * // Batch (returns a completed Job with all results collected)\n * const batch = await client.read({ urls: [\"url1\", \"url2\"] });\n * if (batch.kind === \"job\") {\n * for (const page of batch.data.results) {\n * console.log(page.url, page.markdown?.length);\n * }\n * }\n */\n\nimport type {\n ReaderClientConfig,\n ReadParams,\n ReadResult,\n ScrapeResult,\n Job,\n Credits,\n Page,\n StreamEvent,\n SuccessEnvelope,\n PaginatedEnvelope,\n ErrorEnvelope,\n SessionInfo,\n CreateSessionParams,\n StopSessionResult,\n} from \"./types.js\";\nimport {\n toReaderApiError,\n ReaderApiError,\n ScrapeTimeoutError,\n RateLimitedError,\n} from \"./errors.js\";\n\nconst DEFAULT_BASE_URL = \"https://api.reader.dev\";\nconst DEFAULT_TIMEOUT = 60_000;\nconst DEFAULT_MAX_RETRIES = 2;\nconst DEFAULT_POLL_INTERVAL = 2_000;\nconst DEFAULT_POLL_TIMEOUT = 300_000; // 5 minutes\n\ninterface JobWithPagination {\n data: Job;\n pagination: { total: number; skip: number; limit: number; hasMore: boolean; next?: string };\n}\n\nexport class ReaderClient {\n private apiKey: string;\n private baseUrl: string;\n private timeout: number;\n private maxRetries: number;\n private extraHeaders: Record<string, string>;\n private _sessions: SessionsAPI | null = null;\n\n constructor(config: ReaderClientConfig) {\n if (!config.apiKey) {\n throw new Error(\"API key is required\");\n }\n this.apiKey = config.apiKey;\n this.baseUrl = (config.baseUrl || DEFAULT_BASE_URL).replace(/\\/$/, \"\");\n this.timeout = config.timeout || DEFAULT_TIMEOUT;\n this.maxRetries = config.maxRetries ?? DEFAULT_MAX_RETRIES;\n this.extraHeaders = config.headers || {};\n }\n\n /**\n * Browser sessions API.\n *\n * @example\n * ```typescript\n * const session = await client.sessions.create();\n * const browser = await chromium.connectOverCDP(session.wsEndpoint);\n * // ... use Playwright ...\n * await client.sessions.stop(session.sessionId);\n * ```\n */\n get sessions(): SessionsAPI {\n if (!this._sessions) {\n this._sessions = new SessionsAPI(this.request.bind(this));\n }\n return this._sessions;\n }\n\n /**\n * Read (scrape, batch, or crawl) one or more URLs.\n *\n * - Single URL → sync scrape, returns immediately with `{ kind: \"scrape\", data }`\n * - Multiple URLs or URL + maxDepth/maxPages → async job; this method polls\n * until the job terminates and returns `{ kind: \"job\", data }`.\n */\n async read(params: ReadParams): Promise<ReadResult> {\n const envelope = await this.request<SuccessEnvelope<unknown>>(\n \"POST\",\n \"/v1/read\",\n params,\n );\n\n const data = envelope.data as Record<string, unknown>;\n\n // Async job response: data.id + data.status present, no markdown/html/metadata\n if (\n data &&\n typeof data === \"object\" &&\n \"status\" in data &&\n \"mode\" in data &&\n !(\"markdown\" in data) &&\n !(\"metadata\" in data)\n ) {\n const jobId = String((data as { id: unknown }).id);\n const job = await this.waitForJob(jobId);\n return { kind: \"job\", data: job };\n }\n\n // Synchronous scrape: data has markdown/html/metadata\n return { kind: \"scrape\", data: data as unknown as ScrapeResult };\n }\n\n /**\n * Get job status and a single page of results.\n */\n async getJob(\n jobId: string,\n opts?: { skip?: number; limit?: number },\n ): Promise<{ job: Job; hasMore: boolean; next?: string }> {\n const query = new URLSearchParams();\n if (opts?.skip !== undefined) query.set(\"skip\", String(opts.skip));\n if (opts?.limit !== undefined) query.set(\"limit\", String(opts.limit));\n const qs = query.toString();\n\n const envelope = await this.request<JobWithPagination>(\n \"GET\",\n `/v1/jobs/${jobId}${qs ? `?${qs}` : \"\"}`,\n );\n\n return {\n job: envelope.data,\n hasMore: envelope.pagination.hasMore,\n next: envelope.pagination.next,\n };\n }\n\n /**\n * Fetch all job result pages by following pagination.\n */\n async getAllJobResults(jobId: string): Promise<Page[]> {\n const pages: Page[] = [];\n let skip = 0;\n const limit = 100;\n\n while (true) {\n const { job, hasMore } = await this.getJob(jobId, { skip, limit });\n pages.push(...(job.results ?? []));\n if (!hasMore) break;\n skip += limit;\n }\n\n return pages;\n }\n\n /**\n * Cancel a job. Throws `ConflictError` if the job is already terminal.\n */\n async cancelJob(jobId: string): Promise<void> {\n await this.request(\"DELETE\", `/v1/jobs/${jobId}`);\n }\n\n /**\n * Retry the failed URLs in a job. Throws `InvalidRequestError` if no\n * failed URLs exist.\n */\n async retryJob(jobId: string): Promise<{ id: string; status: string; retrying: number }> {\n const envelope = await this.request<\n SuccessEnvelope<{ id: string; status: string; retrying: number }>\n >(\"POST\", `/v1/jobs/${jobId}/retry`);\n return envelope.data;\n }\n\n /**\n * Poll a job until it completes, fails, or is cancelled. Collects all\n * paginated results when complete.\n */\n async waitForJob(\n jobId: string,\n options?: { pollInterval?: number; timeout?: number },\n ): Promise<Job> {\n const interval = options?.pollInterval ?? DEFAULT_POLL_INTERVAL;\n const timeout = options?.timeout ?? DEFAULT_POLL_TIMEOUT;\n const start = Date.now();\n\n while (Date.now() - start < timeout) {\n const { job } = await this.getJob(jobId, { limit: 1 });\n\n if (\n job.status === \"completed\" ||\n job.status === \"failed\" ||\n job.status === \"cancelled\"\n ) {\n if (job.status === \"completed\") {\n job.results = await this.getAllJobResults(jobId);\n }\n return job;\n }\n\n await sleep(interval);\n }\n\n throw new ScrapeTimeoutError(\n {\n code: \"scrape_timeout\",\n message: `Job ${jobId} polling timed out after ${timeout}ms`,\n details: { timeoutMs: timeout },\n },\n 504,\n );\n }\n\n /**\n * Stream job results as they arrive via polling.\n *\n * @example\n * for await (const event of client.stream(jobId)) {\n * if (event.type === \"page\") console.log(event.data.url);\n * if (event.type === \"done\") break;\n * }\n */\n async *stream(\n jobId: string,\n options?: { pollInterval?: number; timeout?: number },\n ): AsyncGenerator<StreamEvent> {\n const interval = options?.pollInterval ?? DEFAULT_POLL_INTERVAL;\n const timeout = options?.timeout ?? DEFAULT_POLL_TIMEOUT;\n const start = Date.now();\n let lastCompleted = 0;\n\n while (Date.now() - start < timeout) {\n const { job } = await this.getJob(jobId, { skip: lastCompleted, limit: 100 });\n\n yield {\n type: \"progress\",\n completed: job.completed,\n total: job.total,\n status: job.status,\n };\n\n for (const page of job.results ?? []) {\n if (page.error) {\n yield { type: \"error\", url: page.url, error: page.error };\n } else {\n yield { type: \"page\", data: page };\n }\n lastCompleted += 1;\n }\n\n if (\n job.status === \"completed\" ||\n job.status === \"failed\" ||\n job.status === \"cancelled\"\n ) {\n yield {\n type: \"done\",\n completed: job.completed,\n total: job.total,\n status: job.status,\n };\n return;\n }\n\n await sleep(interval);\n }\n\n throw new ScrapeTimeoutError(\n {\n code: \"scrape_timeout\",\n message: `Job ${jobId} stream timed out`,\n details: { timeoutMs: timeout },\n },\n 504,\n );\n }\n\n /**\n * Get the current credit balance for this workspace.\n */\n async getCredits(): Promise<Credits> {\n const envelope = await this.request<SuccessEnvelope<Credits>>(\"GET\", \"/v1/usage/credits\");\n return envelope.data;\n }\n\n // --- Internal ---\n\n private async request<T>(method: string, path: string, body?: unknown): Promise<T> {\n const url = path.startsWith(\"http\") ? path : `${this.baseUrl}${path}`;\n let lastError: Error | null = null;\n\n for (let attempt = 0; attempt <= this.maxRetries; attempt++) {\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), this.timeout);\n\n const res = await fetch(url, {\n method,\n headers: {\n \"Content-Type\": \"application/json\",\n \"x-api-key\": this.apiKey,\n ...this.extraHeaders,\n },\n body: body ? JSON.stringify(body) : undefined,\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const requestId = res.headers.get(\"x-request-id\") ?? undefined;\n const parsed = (await res.json().catch(() => null)) as\n | SuccessEnvelope<unknown>\n | PaginatedEnvelope<unknown>\n | ErrorEnvelope\n | null;\n\n if (!res.ok) {\n if (parsed && \"error\" in parsed && parsed.error) {\n const err = toReaderApiError(parsed.error, res.status, requestId);\n\n // Don't retry client errors except 429\n if (res.status < 500 && res.status !== 429) throw err;\n\n // Honor Retry-After from the rate-limited response\n if (err instanceof RateLimitedError && err.retryAfterSeconds) {\n await sleep(err.retryAfterSeconds * 1000);\n }\n\n lastError = err;\n } else {\n const genericErr = new ReaderApiError(\n {\n code: \"internal_error\",\n message: `Request failed with status ${res.status}`,\n },\n res.status,\n requestId,\n );\n if (res.status < 500) throw genericErr;\n lastError = genericErr;\n }\n } else {\n return parsed as unknown as T;\n }\n } catch (err) {\n if (err instanceof ReaderApiError) {\n if (err.httpStatus < 500 && err.httpStatus !== 429) throw err;\n lastError = err;\n } else if (err instanceof Error) {\n if (err.name === \"AbortError\") {\n lastError = new ReaderApiError(\n { code: \"scrape_timeout\", message: \"Request timed out\" },\n 504,\n );\n } else {\n lastError = err;\n }\n }\n }\n\n // Exponential backoff before retry\n if (attempt < this.maxRetries) {\n await sleep(Math.pow(2, attempt) * 1000);\n }\n }\n\n throw (\n lastError ??\n new ReaderApiError({ code: \"internal_error\", message: \"Request failed\" }, 500)\n );\n }\n}\n\n// ─── Sessions API ────────────────────────────────────────────────────\n\ntype RequestFn = <T>(method: string, path: string, body?: unknown) => Promise<T>;\n\nclass SessionsAPI {\n constructor(private request: RequestFn) {}\n\n /**\n * Create a browser session. Returns a CDP WebSocket URL for\n * Playwright/Puppeteer connection.\n */\n async create(params?: CreateSessionParams): Promise<SessionInfo> {\n const envelope = await this.request<SuccessEnvelope<SessionInfo>>(\n \"POST\",\n \"/v1/sessions\",\n params ?? {},\n );\n return envelope.data;\n }\n\n /**\n * Get session status.\n */\n async get(sessionId: string): Promise<SessionInfo> {\n const envelope = await this.request<SuccessEnvelope<SessionInfo>>(\n \"GET\",\n `/v1/sessions/${sessionId}`,\n );\n return envelope.data;\n }\n\n /**\n * Stop a browser session.\n */\n async stop(sessionId: string): Promise<StopSessionResult> {\n const envelope = await this.request<SuccessEnvelope<StopSessionResult>>(\n \"DELETE\",\n `/v1/sessions/${sessionId}`,\n );\n return envelope.data;\n }\n\n /**\n * List active sessions.\n */\n async list(): Promise<SessionInfo[]> {\n const envelope = await this.request<SuccessEnvelope<SessionInfo[]>>(\n \"GET\",\n \"/v1/sessions\",\n );\n return envelope.data;\n }\n}\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n"],"mappings":"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","names":[]}
package/package.json ADDED
@@ -0,0 +1,42 @@
1
+ {
2
+ "name": "@vakra-dev/reader-js",
3
+ "version": "0.2.0",
4
+ "description": "JavaScript/TypeScript SDK for the Reader API",
5
+ "type": "module",
6
+ "main": "./dist/index.cjs",
7
+ "module": "./dist/index.js",
8
+ "types": "./dist/index.d.ts",
9
+ "exports": {
10
+ ".": {
11
+ "types": "./dist/index.d.ts",
12
+ "import": "./dist/index.js",
13
+ "require": "./dist/index.cjs"
14
+ }
15
+ },
16
+ "files": [
17
+ "dist",
18
+ "README.md"
19
+ ],
20
+ "scripts": {
21
+ "build": "tsup",
22
+ "dev": "tsup --watch",
23
+ "test": "vitest run",
24
+ "typecheck": "tsc --noEmit"
25
+ },
26
+ "keywords": [
27
+ "reader",
28
+ "scraper",
29
+ "web-scraping",
30
+ "markdown",
31
+ "llm",
32
+ "sdk"
33
+ ],
34
+ "author": "Vakra Dev",
35
+ "license": "MIT",
36
+ "dependencies": {},
37
+ "devDependencies": {
38
+ "tsup": "^8.3.6",
39
+ "typescript": "^5.7.3",
40
+ "vitest": "^4.1.0"
41
+ }
42
+ }