merlyn/server/utils/agents/aibitat/plugins/gmail/send/gmail-send-email.js
Timothy Carambat cec67d77f2
GMail Agent Skill (#5400)
* wip

* remove label tech

* ask to read attachments

* update skills

* Skill ready and tested

* report dynamic citations and generic get mailbox util

* norm translations

* translations

* remove dead code, remove connector in multiUser

* simple refactor - dont ask for drafts

* refactor filesize helper

* norm translations, remove read_messages skill
2026-04-10 14:14:12 -07:00

243 lines
9.0 KiB
JavaScript

const gmailLib = require("../lib.js");
const { prepareAttachment, MAX_TOTAL_ATTACHMENT_SIZE } = require("../lib.js");
const { humanFileSize } = require("../../../../../helpers");
module.exports.GmailSendEmail = {
name: "gmail-send-email",
plugin: function () {
return {
name: "gmail-send-email",
setup(aibitat) {
aibitat.function({
super: aibitat,
name: this.name,
description:
"Send an email immediately through Gmail. " +
"This action sends the email right away and cannot be undone. " +
"For composing emails that need review before sending, use gmail-create-draft instead.",
examples: [
{
prompt: "Send an email to john@example.com about the project",
call: JSON.stringify({
to: "john@example.com",
subject: "Project Update",
body: "Hi John,\n\nHere's the latest update on the project.\n\nBest regards",
}),
},
{
prompt: "Send an email with CC recipients",
call: JSON.stringify({
to: "john@example.com",
subject: "Meeting Notes",
body: "Please find the meeting notes attached.",
cc: "manager@example.com, team@example.com",
}),
},
{
prompt: "Send an email with an attachment",
call: JSON.stringify({
to: "john@example.com",
subject: "Report",
body: "Please find the report attached.",
attachments: ["/Users/me/Documents/report.pdf"],
}),
},
],
parameters: {
$schema: "http://json-schema.org/draft-07/schema#",
type: "object",
properties: {
to: {
type: "string",
description:
"Recipient email address(es). Multiple addresses can be comma-separated.",
},
subject: {
type: "string",
description: "Email subject line.",
},
body: {
type: "string",
description: "Plain text email body content.",
},
cc: {
type: "string",
description: "CC recipient email address(es). Optional.",
},
bcc: {
type: "string",
description: "BCC recipient email address(es). Optional.",
},
htmlBody: {
type: "string",
description: "HTML version of the email body. Optional.",
},
replyTo: {
type: "string",
description: "Reply-to email address. Optional.",
},
attachments: {
type: "array",
items: { type: "string" },
description:
"Array of absolute file paths to attach to the email.",
},
},
required: ["to", "subject", "body"],
additionalProperties: false,
},
handler: async function ({
to,
subject,
body,
cc,
bcc,
htmlBody,
replyTo,
attachments,
}) {
try {
this.super.handlerProps.log(`Using the gmail-send-email tool.`);
if (!to || !subject) {
return "Error: 'to' and 'subject' are required.";
}
const preparedAttachments = [];
const attachmentSummaries = [];
let totalAttachmentSize = 0;
if (Array.isArray(attachments) && attachments.length > 0) {
this.super.introspect(
`${this.caller}: Validating ${attachments.length} attachment(s)...`
);
for (const filePath of attachments) {
const result = prepareAttachment(filePath);
if (!result.success) {
this.super.introspect(
`${this.caller}: Attachment validation failed - ${result.error}`
);
return `Error with attachment: ${result.error}`;
}
totalAttachmentSize += result.fileInfo.size;
if (totalAttachmentSize > MAX_TOTAL_ATTACHMENT_SIZE) {
const totalFormatted = humanFileSize(
totalAttachmentSize,
true
);
this.super.introspect(
`${this.caller}: Total attachment size (${totalFormatted}) exceeds 20MB limit`
);
return `Error: Total attachment size (${totalFormatted}) exceeds the 20MB limit. Please reduce the number or size of attachments.`;
}
if (this.super.requestToolApproval) {
const approval = await this.super.requestToolApproval({
skillName: this.name,
payload: {
fileName: result.fileInfo.name,
fileSize: result.fileInfo.sizeFormatted,
filePath: result.fileInfo.path,
},
description:
`Attach file "${result.fileInfo.name}" (${result.fileInfo.sizeFormatted}) to email? ` +
`This file will be sent to ${to}.`,
});
if (!approval.approved) {
this.super.introspect(
`${this.caller}: User rejected attaching "${result.fileInfo.name}"`
);
return `Attachment rejected by user: ${result.fileInfo.name}. ${approval.message || ""}`;
}
}
preparedAttachments.push(result.attachment);
attachmentSummaries.push(
`${result.fileInfo.name} (${result.fileInfo.sizeFormatted})`
);
this.super.introspect(
`${this.caller}: Prepared attachment "${result.fileInfo.name}"`
);
}
}
if (this.super.requestToolApproval) {
const attachmentNote =
preparedAttachments.length > 0
? ` with ${preparedAttachments.length} attachment(s): ${attachmentSummaries.join(", ")}`
: "";
const approval = await this.super.requestToolApproval({
skillName: this.name,
payload: {
to,
subject,
attachmentCount: preparedAttachments.length,
},
description: `Send email to "${to}" with subject "${subject}"${attachmentNote} - This will send immediately`,
});
if (!approval.approved) {
this.super.introspect(
`${this.caller}: User rejected the ${this.name} request.`
);
return approval.message;
}
}
this.super.introspect(
`${this.caller}: Sending email to ${to}${preparedAttachments.length > 0 ? ` with ${preparedAttachments.length} attachment(s)` : ""}`
);
const options = {};
if (cc) options.cc = cc;
if (bcc) options.bcc = bcc;
if (htmlBody) options.htmlBody = htmlBody;
if (replyTo) options.replyTo = replyTo;
if (preparedAttachments.length > 0) {
options.attachments = preparedAttachments;
}
const result = await gmailLib.sendEmail(
to,
subject,
body,
options
);
if (!result.success) {
this.super.introspect(
`${this.caller}: Failed to send email - ${result.error}`
);
return `Error sending email: ${result.error}`;
}
this.super.introspect(
`${this.caller}: Successfully sent email to ${to}`
);
return (
`Successfully sent email:\n` +
`To: ${to}\n` +
`Subject: ${subject}\n` +
(cc ? `CC: ${cc}\n` : "") +
(preparedAttachments.length > 0
? `Attachments: ${attachmentSummaries.join(", ")}\n`
: "") +
`\nThe email has been sent.`
);
} catch (e) {
this.super.handlerProps.log(
`gmail-send-email error: ${e.message}`
);
this.super.introspect(`Error: ${e.message}`);
return `Error sending email: ${e.message}`;
}
},
});
},
};
},
};