@mathzdevolper/phantomcode-cli 1.0.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/commands/chat.d.ts +9 -0
- package/dist/commands/chat.js +68 -0
- package/dist/commands/chat.js.map +1 -0
- package/dist/commands/config.d.ts +1 -0
- package/dist/commands/config.js +57 -0
- package/dist/commands/config.js.map +1 -0
- package/dist/commands/interactive.d.ts +7 -0
- package/dist/commands/interactive.js +155 -0
- package/dist/commands/interactive.js.map +1 -0
- package/dist/commands/models.d.ts +5 -0
- package/dist/commands/models.js +38 -0
- package/dist/commands/models.js.map +1 -0
- package/dist/commands/session.d.ts +1 -0
- package/dist/commands/session.js +77 -0
- package/dist/commands/session.js.map +1 -0
- package/dist/config/manager.d.ts +14 -0
- package/dist/config/manager.js +49 -0
- package/dist/config/manager.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +90 -0
- package/dist/index.js.map +1 -0
- package/dist/provider/dogrouter.d.ts +28 -0
- package/dist/provider/dogrouter.js +58 -0
- package/dist/provider/dogrouter.js.map +1 -0
- package/dist/session/manager.d.ts +20 -0
- package/dist/session/manager.js +112 -0
- package/dist/session/manager.js.map +1 -0
- package/dist/tui/colors.d.ts +15 -0
- package/dist/tui/colors.js +47 -0
- package/dist/tui/colors.js.map +1 -0
- package/package.json +34 -0
- package/src/commands/chat.ts +81 -0
- package/src/commands/config.ts +74 -0
- package/src/commands/interactive.ts +140 -0
- package/src/commands/models.ts +41 -0
- package/src/commands/session.ts +93 -0
- package/src/config/manager.ts +49 -0
- package/src/index.ts +109 -0
- package/src/provider/dogrouter.ts +82 -0
- package/src/session/manager.ts +93 -0
- package/src/tui/colors.ts +43 -0
- package/tsconfig.json +17 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export interface ChatOptions {
|
|
2
|
+
model?: string;
|
|
3
|
+
systemPrompt?: string;
|
|
4
|
+
temperature?: number;
|
|
5
|
+
maxTokens?: number;
|
|
6
|
+
stream?: boolean;
|
|
7
|
+
}
|
|
8
|
+
export interface ChatResponse {
|
|
9
|
+
content: string;
|
|
10
|
+
model: string;
|
|
11
|
+
usage?: {
|
|
12
|
+
promptTokens: number;
|
|
13
|
+
completionTokens: number;
|
|
14
|
+
totalTokens: number;
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
export declare function chat(apiKey: string, messages: {
|
|
18
|
+
role: "user" | "assistant" | "system";
|
|
19
|
+
content: string;
|
|
20
|
+
}[], options?: ChatOptions): Promise<ChatResponse>;
|
|
21
|
+
export declare function chatStream(apiKey: string, messages: {
|
|
22
|
+
role: "user" | "assistant" | "system";
|
|
23
|
+
content: string;
|
|
24
|
+
}[], options?: ChatOptions): AsyncGenerator<string>;
|
|
25
|
+
export declare function listModels(apiKey: string): Promise<{
|
|
26
|
+
id: string;
|
|
27
|
+
ownedBy: string;
|
|
28
|
+
}[]>;
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.chat = chat;
|
|
7
|
+
exports.chatStream = chatStream;
|
|
8
|
+
exports.listModels = listModels;
|
|
9
|
+
const openai_1 = __importDefault(require("openai"));
|
|
10
|
+
const BASE_URL = "https://api.dogrouter.ai/v1";
|
|
11
|
+
let client = null;
|
|
12
|
+
function getClient(apiKey) {
|
|
13
|
+
if (!client) {
|
|
14
|
+
client = new openai_1.default({ apiKey, baseURL: BASE_URL });
|
|
15
|
+
}
|
|
16
|
+
return client;
|
|
17
|
+
}
|
|
18
|
+
async function chat(apiKey, messages, options = {}) {
|
|
19
|
+
const c = getClient(apiKey);
|
|
20
|
+
const completion = await c.chat.completions.create({
|
|
21
|
+
model: options.model || "dogrouter/auto",
|
|
22
|
+
messages: messages.map((m) => ({ role: m.role, content: m.content })),
|
|
23
|
+
temperature: options.temperature ?? 0.7,
|
|
24
|
+
max_tokens: options.maxTokens,
|
|
25
|
+
});
|
|
26
|
+
return {
|
|
27
|
+
content: completion.choices[0]?.message?.content || "",
|
|
28
|
+
model: completion.model,
|
|
29
|
+
usage: completion.usage
|
|
30
|
+
? {
|
|
31
|
+
promptTokens: completion.usage.prompt_tokens,
|
|
32
|
+
completionTokens: completion.usage.completion_tokens,
|
|
33
|
+
totalTokens: completion.usage.total_tokens,
|
|
34
|
+
}
|
|
35
|
+
: undefined,
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
async function* chatStream(apiKey, messages, options = {}) {
|
|
39
|
+
const c = getClient(apiKey);
|
|
40
|
+
const stream = await c.chat.completions.create({
|
|
41
|
+
model: options.model || "dogrouter/auto",
|
|
42
|
+
messages: messages.map((m) => ({ role: m.role, content: m.content })),
|
|
43
|
+
temperature: options.temperature ?? 0.7,
|
|
44
|
+
max_tokens: options.maxTokens,
|
|
45
|
+
stream: true,
|
|
46
|
+
});
|
|
47
|
+
for await (const chunk of stream) {
|
|
48
|
+
const delta = chunk.choices[0]?.delta?.content;
|
|
49
|
+
if (delta)
|
|
50
|
+
yield delta;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
async function listModels(apiKey) {
|
|
54
|
+
const c = getClient(apiKey);
|
|
55
|
+
const models = await c.models.list();
|
|
56
|
+
return models.data.map((m) => ({ id: m.id, ownedBy: m.owned_by || "dogrouter" }));
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=dogrouter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"dogrouter.js","sourceRoot":"","sources":["../../src/provider/dogrouter.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
interface Message {
|
|
2
|
+
role: "user" | "assistant" | "system";
|
|
3
|
+
content: string;
|
|
4
|
+
timestamp: number;
|
|
5
|
+
}
|
|
6
|
+
interface Session {
|
|
7
|
+
id: string;
|
|
8
|
+
title: string;
|
|
9
|
+
model: string;
|
|
10
|
+
messages: Message[];
|
|
11
|
+
createdAt: number;
|
|
12
|
+
updatedAt: number;
|
|
13
|
+
}
|
|
14
|
+
export declare function createSession(model: string, title?: string): Session;
|
|
15
|
+
export declare function getSession(id: string): Session | null;
|
|
16
|
+
export declare function listSessions(): Session[];
|
|
17
|
+
export declare function addMessage(sessionId: string, role: Message["role"], content: string): Session | null;
|
|
18
|
+
export declare function deleteSession(id: string): void;
|
|
19
|
+
export declare function getMessages(sessionId: string): Message[];
|
|
20
|
+
export {};
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.createSession = createSession;
|
|
37
|
+
exports.getSession = getSession;
|
|
38
|
+
exports.listSessions = listSessions;
|
|
39
|
+
exports.addMessage = addMessage;
|
|
40
|
+
exports.deleteSession = deleteSession;
|
|
41
|
+
exports.getMessages = getMessages;
|
|
42
|
+
const uuid_1 = require("uuid");
|
|
43
|
+
const fs = __importStar(require("fs"));
|
|
44
|
+
const path = __importStar(require("path"));
|
|
45
|
+
const os = __importStar(require("os"));
|
|
46
|
+
const SESSIONS_DIR = path.join(os.homedir(), ".phantomcode", "sessions");
|
|
47
|
+
function ensureDir() {
|
|
48
|
+
if (!fs.existsSync(SESSIONS_DIR)) {
|
|
49
|
+
fs.mkdirSync(SESSIONS_DIR, { recursive: true });
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
function sessionPath(id) {
|
|
53
|
+
return path.join(SESSIONS_DIR, `${id}.json`);
|
|
54
|
+
}
|
|
55
|
+
function createSession(model, title = "Nova sessao") {
|
|
56
|
+
ensureDir();
|
|
57
|
+
const session = {
|
|
58
|
+
id: (0, uuid_1.v4)(),
|
|
59
|
+
title,
|
|
60
|
+
model,
|
|
61
|
+
messages: [],
|
|
62
|
+
createdAt: Date.now(),
|
|
63
|
+
updatedAt: Date.now(),
|
|
64
|
+
};
|
|
65
|
+
fs.writeFileSync(sessionPath(session.id), JSON.stringify(session, null, 2));
|
|
66
|
+
return session;
|
|
67
|
+
}
|
|
68
|
+
function getSession(id) {
|
|
69
|
+
try {
|
|
70
|
+
return JSON.parse(fs.readFileSync(sessionPath(id), "utf-8"));
|
|
71
|
+
}
|
|
72
|
+
catch {
|
|
73
|
+
return null;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
function listSessions() {
|
|
77
|
+
ensureDir();
|
|
78
|
+
const files = fs.readdirSync(SESSIONS_DIR).filter((f) => f.endsWith(".json"));
|
|
79
|
+
return files
|
|
80
|
+
.map((f) => {
|
|
81
|
+
try {
|
|
82
|
+
return JSON.parse(fs.readFileSync(path.join(SESSIONS_DIR, f), "utf-8"));
|
|
83
|
+
}
|
|
84
|
+
catch {
|
|
85
|
+
return null;
|
|
86
|
+
}
|
|
87
|
+
})
|
|
88
|
+
.filter((s) => s !== null)
|
|
89
|
+
.sort((a, b) => b.updatedAt - a.updatedAt);
|
|
90
|
+
}
|
|
91
|
+
function addMessage(sessionId, role, content) {
|
|
92
|
+
const session = getSession(sessionId);
|
|
93
|
+
if (!session)
|
|
94
|
+
return null;
|
|
95
|
+
session.messages.push({ role, content, timestamp: Date.now() });
|
|
96
|
+
session.updatedAt = Date.now();
|
|
97
|
+
if (session.messages.length === 1) {
|
|
98
|
+
session.title = content.slice(0, 50) + (content.length > 50 ? "..." : "");
|
|
99
|
+
}
|
|
100
|
+
fs.writeFileSync(sessionPath(sessionId), JSON.stringify(session, null, 2));
|
|
101
|
+
return session;
|
|
102
|
+
}
|
|
103
|
+
function deleteSession(id) {
|
|
104
|
+
const p = sessionPath(id);
|
|
105
|
+
if (fs.existsSync(p))
|
|
106
|
+
fs.unlinkSync(p);
|
|
107
|
+
}
|
|
108
|
+
function getMessages(sessionId) {
|
|
109
|
+
const session = getSession(sessionId);
|
|
110
|
+
return session?.messages || [];
|
|
111
|
+
}
|
|
112
|
+
//# sourceMappingURL=manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"manager.js","sourceRoot":"","sources":["../../src/session/manager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import chalk from "chalk";
|
|
2
|
+
export declare const style: {
|
|
3
|
+
primary: (s: string) => string;
|
|
4
|
+
success: (s: string) => string;
|
|
5
|
+
warning: (s: string) => string;
|
|
6
|
+
error: (s: string) => string;
|
|
7
|
+
muted: (s: string) => string;
|
|
8
|
+
highlight: (s: string) => string;
|
|
9
|
+
label: (s: string) => string;
|
|
10
|
+
bold: chalk.Chalk;
|
|
11
|
+
dim: chalk.Chalk;
|
|
12
|
+
code: (s: string) => string;
|
|
13
|
+
header: (s: string) => string;
|
|
14
|
+
separator: () => string;
|
|
15
|
+
};
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.style = void 0;
|
|
7
|
+
const chalk_1 = __importDefault(require("chalk"));
|
|
8
|
+
const manager_1 = require("../config/manager");
|
|
9
|
+
const theme = {
|
|
10
|
+
light: {
|
|
11
|
+
primary: chalk_1.default.hex("#2563eb"),
|
|
12
|
+
success: chalk_1.default.hex("#16a34a"),
|
|
13
|
+
warning: chalk_1.default.hex("#ea580c"),
|
|
14
|
+
error: chalk_1.default.hex("#dc2626"),
|
|
15
|
+
muted: chalk_1.default.hex("#64748b"),
|
|
16
|
+
highlight: chalk_1.default.hex("#1e293b"),
|
|
17
|
+
label: chalk_1.default.hex("#475569"),
|
|
18
|
+
},
|
|
19
|
+
dark: {
|
|
20
|
+
primary: chalk_1.default.hex("#60a5fa"),
|
|
21
|
+
success: chalk_1.default.hex("#4ade80"),
|
|
22
|
+
warning: chalk_1.default.hex("#fb923c"),
|
|
23
|
+
error: chalk_1.default.hex("#f87171"),
|
|
24
|
+
muted: chalk_1.default.hex("#94a3b8"),
|
|
25
|
+
highlight: chalk_1.default.hex("#e2e8f0"),
|
|
26
|
+
label: chalk_1.default.hex("#cbd5e1"),
|
|
27
|
+
},
|
|
28
|
+
};
|
|
29
|
+
function getColors() {
|
|
30
|
+
const t = (0, manager_1.getTheme)() === "light" ? theme.light : theme.dark;
|
|
31
|
+
return t;
|
|
32
|
+
}
|
|
33
|
+
exports.style = {
|
|
34
|
+
primary: (s) => getColors().primary(s),
|
|
35
|
+
success: (s) => getColors().success(s),
|
|
36
|
+
warning: (s) => getColors().warning(s),
|
|
37
|
+
error: (s) => getColors().error(s),
|
|
38
|
+
muted: (s) => getColors().muted(s),
|
|
39
|
+
highlight: (s) => getColors().highlight(s),
|
|
40
|
+
label: (s) => getColors().label(s),
|
|
41
|
+
bold: chalk_1.default.bold,
|
|
42
|
+
dim: chalk_1.default.dim,
|
|
43
|
+
code: (s) => chalk_1.default.cyan(s),
|
|
44
|
+
header: (s) => chalk_1.default.bold.hex("#60a5fa")(s),
|
|
45
|
+
separator: () => chalk_1.default.dim("─".repeat(process.stdout.columns || 60)),
|
|
46
|
+
};
|
|
47
|
+
//# sourceMappingURL=colors.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"colors.js","sourceRoot":"","sources":["../../src/tui/colors.ts"],"names":[],"mappings":";;;;;;AAAA,kDAA0B;AAC1B,+CAA6C;AAE7C,MAAM,KAAK,GAAG;IACZ,KAAK,EAAE;QACL,OAAO,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC7B,OAAO,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC7B,OAAO,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC7B,KAAK,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC3B,KAAK,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC3B,SAAS,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC/B,KAAK,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;KAC5B;IACD,IAAI,EAAE;QACJ,OAAO,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC7B,OAAO,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC7B,OAAO,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC7B,KAAK,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC3B,KAAK,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC3B,SAAS,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;QAC/B,KAAK,EAAE,eAAK,CAAC,GAAG,CAAC,SAAS,CAAC;KAC5B;CACF,CAAC;AAEF,SAAS,SAAS;IAChB,MAAM,CAAC,GAAG,IAAA,kBAAQ,GAAE,KAAK,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC;IAC5D,OAAO,CAAC,CAAC;AACX,CAAC;AAEY,QAAA,KAAK,GAAG;IACnB,OAAO,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,OAAO,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,OAAO,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,KAAK,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1C,KAAK,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1C,SAAS,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC;IAClD,KAAK,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,SAAS,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1C,IAAI,EAAE,eAAK,CAAC,IAAI;IAChB,GAAG,EAAE,eAAK,CAAC,GAAG;IACd,IAAI,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,eAAK,CAAC,IAAI,CAAC,CAAC,CAAC;IAClC,MAAM,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,eAAK,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;IACnD,SAAS,EAAE,GAAG,EAAE,CAAC,eAAK,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC;CACrE,CAAC"}
|
package/package.json
ADDED
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@mathzdevolper/phantomcode-cli",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"description": "PhantomCode - CLI para DogRouter AI Integration Platform",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"bin": {
|
|
7
|
+
"phantomcode": "dist/index.js"
|
|
8
|
+
},
|
|
9
|
+
"scripts": {
|
|
10
|
+
"build": "tsc",
|
|
11
|
+
"start": "node dist/index.js",
|
|
12
|
+
"dev": "ts-node src/index.ts",
|
|
13
|
+
"prepublishOnly": "npm run build"
|
|
14
|
+
},
|
|
15
|
+
"keywords": ["dogrouter", "cli", "ai", "phantomcode"],
|
|
16
|
+
"author": "",
|
|
17
|
+
"license": "MIT",
|
|
18
|
+
"type": "commonjs",
|
|
19
|
+
"dependencies": {
|
|
20
|
+
"chalk": "^4.1.2",
|
|
21
|
+
"conf": "^15.1.0",
|
|
22
|
+
"openai": "^6.42.0",
|
|
23
|
+
"ora": "^5.4.1",
|
|
24
|
+
"uuid": "^11.1.0",
|
|
25
|
+
"yargs": "^17.7.2"
|
|
26
|
+
},
|
|
27
|
+
"devDependencies": {
|
|
28
|
+
"@types/node": "^22.12.0",
|
|
29
|
+
"@types/uuid": "^10.0.0",
|
|
30
|
+
"@types/yargs": "^17.0.33",
|
|
31
|
+
"ts-node": "^10.9.2",
|
|
32
|
+
"typescript": "^5.7.3"
|
|
33
|
+
}
|
|
34
|
+
}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { getApiKey, getDefaultModel } from "../config/manager";
|
|
2
|
+
import { chat } from "../provider/dogrouter";
|
|
3
|
+
import { style } from "../tui/colors";
|
|
4
|
+
import { createSession, addMessage, getSession } from "../session/manager";
|
|
5
|
+
|
|
6
|
+
interface ChatArgs {
|
|
7
|
+
message: string;
|
|
8
|
+
model?: string;
|
|
9
|
+
noSave?: boolean;
|
|
10
|
+
session?: string;
|
|
11
|
+
system?: string;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export async function handleChat(args: ChatArgs): Promise<void> {
|
|
15
|
+
let apiKey: string;
|
|
16
|
+
try {
|
|
17
|
+
apiKey = getApiKey();
|
|
18
|
+
} catch {
|
|
19
|
+
console.error(style.error("✗ API key nao configurada"));
|
|
20
|
+
console.log(style.muted(" Execute: phantomcode config set <sua-api-key>"));
|
|
21
|
+
process.exit(1);
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const model = args.model || getDefaultModel();
|
|
25
|
+
const messages: { role: "user" | "assistant" | "system"; content: string }[] = [];
|
|
26
|
+
|
|
27
|
+
if (args.session) {
|
|
28
|
+
const session = getSession(args.session);
|
|
29
|
+
if (session) {
|
|
30
|
+
messages.push(...session.messages.map((m) => ({ role: m.role, content: m.content })));
|
|
31
|
+
console.log(style.muted(`↻ Continuando sessao: ${session.title}`));
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if (args.system) {
|
|
36
|
+
messages.unshift({ role: "system", content: args.system });
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
messages.push({ role: "user", content: args.message });
|
|
40
|
+
|
|
41
|
+
console.log(style.separator());
|
|
42
|
+
console.log(`${style.label("Voce:")} ${args.message}`);
|
|
43
|
+
console.log(style.separator());
|
|
44
|
+
|
|
45
|
+
try {
|
|
46
|
+
const response = await chat(apiKey, messages, { model });
|
|
47
|
+
|
|
48
|
+
console.log(`${style.primary("Phantom:")} ${response.content}`);
|
|
49
|
+
|
|
50
|
+
if (response.usage) {
|
|
51
|
+
console.log(style.separator());
|
|
52
|
+
console.log(
|
|
53
|
+
style.muted(
|
|
54
|
+
`↳ Modelo: ${response.model} | Tokens: ${response.usage.promptTokens}→${response.usage.completionTokens} (${response.usage.totalTokens})`
|
|
55
|
+
)
|
|
56
|
+
);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (!args.noSave) {
|
|
60
|
+
if (!args.session) {
|
|
61
|
+
const session = createSession(model, args.message);
|
|
62
|
+
addMessage(session.id, "user", args.message);
|
|
63
|
+
addMessage(session.id, "assistant", response.content);
|
|
64
|
+
console.log(style.muted(`↳ Sessao salva: ${session.id}`));
|
|
65
|
+
} else {
|
|
66
|
+
addMessage(args.session, "user", args.message);
|
|
67
|
+
addMessage(args.session, "assistant", response.content);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
} catch (error: unknown) {
|
|
71
|
+
const err = error as { message?: string; status?: number };
|
|
72
|
+
if (err.status === 401) {
|
|
73
|
+
console.error(style.error("✗ API key invalida. Use: phantomcode config set <api-key>"));
|
|
74
|
+
} else if (err.status === 429) {
|
|
75
|
+
console.error(style.error("✗ Limite de requisicoes excedido. Tente novamente mais tarde"));
|
|
76
|
+
} else {
|
|
77
|
+
console.error(style.error(`✗ Erro: ${err.message || "Erro desconhecido"}`));
|
|
78
|
+
}
|
|
79
|
+
process.exit(1);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { setApiKey, getApiKey, removeApiKey, hasApiKey, getDefaultModel, setDefaultModel, getAll } from "../config/manager";
|
|
2
|
+
import { style } from "../tui/colors";
|
|
3
|
+
|
|
4
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
5
|
+
export function buildConfigCommand(yargs: any): any {
|
|
6
|
+
return yargs
|
|
7
|
+
.command(
|
|
8
|
+
"set <api-key>",
|
|
9
|
+
"Define a API key do DogRouter",
|
|
10
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
11
|
+
(y: any) =>
|
|
12
|
+
y.positional("api-key", {
|
|
13
|
+
describe: "Sua API key do DogRouter",
|
|
14
|
+
type: "string",
|
|
15
|
+
demandOption: true,
|
|
16
|
+
}),
|
|
17
|
+
(args: { apiKey: string }) => {
|
|
18
|
+
setApiKey(args.apiKey);
|
|
19
|
+
console.log(style.success("✓ API key configurada com sucesso"));
|
|
20
|
+
}
|
|
21
|
+
)
|
|
22
|
+
.command(
|
|
23
|
+
"get",
|
|
24
|
+
"Exibe as configuracoes atuais",
|
|
25
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
26
|
+
(y: any) =>
|
|
27
|
+
y
|
|
28
|
+
.option("api-key", { type: "boolean", desc: "Exibir apenas a API key" })
|
|
29
|
+
.option("model", { type: "boolean", desc: "Exibir apenas o modelo padrao" })
|
|
30
|
+
.option("all", { type: "boolean", desc: "Exibir todas as configuracoes" }),
|
|
31
|
+
(args: { apiKey?: boolean; model?: boolean; all?: boolean }) => {
|
|
32
|
+
if (args.apiKey) {
|
|
33
|
+
console.log(style.label(`API Key: ${hasApiKey() ? style.code(getApiKey()) : style.warning("nao definida")}`));
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
if (args.model) {
|
|
37
|
+
console.log(style.label(`Modelo padrao: ${style.code(getDefaultModel())}`));
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
const all = getAll();
|
|
41
|
+
console.log(style.header("┌─ Configuracao ──────────────────────"));
|
|
42
|
+
console.log(style.label(`│ API Key: ${all.apiKey ? style.code(all.apiKey.slice(0, 12) + "...") : style.muted("nao definida")}`));
|
|
43
|
+
console.log(style.label(`│ Modelo padrao: ${style.code(all.defaultModel || "dogrouter/auto")}`));
|
|
44
|
+
console.log(style.label(`│ Tema: ${style.code(all.theme || "dark")}`));
|
|
45
|
+
console.log(style.header("└────────────────────────────────────"));
|
|
46
|
+
}
|
|
47
|
+
)
|
|
48
|
+
.command(
|
|
49
|
+
"reset",
|
|
50
|
+
"Remove a API key configurada",
|
|
51
|
+
() => ({}),
|
|
52
|
+
() => {
|
|
53
|
+
removeApiKey();
|
|
54
|
+
console.log(style.warning("⚠ API key removida"));
|
|
55
|
+
}
|
|
56
|
+
)
|
|
57
|
+
.command(
|
|
58
|
+
"set-model <model>",
|
|
59
|
+
"Define o modelo padrao",
|
|
60
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
61
|
+
(y: any) =>
|
|
62
|
+
y.positional("model", {
|
|
63
|
+
describe: "Nome do modelo (ex: dogrouter/auto)",
|
|
64
|
+
type: "string",
|
|
65
|
+
demandOption: true,
|
|
66
|
+
}),
|
|
67
|
+
(args: { model: string }) => {
|
|
68
|
+
setDefaultModel(args.model);
|
|
69
|
+
console.log(style.success(`✓ Modelo padrao alterado para: ${style.code(args.model)}`));
|
|
70
|
+
}
|
|
71
|
+
)
|
|
72
|
+
.demandCommand(1, "Use: phantomcode config <comando>")
|
|
73
|
+
.help();
|
|
74
|
+
}
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
import * as readline from "readline";
|
|
2
|
+
import { getApiKey, getDefaultModel } from "../config/manager";
|
|
3
|
+
import { chat } from "../provider/dogrouter";
|
|
4
|
+
import { style } from "../tui/colors";
|
|
5
|
+
import { createSession, addMessage, getSession } from "../session/manager";
|
|
6
|
+
|
|
7
|
+
interface InteractiveArgs {
|
|
8
|
+
model?: string;
|
|
9
|
+
session?: string;
|
|
10
|
+
system?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export async function runInteractive(args: InteractiveArgs): Promise<void> {
|
|
14
|
+
let apiKey: string;
|
|
15
|
+
try {
|
|
16
|
+
apiKey = getApiKey();
|
|
17
|
+
} catch {
|
|
18
|
+
console.error(style.error("✗ API key nao configurada"));
|
|
19
|
+
console.log(style.muted(" Execute: phantomcode config set <sua-api-key>"));
|
|
20
|
+
process.exit(1);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const model = args.model || getDefaultModel();
|
|
24
|
+
let sessionId: string | null = null;
|
|
25
|
+
const messages: { role: "user" | "assistant" | "system"; content: string }[] = [];
|
|
26
|
+
|
|
27
|
+
if (args.session) {
|
|
28
|
+
const existing = getSession(args.session);
|
|
29
|
+
if (existing) {
|
|
30
|
+
sessionId = existing.id;
|
|
31
|
+
messages.push(...existing.messages.map((m) => ({ role: m.role, content: m.content })));
|
|
32
|
+
console.log(style.muted(`↻ Continuando sessao: ${existing.title}\n`));
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (args.system) {
|
|
37
|
+
messages.unshift({ role: "system", content: args.system });
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
printBanner();
|
|
41
|
+
|
|
42
|
+
const rl = readline.createInterface({
|
|
43
|
+
input: process.stdin,
|
|
44
|
+
output: process.stdout,
|
|
45
|
+
prompt: style.primary("» "),
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
rl.prompt();
|
|
49
|
+
|
|
50
|
+
rl.on("line", async (line: string) => {
|
|
51
|
+
const input = line.trim();
|
|
52
|
+
|
|
53
|
+
if (input === "" || input === "/exit" || input === "/quit") {
|
|
54
|
+
rl.close();
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
if (input === "/help") {
|
|
59
|
+
printHelp();
|
|
60
|
+
rl.prompt();
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
if (input === "/clear") {
|
|
65
|
+
console.clear();
|
|
66
|
+
printBanner();
|
|
67
|
+
rl.prompt();
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
if (input === "/new") {
|
|
72
|
+
messages.length = 0;
|
|
73
|
+
if (args.system) messages.unshift({ role: "system", content: args.system });
|
|
74
|
+
sessionId = null;
|
|
75
|
+
console.log(style.success("✓ Nova conversa iniciada\n"));
|
|
76
|
+
rl.prompt();
|
|
77
|
+
return;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
if (input.startsWith("/model ")) {
|
|
81
|
+
const newModel = input.slice(7).trim();
|
|
82
|
+
if (newModel) {
|
|
83
|
+
console.log(style.muted(`Modelo alterado para: ${style.code(newModel)}\n`));
|
|
84
|
+
}
|
|
85
|
+
rl.prompt();
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
messages.push({ role: "user", content: input });
|
|
90
|
+
|
|
91
|
+
try {
|
|
92
|
+
const response = await chat(apiKey, messages, { model });
|
|
93
|
+
console.log(`\n${style.primary("Phantom:")} ${response.content}\n`);
|
|
94
|
+
messages.push({ role: "assistant", content: response.content });
|
|
95
|
+
|
|
96
|
+
if (sessionId) {
|
|
97
|
+
addMessage(sessionId, "user", input);
|
|
98
|
+
addMessage(sessionId, "assistant", response.content);
|
|
99
|
+
} else {
|
|
100
|
+
const session = createSession(model, input);
|
|
101
|
+
sessionId = session.id;
|
|
102
|
+
addMessage(sessionId, "user", input);
|
|
103
|
+
addMessage(sessionId, "assistant", response.content);
|
|
104
|
+
}
|
|
105
|
+
} catch (error: unknown) {
|
|
106
|
+
const err = error as { message?: string };
|
|
107
|
+
console.error(`\n${style.error(`✗ ${err.message || "Erro"}`)}\n`);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
rl.prompt();
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
rl.on("close", () => {
|
|
114
|
+
console.log(`\n${style.muted("Sessao encerrada")}`);
|
|
115
|
+
if (sessionId) {
|
|
116
|
+
console.log(style.muted(`ID da sessao: ${sessionId}`));
|
|
117
|
+
}
|
|
118
|
+
process.exit(0);
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
function printBanner(): void {
|
|
123
|
+
console.clear();
|
|
124
|
+
console.log(style.header("╔══════════════════════════════════════════════╗"));
|
|
125
|
+
console.log(style.header("║ PhantomCode - Modo Interativo ║"));
|
|
126
|
+
console.log(style.header("║ DogRouter AI Integration ║"));
|
|
127
|
+
console.log(style.header("╚══════════════════════════════════════════════╝"));
|
|
128
|
+
console.log(style.muted(" /help /exit /clear /new /model <nome>\n"));
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
function printHelp(): void {
|
|
132
|
+
console.log(style.header("\n┌─ Comandos Disponiveis ──────────────────────"));
|
|
133
|
+
console.log(` ${style.code("/exit")} ${style.muted("Sair do modo interativo")}`);
|
|
134
|
+
console.log(` ${style.code("/quit")} ${style.muted("Sair do modo interativo")}`);
|
|
135
|
+
console.log(` ${style.code("/clear")} ${style.muted("Limpar a tela")}`);
|
|
136
|
+
console.log(` ${style.code("/new")} ${style.muted("Iniciar nova conversa")}`);
|
|
137
|
+
console.log(` ${style.code("/model")} ${style.muted("Alterar o modelo (ex: /model dogrouter/auto)")}`);
|
|
138
|
+
console.log(` ${style.code("/help")} ${style.muted("Exibir esta ajuda")}`);
|
|
139
|
+
console.log(style.header("└───────────────────────────────────────────────┘\n"));
|
|
140
|
+
}
|