Files
anything-llm/server/endpoints/utils/outlookAgentUtils.js
Timothy Carambat 4172751858 5427 translations (#5429)
* Outlook agent via Entra Application

* translations
2026-04-14 13:38:03 -07:00

191 lines
6.4 KiB
JavaScript

const { reqBody } = require("../../utils/http");
const {
isSingleUserMode,
} = require("../../utils/middleware/multiUserProtected");
const { validatedRequest } = require("../../utils/middleware/validatedRequest");
/**
* Constructs the OAuth redirect URI from the request headers.
* @param {Object} request - Express request object
* @returns {string} The redirect URI for OAuth callback
*/
function getOutlookRedirectUri(request) {
const protocol = request.headers["x-forwarded-proto"] || request.protocol;
const host = request.headers["x-forwarded-host"] || request.get("host");
return `${protocol}://${host}/api/agent-skills/outlook/auth-callback`;
}
function outlookAgentEndpoints(app) {
if (!app) return;
app.post(
"/admin/agent-skills/outlook/auth-url",
[validatedRequest, isSingleUserMode],
async (request, response) => {
try {
const { clientId, tenantId, clientSecret, authType } = reqBody(request);
if (!clientId || !clientSecret) {
return response.status(400).json({
success: false,
error: "Client ID and Client Secret are required.",
});
}
const outlookLib = require("../../utils/agents/aibitat/plugins/outlook/lib");
const { AUTH_TYPES } = outlookLib;
const validAuthType = Object.values(AUTH_TYPES).includes(authType)
? authType
: AUTH_TYPES.common;
if (validAuthType === AUTH_TYPES.organization && !tenantId) {
return response.status(400).json({
success: false,
error:
"Tenant ID is required for organization-only authentication.",
});
}
const existingConfig = await outlookLib.OutlookBridge.getConfig();
const configUpdate = {
...existingConfig,
clientId: clientId.trim(),
tenantId: tenantId?.trim() || "",
authType: validAuthType,
};
if (!/^\*+$/.test(clientSecret))
configUpdate.clientSecret = clientSecret.trim();
// If auth type changed, clear tokens as they won't work with different authority
if (
existingConfig.authType &&
existingConfig.authType !== validAuthType
) {
delete configUpdate.accessToken;
delete configUpdate.refreshToken;
delete configUpdate.tokenExpiry;
}
await outlookLib.OutlookBridge.updateConfig(configUpdate);
outlookLib.reset();
const redirectUri = getOutlookRedirectUri(request);
const result = await outlookLib.getAuthUrl(redirectUri);
if (!result.success) {
return response
.status(400)
.json({ success: false, error: result.error });
}
return response.status(200).json({ success: true, url: result.url });
} catch (e) {
console.error("Outlook auth URL error:", e);
response.status(500).json({ success: false, error: e.message });
}
}
);
app.get(
"/agent-skills/outlook/auth-callback",
[validatedRequest, isSingleUserMode],
async (request, response) => {
try {
const { code, error, error_description } = request.query;
if (error) {
console.error("Outlook OAuth error:", error, error_description);
return response.redirect(
`/?outlook_auth=error&message=${encodeURIComponent(error_description || error)}`
);
}
if (!code) {
return response.redirect(
"/?outlook_auth=error&message=No authorization code received"
);
}
const outlookLib = require("../../utils/agents/aibitat/plugins/outlook/lib");
const redirectUri = getOutlookRedirectUri(request);
const result = await outlookLib.exchangeCodeForToken(code, redirectUri);
const frontendUrl =
process.env.NODE_ENV === "development" ? "http://localhost:3000" : "";
if (!result.success) {
return response.redirect(
`${frontendUrl}/settings/agents?outlook_auth=error&message=${encodeURIComponent(result.error)}`
);
}
return response.redirect(
`${frontendUrl}/settings/agents?outlook_auth=success`
);
} catch (e) {
console.error("Outlook OAuth callback error:", e);
response.redirect(
`/?outlook_auth=error&message=${encodeURIComponent(e.message)}`
);
}
}
);
app.get(
"/admin/agent-skills/outlook/status",
[validatedRequest, isSingleUserMode],
async (_request, response) => {
try {
const outlookLib = require("../../utils/agents/aibitat/plugins/outlook/lib");
const { AUTH_TYPES, normalizeTokenExpiry } = outlookLib;
const config = await outlookLib.OutlookBridge.getConfig();
const isConfigured = await outlookLib.OutlookBridge.isToolAvailable();
const authType = config.authType || AUTH_TYPES.common;
let hasCredentials = !!(config.clientId && config.clientSecret);
if (authType === AUTH_TYPES.organization) {
hasCredentials = hasCredentials && !!config.tenantId;
}
const safeConfig = {
clientId: config.clientId || "",
tenantId: config.tenantId || "",
clientSecret: config.clientSecret ? "********" : "",
authType: config.authType || AUTH_TYPES.common,
};
return response.status(200).json({
success: true,
isConfigured,
hasCredentials,
isAuthenticated: !!config.accessToken,
tokenExpiry: normalizeTokenExpiry(config.tokenExpiry),
config: safeConfig,
});
} catch (e) {
console.error("Outlook status error:", e);
response.status(500).json({ success: false, error: e.message });
}
}
);
app.post(
"/admin/agent-skills/outlook/revoke",
[validatedRequest, isSingleUserMode],
async (_request, response) => {
try {
const outlookLib = require("../../utils/agents/aibitat/plugins/outlook/lib");
const { SystemSettings } = require("../../models/systemSettings");
await SystemSettings.delete({ label: "outlook_agent_config" });
outlookLib.reset();
return response.status(200).json({ success: true });
} catch (e) {
console.error("Outlook revoke error:", e);
response.status(500).json({ success: false, error: e.message });
}
}
);
}
module.exports = { outlookAgentEndpoints };