discourse-ai/spec/lib/completions/dialects/nova_spec.rb

191 lines
6.0 KiB
Ruby

# frozen_string_literal: true
RSpec.describe DiscourseAi::Completions::Dialects::Nova do
fab!(:llm_model) { Fabricate(:nova_model, vision_enabled: true) }
let(:nova_dialect_klass) { DiscourseAi::Completions::Dialects::Dialect.dialect_for(llm_model) }
it "finds the right dialect" do
expect(nova_dialect_klass).to eq(DiscourseAi::Completions::Dialects::Nova)
end
describe "#translate" do
it "properly formats a basic conversation" do
messages = [
{ type: :user, id: "user1", content: "Hello" },
{ type: :model, content: "Hi there!" },
]
prompt = DiscourseAi::Completions::Prompt.new("You are a helpful bot", messages: messages)
dialect = nova_dialect_klass.new(prompt, llm_model)
translated = dialect.translate
expect(translated.system).to eq([{ text: "You are a helpful bot" }])
expect(translated.messages).to eq(
[
{ role: "user", content: [{ text: "Hello" }] },
{ role: "assistant", content: [{ text: "Hi there!" }] },
],
)
end
context "with image content" do
let(:image100x100) { plugin_file_from_fixtures("100x100.jpg") }
let(:upload) do
UploadCreator.new(image100x100, "image.jpg").create_for(Discourse.system_user.id)
end
it "properly formats messages with images" do
messages = [
{ type: :user, id: "user1", content: "What's in this image?", upload_ids: [upload.id] },
]
prompt = DiscourseAi::Completions::Prompt.new(messages: messages)
dialect = nova_dialect_klass.new(prompt, llm_model)
translated = dialect.translate
encoded = prompt.encoded_uploads(messages.first).first[:base64]
expect(translated.messages.first[:content]).to eq(
[
{ text: "What's in this image?" },
{ image: { format: "jpeg", source: { bytes: encoded } } },
],
)
end
end
context "with tools" do
it "properly formats tool configuration" do
tools = [
{
name: "get_weather",
description: "Get the weather in a city",
parameters: [
{ name: "location", type: "string", description: "the city name", required: true },
],
},
]
messages = [{ type: :user, content: "What's the weather?" }]
prompt =
DiscourseAi::Completions::Prompt.new(
"You are a helpful bot",
messages: messages,
tools: tools,
)
dialect = nova_dialect_klass.new(prompt, llm_model)
translated = dialect.translate
expect(translated.tool_config).to eq(
{
tools: [
{
toolSpec: {
name: "get_weather",
description: "Get the weather in a city",
inputSchema: {
json: {
type: "object",
properties: {
"location" => {
type: "string",
required: true,
},
},
},
},
},
},
],
},
)
end
end
context "with inference configuration" do
it "includes inference configuration when provided" do
messages = [{ type: :user, content: "Hello" }]
prompt = DiscourseAi::Completions::Prompt.new("You are a helpful bot", messages: messages)
dialect = nova_dialect_klass.new(prompt, llm_model)
options = { temperature: 0.7, top_p: 0.9, max_tokens: 100, stop_sequences: ["STOP"] }
translated = dialect.translate
expected = {
system: [{ text: "You are a helpful bot" }],
messages: [{ role: "user", content: [{ text: "Hello" }] }],
inferenceConfig: {
temperature: 0.7,
top_p: 0.9,
stopSequences: ["STOP"],
max_new_tokens: 100,
},
}
expect(translated.to_payload(options)).to eq(expected)
end
it "omits inference configuration when not provided" do
messages = [{ type: :user, content: "Hello" }]
prompt = DiscourseAi::Completions::Prompt.new("You are a helpful bot", messages: messages)
dialect = nova_dialect_klass.new(prompt, llm_model)
translated = dialect.translate
expect(translated.inference_config).to be_nil
end
end
it "handles tool calls and responses" do
tool_call_prompt = { name: "get_weather", arguments: { location: "London" } }
messages = [
{ type: :user, id: "user1", content: "What's the weather in London?" },
{ type: :tool_call, name: "get_weather", id: "tool_id", content: tool_call_prompt.to_json },
{ type: :tool, id: "tool_id", content: "Sunny, 22°C".to_json },
{ type: :model, content: "The weather in London is sunny with 22°C" },
]
prompt =
DiscourseAi::Completions::Prompt.new(
"You are a helpful bot",
messages: messages,
tools: [
{
name: "get_weather",
description: "Get the weather in a city",
parameters: [
{ name: "location", type: "string", description: "the city name", required: true },
],
},
],
)
dialect = nova_dialect_klass.new(prompt, llm_model)
translated = dialect.translate
expect(translated.messages.map { |m| m[:role] }).to eq(%w[user assistant user assistant])
expect(translated.messages.last[:content]).to eq(
[{ text: "The weather in London is sunny with 22°C" }],
)
end
end
describe "#max_prompt_tokens" do
it "returns the model's max prompt tokens" do
prompt = DiscourseAi::Completions::Prompt.new("You are a helpful bot")
dialect = nova_dialect_klass.new(prompt, llm_model)
expect(dialect.max_prompt_tokens).to eq(llm_model.max_prompt_tokens)
end
end
end