Class: RubyLLM::Chat

Inherits:
Object
  • Object
show all
Includes:
Enumerable
Defined in:
lib/ruby_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.



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

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

  @context = context
  @config = context&.config || RubyLLM.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
  }
  @callbacks = Hash.new { |callbacks, name| callbacks[name] = [] }
end

Instance Attribute Details

#headersObject (readonly)

Returns the value of attribute headers.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def headers
  @headers
end

#messagesObject (readonly)

Returns the value of attribute messages.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def messages
  @messages
end

#modelObject (readonly)

Returns the value of attribute model.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def model
  @model
end

#paramsObject (readonly)

Returns the value of attribute params.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def params
  @params
end

#schemaObject (readonly)

Returns the value of attribute schema.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def schema
  @schema
end

#tool_prefsObject (readonly)

Returns the value of attribute tool_prefs.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def tool_prefs
  @tool_prefs
end

#toolsObject (readonly)

Returns the value of attribute tools.



8
9
10
# File 'lib/ruby_llm/chat.rb', line 8

def tools
  @tools
end

Instance Method Details

#add_message(message_or_attributes) ⇒ Object



190
191
192
193
194
# File 'lib/ruby_llm/chat.rb', line 190

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

#after_messageObject



136
137
138
# File 'lib/ruby_llm/chat.rb', line 136

def after_message(&)
  add_callback(:after_message, &)
end

#after_tool_resultObject



144
145
146
# File 'lib/ruby_llm/chat.rb', line 144

def after_tool_result(&)
  add_callback(:after_tool_result, &)
end

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



36
37
38
39
# File 'lib/ruby_llm/chat.rb', line 36

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

#before_messageObject



132
133
134
# File 'lib/ruby_llm/chat.rb', line 132

def before_message(&)
  add_callback(:before_message, &)
end

#before_tool_callObject



140
141
142
# File 'lib/ruby_llm/chat.rb', line 140

def before_tool_call(&)
  add_callback(:before_tool_call, &)
end

#completeObject



156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
# File 'lib/ruby_llm/chat.rb', line 156

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(&)
  )

  run_callbacks(:before_message, :new_message) unless block_given?

  if @schema && response.content.is_a?(String) && !response.tool_call?
    begin
      response.content = JSON.parse(response.content)
    rescue JSON::ParserError
      # If parsing fails, keep content as string
    end
  end

  add_message response
  run_callbacks(:after_message, :end_message, response)

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

#costObject



152
153
154
# File 'lib/ruby_llm/chat.rb', line 152

def cost
  Cost.aggregate(messages.map(&:cost))
end

#eachObject



148
149
150
# File 'lib/ruby_llm/chat.rb', line 148

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

#instance_variablesObject



200
201
202
# File 'lib/ruby_llm/chat.rb', line 200

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

#on_end_messageObject



120
121
122
# File 'lib/ruby_llm/chat.rb', line 120

def on_end_message(&)
  set_legacy_callback(:end_message, :on_end_message, :after_message, &)
end

#on_new_messageObject



116
117
118
# File 'lib/ruby_llm/chat.rb', line 116

def on_new_message(&)
  set_legacy_callback(:new_message, :on_new_message, :before_message, &)
end

#on_tool_callObject



124
125
126
# File 'lib/ruby_llm/chat.rb', line 124

def on_tool_call(&)
  set_legacy_callback(:tool_call, :on_tool_call, :before_tool_call, &)
end

#on_tool_resultObject



128
129
130
# File 'lib/ruby_llm/chat.rb', line 128

def on_tool_result(&)
  set_legacy_callback(:tool_result, :on_tool_result, :after_tool_result, &)
end

#reset_messages!Object



196
197
198
# File 'lib/ruby_llm/chat.rb', line 196

def reset_messages!
  @messages.clear
end

#with_context(context) ⇒ Object



89
90
91
92
93
94
# File 'lib/ruby_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/ruby_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/ruby_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/ruby_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/ruby_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/ruby_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/ruby_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/ruby_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/ruby_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/ruby_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