nolo-cli 0.1.8 → 0.1.9
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/README.md +32 -0
- package/agentRuntimeCommands.ts +3 -3
- package/ai/agent/_executeModel.ts +118 -0
- package/ai/agent/agentSlice.ts +525 -0
- package/ai/agent/appWorkingMemory.ts +126 -0
- package/ai/agent/avatarUtils.ts +24 -0
- package/ai/agent/buildEditingContext.ts +373 -0
- package/ai/agent/buildSystemPrompt.ts +532 -0
- package/ai/agent/cleanAgentMessages.ts +140 -0
- package/ai/agent/cliChatClient.ts +119 -0
- package/ai/agent/cliExecutor.ts +733 -0
- package/ai/agent/cliPrompt.ts +10 -0
- package/ai/agent/contextCompiler.ts +107 -0
- package/ai/agent/contextLayerContract.ts +44 -0
- package/ai/agent/createAgentSchema.ts +234 -0
- package/ai/agent/executeToolCall.ts +58 -0
- package/ai/agent/fetchAgentContexts.ts +42 -0
- package/ai/agent/generatePrompt.ts +3 -0
- package/ai/agent/getFullChatContextKeys.ts +168 -0
- package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
- package/ai/agent/hooks/useAgentConfig.ts +61 -0
- package/ai/agent/hooks/useAgentDialog.ts +35 -0
- package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
- package/ai/agent/hooks/usePublicAgents.ts +473 -0
- package/ai/agent/machineRunPermissions.ts +95 -0
- package/ai/agent/persistMessageWithFixedId.ts +37 -0
- package/ai/agent/planSlice.ts +259 -0
- package/ai/agent/referenceUtils.ts +229 -0
- package/ai/agent/runAgentBackground.ts +238 -0
- package/ai/agent/runAgentClientLoop.ts +138 -0
- package/ai/agent/runtimeGuidance.ts +97 -0
- package/ai/agent/runtimeServerBase.ts +37 -0
- package/ai/agent/server/fetchPublicAgents.ts +128 -0
- package/ai/agent/startParallelAgentStreams.ts +424 -0
- package/ai/agent/startupProtocol.ts +53 -0
- package/ai/agent/streamAgentChatTurn.ts +1278 -0
- package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
- package/ai/agent/types.ts +71 -0
- package/ai/agent/utils/imageOutput.ts +33 -0
- package/ai/agent/utils/sortUtils.ts +250 -0
- package/ai/agent/web/referencePickerUtils.ts +146 -0
- package/ai/ai.locale.ts +1075 -0
- package/ai/chat/accumulateToolCallChunks.ts +95 -0
- package/ai/chat/fetchUtils.native.ts +276 -0
- package/ai/chat/fetchUtils.ts +153 -0
- package/ai/chat/parseApiError.ts +64 -0
- package/ai/chat/parseMultilineSSE.ts +95 -0
- package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
- package/ai/chat/sendOpenAICompletionsRequest.ts +703 -0
- package/ai/chat/sendOpenAIResponseRequest.ts +491 -0
- package/ai/chat/shouldUseServerProxy.ts +18 -0
- package/ai/chat/sseClient.native.ts +91 -0
- package/ai/chat/sseClient.ts +67 -0
- package/ai/chat/streamReader.native.ts +31 -0
- package/ai/chat/streamReader.ts +62 -0
- package/ai/chat/updateTotalUsage.ts +72 -0
- package/ai/context/buildReferenceContext.ts +437 -0
- package/ai/context/calculateContextUsage.ts +133 -0
- package/ai/context/retention.ts +165 -0
- package/ai/context/tokenUtils.ts +78 -0
- package/ai/index.ts +1 -0
- package/ai/llm/calculateGeminiImageTokens.ts +57 -0
- package/ai/llm/deepinfra.ts +28 -0
- package/ai/llm/fireworks.ts +50 -0
- package/ai/llm/generateRequestBody.ts +165 -0
- package/ai/llm/getModelContextWindow.ts +84 -0
- package/ai/llm/getNoloKey.ts +31 -0
- package/ai/llm/getPricing.ts +199 -0
- package/ai/llm/hooks/useModelPricing.ts +75 -0
- package/ai/llm/imagePricing.ts +40 -0
- package/ai/llm/isResponseAPIModel.ts +13 -0
- package/ai/llm/mimo.ts +71 -0
- package/ai/llm/mistral.ts +22 -0
- package/ai/llm/modelAvatar.ts +427 -0
- package/ai/llm/models.ts +45 -0
- package/ai/llm/openrouterModels.ts +269 -0
- package/ai/llm/providers.ts +306 -0
- package/ai/llm/reasoningModels.ts +28 -0
- package/ai/llm/types.ts +59 -0
- package/ai/llm/usageRequestOptions.ts +59 -0
- package/ai/memory/capture.ts +148 -0
- package/ai/memory/consolidate.ts +104 -0
- package/ai/memory/delete.ts +147 -0
- package/ai/memory/overlay.ts +84 -0
- package/ai/memory/query.ts +38 -0
- package/ai/memory/queryShared.ts +160 -0
- package/ai/memory/rank.ts +105 -0
- package/ai/memory/recentRelationshipRecap.ts +249 -0
- package/ai/memory/remember.ts +167 -0
- package/ai/memory/runtime.ts +76 -0
- package/ai/memory/store.ts +20 -0
- package/ai/memory/storeShared.ts +76 -0
- package/ai/memory/types.ts +46 -0
- package/ai/memory/understanding.ts +349 -0
- package/ai/memory/understandingGreeting.ts +264 -0
- package/ai/messages/type.ts +20 -0
- package/ai/policy/personalizationDialog.ts +333 -0
- package/ai/policy/runtimePolicy.ts +440 -0
- package/ai/policy/selfUpdateFields.ts +48 -0
- package/ai/policy/types.ts +64 -0
- package/ai/skills/referenceRuntime.ts +274 -0
- package/ai/skills/skillDiagnostics.ts +251 -0
- package/ai/skills/skillDocBuilder.ts +139 -0
- package/ai/skills/skillDocProtocol.ts +434 -0
- package/ai/skills/skillReferenceSummary.ts +63 -0
- package/ai/skills/skillSummaryMarker.ts +26 -0
- package/ai/token/calculatePrice.ts +544 -0
- package/ai/token/db.ts +98 -0
- package/ai/token/externalToolCost.ts +330 -0
- package/ai/token/hooks/useRecords.ts +65 -0
- package/ai/token/missingUsageEstimate.ts +42 -0
- package/ai/token/modelUsageQuery.ts +252 -0
- package/ai/token/normalizeUsage.ts +84 -0
- package/ai/token/openaiImageGenerationUsage.ts +56 -0
- package/ai/token/prepareTokenUsageData.ts +88 -0
- package/ai/token/query.ts +88 -0
- package/ai/token/queryUserTokens.ts +59 -0
- package/ai/token/resolveBillingTarget.ts +52 -0
- package/ai/token/saveTokenRecord.ts +53 -0
- package/ai/token/serverDialogProjection.ts +78 -0
- package/ai/token/serverTokenWriter.ts +143 -0
- package/ai/token/stats.ts +21 -0
- package/ai/token/tokenThunks.ts +24 -0
- package/ai/token/types.ts +93 -0
- package/ai/tools/agent/agentTools.ts +176 -0
- package/ai/tools/agent/agentUpdateShared.ts +311 -0
- package/ai/tools/agent/callAgentTool.ts +139 -0
- package/ai/tools/agent/createAgentTool.ts +512 -0
- package/ai/tools/agent/createDialogTool.ts +69 -0
- package/ai/tools/agent/createSkillAgentTool.ts +62 -0
- package/ai/tools/agent/parallelBudget.ts +221 -0
- package/ai/tools/agent/presets/appBuilderPreset.ts +145 -0
- package/ai/tools/agent/runLlmTool.ts +96 -0
- package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
- package/ai/tools/agent/skillAgentArgs.ts +106 -0
- package/ai/tools/agent/skillAgentPreset.ts +89 -0
- package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
- package/ai/tools/agent/updateAgentTool.ts +96 -0
- package/ai/tools/agent/updateSelfTool.ts +113 -0
- package/ai/tools/amazonProductScraperTool.ts +86 -0
- package/ai/tools/apifyActorClient.ts +45 -0
- package/ai/tools/appEditGuard.ts +372 -0
- package/ai/tools/appReadSnapshot.ts +153 -0
- package/ai/tools/appTools.ts +1549 -0
- package/ai/tools/applyEditTool.ts +256 -0
- package/ai/tools/applyLineEditsTool.ts +312 -0
- package/ai/tools/browserTools/click.ts +33 -0
- package/ai/tools/browserTools/closeSession.ts +29 -0
- package/ai/tools/browserTools/common.ts +27 -0
- package/ai/tools/browserTools/openSession.ts +48 -0
- package/ai/tools/browserTools/readContent.ts +38 -0
- package/ai/tools/browserTools/selectOption.ts +46 -0
- package/ai/tools/browserTools/typeText.ts +42 -0
- package/ai/tools/category/createCategoryTool.ts +66 -0
- package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
- package/ai/tools/category/updateContentCategoryTool.ts +75 -0
- package/ai/tools/cfBrowserTools.ts +319 -0
- package/ai/tools/cfSpeechToTextTool.ts +49 -0
- package/ai/tools/checkEnvTool.ts +65 -0
- package/ai/tools/cloudflareCrawlTool.ts +289 -0
- package/ai/tools/codeSearchTool.ts +111 -0
- package/ai/tools/codeTools.ts +101 -0
- package/ai/tools/createDocTool.ts +132 -0
- package/ai/tools/createPlanTool.ts +999 -0
- package/ai/tools/createSkillDocTool.ts +155 -0
- package/ai/tools/createWorkflowTool.ts +154 -0
- package/ai/tools/deepseekOcrTool.ts +34 -0
- package/ai/tools/delayTool.ts +31 -0
- package/ai/tools/deleteSpacesTool.ts +325 -0
- package/ai/tools/deleteSpacesToolModel.ts +159 -0
- package/ai/tools/devReloadUtils.ts +29 -0
- package/ai/tools/dialogMessageSearch.ts +137 -0
- package/ai/tools/doctorSkillTool.ts +72 -0
- package/ai/tools/ecommerceScraperTool.ts +86 -0
- package/ai/tools/emailTools.ts +549 -0
- package/ai/tools/evalSkillTool.ts +92 -0
- package/ai/tools/exaSearchTool.ts +64 -0
- package/ai/tools/execBashTool.ts +379 -0
- package/ai/tools/executeSqlTool.ts +192 -0
- package/ai/tools/fetchWebpageSupport.ts +309 -0
- package/ai/tools/fetchWebpageTool.ts +84 -0
- package/ai/tools/geminiImagePreviewTool.ts +361 -0
- package/ai/tools/generateDocxTool.ts +215 -0
- package/ai/tools/googleSearchScraperTool.ts +106 -0
- package/ai/tools/importDataTool.ts +133 -0
- package/ai/tools/importSkillTool.ts +162 -0
- package/ai/tools/index.ts +1858 -0
- package/ai/tools/listFilesTool.ts +82 -0
- package/ai/tools/listUserSpacesTool.ts +113 -0
- package/ai/tools/modelUsageTools.ts +142 -0
- package/ai/tools/olmOcrTool.ts +34 -0
- package/ai/tools/openaiImageTool.ts +218 -0
- package/ai/tools/paddleOcrTool.ts +34 -0
- package/ai/tools/prepareTools.ts +23 -0
- package/ai/tools/readDocTool.ts +84 -0
- package/ai/tools/readFileTool.ts +211 -0
- package/ai/tools/readTool.ts +163 -0
- package/ai/tools/readXPostTool.ts +233 -0
- package/ai/tools/rememberMemoryTool.ts +84 -0
- package/ai/tools/remotionVideoTool.ts +151 -0
- package/ai/tools/searchDialogMessagesTool.ts +222 -0
- package/ai/tools/searchRepoTool.ts +115 -0
- package/ai/tools/searchWorkspaceTool.ts +259 -0
- package/ai/tools/skillFollowup.ts +86 -0
- package/ai/tools/surfWeatherTool.ts +169 -0
- package/ai/tools/table/addTableRowTool.ts +217 -0
- package/ai/tools/table/createTableTool.ts +315 -0
- package/ai/tools/table/rowTools.ts +366 -0
- package/ai/tools/table/schemaTools.ts +244 -0
- package/ai/tools/table/shareTableTool.ts +148 -0
- package/ai/tools/table/toolShared.ts +129 -0
- package/ai/tools/toolApiClient.ts +198 -0
- package/ai/tools/toolNameAliases.ts +57 -0
- package/ai/tools/toolResultError.ts +42 -0
- package/ai/tools/toolRunSlice.ts +303 -0
- package/ai/tools/toolSchemaCompatibility.ts +53 -0
- package/ai/tools/toolVisibility.ts +4 -0
- package/ai/tools/types.ts +20 -0
- package/ai/tools/uiAskChoiceTool.ts +104 -0
- package/ai/tools/updateContentTitleTool.ts +84 -0
- package/ai/tools/updateDocTool.ts +105 -0
- package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
- package/ai/tools/whisperTool.ts +77 -0
- package/ai/tools/writeFileTool.ts +210 -0
- package/ai/tools/youtubeScraperTool.ts +116 -0
- package/ai/tools/ziweiChartTool.ts +678 -0
- package/ai/types.ts +55 -0
- package/ai/workflow/workflowExecutor.ts +323 -0
- package/ai/workflow/workflowSlice.ts +73 -0
- package/ai/workflow/workflowTypes.ts +106 -0
- package/client/compactDialog.ts +222 -0
- package/connector-experimental/capabilities.ts +73 -0
- package/connector-experimental/codexBinary.ts +41 -0
- package/connector-experimental/heartbeatLoop.ts +22 -0
- package/connector-experimental/index.ts +5 -0
- package/connector-experimental/machineInfo.ts +46 -0
- package/connector-experimental/protocol.ts +54 -0
- package/machineCommands.ts +4 -4
- package/package.json +8 -6
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import type { Agent } from "app/types";
|
|
2
|
+
import { getModelConfig } from "ai/llm/providers";
|
|
3
|
+
import {
|
|
4
|
+
isOpenAIFlexPricedModel,
|
|
5
|
+
resolveOpenAIServiceTier,
|
|
6
|
+
} from "integrations/openai/flexTier";
|
|
7
|
+
import { getApproxPricePerImage } from "./imagePricing";
|
|
8
|
+
|
|
9
|
+
interface ModelPricing {
|
|
10
|
+
inputPrice: number;
|
|
11
|
+
inputCacheHitPrice: number;
|
|
12
|
+
outputPrice: number;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
interface Prices {
|
|
16
|
+
cybotInput: number;
|
|
17
|
+
cybotOutput: number;
|
|
18
|
+
serverInput: number;
|
|
19
|
+
serverOutput: number;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const MAX_OUTPUT_TOKENS = 8192; // 单次返回最大 token 数
|
|
23
|
+
|
|
24
|
+
export const getModelPricing = (
|
|
25
|
+
provider: string,
|
|
26
|
+
modelName: string,
|
|
27
|
+
requestedServiceTier?: string
|
|
28
|
+
): ModelPricing | null => {
|
|
29
|
+
let model;
|
|
30
|
+
try {
|
|
31
|
+
model = getModelConfig(provider as any, modelName);
|
|
32
|
+
} catch {
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (!model?.price) return null;
|
|
37
|
+
|
|
38
|
+
return getModelPricingForModel(provider, modelName, model, requestedServiceTier);
|
|
39
|
+
};
|
|
40
|
+
|
|
41
|
+
export const getModelPricingForModel = (
|
|
42
|
+
provider: string,
|
|
43
|
+
modelName: string,
|
|
44
|
+
model: { price?: { input: number; inputCacheHit?: number; output: number } | null },
|
|
45
|
+
requestedServiceTier?: string
|
|
46
|
+
): ModelPricing | null => {
|
|
47
|
+
if (!model.price) return null;
|
|
48
|
+
const normalizedRequestedServiceTier =
|
|
49
|
+
requestedServiceTier === "flex" || requestedServiceTier === "default"
|
|
50
|
+
? requestedServiceTier
|
|
51
|
+
: undefined;
|
|
52
|
+
const shouldApplyFlexDiscount =
|
|
53
|
+
provider === "openai" &&
|
|
54
|
+
(normalizedRequestedServiceTier === "flex" ||
|
|
55
|
+
resolveOpenAIServiceTier({
|
|
56
|
+
providerName: provider,
|
|
57
|
+
model: modelName,
|
|
58
|
+
}) === "flex") &&
|
|
59
|
+
!isOpenAIFlexPricedModel(modelName);
|
|
60
|
+
const priceMultiplier = shouldApplyFlexDiscount ? 0.5 : 1;
|
|
61
|
+
|
|
62
|
+
return {
|
|
63
|
+
inputPrice: model.price.input * priceMultiplier,
|
|
64
|
+
inputCacheHitPrice:
|
|
65
|
+
typeof model.price.inputCacheHit === "number"
|
|
66
|
+
? model.price.inputCacheHit * priceMultiplier
|
|
67
|
+
: 0,
|
|
68
|
+
outputPrice: model.price.output * priceMultiplier,
|
|
69
|
+
};
|
|
70
|
+
};
|
|
71
|
+
|
|
72
|
+
export const getPrices = (config: any, serverPrices: any): Prices => ({
|
|
73
|
+
cybotInput: Number(config?.inputPrice ?? 0),
|
|
74
|
+
cybotOutput: Number(config?.outputPrice ?? 0),
|
|
75
|
+
serverInput: Number(serverPrices?.inputPrice ?? 0),
|
|
76
|
+
serverOutput: Number(serverPrices?.outputPrice ?? 0),
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* 计算最终价格:
|
|
81
|
+
* - 从所有价格中取「每百万 tokens 价格」的最大值
|
|
82
|
+
* - 再换算成单 token 价格
|
|
83
|
+
* - 再乘以最大输出 token 数
|
|
84
|
+
*/
|
|
85
|
+
export const getFinalPrice = (prices: Prices): number => {
|
|
86
|
+
// 1. 取出所有价格字段
|
|
87
|
+
const rawValues = Object.values(prices);
|
|
88
|
+
|
|
89
|
+
// 2. 过滤出合法数字(去掉 NaN、Infinity、null/undefined 等)
|
|
90
|
+
const validValues = rawValues.filter(
|
|
91
|
+
(value) => typeof value === "number" && Number.isFinite(value)
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
// 3. 没有合法值时返回 0,避免 Math.max(...[]) 抛错
|
|
95
|
+
if (validValues.length === 0) {
|
|
96
|
+
return 0;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
// 4. 找到每百万 token 的最高单价
|
|
100
|
+
const maxPricePerMillion = Math.max(...validValues);
|
|
101
|
+
|
|
102
|
+
// 5. 换算成单 token 价格
|
|
103
|
+
const maxPricePerToken = maxPricePerMillion / 1_000_000;
|
|
104
|
+
|
|
105
|
+
// 6. 计算 8192 个 token 的费用
|
|
106
|
+
return maxPricePerToken * MAX_OUTPUT_TOKENS;
|
|
107
|
+
};
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* 每次对话估算的典型 token 用量(输入含历史上下文,输出为单次回复)
|
|
111
|
+
* 仅用于 UI 展示的估算,并非计费依据。
|
|
112
|
+
*/
|
|
113
|
+
const TYPICAL_INPUT_TOKENS = 500;
|
|
114
|
+
const TYPICAL_OUTPUT_TOKENS = 300;
|
|
115
|
+
|
|
116
|
+
export type AgentPriceHint =
|
|
117
|
+
| { type: "per_image"; amount: number }
|
|
118
|
+
| { type: "per_turn"; amount: number };
|
|
119
|
+
|
|
120
|
+
export interface CompactTurnPriceDisplay {
|
|
121
|
+
amountText: string;
|
|
122
|
+
unitCount: number;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
const trimTrailingZeros = (value: string): string => {
|
|
126
|
+
if (!value.includes(".")) return value;
|
|
127
|
+
return value.replace(/\.?0+$/, "");
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* 根据 agent 配置返回用户可读的价格提示:
|
|
132
|
+
* - 图像 agent(imageConfig.enabled):从 model config 查 pricePerImage,以每张为单位
|
|
133
|
+
* - 文字 agent:用典型 token 用量估算每次对话费用
|
|
134
|
+
* - 无价格信息则返回 null
|
|
135
|
+
*/
|
|
136
|
+
export const getAgentPriceHint = (
|
|
137
|
+
agent: Pick<Agent, "inputPrice" | "outputPrice" | "model" | "provider" | "imageConfig">
|
|
138
|
+
): AgentPriceHint | null => {
|
|
139
|
+
if (agent.imageConfig?.enabled && agent.provider && agent.model) {
|
|
140
|
+
try {
|
|
141
|
+
const model = getModelConfig(agent.provider as any, agent.model);
|
|
142
|
+
const pricePerImage = getApproxPricePerImage(
|
|
143
|
+
model,
|
|
144
|
+
agent.imageConfig?.imageSize
|
|
145
|
+
);
|
|
146
|
+
if (typeof pricePerImage === "number") {
|
|
147
|
+
return { type: "per_image", amount: pricePerImage };
|
|
148
|
+
}
|
|
149
|
+
} catch {
|
|
150
|
+
// Fall through to token-based pricing or null.
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
const inputPrice = agent.inputPrice ?? 0;
|
|
155
|
+
const outputPrice = agent.outputPrice ?? 0;
|
|
156
|
+
if (inputPrice === 0 && outputPrice === 0) return null;
|
|
157
|
+
|
|
158
|
+
const amount =
|
|
159
|
+
(inputPrice * TYPICAL_INPUT_TOKENS + outputPrice * TYPICAL_OUTPUT_TOKENS) /
|
|
160
|
+
1_000_000;
|
|
161
|
+
return { type: "per_turn", amount };
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
/** 将金额格式化为用户可读字符串(2 位有效数字,不使用科学计数法)*/
|
|
165
|
+
export const formatPriceAmount = (amount: number): string => {
|
|
166
|
+
if (amount >= 0.1) return amount.toFixed(2);
|
|
167
|
+
if (amount >= 0.01) return amount.toFixed(3);
|
|
168
|
+
if (amount >= 0.001) return amount.toFixed(4);
|
|
169
|
+
if (amount >= 0.0001) return amount.toFixed(5);
|
|
170
|
+
return amount.toFixed(6);
|
|
171
|
+
};
|
|
172
|
+
|
|
173
|
+
export const formatCompactTurnPrice = (
|
|
174
|
+
amount: number
|
|
175
|
+
): CompactTurnPriceDisplay => {
|
|
176
|
+
if (amount <= 0) {
|
|
177
|
+
return { amountText: "0", unitCount: 1 };
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
if (amount >= 0.01) {
|
|
181
|
+
return {
|
|
182
|
+
amountText: trimTrailingZeros(formatPriceAmount(amount)),
|
|
183
|
+
unitCount: 1,
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
const perHundred = amount * 100;
|
|
188
|
+
if (perHundred >= 0.01) {
|
|
189
|
+
return {
|
|
190
|
+
amountText: trimTrailingZeros(formatPriceAmount(perHundred)),
|
|
191
|
+
unitCount: 100,
|
|
192
|
+
};
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
return {
|
|
196
|
+
amountText: trimTrailingZeros(formatPriceAmount(amount * 1000)),
|
|
197
|
+
unitCount: 1000,
|
|
198
|
+
};
|
|
199
|
+
};
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { useState, useEffect, useRef } from "react";
|
|
2
|
+
import { getModelsByProvider, getProviderByModelName } from "ai/llm/providers";
|
|
3
|
+
import type { Model, ModelPrice } from "ai/llm/types";
|
|
4
|
+
|
|
5
|
+
type ModelWithLegacyPricing = Partial<Model> & {
|
|
6
|
+
pricing?: ModelPrice | null;
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
export const resolveModelPrice = (
|
|
10
|
+
model?: ModelWithLegacyPricing | null
|
|
11
|
+
): ModelPrice => {
|
|
12
|
+
const price = model?.price ?? model?.pricing;
|
|
13
|
+
return {
|
|
14
|
+
input: typeof price?.input === "number" ? price.input : 0,
|
|
15
|
+
output: typeof price?.output === "number" ? price.output : 0,
|
|
16
|
+
cachingWrite: price?.cachingWrite,
|
|
17
|
+
cachingRead: price?.cachingRead,
|
|
18
|
+
inputCacheHit: price?.inputCacheHit,
|
|
19
|
+
};
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
const useModelPricing = (
|
|
23
|
+
provider: string,
|
|
24
|
+
modelName: string,
|
|
25
|
+
setValue?: (name: string, value: number) => void
|
|
26
|
+
) => {
|
|
27
|
+
const [models, setModels] = useState<Model[]>([]);
|
|
28
|
+
const [inputPrice, setInputPrice] = useState<number>(0);
|
|
29
|
+
const [outputPrice, setOutputPrice] = useState<number>(0);
|
|
30
|
+
const setValueRef = useRef(setValue);
|
|
31
|
+
|
|
32
|
+
useEffect(() => {
|
|
33
|
+
setValueRef.current = setValue;
|
|
34
|
+
}, [setValue]);
|
|
35
|
+
|
|
36
|
+
useEffect(() => {
|
|
37
|
+
// 若 provider 为空,从模型名反查(防止 provider 漏传导致定价缺失)
|
|
38
|
+
const resolvedProvider = provider || getProviderByModelName(modelName);
|
|
39
|
+
if (!resolvedProvider) return;
|
|
40
|
+
setModels(getModelsByProvider(resolvedProvider));
|
|
41
|
+
}, [provider, modelName]);
|
|
42
|
+
|
|
43
|
+
useEffect(() => {
|
|
44
|
+
const selectedModel = models.find((model) => model.name === modelName);
|
|
45
|
+
if (selectedModel) {
|
|
46
|
+
const price = resolveModelPrice(selectedModel as ModelWithLegacyPricing);
|
|
47
|
+
setInputPrice((current) => current === price.input ? current : price.input);
|
|
48
|
+
setOutputPrice((current) => current === price.output ? current : price.output);
|
|
49
|
+
|
|
50
|
+
if (setValueRef.current) {
|
|
51
|
+
setValueRef.current("inputPrice", price.input);
|
|
52
|
+
setValueRef.current("outputPrice", price.output);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}, [models, modelName]);
|
|
56
|
+
|
|
57
|
+
const updateInputPrice = (value: number) => {
|
|
58
|
+
setInputPrice(value);
|
|
59
|
+
if (setValue) setValue("inputPrice", value);
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
const updateOutputPrice = (value: number) => {
|
|
63
|
+
setOutputPrice(value);
|
|
64
|
+
if (setValue) setValue("outputPrice", value);
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
return {
|
|
68
|
+
inputPrice,
|
|
69
|
+
outputPrice,
|
|
70
|
+
setInputPrice: updateInputPrice,
|
|
71
|
+
setOutputPrice: updateOutputPrice,
|
|
72
|
+
};
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
export default useModelPricing;
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import type { Model } from "./types";
|
|
2
|
+
|
|
3
|
+
export type ImageSizeKey = "1K" | "2K" | "4K";
|
|
4
|
+
|
|
5
|
+
type ImagePricingModel = Pick<
|
|
6
|
+
Model,
|
|
7
|
+
"pricePerImage" | "imageTokenPricePerMillion" | "imageOutputTokenEstimateBySize"
|
|
8
|
+
>;
|
|
9
|
+
|
|
10
|
+
const DEFAULT_IMAGE_SIZE: ImageSizeKey = "1K";
|
|
11
|
+
|
|
12
|
+
export const getApproxPricePerImage = (
|
|
13
|
+
model: ImagePricingModel | null | undefined,
|
|
14
|
+
requestedSize?: ImageSizeKey
|
|
15
|
+
): number | undefined => {
|
|
16
|
+
if (!model) return undefined;
|
|
17
|
+
|
|
18
|
+
if (typeof model.pricePerImage === "number") {
|
|
19
|
+
return model.pricePerImage;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
if (typeof model.imageTokenPricePerMillion !== "number") {
|
|
23
|
+
return undefined;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
const tokenEstimates = model.imageOutputTokenEstimateBySize;
|
|
27
|
+
if (!tokenEstimates) return undefined;
|
|
28
|
+
|
|
29
|
+
const resolvedSize =
|
|
30
|
+
requestedSize && typeof tokenEstimates[requestedSize] === "number"
|
|
31
|
+
? requestedSize
|
|
32
|
+
: DEFAULT_IMAGE_SIZE;
|
|
33
|
+
const outputTokens = tokenEstimates[resolvedSize];
|
|
34
|
+
|
|
35
|
+
if (typeof outputTokens !== "number") {
|
|
36
|
+
return undefined;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
return (model.imageTokenPricePerMillion * outputTokens) / 1_000_000;
|
|
40
|
+
};
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { getModelConfig } from "./providers";
|
|
2
|
+
|
|
3
|
+
export const isResponseAPIModel = (agentConfig) => {
|
|
4
|
+
if (agentConfig.provider !== "openai") return false;
|
|
5
|
+
if (agentConfig.endpointKey === "responses") return true;
|
|
6
|
+
if (!agentConfig.model) return false;
|
|
7
|
+
|
|
8
|
+
try {
|
|
9
|
+
return getModelConfig("openai", agentConfig.model).endpointKey === "responses";
|
|
10
|
+
} catch {
|
|
11
|
+
return false;
|
|
12
|
+
}
|
|
13
|
+
};
|
package/ai/llm/mimo.ts
ADDED
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import type { Model } from "./types";
|
|
2
|
+
|
|
3
|
+
const MIMO_TEXT_PRICE = {
|
|
4
|
+
input: 1 * 8,
|
|
5
|
+
output: 3 * 8,
|
|
6
|
+
cachingRead: 0.2 * 8,
|
|
7
|
+
cachingWrite: 0,
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
const MIMO_TEXT_MODELS = [
|
|
11
|
+
{
|
|
12
|
+
name: "mimo-v2.5-pro",
|
|
13
|
+
displayName: "Xiaomi: MiMo V2.5 Pro",
|
|
14
|
+
},
|
|
15
|
+
{
|
|
16
|
+
name: "mimo-v2.5",
|
|
17
|
+
displayName: "Xiaomi: MiMo V2.5",
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
name: "mimo-v2-pro",
|
|
21
|
+
displayName: "Xiaomi: MiMo V2 Pro",
|
|
22
|
+
},
|
|
23
|
+
{
|
|
24
|
+
name: "mimo-v2-omni",
|
|
25
|
+
displayName: "Xiaomi: MiMo V2 Omni",
|
|
26
|
+
hasAudio: true,
|
|
27
|
+
},
|
|
28
|
+
];
|
|
29
|
+
|
|
30
|
+
const MIMO_TTS_MODELS = [
|
|
31
|
+
{
|
|
32
|
+
name: "mimo-v2.5-tts-voiceclone",
|
|
33
|
+
displayName: "Xiaomi: MiMo V2.5 TTS VoiceClone",
|
|
34
|
+
},
|
|
35
|
+
{
|
|
36
|
+
name: "mimo-v2.5-tts-voicedesign",
|
|
37
|
+
displayName: "Xiaomi: MiMo V2.5 TTS VoiceDesign",
|
|
38
|
+
},
|
|
39
|
+
{
|
|
40
|
+
name: "mimo-v2.5-tts",
|
|
41
|
+
displayName: "Xiaomi: MiMo V2.5 TTS",
|
|
42
|
+
},
|
|
43
|
+
{
|
|
44
|
+
name: "mimo-v2-tts",
|
|
45
|
+
displayName: "Xiaomi: MiMo V2 TTS",
|
|
46
|
+
},
|
|
47
|
+
];
|
|
48
|
+
|
|
49
|
+
export const mimoModels: Model[] = [
|
|
50
|
+
...MIMO_TEXT_MODELS.map((model): Model => ({
|
|
51
|
+
...model,
|
|
52
|
+
hasVision: true,
|
|
53
|
+
price: MIMO_TEXT_PRICE,
|
|
54
|
+
maxOutputTokens: 131072,
|
|
55
|
+
contextWindow: 1048576,
|
|
56
|
+
supportsTool: true,
|
|
57
|
+
supportsReasoningEffort: true,
|
|
58
|
+
description: "MiMo platform chat model.",
|
|
59
|
+
})),
|
|
60
|
+
...MIMO_TTS_MODELS.map((model): Model => ({
|
|
61
|
+
...model,
|
|
62
|
+
hasVision: false,
|
|
63
|
+
hasAudio: true,
|
|
64
|
+
price: MIMO_TEXT_PRICE,
|
|
65
|
+
maxOutputTokens: 32768,
|
|
66
|
+
contextWindow: 32768,
|
|
67
|
+
supportsTool: false,
|
|
68
|
+
supportsReasoningEffort: false,
|
|
69
|
+
description: "MiMo platform TTS model.",
|
|
70
|
+
})),
|
|
71
|
+
];
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
// ai/llm/mistral.ts
|
|
2
|
+
|
|
3
|
+
import type { Model } from "./types";
|
|
4
|
+
|
|
5
|
+
export const mistralModels: Model[] = [
|
|
6
|
+
{
|
|
7
|
+
// 实际使用时建议改成你真实调用用的 model id
|
|
8
|
+
name: "devstral-2512",
|
|
9
|
+
displayName: "Mistral: Devstral 2",
|
|
10
|
+
hasVision: false,
|
|
11
|
+
price: {
|
|
12
|
+
// 单位:$/M tokens,按你给的文档:input 0.4,output 2
|
|
13
|
+
input: 0.1,
|
|
14
|
+
output: 0.1,
|
|
15
|
+
},
|
|
16
|
+
maxOutputTokens: 262144,
|
|
17
|
+
contextWindow: 262144, // 256k ≈ 256 * 1024
|
|
18
|
+
supportsTool: true,
|
|
19
|
+
// 如果后面确认支持 reasoning effort,可以加上:
|
|
20
|
+
// supportsReasoningEffort: true,
|
|
21
|
+
},
|
|
22
|
+
];
|