Class: Legion::Extensions::Llm::Chat

Inherits:
Object
  • Object
show all
Includes:
Enumerable, Logging::Helper
Defined in:
lib/legion/extensions/llm/chat.rb

Overview

Represents a conversation with an AI model

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(model: nil, provider: nil, assume_model_exists: false, context: nil) ⇒ Chat

Returns a new instance of Chat.

Raises:

  • (ArgumentError)


13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
# File 'lib/legion/extensions/llm/chat.rb', line 13

def initialize(model: nil, provider: nil, assume_model_exists: false, context: nil)
  raise ArgumentError, 'Provider must be specified if assume_model_exists is true' if assume_model_exists && !provider

  @context = context
  @config = context&.config || Legion::Extensions::Llm.config
  model_id = model || @config.default_model
  with_model(model_id, provider: provider, assume_exists: assume_model_exists)
  @temperature = nil
  @messages = []
  @tools = {}
  @tool_prefs = { choice: nil, calls: nil }
  @params = {}
  @headers = {}
  @schema = nil
  @thinking = nil
  @on = {
    new_message: nil,
    end_message: nil,
    tool_call: nil,
    tool_result: nil
  }
end

Instance Attribute Details

#headersObject (readonly)

Returns the value of attribute headers.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def headers
  @headers
end

#messagesObject (readonly)

Returns the value of attribute messages.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def messages
  @messages
end

#modelObject (readonly)

Returns the value of attribute model.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def model
  @model
end

#paramsObject (readonly)

Returns the value of attribute params.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def params
  @params
end

#schemaObject (readonly)

Returns the value of attribute schema.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def schema
  @schema
end

#tool_prefsObject (readonly)

Returns the value of attribute tool_prefs.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def tool_prefs
  @tool_prefs
end

#toolsObject (readonly)

Returns the value of attribute tools.



11
12
13
# File 'lib/legion/extensions/llm/chat.rb', line 11

def tools
  @tools
end

Instance Method Details

#add_message(message_or_attributes) ⇒ Object



174
175
176
177
178
# File 'lib/legion/extensions/llm/chat.rb', line 174

def add_message(message_or_attributes)
  message = message_or_attributes.is_a?(Message) ? message_or_attributes : Message.new(message_or_attributes)
  messages << message
  message
end

#ask(message = nil, with: nil) ⇒ Object Also known as: say



36
37
38
39
# File 'lib/legion/extensions/llm/chat.rb', line 36

def ask(message = nil, with: nil, &)
  add_message role: :user, content: build_content(message, with)
  complete(&)
end

#completeObject



140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
# File 'lib/legion/extensions/llm/chat.rb', line 140

def complete(&)
  response = @provider.complete(
    messages,
    tools: @tools,
    tool_prefs: @tool_prefs,
    temperature: @temperature,
    model: @model,
    params: @params,
    headers: @headers,
    schema: @schema,
    thinking: @thinking,
    &wrap_streaming_block(&)
  )

  @on[:new_message]&.call unless block_given?

  if @schema && response.content.is_a?(String) && !response.tool_call?
    begin
      response.content = Legion::JSON.parse(response.content, symbolize_names: false)
    rescue Legion::JSON::ParseError => e
      handle_exception(e, level: :warn, handled: true, operation: 'llm.chat.complete')
    end
  end

  add_message response
  @on[:end_message]&.call(response)

  if response.tool_call?
    handle_tool_calls(response, &)
  else
    response
  end
end

#eachObject



136
137
138
# File 'lib/legion/extensions/llm/chat.rb', line 136

def each(&)
  messages.each(&)
end

#instance_variablesObject



184
185
186
# File 'lib/legion/extensions/llm/chat.rb', line 184

def instance_variables
  super - %i[@connection @config]
end

#on_end_message(&block) ⇒ Object



121
122
123
124
# File 'lib/legion/extensions/llm/chat.rb', line 121

def on_end_message(&block)
  @on[:end_message] = block
  self
end

#on_new_message(&block) ⇒ Object



116
117
118
119
# File 'lib/legion/extensions/llm/chat.rb', line 116

def on_new_message(&block)
  @on[:new_message] = block
  self
end

#on_tool_call(&block) ⇒ Object



126
127
128
129
# File 'lib/legion/extensions/llm/chat.rb', line 126

def on_tool_call(&block)
  @on[:tool_call] = block
  self
end

#on_tool_result(&block) ⇒ Object



131
132
133
134
# File 'lib/legion/extensions/llm/chat.rb', line 131

def on_tool_result(&block)
  @on[:tool_result] = block
  self
end

#reset_messages!Object



180
181
182
# File 'lib/legion/extensions/llm/chat.rb', line 180

def reset_messages!
  @messages.clear
end

#with_context(context) ⇒ Object



89
90
91
92
93
94
# File 'lib/legion/extensions/llm/chat.rb', line 89

def with_context(context)
  @context = context
  @config = context.config
  with_model(@model.id, provider: @provider.slug, assume_exists: true)
  self
end

#with_headers(**headers) ⇒ Object



101
102
103
104
# File 'lib/legion/extensions/llm/chat.rb', line 101

def with_headers(**headers)
  @headers = headers
  self
end

#with_instructions(instructions, append: false, replace: nil) ⇒ Object



43
44
45
46
47
48
49
50
51
52
53
# File 'lib/legion/extensions/llm/chat.rb', line 43

def with_instructions(instructions, append: false, replace: nil)
  append ||= (replace == false) unless replace.nil?

  if append
    append_system_instruction(instructions)
  else
    replace_system_instruction(instructions)
  end

  self
end

#with_model(model_id, provider: nil, assume_exists: false) ⇒ Object



71
72
73
74
75
# File 'lib/legion/extensions/llm/chat.rb', line 71

def with_model(model_id, provider: nil, assume_exists: false)
  @model, @provider = Models.resolve(model_id, provider:, assume_exists:, config: @config)
  @connection = @provider.connection
  self
end

#with_params(**params) ⇒ Object



96
97
98
99
# File 'lib/legion/extensions/llm/chat.rb', line 96

def with_params(**params)
  @params = params
  self
end

#with_schema(schema) ⇒ Object



106
107
108
109
110
111
112
113
114
# File 'lib/legion/extensions/llm/chat.rb', line 106

def with_schema(schema)
  schema_instance = schema.is_a?(Class) ? schema.new : schema

  @schema = normalize_schema_payload(
    schema_instance.respond_to?(:to_json_schema) ? schema_instance.to_json_schema : schema_instance
  )

  self
end

#with_temperature(temperature) ⇒ Object



77
78
79
80
# File 'lib/legion/extensions/llm/chat.rb', line 77

def with_temperature(temperature)
  @temperature = temperature
  self
end

#with_thinking(effort: nil, budget: nil) ⇒ Object

Raises:

  • (ArgumentError)


82
83
84
85
86
87
# File 'lib/legion/extensions/llm/chat.rb', line 82

def with_thinking(effort: nil, budget: nil)
  raise ArgumentError, 'with_thinking requires :effort or :budget' if effort.nil? && budget.nil?

  @thinking = Thinking::Config.new(effort: effort, budget: budget)
  self
end

#with_tool(tool, choice: nil, calls: nil) ⇒ Object



55
56
57
58
59
60
61
62
# File 'lib/legion/extensions/llm/chat.rb', line 55

def with_tool(tool, choice: nil, calls: nil)
  unless tool.nil?
    tool_instance = tool.is_a?(Class) ? tool.new : tool
    @tools[tool_instance.name.to_sym] = tool_instance
  end
  update_tool_options(choice:, calls:)
  self
end

#with_tools(*tools, replace: false, choice: nil, calls: nil) ⇒ Object



64
65
66
67
68
69
# File 'lib/legion/extensions/llm/chat.rb', line 64

def with_tools(*tools, replace: false, choice: nil, calls: nil)
  @tools.clear if replace
  tools.compact.each { |tool| with_tool tool }
  update_tool_options(choice:, calls:)
  self
end