350 lines
11 KiB
Ruby
350 lines
11 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
class AiTool < ActiveRecord::Base
|
|
validates :name, presence: true, length: { maximum: 100 }
|
|
validates :description, presence: true, length: { maximum: 1000 }
|
|
validates :summary, presence: true, length: { maximum: 255 }
|
|
validates :script, presence: true, length: { maximum: 100_000 }
|
|
validates :created_by_id, presence: true
|
|
belongs_to :created_by, class_name: "User"
|
|
has_many :rag_document_fragments, dependent: :destroy, as: :target
|
|
has_many :upload_references, as: :target, dependent: :destroy
|
|
has_many :uploads, through: :upload_references
|
|
before_update :regenerate_rag_fragments
|
|
|
|
def signature
|
|
{ name: name, description: description, parameters: parameters.map(&:symbolize_keys) }
|
|
end
|
|
|
|
def runner(parameters, llm:, bot_user:, context: {})
|
|
DiscourseAi::AiBot::ToolRunner.new(
|
|
parameters: parameters,
|
|
llm: llm,
|
|
bot_user: bot_user,
|
|
context: context,
|
|
tool: self,
|
|
)
|
|
end
|
|
|
|
after_commit :bump_persona_cache
|
|
|
|
def bump_persona_cache
|
|
AiPersona.persona_cache.flush!
|
|
end
|
|
|
|
def regenerate_rag_fragments
|
|
if rag_chunk_tokens_changed? || rag_chunk_overlap_tokens_changed?
|
|
RagDocumentFragment.where(target: self).delete_all
|
|
end
|
|
end
|
|
|
|
def self.preamble
|
|
<<~JS
|
|
/**
|
|
* Tool API Quick Reference
|
|
*
|
|
* Entry Functions
|
|
*
|
|
* invoke(parameters): Main function. Receives parameters (Object). Must return a JSON-serializable value.
|
|
* Example:
|
|
* function invoke(parameters) { return "result"; }
|
|
*
|
|
* details(): Optional. Returns a string describing the tool.
|
|
* Example:
|
|
* function details() { return "Tool description."; }
|
|
*
|
|
* Provided Objects
|
|
*
|
|
* 1. http
|
|
* http.get(url, options?): Performs an HTTP GET request.
|
|
* Parameters:
|
|
* url (string): The request URL.
|
|
* options (Object, optional):
|
|
* headers (Object): Request headers.
|
|
* Returns:
|
|
* { status: number, body: string }
|
|
*
|
|
* http.post(url, options?): Performs an HTTP POST request.
|
|
* Parameters:
|
|
* url (string): The request URL.
|
|
* options (Object, optional):
|
|
* headers (Object): Request headers.
|
|
* body (string): Request body.
|
|
* Returns:
|
|
* { status: number, body: string }
|
|
*
|
|
* Note: Max 20 HTTP requests per execution.
|
|
*
|
|
* 2. llm
|
|
* llm.truncate(text, length): Truncates text to a specified token length.
|
|
* Parameters:
|
|
* text (string): Text to truncate.
|
|
* length (number): Max tokens.
|
|
* Returns:
|
|
* Truncated string.
|
|
*
|
|
* 3. index
|
|
* index.search(query, options?): Searches indexed documents.
|
|
* Parameters:
|
|
* query (string): Search query.
|
|
* options (Object, optional):
|
|
* filenames (Array): Limit search to specific files.
|
|
* limit (number): Max fragments (up to 200).
|
|
* Returns:
|
|
* Array of { fragment: string, metadata: string }
|
|
*
|
|
* 4. upload
|
|
* upload.create(filename, base_64_content): Uploads a file.
|
|
* Parameters:
|
|
* filename (string): Name of the file.
|
|
* base_64_content (string): Base64 encoded file content.
|
|
* Returns:
|
|
* { id: number, short_url: string }
|
|
*
|
|
* 5. chain
|
|
* chain.setCustomRaw(raw): Sets the body of the post and exist chain.
|
|
* Parameters:
|
|
* raw (string): raw content to add to post.
|
|
*
|
|
* Constraints
|
|
*
|
|
* Execution Time: ≤ 2000ms
|
|
* Memory: ≤ 10MB
|
|
* HTTP Requests: ≤ 20 per execution
|
|
* Exceeding limits will result in errors or termination.
|
|
*
|
|
* Security
|
|
*
|
|
* Sandboxed Environment: No access to system or global objects.
|
|
* No File System Access: Cannot read or write files.
|
|
*/
|
|
JS
|
|
end
|
|
|
|
def self.presets
|
|
[
|
|
{
|
|
preset_id: "browse_web_jina",
|
|
name: "browse_web",
|
|
description: "Browse the web as a markdown document",
|
|
parameters: [
|
|
{ name: "url", type: "string", required: true, description: "The URL to browse" },
|
|
],
|
|
script: <<~SCRIPT,
|
|
#{preamble}
|
|
let url;
|
|
function invoke(p) {
|
|
url = p.url;
|
|
result = http.get(`https://r.jina.ai/${url}`);
|
|
// truncates to 15000 tokens
|
|
return llm.truncate(result.body, 15000);
|
|
}
|
|
function details() {
|
|
return "Read: " + url
|
|
}
|
|
SCRIPT
|
|
},
|
|
{
|
|
preset_id: "exchange_rate",
|
|
name: "exchange_rate",
|
|
description: "Get current exchange rates for various currencies",
|
|
parameters: [
|
|
{
|
|
name: "base_currency",
|
|
type: "string",
|
|
required: true,
|
|
description: "The base currency code (e.g., USD, EUR)",
|
|
},
|
|
{
|
|
name: "target_currency",
|
|
type: "string",
|
|
required: true,
|
|
description: "The target currency code (e.g., EUR, JPY)",
|
|
},
|
|
{ name: "amount", type: "number", description: "Amount to convert eg: 123.45" },
|
|
],
|
|
script: <<~SCRIPT,
|
|
#{preamble}
|
|
// note: this script uses the open.er-api.com service, it is only updated
|
|
// once every 24 hours, for more up to date rates see: https://www.exchangerate-api.com
|
|
function invoke(params) {
|
|
const url = `https://open.er-api.com/v6/latest/${params.base_currency}`;
|
|
const result = http.get(url);
|
|
if (result.status !== 200) {
|
|
return { error: "Failed to fetch exchange rates" };
|
|
}
|
|
const data = JSON.parse(result.body);
|
|
const rate = data.rates[params.target_currency];
|
|
if (!rate) {
|
|
return { error: "Target currency not found" };
|
|
}
|
|
|
|
const rval = {
|
|
base_currency: params.base_currency,
|
|
target_currency: params.target_currency,
|
|
exchange_rate: rate,
|
|
last_updated: data.time_last_update_utc
|
|
};
|
|
|
|
if (params.amount) {
|
|
rval.original_amount = params.amount;
|
|
rval.converted_amount = params.amount * rate;
|
|
}
|
|
|
|
return rval;
|
|
}
|
|
|
|
function details() {
|
|
return "<a href='https://www.exchangerate-api.com'>Rates By Exchange Rate API</a>";
|
|
}
|
|
SCRIPT
|
|
summary: "Get current exchange rates between two currencies",
|
|
},
|
|
{
|
|
preset_id: "stock_quote",
|
|
name: "stock_quote",
|
|
description: "Get real-time stock quote information using AlphaVantage API",
|
|
parameters: [
|
|
{
|
|
name: "symbol",
|
|
type: "string",
|
|
required: true,
|
|
description: "The stock symbol (e.g., AAPL, GOOGL)",
|
|
},
|
|
],
|
|
script: <<~SCRIPT,
|
|
#{preamble}
|
|
function invoke(params) {
|
|
const apiKey = 'YOUR_ALPHAVANTAGE_API_KEY'; // Replace with your actual API key
|
|
const url = `https://www.alphavantage.co/query?function=GLOBAL_QUOTE&symbol=${params.symbol}&apikey=${apiKey}`;
|
|
|
|
const result = http.get(url);
|
|
if (result.status !== 200) {
|
|
return { error: "Failed to fetch stock quote" };
|
|
}
|
|
|
|
const data = JSON.parse(result.body);
|
|
if (data['Error Message']) {
|
|
return { error: data['Error Message'] };
|
|
}
|
|
|
|
const quote = data['Global Quote'];
|
|
if (!quote || Object.keys(quote).length === 0) {
|
|
return { error: "No data found for the given symbol" };
|
|
}
|
|
|
|
return {
|
|
symbol: quote['01. symbol'],
|
|
price: parseFloat(quote['05. price']),
|
|
change: parseFloat(quote['09. change']),
|
|
change_percent: quote['10. change percent'],
|
|
volume: parseInt(quote['06. volume']),
|
|
latest_trading_day: quote['07. latest trading day']
|
|
};
|
|
}
|
|
|
|
function details() {
|
|
return "<a href='https://www.alphavantage.co'>Stock data provided by AlphaVantage</a>";
|
|
}
|
|
SCRIPT
|
|
summary: "Get real-time stock quotes using AlphaVantage API",
|
|
},
|
|
{
|
|
preset_id: "image_generation",
|
|
name: "image_generation",
|
|
description:
|
|
"Generate images using the FLUX model from Black Forest Labs using together.ai",
|
|
parameters: [
|
|
{
|
|
name: "prompt",
|
|
type: "string",
|
|
required: true,
|
|
description: "The text prompt for image generation",
|
|
},
|
|
{
|
|
name: "seed",
|
|
type: "number",
|
|
required: false,
|
|
description: "Optional seed for random number generation",
|
|
},
|
|
],
|
|
script: <<~SCRIPT,
|
|
#{preamble}
|
|
const apiKey = "YOUR_KEY";
|
|
const model = "black-forest-labs/FLUX.1.1-pro";
|
|
|
|
function invoke(params) {
|
|
let seed = parseInt(params.seed);
|
|
if (!(seed > 0)) {
|
|
seed = Math.floor(Math.random() * 1000000) + 1;
|
|
}
|
|
|
|
const prompt = params.prompt;
|
|
const body = {
|
|
model: model,
|
|
prompt: prompt,
|
|
width: 1024,
|
|
height: 768,
|
|
steps: 10,
|
|
n: 1,
|
|
seed: seed,
|
|
response_format: "b64_json",
|
|
};
|
|
|
|
const result = http.post("https://api.together.xyz/v1/images/generations", {
|
|
headers: {
|
|
"Authorization": `Bearer ${apiKey}`,
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
const base64Image = JSON.parse(result.body).data[0].b64_json;
|
|
const image = upload.create("generated_image.png", base64Image);
|
|
const raw = `\n![${prompt}](${image.short_url})\n`;
|
|
chain.setCustomRaw(raw);
|
|
|
|
return { result: "Image generated successfully", seed: seed };
|
|
}
|
|
|
|
function details() {
|
|
return "Generates images based on a text prompt using the FLUX model.";
|
|
}
|
|
SCRIPT
|
|
summary: "Generate image",
|
|
},
|
|
{ preset_id: "empty_tool", script: <<~SCRIPT },
|
|
#{preamble}
|
|
function invoke(params) {
|
|
// logic here
|
|
return params;
|
|
}
|
|
function details() {
|
|
return "Details about this tool";
|
|
}
|
|
SCRIPT
|
|
].map do |preset|
|
|
preset[:preset_name] = I18n.t("discourse_ai.tools.presets.#{preset[:preset_id]}.name")
|
|
preset
|
|
end
|
|
end
|
|
end
|
|
|
|
# == Schema Information
|
|
#
|
|
# Table name: ai_tools
|
|
#
|
|
# id :bigint not null, primary key
|
|
# name :string not null
|
|
# description :string not null
|
|
# summary :string not null
|
|
# parameters :jsonb not null
|
|
# script :text not null
|
|
# created_by_id :integer not null
|
|
# enabled :boolean default(TRUE), not null
|
|
# created_at :datetime not null
|
|
# updated_at :datetime not null
|
|
# rag_chunk_tokens :integer default(374), not null
|
|
# rag_chunk_overlap_tokens :integer default(10), not null
|
|
#
|