prompt_manager 0.4.1 → 0.5.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.
@@ -3,36 +3,45 @@
3
3
  # Use the local (or remote) file system as a place to
4
4
  # store and access prompts.
5
5
  #
6
- # Adds two additional methods to the Promp class:
6
+ # Adds two additional methods to the Prompt class:
7
7
  # list - returns Array of prompt IDs
8
- # path = returns a Pathname object to the prompt's text file
8
+ # path - returns a Pathname object to the prompt's text file
9
9
  # path(prompt_id) - same as path on the prompt instance
10
10
  #
11
11
  # Allows sub-directories of the prompts_dir to be
12
- # used like categories. For example the prompt_id "toy/magic"
12
+ # used like categories. For example the prompt_id "toy/magic"
13
13
  # is found in the `magic.txt` file inside the `toy` sub-directory
14
14
  # of the prompts_dir.
15
15
  #
16
- # There can man be many layers of categories (sub-directories)
16
+ # There can be many layers of categories (sub-directories)
17
17
  #
18
+ # This adapter serves as the file-based storage backend for PromptManager::Prompt,
19
+ # enabling prompt retrieval and persistence using file system operations.
18
20
 
19
21
  require 'json' # basic serialization of parameters
20
22
  require 'pathname'
23
+ require 'fileutils'
21
24
 
22
25
  class PromptManager::Storage::FileSystemAdapter
23
- SEARCH_PROC = nil # placeholder
26
+ # Placeholder for search proc
27
+ SEARCH_PROC = nil
28
+ # File extension for parameters
24
29
  PARAMS_EXTENSION = '.json'.freeze
30
+ # File extension for prompts
25
31
  PROMPT_EXTENSION = '.txt'.freeze
32
+ # Regular expression for valid prompt IDs
26
33
  PROMPT_ID_FORMAT = /^[a-zA-Z0-9\-\/_]+$/
27
34
 
28
35
  class << self
29
- attr_accessor :prompts_dir, :search_proc,
36
+ # Accessors for configuration options
37
+ attr_accessor :prompts_dir, :search_proc,
30
38
  :params_extension, :prompt_extension
31
-
39
+
40
+ # Configure the adapter
32
41
  def config
33
42
  if block_given?
34
43
  yield self
35
- validate_configuration
44
+ validate_configuration
36
45
  else
37
46
  raise ArgumentError, "No block given to config"
38
47
  end
@@ -48,7 +57,6 @@ class PromptManager::Storage::FileSystemAdapter
48
57
  new.list
49
58
  end
50
59
 
51
-
52
60
  def path(prompt_id)
53
61
  new.path(prompt_id)
54
62
  end
@@ -56,6 +64,7 @@ class PromptManager::Storage::FileSystemAdapter
56
64
  #################################################
57
65
  private
58
66
 
67
+ # Validate the configuration
59
68
  def validate_configuration
60
69
  validate_prompts_dir
61
70
  validate_search_proc
@@ -63,28 +72,33 @@ class PromptManager::Storage::FileSystemAdapter
63
72
  validate_params_extension
64
73
  end
65
74
 
66
-
75
+ # Validate the prompts directory
67
76
  def validate_prompts_dir
68
- # This is a work around for a Ruby scope issue where the
69
- # class getter/setter method is becoming confused with a
70
- # local variable when anything other than plain 'ol get and
71
- # set are used.'This error is in both Ruby v3.2.2 and
77
+ # This is a work around for a Ruby scope issue where the
78
+ # class getter/setter method is becoming confused with a
79
+ # local variable when anything other than plain 'ol get and
80
+ # set are used. This error is in both Ruby v3.2.2 and
72
81
  # v3.3.0-preview3.
73
82
  #
74
83
  prompts_dir_local = self.prompts_dir
75
84
 
85
+ raise ArgumentError, "prompts_dir must be set" if prompts_dir_local.nil? || prompts_dir_local.to_s.strip.empty?
86
+
76
87
  unless prompts_dir_local.is_a?(Pathname)
77
- prompts_dir_local = Pathname.new(prompts_dir_local) unless prompts_dir_local.nil?
88
+ prompts_dir_local = Pathname.new(prompts_dir_local)
78
89
  end
79
90
 
80
91
  prompts_dir_local = prompts_dir_local.expand_path
81
92
 
82
- raise(ArgumentError, "prompts_dir: #{prompts_dir_local}") unless prompts_dir_local.exist? && prompts_dir_local.directory?
83
-
93
+ unless prompts_dir_local.exist?
94
+ FileUtils.mkdir_p(prompts_dir_local)
95
+ end
96
+ raise(ArgumentError, "prompts_dir: #{prompts_dir_local} is not a directory") unless prompts_dir_local.directory?
97
+
84
98
  self.prompts_dir = prompts_dir_local
85
99
  end
86
100
 
87
-
101
+ # Validate the search proc
88
102
  def validate_search_proc
89
103
  search_proc_local = self.search_proc
90
104
 
@@ -97,7 +111,7 @@ class PromptManager::Storage::FileSystemAdapter
97
111
  self.search_proc = search_proc_local
98
112
  end
99
113
 
100
-
114
+ # Validate the prompt extension
101
115
  def validate_prompt_extension
102
116
  prompt_extension_local = self.prompt_extension
103
117
 
@@ -113,7 +127,7 @@ class PromptManager::Storage::FileSystemAdapter
113
127
  self.prompt_extension = prompt_extension_local
114
128
  end
115
129
 
116
-
130
+ # Validate the params extension
117
131
  def validate_params_extension
118
132
  params_extension_local = self.params_extension
119
133
 
@@ -130,25 +144,25 @@ class PromptManager::Storage::FileSystemAdapter
130
144
  end
131
145
  end
132
146
 
133
-
134
147
  ##################################################
135
148
  ###
136
149
  ## Instance
137
150
  #
138
151
 
152
+ # Accessors for instance variables
139
153
  def prompts_dir = self.class.prompts_dir
140
154
  def search_proc = self.class.search_proc
141
155
  def prompt_extension = self.class.prompt_extension
142
156
  def params_extension = self.class.params_extension
143
157
 
144
-
158
+ # Initialize the adapter
145
159
  def initialize
146
160
  # NOTE: validate because main program may have made
147
161
  # changes outside of the config block
148
162
  self.class.send(:validate_configuration) # send gets around private designations of a method
149
163
  end
150
164
 
151
-
165
+ # Get a prompt by ID
152
166
  def get(id:)
153
167
  validate_id(id)
154
168
  verify_id(id)
@@ -160,17 +174,15 @@ class PromptManager::Storage::FileSystemAdapter
160
174
  }
161
175
  end
162
176
 
163
-
164
177
  # Retrieve prompt text by its id
165
178
  def prompt_text(prompt_id)
166
179
  read_file(file_path(prompt_id, prompt_extension))
167
180
  end
168
181
 
169
-
170
182
  # Retrieve parameter values by its id
171
183
  def parameter_values(prompt_id)
172
184
  params_path = file_path(prompt_id, params_extension)
173
-
185
+
174
186
  if params_path.exist?
175
187
  parms_content = read_file(params_path)
176
188
  deserialize(parms_content)
@@ -179,35 +191,33 @@ class PromptManager::Storage::FileSystemAdapter
179
191
  end
180
192
  end
181
193
 
182
-
183
194
  # Save prompt text and parameter values to corresponding files
184
195
  def save(
185
- id:,
186
- text: "",
196
+ id:,
197
+ text: "",
187
198
  parameters: {}
188
199
  )
189
200
  validate_id(id)
190
201
 
191
202
  prompt_filepath = file_path(id, prompt_extension)
192
203
  params_filepath = file_path(id, params_extension)
193
-
204
+
194
205
  write_with_error_handling(prompt_filepath, text)
195
206
  write_with_error_handling(params_filepath, serialize(parameters))
196
207
  end
197
208
 
198
-
199
- # Delete prompted text and parameter values files
209
+ # Delete prompt text and parameter values files
200
210
  def delete(id:)
201
211
  validate_id(id)
202
212
 
203
213
  prompt_filepath = file_path(id, prompt_extension)
204
214
  params_filepath = file_path(id, params_extension)
205
-
215
+
206
216
  delete_with_error_handling(prompt_filepath)
207
217
  delete_with_error_handling(params_filepath)
208
218
  end
209
219
 
210
-
220
+ # Search for prompts
211
221
  def search(for_what)
212
222
  search_term = for_what.downcase
213
223
 
@@ -218,25 +228,23 @@ class PromptManager::Storage::FileSystemAdapter
218
228
  end
219
229
  end
220
230
 
221
-
222
231
  # Return an Array of prompt IDs
223
232
  def list(*)
224
233
  prompt_ids = []
225
-
234
+
226
235
  Pathname.glob(prompts_dir.join("**/*#{prompt_extension}")).each do |file_path|
227
236
  prompt_id = file_path.relative_path_from(prompts_dir).to_s.gsub(prompt_extension, '')
228
237
  prompt_ids << prompt_id
229
238
  end
230
239
 
231
- prompt_ids
240
+ prompt_ids.sort
232
241
  end
233
242
 
234
-
235
243
  # Returns a Pathname object for a prompt ID text file
236
244
  # However, it is possible that the file does not exist.
237
245
  def path(id)
238
246
  validate_id(id)
239
- file_path(id, prompt_extension)
247
+ file_path(id, prompt_extension)
240
248
  end
241
249
 
242
250
  ##########################################
@@ -247,14 +255,14 @@ class PromptManager::Storage::FileSystemAdapter
247
255
  raise ArgumentError, "Invalid ID format id: #{id}" unless id =~ PROMPT_ID_FORMAT
248
256
  end
249
257
 
250
-
258
+ # Verify that the ID exists
251
259
  def verify_id(id)
252
260
  unless file_path(id, prompt_extension).exist?
253
261
  raise ArgumentError, "Invalid prompt_id: #{id}"
254
262
  end
255
263
  end
256
264
 
257
-
265
+ # Write to a file with error handling
258
266
  def write_with_error_handling(file_path, content)
259
267
  begin
260
268
  file_path.write content
@@ -264,8 +272,7 @@ class PromptManager::Storage::FileSystemAdapter
264
272
  end
265
273
  end
266
274
 
267
-
268
- # file_path (Pathname)
275
+ # Delete a file with error handling
269
276
  def delete_with_error_handling(file_path)
270
277
  begin
271
278
  file_path.delete
@@ -275,39 +282,37 @@ class PromptManager::Storage::FileSystemAdapter
275
282
  end
276
283
  end
277
284
 
278
-
285
+ # Get the file path for a prompt ID and extension
279
286
  def file_path(id, extension)
280
287
  prompts_dir + "#{id}#{extension}"
281
288
  end
282
289
 
283
-
290
+ # Read a file
284
291
  def read_file(full_path)
285
292
  raise IOError, 'File does not exist' unless File.exist?(full_path)
286
293
  File.read(full_path)
287
294
  end
288
295
 
289
-
296
+ # Search for prompts
290
297
  def search_prompts(search_term)
291
298
  prompt_ids = []
292
-
299
+
293
300
  Pathname.glob(prompts_dir.join("**/*#{prompt_extension}")).each do |prompt_path|
294
301
  if prompt_path.read.downcase.include?(search_term)
295
- prompt_id = prompt_path.relative_path_from(prompts_dir).to_s.gsub(prompt_extension, '')
302
+ prompt_id = prompt_path.relative_path_from(prompts_dir).to_s.gsub(prompt_extension, '')
296
303
  prompt_ids << prompt_id
297
304
  end
298
305
  end
299
306
 
300
- prompt_ids
307
+ prompt_ids.sort
301
308
  end
302
309
 
303
-
304
- # TODO: Should the serializer be generic?
305
-
310
+ # Serialize data to JSON
306
311
  def serialize(data)
307
312
  data.to_json
308
313
  end
309
314
 
310
-
315
+ # Deserialize JSON data
311
316
  def deserialize(data)
312
317
  JSON.parse(data)
313
318
  end
@@ -1,7 +1,34 @@
1
1
  # prompt_manager/lib/prompt_manager/storage.rb
2
2
 
3
+ # The Storage module provides a namespace for different storage adapters
4
+ # that handle persistence of prompts. Each adapter implements a common
5
+ # interface for saving, retrieving, searching, and deleting prompts.
6
+ #
7
+ # Available adapters:
8
+ # - FileSystemAdapter: Stores prompts in text files on the local filesystem
9
+ # - ActiveRecordAdapter: Stores prompts in a database using ActiveRecord
10
+ #
11
+ # To use an adapter, configure it before using PromptManager::
12
+ #
13
+ # Example with FileSystemAdapter:
14
+ # PromptManager::Storage::FileSystemAdapter.config do |config|
15
+ # config.prompts_dir = Pathname.new('/path/to/prompts')
16
+ # end
17
+ # PromptManager::Prompt.storage_adapter = PromptManager::Storage::FileSystemAdapter.new
18
+ #
19
+ # Example with ActiveRecordAdapter:
20
+ # PromptManager::Storage::ActiveRecordAdapter.config do |config|
21
+ # config.model = MyPromptModel
22
+ # config.id_column = :prompt_id
23
+ # config.text_column = :content
24
+ # config.parameters_column = :params
25
+ # end
26
+ # PromptManager::Prompt.storage_adapter = PromptManager::Storage::ActiveRecordAdapter.new
3
27
  module PromptManager
28
+ # The Storage module provides adapters for different storage backends.
29
+ # Each adapter implements a common interface for managing prompts.
30
+ # Note: PromptManager::Prompt uses one of these adapters as its storage backend to
31
+ # perform all CRUD operations on prompt data.
4
32
  module Storage
5
33
  end
6
34
  end
7
-
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module PromptManager
4
- VERSION = "0.4.1"
4
+ VERSION = "0.5.0"
5
5
  end
@@ -2,11 +2,23 @@
2
2
  #
3
3
  # frozen_string_literal: true
4
4
 
5
+ require 'ostruct'
6
+
5
7
  require_relative "prompt_manager/version"
6
- require_relative "prompt_manager/storage"
7
8
  require_relative "prompt_manager/prompt"
9
+ require_relative "prompt_manager/storage"
10
+ require_relative "prompt_manager/storage/file_system_adapter"
8
11
 
12
+ # The PromptManager module provides functionality for managing, storing,
13
+ # retrieving, and parameterizing text prompts used with generative AI systems.
14
+ # It supports different storage backends through adapters and offers features
15
+ # like parameter substitution, directives processing, and comment handling.
9
16
  module PromptManager
17
+ # Base error class for all PromptManager-specific errors
10
18
  class Error < StandardError; end
11
- # TODO: Add some more module specific errors here
19
+
20
+ # TODO: Add additional module-specific error classes such as:
21
+ # - StorageError - For issues with storing or retrieving prompts
22
+ # - ParameterError - For issues with parameter substitution
23
+ # - ConfigurationError - For setup and configuration issues
12
24
  end
metadata CHANGED
@@ -1,14 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: prompt_manager
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.4.1
4
+ version: 0.5.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Dewayne VanHoozer
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2023-12-29 00:00:00.000000000 Z
10
+ date: 2025-03-30 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
13
  name: activerecord
@@ -66,6 +65,20 @@ dependencies:
66
65
  - - ">="
67
66
  - !ruby/object:Gem::Version
68
67
  version: '0'
68
+ - !ruby/object:Gem::Dependency
69
+ name: ostruct
70
+ requirement: !ruby/object:Gem::Requirement
71
+ requirements:
72
+ - - ">="
73
+ - !ruby/object:Gem::Version
74
+ version: '0'
75
+ type: :development
76
+ prerelease: false
77
+ version_requirements: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - ">="
80
+ - !ruby/object:Gem::Version
81
+ version: '0'
69
82
  - !ruby/object:Gem::Dependency
70
83
  name: tocer
71
84
  requirement: !ruby/object:Gem::Requirement
@@ -80,8 +93,36 @@ dependencies:
80
93
  - - ">="
81
94
  - !ruby/object:Gem::Version
82
95
  version: '0'
96
+ - !ruby/object:Gem::Dependency
97
+ name: simplecov
98
+ requirement: !ruby/object:Gem::Requirement
99
+ requirements:
100
+ - - ">="
101
+ - !ruby/object:Gem::Version
102
+ version: '0'
103
+ type: :development
104
+ prerelease: false
105
+ version_requirements: !ruby/object:Gem::Requirement
106
+ requirements:
107
+ - - ">="
108
+ - !ruby/object:Gem::Version
109
+ version: '0'
110
+ - !ruby/object:Gem::Dependency
111
+ name: sqlite3
112
+ requirement: !ruby/object:Gem::Requirement
113
+ requirements:
114
+ - - ">="
115
+ - !ruby/object:Gem::Version
116
+ version: '0'
117
+ type: :development
118
+ prerelease: false
119
+ version_requirements: !ruby/object:Gem::Requirement
120
+ requirements:
121
+ - - ">="
122
+ - !ruby/object:Gem::Version
123
+ version: '0'
83
124
  description: "Manage the parameterized prompts (text) used in generative AI (aka chatGPT,
84
- \nOpenAI, et.al.) using storage adapters such as FileSystemAdapter, \nSqliteAdapter
125
+ \nOpenAI, et.al.) using storage adapters such as FileSystemAdapter, \nSqliteAdapter,
85
126
  and ActiveRecordAdapter.\n"
86
127
  email:
87
128
  - dvanhoozer@gmail.com
@@ -90,11 +131,13 @@ extensions: []
90
131
  extra_rdoc_files: []
91
132
  files:
92
133
  - ".envrc"
134
+ - ".irbrc"
93
135
  - CHANGELOG.md
94
136
  - LICENSE
95
137
  - LICENSE.txt
96
138
  - README.md
97
139
  - Rakefile
140
+ - examples/directives.rb
98
141
  - examples/prompts_dir/todo.json
99
142
  - examples/prompts_dir/todo.txt
100
143
  - examples/prompts_dir/toy/8-ball.txt
@@ -102,6 +145,7 @@ files:
102
145
  - examples/simple.rb
103
146
  - examples/using_search_proc.rb
104
147
  - lib/prompt_manager.rb
148
+ - lib/prompt_manager/directive_processor.rb
105
149
  - lib/prompt_manager/prompt.rb
106
150
  - lib/prompt_manager/storage.rb
107
151
  - lib/prompt_manager/storage/active_record_adapter.rb
@@ -115,7 +159,6 @@ metadata:
115
159
  homepage_uri: https://github.com/MadBomber/prompt_manager
116
160
  source_code_uri: https://github.com/MadBomber/prompt_manager
117
161
  changelog_uri: https://github.com/MadBomber/prompt_manager
118
- post_install_message:
119
162
  rdoc_options: []
120
163
  require_paths:
121
164
  - lib
@@ -130,8 +173,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
130
173
  - !ruby/object:Gem::Version
131
174
  version: '0'
132
175
  requirements: []
133
- rubygems_version: 3.5.3
134
- signing_key:
176
+ rubygems_version: 3.6.6
135
177
  specification_version: 4
136
178
  summary: Manage prompts for use with gen-AI processes
137
179
  test_files: []