mirror of
https://github.com/NoFxAiOS/nofx.git
synced 2026-07-05 20:11:13 +08:00
591 lines
24 KiB
Go
591 lines
24 KiB
Go
package agent
|
||
|
||
import (
|
||
"context"
|
||
"encoding/json"
|
||
"fmt"
|
||
"sort"
|
||
"strings"
|
||
|
||
"nofx/mcp"
|
||
)
|
||
|
||
type llmFlowExtractionTask struct {
|
||
Skill string `json:"skill,omitempty"`
|
||
Action string `json:"action,omitempty"`
|
||
Fields map[string]string `json:"fields,omitempty"`
|
||
}
|
||
|
||
type llmFlowExtractionResult struct {
|
||
Intent string `json:"intent,omitempty"`
|
||
TargetSnapshotID string `json:"target_snapshot_id,omitempty"`
|
||
InlineSubIntent string `json:"inline_sub_intent,omitempty"`
|
||
Fields map[string]string `json:"fields,omitempty"`
|
||
Tasks []llmFlowExtractionTask `json:"tasks,omitempty"`
|
||
Reason string `json:"reason,omitempty"`
|
||
}
|
||
|
||
type llmFlowFieldSpec struct {
|
||
Key string `json:"key"`
|
||
Description string `json:"description"`
|
||
Required bool `json:"required,omitempty"`
|
||
}
|
||
|
||
func buildActiveFlowExtractionPrompt(lang, flowLabel, flowContext string, text string, recentConversationCtx string, currentRefs any, suspendedSnapshots any, extraSections []string) (string, string) {
|
||
systemPrompt := `You extract structured continuation input for an active NOFXi flow.
|
||
Return JSON only. No markdown.
|
||
|
||
You must decide one of:
|
||
- "continue": the user is continuing the current flow and may have supplied fields
|
||
- "switch": the user is switching away to another task
|
||
- "cancel": the user is cancelling the current flow
|
||
- "instant_reply": the user is only chatting / greeting and no task fields should be written
|
||
|
||
Rules:
|
||
- Prefer "continue" only when the message clearly contributes to the current flow.
|
||
- Set target_snapshot_id only when the user is clearly referring to one suspended snapshot from Suspended snapshots JSON.
|
||
- For greetings, thanks, and casual chat, use "instant_reply".
|
||
- Consider Current references JSON and Suspended snapshots JSON when resolving vague references like "那个", "刚才那个", or "前面那个".
|
||
- Treat this as semantic slot filling, not keyword copying.
|
||
- Users will often speak in natural language, shorthand, colloquial labels, translated labels, or mild misspellings instead of exact schema keys.
|
||
- Your job is to decide which allowed canonical field each value belongs to based on the active flow, field descriptions, current missing fields, and conversation context.
|
||
- Never require the user to say the exact internal field key.
|
||
- In task.fields, always emit the canonical field keys from Allowed field spec JSON, never aliases, paraphrases, or user wording.
|
||
- If the user clearly supplied a value for one allowed field, normalize it to that canonical key before returning JSON.`
|
||
|
||
sections := []string{
|
||
fmt.Sprintf("Language: %s", lang),
|
||
fmt.Sprintf("Active flow label: %s", flowLabel),
|
||
flowContext,
|
||
fmt.Sprintf("Current references JSON: %s", mustMarshalJSON(currentRefs)),
|
||
fmt.Sprintf("Suspended snapshots JSON: %s", mustMarshalJSON(suspendedSnapshots)),
|
||
}
|
||
sections = append(sections, extraSections...)
|
||
sections = append(sections, fmt.Sprintf("User message: %s", text), fmt.Sprintf("Recent conversation:\n%s", recentConversationCtx))
|
||
return systemPrompt, strings.Join(sections, "\n")
|
||
}
|
||
|
||
func (a *Agent) extractSkillSessionFieldsWithLLM(ctx context.Context, userID int64, lang, text string, session skillSession) llmFlowExtractionResult {
|
||
if a == nil || a.aiClient == nil {
|
||
return llmFlowExtractionResult{}
|
||
}
|
||
text = strings.TrimSpace(text)
|
||
if text == "" {
|
||
return llmFlowExtractionResult{}
|
||
}
|
||
|
||
flowSummary, fieldSpecs, currentValues, missingFields := skillSessionExtractionContext(session, lang)
|
||
recentConversationCtx := a.buildRecentConversationContext(userID, text)
|
||
state := a.getExecutionState(userID)
|
||
currentRefs := state.CurrentReferences
|
||
if currentRefs == nil {
|
||
currentRefs = a.semanticCurrentReferences(userID)
|
||
}
|
||
skillContext := buildCurrentSkillExecutionContext(lang, session)
|
||
systemPrompt, userPrompt := buildActiveFlowExtractionPrompt(
|
||
lang,
|
||
"skill_session",
|
||
flowSummary,
|
||
text,
|
||
recentConversationCtx,
|
||
currentRefs,
|
||
a.SnapshotManager(userID).List(),
|
||
[]string{
|
||
skillContext,
|
||
fmt.Sprintf("Allowed field spec JSON: %s", mustMarshalJSON(fieldSpecs)),
|
||
fmt.Sprintf("Current flow field values JSON: %s", mustMarshalJSON(currentValues)),
|
||
fmt.Sprintf("Current missing fields JSON: %s", mustMarshalJSON(missingFields)),
|
||
},
|
||
)
|
||
waitingHint := ""
|
||
if len(missingFields) > 0 {
|
||
waitingHint = fmt.Sprintf("\n- The flow is currently waiting for the user to provide: [%s]. Before deciding \"switch\", first check whether the user message can fill any of these fields — even without an explicit prefix or keyword.", strings.Join(missingFields, ", "))
|
||
}
|
||
systemPrompt += `
|
||
- This is the structured continuation input for an active NOFXi task flow.
|
||
- For "continue", return exactly one task for the active skill/action and place extracted field values in task.fields.
|
||
- Only extract fields from the allowed field spec list.
|
||
- Treat Allowed field spec JSON as the canonical output schema.
|
||
- If a user-provided value does not fit one of those canonical keys, omit it; never create another key.
|
||
- Use field descriptions plus current missing fields to infer the best canonical destination field for each user-provided value.
|
||
- When the user supplies a credential, endpoint, name, toggle, or config value in natural language, map it to the most plausible allowed canonical field instead of echoing the user's label.
|
||
- Do not return near-match keys, guessed aliases, or raw user labels as JSON keys.
|
||
- Do not invent values that were not supported by the user message or strong context.
|
||
- If the user explicitly says "you choose one for me", you may leave that field empty and explain it in reason.
|
||
- If the active skill dependency summary says the current flow depends on other resource configs, treat dependency repair as continuation of the active flow instead of a new peer task.
|
||
- When the user clearly wants to create a new dependency resource (e.g. "选(2)", "新建策略", "创建交易所"), set inline_sub_intent="create_sub_resource".
|
||
- When the user clearly wants to edit/update an existing dependency resource (e.g. "编辑策略", "改一下模型"), set inline_sub_intent="edit_sub_resource".
|
||
- For "switch", you may return tasks for the new request if they are clear enough.
|
||
- If no field can be safely extracted for the current flow, return "switch" or "instant_reply", not fake fields.` + waitingHint + `
|
||
|
||
Return JSON with this shape:
|
||
{"intent":"continue|switch|cancel|instant_reply","target_snapshot_id":"","inline_sub_intent":"","tasks":[{"skill":"","action":"","fields":{}}],"reason":""}`
|
||
|
||
stageCtx, cancel := withPlannerStageTimeout(ctx, directReplyTimeout)
|
||
defer cancel()
|
||
raw, err := a.aiClient.CallWithRequest(&mcp.Request{
|
||
Messages: []mcp.Message{
|
||
mcp.NewSystemMessage(systemPrompt),
|
||
mcp.NewUserMessage(userPrompt),
|
||
},
|
||
Ctx: stageCtx,
|
||
})
|
||
if err != nil {
|
||
return llmFlowExtractionResult{}
|
||
}
|
||
return filterLLMFlowExtractionFields(parseLLMFlowExtractionResult(raw), fieldSpecs)
|
||
}
|
||
|
||
func parseLLMFlowExtractionResult(raw string) llmFlowExtractionResult {
|
||
out, ok := parseRawFlowExtractionEnvelope(raw)
|
||
if !ok {
|
||
return llmFlowExtractionResult{}
|
||
}
|
||
switch out.Intent {
|
||
case "continue", "switch", "cancel", "instant_reply":
|
||
return out
|
||
default:
|
||
return llmFlowExtractionResult{}
|
||
}
|
||
}
|
||
|
||
func parseRawFlowExtractionEnvelope(raw string) (llmFlowExtractionResult, bool) {
|
||
raw = strings.TrimSpace(raw)
|
||
raw = strings.TrimPrefix(raw, "```json")
|
||
raw = strings.TrimPrefix(raw, "```")
|
||
raw = strings.TrimSuffix(raw, "```")
|
||
raw = strings.TrimSpace(raw)
|
||
|
||
var out llmFlowExtractionResult
|
||
if err := json.Unmarshal([]byte(raw), &out); err != nil {
|
||
start := strings.Index(raw, "{")
|
||
end := strings.LastIndex(raw, "}")
|
||
if start < 0 || end <= start || json.Unmarshal([]byte(raw[start:end+1]), &out) != nil {
|
||
return llmFlowExtractionResult{}, false
|
||
}
|
||
}
|
||
|
||
out.Intent = strings.TrimSpace(strings.ToLower(out.Intent))
|
||
out.TargetSnapshotID = strings.TrimSpace(out.TargetSnapshotID)
|
||
out.Reason = strings.TrimSpace(out.Reason)
|
||
if len(out.Fields) > 0 {
|
||
clean := make(map[string]string, len(out.Fields))
|
||
for key, value := range out.Fields {
|
||
key = strings.TrimSpace(key)
|
||
value = strings.TrimSpace(value)
|
||
if key == "" || value == "" {
|
||
continue
|
||
}
|
||
clean[key] = value
|
||
}
|
||
out.Fields = clean
|
||
}
|
||
cleanTasks := make([]llmFlowExtractionTask, 0, len(out.Tasks))
|
||
for _, task := range out.Tasks {
|
||
task.Skill = strings.TrimSpace(task.Skill)
|
||
task.Action = strings.TrimSpace(task.Action)
|
||
if len(task.Fields) > 0 {
|
||
clean := make(map[string]string, len(task.Fields))
|
||
for key, value := range task.Fields {
|
||
key = strings.TrimSpace(key)
|
||
value = strings.TrimSpace(value)
|
||
if key == "" || value == "" {
|
||
continue
|
||
}
|
||
clean[key] = value
|
||
}
|
||
task.Fields = clean
|
||
}
|
||
cleanTasks = append(cleanTasks, task)
|
||
}
|
||
out.Tasks = cleanTasks
|
||
return out, out.Intent != ""
|
||
}
|
||
|
||
func filterLLMFlowExtractionFields(result llmFlowExtractionResult, specs []llmFlowFieldSpec) llmFlowExtractionResult {
|
||
if len(specs) == 0 {
|
||
result.Fields = nil
|
||
for i := range result.Tasks {
|
||
result.Tasks[i].Fields = nil
|
||
}
|
||
return result
|
||
}
|
||
allowed := make(map[string]struct{}, len(specs))
|
||
for _, spec := range specs {
|
||
key := strings.TrimSpace(spec.Key)
|
||
if key != "" {
|
||
allowed[key] = struct{}{}
|
||
}
|
||
}
|
||
filter := func(fields map[string]string) map[string]string {
|
||
if len(fields) == 0 {
|
||
return fields
|
||
}
|
||
clean := make(map[string]string, len(fields))
|
||
for key, value := range fields {
|
||
if _, ok := allowed[key]; !ok {
|
||
continue
|
||
}
|
||
clean[key] = value
|
||
}
|
||
if len(clean) == 0 {
|
||
return nil
|
||
}
|
||
return clean
|
||
}
|
||
result.Fields = filter(result.Fields)
|
||
for i := range result.Tasks {
|
||
result.Tasks[i].Fields = filter(result.Tasks[i].Fields)
|
||
}
|
||
return result
|
||
}
|
||
|
||
func skillSessionExtractionContext(session skillSession, lang string) (string, []llmFlowFieldSpec, map[string]string, []string) {
|
||
currentStep, _ := currentSkillDAGStep(session)
|
||
fieldSpecs := allowedFieldSpecsForSkillSession(session, lang)
|
||
currentValues := currentFieldValuesForSkillSession(session)
|
||
missing := missingFieldKeysForSkillSession(session)
|
||
summary := fmt.Sprintf("Active flow type: skill_session\nSkill: %s\nAction: %s\nCurrent DAG step: %s", session.Name, session.Action, currentStep.ID)
|
||
return summary, fieldSpecs, currentValues, missing
|
||
}
|
||
|
||
func allowedFieldSpecsForSkillSession(session skillSession, lang string) []llmFlowFieldSpec {
|
||
add := func(out *[]llmFlowFieldSpec, key, description string, required bool) {
|
||
*out = append(*out, llmFlowFieldSpec{Key: key, Description: description, Required: required})
|
||
}
|
||
out := make([]llmFlowFieldSpec, 0, 24)
|
||
if actionRequiresSlot(session.Name, session.Action, "target_ref") {
|
||
add(&out, "target_ref_id", slotDisplayName("target_ref", lang)+" ID", true)
|
||
add(&out, "target_ref_name", slotDisplayName("target_ref", lang), true)
|
||
}
|
||
if supportsBulkTargetSelection(session.Name, session.Action) {
|
||
add(&out, "bulk_scope", "bulk deletion scope, use all only when the user clearly requested all targets", false)
|
||
}
|
||
switch session.Name {
|
||
case "model_management":
|
||
required := map[string]bool{"provider": true}
|
||
if strings.HasPrefix(session.Action, "update") {
|
||
add(&out, "update_field", displayCatalogFieldName("update_field", lang), false)
|
||
}
|
||
add(&out, "provider", slotDisplayName("provider", lang), required["provider"])
|
||
add(&out, "name", displayCatalogFieldName("name", lang), required["name"])
|
||
add(&out, "custom_model_name", displayCatalogFieldName("custom_model_name", lang), required["custom_model_name"])
|
||
add(&out, "api_key", displayCatalogFieldName("api_key", lang), required["api_key"])
|
||
add(&out, "custom_api_url", displayCatalogFieldName("custom_api_url", lang), false)
|
||
add(&out, "enabled", displayCatalogFieldName("enabled", lang), false)
|
||
case "exchange_management":
|
||
required := map[string]bool{"exchange_type": true, "account_name": true}
|
||
if strings.HasPrefix(session.Action, "update") {
|
||
add(&out, "update_field", displayCatalogFieldName("update_field", lang), false)
|
||
}
|
||
add(&out, "exchange_type", slotDisplayName("exchange_type", lang), required["exchange_type"])
|
||
add(&out, "account_name", displayCatalogFieldName("account_name", lang), required["account_name"])
|
||
add(&out, "api_key", displayCatalogFieldName("api_key", lang), false)
|
||
add(&out, "secret_key", displayCatalogFieldName("secret_key", lang), false)
|
||
add(&out, "passphrase", displayCatalogFieldName("passphrase", lang), false)
|
||
add(&out, "testnet", displayCatalogFieldName("testnet", lang), false)
|
||
add(&out, "enabled", displayCatalogFieldName("enabled", lang), false)
|
||
add(&out, "hyperliquid_wallet_addr", displayCatalogFieldName("hyperliquid_wallet_addr", lang), false)
|
||
add(&out, "aster_user", displayCatalogFieldName("aster_user", lang), false)
|
||
add(&out, "aster_signer", displayCatalogFieldName("aster_signer", lang), false)
|
||
add(&out, "aster_private_key", displayCatalogFieldName("aster_private_key", lang), false)
|
||
add(&out, "lighter_wallet_addr", displayCatalogFieldName("lighter_wallet_addr", lang), false)
|
||
add(&out, "lighter_api_key_private_key", displayCatalogFieldName("lighter_api_key_private_key", lang), false)
|
||
add(&out, "lighter_api_key_index", displayCatalogFieldName("lighter_api_key_index", lang), false)
|
||
case "trader_management":
|
||
if strings.HasPrefix(session.Action, "update") {
|
||
add(&out, "update_field", displayCatalogFieldName("update_field", lang), false)
|
||
}
|
||
add(&out, "name", slotDisplayName("name", lang), true)
|
||
add(&out, "exchange_id", slotDisplayName("exchange", lang)+" ID", false)
|
||
add(&out, "exchange_name", slotDisplayName("exchange", lang), true)
|
||
add(&out, "model_id", slotDisplayName("model", lang)+" ID", false)
|
||
add(&out, "model_name", slotDisplayName("model", lang), true)
|
||
add(&out, "strategy_id", slotDisplayName("strategy", lang)+" ID", false)
|
||
add(&out, "strategy_name", slotDisplayName("strategy", lang), true)
|
||
add(&out, "auto_start", "auto_start", false)
|
||
add(&out, "scan_interval_minutes", displayCatalogFieldName("scan_interval_minutes", lang), false)
|
||
add(&out, "is_cross_margin", displayCatalogFieldName("is_cross_margin", lang), false)
|
||
add(&out, "show_in_competition", displayCatalogFieldName("show_in_competition", lang), false)
|
||
case "strategy_management":
|
||
if session.Action == "update_config" {
|
||
add(&out, "config_field", strategyConfigFieldDisplayName("config_field", lang), false)
|
||
add(&out, "config_value", strategyConfigFieldDisplayName("config_value", lang), false)
|
||
}
|
||
add(&out, "name", slotDisplayName("name", lang), true)
|
||
for _, key := range manualStrategyEditableFieldKeys() {
|
||
add(&out, key, strategyConfigFieldDisplayName(key, lang), false)
|
||
}
|
||
}
|
||
return out
|
||
}
|
||
|
||
func currentFieldValuesForSkillSession(session skillSession) map[string]string {
|
||
values := map[string]string{}
|
||
for key, value := range session.Fields {
|
||
if trimmed := strings.TrimSpace(value); trimmed != "" {
|
||
values[key] = trimmed
|
||
}
|
||
}
|
||
if session.TargetRef != nil {
|
||
if session.TargetRef.ID != "" {
|
||
values["target_ref_id"] = session.TargetRef.ID
|
||
}
|
||
if session.TargetRef.Name != "" {
|
||
values["target_ref_name"] = session.TargetRef.Name
|
||
}
|
||
}
|
||
for _, key := range []string{"name", "exchange_id", "exchange_name", "model_id", "model_name", "strategy_id", "strategy_name", "auto_start"} {
|
||
if value := fieldValue(session, key); value != "" {
|
||
values[key] = value
|
||
}
|
||
}
|
||
return values
|
||
}
|
||
|
||
func missingFieldKeysForSkillSession(session skillSession) []string {
|
||
missing := make([]string, 0, 8)
|
||
switch session.Name {
|
||
case "model_management":
|
||
if session.Action != "create" && session.Action != "query_list" && session.Action != "query" && session.Action != "query_detail" && session.TargetRef == nil {
|
||
missing = append(missing, "target_ref")
|
||
}
|
||
if strings.HasPrefix(session.Action, "update") {
|
||
if session.Action == "update_status" {
|
||
if fieldValue(session, "enabled") == "" {
|
||
missing = append(missing, "enabled")
|
||
}
|
||
} else if session.Action == "update_endpoint" {
|
||
if fieldValue(session, "custom_api_url") == "" {
|
||
missing = append(missing, "custom_api_url")
|
||
}
|
||
} else {
|
||
if fieldValue(session, "update_field") == "" {
|
||
missing = append(missing, "update_field")
|
||
}
|
||
}
|
||
} else {
|
||
for _, key := range []string{"provider"} {
|
||
if fieldValue(session, key) == "" {
|
||
missing = append(missing, key)
|
||
}
|
||
}
|
||
if fieldValue(session, "api_key") == "" {
|
||
missing = append(missing, "api_key")
|
||
}
|
||
}
|
||
case "exchange_management":
|
||
if session.Action != "create" && session.Action != "query_list" && session.Action != "query" && session.Action != "query_detail" && session.TargetRef == nil {
|
||
missing = append(missing, "target_ref")
|
||
}
|
||
if strings.HasPrefix(session.Action, "update") {
|
||
if session.Action == "update_status" {
|
||
if fieldValue(session, "enabled") == "" {
|
||
missing = append(missing, "enabled")
|
||
}
|
||
} else {
|
||
if fieldValue(session, "update_field") == "" {
|
||
missing = append(missing, "update_field")
|
||
}
|
||
}
|
||
} else {
|
||
for _, key := range []string{"exchange_type", "account_name", "api_key", "secret_key"} {
|
||
if fieldValue(session, key) == "" {
|
||
missing = append(missing, key)
|
||
}
|
||
}
|
||
}
|
||
case "trader_management":
|
||
if strings.HasPrefix(session.Action, "update") || strings.HasPrefix(session.Action, "configure_") {
|
||
if session.TargetRef == nil {
|
||
missing = append(missing, "target_ref")
|
||
}
|
||
if session.Action == "update_bindings" || session.Action == "configure_strategy" || session.Action == "configure_exchange" || session.Action == "configure_model" {
|
||
switch session.Action {
|
||
case "configure_strategy":
|
||
if fieldValue(session, "strategy_id") == "" && fieldValue(session, "strategy_name") == "" {
|
||
missing = append(missing, "strategy_name")
|
||
}
|
||
break
|
||
case "configure_exchange":
|
||
if fieldValue(session, "exchange_id") == "" && fieldValue(session, "exchange_name") == "" {
|
||
missing = append(missing, "exchange_name")
|
||
}
|
||
break
|
||
case "configure_model":
|
||
if fieldValue(session, "model_id") == "" && fieldValue(session, "model_name") == "" {
|
||
missing = append(missing, "model_name")
|
||
}
|
||
break
|
||
}
|
||
if len(missing) > 0 {
|
||
break
|
||
}
|
||
if fieldValue(session, "model_id") == "" && fieldValue(session, "exchange_id") == "" && fieldValue(session, "strategy_id") == "" &&
|
||
fieldValue(session, "model_name") == "" && fieldValue(session, "exchange_name") == "" && fieldValue(session, "strategy_name") == "" {
|
||
missing = append(missing, "update_field")
|
||
}
|
||
} else {
|
||
if fieldValue(session, "update_field") == "" {
|
||
missing = append(missing, "update_field")
|
||
}
|
||
}
|
||
} else {
|
||
if fieldValue(session, "name") == "" {
|
||
missing = append(missing, "name")
|
||
}
|
||
if fieldValue(session, "exchange_id") == "" && fieldValue(session, "exchange_name") == "" {
|
||
missing = append(missing, "exchange_name")
|
||
}
|
||
if fieldValue(session, "model_id") == "" && fieldValue(session, "model_name") == "" {
|
||
missing = append(missing, "model_name")
|
||
}
|
||
if fieldValue(session, "strategy_id") == "" && fieldValue(session, "strategy_name") == "" {
|
||
missing = append(missing, "strategy_name")
|
||
}
|
||
}
|
||
case "strategy_management":
|
||
if session.Action != "create" && session.Action != "query_list" && session.Action != "query" && session.Action != "query_detail" && session.TargetRef == nil {
|
||
missing = append(missing, "target_ref")
|
||
}
|
||
switch session.Action {
|
||
case "update_config":
|
||
if fieldValue(session, "config_field") == "" {
|
||
missing = append(missing, "config_field")
|
||
} else if fieldValue(session, "config_value") == "" {
|
||
missing = append(missing, "config_value")
|
||
}
|
||
default:
|
||
if fieldValue(session, "name") == "" {
|
||
missing = append(missing, "name")
|
||
}
|
||
}
|
||
}
|
||
sort.Strings(missing)
|
||
return missing
|
||
}
|
||
|
||
func providerExplicitlyMentionedInText(provider, text string) bool {
|
||
provider = strings.ToLower(strings.TrimSpace(provider))
|
||
lower := strings.ToLower(strings.TrimSpace(text))
|
||
if provider == "" || lower == "" {
|
||
return false
|
||
}
|
||
spec, _ := modelProviderSpecByID(provider)
|
||
candidates := []string{provider, strings.ToLower(strings.TrimSpace(spec.DisplayName))}
|
||
switch provider {
|
||
case "blockrun-base":
|
||
candidates = append(candidates, "blockrun", "blockrun base", "base wallet")
|
||
case "blockrun-sol":
|
||
candidates = append(candidates, "blockrun", "blockrun sol", "solana wallet")
|
||
case "claw402":
|
||
candidates = append(candidates, "claw 402")
|
||
}
|
||
for _, candidate := range candidates {
|
||
candidate = strings.TrimSpace(candidate)
|
||
if candidate != "" && strings.Contains(lower, candidate) {
|
||
return true
|
||
}
|
||
}
|
||
return false
|
||
}
|
||
|
||
func sanitizeLLMExtractionForSkillSession(text string, session skillSession, result llmFlowExtractionResult) llmFlowExtractionResult {
|
||
if session.Name != "model_management" || len(result.Tasks) == 0 {
|
||
return result
|
||
}
|
||
task := result.Tasks[0]
|
||
if task.Fields == nil {
|
||
return result
|
||
}
|
||
if provider := strings.TrimSpace(task.Fields["provider"]); provider != "" && !providerExplicitlyMentionedInText(provider, text) {
|
||
delete(task.Fields, "provider")
|
||
result.Tasks[0] = task
|
||
}
|
||
return result
|
||
}
|
||
|
||
func (a *Agent) applyLLMExtractionToSkillSession(storeUserID string, session *skillSession, result llmFlowExtractionResult, lang string, text string) {
|
||
if session == nil {
|
||
return
|
||
}
|
||
result = sanitizeLLMExtractionForSkillSession(text, *session, result)
|
||
if sub := strings.TrimSpace(result.InlineSubIntent); sub == "create_sub_resource" || sub == "edit_sub_resource" {
|
||
setField(session, "inline_sub_intent", sub)
|
||
}
|
||
if len(result.Tasks) == 0 {
|
||
return
|
||
}
|
||
task := result.Tasks[0]
|
||
if task.Skill != "" && task.Skill != session.Name {
|
||
return
|
||
}
|
||
if task.Action != "" && session.Action != "" && task.Action != session.Action {
|
||
return
|
||
}
|
||
for key, value := range task.Fields {
|
||
value = strings.TrimSpace(value)
|
||
if value == "" {
|
||
continue
|
||
}
|
||
switch key {
|
||
case "target_ref_id":
|
||
if session.TargetRef == nil {
|
||
session.TargetRef = &EntityReference{}
|
||
}
|
||
session.TargetRef.ID = value
|
||
if session.TargetRef.Source == "" {
|
||
session.TargetRef.Source = "llm_extraction"
|
||
}
|
||
continue
|
||
case "target_ref_name":
|
||
if session.TargetRef == nil {
|
||
session.TargetRef = &EntityReference{}
|
||
}
|
||
session.TargetRef.Name = value
|
||
if session.TargetRef.Source == "" {
|
||
session.TargetRef.Source = "llm_extraction"
|
||
}
|
||
continue
|
||
}
|
||
switch session.Name {
|
||
case "model_management":
|
||
if key == "provider" || key == "name" || key == "custom_model_name" || key == "api_key" || key == "custom_api_url" || key == "enabled" || key == "update_field" {
|
||
setField(session, key, value)
|
||
}
|
||
case "exchange_management":
|
||
switch key {
|
||
case "exchange_type", "account_name", "api_key", "secret_key", "passphrase", "testnet", "enabled", "update_field":
|
||
setField(session, key, value)
|
||
}
|
||
case "trader_management":
|
||
switch key {
|
||
case "update_field":
|
||
setField(session, key, value)
|
||
case "name", "exchange_id", "exchange_name", "model_id", "ai_model_id", "model_name", "strategy_id", "strategy_name", "auto_start":
|
||
setField(session, key, value)
|
||
case "scan_interval_minutes", "is_cross_margin", "show_in_competition":
|
||
setField(session, key, value)
|
||
}
|
||
case "strategy_management":
|
||
if key == "name" {
|
||
setField(session, "name", value)
|
||
continue
|
||
}
|
||
if key == "config_field" || key == "config_value" {
|
||
setField(session, key, value)
|
||
continue
|
||
}
|
||
if session.Action == "update_config" {
|
||
setField(session, "config_field", key)
|
||
setField(session, "config_value", value)
|
||
continue
|
||
}
|
||
cfg := unmarshalStrategyCreateDraft(fieldValue(*session, strategyCreateDraftConfigField), lang)
|
||
if err := applyStrategyConfigPatch(&cfg, key, value); err == nil {
|
||
setField(session, strategyCreateDraftConfigField, marshalStrategyCreateDraft(cfg))
|
||
}
|
||
}
|
||
}
|
||
}
|