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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 91badda19121014eee4d6c28b1cb882347d951af971d9afb021b00e558cb3fdc
4
- data.tar.gz: 500c401e8ced2be25adacaf374a6e18e73c42ec811e7bb632155de9d5626370c
3
+ metadata.gz: 267e48a65e377f2fc3390d3bbff69a6e1f241c96951fc86437f5ab02d2e93daa
4
+ data.tar.gz: db8af7d8515e72bf12f847e3544a7413a0acb0707a8321bcae7f13e1ffa1ff86
5
5
  SHA512:
6
- metadata.gz: 89c274000ee45ddb047eb33ab232d0e300fab51be512b1181e6d9d9dbf6abe5213013229bfd0450e948b78b09aa00aab882131ed066c7529e27c5865b8251647
7
- data.tar.gz: 4fc94add6f43ab7b0f289849e68c3c6c9f35ca24a90983a459cd2ea9043de03b8b2f9ed6701a3a614be7efecd1fc5557ab0fb0d252e84dc628d38f9f80d2d9ba
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
- ## [0.4.1] = 2023-12-29
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
- ## [0.4.0] = 2023-12-19
19
+ ### [0.4.0] = 2023-12-19
6
20
  - Add "//directives param(s)" with keywords just like the prompt text.
7
21
 
8
- ## [0.3.3] = 2023-12-01
22
+ ### [0.3.3] = 2023-12-01
9
23
  - Added example of using the `search_proc` config parameter with the FileSystemAdapter.
10
24
 
11
- ## [0.3.2] = 2023-12-01
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
- ## [0.3.0] = 2023-11-28
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
- ## [0.2.0] - 2023-11-21
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
- ## [0.1.0] - 2023-11-16
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
- - [prompts_dir](#prompts_dir)
27
- - [search_proc](#search_proc)
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
- - [id_column](#id_column)
36
- - [text_column](#text_column)
37
- - [parameters_column](#parameters_column)
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 somthing 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.
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
- The current hard-coded REGEX for a [KEYWORD] identifies any all [UPPERCASE_TEXT] enclosed in square brackets as a keyword. [KEYWORDS CAN ALSO HAVE SPACES] as well as the underscore character.
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
- This is just the initial convention adopted by prompt_manager. It is intended that this REGEX be configurable so that the prompt_manager can be used with other conventions.
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 as easy as getting the kewyords:
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 amd removing comments.
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
- puts prompt.to_s
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
@@ -8,7 +8,6 @@ end
8
8
 
9
9
  Tocer::Rake::Register.call
10
10
 
11
-
12
11
  require "bundler/gem_tasks"
13
12
  require "rake/testtask"
14
13
 
@@ -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
+
@@ -10,9 +10,6 @@
10
10
  ##
11
11
  #
12
12
 
13
- require 'debug_me'
14
- include DebugMe
15
-
16
13
  require 'prompt_manager'
17
14
  require 'prompt_manager/storage/file_system_adapter'
18
15
 
@@ -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