camo-gemini 0.1.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 (139) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +242 -0
  3. package/dist/client/camofox-client.d.ts +21 -0
  4. package/dist/client/camofox-client.d.ts.map +1 -0
  5. package/dist/client/camofox-client.js +260 -0
  6. package/dist/client/camofox-client.js.map +1 -0
  7. package/dist/config.d.ts +3 -0
  8. package/dist/config.d.ts.map +1 -0
  9. package/dist/config.js +94 -0
  10. package/dist/config.js.map +1 -0
  11. package/dist/core/browser-js.d.ts +27 -0
  12. package/dist/core/browser-js.d.ts.map +1 -0
  13. package/dist/core/browser-js.js +172 -0
  14. package/dist/core/browser-js.js.map +1 -0
  15. package/dist/core/constants.d.ts +78 -0
  16. package/dist/core/constants.d.ts.map +1 -0
  17. package/dist/core/constants.js +115 -0
  18. package/dist/core/constants.js.map +1 -0
  19. package/dist/core/error-messages.d.ts +8 -0
  20. package/dist/core/error-messages.d.ts.map +1 -0
  21. package/dist/core/error-messages.js +121 -0
  22. package/dist/core/error-messages.js.map +1 -0
  23. package/dist/core/failover.d.ts +21 -0
  24. package/dist/core/failover.d.ts.map +1 -0
  25. package/dist/core/failover.js +47 -0
  26. package/dist/core/failover.js.map +1 -0
  27. package/dist/core/logger.d.ts +13 -0
  28. package/dist/core/logger.d.ts.map +1 -0
  29. package/dist/core/logger.js +37 -0
  30. package/dist/core/logger.js.map +1 -0
  31. package/dist/core/request-builder.d.ts +20 -0
  32. package/dist/core/request-builder.d.ts.map +1 -0
  33. package/dist/core/request-builder.js +210 -0
  34. package/dist/core/request-builder.js.map +1 -0
  35. package/dist/core/response-parser.d.ts +12 -0
  36. package/dist/core/response-parser.d.ts.map +1 -0
  37. package/dist/core/response-parser.js +221 -0
  38. package/dist/core/response-parser.js.map +1 -0
  39. package/dist/core/retry.d.ts +10 -0
  40. package/dist/core/retry.d.ts.map +1 -0
  41. package/dist/core/retry.js +132 -0
  42. package/dist/core/retry.js.map +1 -0
  43. package/dist/core/stream-parser.d.ts +9 -0
  44. package/dist/core/stream-parser.d.ts.map +1 -0
  45. package/dist/core/stream-parser.js +69 -0
  46. package/dist/core/stream-parser.js.map +1 -0
  47. package/dist/core/token-manager.d.ts +9 -0
  48. package/dist/core/token-manager.d.ts.map +1 -0
  49. package/dist/core/token-manager.js +64 -0
  50. package/dist/core/token-manager.js.map +1 -0
  51. package/dist/core/utils.d.ts +2 -0
  52. package/dist/core/utils.d.ts.map +1 -0
  53. package/dist/core/utils.js +15 -0
  54. package/dist/core/utils.js.map +1 -0
  55. package/dist/dashboard/server.d.ts +31 -0
  56. package/dist/dashboard/server.d.ts.map +1 -0
  57. package/dist/dashboard/server.js +190 -0
  58. package/dist/dashboard/server.js.map +1 -0
  59. package/dist/errors.d.ts +32 -0
  60. package/dist/errors.d.ts.map +1 -0
  61. package/dist/errors.js +59 -0
  62. package/dist/errors.js.map +1 -0
  63. package/dist/index.d.ts +3 -0
  64. package/dist/index.d.ts.map +1 -0
  65. package/dist/index.js +36 -0
  66. package/dist/index.js.map +1 -0
  67. package/dist/server.d.ts +31 -0
  68. package/dist/server.d.ts.map +1 -0
  69. package/dist/server.js +54 -0
  70. package/dist/server.js.map +1 -0
  71. package/dist/services/account.d.ts +16 -0
  72. package/dist/services/account.d.ts.map +1 -0
  73. package/dist/services/account.js +55 -0
  74. package/dist/services/account.js.map +1 -0
  75. package/dist/services/auth.d.ts +39 -0
  76. package/dist/services/auth.d.ts.map +1 -0
  77. package/dist/services/auth.js +285 -0
  78. package/dist/services/auth.js.map +1 -0
  79. package/dist/services/chat.d.ts +18 -0
  80. package/dist/services/chat.d.ts.map +1 -0
  81. package/dist/services/chat.js +151 -0
  82. package/dist/services/chat.js.map +1 -0
  83. package/dist/services/cookie-rotation.d.ts +25 -0
  84. package/dist/services/cookie-rotation.d.ts.map +1 -0
  85. package/dist/services/cookie-rotation.js +138 -0
  86. package/dist/services/cookie-rotation.js.map +1 -0
  87. package/dist/services/gems.d.ts +33 -0
  88. package/dist/services/gems.d.ts.map +1 -0
  89. package/dist/services/gems.js +202 -0
  90. package/dist/services/gems.js.map +1 -0
  91. package/dist/services/generate.d.ts +27 -0
  92. package/dist/services/generate.d.ts.map +1 -0
  93. package/dist/services/generate.js +149 -0
  94. package/dist/services/generate.js.map +1 -0
  95. package/dist/services/health.d.ts +20 -0
  96. package/dist/services/health.d.ts.map +1 -0
  97. package/dist/services/health.js +77 -0
  98. package/dist/services/health.js.map +1 -0
  99. package/dist/services/upload.d.ts +17 -0
  100. package/dist/services/upload.d.ts.map +1 -0
  101. package/dist/services/upload.js +120 -0
  102. package/dist/services/upload.js.map +1 -0
  103. package/dist/setup.d.ts +7 -0
  104. package/dist/setup.d.ts.map +1 -0
  105. package/dist/setup.js +165 -0
  106. package/dist/setup.js.map +1 -0
  107. package/dist/state.d.ts +57 -0
  108. package/dist/state.d.ts.map +1 -0
  109. package/dist/state.js +260 -0
  110. package/dist/state.js.map +1 -0
  111. package/dist/tools/account.d.ts +4 -0
  112. package/dist/tools/account.d.ts.map +1 -0
  113. package/dist/tools/account.js +29 -0
  114. package/dist/tools/account.js.map +1 -0
  115. package/dist/tools/auth.d.ts +4 -0
  116. package/dist/tools/auth.d.ts.map +1 -0
  117. package/dist/tools/auth.js +56 -0
  118. package/dist/tools/auth.js.map +1 -0
  119. package/dist/tools/chat.d.ts +4 -0
  120. package/dist/tools/chat.d.ts.map +1 -0
  121. package/dist/tools/chat.js +101 -0
  122. package/dist/tools/chat.js.map +1 -0
  123. package/dist/tools/gems.d.ts +4 -0
  124. package/dist/tools/gems.d.ts.map +1 -0
  125. package/dist/tools/gems.js +93 -0
  126. package/dist/tools/gems.js.map +1 -0
  127. package/dist/tools/health.d.ts +4 -0
  128. package/dist/tools/health.d.ts.map +1 -0
  129. package/dist/tools/health.js +13 -0
  130. package/dist/tools/health.js.map +1 -0
  131. package/dist/tools/media.d.ts +4 -0
  132. package/dist/tools/media.d.ts.map +1 -0
  133. package/dist/tools/media.js +65 -0
  134. package/dist/tools/media.js.map +1 -0
  135. package/dist/types.d.ts +292 -0
  136. package/dist/types.d.ts.map +1 -0
  137. package/dist/types.js +2 -0
  138. package/dist/types.js.map +1 -0
  139. package/package.json +66 -0
@@ -0,0 +1,132 @@
1
+ import { RetryExhaustedError } from "../errors.js";
2
+ import { DEFAULT_MAX_RETRIES, DEFAULT_RETRY_DELAY_FACTOR } from "./constants.js";
3
+ const DEFAULT_RETRY_OPTIONS = {
4
+ maxRetries: DEFAULT_MAX_RETRIES,
5
+ delayFactor: DEFAULT_RETRY_DELAY_FACTOR
6
+ };
7
+ const RETRYABLE_PARSE_CODES = ["TEMPORARY_ERROR"];
8
+ const RATE_LIMIT_PARSE_CODES = ["RATE_LIMITED", "USAGE_LIMIT_EXCEEDED"];
9
+ const FAILOVER_ONLY_PARSE_CODES = ["IMAGE_GEN_BLOCKED"];
10
+ const FATAL_PARSE_CODES = [
11
+ "IP_BLOCKED",
12
+ "MODEL_INCONSISTENT",
13
+ "MODEL_HEADER_INVALID",
14
+ "UNKNOWN_API_ERROR",
15
+ "PARSE_ERROR"
16
+ ];
17
+ function sleep(ms) {
18
+ return new Promise((resolve) => setTimeout(resolve, ms));
19
+ }
20
+ function toError(value) {
21
+ if (value instanceof Error) {
22
+ return value;
23
+ }
24
+ if (typeof value === "string") {
25
+ return new Error(value);
26
+ }
27
+ if (typeof value === "object" && value !== null && "message" in value) {
28
+ const maybeMessage = value.message;
29
+ if (typeof maybeMessage === "string" && maybeMessage.length > 0) {
30
+ return new Error(maybeMessage);
31
+ }
32
+ }
33
+ return new Error("Unknown error");
34
+ }
35
+ function readParseCode(error) {
36
+ const asObj = error;
37
+ if (typeof asObj?.code === "string") {
38
+ return asObj.code;
39
+ }
40
+ if (asObj?.ok === false && typeof asObj.error === "object" && asObj.error !== null) {
41
+ const nestedCode = asObj.error.code;
42
+ if (typeof nestedCode === "string") {
43
+ return nestedCode;
44
+ }
45
+ }
46
+ return null;
47
+ }
48
+ function has429Status(error) {
49
+ const asObj = error;
50
+ if (asObj?.status === 429) {
51
+ return true;
52
+ }
53
+ const nestedStatus = asObj?.error?.status;
54
+ if (nestedStatus === 429) {
55
+ return true;
56
+ }
57
+ const message = toError(error).message;
58
+ return /\b429\b/.test(message) || /rate\s*limit/i.test(message);
59
+ }
60
+ export function classifyError(error) {
61
+ const parseCode = readParseCode(error);
62
+ if (parseCode) {
63
+ if (RETRYABLE_PARSE_CODES.includes(parseCode)) {
64
+ return "retryable";
65
+ }
66
+ if (RATE_LIMIT_PARSE_CODES.includes(parseCode)) {
67
+ return "rate_limited";
68
+ }
69
+ if (FAILOVER_ONLY_PARSE_CODES.includes(parseCode)) {
70
+ return "failover_only";
71
+ }
72
+ if (FATAL_PARSE_CODES.includes(parseCode)) {
73
+ return "fatal";
74
+ }
75
+ }
76
+ const asObj = error;
77
+ if (typeof asObj?.errorType === "string") {
78
+ if (asObj.errorType === "timeout") {
79
+ return "retryable";
80
+ }
81
+ if (asObj.errorType === "js_error") {
82
+ return "retryable";
83
+ }
84
+ }
85
+ if (has429Status(error)) {
86
+ return "rate_limited";
87
+ }
88
+ const normalized = toError(error);
89
+ if (/timeout|timed out|econnreset|enotfound|network|failed to fetch|networkerror|fetch error/i.test(normalized.message)) {
90
+ return "retryable";
91
+ }
92
+ if (normalized.name === "AbortError") {
93
+ return "retryable";
94
+ }
95
+ return "fatal";
96
+ }
97
+ export function calculateDelay(attempt, maxRetries, delayFactor) {
98
+ // Intentionally decreasing delay to match Python implementation:
99
+ // (maxRetries - currentAttempt + 1) * delayFactor.
100
+ return Math.max(0, (maxRetries - attempt + 1) * delayFactor);
101
+ }
102
+ export async function withRetry(fn, options = {}) {
103
+ const retryOptions = {
104
+ ...DEFAULT_RETRY_OPTIONS,
105
+ ...options
106
+ };
107
+ for (let attempt = 0; attempt <= retryOptions.maxRetries; attempt += 1) {
108
+ try {
109
+ return await fn();
110
+ }
111
+ catch (error) {
112
+ const normalized = toError(error);
113
+ const classification = classifyError(error);
114
+ const isExhausted = attempt >= retryOptions.maxRetries;
115
+ if (classification === "fatal") {
116
+ throw normalized;
117
+ }
118
+ if (classification === "failover_only") {
119
+ throw new RetryExhaustedError(attempt + 1, normalized);
120
+ }
121
+ if (isExhausted) {
122
+ throw new RetryExhaustedError(attempt + 1, normalized);
123
+ }
124
+ const baseDelay = calculateDelay(attempt, retryOptions.maxRetries, retryOptions.delayFactor);
125
+ const delay = classification === "rate_limited" ? baseDelay * 2 : baseDelay;
126
+ retryOptions.onRetry?.(attempt + 1, normalized, delay);
127
+ await sleep(delay * 1000);
128
+ }
129
+ }
130
+ throw new Error("Unreachable retry state");
131
+ }
132
+ //# sourceMappingURL=retry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retry.js","sourceRoot":"","sources":["../../src/core/retry.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ export interface FrameParseResult {
2
+ frames: unknown[];
3
+ remainder: string;
4
+ }
5
+ export declare class StreamParser {
6
+ extractFrames(rawResponse: string): unknown[];
7
+ parseFrameBuffer(content: string): FrameParseResult;
8
+ }
9
+ //# sourceMappingURL=stream-parser.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream-parser.d.ts","sourceRoot":"","sources":["../../src/core/stream-parser.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,gBAAgB;IAC/B,MAAM,EAAE,OAAO,EAAE,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,qBAAa,YAAY;IACvB,aAAa,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,EAAE;IAoB7C,gBAAgB,CAAC,OAAO,EAAE,MAAM,GAAG,gBAAgB;CAsDpD"}
@@ -0,0 +1,69 @@
1
+ import { ANTI_XSSI_PREFIX } from "./constants.js";
2
+ export class StreamParser {
3
+ extractFrames(rawResponse) {
4
+ let content = rawResponse;
5
+ if (content.startsWith(ANTI_XSSI_PREFIX)) {
6
+ content = content.slice(ANTI_XSSI_PREFIX.length);
7
+ }
8
+ content = content.trimStart();
9
+ const { frames } = this.parseFrameBuffer(content);
10
+ if (frames.length > 0) {
11
+ return frames;
12
+ }
13
+ try {
14
+ const parsed = JSON.parse(content.trim());
15
+ return Array.isArray(parsed) ? parsed : [parsed];
16
+ }
17
+ catch {
18
+ return [];
19
+ }
20
+ }
21
+ parseFrameBuffer(content) {
22
+ const frames = [];
23
+ let consumedPos = 0;
24
+ const lengthMarkerPattern = /(\d+)/y;
25
+ while (consumedPos < content.length) {
26
+ while (consumedPos < content.length && /\s/.test(content[consumedPos])) {
27
+ consumedPos += 1;
28
+ }
29
+ if (consumedPos >= content.length) {
30
+ break;
31
+ }
32
+ lengthMarkerPattern.lastIndex = consumedPos;
33
+ const match = lengthMarkerPattern.exec(content);
34
+ if (!match) {
35
+ break;
36
+ }
37
+ const lengthValue = match[1];
38
+ const length = Number.parseInt(lengthValue, 10);
39
+ const startContent = match.index + lengthValue.length;
40
+ const endContent = startContent + length;
41
+ if (endContent > content.length) {
42
+ consumedPos = match.index;
43
+ break;
44
+ }
45
+ const chunk = content.slice(startContent, endContent).trim();
46
+ consumedPos = endContent;
47
+ if (!chunk) {
48
+ continue;
49
+ }
50
+ try {
51
+ const parsed = JSON.parse(chunk);
52
+ if (Array.isArray(parsed)) {
53
+ frames.push(...parsed);
54
+ }
55
+ else {
56
+ frames.push(parsed);
57
+ }
58
+ }
59
+ catch {
60
+ continue;
61
+ }
62
+ }
63
+ return {
64
+ frames,
65
+ remainder: content.slice(consumedPos)
66
+ };
67
+ }
68
+ }
69
+ //# sourceMappingURL=stream-parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream-parser.js","sourceRoot":"","sources":["../../src/core/stream-parser.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { CamofoxClient } from "../client/camofox-client.js";
2
+ import type { GeminiTokens } from "../types.js";
3
+ export declare class TokenManager {
4
+ private client;
5
+ constructor(client: CamofoxClient);
6
+ extractTokens(tabId: string, userId: string): Promise<GeminiTokens>;
7
+ isValid(tokens: GeminiTokens): boolean;
8
+ }
9
+ //# sourceMappingURL=token-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-manager.d.ts","sourceRoot":"","sources":["../../src/core/token-manager.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,6BAA6B,CAAC;AAGjE,OAAO,KAAK,EAAE,YAAY,EAAyB,MAAM,aAAa,CAAC;AAEvE,qBAAa,YAAY;IACX,OAAO,CAAC,MAAM;gBAAN,MAAM,EAAE,aAAa;IAEnC,aAAa,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,CAAC;IAiEzE,OAAO,CAAC,MAAM,EAAE,YAAY,GAAG,OAAO;CAIvC"}
@@ -0,0 +1,64 @@
1
+ import { TOKEN_VALIDITY_MS } from "./constants.js";
2
+ import { AppError, GEMINI_ERROR } from "../errors.js";
3
+ export class TokenManager {
4
+ client;
5
+ constructor(client) {
6
+ this.client = client;
7
+ }
8
+ async extractTokens(tabId, userId) {
9
+ const extractionJs = `(() => {
10
+ const url = window.location.href;
11
+ if (url.includes('consent.google.com') || url.includes('accounts.google.com/signin') || url.includes('accounts.google.com/v3')) {
12
+ return { ok: false, error: 'not_authenticated', url: url };
13
+ }
14
+
15
+ const html = document.documentElement.innerHTML;
16
+ const snlm0eMatch = html.match(/"SNlM0e"\\s*:\\s*"(.*?)"/);
17
+ const cfb2hMatch = html.match(/"cfb2h"\\s*:\\s*"(.*?)"/);
18
+ const fdrfjeMatch = html.match(/"FdrFJe"\\s*:\\s*"(.*?)"/);
19
+
20
+ const snlm0e = snlm0eMatch ? snlm0eMatch[1] : '';
21
+ const cfb2h = cfb2hMatch ? cfb2hMatch[1] : null;
22
+ const fdrfje = fdrfjeMatch ? fdrfjeMatch[1] : null;
23
+
24
+ if (!cfb2h && !fdrfje) {
25
+ return {
26
+ ok: false,
27
+ error: 'no_tokens_found',
28
+ url: url,
29
+ hint: 'Page may not be fully loaded or user is not authenticated'
30
+ };
31
+ }
32
+
33
+ return {
34
+ ok: true,
35
+ tokens: { snlm0e: snlm0e, cfb2h: cfb2h, fdrfje: fdrfje },
36
+ url: url,
37
+ extractedAt: Date.now()
38
+ };
39
+ })()`;
40
+ const result = await this.client.evaluate(tabId, extractionJs, userId);
41
+ if (!result.ok) {
42
+ throw new AppError(GEMINI_ERROR.TOKEN_EXTRACTION_FAILED, `Token extraction evaluate failed: ${result.error ?? "unknown error"}`);
43
+ }
44
+ const data = result.result;
45
+ if (!data?.ok) {
46
+ if (data?.error === "not_authenticated") {
47
+ throw new AppError(GEMINI_ERROR.NOT_AUTHENTICATED, "User is not authenticated. Please log in to Gemini via CamoFox browser.");
48
+ }
49
+ const hint = data?.hint ? ` — ${data.hint}` : "";
50
+ throw new AppError(GEMINI_ERROR.TOKEN_EXTRACTION_FAILED, `Token extraction failed: ${data?.error ?? "unknown"}${hint}`);
51
+ }
52
+ return {
53
+ snlm0e: data.tokens?.snlm0e ?? "",
54
+ cfb2h: data.tokens?.cfb2h ?? "",
55
+ fdrfje: data.tokens?.fdrfje ?? "",
56
+ extractedAt: data.extractedAt ?? Date.now()
57
+ };
58
+ }
59
+ isValid(tokens) {
60
+ const age = Date.now() - tokens.extractedAt;
61
+ return age < TOKEN_VALIDITY_MS;
62
+ }
63
+ }
64
+ //# sourceMappingURL=token-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-manager.js","sourceRoot":"","sources":["../../src/core/token-manager.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAGtD,MAAM,OAAO,YAAY;IACH;IAApB,YAAoB,MAAqB;QAArB,WAAM,GAAN,MAAM,CAAe;IAAG,CAAC;IAE7C,KAAK,CAAC,aAAa,CAAC,KAAa,EAAE,MAAc;QAC/C,MAAM,YAAY,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;SA8BhB,CAAC;QAEN,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,EAAE,YAAY,EAAE,MAAM,CAAC,CAAC;QACvE,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;YACf,MAAM,IAAI,QAAQ,CAChB,YAAY,CAAC,uBAAuB,EACpC,qCAAqC,MAAM,CAAC,KAAK,IAAI,eAAe,EAAE,CACvE,CAAC;QACJ,CAAC;QAED,MAAM,IAAI,GAAG,MAAM,CAAC,MAA+B,CAAC;QACpD,IAAI,CAAC,IAAI,EAAE,EAAE,EAAE,CAAC;YACd,IAAI,IAAI,EAAE,KAAK,KAAK,mBAAmB,EAAE,CAAC;gBACxC,MAAM,IAAI,QAAQ,CAChB,YAAY,CAAC,iBAAiB,EAC9B,yEAAyE,CAC1E,CAAC;YACJ,CAAC;YAED,MAAM,IAAI,GAAG,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACjD,MAAM,IAAI,QAAQ,CAChB,YAAY,CAAC,uBAAuB,EACpC,4BAA4B,IAAI,EAAE,KAAK,IAAI,SAAS,GAAG,IAAI,EAAE,CAC9D,CAAC;QACJ,CAAC;QAED,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,MAAM,IAAI,EAAE;YACjC,KAAK,EAAE,IAAI,CAAC,MAAM,EAAE,KAAK,IAAI,EAAE;YAC/B,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,MAAM,IAAI,EAAE;YACjC,WAAW,EAAE,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,GAAG,EAAE;SAC5C,CAAC;IACJ,CAAC;IAED,OAAO,CAAC,MAAoB;QAC1B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,MAAM,CAAC,WAAW,CAAC;QAC5C,OAAO,GAAG,GAAG,iBAAiB,CAAC;IACjC,CAAC;CACF"}
@@ -0,0 +1,2 @@
1
+ export declare function getNestedValue<T = unknown>(data: unknown, path: (number | string)[], fallback?: T): T | undefined;
2
+ //# sourceMappingURL=utils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../src/core/utils.ts"],"names":[],"mappings":"AAAA,wBAAgB,cAAc,CAAC,CAAC,GAAG,OAAO,EACxC,IAAI,EAAE,OAAO,EACb,IAAI,EAAE,CAAC,MAAM,GAAG,MAAM,CAAC,EAAE,EACzB,QAAQ,CAAC,EAAE,CAAC,GACX,CAAC,GAAG,SAAS,CAaf"}
@@ -0,0 +1,15 @@
1
+ export function getNestedValue(data, path, fallback) {
2
+ let current = data;
3
+ for (const key of path) {
4
+ if (current == null || typeof current !== "object") {
5
+ return fallback;
6
+ }
7
+ current = current[key];
8
+ }
9
+ // Intentional TS behavior: a final resolved value of `null` is returned as-is,
10
+ // while only `undefined` triggers `fallback`. This differs from Python
11
+ // `get_nested_value` (`None` -> default). Callers wanting Python-equivalent
12
+ // semantics should check `result == null`.
13
+ return (current === undefined ? fallback : current);
14
+ }
15
+ //# sourceMappingURL=utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/core/utils.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,cAAc,CAC5B,IAAa,EACb,IAAyB,EACzB,QAAY;IAEZ,IAAI,OAAO,GAAY,IAAI,CAAC;IAC5B,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,IAAI,OAAO,IAAI,IAAI,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE,CAAC;YACnD,OAAO,QAAQ,CAAC;QAClB,CAAC;QACD,OAAO,GAAI,OAA4C,CAAC,GAAG,CAAC,CAAC;IAC/D,CAAC;IACD,+EAA+E;IAC/E,uEAAuE;IACvE,4EAA4E;IAC5E,2CAA2C;IAC3C,OAAO,CAAC,OAAO,KAAK,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAkB,CAAC;AACvE,CAAC"}
@@ -0,0 +1,31 @@
1
+ import type { AccountService } from "../services/account.js";
2
+ import type { AuthService } from "../services/auth.js";
3
+ import type { GenerateService } from "../services/generate.js";
4
+ import type { HealthService } from "../services/health.js";
5
+ import type { StateManager } from "../state.js";
6
+ export interface DashboardDeps {
7
+ state: StateManager;
8
+ auth: AuthService;
9
+ account: AccountService;
10
+ health: HealthService;
11
+ generate: GenerateService;
12
+ }
13
+ export declare class DashboardServer {
14
+ private deps;
15
+ private port;
16
+ private server;
17
+ private dashboardHtml;
18
+ constructor(deps: DashboardDeps, port?: number);
19
+ start(): Promise<void>;
20
+ stop(): void;
21
+ private handleRequest;
22
+ private handleHealth;
23
+ private handleListAccounts;
24
+ private handleLogin;
25
+ private handleLogout;
26
+ private handleRemoveAccount;
27
+ private handleChat;
28
+ private readBody;
29
+ private jsonResponse;
30
+ }
31
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/dashboard/server.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAC7D,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AACvD,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAC/D,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AAC3D,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAIhD,MAAM,WAAW,aAAa;IAC5B,KAAK,EAAE,YAAY,CAAC;IACpB,IAAI,EAAE,WAAW,CAAC;IAClB,OAAO,EAAE,cAAc,CAAC;IACxB,MAAM,EAAE,aAAa,CAAC;IACtB,QAAQ,EAAE,eAAe,CAAC;CAC3B;AAED,qBAAa,eAAe;IAKxB,OAAO,CAAC,IAAI;IACZ,OAAO,CAAC,IAAI;IALd,OAAO,CAAC,MAAM,CAAgD;IAC9D,OAAO,CAAC,aAAa,CAAuB;gBAGlC,IAAI,EAAE,aAAa,EACnB,IAAI,SAAO;IAGf,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAoB5B,IAAI,IAAI,IAAI;YAKE,aAAa;YAmEb,YAAY;IAK1B,OAAO,CAAC,kBAAkB;YAKZ,WAAW;YAUX,YAAY;YAUZ,mBAAmB;YAUnB,UAAU;IAoCxB,OAAO,CAAC,QAAQ;IAqBhB,OAAO,CAAC,YAAY;CAIrB"}
@@ -0,0 +1,190 @@
1
+ import { createServer } from "node:http";
2
+ import { readFile } from "node:fs/promises";
3
+ import { dirname, join } from "node:path";
4
+ import { fileURLToPath } from "node:url";
5
+ const __dirname = dirname(fileURLToPath(import.meta.url));
6
+ export class DashboardServer {
7
+ deps;
8
+ port;
9
+ server = null;
10
+ dashboardHtml = null;
11
+ constructor(deps, port = 9378) {
12
+ this.deps = deps;
13
+ this.port = port;
14
+ }
15
+ async start() {
16
+ try {
17
+ this.dashboardHtml = await readFile(join(__dirname, "index.html"), "utf-8");
18
+ }
19
+ catch {
20
+ this.dashboardHtml = "<html><body><h1>Dashboard HTML not found</h1></body></html>";
21
+ }
22
+ this.server = createServer((req, res) => {
23
+ void this.handleRequest(req, res);
24
+ });
25
+ await new Promise((resolve, reject) => {
26
+ this.server?.on("error", reject);
27
+ this.server?.listen(this.port, () => {
28
+ process.stderr.write(`[camo-gemini] Dashboard: http://localhost:${this.port}\n`);
29
+ resolve();
30
+ });
31
+ });
32
+ }
33
+ stop() {
34
+ this.server?.close();
35
+ this.server = null;
36
+ }
37
+ async handleRequest(req, res) {
38
+ const url = new URL(req.url ?? "/", `http://localhost:${this.port}`);
39
+ const path = url.pathname;
40
+ const method = req.method ?? "GET";
41
+ res.setHeader("Access-Control-Allow-Origin", "*");
42
+ res.setHeader("Access-Control-Allow-Methods", "GET, POST, DELETE, OPTIONS");
43
+ res.setHeader("Access-Control-Allow-Headers", "Content-Type");
44
+ if (method === "OPTIONS") {
45
+ res.writeHead(204);
46
+ res.end();
47
+ return;
48
+ }
49
+ try {
50
+ if (path === "/api/health" && method === "GET") {
51
+ await this.handleHealth(res);
52
+ return;
53
+ }
54
+ if (path === "/api/accounts" && method === "GET") {
55
+ this.handleListAccounts(res);
56
+ return;
57
+ }
58
+ if (/^\/api\/accounts\/\d+\/login$/.test(path) && method === "POST") {
59
+ const idx = Number.parseInt(path.split("/")[3] ?? "", 10);
60
+ await this.handleLogin(res, idx);
61
+ return;
62
+ }
63
+ if (/^\/api\/accounts\/\d+\/logout$/.test(path) && method === "POST") {
64
+ const idx = Number.parseInt(path.split("/")[3] ?? "", 10);
65
+ await this.handleLogout(res, idx);
66
+ return;
67
+ }
68
+ if (/^\/api\/accounts\/\d+$/.test(path) && method === "DELETE") {
69
+ const idx = Number.parseInt(path.split("/")[3] ?? "", 10);
70
+ await this.handleRemoveAccount(res, idx);
71
+ return;
72
+ }
73
+ if (path === "/api/chat" && method === "POST") {
74
+ await this.handleChat(req, res);
75
+ return;
76
+ }
77
+ if (path.startsWith("/api/")) {
78
+ this.jsonResponse(res, 404, { error: "Not found" });
79
+ return;
80
+ }
81
+ if (method === "GET") {
82
+ res.writeHead(200, { "Content-Type": "text/html; charset=utf-8" });
83
+ res.end(this.dashboardHtml);
84
+ return;
85
+ }
86
+ this.jsonResponse(res, 404, { error: "Not found" });
87
+ }
88
+ catch (err) {
89
+ const msg = err instanceof Error ? err.message : String(err);
90
+ this.jsonResponse(res, 500, { error: msg });
91
+ }
92
+ }
93
+ async handleHealth(res) {
94
+ const result = await this.deps.health.checkAllAccounts();
95
+ this.jsonResponse(res, 200, result);
96
+ }
97
+ handleListAccounts(res) {
98
+ const accounts = this.deps.account.listAccounts();
99
+ this.jsonResponse(res, 200, { accounts });
100
+ }
101
+ async handleLogin(res, accountIndex) {
102
+ try {
103
+ const info = await this.deps.account.addAccount(accountIndex);
104
+ this.jsonResponse(res, 200, info);
105
+ }
106
+ catch (err) {
107
+ const msg = err instanceof Error ? err.message : String(err);
108
+ this.jsonResponse(res, 400, { error: msg });
109
+ }
110
+ }
111
+ async handleLogout(res, accountIndex) {
112
+ try {
113
+ await this.deps.auth.logout(accountIndex);
114
+ this.jsonResponse(res, 200, { ok: true });
115
+ }
116
+ catch (err) {
117
+ const msg = err instanceof Error ? err.message : String(err);
118
+ this.jsonResponse(res, 400, { error: msg });
119
+ }
120
+ }
121
+ async handleRemoveAccount(res, accountIndex) {
122
+ try {
123
+ await this.deps.account.removeAccount(accountIndex);
124
+ this.jsonResponse(res, 200, { ok: true });
125
+ }
126
+ catch (err) {
127
+ const msg = err instanceof Error ? err.message : String(err);
128
+ this.jsonResponse(res, 400, { error: msg });
129
+ }
130
+ }
131
+ async handleChat(req, res) {
132
+ let body;
133
+ try {
134
+ body = await this.readBody(req);
135
+ }
136
+ catch {
137
+ this.jsonResponse(res, 413, { error: "Request body too large" });
138
+ return;
139
+ }
140
+ let parsed;
141
+ try {
142
+ parsed = JSON.parse(body);
143
+ }
144
+ catch {
145
+ this.jsonResponse(res, 400, { error: "Invalid JSON body" });
146
+ return;
147
+ }
148
+ const { prompt, accountIndex } = parsed;
149
+ if (typeof prompt !== "string" || !prompt) {
150
+ this.jsonResponse(res, 400, { error: "prompt is required" });
151
+ return;
152
+ }
153
+ try {
154
+ const result = await this.deps.generate.generate({ prompt, accountIndex });
155
+ this.jsonResponse(res, 200, {
156
+ text: result.output.candidates[result.output.chosenIndex]?.text ?? "",
157
+ candidates: result.output.candidates
158
+ });
159
+ }
160
+ catch (err) {
161
+ const msg = err instanceof Error ? err.message : String(err);
162
+ this.jsonResponse(res, 500, { error: msg });
163
+ }
164
+ }
165
+ readBody(req) {
166
+ return new Promise((resolve, reject) => {
167
+ const chunks = [];
168
+ let size = 0;
169
+ const MAX_BODY = 1_048_576;
170
+ req.on("data", (chunk) => {
171
+ size += chunk.length;
172
+ if (size > MAX_BODY) {
173
+ req.destroy();
174
+ reject(new Error("Request body too large"));
175
+ return;
176
+ }
177
+ chunks.push(chunk);
178
+ });
179
+ req.on("end", () => {
180
+ resolve(Buffer.concat(chunks).toString("utf-8"));
181
+ });
182
+ req.on("error", reject);
183
+ });
184
+ }
185
+ jsonResponse(res, status, data) {
186
+ res.writeHead(status, { "Content-Type": "application/json" });
187
+ res.end(JSON.stringify(data));
188
+ }
189
+ }
190
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/dashboard/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,32 @@
1
+ export type ErrorCode = "CONNECTION_REFUSED" | "TAB_NOT_FOUND" | "AUTH_REQUIRED" | "API_KEY_REQUIRED" | "TIMEOUT" | "VALIDATION_ERROR" | "NOT_IMPLEMENTED" | "GEMINI_AUTH_FAILED" | "GEMINI_RATE_LIMITED" | "GEMINI_TEMPORARY_ERROR" | "NOT_AUTHENTICATED" | "TOKEN_EXTRACTION_FAILED" | "COOKIE_ROTATION_FAILED" | "SESSION_EXPIRED" | "CAMOFOX_UNAVAILABLE" | "RETRY_EXHAUSTED" | "INTERNAL_ERROR";
2
+ export declare const GEMINI_ERROR: {
3
+ readonly NOT_AUTHENTICATED: "NOT_AUTHENTICATED";
4
+ readonly TOKEN_EXTRACTION_FAILED: "TOKEN_EXTRACTION_FAILED";
5
+ readonly COOKIE_ROTATION_FAILED: "COOKIE_ROTATION_FAILED";
6
+ readonly SESSION_EXPIRED: "SESSION_EXPIRED";
7
+ readonly CAMOFOX_UNAVAILABLE: "CAMOFOX_UNAVAILABLE";
8
+ };
9
+ export declare class AppError extends Error {
10
+ readonly code: ErrorCode;
11
+ readonly status?: number;
12
+ constructor(code: ErrorCode, message: string, status?: number);
13
+ }
14
+ export declare class RetryExhaustedError extends AppError {
15
+ constructor(attempts: number, lastError: Error);
16
+ }
17
+ export interface ToolResult {
18
+ [key: string]: unknown;
19
+ isError?: boolean;
20
+ content: Array<{
21
+ type: "text";
22
+ text: string;
23
+ } | {
24
+ type: "image";
25
+ data: string;
26
+ mimeType: string;
27
+ }>;
28
+ }
29
+ export declare function okResult(data: unknown): ToolResult;
30
+ export declare function toErrorResult(error: unknown): ToolResult;
31
+ export declare function normalizeError(error: unknown): AppError;
32
+ //# sourceMappingURL=errors.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../src/errors.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,SAAS,GACjB,oBAAoB,GACpB,eAAe,GACf,eAAe,GACf,kBAAkB,GAClB,SAAS,GACT,kBAAkB,GAClB,iBAAiB,GACjB,oBAAoB,GACpB,qBAAqB,GACrB,wBAAwB,GACxB,mBAAmB,GACnB,yBAAyB,GACzB,wBAAwB,GACxB,iBAAiB,GACjB,qBAAqB,GACrB,iBAAiB,GACjB,gBAAgB,CAAC;AAErB,eAAO,MAAM,YAAY;;;;;;CAMf,CAAC;AAEX,qBAAa,QAAS,SAAQ,KAAK;IACjC,SAAgB,IAAI,EAAE,SAAS,CAAC;IAEhC,SAAgB,MAAM,CAAC,EAAE,MAAM,CAAC;gBAEpB,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM;CAM9D;AAED,qBAAa,mBAAoB,SAAQ,QAAQ;gBACnC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,KAAK;CAK/C;AAED,MAAM,WAAW,UAAU;IACzB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;IACvB,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,OAAO,EAAE,KAAK,CACV;QACE,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;KACd,GACD;QACE,IAAI,EAAE,OAAO,CAAC;QACd,IAAI,EAAE,MAAM,CAAC;QACb,QAAQ,EAAE,MAAM,CAAC;KAClB,CACJ,CAAC;CACH;AAED,wBAAgB,QAAQ,CAAC,IAAI,EAAE,OAAO,GAAG,UAAU,CAIlD;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,OAAO,GAAG,UAAU,CAYxD;AAED,wBAAgB,cAAc,CAAC,KAAK,EAAE,OAAO,GAAG,QAAQ,CAiBvD"}