@hasna/connectors 0.4.2 → 0.5.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/bin/index.js +113 -1
- package/bin/mcp.js +113 -1
- package/bin/serve.js +112 -0
- package/connectors/connect-assemblyai/.env.example +11 -0
- package/connectors/connect-assemblyai/CLAUDE.md +128 -0
- package/connectors/connect-assemblyai/README.md +193 -0
- package/connectors/connect-assemblyai/package.json +50 -0
- package/connectors/connect-assemblyai/src/api/client.ts +192 -0
- package/connectors/connect-assemblyai/src/api/index.ts +71 -0
- package/connectors/connect-assemblyai/src/cli/index.ts +384 -0
- package/connectors/connect-assemblyai/src/index.ts +19 -0
- package/connectors/connect-assemblyai/src/types/index.ts +277 -0
- package/connectors/connect-assemblyai/src/utils/config.ts +103 -0
- package/connectors/connect-assemblyai/src/utils/output.ts +119 -0
- package/connectors/connect-assemblyai/tsconfig.json +16 -0
- package/connectors/connect-baseten/.env.example +11 -0
- package/connectors/connect-baseten/CLAUDE.md +128 -0
- package/connectors/connect-baseten/README.md +193 -0
- package/connectors/connect-baseten/package.json +51 -0
- package/connectors/connect-baseten/src/api/client.ts +71 -0
- package/connectors/connect-baseten/src/api/index.ts +40 -0
- package/connectors/connect-baseten/src/cli/index.ts +244 -0
- package/connectors/connect-baseten/src/index.ts +19 -0
- package/connectors/connect-baseten/src/types/index.ts +55 -0
- package/connectors/connect-baseten/src/utils/config.ts +103 -0
- package/connectors/connect-baseten/src/utils/output.ts +119 -0
- package/connectors/connect-baseten/tsconfig.json +16 -0
- package/connectors/connect-cerebras/.env.example +11 -0
- package/connectors/connect-cerebras/CLAUDE.md +128 -0
- package/connectors/connect-cerebras/README.md +193 -0
- package/connectors/connect-cerebras/package.json +51 -0
- package/connectors/connect-cerebras/src/api/client.ts +64 -0
- package/connectors/connect-cerebras/src/api/index.ts +32 -0
- package/connectors/connect-cerebras/src/cli/index.ts +244 -0
- package/connectors/connect-cerebras/src/index.ts +19 -0
- package/connectors/connect-cerebras/src/types/index.ts +65 -0
- package/connectors/connect-cerebras/src/utils/config.ts +103 -0
- package/connectors/connect-cerebras/src/utils/output.ts +119 -0
- package/connectors/connect-cerebras/tsconfig.json +16 -0
- package/connectors/connect-cohere/.env.example +11 -0
- package/connectors/connect-cohere/CLAUDE.md +128 -0
- package/connectors/connect-cohere/README.md +193 -0
- package/connectors/connect-cohere/package.json +53 -0
- package/connectors/connect-cohere/src/api/client.ts +109 -0
- package/connectors/connect-cohere/src/api/index.ts +59 -0
- package/connectors/connect-cohere/src/cli/index.ts +255 -0
- package/connectors/connect-cohere/src/index.ts +19 -0
- package/connectors/connect-cohere/src/types/index.ts +132 -0
- package/connectors/connect-cohere/src/utils/config.ts +197 -0
- package/connectors/connect-cohere/src/utils/output.ts +119 -0
- package/connectors/connect-cohere/tsconfig.json +16 -0
- package/connectors/connect-deepgram/.env.example +11 -0
- package/connectors/connect-deepgram/CLAUDE.md +128 -0
- package/connectors/connect-deepgram/README.md +193 -0
- package/connectors/connect-deepgram/package.json +51 -0
- package/connectors/connect-deepgram/src/api/client.ts +235 -0
- package/connectors/connect-deepgram/src/api/index.ts +57 -0
- package/connectors/connect-deepgram/src/cli/index.ts +339 -0
- package/connectors/connect-deepgram/src/index.ts +19 -0
- package/connectors/connect-deepgram/src/types/index.ts +232 -0
- package/connectors/connect-deepgram/src/utils/config.ts +103 -0
- package/connectors/connect-deepgram/src/utils/output.ts +119 -0
- package/connectors/connect-deepgram/tsconfig.json +16 -0
- package/connectors/connect-deepseek/.env.example +11 -0
- package/connectors/connect-deepseek/CLAUDE.md +128 -0
- package/connectors/connect-deepseek/README.md +193 -0
- package/connectors/connect-deepseek/package.json +51 -0
- package/connectors/connect-deepseek/src/api/client.ts +108 -0
- package/connectors/connect-deepseek/src/api/index.ts +36 -0
- package/connectors/connect-deepseek/src/cli/index.ts +167 -0
- package/connectors/connect-deepseek/src/index.ts +19 -0
- package/connectors/connect-deepseek/src/types/index.ts +72 -0
- package/connectors/connect-deepseek/src/utils/config.ts +103 -0
- package/connectors/connect-deepseek/src/utils/output.ts +119 -0
- package/connectors/connect-deepseek/tsconfig.json +16 -0
- package/connectors/connect-fal/.env.example +11 -0
- package/connectors/connect-fal/CLAUDE.md +128 -0
- package/connectors/connect-fal/README.md +193 -0
- package/connectors/connect-fal/package.json +51 -0
- package/connectors/connect-fal/src/api/client.ts +172 -0
- package/connectors/connect-fal/src/api/index.ts +55 -0
- package/connectors/connect-fal/src/cli/index.ts +341 -0
- package/connectors/connect-fal/src/index.ts +19 -0
- package/connectors/connect-fal/src/types/index.ts +135 -0
- package/connectors/connect-fal/src/utils/config.ts +103 -0
- package/connectors/connect-fal/src/utils/output.ts +119 -0
- package/connectors/connect-fal/tsconfig.json +16 -0
- package/connectors/connect-fireworks/.env.example +11 -0
- package/connectors/connect-fireworks/CLAUDE.md +128 -0
- package/connectors/connect-fireworks/README.md +193 -0
- package/connectors/connect-fireworks/package.json +51 -0
- package/connectors/connect-fireworks/src/api/client.ts +63 -0
- package/connectors/connect-fireworks/src/api/index.ts +36 -0
- package/connectors/connect-fireworks/src/cli/index.ts +244 -0
- package/connectors/connect-fireworks/src/index.ts +19 -0
- package/connectors/connect-fireworks/src/types/index.ts +70 -0
- package/connectors/connect-fireworks/src/utils/config.ts +103 -0
- package/connectors/connect-fireworks/src/utils/output.ts +119 -0
- package/connectors/connect-fireworks/tsconfig.json +16 -0
- package/connectors/connect-groq/.env.example +11 -0
- package/connectors/connect-groq/CLAUDE.md +128 -0
- package/connectors/connect-groq/README.md +193 -0
- package/connectors/connect-groq/package.json +52 -0
- package/connectors/connect-groq/src/api/client.ts +108 -0
- package/connectors/connect-groq/src/api/index.ts +36 -0
- package/connectors/connect-groq/src/cli/index.ts +171 -0
- package/connectors/connect-groq/src/index.ts +19 -0
- package/connectors/connect-groq/src/types/index.ts +69 -0
- package/connectors/connect-groq/src/utils/config.ts +103 -0
- package/connectors/connect-groq/src/utils/output.ts +119 -0
- package/connectors/connect-groq/tsconfig.json +16 -0
- package/connectors/connect-luma/.env.example +11 -0
- package/connectors/connect-luma/CLAUDE.md +128 -0
- package/connectors/connect-luma/README.md +193 -0
- package/connectors/connect-luma/package.json +53 -0
- package/connectors/connect-luma/src/api/client.ts +85 -0
- package/connectors/connect-luma/src/api/index.ts +44 -0
- package/connectors/connect-luma/src/cli/index.ts +300 -0
- package/connectors/connect-luma/src/index.ts +19 -0
- package/connectors/connect-luma/src/types/index.ts +60 -0
- package/connectors/connect-luma/src/utils/config.ts +103 -0
- package/connectors/connect-luma/src/utils/output.ts +119 -0
- package/connectors/connect-luma/tsconfig.json +16 -0
- package/connectors/connect-modal/.env.example +11 -0
- package/connectors/connect-modal/CLAUDE.md +128 -0
- package/connectors/connect-modal/README.md +193 -0
- package/connectors/connect-modal/package.json +51 -0
- package/connectors/connect-modal/src/api/client.ts +119 -0
- package/connectors/connect-modal/src/api/index.ts +69 -0
- package/connectors/connect-modal/src/cli/index.ts +224 -0
- package/connectors/connect-modal/src/index.ts +21 -0
- package/connectors/connect-modal/src/types/index.ts +60 -0
- package/connectors/connect-modal/src/utils/config.ts +114 -0
- package/connectors/connect-modal/src/utils/output.ts +119 -0
- package/connectors/connect-modal/tsconfig.json +16 -0
- package/connectors/connect-perplexity/.env.example +4 -0
- package/connectors/connect-perplexity/CLAUDE.md +156 -0
- package/connectors/connect-perplexity/README.md +184 -0
- package/connectors/connect-perplexity/package.json +58 -0
- package/connectors/connect-perplexity/scripts/publish.ts +210 -0
- package/connectors/connect-perplexity/src/api/client.ts +119 -0
- package/connectors/connect-perplexity/src/api/example.ts +118 -0
- package/connectors/connect-perplexity/src/api/index.ts +48 -0
- package/connectors/connect-perplexity/src/cli/index.ts +421 -0
- package/connectors/connect-perplexity/src/index.ts +24 -0
- package/connectors/connect-perplexity/src/types/index.ts +140 -0
- package/connectors/connect-perplexity/src/utils/config.ts +208 -0
- package/connectors/connect-perplexity/src/utils/output.ts +119 -0
- package/connectors/connect-perplexity/tsconfig.json +16 -0
- package/connectors/connect-replicate/.env.example +11 -0
- package/connectors/connect-replicate/CLAUDE.md +128 -0
- package/connectors/connect-replicate/README.md +193 -0
- package/connectors/connect-replicate/package.json +51 -0
- package/connectors/connect-replicate/src/api/client.ts +109 -0
- package/connectors/connect-replicate/src/api/index.ts +71 -0
- package/connectors/connect-replicate/src/cli/index.ts +250 -0
- package/connectors/connect-replicate/src/index.ts +19 -0
- package/connectors/connect-replicate/src/types/index.ts +85 -0
- package/connectors/connect-replicate/src/utils/config.ts +103 -0
- package/connectors/connect-replicate/src/utils/output.ts +119 -0
- package/connectors/connect-replicate/tsconfig.json +16 -0
- package/connectors/connect-roboflow/.env.example +11 -0
- package/connectors/connect-roboflow/CLAUDE.md +272 -0
- package/connectors/connect-roboflow/README.md +193 -0
- package/connectors/connect-roboflow/package.json +51 -0
- package/connectors/connect-roboflow/scripts/release.ts +179 -0
- package/connectors/connect-roboflow/src/api/client.ts +213 -0
- package/connectors/connect-roboflow/src/api/example.ts +48 -0
- package/connectors/connect-roboflow/src/api/index.ts +51 -0
- package/connectors/connect-roboflow/src/cli/index.ts +254 -0
- package/connectors/connect-roboflow/src/index.ts +103 -0
- package/connectors/connect-roboflow/src/types/index.ts +237 -0
- package/connectors/connect-roboflow/src/utils/auth.ts +274 -0
- package/connectors/connect-roboflow/src/utils/bulk.ts +212 -0
- package/connectors/connect-roboflow/src/utils/config.ts +326 -0
- package/connectors/connect-roboflow/src/utils/output.ts +175 -0
- package/connectors/connect-roboflow/src/utils/settings.ts +114 -0
- package/connectors/connect-roboflow/src/utils/storage.ts +198 -0
- package/connectors/connect-roboflow/tsconfig.json +16 -0
- package/connectors/connect-runway/.env.example +11 -0
- package/connectors/connect-runway/CLAUDE.md +128 -0
- package/connectors/connect-runway/README.md +193 -0
- package/connectors/connect-runway/package.json +52 -0
- package/connectors/connect-runway/src/api/client.ts +78 -0
- package/connectors/connect-runway/src/api/index.ts +40 -0
- package/connectors/connect-runway/src/cli/index.ts +283 -0
- package/connectors/connect-runway/src/index.ts +19 -0
- package/connectors/connect-runway/src/types/index.ts +52 -0
- package/connectors/connect-runway/src/utils/config.ts +103 -0
- package/connectors/connect-runway/src/utils/output.ts +119 -0
- package/connectors/connect-runway/tsconfig.json +16 -0
- package/connectors/connect-together/.env.example +11 -0
- package/connectors/connect-together/CLAUDE.md +128 -0
- package/connectors/connect-together/README.md +193 -0
- package/connectors/connect-together/package.json +52 -0
- package/connectors/connect-together/src/api/client.ts +106 -0
- package/connectors/connect-together/src/api/index.ts +47 -0
- package/connectors/connect-together/src/cli/index.ts +228 -0
- package/connectors/connect-together/src/index.ts +19 -0
- package/connectors/connect-together/src/types/index.ts +91 -0
- package/connectors/connect-together/src/utils/config.ts +142 -0
- package/connectors/connect-together/src/utils/output.ts +119 -0
- package/connectors/connect-together/tsconfig.json +16 -0
- package/dist/index.js +112 -0
- package/package.json +1 -1
|
@@ -0,0 +1,421 @@
|
|
|
1
|
+
#!/usr/bin/env bun
|
|
2
|
+
import { Command } from 'commander';
|
|
3
|
+
import chalk from 'chalk';
|
|
4
|
+
import { Perplexity } from '../api';
|
|
5
|
+
import { PERPLEXITY_MODELS } from '../types';
|
|
6
|
+
import type { PerplexityModel } from '../types';
|
|
7
|
+
import {
|
|
8
|
+
getApiKey,
|
|
9
|
+
setApiKey,
|
|
10
|
+
clearConfig,
|
|
11
|
+
getConfigDir,
|
|
12
|
+
setProfileOverride,
|
|
13
|
+
getCurrentProfile,
|
|
14
|
+
setCurrentProfile,
|
|
15
|
+
listProfiles,
|
|
16
|
+
createProfile,
|
|
17
|
+
deleteProfile,
|
|
18
|
+
profileExists,
|
|
19
|
+
loadProfile,
|
|
20
|
+
getDefaultModel,
|
|
21
|
+
setDefaultModel,
|
|
22
|
+
} from '../utils/config';
|
|
23
|
+
import type { OutputFormat } from '../utils/output';
|
|
24
|
+
import { success, error, info, print, warn } from '../utils/output';
|
|
25
|
+
|
|
26
|
+
const CONNECTOR_NAME = 'connect-perplexity';
|
|
27
|
+
const VERSION = '0.1.0';
|
|
28
|
+
|
|
29
|
+
const program = new Command();
|
|
30
|
+
|
|
31
|
+
program
|
|
32
|
+
.name(CONNECTOR_NAME)
|
|
33
|
+
.description('Perplexity AI API connector - Chat completions with web search grounding')
|
|
34
|
+
.version(VERSION)
|
|
35
|
+
.option('-k, --api-key <key>', 'API key (overrides config)')
|
|
36
|
+
.option('-f, --format <format>', 'Output format (json, pretty)', 'pretty')
|
|
37
|
+
.option('-p, --profile <profile>', 'Use a specific profile')
|
|
38
|
+
.hook('preAction', (thisCommand) => {
|
|
39
|
+
const opts = thisCommand.opts();
|
|
40
|
+
// Set profile override before any command runs
|
|
41
|
+
if (opts.profile) {
|
|
42
|
+
if (!profileExists(opts.profile)) {
|
|
43
|
+
error(`Profile "${opts.profile}" does not exist. Create it with "${CONNECTOR_NAME} profile create ${opts.profile}"`);
|
|
44
|
+
process.exit(1);
|
|
45
|
+
}
|
|
46
|
+
setProfileOverride(opts.profile);
|
|
47
|
+
}
|
|
48
|
+
// Set API key from flag if provided
|
|
49
|
+
if (opts.apiKey) {
|
|
50
|
+
process.env.PERPLEXITY_API_KEY = opts.apiKey;
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
// Helper to get output format
|
|
55
|
+
function getFormat(cmd: Command): OutputFormat {
|
|
56
|
+
const parent = cmd.parent;
|
|
57
|
+
return (parent?.opts().format || 'pretty') as OutputFormat;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Helper to get authenticated client
|
|
61
|
+
function getClient(): Perplexity {
|
|
62
|
+
const apiKey = getApiKey();
|
|
63
|
+
if (!apiKey) {
|
|
64
|
+
error(`No API key configured. Run "${CONNECTOR_NAME} config set-key <key>" or set PERPLEXITY_API_KEY environment variable.`);
|
|
65
|
+
process.exit(1);
|
|
66
|
+
}
|
|
67
|
+
return new Perplexity({ apiKey });
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// ============================================
|
|
71
|
+
// Profile Commands
|
|
72
|
+
// ============================================
|
|
73
|
+
const profileCmd = program
|
|
74
|
+
.command('profile')
|
|
75
|
+
.description('Manage configuration profiles');
|
|
76
|
+
|
|
77
|
+
profileCmd
|
|
78
|
+
.command('list')
|
|
79
|
+
.description('List all profiles')
|
|
80
|
+
.action(() => {
|
|
81
|
+
const profiles = listProfiles();
|
|
82
|
+
const current = getCurrentProfile();
|
|
83
|
+
|
|
84
|
+
if (profiles.length === 0) {
|
|
85
|
+
info('No profiles found. Use "profile create <name>" to create one.');
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
success(`Profiles:`);
|
|
90
|
+
profiles.forEach(p => {
|
|
91
|
+
const isActive = p === current ? chalk.green(' (active)') : '';
|
|
92
|
+
console.log(` ${p}${isActive}`);
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
profileCmd
|
|
97
|
+
.command('use <name>')
|
|
98
|
+
.description('Switch to a profile')
|
|
99
|
+
.action((name: string) => {
|
|
100
|
+
if (!profileExists(name)) {
|
|
101
|
+
error(`Profile "${name}" does not exist. Create it with "profile create ${name}"`);
|
|
102
|
+
process.exit(1);
|
|
103
|
+
}
|
|
104
|
+
setCurrentProfile(name);
|
|
105
|
+
success(`Switched to profile: ${name}`);
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
profileCmd
|
|
109
|
+
.command('create <name>')
|
|
110
|
+
.description('Create a new profile')
|
|
111
|
+
.option('--api-key <key>', 'API key')
|
|
112
|
+
.option('--use', 'Switch to this profile after creation')
|
|
113
|
+
.action((name: string, opts) => {
|
|
114
|
+
if (profileExists(name)) {
|
|
115
|
+
error(`Profile "${name}" already exists`);
|
|
116
|
+
process.exit(1);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
createProfile(name, {
|
|
120
|
+
apiKey: opts.apiKey,
|
|
121
|
+
});
|
|
122
|
+
success(`Profile "${name}" created`);
|
|
123
|
+
|
|
124
|
+
if (opts.use) {
|
|
125
|
+
setCurrentProfile(name);
|
|
126
|
+
info(`Switched to profile: ${name}`);
|
|
127
|
+
}
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
profileCmd
|
|
131
|
+
.command('delete <name>')
|
|
132
|
+
.description('Delete a profile')
|
|
133
|
+
.action((name: string) => {
|
|
134
|
+
if (name === 'default') {
|
|
135
|
+
error('Cannot delete the default profile');
|
|
136
|
+
process.exit(1);
|
|
137
|
+
}
|
|
138
|
+
if (deleteProfile(name)) {
|
|
139
|
+
success(`Profile "${name}" deleted`);
|
|
140
|
+
} else {
|
|
141
|
+
error(`Profile "${name}" not found`);
|
|
142
|
+
process.exit(1);
|
|
143
|
+
}
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
profileCmd
|
|
147
|
+
.command('show [name]')
|
|
148
|
+
.description('Show profile configuration')
|
|
149
|
+
.action((name?: string) => {
|
|
150
|
+
const profileName = name || getCurrentProfile();
|
|
151
|
+
const config = loadProfile(profileName);
|
|
152
|
+
const active = getCurrentProfile();
|
|
153
|
+
|
|
154
|
+
console.log(chalk.bold(`Profile: ${profileName}${profileName === active ? chalk.green(' (active)') : ''}`));
|
|
155
|
+
info(`API Key: ${config.apiKey ? `${config.apiKey.substring(0, 8)}...` : chalk.gray('not set')}`);
|
|
156
|
+
info(`Default Model: ${config.defaultModel || chalk.gray('sonar')}`);
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
// ============================================
|
|
160
|
+
// Config Commands
|
|
161
|
+
// ============================================
|
|
162
|
+
const configCmd = program
|
|
163
|
+
.command('config')
|
|
164
|
+
.description('Manage CLI configuration (for active profile)');
|
|
165
|
+
|
|
166
|
+
configCmd
|
|
167
|
+
.command('set-key <apiKey>')
|
|
168
|
+
.description('Set API key')
|
|
169
|
+
.action((apiKey: string) => {
|
|
170
|
+
setApiKey(apiKey);
|
|
171
|
+
success(`API key saved to profile: ${getCurrentProfile()}`);
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
configCmd
|
|
175
|
+
.command('set-model <model>')
|
|
176
|
+
.description('Set default model')
|
|
177
|
+
.action((model: string) => {
|
|
178
|
+
if (!PERPLEXITY_MODELS.includes(model as PerplexityModel)) {
|
|
179
|
+
error(`Invalid model. Available models: ${PERPLEXITY_MODELS.join(', ')}`);
|
|
180
|
+
process.exit(1);
|
|
181
|
+
}
|
|
182
|
+
setDefaultModel(model);
|
|
183
|
+
success(`Default model set to: ${model}`);
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
configCmd
|
|
187
|
+
.command('show')
|
|
188
|
+
.description('Show current configuration')
|
|
189
|
+
.action(() => {
|
|
190
|
+
const profileName = getCurrentProfile();
|
|
191
|
+
const apiKey = getApiKey();
|
|
192
|
+
const model = getDefaultModel();
|
|
193
|
+
|
|
194
|
+
console.log(chalk.bold(`Active Profile: ${profileName}`));
|
|
195
|
+
info(`Config directory: ${getConfigDir()}`);
|
|
196
|
+
info(`API Key: ${apiKey ? `${apiKey.substring(0, 8)}...` : chalk.gray('not set')}`);
|
|
197
|
+
info(`Default Model: ${model || chalk.gray('sonar')}`);
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
configCmd
|
|
201
|
+
.command('clear')
|
|
202
|
+
.description('Clear configuration for active profile')
|
|
203
|
+
.action(() => {
|
|
204
|
+
clearConfig();
|
|
205
|
+
success(`Configuration cleared for profile: ${getCurrentProfile()}`);
|
|
206
|
+
});
|
|
207
|
+
|
|
208
|
+
// ============================================
|
|
209
|
+
// Chat Commands
|
|
210
|
+
// ============================================
|
|
211
|
+
const chatCmd = program
|
|
212
|
+
.command('chat')
|
|
213
|
+
.description('Chat completion commands');
|
|
214
|
+
|
|
215
|
+
chatCmd
|
|
216
|
+
.command('ask <question>')
|
|
217
|
+
.description('Ask a question')
|
|
218
|
+
.option('-m, --model <model>', `Model to use (${PERPLEXITY_MODELS.join(', ')})`, getDefaultModel() || 'sonar')
|
|
219
|
+
.option('-t, --temperature <temp>', 'Temperature (0-2)', '0.7')
|
|
220
|
+
.option('--max-tokens <tokens>', 'Maximum tokens')
|
|
221
|
+
.option('-s, --system <prompt>', 'System prompt')
|
|
222
|
+
.option('--recency <filter>', 'Search recency filter (hour, day, week, month)')
|
|
223
|
+
.action(async (question: string, opts) => {
|
|
224
|
+
try {
|
|
225
|
+
const client = getClient();
|
|
226
|
+
const response = await client.chat.ask(question, {
|
|
227
|
+
model: opts.model as PerplexityModel,
|
|
228
|
+
temperature: parseFloat(opts.temperature),
|
|
229
|
+
maxTokens: opts.maxTokens ? parseInt(opts.maxTokens) : undefined,
|
|
230
|
+
systemPrompt: opts.system,
|
|
231
|
+
searchRecencyFilter: opts.recency,
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
const format = getFormat(chatCmd);
|
|
235
|
+
if (format === 'json') {
|
|
236
|
+
print(response, format);
|
|
237
|
+
} else {
|
|
238
|
+
// Pretty print the response
|
|
239
|
+
const content = response.choices[0]?.message?.content || '';
|
|
240
|
+
console.log(chalk.cyan('\nAnswer:\n'));
|
|
241
|
+
console.log(content);
|
|
242
|
+
|
|
243
|
+
if (response.citations && response.citations.length > 0) {
|
|
244
|
+
console.log(chalk.cyan('\nCitations:'));
|
|
245
|
+
response.citations.forEach((citation, i) => {
|
|
246
|
+
console.log(chalk.gray(` [${i + 1}] ${citation}`));
|
|
247
|
+
});
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
console.log(chalk.gray(`\n(${response.usage.total_tokens} tokens, model: ${response.model})`));
|
|
251
|
+
}
|
|
252
|
+
} catch (err) {
|
|
253
|
+
error(String(err));
|
|
254
|
+
process.exit(1);
|
|
255
|
+
}
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
chatCmd
|
|
259
|
+
.command('search <query>')
|
|
260
|
+
.description('Search the web and get an answer')
|
|
261
|
+
.option('-m, --model <model>', `Model to use`, 'sonar-pro')
|
|
262
|
+
.option('--recency <filter>', 'Search recency filter (hour, day, week, month)')
|
|
263
|
+
.action(async (query: string, opts) => {
|
|
264
|
+
try {
|
|
265
|
+
const client = getClient();
|
|
266
|
+
const response = await client.chat.search(query, {
|
|
267
|
+
model: opts.model as PerplexityModel,
|
|
268
|
+
recency: opts.recency,
|
|
269
|
+
});
|
|
270
|
+
|
|
271
|
+
const format = getFormat(chatCmd);
|
|
272
|
+
if (format === 'json') {
|
|
273
|
+
print(response, format);
|
|
274
|
+
} else {
|
|
275
|
+
const content = response.choices[0]?.message?.content || '';
|
|
276
|
+
console.log(chalk.cyan('\nSearch Results:\n'));
|
|
277
|
+
console.log(content);
|
|
278
|
+
|
|
279
|
+
if (response.citations && response.citations.length > 0) {
|
|
280
|
+
console.log(chalk.cyan('\nSources:'));
|
|
281
|
+
response.citations.forEach((citation, i) => {
|
|
282
|
+
console.log(chalk.gray(` [${i + 1}] ${citation}`));
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
console.log(chalk.gray(`\n(${response.usage.total_tokens} tokens)`));
|
|
287
|
+
}
|
|
288
|
+
} catch (err) {
|
|
289
|
+
error(String(err));
|
|
290
|
+
process.exit(1);
|
|
291
|
+
}
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
chatCmd
|
|
295
|
+
.command('research <topic>')
|
|
296
|
+
.description('Deep research on a topic (uses sonar-deep-research)')
|
|
297
|
+
.option('--max-tokens <tokens>', 'Maximum tokens')
|
|
298
|
+
.action(async (topic: string, opts) => {
|
|
299
|
+
try {
|
|
300
|
+
info('Starting deep research (this may take a moment)...');
|
|
301
|
+
const client = getClient();
|
|
302
|
+
const response = await client.chat.research(topic, {
|
|
303
|
+
maxTokens: opts.maxTokens ? parseInt(opts.maxTokens) : undefined,
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
const format = getFormat(chatCmd);
|
|
307
|
+
if (format === 'json') {
|
|
308
|
+
print(response, format);
|
|
309
|
+
} else {
|
|
310
|
+
const content = response.choices[0]?.message?.content || '';
|
|
311
|
+
console.log(chalk.cyan('\nResearch Report:\n'));
|
|
312
|
+
console.log(content);
|
|
313
|
+
|
|
314
|
+
if (response.citations && response.citations.length > 0) {
|
|
315
|
+
console.log(chalk.cyan('\nReferences:'));
|
|
316
|
+
response.citations.forEach((citation, i) => {
|
|
317
|
+
console.log(chalk.gray(` [${i + 1}] ${citation}`));
|
|
318
|
+
});
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
console.log(chalk.gray(`\n(${response.usage.total_tokens} tokens, model: ${response.model})`));
|
|
322
|
+
}
|
|
323
|
+
} catch (err) {
|
|
324
|
+
error(String(err));
|
|
325
|
+
process.exit(1);
|
|
326
|
+
}
|
|
327
|
+
});
|
|
328
|
+
|
|
329
|
+
chatCmd
|
|
330
|
+
.command('reason <prompt>')
|
|
331
|
+
.description('Reasoning task (uses sonar-reasoning-pro)')
|
|
332
|
+
.option('--max-tokens <tokens>', 'Maximum tokens')
|
|
333
|
+
.action(async (prompt: string, opts) => {
|
|
334
|
+
try {
|
|
335
|
+
const client = getClient();
|
|
336
|
+
const response = await client.chat.reason(prompt, {
|
|
337
|
+
maxTokens: opts.maxTokens ? parseInt(opts.maxTokens) : undefined,
|
|
338
|
+
});
|
|
339
|
+
|
|
340
|
+
const format = getFormat(chatCmd);
|
|
341
|
+
if (format === 'json') {
|
|
342
|
+
print(response, format);
|
|
343
|
+
} else {
|
|
344
|
+
const content = response.choices[0]?.message?.content || '';
|
|
345
|
+
console.log(chalk.cyan('\nReasoning:\n'));
|
|
346
|
+
console.log(content);
|
|
347
|
+
|
|
348
|
+
console.log(chalk.gray(`\n(${response.usage.total_tokens} tokens, model: ${response.model})`));
|
|
349
|
+
}
|
|
350
|
+
} catch (err) {
|
|
351
|
+
error(String(err));
|
|
352
|
+
process.exit(1);
|
|
353
|
+
}
|
|
354
|
+
});
|
|
355
|
+
|
|
356
|
+
// ============================================
|
|
357
|
+
// Quick Commands (shortcuts)
|
|
358
|
+
// ============================================
|
|
359
|
+
program
|
|
360
|
+
.command('ask <question>')
|
|
361
|
+
.description('Quick ask (shortcut for "chat ask")')
|
|
362
|
+
.option('-m, --model <model>', `Model to use`, 'sonar')
|
|
363
|
+
.action(async (question: string, opts) => {
|
|
364
|
+
try {
|
|
365
|
+
const client = getClient();
|
|
366
|
+
const response = await client.chat.ask(question, {
|
|
367
|
+
model: opts.model as PerplexityModel,
|
|
368
|
+
});
|
|
369
|
+
|
|
370
|
+
const content = response.choices[0]?.message?.content || '';
|
|
371
|
+
console.log(content);
|
|
372
|
+
|
|
373
|
+
if (response.citations && response.citations.length > 0) {
|
|
374
|
+
console.log(chalk.cyan('\nSources:'));
|
|
375
|
+
response.citations.forEach((citation, i) => {
|
|
376
|
+
console.log(chalk.gray(` [${i + 1}] ${citation}`));
|
|
377
|
+
});
|
|
378
|
+
}
|
|
379
|
+
} catch (err) {
|
|
380
|
+
error(String(err));
|
|
381
|
+
process.exit(1);
|
|
382
|
+
}
|
|
383
|
+
});
|
|
384
|
+
|
|
385
|
+
program
|
|
386
|
+
.command('search <query>')
|
|
387
|
+
.description('Quick search (shortcut for "chat search")')
|
|
388
|
+
.action(async (query: string) => {
|
|
389
|
+
try {
|
|
390
|
+
const client = getClient();
|
|
391
|
+
const response = await client.chat.search(query);
|
|
392
|
+
|
|
393
|
+
const content = response.choices[0]?.message?.content || '';
|
|
394
|
+
console.log(content);
|
|
395
|
+
|
|
396
|
+
if (response.citations && response.citations.length > 0) {
|
|
397
|
+
console.log(chalk.cyan('\nSources:'));
|
|
398
|
+
response.citations.forEach((citation, i) => {
|
|
399
|
+
console.log(chalk.gray(` [${i + 1}] ${citation}`));
|
|
400
|
+
});
|
|
401
|
+
}
|
|
402
|
+
} catch (err) {
|
|
403
|
+
error(String(err));
|
|
404
|
+
process.exit(1);
|
|
405
|
+
}
|
|
406
|
+
});
|
|
407
|
+
|
|
408
|
+
program
|
|
409
|
+
.command('models')
|
|
410
|
+
.description('List available models')
|
|
411
|
+
.action(() => {
|
|
412
|
+
console.log(chalk.bold('Available Perplexity Models:\n'));
|
|
413
|
+
console.log(` ${chalk.cyan('sonar')} - Standard model for general queries`);
|
|
414
|
+
console.log(` ${chalk.cyan('sonar-pro')} - Enhanced model with better accuracy`);
|
|
415
|
+
console.log(` ${chalk.cyan('sonar-reasoning')} - Model optimized for reasoning tasks`);
|
|
416
|
+
console.log(` ${chalk.cyan('sonar-reasoning-pro')} - Enhanced reasoning model`);
|
|
417
|
+
console.log(` ${chalk.cyan('sonar-deep-research')} - Model for comprehensive research`);
|
|
418
|
+
});
|
|
419
|
+
|
|
420
|
+
// Parse and execute
|
|
421
|
+
program.parse();
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
// Perplexity AI API Connector
|
|
2
|
+
// A TypeScript wrapper for Perplexity's chat completions with web search grounding
|
|
3
|
+
|
|
4
|
+
export { Perplexity } from './api';
|
|
5
|
+
export * from './types';
|
|
6
|
+
|
|
7
|
+
// Re-export individual API classes for advanced usage
|
|
8
|
+
export { PerplexityClient, ChatApi } from './api';
|
|
9
|
+
|
|
10
|
+
// Export config utilities
|
|
11
|
+
export {
|
|
12
|
+
getApiKey,
|
|
13
|
+
setApiKey,
|
|
14
|
+
getDefaultModel,
|
|
15
|
+
setDefaultModel,
|
|
16
|
+
getCurrentProfile,
|
|
17
|
+
setCurrentProfile,
|
|
18
|
+
listProfiles,
|
|
19
|
+
createProfile,
|
|
20
|
+
deleteProfile,
|
|
21
|
+
loadProfile,
|
|
22
|
+
saveProfile,
|
|
23
|
+
clearConfig,
|
|
24
|
+
} from './utils/config';
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
// Perplexity AI API Types
|
|
2
|
+
|
|
3
|
+
// ============================================
|
|
4
|
+
// Configuration
|
|
5
|
+
// ============================================
|
|
6
|
+
|
|
7
|
+
export interface PerplexityConfig {
|
|
8
|
+
apiKey: string;
|
|
9
|
+
baseUrl?: string;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
// ============================================
|
|
13
|
+
// Models
|
|
14
|
+
// ============================================
|
|
15
|
+
|
|
16
|
+
export type PerplexityModel =
|
|
17
|
+
| 'sonar'
|
|
18
|
+
| 'sonar-pro'
|
|
19
|
+
| 'sonar-reasoning'
|
|
20
|
+
| 'sonar-reasoning-pro'
|
|
21
|
+
| 'sonar-deep-research';
|
|
22
|
+
|
|
23
|
+
export const PERPLEXITY_MODELS: PerplexityModel[] = [
|
|
24
|
+
'sonar',
|
|
25
|
+
'sonar-pro',
|
|
26
|
+
'sonar-reasoning',
|
|
27
|
+
'sonar-reasoning-pro',
|
|
28
|
+
'sonar-deep-research',
|
|
29
|
+
];
|
|
30
|
+
|
|
31
|
+
// ============================================
|
|
32
|
+
// Chat Completions
|
|
33
|
+
// ============================================
|
|
34
|
+
|
|
35
|
+
export type MessageRole = 'system' | 'user' | 'assistant';
|
|
36
|
+
|
|
37
|
+
export interface ChatMessage {
|
|
38
|
+
role: MessageRole;
|
|
39
|
+
content: string;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export interface ChatCompletionRequest {
|
|
43
|
+
model: PerplexityModel;
|
|
44
|
+
messages: ChatMessage[];
|
|
45
|
+
max_tokens?: number;
|
|
46
|
+
temperature?: number;
|
|
47
|
+
top_p?: number;
|
|
48
|
+
top_k?: number;
|
|
49
|
+
stream?: boolean;
|
|
50
|
+
presence_penalty?: number;
|
|
51
|
+
frequency_penalty?: number;
|
|
52
|
+
search_domain_filter?: string[];
|
|
53
|
+
return_images?: boolean;
|
|
54
|
+
return_related_questions?: boolean;
|
|
55
|
+
search_recency_filter?: 'month' | 'week' | 'day' | 'hour';
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
export interface SearchResult {
|
|
59
|
+
title: string;
|
|
60
|
+
url: string;
|
|
61
|
+
date?: string;
|
|
62
|
+
last_updated?: string;
|
|
63
|
+
snippet: string;
|
|
64
|
+
source: 'web' | string;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export interface ChatCompletionChoice {
|
|
68
|
+
index: number;
|
|
69
|
+
message: ChatMessage;
|
|
70
|
+
finish_reason: 'stop' | 'length' | 'content_filter' | null;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export interface ChatCompletionUsage {
|
|
74
|
+
prompt_tokens: number;
|
|
75
|
+
completion_tokens: number;
|
|
76
|
+
total_tokens: number;
|
|
77
|
+
citation_tokens?: number;
|
|
78
|
+
num_search_queries?: number;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
export interface ChatCompletionResponse {
|
|
82
|
+
id: string;
|
|
83
|
+
model: string;
|
|
84
|
+
created: number;
|
|
85
|
+
object: 'chat.completion';
|
|
86
|
+
choices: ChatCompletionChoice[];
|
|
87
|
+
usage: ChatCompletionUsage;
|
|
88
|
+
citations?: string[];
|
|
89
|
+
search_results?: SearchResult[];
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Streaming types
|
|
93
|
+
export interface ChatCompletionChunkDelta {
|
|
94
|
+
role?: MessageRole;
|
|
95
|
+
content?: string;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
export interface ChatCompletionChunkChoice {
|
|
99
|
+
index: number;
|
|
100
|
+
delta: ChatCompletionChunkDelta;
|
|
101
|
+
finish_reason: 'stop' | 'length' | 'content_filter' | null;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
export interface ChatCompletionChunk {
|
|
105
|
+
id: string;
|
|
106
|
+
model: string;
|
|
107
|
+
created: number;
|
|
108
|
+
object: 'chat.completion.chunk';
|
|
109
|
+
choices: ChatCompletionChunkChoice[];
|
|
110
|
+
usage?: ChatCompletionUsage;
|
|
111
|
+
citations?: string[];
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
// ============================================
|
|
115
|
+
// Common Types
|
|
116
|
+
// ============================================
|
|
117
|
+
|
|
118
|
+
export type OutputFormat = 'json' | 'pretty';
|
|
119
|
+
|
|
120
|
+
// ============================================
|
|
121
|
+
// API Error Types
|
|
122
|
+
// ============================================
|
|
123
|
+
|
|
124
|
+
export interface ApiErrorDetail {
|
|
125
|
+
code: string;
|
|
126
|
+
message: string;
|
|
127
|
+
field?: string;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
export class PerplexityApiError extends Error {
|
|
131
|
+
public readonly statusCode: number;
|
|
132
|
+
public readonly errors?: ApiErrorDetail[];
|
|
133
|
+
|
|
134
|
+
constructor(message: string, statusCode: number, errors?: ApiErrorDetail[]) {
|
|
135
|
+
super(message);
|
|
136
|
+
this.name = 'PerplexityApiError';
|
|
137
|
+
this.statusCode = statusCode;
|
|
138
|
+
this.errors = errors;
|
|
139
|
+
}
|
|
140
|
+
}
|