omniai 2.0.0 → 2.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 116095cbb73cb4e826fe28a4c644e37ea1ecb887d601b0ca4cbdc3a70e0ca81e
4
- data.tar.gz: d17b5078d80e7cf6858b0e77e22d5949c157841672aa5bbe54a2782e8e10df31
3
+ metadata.gz: c58b1fbebfaaaa7bcd77fe9754d0bf0bddc7b55c07b76ae31cbae0ae24b2024f
4
+ data.tar.gz: d4462dbf00a04ce81db7f8bf7c4f45018aaab994fd2699483e8b2de324546ab2
5
5
  SHA512:
6
- metadata.gz: 5b3a610780db624592fab65b2437ac552de7dd9f7d5c4f6ec62aff546720db832a40f4a7e24e68d64f42ce84dff24ed73a3105a08c37c61f00faa83d65fd55ca
7
- data.tar.gz: 5000a20feca11ee1e29b03d71885d4e75aa6792a39ac828ae78b3fcae5783430f65fe93b209350aa3399dfc7cce1ccbadc0a844518e0ccb772d0b8704f304cc3
6
+ metadata.gz: b7857bee4ec80f63d8e2a8ae3a6534c36f21ae622ba3a672653ae6aa2017e929baa2d4f89d9bcd783ab645bf17c9e65c6dfa8fa18fc757a38106d59375305bb8
7
+ data.tar.gz: d0c5f8a0ab434740df1a6abc4f5f9d96590d067fe5d6e4e1a9849edea9647bcfa4c5768fdfe74c14a520ff6a9849e8b51f56b764ca8f28424bdb614a87d47f0e
data/README.md CHANGED
@@ -16,124 +16,161 @@ OmniAI provides a unified Ruby API for integrating with multiple AI providers, i
16
16
 
17
17
  ## Examples
18
18
 
19
- ### Example #1: [Chat](https://github.com/ksylvest/omniai/blob/main/examples/chat)
19
+ ### Example #1: [Chat w/ Text](https://github.com/ksylvest/omniai/blob/main/examples/chat_with_text)
20
20
 
21
- This example demonstrates using `OmniAI` with **Anthropic** to prompt a “biologist” for an analysis of photos, identifying the animals within each one. A system and user message are provided, and the response is streamed in real time.
21
+ This example demonstrates using `OmniAI` with **Anthropic** to ask for a joke. The response is parsed and printed.
22
22
 
23
23
  ```ruby
24
24
  require 'omniai/anthropic'
25
25
 
26
- CLIENT = OmniAI::Anthropic::Client.new
27
-
28
- CAT_URL = 'https://images.unsplash.com/photo-1472491235688-bdc81a63246e?q=80&w=1024&h=1024&fit=crop&fm=jpg'
29
- DOG_URL = 'https://images.unsplash.com/photo-1517849845537-4d257902454a?q=80&w=1024&h=1024&fit=crop&fm=jpg'
26
+ client = OmniAI::Anthropic::Client.new
30
27
 
31
- CLIENT.chat(stream: $stdout) do |prompt|
32
- prompt.system('You are a helpful biologist with an expertise in animals that responds with the latin names.')
33
- prompt.user do |message|
34
- message.text('What animals are in the attached photos?')
35
- message.url(CAT_URL, 'image/jpeg')
36
- message.url(DOG_URL, 'image/jpeg')
37
- end
38
- end
28
+ puts client.chat("Tell me a joke").text
39
29
  ```
40
30
 
41
31
  ```
42
- The animals in the photos are:
43
-
44
- 1. A cat (*Felis catus*).
45
- 2. A dog (*Canis familiaris*).
32
+ Why don't scientists trust atoms? Because they make up everything!
46
33
  ```
47
34
 
48
- ### Example #2: [Text-to-Speech](https://github.com/ksylvest/omniai/blob/main/examples/text_to_speech)
35
+ ### Example #2: [Chat w/ Prompt](https://github.com/ksylvest/omniai/blob/main/examples/chat_with_prompt)
49
36
 
50
- This example demonstrates using `OmniAI` with **OpenAI** to convert text to speech and save it to a file.
37
+ This example demonstrates using `OmniAI` with **Mistral** to ask for the fastest animal. It includes a system and user message in the prompt. The response is streamed in real time.
51
38
 
52
39
  ```ruby
53
- require 'omniai/openai'
40
+ require "omniai/mistral"
54
41
 
55
- CLIENT = OmniAI::OpenAI::Client.new
42
+ client = OmniAI::Mistral::Client.new
56
43
 
57
- File.open(File.join(__dir__, 'audio.wav'), 'wb') do |file|
58
- CLIENT.speak('Sally sells seashells by the seashore.', format: OmniAI::Speak::Format::WAV) do |chunk|
59
- file << chunk
60
- end
44
+ client.chat(stream: $stdout) do |prompt|
45
+ prompt.system "Respond in both English and French."
46
+ prompt.user "What is the fastest animal?"
61
47
  end
62
48
  ```
63
49
 
64
- ### Example #3: [Speech-to-Text](https://github.com/ksylvest/omniai/blob/main/examples/speech_to_text)
50
+ ```
51
+ **English**: The peregrine falcon is generally considered the fastest animal, reaching speeds of over 390 km/h.
52
+ **French**: Le faucon pèlerin est généralement considéré comme l'animal le plus rapide, atteignant des vitesses de plus de 390 km/h.
53
+ ```
65
54
 
66
- This example demonstrates using `OmniAI` with **OpenAI** to convert speech to text.
55
+ ### Example #3: [Chat w/ Vision](https://github.com/ksylvest/omniai/blob/main/examples/chat_with_vision)
56
+
57
+ This example demonstrates using `OmniAI` with **OpenAI** to prompt a “biologist” for an analysis of photos, identifying the animals within each one. A system and user message are provided, and the response is streamed in real time.
67
58
 
68
59
  ```ruby
69
- require 'omniai/openai'
60
+ require "omniai/openai"
70
61
 
71
- CLIENT = OmniAI::OpenAI::Client.new
62
+ client = OmniAI::OpenAI::Client.new
72
63
 
73
- File.open(File.join(__dir__, 'audio.wav'), 'rb') do |file|
74
- transcription = CLIENT.transcribe(file)
75
- puts(transcription.text)
64
+ CAT_URL = "https://images.unsplash.com/photo-1472491235688-bdc81a63246e?q=80&w=1024&h=1024&fit=crop&fm=jpg"
65
+ DOG_URL = "https://images.unsplash.com/photo-1517849845537-4d257902454a?q=80&w=1024&h=1024&fit=crop&fm=jpg"
66
+
67
+ client.chat(stream: $stdout) do |prompt|
68
+ prompt.system("You are a helpful biologist with expertise in animals who responds with the Latin names.")
69
+ prompt.user do |message|
70
+ message.text("What animals are in the attached photos?")
71
+ message.url(CAT_URL, "image/jpeg")
72
+ message.url(DOG_URL, "image/jpeg")
73
+ end
76
74
  end
77
75
  ```
78
76
 
79
- ### Example #4: [Tools](https://github.com/ksylvest/omniai/blob/main/examples/tools)
77
+ ```
78
+ The first photo is of a cat, *Felis Catus*.
79
+ The second photo is of a dog, *Canis Familiaris*.
80
+ ```
81
+
82
+ ### Example #4: [Chat w/ Tools](https://github.com/ksylvest/omniai/blob/main/examples/chat_with_tools)
80
83
 
81
- This example demonstrates how to use `OmniAI` with **Google** to call a custom tool that generates a weather report. The tool accepts a list of locations (city and country) and returns a temperature randomly generated for demonstration purposes based on the provided parameters.
84
+ This example demonstrates using `OmniAI` with **Google** to ask for the weather. A tool “Weather” is provided. The tool accepts a location and unit (Celsius or Fahrenheit) then calculates the weather. The LLM makes multiple tool-call requests and is automatically provided with a tool-call response prior to streaming in real-time the result.
82
85
 
83
86
  ```ruby
84
87
  require 'omniai/google'
85
88
 
86
- CLIENT = OmniAI::Google::Client.new
89
+ client = OmniAI::Google::Client.new
87
90
 
88
- LOCATION = OmniAI::Tool::Property.object(
89
- properties: {
90
- city: OmniAI::Tool::Property.string(description: 'e.g. "Toronto"'),
91
- country: OmniAI::Tool::Property.string(description: 'e.g. "Canada"'),
92
- },
93
- required: %i[city country]
94
- )
91
+ class Weather < OmniAI::Tool
92
+ description "Lookup the weather for a location"
95
93
 
96
- LOCATIONS = OmniAI::Tool::Property.array(
97
- min_items: 1,
98
- max_items: 5,
99
- items: LOCATION
100
- )
94
+ parameter :location, :string, description: "A location (e.g. 'Toronto, Canada')."
95
+ parameter :unit, :string, enum: %w[Celsius Fahrenheit], description: "The unit of measurement."
96
+ required %i[location]
101
97
 
102
- UNIT = OmniAI::Tool::Property.string(enum: %w[celcius fahrenheit])
98
+ # @param location [String] required
99
+ # @param unit [String] optional - "Celcius" or "Fahrenheit"
100
+ # @return [String]
101
+ def execute(location:, unit: "Celsius")
102
+ puts "[weather] location=#{location} unit=#{unit}"
103
+ "#{rand(20..50)}° #{unit} at #{location}"
104
+ end
105
+ end
103
106
 
104
- WEATHER = proc do |locations:, unit: 'celsius'|
105
- locations.map do |location|
106
- "#{rand(20..50)}° #{unit} in #{location[:city]}, #{location[:country]}"
107
- end.join("\n")
107
+ client.chat(stream: $stdout, tools: [Weather.new]) do |prompt|
108
+ prompt.system "You are an expert in weather."
109
+ prompt.user 'What is the weather in "London" in Celsius and "Madrid" in Fahrenheit?'
108
110
  end
111
+ ```
109
112
 
110
- TOOL = OmniAI::Tool.new(
111
- WEATHER,
112
- name: 'Weather',
113
- description: 'Lookup the weather in a location',
114
- parameters: OmniAI::Tool::Parameters.new(
115
- properties: {
116
- locations: LOCATIONS,
117
- unit: UNIT,
118
- },
119
- required: %i[locations]
120
- )
121
- )
113
+ ```
114
+ [weather] location=London unit=Celsius
115
+ [weather] location=Madrid unit=Fahrenheit
116
+ ```
122
117
 
123
- completion = CLIENT.chat(tools: [TOOL]) do |prompt|
124
- prompt.user do |message|
125
- message.text('What is the weather in "London" in celcius and "Seattle" in fahrenheit?')
126
- end
127
- end
118
+ ```
119
+ The weather is 24° Celsius in London and 42° Fahrenheit in Madrid.
120
+ ```
121
+
122
+ ### Example #5: Chat w/ CLI
123
+
124
+ The `OmniAI` gem also ships with a CLI to simplify quick tests.
128
125
 
129
- puts(completion.text)
126
+ ```bash
127
+ omniai chat "Who designed the Ruby programming language?"
130
128
  ```
131
129
 
132
130
  ```
133
- The weather is 24° celcius in London and 42° fahrenheit in Seattle.
131
+ The Ruby programming language was created by Yukihiro Matsumoto, often known as "Matz."
132
+ ```
133
+
134
+ ```bash
135
+ omniai chat --provider="google" --model="gemini-2.0-flash" "Who are you?"
136
+ ```
137
+
138
+ ```
139
+ I am a large language model, trained by Google.
140
+ ```
141
+
142
+ ### Example #6: [Text-to-Speech](https://github.com/ksylvest/omniai/blob/main/examples/text_to_speech)
143
+
144
+ This example demonstrates using `OmniAI` with **OpenAI** to convert text to speech and save it to a file.
145
+
146
+ ```ruby
147
+ require 'omniai/openai'
148
+
149
+ client = OmniAI::OpenAI::Client.new
150
+
151
+ File.open(File.join(__dir__, 'audio.wav'), 'wb') do |file|
152
+ client.speak('Sally sells seashells by the seashore.', format: OmniAI::Speak::Format::WAV) do |chunk|
153
+ file << chunk
154
+ end
155
+ end
156
+ ```
157
+
158
+ ### Example #7: [Speech-to-Text](https://github.com/ksylvest/omniai/blob/main/examples/speech_to_text)
159
+
160
+ This example demonstrates using `OmniAI` with **OpenAI** to convert speech to text.
161
+
162
+ ```ruby
163
+ require 'omniai/openai'
164
+
165
+ client = OmniAI::OpenAI::Client.new
166
+
167
+ File.open(File.join(__dir__, 'audio.wav'), 'rb') do |file|
168
+ transcription = client.transcribe(file)
169
+ puts(transcription.text)
170
+ end
134
171
  ```
135
172
 
136
- ### Example #5: [Embeddings](https://github.com/ksylvest/omniai/blob/main/examples/embeddings)
173
+ ### Example #8: [Embeddings](https://github.com/ksylvest/omniai/blob/main/examples/embeddings)
137
174
 
138
175
  This example demonstrates using `OmniAI` with **Mistral** to generate embeddings for a dataset. It defines a set of entries (e.g. "George is a teacher." or "Ringo is a doctor.") and then compares the embeddings generated from a query (e.g. "What does George do?" or "Who is a doctor?") to rank the entries by relevance.
139
176
 
@@ -282,7 +319,7 @@ logger = Logger.new(STDOUT)
282
319
  client = OmniAI::OpenAI::Client.new(timeout: 8) # i.e. 8 seconds
283
320
  ```
284
321
 
285
- Timeouts are also be configurable by passing a `timeout` hash with `timeout` / `read` / `write` / `keys using:
322
+ Timeouts are also configurable by passing a `timeout` hash with `timeout` / `read` / `write` / keys using:
286
323
 
287
324
  ```ruby
288
325
  require 'omniai/openai'
@@ -351,18 +388,18 @@ A chat can also be initialized with tools:
351
388
 
352
389
  ```ruby
353
390
  tool = OmniAI::Tool.new(
354
- proc { |location:, unit: 'celsius'| "#{rand(20..50)}° #{unit} in #{location}" },
391
+ proc { |location:, unit: 'Celsius'| "#{rand(20..50)}° #{unit} in #{location}" },
355
392
  name: 'Weather',
356
393
  description: 'Lookup the weather in a location',
357
394
  parameters: OmniAI::Tool::Parameters.new(
358
395
  properties: {
359
396
  location: OmniAI::Tool::Property.string(description: 'e.g. Toronto'),
360
- unit: OmniAI::Tool::Property.string(enum: %w[celcius fahrenheit]),
397
+ unit: OmniAI::Tool::Property.string(enum: %w[Celsius Fahrenheit]),
361
398
  },
362
399
  required: %i[location]
363
400
  )
364
401
  )
365
- client.chat('What is the weather in "London" in celcius and "Paris" in fahrenheit?', tools: [tool])
402
+ client.chat('What is the weather in "London" in Celsius and "Paris" in Fahrenheit?', tools: [tool])
366
403
  ```
367
404
 
368
405
  ### Transcribe
@@ -28,6 +28,19 @@ module OmniAI
28
28
  # @return [Array<String>, nil]
29
29
  attr_reader :enum
30
30
 
31
+ # @param kind [Symbol]
32
+ # @return [OmniAI::Tool::Property]
33
+ def self.build(kind, **args)
34
+ case kind
35
+ when :array then array(**args)
36
+ when :object then object(**args)
37
+ when :boolean then boolean(**args)
38
+ when :integer then integer(**args)
39
+ when :string then string(**args)
40
+ when :number then number(**args)
41
+ end
42
+ end
43
+
31
44
  # @example
32
45
  # property = OmniAI::Tool::Property.array(
33
46
  # items: OmniAI::Tool::Property.string(description: 'The name of the person.'),
data/lib/omniai/tool.rb CHANGED
@@ -3,40 +3,82 @@
3
3
  module OmniAI
4
4
  # Usage:
5
5
  #
6
- # fibonacci = proc do |n:|
7
- # next(0) if n == 0
8
- # next(1) if n == 1
9
- # fibonacci.call(n: n - 1) + fibonacci.call(n: n - 2)
10
- # end
6
+ # class Weather < OmniAI::Tool
7
+ # description 'Find the weather for a location'
8
+ #
9
+ # parameter :location, :string, description: 'The location to find the weather for (e.g. "Toronto, Canada").'
10
+ # parameter :unit, :string, description: 'The unit of measurement (e.g. "Celcius" or "Fahrenheit").'
11
+ # required %i[location]
11
12
  #
12
- # tool = OmniAI::Tool.new(fibonacci,
13
- # name: 'Fibonacci',
14
- # description: 'Cacluate the nth Fibonacci',
15
- # parameters: OmniAI::Tool::Parameters.new(
16
- # properties: {
17
- # n: OmniAI::Tool::Property.integer(description: 'The nth Fibonacci number to calculate')
18
- # },
19
- # required: %i[n],
20
- # )
21
- # )
13
+ # def execute!(location:)
14
+ # # ...
15
+ # end
16
+ # end
22
17
  class Tool
23
- # @return [Proc]
18
+ class << self
19
+ # @param description [String]
20
+ def description(description = nil)
21
+ return @description if description.nil?
22
+
23
+ @description = description
24
+ end
25
+
26
+ # @return [OmniAI::Tool::Parameters]
27
+ def parameters
28
+ @parameters ||= Parameters.new
29
+ end
30
+
31
+ # @param name [Symbol]
32
+ # @param kind [Symbol]
33
+ def parameter(name, kind, **)
34
+ parameters.properties[name] = Property.build(kind, **)
35
+ end
36
+
37
+ # @param names [Array<Symbol>]
38
+ def required(names)
39
+ parameters.required = names
40
+ end
41
+
42
+ # Converts a class name to a tool:
43
+ # - e.g. "IBM::Watson::SearchTool" => "ibm_watson_search"
44
+ #
45
+ # @return [String]
46
+ def namify
47
+ name
48
+ .gsub("::", "_")
49
+ .gsub(/(?<prefix>[A-Z+])(?<suffix>[A-Z][a-z])/, '\k<prefix>_\k<suffix>')
50
+ .gsub(/(?<prefix>[a-z])(?<suffix>[A-Z])/, '\k<prefix>_\k<suffix>')
51
+ .gsub(/_tool$/i, "")
52
+ .downcase
53
+ end
54
+ end
55
+
56
+ # @!attribute [rw] function
57
+ # @return [Proc]
24
58
  attr_accessor :function
25
59
 
26
- # @return [String]
60
+ # @!attribute [rw] name
61
+ # @return [String]
27
62
  attr_accessor :name
28
63
 
29
- # @return [String, nil]
64
+ # @!attribute [description]
65
+ # @return [String, nil]
30
66
  attr_accessor :description
31
67
 
32
- # @return [Hash, nil]
68
+ # @!attribute[parameters]
69
+ # @return [Hash, nil]
33
70
  attr_accessor :parameters
34
71
 
35
72
  # @param function [Proc]
36
73
  # @param name [String]
37
74
  # @param description [String]
38
75
  # @param parameters [Hash]
39
- def initialize(function, name:, description: nil, parameters: nil)
76
+ def initialize(
77
+ function = method(:execute),
78
+ name: self.class.namify,
79
+ description: self.class.description,
80
+ parameters: self.class.parameters
81
+ )
40
82
  @function = function
41
83
  @name = name
42
84
  @description = description
@@ -79,6 +121,10 @@ module OmniAI
79
121
  }
80
122
  end
81
123
 
124
+ def execute(...)
125
+ raise NotImplementedError, "#{self.class}#execute undefined"
126
+ end
127
+
82
128
  # @example
83
129
  # tool.call({ "n" => 6 })
84
130
  # #=> 8
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module OmniAI
4
- VERSION = "2.0.0"
4
+ VERSION = "2.1.0"
5
5
  end
metadata CHANGED
@@ -1,13 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: omniai
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.0.0
4
+ version: 2.1.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Kevin Sylvestre
8
8
  bindir: exe
9
9
  cert_chain: []
10
- date: 2025-03-03 00:00:00.000000000 Z
10
+ date: 2025-03-17 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: event_stream_parser