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.
- checksums.yaml +4 -4
- data/.irbrc +14 -0
- data/CHANGELOG.md +21 -7
- data/README.md +54 -22
- data/Rakefile +0 -1
- data/examples/directives.rb +98 -0
- data/examples/using_search_proc.rb +0 -3
- data/lib/prompt_manager/directive_processor.rb +47 -0
- data/lib/prompt_manager/prompt.rb +80 -152
- data/lib/prompt_manager/storage/active_record_adapter.rb +46 -35
- data/lib/prompt_manager/storage/file_system_adapter.rb +58 -53
- data/lib/prompt_manager/storage.rb +28 -1
- data/lib/prompt_manager/version.rb +1 -1
- data/lib/prompt_manager.rb +14 -2
- metadata +49 -7
@@ -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
|
6
|
+
# Adds two additional methods to the Prompt class:
|
7
7
|
# list - returns Array of prompt IDs
|
8
|
-
# path
|
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.
|
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
|
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
|
-
|
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
|
-
|
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.
|
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)
|
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
|
-
|
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
|
-
|
data/lib/prompt_manager.rb
CHANGED
@@ -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
|
-
|
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
|
+
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:
|
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.
|
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: []
|