@brianbuie/node-kit 0.9.1 → 0.9.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.mts","names":[],"sources":["../src/File.ts","../src/Dir.ts","../src/Cache.ts","../src/Fetcher.ts","../src/Log.ts","../src/snapshot.ts","../src/timeout.ts","../src/TypeWriter.ts"],"sourcesContent":[],"mappings":"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"}
package/dist/index.mjs ADDED
@@ -0,0 +1,562 @@
1
+ import * as fs from "node:fs";
2
+ import * as path from "node:path";
3
+ import sanitizeFilename from "sanitize-filename";
4
+ import { Readable } from "node:stream";
5
+ import { finished } from "node:stream/promises";
6
+ import { parseStream, writeToStream } from "fast-csv";
7
+ import { isObjectLike, merge } from "lodash-es";
8
+ import { add, isAfter } from "date-fns";
9
+ import extractDomain from "extract-domain";
10
+ import { inspect } from "node:util";
11
+ import chalk from "chalk";
12
+ import * as qt from "quicktype-core";
13
+
14
+ //#region src/snapshot.ts
15
+ /**
16
+ * Allows special objects (Error, Headers, Set) to be included in JSON.stringify output
17
+ * functions are removed
18
+ */
19
+ function snapshot(i, max = 50, depth = 0) {
20
+ if (Array.isArray(i)) {
21
+ if (depth === max) return [];
22
+ return i.map((c) => snapshot(c, max, depth + 1));
23
+ }
24
+ if (typeof i === "function") return void 0;
25
+ if (!isObjectLike(i)) return i;
26
+ if (depth === max) return {};
27
+ let output = {};
28
+ if (typeof i.entries === "function") {
29
+ for (let [k, v] of i.entries()) output[k] = snapshot(v, max, depth + 1);
30
+ return output;
31
+ }
32
+ const obj = i;
33
+ for (let key in obj) output[key] = snapshot(obj[key], max, depth + 1);
34
+ Object.getOwnPropertyNames(obj).forEach((key) => {
35
+ output[key] = snapshot(obj[key], max, depth + 1);
36
+ });
37
+ return output;
38
+ }
39
+
40
+ //#endregion
41
+ //#region src/File.ts
42
+ /**
43
+ * WARNING: API will change!
44
+ */
45
+ var File = class {
46
+ path;
47
+ constructor(filepath) {
48
+ this.path = filepath;
49
+ }
50
+ get exists() {
51
+ return fs.existsSync(this.path);
52
+ }
53
+ delete() {
54
+ fs.rmSync(this.path, { force: true });
55
+ }
56
+ read() {
57
+ return this.exists ? fs.readFileSync(this.path, "utf8") : void 0;
58
+ }
59
+ /**
60
+ * @returns lines as strings, removes trailing '\n'
61
+ */
62
+ lines() {
63
+ const contents = (this.read() || "").split("\n");
64
+ return contents.slice(0, contents.length - 1);
65
+ }
66
+ get readStream() {
67
+ return this.exists ? fs.createReadStream(this.path) : Readable.from([]);
68
+ }
69
+ get writeStream() {
70
+ fs.mkdirSync(path.parse(this.path).dir, { recursive: true });
71
+ return fs.createWriteStream(this.path);
72
+ }
73
+ write(contents) {
74
+ fs.mkdirSync(path.parse(this.path).dir, { recursive: true });
75
+ if (typeof contents === "string") return fs.writeFileSync(this.path, contents);
76
+ if (contents instanceof ReadableStream) return finished(Readable.from(contents).pipe(this.writeStream));
77
+ throw new Error(`Invalid content type: ${typeof contents}`);
78
+ }
79
+ /**
80
+ * creates file if it doesn't exist, appends string or array of strings as new lines.
81
+ * File always ends with '\n', so contents don't need to be read before appending
82
+ */
83
+ append(lines) {
84
+ if (!this.exists) this.write("");
85
+ const contents = Array.isArray(lines) ? lines.join("\n") : lines;
86
+ fs.appendFileSync(this.path, contents + "\n");
87
+ }
88
+ static get FileType() {
89
+ return FileType;
90
+ }
91
+ json(contents) {
92
+ return new FileTypeJson(this.path, contents);
93
+ }
94
+ static get json() {
95
+ return FileTypeJson;
96
+ }
97
+ ndjson(lines) {
98
+ return new FileTypeNdjson(this.path, lines);
99
+ }
100
+ static get ndjson() {
101
+ return FileTypeNdjson;
102
+ }
103
+ async csv(rows, keys) {
104
+ const csvFile = new FileTypeCsv(this.path);
105
+ if (rows) await csvFile.write(rows, keys);
106
+ return csvFile;
107
+ }
108
+ static get csv() {
109
+ return FileTypeCsv;
110
+ }
111
+ };
112
+ /**
113
+ * A generic file adaptor, extended by specific file type implementations
114
+ */
115
+ var FileType = class {
116
+ file;
117
+ constructor(filepath, contents) {
118
+ this.file = new File(filepath);
119
+ if (contents) this.file.write(contents);
120
+ }
121
+ get exists() {
122
+ return this.file.exists;
123
+ }
124
+ get path() {
125
+ return this.file.path;
126
+ }
127
+ delete() {
128
+ this.file.delete();
129
+ }
130
+ };
131
+ /**
132
+ * A .json file that maintains data type when reading/writing.
133
+ * This is unsafe! Type is not checked at runtime, avoid using on files manipulated outside of your application.
134
+ */
135
+ var FileTypeJson = class extends FileType {
136
+ constructor(filepath, contents) {
137
+ super(filepath.endsWith(".json") ? filepath : filepath + ".json");
138
+ if (contents) this.write(contents);
139
+ }
140
+ read() {
141
+ const contents = this.file.read();
142
+ return contents ? JSON.parse(contents) : void 0;
143
+ }
144
+ write(contents) {
145
+ this.file.write(JSON.stringify(snapshot(contents), null, 2));
146
+ }
147
+ };
148
+ /**
149
+ * New-line delimited json file (.ndjson)
150
+ * @see https://jsonltools.com/ndjson-format-specification
151
+ */
152
+ var FileTypeNdjson = class extends FileType {
153
+ constructor(filepath, lines) {
154
+ super(filepath.endsWith(".ndjson") ? filepath : filepath + ".ndjson");
155
+ if (lines) this.append(lines);
156
+ }
157
+ append(lines) {
158
+ this.file.append(Array.isArray(lines) ? lines.map((l) => JSON.stringify(snapshot(l))) : JSON.stringify(snapshot(lines)));
159
+ }
160
+ lines() {
161
+ return this.file.lines().map((l) => JSON.parse(l));
162
+ }
163
+ };
164
+ /**
165
+ * Comma separated values (.csv).
166
+ * Input rows as objects, keys are used as column headers
167
+ */
168
+ var FileTypeCsv = class extends FileType {
169
+ constructor(filepath) {
170
+ super(filepath.endsWith(".csv") ? filepath : filepath + ".csv");
171
+ }
172
+ async write(rows, keys) {
173
+ const headerSet = /* @__PURE__ */ new Set();
174
+ if (keys) for (const key of keys) headerSet.add(key);
175
+ else for (const row of rows) for (const key in row) headerSet.add(key);
176
+ const headers = Array.from(headerSet);
177
+ const outRows = rows.map((row) => headers.map((key) => row[key]));
178
+ return finished(writeToStream(this.file.writeStream, [headers, ...outRows]));
179
+ }
180
+ #parseVal(val) {
181
+ if (val.toLowerCase() === "false") return false;
182
+ if (val.toLowerCase() === "true") return true;
183
+ if (val.length === 0) return null;
184
+ if (/^[\.0-9]+$/.test(val)) return Number(val);
185
+ return val;
186
+ }
187
+ async read() {
188
+ return new Promise((resolve, reject) => {
189
+ const parsed = [];
190
+ parseStream(this.file.readStream, { headers: true }).on("error", (e) => reject(e)).on("data", (raw) => {
191
+ parsed.push(Object.entries(raw).reduce((all, [key, val]) => ({
192
+ ...all,
193
+ [key]: this.#parseVal(val)
194
+ }), {}));
195
+ }).on("end", () => resolve(parsed));
196
+ });
197
+ }
198
+ };
199
+
200
+ //#endregion
201
+ //#region src/Dir.ts
202
+ /**
203
+ * Reference to a specific directory with helpful methods for resolving filepaths,
204
+ * sanitizing filenames, and saving files.
205
+ */
206
+ var Dir = class Dir {
207
+ path;
208
+ /**
209
+ * @param path can be relative to workspace or absolute
210
+ */
211
+ constructor(_path) {
212
+ this.path = _path;
213
+ }
214
+ create() {
215
+ fs.mkdirSync(this.path, { recursive: true });
216
+ }
217
+ /**
218
+ * Create a new Dir inside the current Dir
219
+ * @param subPath to create in current Dir
220
+ * @example
221
+ * const folder = new Dir('example');
222
+ * // folder.path = './example'
223
+ * const child = folder.subDir('path/to/dir');
224
+ * // child.path = './example/path/to/dir'
225
+ */
226
+ dir(subPath) {
227
+ return new Dir(path.resolve(this.path, subPath));
228
+ }
229
+ sanitize(name) {
230
+ return sanitizeFilename(name.replace("https://", "").replace("www.", ""), { replacement: "_" }).slice(-200);
231
+ }
232
+ /**
233
+ * @param base - The file name with extension
234
+ * @example
235
+ * const folder = new Dir('example');
236
+ * const filepath = folder.resolve('file.json');
237
+ * // 'example/file.json'
238
+ */
239
+ filepath(base) {
240
+ return path.resolve(this.path, this.sanitize(base));
241
+ }
242
+ file(base) {
243
+ return new File(this.filepath(base));
244
+ }
245
+ };
246
+ /**
247
+ * Extends Dir class with method to `clear()` contents
248
+ */
249
+ var TempDir = class TempDir extends Dir {
250
+ dir(subPath) {
251
+ return new TempDir(path.resolve(this.path, subPath));
252
+ }
253
+ clear() {
254
+ fs.rmSync(this.path, {
255
+ recursive: true,
256
+ force: true
257
+ });
258
+ this.create();
259
+ }
260
+ };
261
+ /**
262
+ * Common temp dir location
263
+ */
264
+ const temp = new TempDir(".temp");
265
+
266
+ //#endregion
267
+ //#region src/Cache.ts
268
+ const cacheDir = temp.dir("cache");
269
+ /**
270
+ * Save data to a local file with an expiration.
271
+ * Fresh/stale data is returned with a flag for if it's fresh or not,
272
+ * so stale data can still be used if needed.
273
+ */
274
+ var Cache = class {
275
+ file;
276
+ ttl;
277
+ constructor(key, ttl, initialData) {
278
+ this.file = cacheDir.file(key).json();
279
+ this.ttl = typeof ttl === "number" ? { minutes: ttl } : ttl;
280
+ if (initialData) this.write(initialData);
281
+ }
282
+ write(data) {
283
+ this.file.write({
284
+ savedAt: (/* @__PURE__ */ new Date()).toUTCString(),
285
+ data
286
+ });
287
+ }
288
+ read() {
289
+ const { savedAt, data } = this.file.read() || {};
290
+ return [data, Boolean(savedAt && isAfter(add(savedAt, this.ttl), /* @__PURE__ */ new Date()))];
291
+ }
292
+ };
293
+
294
+ //#endregion
295
+ //#region src/Fetcher.ts
296
+ /**
297
+ * Fetcher provides a quick way to set up a basic API connection
298
+ * with options applied to every request.
299
+ * Includes basic methods for requesting and parsing responses
300
+ */
301
+ var Fetcher = class {
302
+ defaultOptions;
303
+ constructor(opts = {}) {
304
+ this.defaultOptions = {
305
+ timeout: 6e4,
306
+ retries: 0,
307
+ retryDelay: 3e3,
308
+ ...opts
309
+ };
310
+ }
311
+ /**
312
+ * Build URL with URLSearchParams if query is provided.
313
+ * Also returns domain, to help with cookies
314
+ */
315
+ buildUrl(route, opts = {}) {
316
+ const mergedOptions = merge({}, this.defaultOptions, opts);
317
+ const params = [];
318
+ Object.entries(mergedOptions.query || {}).forEach(([key, val]) => {
319
+ if (val === void 0) return;
320
+ if (Array.isArray(val)) val.forEach((v) => {
321
+ params.push([key, `${v}`]);
322
+ });
323
+ else params.push([key, `${val}`]);
324
+ });
325
+ const search = params.length > 0 ? "?" + new URLSearchParams(params).toString() : "";
326
+ const url = new URL(route + search, this.defaultOptions.base);
327
+ return [url, extractDomain(url.href)];
328
+ }
329
+ /**
330
+ * Merges options to get headers. Useful when extending the Fetcher class to add custom auth.
331
+ */
332
+ buildHeaders(route, opts = {}) {
333
+ const { headers } = merge({}, this.defaultOptions, opts);
334
+ return headers || {};
335
+ }
336
+ /**
337
+ * Builds request, merging defaultOptions and provided options.
338
+ * Includes Abort signal for timeout
339
+ */
340
+ buildRequest(route, opts = {}) {
341
+ const mergedOptions = merge({}, this.defaultOptions, opts);
342
+ const { query, data, timeout: timeout$1, retries, ...init } = mergedOptions;
343
+ init.headers = this.buildHeaders(route, mergedOptions);
344
+ if (data) {
345
+ init.headers["content-type"] = init.headers["content-type"] || "application/json";
346
+ init.method = init.method || "POST";
347
+ init.body = JSON.stringify(data);
348
+ }
349
+ if (timeout$1) init.signal = AbortSignal.timeout(timeout$1);
350
+ const [url, domain] = this.buildUrl(route, mergedOptions);
351
+ return [
352
+ new Request(url, init),
353
+ mergedOptions,
354
+ domain
355
+ ];
356
+ }
357
+ /**
358
+ * Builds and performs the request, merging provided options with defaultOptions.
359
+ * If `opts.data` is provided, method is updated to POST, content-type json, data is stringified in the body.
360
+ * Retries on local or network error, with increasing backoff.
361
+ */
362
+ async fetch(route, opts = {}) {
363
+ const [_req, options] = this.buildRequest(route, opts);
364
+ const maxAttempts = (options.retries || 0) + 1;
365
+ let attempt = 0;
366
+ while (attempt < maxAttempts) {
367
+ attempt++;
368
+ const [req] = this.buildRequest(route, opts);
369
+ const res = await fetch(req).then((r) => {
370
+ if (!r.ok) throw new Error(r.statusText);
371
+ return r;
372
+ }).catch(async (error) => {
373
+ if (attempt < maxAttempts) {
374
+ const wait = attempt * 3e3;
375
+ console.warn(`${req.method} ${req.url} (attempt ${attempt} of ${maxAttempts})`, error);
376
+ await new Promise((resolve) => setTimeout(resolve, wait));
377
+ } else throw new Error(error);
378
+ });
379
+ if (res) return [res, req];
380
+ }
381
+ throw new Error(`Failed to fetch ${_req.url}`);
382
+ }
383
+ async fetchText(route, opts = {}) {
384
+ return this.fetch(route, opts).then(async ([res, req]) => {
385
+ return [
386
+ await res.text(),
387
+ res,
388
+ req
389
+ ];
390
+ });
391
+ }
392
+ async fetchJson(route, opts = {}) {
393
+ return this.fetchText(route, opts).then(([txt, res, req]) => [
394
+ JSON.parse(txt),
395
+ res,
396
+ req
397
+ ]);
398
+ }
399
+ };
400
+
401
+ //#endregion
402
+ //#region src/Log.ts
403
+ var Log = class {
404
+ static isTest = process.env.npm_package_name === "@brianbuie/node-kit" && process.env.npm_lifecycle_event === "test";
405
+ /**
406
+ * Gcloud parses JSON in stdout
407
+ */
408
+ static #toGcloud(entry) {
409
+ if (entry.details?.length === 1) console.log(JSON.stringify({
410
+ ...entry,
411
+ details: entry.details[0]
412
+ }));
413
+ else console.log(JSON.stringify(entry));
414
+ }
415
+ /**
416
+ * Includes colors and better inspection for logging during dev
417
+ */
418
+ static #toConsole(entry, color) {
419
+ if (entry.message) console.log(color(`[${entry.severity}] ${entry.message}`));
420
+ entry.details?.forEach((detail) => {
421
+ console.log(inspect(detail, {
422
+ depth: 10,
423
+ breakLength: 100,
424
+ compact: true,
425
+ colors: true
426
+ }));
427
+ });
428
+ }
429
+ static #log(options, ...input) {
430
+ const { message, details } = this.prepare(...input);
431
+ if (process.env.K_SERVICE !== void 0 || process.env.CLOUD_RUN_JOB !== void 0) {
432
+ this.#toGcloud({
433
+ message,
434
+ severity: options.severity,
435
+ details
436
+ });
437
+ return {
438
+ message,
439
+ details,
440
+ options
441
+ };
442
+ }
443
+ if (!this.isTest) this.#toConsole({
444
+ message,
445
+ severity: options.severity,
446
+ details
447
+ }, options.color);
448
+ return {
449
+ message,
450
+ details,
451
+ options
452
+ };
453
+ }
454
+ /**
455
+ * Handle first argument being a string or an object with a 'message' prop
456
+ * Also snapshots special objects (eg Error, Response) to keep props in later JSON.stringify output
457
+ */
458
+ static prepare(...input) {
459
+ let [first, ...rest] = input.map((i) => snapshot(i));
460
+ if (typeof first === "string") return {
461
+ message: first,
462
+ details: rest
463
+ };
464
+ if (isObjectLike(first) && typeof first["message"] === "string") {
465
+ const { message, ...firstDetails } = first;
466
+ return {
467
+ message,
468
+ details: [firstDetails, ...rest]
469
+ };
470
+ }
471
+ return { details: input };
472
+ }
473
+ /**
474
+ * Logs error details before throwing
475
+ */
476
+ static error(...input) {
477
+ const { message } = this.#log({
478
+ severity: "ERROR",
479
+ color: chalk.red
480
+ }, ...input);
481
+ throw new Error(message);
482
+ }
483
+ static warn(...input) {
484
+ return this.#log({
485
+ severity: "WARNING",
486
+ color: chalk.yellow
487
+ }, ...input);
488
+ }
489
+ static notice(...input) {
490
+ return this.#log({
491
+ severity: "NOTICE",
492
+ color: chalk.cyan
493
+ }, ...input);
494
+ }
495
+ static info(...input) {
496
+ return this.#log({
497
+ severity: "INFO",
498
+ color: chalk.white
499
+ }, ...input);
500
+ }
501
+ static debug(...input) {
502
+ if (process.argv.some((arg) => arg.includes("--debug")) || process.env.DEBUG !== void 0 || process.env.NODE_ENV !== "production") return this.#log({
503
+ severity: "DEBUG",
504
+ color: chalk.gray
505
+ }, ...input);
506
+ }
507
+ };
508
+
509
+ //#endregion
510
+ //#region src/timeout.ts
511
+ async function timeout(ms) {
512
+ return new Promise((resolve) => {
513
+ setTimeout(resolve, ms);
514
+ });
515
+ }
516
+
517
+ //#endregion
518
+ //#region src/TypeWriter.ts
519
+ var TypeWriter = class {
520
+ moduleName;
521
+ input = qt.jsonInputForTargetLanguage("typescript");
522
+ outDir;
523
+ qtSettings;
524
+ constructor(moduleName, settings = {}) {
525
+ this.moduleName = moduleName;
526
+ const { outDir, ...qtSettings } = settings;
527
+ this.outDir = outDir || "./types";
528
+ this.qtSettings = merge({
529
+ lang: "typescript",
530
+ rendererOptions: {
531
+ "just-types": true,
532
+ "prefer-types": true
533
+ },
534
+ inferEnums: false,
535
+ inferDateTimes: false
536
+ }, qtSettings);
537
+ }
538
+ async addMember(name, _samples) {
539
+ const samples = _samples.map((s) => typeof s === "string" ? s : JSON.stringify(s));
540
+ await this.input.addSource({
541
+ name,
542
+ samples
543
+ });
544
+ }
545
+ async toString() {
546
+ const inputData = new qt.InputData();
547
+ inputData.addInput(this.input);
548
+ return (await qt.quicktype({
549
+ inputData,
550
+ ...this.qtSettings
551
+ })).lines.join("\n");
552
+ }
553
+ async toFile() {
554
+ const result = await this.toString();
555
+ fs.mkdirSync(this.outDir, { recursive: true });
556
+ fs.writeFileSync(`${this.outDir}/${this.moduleName}.d.ts`, result);
557
+ }
558
+ };
559
+
560
+ //#endregion
561
+ export { Cache, Dir, Fetcher, File, FileType, FileTypeCsv, FileTypeJson, FileTypeNdjson, Log, TempDir, TypeWriter, snapshot, temp, timeout };
562
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.mjs","names":["output: Record<string, any>","obj: Record<string, any>","parsed: Row[]","#parseVal","params: [string, string][]","timeout","#toGcloud","#toConsole","#log"],"sources":["../src/snapshot.ts","../src/File.ts","../src/Dir.ts","../src/Cache.ts","../src/Fetcher.ts","../src/Log.ts","../src/timeout.ts","../src/TypeWriter.ts"],"sourcesContent":["import { isObjectLike } from 'lodash-es';\n\n/**\n * Allows special objects (Error, Headers, Set) to be included in JSON.stringify output\n * functions are removed\n */\nexport function snapshot(i: unknown, max = 50, depth = 0): any {\n if (Array.isArray(i)) {\n if (depth === max) return [];\n return i.map((c) => snapshot(c, max, depth + 1));\n }\n if (typeof i === 'function') return undefined;\n if (!isObjectLike(i)) return i;\n\n if (depth === max) return {};\n let output: Record<string, any> = {};\n // @ts-ignore If it has an 'entries' function, use that for looping (eg. Set, Map, Headers)\n if (typeof i.entries === 'function') {\n // @ts-ignore\n for (let [k, v] of i.entries()) {\n output[k] = snapshot(v, max, depth + 1);\n }\n return output;\n }\n\n // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Enumerability_and_ownership_of_properties\n\n // Get Enumerable, inherited properties\n const obj: Record<string, any> = i!;\n for (let key in obj) {\n output[key] = snapshot(obj[key], max, depth + 1);\n }\n\n // Get Non-enumberable, own properties\n Object.getOwnPropertyNames(obj).forEach((key) => {\n output[key] = snapshot(obj[key], max, depth + 1);\n });\n\n return output;\n}\n","import * as fs from 'node:fs';\nimport * as path from 'node:path';\nimport { Readable } from 'node:stream';\nimport { finished } from 'node:stream/promises';\nimport { writeToStream, parseStream } from 'fast-csv';\nimport { snapshot } from './snapshot.ts';\n\n/**\n * WARNING: API will change!\n */\nexport class File {\n path;\n\n constructor(filepath: string) {\n this.path = filepath;\n }\n\n get exists() {\n return fs.existsSync(this.path);\n }\n\n delete() {\n fs.rmSync(this.path, { force: true });\n }\n\n read() {\n return this.exists ? fs.readFileSync(this.path, 'utf8') : undefined;\n }\n\n /**\n * @returns lines as strings, removes trailing '\\n'\n */\n lines() {\n const contents = (this.read() || '').split('\\n');\n return contents.slice(0, contents.length - 1);\n }\n\n get readStream() {\n return this.exists ? fs.createReadStream(this.path) : Readable.from([]);\n }\n\n get writeStream() {\n fs.mkdirSync(path.parse(this.path).dir, { recursive: true });\n return fs.createWriteStream(this.path);\n }\n\n write(contents: string | ReadableStream) {\n fs.mkdirSync(path.parse(this.path).dir, { recursive: true });\n if (typeof contents === 'string') return fs.writeFileSync(this.path, contents);\n if (contents instanceof ReadableStream) return finished(Readable.from(contents).pipe(this.writeStream));\n throw new Error(`Invalid content type: ${typeof contents}`);\n }\n\n /**\n * creates file if it doesn't exist, appends string or array of strings as new lines.\n * File always ends with '\\n', so contents don't need to be read before appending\n */\n append(lines: string | string[]) {\n if (!this.exists) this.write('');\n const contents = Array.isArray(lines) ? lines.join('\\n') : lines;\n fs.appendFileSync(this.path, contents + '\\n');\n }\n\n static get FileType() {\n return FileType;\n }\n\n json<T>(contents?: T) {\n return new FileTypeJson<T>(this.path, contents);\n }\n\n static get json() {\n return FileTypeJson;\n }\n\n ndjson<T extends object>(lines?: T | T[]) {\n return new FileTypeNdjson<T>(this.path, lines);\n }\n\n static get ndjson() {\n return FileTypeNdjson;\n }\n\n async csv<T extends object>(rows?: T[], keys?: (keyof T)[]) {\n const csvFile = new FileTypeCsv<T>(this.path);\n if (rows) await csvFile.write(rows, keys);\n return csvFile;\n }\n\n static get csv() {\n return FileTypeCsv;\n }\n}\n\n/**\n * A generic file adaptor, extended by specific file type implementations\n */\nexport class FileType {\n file;\n\n constructor(filepath: string, contents?: string) {\n this.file = new File(filepath);\n if (contents) this.file.write(contents);\n }\n\n get exists() {\n return this.file.exists;\n }\n\n get path() {\n return this.file.path;\n }\n\n delete() {\n this.file.delete();\n }\n}\n\n/**\n * A .json file that maintains data type when reading/writing.\n * This is unsafe! Type is not checked at runtime, avoid using on files manipulated outside of your application.\n */\nexport class FileTypeJson<T> extends FileType {\n constructor(filepath: string, contents?: T) {\n super(filepath.endsWith('.json') ? filepath : filepath + '.json');\n if (contents) this.write(contents);\n }\n\n read() {\n const contents = this.file.read();\n return contents ? (JSON.parse(contents) as T) : undefined;\n }\n\n write(contents: T) {\n this.file.write(JSON.stringify(snapshot(contents), null, 2));\n }\n}\n\n/**\n * New-line delimited json file (.ndjson)\n * @see https://jsonltools.com/ndjson-format-specification\n */\nexport class FileTypeNdjson<T extends object> extends FileType {\n constructor(filepath: string, lines?: T | T[]) {\n super(filepath.endsWith('.ndjson') ? filepath : filepath + '.ndjson');\n if (lines) this.append(lines);\n }\n\n append(lines: T | T[]) {\n this.file.append(\n Array.isArray(lines) ? lines.map((l) => JSON.stringify(snapshot(l))) : JSON.stringify(snapshot(lines)),\n );\n }\n\n lines() {\n return this.file.lines().map((l) => JSON.parse(l) as T);\n }\n}\n\ntype Key<T extends object> = keyof T;\n\n/**\n * Comma separated values (.csv).\n * Input rows as objects, keys are used as column headers\n */\nexport class FileTypeCsv<Row extends object> extends FileType {\n constructor(filepath: string) {\n super(filepath.endsWith('.csv') ? filepath : filepath + '.csv');\n }\n\n async write(rows: Row[], keys?: Key<Row>[]) {\n const headerSet = new Set<Key<Row>>();\n if (keys) {\n for (const key of keys) headerSet.add(key);\n } else {\n for (const row of rows) {\n for (const key in row) headerSet.add(key);\n }\n }\n const headers = Array.from(headerSet);\n const outRows = rows.map((row) => headers.map((key) => row[key]));\n return finished(writeToStream(this.file.writeStream, [headers, ...outRows]));\n }\n\n #parseVal(val: string) {\n if (val.toLowerCase() === 'false') return false;\n if (val.toLowerCase() === 'true') return true;\n if (val.length === 0) return null;\n if (/^[\\.0-9]+$/.test(val)) return Number(val);\n return val;\n }\n\n async read() {\n return new Promise<Row[]>((resolve, reject) => {\n const parsed: Row[] = [];\n parseStream(this.file.readStream, { headers: true })\n .on('error', (e) => reject(e))\n .on('data', (raw: Record<Key<Row>, string>) => {\n parsed.push(\n Object.entries(raw).reduce(\n (all, [key, val]) => ({\n ...all,\n [key]: this.#parseVal(val as string),\n }),\n {} as Row,\n ),\n );\n })\n .on('end', () => resolve(parsed));\n });\n }\n}\n","import * as fs from 'node:fs';\nimport * as path from 'node:path';\nimport sanitizeFilename from 'sanitize-filename';\nimport { File } from './File.ts';\n\n/**\n * Reference to a specific directory with helpful methods for resolving filepaths,\n * sanitizing filenames, and saving files.\n */\nexport class Dir {\n path;\n\n /**\n * @param path can be relative to workspace or absolute\n */\n constructor(_path: string) {\n this.path = _path;\n }\n\n create() {\n fs.mkdirSync(this.path, { recursive: true });\n }\n\n /**\n * Create a new Dir inside the current Dir\n * @param subPath to create in current Dir\n * @example\n * const folder = new Dir('example');\n * // folder.path = './example'\n * const child = folder.subDir('path/to/dir');\n * // child.path = './example/path/to/dir'\n */\n dir(subPath: string) {\n return new Dir(path.resolve(this.path, subPath));\n }\n\n sanitize(name: string) {\n return sanitizeFilename(name.replace('https://', '').replace('www.', ''), { replacement: '_' }).slice(-200);\n }\n\n /**\n * @param base - The file name with extension\n * @example\n * const folder = new Dir('example');\n * const filepath = folder.resolve('file.json');\n * // 'example/file.json'\n */\n filepath(base: string) {\n return path.resolve(this.path, this.sanitize(base));\n }\n\n file(base: string) {\n return new File(this.filepath(base));\n }\n}\n\n/**\n * Extends Dir class with method to `clear()` contents\n */\nexport class TempDir extends Dir {\n dir(subPath: string) {\n return new TempDir(path.resolve(this.path, subPath));\n }\n\n clear() {\n fs.rmSync(this.path, { recursive: true, force: true });\n this.create();\n }\n}\n\n/**\n * Common temp dir location\n */\nexport const temp = new TempDir('.temp');\n","import { type Duration, isAfter, add } from 'date-fns';\nimport { temp } from './Dir.ts';\n\nconst cacheDir = temp.dir('cache');\n\n/**\n * Save data to a local file with an expiration.\n * Fresh/stale data is returned with a flag for if it's fresh or not,\n * so stale data can still be used if needed.\n */\nexport class Cache<T> {\n file;\n ttl;\n\n constructor(key: string, ttl: number | Duration, initialData?: T) {\n this.file = cacheDir.file(key).json<{ savedAt: string; data: T }>();\n this.ttl = typeof ttl === 'number' ? { minutes: ttl } : ttl;\n if (initialData) this.write(initialData);\n }\n\n write(data: T) {\n this.file.write({ savedAt: new Date().toUTCString(), data });\n }\n\n read(): [T | undefined, boolean] {\n const { savedAt, data } = this.file.read() || {};\n const isFresh = Boolean(savedAt && isAfter(add(savedAt, this.ttl), new Date()));\n return [data, isFresh];\n }\n}\n","import { merge } from 'lodash-es';\nimport extractDomain from 'extract-domain';\n\nexport type Route = string | URL;\n\ntype QueryVal = string | number | boolean | null | undefined;\nexport type Query = Record<string, QueryVal | QueryVal[]>;\n\nexport type FetchOptions = RequestInit & {\n base?: string;\n query?: Query;\n headers?: Record<string, string>;\n data?: any;\n timeout?: number;\n retries?: number;\n retryDelay?: number;\n};\n\n/**\n * Fetcher provides a quick way to set up a basic API connection\n * with options applied to every request.\n * Includes basic methods for requesting and parsing responses\n */\nexport class Fetcher {\n defaultOptions;\n\n constructor(opts: FetchOptions = {}) {\n this.defaultOptions = {\n timeout: 60000,\n retries: 0,\n retryDelay: 3000,\n ...opts,\n };\n }\n\n /**\n * Build URL with URLSearchParams if query is provided.\n * Also returns domain, to help with cookies\n */\n buildUrl(route: Route, opts: FetchOptions = {}): [URL, string] {\n const mergedOptions = merge({}, this.defaultOptions, opts);\n const params: [string, string][] = [];\n Object.entries(mergedOptions.query || {}).forEach(([key, val]) => {\n if (val === undefined) return;\n if (Array.isArray(val)) {\n val.forEach((v) => {\n params.push([key, `${v}`]);\n });\n } else {\n params.push([key, `${val}`]);\n }\n });\n const search = params.length > 0 ? '?' + new URLSearchParams(params).toString() : '';\n const url = new URL(route + search, this.defaultOptions.base);\n const domain = extractDomain(url.href) as string;\n return [url, domain];\n }\n\n /**\n * Merges options to get headers. Useful when extending the Fetcher class to add custom auth.\n */\n buildHeaders(route: Route, opts: FetchOptions = {}) {\n const { headers } = merge({}, this.defaultOptions, opts);\n return headers || {};\n }\n\n /**\n * Builds request, merging defaultOptions and provided options.\n * Includes Abort signal for timeout\n */\n buildRequest(route: Route, opts: FetchOptions = {}): [Request, FetchOptions, string] {\n const mergedOptions = merge({}, this.defaultOptions, opts);\n const { query, data, timeout, retries, ...init } = mergedOptions;\n init.headers = this.buildHeaders(route, mergedOptions);\n if (data) {\n init.headers['content-type'] = init.headers['content-type'] || 'application/json';\n init.method = init.method || 'POST';\n init.body = JSON.stringify(data);\n }\n if (timeout) {\n init.signal = AbortSignal.timeout(timeout);\n }\n const [url, domain] = this.buildUrl(route, mergedOptions);\n const req = new Request(url, init);\n return [req, mergedOptions, domain];\n }\n\n /**\n * Builds and performs the request, merging provided options with defaultOptions.\n * If `opts.data` is provided, method is updated to POST, content-type json, data is stringified in the body.\n * Retries on local or network error, with increasing backoff.\n */\n async fetch(route: Route, opts: FetchOptions = {}): Promise<[Response, Request]> {\n const [_req, options] = this.buildRequest(route, opts);\n const maxAttempts = (options.retries || 0) + 1;\n let attempt = 0;\n while (attempt < maxAttempts) {\n attempt++;\n const [req] = this.buildRequest(route, opts);\n const res = await fetch(req)\n .then((r) => {\n if (!r.ok) throw new Error(r.statusText);\n return r;\n })\n .catch(async (error) => {\n if (attempt < maxAttempts) {\n const wait = attempt * 3000;\n console.warn(`${req.method} ${req.url} (attempt ${attempt} of ${maxAttempts})`, error);\n await new Promise((resolve) => setTimeout(resolve, wait));\n } else {\n throw new Error(error);\n }\n });\n if (res) return [res, req];\n }\n throw new Error(`Failed to fetch ${_req.url}`);\n }\n\n async fetchText(route: Route, opts: FetchOptions = {}): Promise<[string, Response, Request]> {\n return this.fetch(route, opts).then(async ([res, req]) => {\n const text = await res.text();\n return [text, res, req];\n });\n }\n\n async fetchJson<T>(route: Route, opts: FetchOptions = {}): Promise<[T, Response, Request]> {\n return this.fetchText(route, opts).then(([txt, res, req]) => [JSON.parse(txt) as T, res, req]);\n }\n}\n","import { inspect } from 'node:util';\nimport { isObjectLike } from 'lodash-es';\nimport chalk, { type ChalkInstance } from 'chalk';\nimport { snapshot } from './snapshot.ts';\n\ntype Severity = 'DEFAULT' | 'DEBUG' | 'INFO' | 'NOTICE' | 'WARNING' | 'ERROR' | 'CRITICAL' | 'ALERT' | 'EMERGENCY';\n\ntype Options = {\n severity: Severity;\n color: ChalkInstance;\n};\n\ntype Entry = {\n message?: string;\n severity: Severity;\n details?: unknown[];\n};\n\nexport class Log {\n // Only silence logs when THIS package is running its own tests\n static isTest = process.env.npm_package_name === '@brianbuie/node-kit' && process.env.npm_lifecycle_event === 'test';\n\n /**\n * Gcloud parses JSON in stdout\n */\n static #toGcloud(entry: Entry) {\n if (entry.details?.length === 1) {\n console.log(JSON.stringify({ ...entry, details: entry.details[0] }));\n } else {\n console.log(JSON.stringify(entry));\n }\n }\n\n /**\n * Includes colors and better inspection for logging during dev\n */\n static #toConsole(entry: Entry, color: ChalkInstance) {\n if (entry.message) console.log(color(`[${entry.severity}] ${entry.message}`));\n entry.details?.forEach((detail) => {\n console.log(inspect(detail, { depth: 10, breakLength: 100, compact: true, colors: true }));\n });\n }\n\n static #log(options: Options, ...input: unknown[]) {\n const { message, details } = this.prepare(...input);\n // https://cloud.google.com/run/docs/container-contract#env-vars\n const isGcloud = process.env.K_SERVICE !== undefined || process.env.CLOUD_RUN_JOB !== undefined;\n if (isGcloud) {\n this.#toGcloud({ message, severity: options.severity, details });\n return { message, details, options };\n }\n // Hide output while testing this package\n if (!this.isTest) {\n this.#toConsole({ message, severity: options.severity, details }, options.color);\n }\n return { message, details, options };\n }\n\n /**\n * Handle first argument being a string or an object with a 'message' prop\n * Also snapshots special objects (eg Error, Response) to keep props in later JSON.stringify output\n */\n static prepare(...input: unknown[]): { message?: string; details: unknown[] } {\n let [first, ...rest] = input.map((i) => snapshot(i));\n if (typeof first === 'string') return { message: first, details: rest };\n // @ts-ignore\n if (isObjectLike(first) && typeof first['message'] === 'string') {\n const { message, ...firstDetails } = first as { message: string };\n return { message, details: [firstDetails, ...rest] };\n }\n return { details: input };\n }\n\n /**\n * Logs error details before throwing\n */\n static error(...input: unknown[]) {\n const { message } = this.#log({ severity: 'ERROR', color: chalk.red }, ...input);\n throw new Error(message);\n }\n\n static warn(...input: unknown[]) {\n return this.#log({ severity: 'WARNING', color: chalk.yellow }, ...input);\n }\n\n static notice(...input: unknown[]) {\n return this.#log({ severity: 'NOTICE', color: chalk.cyan }, ...input);\n }\n\n static info(...input: unknown[]) {\n return this.#log({ severity: 'INFO', color: chalk.white }, ...input);\n }\n\n static debug(...input: unknown[]) {\n const debugging = process.argv.some((arg) => arg.includes('--debug')) || process.env.DEBUG !== undefined;\n if (debugging || process.env.NODE_ENV !== 'production') {\n return this.#log({ severity: 'DEBUG', color: chalk.gray }, ...input);\n }\n }\n}\n","export async function timeout(ms: number) {\n return new Promise((resolve) => {\n setTimeout(resolve, ms);\n });\n}\n","import * as fs from 'node:fs';\nimport { merge } from 'lodash-es';\nimport * as qt from 'quicktype-core';\n\nexport class TypeWriter {\n moduleName;\n input = qt.jsonInputForTargetLanguage('typescript');\n outDir;\n qtSettings;\n\n constructor(moduleName: string, settings: { outDir?: string } & Partial<qt.Options> = {}) {\n this.moduleName = moduleName;\n const { outDir, ...qtSettings } = settings;\n this.outDir = outDir || './types';\n const defaultSettings = {\n lang: 'typescript',\n rendererOptions: {\n 'just-types': true,\n 'prefer-types': true,\n },\n inferEnums: false,\n inferDateTimes: false,\n };\n this.qtSettings = merge(defaultSettings, qtSettings);\n }\n\n async addMember(name: string, _samples: any[]) {\n const samples = _samples.map((s) => (typeof s === 'string' ? s : JSON.stringify(s)));\n await this.input.addSource({ name, samples });\n }\n\n async toString() {\n const inputData = new qt.InputData();\n inputData.addInput(this.input);\n const result = await qt.quicktype({\n inputData,\n ...this.qtSettings,\n });\n return result.lines.join('\\n');\n }\n\n async toFile() {\n const result = await this.toString();\n fs.mkdirSync(this.outDir, { recursive: true });\n fs.writeFileSync(`${this.outDir}/${this.moduleName}.d.ts`, result);\n }\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@brianbuie/node-kit",
3
- "version": "0.9.1",
3
+ "version": "0.9.2",
4
4
  "license": "ISC",
5
5
  "repository": {
6
6
  "type": "git",