donebear 0.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,4 @@
1
+ #!/usr/bin/env node
2
+ import { n as setCurrentWorkspace, t as ensureLocalContext } from "./context-store-BP3Fw5Ki.mjs";
3
+
4
+ export { ensureLocalContext };
@@ -0,0 +1,139 @@
1
+ #!/usr/bin/env node
2
+ import { homedir } from "node:os";
3
+ import { join } from "node:path";
4
+ import { mkdir, readFile, writeFile } from "node:fs/promises";
5
+ import { randomUUID } from "node:crypto";
6
+
7
+ //#region src/constants.ts
8
+ const CLI_NAME = "donebear";
9
+ const DONEBEAR_TOKEN_ENV = "DONEBEAR_TOKEN";
10
+ const DONEBEAR_DEBUG_ENV = "DONEBEAR_DEBUG";
11
+ const DONEBEAR_CONFIG_DIR_ENV = "DONEBEAR_CONFIG_DIR";
12
+ const DONEBEAR_SUPABASE_URL_ENV = "DONEBEAR_SUPABASE_URL";
13
+ const DONEBEAR_SUPABASE_KEY_ENV = "DONEBEAR_SUPABASE_PUBLISHABLE_KEY";
14
+ const DONEBEAR_API_URL_ENV = "DONEBEAR_API_URL";
15
+ const DEFAULT_API_URL = "https://api.donebear.com";
16
+ const DEFAULT_SUPABASE_URL = "https://kdarvputxmlxqemisnbh.supabase.co";
17
+ const DEFAULT_SUPABASE_PUBLISHABLE_KEY = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImtkYXJ2cHV0eG1seHFlbWlzbmJoIiwicm9sZSI6ImFub24iLCJpYXQiOjE3NzIzNTcyNjcsImV4cCI6MjA4NzkzMzI2N30.A8sDPWNe_1EcCy0-GsUw4h1icxSGz_VDc9E9tPTYEU8";
18
+ const CLI_OAUTH_CLIENT_ID = "9cc84e5c-6a0e-4a6d-93a2-334d8d307122";
19
+ const DONEBEAR_OAUTH_CLIENT_ID_ENV = "DONEBEAR_OAUTH_CLIENT_ID";
20
+ const DEFAULT_OAUTH_CALLBACK_PORT = 8787;
21
+ const DEFAULT_OAUTH_CALLBACK_PATH = "/auth/callback";
22
+ const DEFAULT_OAUTH_TIMEOUT_SECONDS = 180;
23
+ const getDefaultConfigBaseDir = () => {
24
+ if (process.platform === "win32") return process.env.APPDATA ?? join(homedir(), "AppData", "Roaming");
25
+ return process.env.XDG_CONFIG_HOME ?? join(homedir(), ".config");
26
+ };
27
+ const configBaseDir = getDefaultConfigBaseDir();
28
+ const CONFIG_DIR = process.env[DONEBEAR_CONFIG_DIR_ENV] ?? join(configBaseDir, CLI_NAME);
29
+ const AUTH_FILE_PATH = join(CONFIG_DIR, "auth.json");
30
+ const CONTEXT_FILE_PATH = join(CONFIG_DIR, "context.json");
31
+
32
+ //#endregion
33
+ //#region src/errors.ts
34
+ const EXIT_CODES = {
35
+ AUTH_REQUIRED: 4,
36
+ CANCELLED: 2,
37
+ ERROR: 1,
38
+ NETWORK: 5,
39
+ SUCCESS: 0,
40
+ VALIDATION: 3
41
+ };
42
+ const EXIT_CODE_TO_ERROR_CODE = {
43
+ [EXIT_CODES.SUCCESS]: "unknown_error",
44
+ [EXIT_CODES.ERROR]: "unknown_error",
45
+ [EXIT_CODES.CANCELLED]: "cancelled",
46
+ [EXIT_CODES.VALIDATION]: "validation_error",
47
+ [EXIT_CODES.AUTH_REQUIRED]: "auth_error",
48
+ [EXIT_CODES.NETWORK]: "network_error"
49
+ };
50
+ const exitCodeToErrorCode = (exitCode) => EXIT_CODE_TO_ERROR_CODE[exitCode] ?? "unknown_error";
51
+ var CliError = class extends Error {
52
+ exitCode;
53
+ code;
54
+ hint;
55
+ constructor(message, exitCode = EXIT_CODES.ERROR, hint) {
56
+ super(message);
57
+ this.name = "CliError";
58
+ this.exitCode = exitCode;
59
+ this.code = exitCodeToErrorCode(exitCode);
60
+ this.hint = hint ?? null;
61
+ }
62
+ };
63
+ const toCliError = (error) => {
64
+ if (error instanceof CliError) return error;
65
+ if (error instanceof Error) {
66
+ if (error instanceof TypeError && error.message.includes("fetch")) return new CliError("Cannot connect to Done Bear API.", EXIT_CODES.NETWORK, "Check your internet connection and API URL configuration.");
67
+ if (error.name === "TimeoutError" || error.name === "AbortError") return new CliError("Request timed out.", EXIT_CODES.NETWORK, "The API may be unavailable. Try again later.");
68
+ return new CliError(error.message, EXIT_CODES.ERROR);
69
+ }
70
+ return new CliError("Unknown error", EXIT_CODES.ERROR);
71
+ };
72
+
73
+ //#endregion
74
+ //#region src/context-store.ts
75
+ const isRecord = (value) => typeof value === "object" && value !== null;
76
+ const parseLocalContext = (value) => {
77
+ if (!isRecord(value)) return null;
78
+ if (value.workspaceId !== null && typeof value.workspaceId !== "string") return null;
79
+ if (typeof value.clientId !== "string" || value.clientId.length === 0) return null;
80
+ return {
81
+ clientId: value.clientId,
82
+ workspaceId: value.workspaceId
83
+ };
84
+ };
85
+ const parseContextFile = (payload) => {
86
+ const parsed = JSON.parse(payload);
87
+ if (!isRecord(parsed) || parsed.version !== 1) throw new CliError(`Invalid CLI context format in ${CONTEXT_FILE_PATH}`, EXIT_CODES.ERROR);
88
+ const context = parseLocalContext(parsed.context);
89
+ if (!context) throw new CliError(`Invalid CLI context in ${CONTEXT_FILE_PATH}`, EXIT_CODES.ERROR);
90
+ return {
91
+ context,
92
+ version: 1
93
+ };
94
+ };
95
+ const writeContext = async (context) => {
96
+ const payload = {
97
+ context,
98
+ version: 1
99
+ };
100
+ await mkdir(CONFIG_DIR, {
101
+ mode: 448,
102
+ recursive: true
103
+ });
104
+ await writeFile(CONTEXT_FILE_PATH, `${JSON.stringify(payload, null, 2)}\n`, {
105
+ encoding: "utf8",
106
+ mode: 384
107
+ });
108
+ };
109
+ const readLocalContext = async () => {
110
+ try {
111
+ return parseContextFile(await readFile(CONTEXT_FILE_PATH, "utf8")).context;
112
+ } catch (error) {
113
+ if (isRecord(error) && error.code === "ENOENT") return null;
114
+ if (error instanceof CliError) throw error;
115
+ throw new CliError(`Failed to read context from ${CONTEXT_FILE_PATH}`, EXIT_CODES.ERROR);
116
+ }
117
+ };
118
+ const ensureLocalContext = async () => {
119
+ const existing = await readLocalContext();
120
+ if (existing) return existing;
121
+ const created = {
122
+ clientId: randomUUID(),
123
+ workspaceId: null
124
+ };
125
+ await writeContext(created);
126
+ return created;
127
+ };
128
+ const setCurrentWorkspace = async (workspaceId) => {
129
+ const nextContext = {
130
+ ...await ensureLocalContext(),
131
+ workspaceId
132
+ };
133
+ await writeContext(nextContext);
134
+ return nextContext;
135
+ };
136
+
137
+ //#endregion
138
+ export { DONEBEAR_DEBUG_ENV as _, toCliError as a, DONEBEAR_SUPABASE_URL_ENV as b, CLI_OAUTH_CLIENT_ID as c, DEFAULT_OAUTH_CALLBACK_PATH as d, DEFAULT_OAUTH_CALLBACK_PORT as f, DONEBEAR_API_URL_ENV as g, DEFAULT_SUPABASE_URL as h, EXIT_CODES as i, CONFIG_DIR as l, DEFAULT_SUPABASE_PUBLISHABLE_KEY as m, setCurrentWorkspace as n, AUTH_FILE_PATH as o, DEFAULT_OAUTH_TIMEOUT_SECONDS as p, CliError as r, CLI_NAME as s, ensureLocalContext as t, DEFAULT_API_URL as u, DONEBEAR_OAUTH_CLIENT_ID_ENV as v, DONEBEAR_TOKEN_ENV as x, DONEBEAR_SUPABASE_KEY_ENV as y };
139
+ //# sourceMappingURL=context-store-BP3Fw5Ki.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-store-BP3Fw5Ki.mjs","names":[],"sources":["../src/constants.ts","../src/errors.ts","../src/context-store.ts"],"sourcesContent":["import { homedir } from \"node:os\";\nimport { join } from \"node:path\";\n\nexport const CLI_NAME = \"donebear\";\n\nexport const DONEBEAR_TOKEN_ENV = \"DONEBEAR_TOKEN\";\nexport const DONEBEAR_DEBUG_ENV = \"DONEBEAR_DEBUG\";\nconst DONEBEAR_CONFIG_DIR_ENV = \"DONEBEAR_CONFIG_DIR\";\nexport const DONEBEAR_SUPABASE_URL_ENV = \"DONEBEAR_SUPABASE_URL\";\nexport const DONEBEAR_SUPABASE_KEY_ENV = \"DONEBEAR_SUPABASE_PUBLISHABLE_KEY\";\nexport const DONEBEAR_API_URL_ENV = \"DONEBEAR_API_URL\";\n\nexport const DEFAULT_API_URL = \"https://api.donebear.com\";\n\nexport const DEFAULT_SUPABASE_URL = \"https://kdarvputxmlxqemisnbh.supabase.co\";\nexport const DEFAULT_SUPABASE_PUBLISHABLE_KEY =\n \"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImtkYXJ2cHV0eG1seHFlbWlzbmJoIiwicm9sZSI6ImFub24iLCJpYXQiOjE3NzIzNTcyNjcsImV4cCI6MjA4NzkzMzI2N30.A8sDPWNe_1EcCy0-GsUw4h1icxSGz_VDc9E9tPTYEU8\";\n\nexport const CLI_OAUTH_CLIENT_ID = \"9cc84e5c-6a0e-4a6d-93a2-334d8d307122\";\nexport const DONEBEAR_OAUTH_CLIENT_ID_ENV = \"DONEBEAR_OAUTH_CLIENT_ID\";\n\nexport const DEFAULT_OAUTH_CALLBACK_PORT = 8787;\nexport const DEFAULT_OAUTH_CALLBACK_PATH = \"/auth/callback\";\nexport const DEFAULT_OAUTH_TIMEOUT_SECONDS = 180;\n\nconst getDefaultConfigBaseDir = (): string => {\n if (process.platform === \"win32\") {\n return process.env.APPDATA ?? join(homedir(), \"AppData\", \"Roaming\");\n }\n\n return process.env.XDG_CONFIG_HOME ?? join(homedir(), \".config\");\n};\n\nconst configBaseDir = getDefaultConfigBaseDir();\n\nexport const CONFIG_DIR =\n process.env[DONEBEAR_CONFIG_DIR_ENV] ?? join(configBaseDir, CLI_NAME);\n\nexport const AUTH_FILE_PATH = join(CONFIG_DIR, \"auth.json\");\nexport const CONTEXT_FILE_PATH = join(CONFIG_DIR, \"context.json\");\n","export const EXIT_CODES = {\n AUTH_REQUIRED: 4,\n CANCELLED: 2,\n ERROR: 1,\n NETWORK: 5,\n SUCCESS: 0,\n VALIDATION: 3,\n} as const;\n\nexport type ExitCode = (typeof EXIT_CODES)[keyof typeof EXIT_CODES];\n\nexport type ErrorCode =\n | \"unknown_error\"\n | \"cancelled\"\n | \"validation_error\"\n | \"auth_error\"\n | \"network_error\";\n\nconst EXIT_CODE_TO_ERROR_CODE: Record<ExitCode, ErrorCode> = {\n [EXIT_CODES.SUCCESS]: \"unknown_error\",\n [EXIT_CODES.ERROR]: \"unknown_error\",\n [EXIT_CODES.CANCELLED]: \"cancelled\",\n [EXIT_CODES.VALIDATION]: \"validation_error\",\n [EXIT_CODES.AUTH_REQUIRED]: \"auth_error\",\n [EXIT_CODES.NETWORK]: \"network_error\",\n};\n\nexport const exitCodeToErrorCode = (exitCode: ExitCode): ErrorCode =>\n EXIT_CODE_TO_ERROR_CODE[exitCode] ?? \"unknown_error\";\n\nexport class CliError extends Error {\n readonly exitCode: ExitCode;\n readonly code: ErrorCode;\n readonly hint: string | null;\n\n constructor(\n message: string,\n exitCode: ExitCode = EXIT_CODES.ERROR,\n hint?: string\n ) {\n super(message);\n this.name = \"CliError\";\n this.exitCode = exitCode;\n this.code = exitCodeToErrorCode(exitCode);\n this.hint = hint ?? null;\n }\n}\n\nexport const toCliError = (error: unknown): CliError => {\n if (error instanceof CliError) {\n return error;\n }\n\n if (error instanceof Error) {\n // Network errors (fetch failures)\n if (error instanceof TypeError && error.message.includes(\"fetch\")) {\n return new CliError(\n \"Cannot connect to Done Bear API.\",\n EXIT_CODES.NETWORK,\n \"Check your internet connection and API URL configuration.\"\n );\n }\n\n // Timeout errors\n if (error.name === \"TimeoutError\" || error.name === \"AbortError\") {\n return new CliError(\n \"Request timed out.\",\n EXIT_CODES.NETWORK,\n \"The API may be unavailable. Try again later.\"\n );\n }\n\n return new CliError(error.message, EXIT_CODES.ERROR);\n }\n\n return new CliError(\"Unknown error\", EXIT_CODES.ERROR);\n};\n","import { randomUUID } from \"node:crypto\";\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\n\nimport { CONFIG_DIR, CONTEXT_FILE_PATH } from \"./constants\";\nimport { CliError, EXIT_CODES } from \"./errors\";\nimport type { CliContextFileData, LocalCliContext } from \"./types\";\n\nconst isRecord = (value: unknown): value is Record<string, unknown> =>\n typeof value === \"object\" && value !== null;\n\nconst parseLocalContext = (value: unknown): LocalCliContext | null => {\n if (!isRecord(value)) {\n return null;\n }\n\n if (value.workspaceId !== null && typeof value.workspaceId !== \"string\") {\n return null;\n }\n\n if (typeof value.clientId !== \"string\" || value.clientId.length === 0) {\n return null;\n }\n\n return {\n clientId: value.clientId,\n workspaceId: value.workspaceId,\n };\n};\n\nconst parseContextFile = (payload: string): CliContextFileData => {\n const parsed = JSON.parse(payload) as unknown;\n\n if (!isRecord(parsed) || parsed.version !== 1) {\n throw new CliError(\n `Invalid CLI context format in ${CONTEXT_FILE_PATH}`,\n EXIT_CODES.ERROR\n );\n }\n\n const context = parseLocalContext(parsed.context);\n\n if (!context) {\n throw new CliError(\n `Invalid CLI context in ${CONTEXT_FILE_PATH}`,\n EXIT_CODES.ERROR\n );\n }\n\n return {\n context,\n version: 1,\n };\n};\n\nconst writeContext = async (context: LocalCliContext): Promise<void> => {\n const payload: CliContextFileData = {\n context,\n version: 1,\n };\n\n await mkdir(CONFIG_DIR, {\n mode: 0o700,\n recursive: true,\n });\n\n await writeFile(CONTEXT_FILE_PATH, `${JSON.stringify(payload, null, 2)}\\n`, {\n encoding: \"utf8\",\n mode: 0o600,\n });\n};\n\nconst readLocalContext = async (): Promise<LocalCliContext | null> => {\n try {\n const raw = await readFile(CONTEXT_FILE_PATH, \"utf8\");\n return parseContextFile(raw).context;\n } catch (error) {\n if (isRecord(error) && error.code === \"ENOENT\") {\n return null;\n }\n\n if (error instanceof CliError) {\n throw error;\n }\n\n throw new CliError(\n `Failed to read context from ${CONTEXT_FILE_PATH}`,\n EXIT_CODES.ERROR\n );\n }\n};\n\nexport const ensureLocalContext = async (): Promise<LocalCliContext> => {\n const existing = await readLocalContext();\n\n if (existing) {\n return existing;\n }\n\n const created: LocalCliContext = {\n clientId: randomUUID(),\n workspaceId: null,\n };\n\n await writeContext(created);\n return created;\n};\n\nexport const setCurrentWorkspace = async (\n workspaceId: string | null\n): Promise<LocalCliContext> => {\n const context = await ensureLocalContext();\n\n const nextContext: LocalCliContext = {\n ...context,\n workspaceId,\n };\n\n await writeContext(nextContext);\n return nextContext;\n};\n"],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export { };
package/dist/index.mjs ADDED
@@ -0,0 +1 @@
1
+ export { };
@@ -0,0 +1,4 @@
1
+ #!/usr/bin/env node
2
+ import { n as readStdin, t as promptForValue } from "./prompt-JbRe_zaa.mjs";
3
+
4
+ export { promptForValue };
@@ -0,0 +1,37 @@
1
+ #!/usr/bin/env node
2
+ import { createInterface } from "node:readline";
3
+
4
+ //#region src/prompt.ts
5
+ /**
6
+ * Prompt for a value interactively.
7
+ * Returns null if not in a TTY or if the user enters an empty string.
8
+ */
9
+ const promptForValue = (context, label) => {
10
+ if (!process.stdin.isTTY || context.json || context.quiet) return Promise.resolve(null);
11
+ const rl = createInterface({
12
+ input: process.stdin,
13
+ output: process.stderr
14
+ });
15
+ return new Promise((resolve) => {
16
+ rl.question(`${label}: `, (answer) => {
17
+ rl.close();
18
+ const trimmed = answer.trim();
19
+ resolve(trimmed.length > 0 ? trimmed : null);
20
+ });
21
+ });
22
+ };
23
+ /**
24
+ * Read all of stdin as a string. Returns null if stdin is a TTY
25
+ * (nothing piped) or if it's empty.
26
+ */
27
+ const readStdin = async () => {
28
+ if (process.stdin.isTTY) return null;
29
+ const chunks = [];
30
+ for await (const chunk of process.stdin) chunks.push(chunk);
31
+ const text = Buffer.concat(chunks).toString("utf8").trim();
32
+ return text.length > 0 ? text : null;
33
+ };
34
+
35
+ //#endregion
36
+ export { readStdin as n, promptForValue as t };
37
+ //# sourceMappingURL=prompt-JbRe_zaa.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompt-JbRe_zaa.mjs","names":[],"sources":["../src/prompt.ts"],"sourcesContent":["import { createInterface } from \"node:readline\";\n\nimport type { CliContext } from \"./types\";\n\n/**\n * Prompt for a value interactively.\n * Returns null if not in a TTY or if the user enters an empty string.\n */\nexport const promptForValue = (\n context: CliContext,\n label: string\n): Promise<string | null> => {\n if (!process.stdin.isTTY || context.json || context.quiet) {\n return Promise.resolve(null);\n }\n\n const rl = createInterface({\n input: process.stdin,\n output: process.stderr,\n });\n\n // oxlint-disable-next-line avoid-new -- wrapping callback API in promise\n return new Promise((resolve) => {\n rl.question(`${label}: `, (answer) => {\n rl.close();\n const trimmed = answer.trim();\n resolve(trimmed.length > 0 ? trimmed : null);\n });\n });\n};\n\n/**\n * Read all of stdin as a string. Returns null if stdin is a TTY\n * (nothing piped) or if it's empty.\n */\nexport const readStdin = async (): Promise<string | null> => {\n if (process.stdin.isTTY) {\n return null;\n }\n\n const chunks: Buffer[] = [];\n\n for await (const chunk of process.stdin) {\n chunks.push(chunk as Buffer);\n }\n\n const text = Buffer.concat(chunks).toString(\"utf8\").trim();\n return text.length > 0 ? text : null;\n};\n"],"mappings":";;;;;;;;AAQA,MAAa,kBACX,SACA,UAC2B;AAC3B,KAAI,CAAC,QAAQ,MAAM,SAAS,QAAQ,QAAQ,QAAQ,MAClD,QAAO,QAAQ,QAAQ,KAAK;CAG9B,MAAM,KAAK,gBAAgB;EACzB,OAAO,QAAQ;EACf,QAAQ,QAAQ;EACjB,CAAC;AAGF,QAAO,IAAI,SAAS,YAAY;AAC9B,KAAG,SAAS,GAAG,MAAM,MAAM,WAAW;AACpC,MAAG,OAAO;GACV,MAAM,UAAU,OAAO,MAAM;AAC7B,WAAQ,QAAQ,SAAS,IAAI,UAAU,KAAK;IAC5C;GACF;;;;;;AAOJ,MAAa,YAAY,YAAoC;AAC3D,KAAI,QAAQ,MAAM,MAChB,QAAO;CAGT,MAAM,SAAmB,EAAE;AAE3B,YAAW,MAAM,SAAS,QAAQ,MAChC,QAAO,KAAK,MAAgB;CAG9B,MAAM,OAAO,OAAO,OAAO,OAAO,CAAC,SAAS,OAAO,CAAC,MAAM;AAC1D,QAAO,KAAK,SAAS,IAAI,OAAO"}
package/package.json ADDED
@@ -0,0 +1,64 @@
1
+ {
2
+ "name": "donebear",
3
+ "version": "0.3.1",
4
+ "description": "Done Bear CLI with OAuth authentication",
5
+ "homepage": "https://github.com/mblode/donebear#readme",
6
+ "bugs": {
7
+ "url": "https://github.com/mblode/donebear/tasks"
8
+ },
9
+ "repository": {
10
+ "type": "git",
11
+ "url": "git+https://github.com/mblode/donebear.git",
12
+ "directory": "apps/cli"
13
+ },
14
+ "bin": {
15
+ "donebear": "./dist/cli.mjs"
16
+ },
17
+ "files": [
18
+ "dist",
19
+ "README.md",
20
+ "AGENTS.md"
21
+ ],
22
+ "type": "module",
23
+ "main": "./dist/index.mjs",
24
+ "types": "./dist/index.d.mts",
25
+ "exports": {
26
+ ".": {
27
+ "types": "./dist/index.d.mts",
28
+ "import": "./dist/index.mjs"
29
+ }
30
+ },
31
+ "publishConfig": {
32
+ "access": "public"
33
+ },
34
+ "scripts": {
35
+ "build": "tsdown",
36
+ "dev": "tsdown --watch",
37
+ "start": "node dist/cli.mjs",
38
+ "lint": "oxlint .",
39
+ "lint:fix": "oxlint --fix .",
40
+ "format": "oxfmt --write .",
41
+ "format:check": "oxfmt --check .",
42
+ "check-types": "tsc --noEmit",
43
+ "test": "vitest run"
44
+ },
45
+ "dependencies": {
46
+ "better-sqlite3": "^12.6.2",
47
+ "commander": "^14.0.2",
48
+ "dotenv": "^17.2.3",
49
+ "jmespath": "^0.16.0",
50
+ "rrule": "^2.8.1",
51
+ "yaml": "^2.8.2"
52
+ },
53
+ "devDependencies": {
54
+ "@types/better-sqlite3": "^7.6.13",
55
+ "@types/jmespath": "^0.15.2",
56
+ "@types/node": "^25.1.0",
57
+ "tsdown": "^0.20.3",
58
+ "typescript": "^5.9.3",
59
+ "vitest": "^4.0.18"
60
+ },
61
+ "engines": {
62
+ "node": ">=18"
63
+ }
64
+ }