openai_helper_gem 0.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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 95b6f0c9a4e8ac865a75b836fe47e24af0e2f55842ca4b61693b024173f25141
4
+ data.tar.gz: b3839baaebd14496259c50847bd95cb437b89bef635106f53dd6173509445c7d
5
+ SHA512:
6
+ metadata.gz: 6d5e9d057dc8d9fa200a0b3f99ca769203af14b6b32e8914b3da08ee06404a3fe8b8fcc8560ba39ac77a53793c1fd7c0e5a5a06c406a94c14be532ea9ad72568
7
+ data.tar.gz: a0e02ec7bc677431b0a00a9268647d8f9740d9573288e50e5135e02587be3de13b7c08410af792ac9c4bd0c05eb291aa2d248dc326702fc067bacdb287efc5ed
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2024 Miguel Regedor
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,127 @@
1
+ # OpenAIHelperGem
2
+
3
+ `OpenAIHelperGem` is a Ruby module designed to simplify interactions with OpenAI's API. It provides functionality to upload files, send messages with structured outputs, generate speech from text, and automatically play the resulting audio on macOS.
4
+
5
+ ## Features
6
+
7
+ - **File Upload:** Upload files to OpenAI and retrieve `file_id` for further use.
8
+ - **Text-to-Speech (TTS):** Convert text to high-quality speech and save the audio locally.
9
+ - **Autoplay Audio:** Automatically play the generated audio after saving (configurable).
10
+ - **Structured Output Extraction:** Send messages and extract structured outputs from OpenAI responses.
11
+ - **Custom Notifications:** Handle errors with notifications (macOS, terminal output, or none).
12
+
13
+ ## Installation
14
+
15
+ Add the module to your project:
16
+
17
+ ```ruby
18
+ require_relative 'path_to_openai_helper_gem/openai_helper_gem'
19
+ ```
20
+
21
+ Ensure you have the following environment variables set:
22
+
23
+ - `OPENAI_API_KEY`: Your OpenAI API key.
24
+ - `GPT_MODEL` (optional): The OpenAI model you wish to use (defaults to gpt-3.5-turbo).
25
+ - `OPENAI_AUDIO_PATH` (optional): Custom path to save generated audio (defaults to the user's home directory).
26
+
27
+ ## Usage
28
+
29
+ ### Basic Example
30
+
31
+ Send a message and extract structured output:
32
+
33
+ ```ruby
34
+ require_relative 'openai_helper_gem'
35
+
36
+ messages = [{ role: 'user', content: 'Tell me a joke.' }]
37
+ schema = {
38
+ type: "object",
39
+ properties: {
40
+ joke: { type: "string" }
41
+ },
42
+ required: ["joke"]
43
+ }
44
+
45
+ response = OpenAIHelperGem::Client.perform_request(messages, schema)
46
+ puts response['joke']
47
+ ```
48
+
49
+ ### File Upload
50
+
51
+ Upload a file to OpenAI and retrieve the `file_id`:
52
+
53
+ ```ruby
54
+ file_id = OpenAIHelperGem::Client.upload_file('path_to_your_file.pdf')
55
+ ```
56
+
57
+ ### Generate Speech
58
+
59
+ Generate speech from text and automatically play the audio:
60
+
61
+ ```ruby
62
+ OpenAIHelperGem::Client.generate_speech("This is a test message.")
63
+ ```
64
+
65
+ Generate speech without playing it immediately:
66
+
67
+ ```ruby
68
+ OpenAIHelperGem::Client.generate_speech("This is a test message.", "output.mp3", autoplay: false)
69
+ ```
70
+
71
+ ### Notifications
72
+
73
+ Customize how errors and other notifications are handled:
74
+
75
+ Use terminal output:
76
+
77
+ ```ruby
78
+ OpenAIHelperGem::Client.notification_method = :puts
79
+ ```
80
+
81
+ Disable notifications:
82
+
83
+ ```ruby
84
+ OpenAIHelperGem::Client.notification_method = :none
85
+ ```
86
+
87
+ ### Default Paths
88
+
89
+ The module saves generated audio files to a default directory in the user's home folder:
90
+
91
+ ```ruby
92
+ OpenAIHelperGem::Client.default_audio_path = ENV['OPENAI_AUDIO_PATH'] || File.join(Dir.home, 'Local Resources', 'API Logs', 'OpenAI Whisper')
93
+ ```
94
+
95
+ ## Complex Example
96
+
97
+ Here’s a more comprehensive example that combines file upload, schema validation, and text-to-speech generation:
98
+
99
+ ```ruby
100
+ require_relative 'openai_helper_gem'
101
+
102
+ # Step 1: Define messages and schema
103
+ messages = [
104
+ { role: 'user', content: 'Please summarize the attached document.' }
105
+ ]
106
+
107
+ schema = {
108
+ type: "object",
109
+ properties: {
110
+ summary: { type: "string" }
111
+ },
112
+ required: ["summary"]
113
+ }
114
+
115
+ # Step 2: Send message with the file and extract structured output
116
+ response = OpenAIHelperGem::Client.perform_request(messages, schema, 'path_to_your_document.pdf')
117
+
118
+ # Step 3: Output the summary
119
+ puts response['summary']
120
+
121
+ # Step 4: Convert the summary to speech and save it as an audio file
122
+ OpenAIHelperGem::Client.generate_speech(response['summary'], "document_summary.mp3")
123
+ ```
124
+
125
+ ## License
126
+
127
+ This module is free to use and modify under the MIT License. See the `LICENSE.txt` file for more details.
@@ -0,0 +1,129 @@
1
+ require 'net/http'
2
+ require 'json'
3
+ require 'uri'
4
+ require 'fileutils'
5
+
6
+ module OpenAIHelperGem
7
+ class Client
8
+ @notification_method = :mac_os
9
+ @default_audio_path = ENV['OPENAI_AUDIO_PATH'] || File.join(Dir.home, 'Local Resources', 'API Logs', 'OpenAI Whisper')
10
+ @api_key = ENV['OPENAI_API_KEY']
11
+ @tts_api_endpoint = URI('https://api.openai.com/v1/audio/speech')
12
+ @file_upload_endpoint = URI('https://api.openai.com/v1/files')
13
+ @chat_completion_endpoint = URI('https://api.openai.com/v1/chat/completions')
14
+
15
+ class << self
16
+ attr_accessor :notification_method, :default_audio_path, :api_key
17
+
18
+ def upload_file(file_path)
19
+ file = File.open(file_path, 'rb')
20
+ request_data = { purpose: 'assistant', file: file }
21
+
22
+ begin
23
+ http = Net::HTTP.new(@file_upload_endpoint.host, @file_upload_endpoint.port)
24
+ http.use_ssl = true
25
+ request = Net::HTTP::Post.new(@file_upload_endpoint.path, { 'Authorization': "Bearer #{@api_key}" })
26
+ request.set_form([['file', file]], 'multipart/form-data')
27
+ response = http.request(request)
28
+ file.close
29
+ result = JSON.parse(response.body)
30
+ if result['error']
31
+ notify_error("File Upload Failed", result['error']['message'])
32
+ exit 1
33
+ end
34
+ result['id']
35
+ rescue StandardError => e
36
+ notify_error("File Upload Error", e.message)
37
+ exit 1
38
+ end
39
+ end
40
+
41
+ def send_message_with_file(messages, file_id)
42
+ request_data = { model: ENV['GPT_MODEL'] || 'gpt-3.5-turbo', messages: messages, file_ids: [file_id] }
43
+
44
+ begin
45
+ http = Net::HTTP.new(@chat_completion_endpoint.host, @chat_completion_endpoint.port)
46
+ http.use_ssl = true
47
+ request = Net::HTTP::Post.new(@chat_completion_endpoint.path, {
48
+ 'Content-Type': 'application/json',
49
+ 'Authorization': "Bearer #{@api_key}"
50
+ })
51
+ request.body = request_data.to_json
52
+ response = http.request(request)
53
+ JSON.parse(response.body)
54
+ rescue StandardError => e
55
+ notify_error("API Request Failed", e.message)
56
+ exit 1
57
+ end
58
+ end
59
+
60
+ def generate_speech(text, output_file_name = "output.mp3", autoplay: true)
61
+ request_data = { model: 'tts-1-hd', voice: 'alloy', input: text }
62
+
63
+ begin
64
+ http = Net::HTTP.new(@tts_api_endpoint.host, @tts_api_endpoint.port)
65
+ http.use_ssl = true
66
+ request = Net::HTTP::Post.new(@tts_api_endpoint.path, {
67
+ 'Content-Type': 'application/json',
68
+ 'Authorization': "Bearer #{@api_key}"
69
+ })
70
+ request.body = request_data.to_json
71
+ response = http.request(request)
72
+
73
+ output_path = File.join(@default_audio_path, output_file_name)
74
+ FileUtils.mkdir_p(File.dirname(output_path))
75
+ File.open(output_path, 'wb') { |file| file.write(response.body) }
76
+
77
+ puts "Audio saved to #{output_path}"
78
+ play_audio(output_path) if autoplay
79
+ rescue StandardError => e
80
+ notify_error("Text-to-Speech Error", e.message)
81
+ exit 1
82
+ end
83
+ end
84
+
85
+ def play_audio(file_path)
86
+ system("afplay '#{file_path}'")
87
+ end
88
+
89
+ def extract_structured_output(response)
90
+ if response['error']
91
+ notify_error("API Error", response['error']['message'])
92
+ exit 1
93
+ end
94
+
95
+ function_call = response.dig('choices', 0, 'message', 'function_call')
96
+ unless function_call && function_call['arguments']
97
+ notify_error("Invalid Response", "No structured output found in the response.")
98
+ exit 1
99
+ end
100
+
101
+ begin
102
+ JSON.parse(function_call['arguments'])
103
+ rescue JSON::ParserError => e
104
+ notify_error("JSON Parsing Error", e.message)
105
+ exit 1
106
+ end
107
+ end
108
+
109
+ def perform_request(messages, schema, file_path = nil)
110
+ file_id = upload_file(file_path) if file_path
111
+ response = send_message_with_file(messages, file_id) if file_id
112
+ extract_structured_output(response)
113
+ end
114
+
115
+ def notify_error(title, message)
116
+ case @notification_method
117
+ when :mac_os
118
+ system("osascript -e 'display notification \"#{message}\" with title \"#{title}\"'")
119
+ when :puts
120
+ puts "[#{title}] #{message}"
121
+ when :none
122
+ # No output
123
+ else
124
+ puts "[Unknown Notification Method] #{title}: #{message}"
125
+ end
126
+ end
127
+ end
128
+ end
129
+ end
@@ -0,0 +1,3 @@
1
+ module OpenAIHelperGem
2
+ VERSION = "0.1.0"
3
+ end
@@ -0,0 +1,6 @@
1
+ require "openai_helper_gem/version"
2
+ require "openai_helper_gem/client"
3
+
4
+ module OpenAIHelperGem
5
+ # Your code goes here...
6
+ end
metadata ADDED
@@ -0,0 +1,92 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: openai_helper_gem
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Miguel Regedor
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2024-08-16 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: net-http
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '0.1'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '0.1'
27
+ - !ruby/object:Gem::Dependency
28
+ name: json
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '2.0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '2.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rspec
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '3.0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '3.0'
55
+ description: OpenAIHelper is a Ruby module designed to enhance productivity by integrating
56
+ OpenAI's capabilities into local scripts. It allows file uploads, structured output
57
+ extraction, text-to-speech generation, and automatic audio playback.
58
+ email:
59
+ - miguelregedor@gmail.com
60
+ executables: []
61
+ extensions: []
62
+ extra_rdoc_files: []
63
+ files:
64
+ - LICENSE.txt
65
+ - README.md
66
+ - lib/openai_helper_gem.rb
67
+ - lib/openai_helper_gem/client.rb
68
+ - lib/openai_helper_gem/version.rb
69
+ homepage: http://example.com/openai_helper_gem
70
+ licenses:
71
+ - MIT
72
+ metadata: {}
73
+ post_install_message:
74
+ rdoc_options: []
75
+ require_paths:
76
+ - lib
77
+ required_ruby_version: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - ">="
80
+ - !ruby/object:Gem::Version
81
+ version: '0'
82
+ required_rubygems_version: !ruby/object:Gem::Requirement
83
+ requirements:
84
+ - - ">="
85
+ - !ruby/object:Gem::Version
86
+ version: '0'
87
+ requirements: []
88
+ rubygems_version: 3.5.17
89
+ signing_key:
90
+ specification_version: 4
91
+ summary: A Ruby module to simplify interactions with OpenAI's API.
92
+ test_files: []