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 +4 -4
- data/README.md +114 -77
- data/lib/omniai/tool/property.rb +13 -0
- data/lib/omniai/tool.rb +66 -20
- data/lib/omniai/version.rb +1 -1
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: c58b1fbebfaaaa7bcd77fe9754d0bf0bddc7b55c07b76ae31cbae0ae24b2024f
|
4
|
+
data.tar.gz: d4462dbf00a04ce81db7f8bf7c4f45018aaab994fd2699483e8b2de324546ab2
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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/
|
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
|
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
|
-
|
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
|
-
|
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
|
-
|
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: [
|
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 **
|
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
|
40
|
+
require "omniai/mistral"
|
54
41
|
|
55
|
-
|
42
|
+
client = OmniAI::Mistral::Client.new
|
56
43
|
|
57
|
-
|
58
|
-
|
59
|
-
|
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
|
-
|
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
|
-
|
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
|
60
|
+
require "omniai/openai"
|
70
61
|
|
71
|
-
|
62
|
+
client = OmniAI::OpenAI::Client.new
|
72
63
|
|
73
|
-
|
74
|
-
|
75
|
-
|
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
|
-
|
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
|
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
|
-
|
89
|
+
client = OmniAI::Google::Client.new
|
87
90
|
|
88
|
-
|
89
|
-
|
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
|
-
|
97
|
-
|
98
|
-
|
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
|
-
|
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
|
-
|
105
|
-
|
106
|
-
|
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
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
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
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
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
|
-
|
126
|
+
```bash
|
127
|
+
omniai chat "Who designed the Ruby programming language?"
|
130
128
|
```
|
131
129
|
|
132
130
|
```
|
133
|
-
The
|
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 #
|
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
|
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: '
|
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[
|
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
|
402
|
+
client.chat('What is the weather in "London" in Celsius and "Paris" in Fahrenheit?', tools: [tool])
|
366
403
|
```
|
367
404
|
|
368
405
|
### Transcribe
|
data/lib/omniai/tool/property.rb
CHANGED
@@ -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
|
-
#
|
7
|
-
#
|
8
|
-
#
|
9
|
-
#
|
10
|
-
#
|
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
|
-
#
|
13
|
-
#
|
14
|
-
#
|
15
|
-
#
|
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
|
-
|
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
|
-
#
|
60
|
+
# @!attribute [rw] name
|
61
|
+
# @return [String]
|
27
62
|
attr_accessor :name
|
28
63
|
|
29
|
-
#
|
64
|
+
# @!attribute [description]
|
65
|
+
# @return [String, nil]
|
30
66
|
attr_accessor :description
|
31
67
|
|
32
|
-
#
|
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(
|
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
|
data/lib/omniai/version.rb
CHANGED
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.
|
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-
|
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
|