summaryrefslogtreecommitdiff
path: root/components/providers
diff options
context:
space:
mode:
authoricebaker <icebaker@proton.me>2023-11-18 19:07:10 -0300
committericebaker <icebaker@proton.me>2023-11-18 19:07:10 -0300
commit8ae78b954350755a47a13133668dba93bac15f37 (patch)
tree9cdc3bb770d778bd8d00675fdbc1f27a6e27e37c /components/providers
parentab22d1bbe37093912cb7418b3c945153a15f4255 (diff)
adding support for tools
Diffstat (limited to 'components/providers')
-rw-r--r--components/providers/openai.rb100
-rw-r--r--components/providers/openai/tools.rb73
2 files changed, 163 insertions, 10 deletions
diff --git a/components/providers/openai.rb b/components/providers/openai.rb
index ce6fb33..437114c 100644
--- a/components/providers/openai.rb
+++ b/components/providers/openai.rb
@@ -2,9 +2,14 @@
require 'openai'
-require_relative './base'
+require_relative 'base'
require_relative '../crypto'
+require_relative '../../logic/providers/openai/tools'
+require_relative '../../controllers/interfaces/tools'
+
+require_relative 'openai/tools'
+
module NanoBot
module Components
module Providers
@@ -34,15 +39,23 @@ module NanoBot
def stream(input)
provider = @settings.key?(:stream) ? @settings[:stream] : true
+
+ # TODO: There's a bug here...
interface = input[:interface].key?(:stream) ? input[:interface][:stream] : true
provider && interface
end
- def evaluate(input, &block)
+ def evaluate(input, &feedback)
messages = input[:history].map do |event|
- { role: event[:who] == 'user' ? 'user' : 'assistant',
- content: event[:message] }
+ if event[:message].nil? && event[:meta] && event[:meta][:tool_calls]
+ { role: 'assistant', content: nil, tool_calls: event[:meta][:tool_calls] }
+ elsif event[:who] == 'tool'
+ { role: event[:who], content: event[:message],
+ tool_call_id: event[:meta][:id], name: event[:meta][:name] }
+ else
+ { role: event[:who] == 'user' ? 'user' : 'assistant', content: event[:message] }
+ end
end
%i[instruction backdrop directive].each do |key|
@@ -62,17 +75,65 @@ module NanoBot
payload.delete(:logit_bias) if payload.key?(:logit_bias) && payload[:logit_bias].nil?
+ payload[:tools] = input[:tools].map { |raw| NanoBot::Logic::OpenAI::Tools.adapt(raw) } if input[:tools]
+
if stream(input)
content = ''
+ tools = []
payload[:stream] = proc do |chunk, _bytesize|
- partial = chunk.dig('choices', 0, 'delta', 'content')
- if partial
- content += partial
- block.call({ who: 'AI', message: partial }, false)
+ partial_content = chunk.dig('choices', 0, 'delta', 'content')
+ partial_tools = chunk.dig('choices', 0, 'delta', 'tool_calls')
+
+ if partial_tools
+ partial_tools.each do |partial_tool|
+ tools[partial_tool['index']] = {} if tools[partial_tool['index']].nil?
+
+ partial_tool.keys.reject { |key| ['index'].include?(key) }.each do |key|
+ target = tools[partial_tool['index']]
+
+ if partial_tool[key].is_a?(Hash)
+ target[key] = {} if target[key].nil?
+ partial_tool[key].each_key do |sub_key|
+ target[key][sub_key] = '' if target[key][sub_key].nil?
+
+ target[key][sub_key] += partial_tool[key][sub_key]
+ end
+ else
+ target[key] = '' if target[key].nil?
+
+ target[key] += partial_tool[key]
+ end
+ end
+ end
+ end
+
+ if partial_content
+ content += partial_content
+ feedback.call(
+ { should_be_stored: false,
+ interaction: { who: 'AI', message: partial_content } }
+ )
end
- block.call({ who: 'AI', message: content }, true) if chunk.dig('choices', 0, 'finish_reason')
+ if chunk.dig('choices', 0, 'finish_reason')
+ if tools&.size&.positive?
+ feedback.call(
+ { should_be_stored: true,
+ needs_another_round: true,
+ interaction: { who: 'AI', message: nil, meta: { tool_calls: tools } } }
+ )
+ Tools.apply(input[:tools], tools, feedback).each do |interaction|
+ feedback.call({ should_be_stored: true, needs_another_round: true, interaction: })
+ end
+ end
+
+ feedback.call(
+ { should_be_stored: !(content.nil? || content == ''),
+ interaction: content.nil? || content == '' ? nil : { who: 'AI', message: content },
+ finished: true }
+ )
+ end
end
@client.chat(parameters: payload)
@@ -81,7 +142,26 @@ module NanoBot
raise StandardError, result['error'] if result['error']
- block.call({ who: 'AI', message: result.dig('choices', 0, 'message', 'content') }, true)
+ tools = result.dig('choices', 0, 'message', 'tool_calls')
+
+ if tools&.size&.positive?
+ feedback.call(
+ { should_be_stored: true,
+ needs_another_round: true,
+ interaction: { who: 'AI', message: nil, meta: { tool_calls: tools } } }
+ )
+ Tools.apply(input[:tools], tools, feedback).each do |interaction|
+ feedback.call({ should_be_stored: true, needs_another_round: true, interaction: })
+ end
+ end
+
+ content = result.dig('choices', 0, 'message', 'content')
+
+ feedback.call(
+ { should_be_stored: !(content.nil? || content == ''),
+ interaction: content.nil? || content == '' ? nil : { who: 'AI', message: content },
+ finished: true }
+ )
end
end
diff --git a/components/providers/openai/tools.rb b/components/providers/openai/tools.rb
new file mode 100644
index 0000000..ea34ae6
--- /dev/null
+++ b/components/providers/openai/tools.rb
@@ -0,0 +1,73 @@
+# frozen_string_literal: true
+
+require_relative '../../embedding'
+
+require 'concurrent'
+
+module NanoBot
+ module Components
+ module Providers
+ class OpenAI < Base
+ module Tools
+ def self.apply(cartridge, tools, feedback)
+ prepared_tools = NanoBot::Logic::OpenAI::Tools.prepare(cartridge, tools)
+
+ futures = prepared_tools.map do |tool|
+ Concurrent::Promises.future { process!(tool, feedback) }
+ end
+
+ results = Concurrent::Promises.zip(*futures).value!
+
+ results.map do |applied_tool|
+ {
+ who: 'tool',
+ message: applied_tool[:output],
+ meta: { id: applied_tool[:id], name: applied_tool[:name] }
+ }
+ end
+ end
+
+ def self.process!(tool, feedback)
+ feedback.call(
+ { should_be_stored: false,
+ interaction: { who: 'AI', message: nil, meta: {
+ tool: { action: 'call', id: tool[:id], name: tool[:name], parameters: tool[:parameters] }
+ } } }
+ )
+
+ call = { parameters: %w[parameters], values: [tool[:parameters]], safety: false }
+
+ if %i[fennel lua clojure].count { |key| !tool[:source][key].nil? } > 1
+ raise StandardError, 'conflicting tools'
+ end
+
+ if !tool[:source][:fennel].nil?
+ call[:source] = tool[:source][:fennel]
+ tool[:output] = Components::Embedding.fennel(**call)
+ elsif !tool[:source][:clojure].nil?
+ call[:source] = tool[:source][:clojure]
+ tool[:output] = Components::Embedding.clojure(**call)
+ elsif !tool[:source][:lua].nil?
+ call[:source] = tool[:source][:lua]
+ tool[:output] = Components::Embedding.lua(**call)
+ else
+ raise 'missing source code'
+ end
+
+ feedback.call(
+ { should_be_stored: false,
+ interaction: { who: 'AI', message: nil, meta: {
+ tool: {
+ action: 'response', id: tool[:id], name: tool[:name],
+ parameters: tool[:parameters], output: tool[:output]
+ }
+ } } }
+ )
+
+ tool
+ end
+ end
+ end
+ end
+ end
+end