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
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 267e48a65e377f2fc3390d3bbff69a6e1f241c96951fc86437f5ab02d2e93daa
|
4
|
+
data.tar.gz: db8af7d8515e72bf12f847e3544a7413a0acb0707a8321bcae7f13e1ffa1ff86
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 259cb1494a9390f94d40c669bd87960d9257a9126c3d71cbc6fdcdbaa82dae3e0209f23acc322d019e781abe48090a070fba20761a2ffe8a1818a303c8ef2741
|
7
|
+
data.tar.gz: 97e6ce3eb0743b02804b44c745a2710f15e4048957c7a039c6313b2d061d65660f9d4435465fd0139ff8e348ba841cb4ae543d06a7709391fab19a6419ea1e55
|
data/.irbrc
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
require_relative 'lib/prompt_manager'
|
2
|
+
require_relative 'lib/prompt_manager/storage/file_system_adapter'
|
3
|
+
|
4
|
+
HERE = Pathname.new __dir__
|
5
|
+
|
6
|
+
PromptManager::Storage::FileSystemAdapter.config do |config|
|
7
|
+
config.prompts_dir = HERE + 'examples/prompts_dir'
|
8
|
+
# config.search_proc = nil # default
|
9
|
+
# config.prompt_extension = '.txt' # default
|
10
|
+
# config.parms+_extension = '.json' # default
|
11
|
+
end
|
12
|
+
|
13
|
+
PromptManager::Prompt.storage_adapter = PromptManager::Storage::FileSystemAdapter.new
|
14
|
+
|
data/CHANGELOG.md
CHANGED
@@ -1,27 +1,41 @@
|
|
1
|
-
##
|
1
|
+
## Unreleased
|
2
|
+
|
3
|
+
## Released
|
4
|
+
### [0.5.0] = 2025-03-29
|
5
|
+
- Major refactoring of to improve processing of parameters and directives.
|
6
|
+
- Added PromptManager::DirectiveProcessor as an example of how to implement custom directives.
|
7
|
+
- Added support for //include directive that protects against loops.
|
8
|
+
- Added support for embedding system environment variables.
|
9
|
+
- Added support for ERB processing within a prompt.
|
10
|
+
- Improved test coverage.
|
11
|
+
|
12
|
+
### [0.4.2] = 2024-10-26
|
13
|
+
- Added configurable parameter_regex to customize keyword pattern
|
14
|
+
|
15
|
+
### [0.4.1] = 2023-12-29
|
2
16
|
- Changed @directives from Hash to an Array
|
3
17
|
- Fixed keywords not being substituted in directives
|
4
18
|
|
5
|
-
|
19
|
+
### [0.4.0] = 2023-12-19
|
6
20
|
- Add "//directives param(s)" with keywords just like the prompt text.
|
7
21
|
|
8
|
-
|
22
|
+
### [0.3.3] = 2023-12-01
|
9
23
|
- Added example of using the `search_proc` config parameter with the FileSystemAdapter.
|
10
24
|
|
11
|
-
|
25
|
+
### [0.3.2] = 2023-12-01
|
12
26
|
|
13
27
|
- The ActiveRecordAdapter is passing its unit tests
|
14
28
|
- Dropped the concept of an sqlite3 adapter since active record can be used to access sqlite3 databases as well as the big boys.
|
15
29
|
|
16
|
-
|
30
|
+
### [0.3.0] = 2023-11-28
|
17
31
|
|
18
32
|
- **Breaking change** The value of the parameters Hash for a keyword is now an Array instead of a single value. The last value in the Array is always the most recent value used for the given keyword. This was done to support the use of a Readline::History object editing in the [aia](https://github.com/MadBomber/aia) CLI tool
|
19
33
|
|
20
|
-
|
34
|
+
### [0.2.0] - 2023-11-21
|
21
35
|
|
22
36
|
- **Breaking change to FileSystemAdapter config process**
|
23
37
|
- added list and path as extra methods in FileSystemAdapter
|
24
38
|
|
25
|
-
|
39
|
+
### [0.1.0] - 2023-11-16
|
26
40
|
|
27
41
|
- Initial release using the FileSystemAdapter
|
data/README.md
CHANGED
@@ -9,6 +9,8 @@ Manage the parameterized prompts (text) used in generative AI (aka chatGPT, Open
|
|
9
9
|
|
10
10
|
## Table of Contents
|
11
11
|
|
12
|
+
- [PromptManager](#promptmanager)
|
13
|
+
- [Table of Contents](#table-of-contents)
|
12
14
|
- [Installation](#installation)
|
13
15
|
- [Usage](#usage)
|
14
16
|
- [Overview](#overview)
|
@@ -16,15 +18,15 @@ Manage the parameterized prompts (text) used in generative AI (aka chatGPT, Open
|
|
16
18
|
- [What does a keyword look like?](#what-does-a-keyword-look-like)
|
17
19
|
- [All about directives](#all-about-directives)
|
18
20
|
- [Example Prompt with Directives](#example-prompt-with-directives)
|
19
|
-
- [Accessing Directives](#accessing-directives)
|
21
|
+
- [Accessing Directives and Setting Parameter Values](#accessing-directives-and-setting-parameter-values)
|
20
22
|
- [Dynamic Directives](#dynamic-directives)
|
21
23
|
- [Executing Directives](#executing-directives)
|
22
24
|
- [Comments Are Ignored](#comments-are-ignored)
|
23
25
|
- [Storage Adapters](#storage-adapters)
|
24
26
|
- [FileSystemAdapter](#filesystemadapter)
|
25
27
|
- [Configuration](#configuration)
|
26
|
-
- [
|
27
|
-
- [
|
28
|
+
- [prompts\_dir](#prompts_dir)
|
29
|
+
- [search\_proc](#search_proc)
|
28
30
|
- [File Extensions](#file-extensions)
|
29
31
|
- [Example Prompt Text File](#example-prompt-text-file)
|
30
32
|
- [Example Prompt Parameters JSON File](#example-prompt-parameters-json-file)
|
@@ -32,15 +34,21 @@ Manage the parameterized prompts (text) used in generative AI (aka chatGPT, Open
|
|
32
34
|
- [ActiveRecordAdapter](#activerecordadapter)
|
33
35
|
- [Configuration](#configuration-1)
|
34
36
|
- [model](#model)
|
35
|
-
- [
|
36
|
-
- [
|
37
|
-
- [
|
37
|
+
- [id\_column](#id_column)
|
38
|
+
- [text\_column](#text_column)
|
39
|
+
- [parameters\_column](#parameters_column)
|
38
40
|
- [Other Potential Storage Adapters](#other-potential-storage-adapters)
|
39
41
|
- [Development](#development)
|
40
42
|
- [Contributing](#contributing)
|
41
43
|
- [License](#license)
|
42
44
|
|
43
45
|
<!-- Tocer[finish]: Auto-generated, don't remove. -->
|
46
|
+
### Latest Capabilities
|
47
|
+
- **Directive Processing:** Processes directives such as `//include` (aliased as `//import`) with loop protection.
|
48
|
+
- **ERB Processing:** Supports ERB templating within prompts.
|
49
|
+
- **Environment Variable Embedding:** Automatically substitutes system environment variables in prompts.
|
50
|
+
- **Improved Parameter Handling:** Refactored to maintain a history of parameter values.
|
51
|
+
- **ActiveRecord Adapter:** Facilitates storing and retrieving prompts via an ActiveRecord model.
|
44
52
|
|
45
53
|
## Installation
|
46
54
|
|
@@ -60,25 +68,39 @@ See also [examples/using_search_proc.rb](examples/using_search_proc.rb)
|
|
60
68
|
|
61
69
|
## Overview
|
62
70
|
|
71
|
+
### Prompt Initialization Options
|
72
|
+
- `id`: A String name for the prompt.
|
73
|
+
- `context`: An Array for additional context.
|
74
|
+
- `directives_processor`: An instance of PromptManager::DirectiveProcessor (default), can be customized.
|
75
|
+
- `external_binding`: A Ruby binding to be used for ERB processing.
|
76
|
+
- `erb_flag`: Boolean flag to enable ERB processing in the prompt text.
|
77
|
+
- `envar_flag`: Boolean flag to enable environment variable substitution in the prompt text.
|
78
|
+
|
63
79
|
The `prompt_manager` gem provides functionality to manage prompts that have keywords and directives for use with generative AI processes.
|
64
80
|
|
65
81
|
### Generative AI (gen-AI)
|
66
82
|
|
67
|
-
Gen-AI deals with the conversion (some would say execution) of a human natural language text (the "prompt") into
|
83
|
+
Gen-AI deals with the conversion (some would say execution) of a human natural language text (the "prompt") into something else using what are known as large language models (LLM) such as those available from OpenAI. A parameterized prompt is one in which there are embedded keywords (parameters) which are place holders for other text to be inserted into the prompt.
|
68
84
|
|
69
85
|
The prompt_manager uses a regular expression to identify these keywords within the prompt. It uses the keywords as keys in a `parameters` Hash which is stored with the prompt text in a serialized form - for example as JSON.
|
70
86
|
|
71
87
|
#### What does a keyword look like?
|
72
88
|
|
73
|
-
|
89
|
+
By default, any text matching `[UPPERCASE_TEXT]` enclosed in square brackets is treated as a keyword. [KEYWORDS CAN ALSO HAVE SPACES] as well as the underscore character.
|
90
|
+
|
91
|
+
You can customize the keyword pattern by setting a different regular expression:
|
74
92
|
|
75
|
-
|
93
|
+
```ruby
|
94
|
+
# Use {{param}} style instead of [PARAM]
|
95
|
+
PromptManager::Prompt.parameter_regex = /(\{\{[A-Za-z_]+\}\})/
|
96
|
+
```
|
76
97
|
|
98
|
+
The regex must include capturing parentheses () to extract the keyword. The default regex is `/(\[[A-Z _|]+\])/`.
|
77
99
|
#### All about directives
|
78
100
|
|
79
101
|
A directive is a line in the prompt text that starts with the two characters '//' - slash slash - just like in the old days of IBM JCL - Job Control Language. A prompt can have zero or more directives. Directives can have parameters and can make use of keywords.
|
80
102
|
|
81
|
-
The `prompt_manager` only collects directives. It extracts keywords from directive lines and provides the substitution of those keywords with other text just like it does for the prompt.
|
103
|
+
The `prompt_manager` only collects directives. It extracts keywords from directive lines and provides the substitution of those keywords with other text just like it does for the prompt.
|
82
104
|
|
83
105
|
##### Example Prompt with Directives
|
84
106
|
|
@@ -96,20 +118,30 @@ __END__
|
|
96
118
|
Computers will never replace Frank Sinatra
|
97
119
|
```
|
98
120
|
|
99
|
-
##### Accessing Directives
|
121
|
+
##### Accessing Directives and Setting Parameter Values
|
100
122
|
|
101
|
-
Getting directives from a prompt is
|
123
|
+
Getting directives and keywords from a prompt is straightforward:
|
102
124
|
|
103
125
|
```ruby
|
104
|
-
prompt = PromptManager::Prompt.new(
|
105
|
-
prompt.keywords #=> an Array
|
126
|
+
prompt = PromptManager::Prompt.new(id: 'some_id')
|
127
|
+
prompt.keywords #=> an Array of keywords found in the prompt text
|
106
128
|
prompt.directives #=> an Array of entries like: ['directive', 'parameters']
|
107
129
|
|
130
|
+
# Or directly update the parameters hash
|
131
|
+
prompt.parameters = {
|
132
|
+
"[KEYWORD1]" => "value1",
|
133
|
+
"[KEYWORD2]" => "value2"
|
134
|
+
}
|
135
|
+
|
136
|
+
# After setting parameter values, call to_s to build the final prompt
|
108
137
|
# to_s builds the prompt by substituting
|
109
|
-
# values for keywords
|
138
|
+
# values for keywords and removing comments.
|
110
139
|
# The resulting text contains directives and
|
111
140
|
# prompt text ready for the LLM process.
|
112
|
-
|
141
|
+
final_text = prompt.to_s
|
142
|
+
|
143
|
+
# To persist any parameter changes to storage
|
144
|
+
prompt.save
|
113
145
|
```
|
114
146
|
|
115
147
|
The entries in the Array returned by the `prompt.directives` method is in the order that the directives were defined within the prompt. Each entry has two elements:
|
@@ -132,7 +164,7 @@ Since directies are collected after the keywords in the prompt have been substit
|
|
132
164
|
|
133
165
|
The `prompt_manager` gem only collects directives. Executing those directives is left up to some down stream process. Here are some ideas on how directives could be used in prompt downstream process:
|
134
166
|
|
135
|
-
- "//model gpt-5" could be used to set the LLM model to be used for a specific prompt.
|
167
|
+
- "//model gpt-5" could be used to set the LLM model to be used for a specific prompt.
|
136
168
|
- "//backend mods" could be used to set the backend prompt processor on the command line to be the `mods` utility.
|
137
169
|
- "//include path_to_file" could be used to add the contents of a file to the prompt.
|
138
170
|
- "//chat" could be used to send the prompts and then start up a chat session about the prompt and its response.
|
@@ -166,8 +198,8 @@ Use a `config` block to establish the configuration for the class.
|
|
166
198
|
|
167
199
|
```ruby
|
168
200
|
PromptManager::Storage::FileSystemAdapter.config do |o|
|
169
|
-
o.prompts_dir = "path/to/prompts_directory"
|
170
|
-
o.search_proc = nil # default
|
201
|
+
o.prompts_dir = "path/to/prompts_directory"
|
202
|
+
o.search_proc = nil # default
|
171
203
|
o.prompt_extension = '.txt' # default
|
172
204
|
o.params_extension = '.json' # default
|
173
205
|
end
|
@@ -177,7 +209,7 @@ The `config` block returns `self` so that means you can do this to setup the sto
|
|
177
209
|
|
178
210
|
```ruby
|
179
211
|
PromptManager::Prompt
|
180
|
-
.storage_adapter =
|
212
|
+
.storage_adapter =
|
181
213
|
PromptManager::Storage::FileSystemAdapter
|
182
214
|
.config do |config|
|
183
215
|
config.prompts_dir = 'path/to/prompts_dir'
|
@@ -192,7 +224,7 @@ An `ArgumentError` will be raised when `prompts_dir` does not exist or if it is
|
|
192
224
|
|
193
225
|
##### search_proc
|
194
226
|
|
195
|
-
The default for `search_proc` is nil which means that the search will be preformed by a default `search` method which is basically reading all the prompt files to see which ones contain the search term. It will return an Array of prompt IDs for each prompt file found that contains the search term. Its up to the application to select which returned prompt ID to use.
|
227
|
+
The default for `search_proc` is nil which means that the search will be preformed by a default `search` method which is basically reading all the prompt files to see which ones contain the search term. It will return an Array of prompt IDs for each prompt file found that contains the search term. Its up to the application to select which returned prompt ID to use.
|
196
228
|
|
197
229
|
There are faster ways to search and select files. For example there are specialized search and selection utilities that are available for the command line. The `examples` directory contains a `bash` script named `rgfzf` that uses `rg` (aka `ripgrep`) to do the searching and `fzf` to do the selecting.
|
198
230
|
|
@@ -267,7 +299,7 @@ The `PromptManager::Prompt` class expects an instance of a storage adapter class
|
|
267
299
|
|
268
300
|
```ruby
|
269
301
|
PromptManager::Prompt
|
270
|
-
.storage_adapter =
|
302
|
+
.storage_adapter =
|
271
303
|
PromptManager::Storage::ActiveRecordAdapter.config do |config|
|
272
304
|
config.model = DbPromptModel # any ActiveRecord::Base model
|
273
305
|
config.id_column = :prompt_name
|
data/Rakefile
CHANGED
@@ -0,0 +1,98 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# encoding: utf-8
|
3
|
+
# frozen_string_literal: true
|
4
|
+
# warn_indent: true
|
5
|
+
##########################################################
|
6
|
+
###
|
7
|
+
## File: directives.rb
|
8
|
+
## Desc: Demo of the PromptManager and FileStorageAdapter
|
9
|
+
## By: Dewayne VanHoozer (dvanhoozer@gmail.com)
|
10
|
+
##
|
11
|
+
#
|
12
|
+
|
13
|
+
param1 = "param_one"
|
14
|
+
param2 = "param_two"
|
15
|
+
|
16
|
+
|
17
|
+
class MyDirectives
|
18
|
+
def self.good_directive(*args)
|
19
|
+
puts "inside #{__method__} with these parameters:"
|
20
|
+
puts args.join(",\n")
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
def concept_break = print "\n------------------------\n\n\n"
|
25
|
+
|
26
|
+
require_relative '../lib/prompt_manager'
|
27
|
+
require_relative '../lib/prompt_manager/storage/file_system_adapter'
|
28
|
+
|
29
|
+
require 'amazing_print'
|
30
|
+
require 'pathname'
|
31
|
+
|
32
|
+
require 'debug_me'
|
33
|
+
include DebugMe
|
34
|
+
|
35
|
+
HERE = Pathname.new( __dir__ )
|
36
|
+
PROMPTS_DIR = HERE + "prompts_dir"
|
37
|
+
|
38
|
+
|
39
|
+
######################################################
|
40
|
+
# Main
|
41
|
+
|
42
|
+
at_exit do
|
43
|
+
puts
|
44
|
+
puts "Done."
|
45
|
+
puts
|
46
|
+
end
|
47
|
+
|
48
|
+
# Configure the Storage Adapter to use
|
49
|
+
PromptManager::Storage::FileSystemAdapter.config do |config|
|
50
|
+
config.prompts_dir = PROMPTS_DIR
|
51
|
+
# config.search_proc = nil # default
|
52
|
+
# config.prompt_extension = '.txt' # default
|
53
|
+
# config.parms+_extension = '.json' # default
|
54
|
+
end
|
55
|
+
|
56
|
+
PromptManager::Prompt.storage_adapter = PromptManager::Storage::FileSystemAdapter.new
|
57
|
+
|
58
|
+
# Use {parameter name} brackets to define a parameter
|
59
|
+
PromptManager::Prompt.parameter_regex = /\{[A-Za-z _|]+\}/
|
60
|
+
|
61
|
+
# Retrieve a prompt
|
62
|
+
prompt = PromptManager::Prompt.get(id: 'directive_example')
|
63
|
+
|
64
|
+
# Shows prompt without comments or directives
|
65
|
+
# It still has its parameter placeholders
|
66
|
+
puts prompt
|
67
|
+
concept_break
|
68
|
+
|
69
|
+
puts "Directives in the prompt:"
|
70
|
+
ap prompt.directives
|
71
|
+
|
72
|
+
puts "Processing directives ..."
|
73
|
+
prompt.directives.each do |entry|
|
74
|
+
if MyDirectives.respond_to? entry.first.to_sym
|
75
|
+
ruby = "MyDirectives.#{entry.first}(#{entry.last.gsub(' ', ',')})"
|
76
|
+
eval "#{ruby}"
|
77
|
+
else
|
78
|
+
puts "ERROR: there is no method: #{entry.first}"
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
concept_break
|
83
|
+
|
84
|
+
|
85
|
+
|
86
|
+
puts "Parameters in the prompt:"
|
87
|
+
ap prompt.parameters
|
88
|
+
puts "-"*16
|
89
|
+
|
90
|
+
puts "keywords:"
|
91
|
+
ap prompt.keywords
|
92
|
+
concept_break
|
93
|
+
|
94
|
+
prompt.parameters['{language}'] << 'French'
|
95
|
+
|
96
|
+
puts "After Substitution"
|
97
|
+
puts prompt
|
98
|
+
|
@@ -0,0 +1,47 @@
|
|
1
|
+
# lib/prompt_manager/directive_processor.rb
|
2
|
+
|
3
|
+
# This is an example of a directive processor class.
|
4
|
+
# It only supports the //include directive which is also
|
5
|
+
# aliased as //import.
|
6
|
+
|
7
|
+
module PromptManager
|
8
|
+
class DirectiveProcessor
|
9
|
+
EXCLUDED_METHODS = %w[ run initialize ]
|
10
|
+
|
11
|
+
def initialize
|
12
|
+
@prefix_size = PromptManager::Prompt::DIRECTIVE_SIGNAL.size
|
13
|
+
@included_files = []
|
14
|
+
end
|
15
|
+
|
16
|
+
def run(directives)
|
17
|
+
return {} if directives.nil? || directives.empty?
|
18
|
+
directives.each do |key, _|
|
19
|
+
sans_prefix = key[@prefix_size..]
|
20
|
+
args = sans_prefix.split(' ')
|
21
|
+
method_name = args.shift
|
22
|
+
|
23
|
+
if EXCLUDED_METHODS.include?(method_name)
|
24
|
+
directives[key] = "Error: #{method_name} is not a valid directive: #{key}"
|
25
|
+
elsif respond_to?(method_name)
|
26
|
+
directives[key] = send(method_name, *args)
|
27
|
+
else
|
28
|
+
directives[key] = "Error: Unknown directive '#{key}'"
|
29
|
+
end
|
30
|
+
end
|
31
|
+
directives
|
32
|
+
end
|
33
|
+
|
34
|
+
def include(file_path)
|
35
|
+
if File.exist?(file_path) &&
|
36
|
+
File.readable?(file_path) &&
|
37
|
+
!@included_files.include?(file_path)
|
38
|
+
content = File.read(file_path)
|
39
|
+
@included_files << file_path
|
40
|
+
content
|
41
|
+
else
|
42
|
+
"Error: File '#{file_path}' not accessible"
|
43
|
+
end
|
44
|
+
end
|
45
|
+
alias_method :import, :include
|
46
|
+
end
|
47
|
+
end
|