merlyn/server/utils/agents/defaults.js
Timothy Carambat 1c0d0301b0
Outlook agent via Entra Application (#5427)
* Outlook agent via Entra Application

* translations (#5437)
2026-04-14 14:05:19 -07:00

160 lines
4.9 KiB
JavaScript

const AgentPlugins = require("./aibitat/plugins");
const { SystemSettings } = require("../../models/systemSettings");
const { safeJsonParse } = require("../http");
const Provider = require("./aibitat/providers/ai-provider");
const ImportedPlugin = require("./imported");
const { AgentFlows } = require("../agentFlows");
const MCPCompatibilityLayer = require("../MCP");
// This is a list of skills that are built-in and default enabled.
const DEFAULT_SKILLS = [
AgentPlugins.memory.name,
AgentPlugins.docSummarizer.name,
AgentPlugins.webScraping.name,
];
/**
* Configuration for agent skills that require availability checks and disabled sub-skill lists.
* Each entry maps a skill name to its availability checker and disabled skills list key.
*/
const SKILL_FILTER_CONFIG = {
"filesystem-agent": {
getAvailability: () =>
require("./aibitat/plugins/filesystem/lib").isToolAvailable(),
disabledSettingKey: "disabled_filesystem_skills",
},
"create-files-agent": {
getAvailability: () =>
require("./aibitat/plugins/create-files/lib").isToolAvailable(),
disabledSettingKey: "disabled_create_files_skills",
},
"gmail-agent": {
getAvailability: async () =>
require("./aibitat/plugins/gmail/lib").GmailBridge.isToolAvailable(),
disabledSettingKey: "disabled_gmail_skills",
},
"outlook-agent": {
getAvailability: async () =>
require("./aibitat/plugins/outlook/lib").OutlookBridge.isToolAvailable(),
disabledSettingKey: "disabled_outlook_skills",
},
};
const USER_AGENT = {
name: "USER",
getDefinition: () => {
return {
interrupt: "ALWAYS",
role: "I am the human monitor and oversee this chat. Any questions on action or decision making should be directed to me.",
};
},
};
const WORKSPACE_AGENT = {
name: "@agent",
/**
* Get the definition for the workspace agent with its role (prompt) and functions in Aibitat format
* @param {string} provider
* @param {import("@prisma/client").workspaces | null} workspace
* @param {import("@prisma/client").users | null} user
* @returns {Promise<{ role: string, functions: object[] }>}
*/
getDefinition: async (provider = null, workspace = null, user = null) => {
const basePrompt = await Provider.systemPrompt({
provider,
workspace,
user,
});
return {
role: basePrompt,
functions: [
...(await agentSkillsFromSystemSettings()),
...ImportedPlugin.activeImportedPlugins(),
...AgentFlows.activeFlowPlugins(),
...(await new MCPCompatibilityLayer().activeMCPServers()),
],
};
},
};
/**
* Fetches and preloads the names/identifiers for plugins that will be dynamically
* loaded later
* @returns {Promise<string[]>}
*/
async function agentSkillsFromSystemSettings() {
const systemFunctions = [];
// Load non-imported built-in skills that are configurable, but are default enabled.
const _disabledDefaultSkills = safeJsonParse(
await SystemSettings.getValueOrFallback(
{ label: "disabled_agent_skills" },
"[]"
),
[]
);
DEFAULT_SKILLS.forEach((skill) => {
if (!_disabledDefaultSkills.includes(skill))
systemFunctions.push(AgentPlugins[skill].name);
});
// Load non-imported built-in skills that are configurable.
const _setting = safeJsonParse(
await SystemSettings.getValueOrFallback(
{ label: "default_agent_skills" },
"[]"
),
[]
);
// Pre-load disabled sub-skills and availability for configured skills
const skillFilterState = {};
for (const skillName of Object.keys(SKILL_FILTER_CONFIG)) {
if (!_setting.includes(skillName)) continue;
const config = SKILL_FILTER_CONFIG[skillName];
skillFilterState[skillName] = {
available: await config.getAvailability(),
disabledSubSkills: safeJsonParse(
await SystemSettings.getValueOrFallback(
{ label: config.disabledSettingKey },
"[]"
),
[]
),
};
}
for (const skillName of _setting) {
if (!AgentPlugins.hasOwnProperty(skillName)) continue;
// This is a plugin module with many sub-children plugins who
// need to be named via `${parent}#${child}` naming convention
if (Array.isArray(AgentPlugins[skillName].plugin)) {
for (const subPlugin of AgentPlugins[skillName].plugin) {
// Check if this skill has filter configuration
const filterState = skillFilterState[skillName];
if (filterState) {
if (!filterState.available) continue;
if (filterState.disabledSubSkills.includes(subPlugin.name)) continue;
}
systemFunctions.push(
`${AgentPlugins[skillName].name}#${subPlugin.name}`
);
}
continue;
}
// This is normal single-stage plugin
systemFunctions.push(AgentPlugins[skillName].name);
}
return systemFunctions;
}
module.exports = {
USER_AGENT,
WORKSPACE_AGENT,
agentSkillsFromSystemSettings,
};