gpterm 0.4.2 → 0.6.1

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.
Files changed (6) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +18 -3
  3. data/config/prompts.yml +113 -0
  4. data/lib/client.rb +43 -115
  5. data/lib/gpterm.rb +67 -32
  6. metadata +17 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 4c034c1090bea2a09ffb99c7e44c5246a600ead24bff1eebcbfd200ef0c35008
4
- data.tar.gz: 300c572c96cb84a04cdff33a8f292a515de1deb0dfd28df0f3a4a0a2da3c63e2
3
+ metadata.gz: eefadbe4a618d7d95cef501fcf44e42db0487adb79d8e8461cd22cc6eefd655b
4
+ data.tar.gz: 523ac93f6e14d5298931aba6d93209cf179ad54f5af295846670ebcd1b2932fe
5
5
  SHA512:
6
- metadata.gz: ef3454943ee1cb69788e4365d23f7fae99acd8eaf2b5278041677cad1b6caa7465df9d099d8ac616c900fba58f4c23a8756a77e1f69a9c46a0b2ff67d3b344e6
7
- data.tar.gz: 067fe2e8b6405f8634fcc1d1d63bab4c66752280d472f0b9e59bfdaf102f338835b53827ca1d9d9438daabbe0872c35966c72c57bee5aa658922842ba9ff8184
6
+ metadata.gz: baff99e03558c78a074a43df8be57ea460be0852aea9e48fdc9eb11a5594bde329e92789f7de9d4158e85f06b26c2f13eff303aa9e484b2eef98c1c2361399c3
7
+ data.tar.gz: bf12e985975db7eef7bb533bae732a0a992d68d97d8af69257eb9b73f3ca71fc159f083138c56539a0fb84351e37acb3573d16e269b00b0f55ce2d1b5bbbb161
data/README.md CHANGED
@@ -1,7 +1,9 @@
1
- # gpterm
1
+ # gpterm: a natural language interface for your terminal
2
2
 
3
3
  **WARNING:** `gpterm` has very few guardrails. If used indiscriminately, it can wipe your entire system or leak information.
4
4
 
5
+ ![gpterm-fb-for-dogs](https://github.com/basicallydan/gpterm/assets/516325/7a5bed4f-6f41-4d0a-85d9-79fb071c1aaf)
6
+
5
7
  `gpterm` is a powerful, flexible and dangerous command-line tool designed to help you generate commands for your terminal using OpenAI's Chat Completions. It will not execute commands without your consent, but please do check which commands it is presenting before you let it execute them. Like so:
6
8
 
7
9
  ```bash
@@ -42,6 +44,15 @@ On first run, you'll be prompted to enter your OpenAI API key. This is required
42
44
  You can save and reuse preset prompts for common or repeated tasks. To create a preset, use the `-s` option followed by a name and the prompt, separated by a comma.
43
45
  To use a preset, simply pass its name as an argument when starting gpterm.
44
46
 
47
+ ### Example Presets
48
+
49
+ I have been using the following presets while developing `gpterm` and have found them to be quite helpful.
50
+
51
+ - `gc`: **Makes and pushes a commit describing the most recent changes.**
52
+ - Prompt: `"Commit all the latest changes to main, describing specifically what the changes are in the commit message. Please determine what the changes are before you write the commit message. You should use git diff and git status for information gathering"`
53
+ - `minorv`: **Does all the steps necessary in publishing a new version of this gem, including updating the changelog.**
54
+ - Prompt: `"Bump to a new minor version of this gem (change the y and reset z in version number x.y.z). That means you need to add a new section to the changelog with 1-5 bullet points summarising the changes (check git logs since the most recent git tag), then commit that, then create a new git tag, pushing the tag to the remote, update the gemspec, build the gem and publish it to rubygems. DO NOT try to add or commit any .gemspec files to the repo, or any other files which are listed in .gitignore"`
55
+
45
56
  ## Contributing
46
57
 
47
58
  Contributions are welcome! Feel free to open an issue or pull request.
@@ -50,6 +61,10 @@ Contributions are welcome! Feel free to open an issue or pull request.
50
61
 
51
62
  gpterm is open-source software licensed under the MIT license.
52
63
 
53
- ## Author
64
+ ## Credit
65
+
66
+ [Dan Hough](https://danhough.com)/@basicallydan built this application.
67
+
68
+ [Alex Rudall](https://github.com/alexrudall)/@alexrudall developed [ruby-openai](https://github.com/alexrudall/ruby-openai) around which this application was built.
54
69
 
55
- [Dan Hough](https://danhough.com)
70
+ [OpenAI](https://openai.com/)/@openai built a powerful API full of great AI models upon which this whole concept relies.
@@ -0,0 +1,113 @@
1
+ system: |
2
+ You are an AI running in a command-line application being executed inside of a directory on a user's computer. You are executed by running `gpterm` in the terminal or shell. You are an expert in POSIX-compliant command-line tools and utilities, and you are able to run any command that this system can run. You are also able to understand the output of those commands.
3
+
4
+ You are executed by running `gpterm` in the shell, and you are provided with a GOAL PROMPT.
5
+
6
+ The user's GOAL PROMPT will be a natural-language string which describes a goal that the user would like you to help them achieve by running some commands in their shell.
7
+
8
+ As part of the conversation, you will be given COMMAND PROMPTS and QUESTION PROMPTS. You must respond to each of these with the appropriate type of response.
9
+
10
+ # COMMAND PROMPTS
11
+
12
+ When you are given a prompt ending with "COMMANDS:", you MUST respond with either:
13
+ - One or more commands which can be executed on the shell
14
+ - An INSTRUCTION CODE which indicates to the application that no commands can be generated or are necessary
15
+
16
+ When responding with commands:
17
+ - The response MUST be a string containing one or more shell commands, separated by newlines, and nothing else
18
+ - The response MUST NOT contain any comments or extraneous information
19
+ - The response MUST NOT start with backticks, or end with backticks
20
+ - The response MUST keep in mind that each line of the response will be executed in the shell in a subshell, and the output of each command will be captured
21
+ - The commands MUST NOT contain any placeholders which the user is expected to replace with their own values
22
+ - If a command needs to be run in another directory, the command to change directory MUST be part of that command. To execute a command in a different directory, you must chain the cd command with the command you want to run, like so: `cd /path/to/directory && command`. You will need to do the same for any command that requires a different working directory, even if you have used cd in a previous command
23
+
24
+ When responding with an INSTRUCTION CODE indicating that no commands can be generated or are necessary:
25
+ - The response MUST start with $$ and end with $$, and be a single line with only alphanumeric characters and underscores
26
+ - The response MUST match one of the codes provided in the prompt
27
+
28
+ # QUESTION PROMPTS
29
+
30
+ When you are given a prompt ending with "QUESTION:", you MUST respond with either:
31
+ - A string which contains a question that the user can answer to provide you with some information you need to generate the commands to accomplish the goal
32
+ - An INSTRUCTION CODE which indicates to the application that no question can be generated or is necessary
33
+
34
+ When responding with a question:
35
+ - The response MUST be a string containing a question that gathers ONE piece of information. If you need multiple pieces of information, you can ask a follow-up question after the user responds to the one you are currently asking
36
+
37
+ When responding with an INSTRUCTION CODE indicating that no question can be generated or is necessary:
38
+ - The response MUST start with $$ and end with $$, and be a single line with only alphanumeric characters and underscores
39
+ - The response MUST match one of the codes provided in the prompt
40
+ info_gathering: |
41
+ Your FIRST response should be a list of commands that will be automatically executed to gather more information about the user's system. For this response, additonal formatting rules apply:
42
+ - The response MUST NOT contain any plain language instructions, and must not start with or end with backticks to indicate code.
43
+ - The commands MUST NOT make any changes to the user's system.
44
+ - The commands MUST NOT make any changes to any files on the user's system.
45
+ - The commands MUST NOT write to any files using the > or >> operators.
46
+ - The commands MUST NOT use the touch command.
47
+ - The commands MUST NOT use echo or any other command to write into files using the > or >> operators.
48
+ - The commands MUST NOT send any data to any external servers.
49
+ - The commands MAY gather information about the user's system, such as the version of a software package, or the contents of a file.
50
+ - The commands CAN pipe their output into other commands.
51
+ - The commands SHOULD tend to gather more verbose information INSTEAD OF more concise information.
52
+
53
+ This will help you to provide a more accurate response to the user's goal.
54
+
55
+ VALID example response. These commands are examples of commands which CAN be included in your FIRST response:
56
+
57
+ for file in *; do cat "$file"; done
58
+ which ls
59
+ which git
60
+ which brew
61
+ git diff
62
+ git status
63
+
64
+ INVALID example response. These commands are examples of commands which MUST NOT be included in your FIRST response:
65
+
66
+ touch file.txt
67
+ git add .
68
+ git push
69
+
70
+ Alternatively for this response, you may respond with one of the following INSTRUCTION CODES and NO OTHER INSTRUCTIONS CODES:
71
+ - $$cannot_compute$$ - You cannot create a VALID response to this prompt. The user will be asked to provide a new prompt.
72
+ - $$no_gathering_needed$$ - You do not need to gather more information. The next step will be executed.
73
+
74
+ Please note that in many cases, you will need to gather information by running commands.
75
+
76
+ If you need to gather information directly from the user, you will be able to do so in the next step by asking questions.
77
+ user_question: |
78
+ Before you provide the user with the next command, you have the opportunity to ask the user to provide more information so you can better tailor your response to their needs.
79
+
80
+ If you would like to ask the user for more information, please provide a prompt that asks the user for the information you need.
81
+
82
+ Alternatively for this response, you may respond with one of the following INSTRUCTION CODES and NO OTHER INSTRUCTIONS CODES:
83
+ - $$no_more_information_needed$$ - You do not need to gather any further information. The next step will be executed.
84
+
85
+ QUESTION:
86
+ goal_commands: |
87
+ Your NEXT response should be a list of commands that will be automatically executed to fulfill the user's goal. For this response, additonal formatting rules apply:
88
+ - The commands ARE ALLOWED to make changes to the user's system.
89
+ - The commands ARE ALLOWED to install new software using package managers like Homebrew
90
+
91
+ The following commands are FORBIDDEN in this response. Outputting ANY of these commands will result in a rejection of your response:
92
+ - rm -rf /
93
+ - sudo rm -rf /
94
+ - dd if=/dev/random of=/dev/sda
95
+ - :(){ :|: & };:
96
+ - chmod -R 777 /
97
+ - mkfs.ext4 /dev/sda1
98
+ - dd if=/dev/zero of=/dev/sda bs=1M count=1
99
+ - mv / /dev/null
100
+ - cat /etc/shadow
101
+ - echo "malicious_code" > ~/.bash_profile
102
+
103
+ VALID example response. These commands are examples of commands which CAN be included in your FINAL response:
104
+
105
+ ls
106
+ mkdir new_directory
107
+ brew install git
108
+ git commit -m "This is a great commit message"
109
+
110
+ Alternatively, you may respond with the following INSTRUCTION CODES:
111
+ - $$cannot_compute$$ - You cannot create a VALID response to this prompt. The user will be asked to provide a new prompt.
112
+
113
+ COMMANDS:
data/lib/client.rb CHANGED
@@ -1,159 +1,87 @@
1
1
  require "openai"
2
+ require 'yaml'
2
3
 
3
4
  class Client
4
- attr_reader :openapi_client
5
+ attr_reader :openai_client
5
6
  attr_reader :config
6
7
 
7
8
  def initialize(config)
8
9
  @config = config
9
- @openapi_client = OpenAI::Client.new(access_token: config["openapi_key"])
10
+ @openai_client = OpenAI::Client.new(access_token: config["openapi_key"])
11
+ @prompts = YAML.load_file(File.join(__dir__, '..', 'config', 'prompts.yml'))
10
12
  end
11
13
 
12
- def first_prompt(prompt)
13
- system_prompt = <<~PROMPT
14
- You are a command-line application being executed inside of a directory in a macOS environment, on the user's terminal command line.
15
-
16
- You are executed by running `gpterm` in the terminal, and you are provided with a prompt to respond to with the -p flag.
17
-
18
- Users can add a preset prompt by running `gpterm -s <name>,<prompt>`.
19
-
20
- The eventual output to the user would be a list of commands that they can run in their terminal to accomplish a task.
21
-
22
- You have the ability to run any command that this system can run, and you can read the output of those commands.
23
-
24
- However, any command which would ordinarily change the directory, such as cd, will not change the location of the directory in which you are running. To execute a command in a different directory, you must chain the cd command with the command you want to run, like so: `cd /path/to/directory && command`. You will need to do the same for any command that requires a different working directory, even if you have used cd in a previous command.
25
-
26
- The user is trying to accomplish a task using the terminal, but they are not sure how to do it.
27
- PROMPT
14
+ def first_prompt(user_goal_prompt)
15
+ system_prompt = @prompts["system"]
28
16
 
29
17
  if @config["send_path"]
30
18
  system_prompt += <<~PROMPT
19
+ # ADDITIONAL CONTEXT:
20
+
31
21
  The user's PATH environment variable is:
32
22
  #{ENV["PATH"]}
33
23
  PROMPT
34
24
  end
35
25
 
36
- full_prompt = <<~PROMPT
37
- Your FIRST response should be a list of commands that will be automatically executed to gather more information about the user's system.
38
- - The response MUST NOT contain any plain language instructions, and must not start with or end with backticks to indicate code.
39
- - The commands MUST NOT make any changes to the user's system.
40
- - The commands MUST NOT make any changes to any files on the user's system.
41
- - The commands MUST NOT write to any files using the > or >> operators.
42
- - The commands MUST NOT use the touch command.
43
- - The commands MUST NOT use echo or any other command to write into files using the > or >> operators.
44
- - The commands MUST NOT send any data to any external servers.
45
- - The commands MUST NOT contain any placeholders in angle brackets like <this>.
46
- - The commands MAY gather information about the user's system, such as the version of a software package, or the contents of a file.
47
- - The commands CAN pipe their output into other commands.
48
- - The commands SHOULD tend to gather more verbose information INSTEAD OF more concise information.
49
- This will help you to provide a more accurate response to the user's goal.
50
- Therefore your FIRST response MUST contain ONLY a list of commands and nothing else.
51
-
52
- VALID example response. These commands are examples of commands which CAN be included in your FIRST response:
53
-
54
- for file in *; do cat "$file"; done
55
- which ls
56
- which git
57
- which brew
58
- git diff
59
- git status
60
-
61
- INVALID example response. These commands are examples of commands which MUST NOT be included in your FIRST response:
62
-
63
- touch file.txt
64
- git add .
65
- git push
66
-
67
- If you cannot create a VALID response, simply return the string "$$cannot_compute$$" and the user will be asked to provide a new prompt.
68
- If you do not need to gather more information, simply return the string "$$no_gathering_needed$$" and the next step will be executed.
69
- You probably will need to gather information.
70
- If you need to gather information directly from the user, you will be able to do so in the next step.
71
-
72
- The user's goal prompt is:
73
- "#{prompt}"
74
- Commands to execute to gather more information about the user's system before providing the response which will accomplish the user's goal:
26
+ user_prompt = @prompts["info_gathering"]
27
+ user_prompt += <<~PROMPT
28
+ The user's GOAL PROMPT is:
29
+
30
+ "#{user_goal_prompt}"
31
+
32
+ Please respond with one or more commands to execute to gather more information about the user's system before providing the response which will accomplish the user's goal.
33
+
34
+ COMMANDS:
75
35
  PROMPT
76
36
 
77
37
  @messages = [
78
- { role: "system", content: system_prompt },
79
- { role: "user", content: full_prompt }
38
+ { role: "system", content: system_prompt }
80
39
  ]
81
40
 
82
- response = openapi_client.chat(
83
- parameters: {
84
- model: "gpt-4-turbo-preview",
85
- messages: @messages,
86
- temperature: 0.6,
87
- }
88
- )
89
- content = response.dig("choices", 0, "message", "content")
90
-
91
- @messages << { role: "assistant", content: content }
92
-
93
- content
41
+ continue_conversation(user_prompt)
94
42
  end
95
43
 
96
- def offer_information_prompt(prompt)
97
- full_prompt = <<~PROMPT
98
- This is the output of the command you provided to the user in the previous step.
99
-
100
- #{prompt}
101
-
102
- Before you provide the user with the next command, you have the opportunity to ask the user to provide more information so you can better tailor your response to their needs.
103
-
104
- If you would like to ask the user for more information, please provide a prompt that asks the user for the information you need.
105
- - Your prompt MUST ONLY contain one question. You will be able to ask another question in the next step.
106
- If you have all the information you need, simply return the string "$$no_more_information_needed$$" and the next step will be executed.
107
- PROMPT
44
+ def offer_information_prompt(previous_output, previous_output_type = :question_response)
45
+ question_prompt = if previous_output_type == :question_response
46
+ <<~PROMPT
47
+ This is the output of the question you asked the user in the previous step.
108
48
 
109
- @messages << { role: "user", content: full_prompt }
110
-
111
- response = openapi_client.chat(
112
- parameters: {
113
- model: "gpt-4-turbo-preview",
114
- messages: @messages,
115
- temperature: 0.6,
116
- }
117
- )
49
+ #{previous_output}
50
+ PROMPT
51
+ else
52
+ <<~PROMPT
53
+ This is the output of the command you provided to the user in the previous step.
118
54
 
119
- content = response.dig("choices", 0, "message", "content")
55
+ #{previous_output}
56
+ PROMPT
57
+ end
120
58
 
121
- @messages << { role: "assistant", content: content }
59
+ question_prompt += @prompts["user_question"]
122
60
 
123
- content
61
+ continue_conversation(question_prompt)
124
62
  end
125
63
 
126
64
  def final_prompt(prompt)
127
- full_prompt = <<~PROMPT
65
+ goal_commands_prompt = <<~PROMPT
128
66
  This is the output of the command you provided to the user in the previous step.
129
67
 
130
68
  #{prompt}
131
69
 
132
- Your NEXT response should be a list of commands that will be automatically executed to fulfill the user's goal.
133
- - The commands may make changes to the user's system.
134
- - The commands may install new software using package managers like Homebrew
135
- - The commands MUST all start with a valid command that you would run in the terminal
136
- - The commands MUST NOT contain any placeholders in angle brackets like <this>.
137
- - The response MUST NOT contain any plain language instructions, or backticks indicating where the commands begin or end.
138
- - THe response MUST NOT start or end with backticks.
139
- - The response MUST NOT end with a newline character.
140
- Therefore your NEXT response MUST contain ONLY a list of commands and nothing else.
70
+ PROMPT
141
71
 
142
- VALID example response. These commands are examples of commands which CAN be included in your FINAL response:
72
+ goal_commands_prompt += @prompts["goal_commands"]
143
73
 
144
- ls
145
- mkdir new_directory
146
- brew install git
147
- git commit -m "This is a great commit message"
74
+ continue_conversation(goal_commands_prompt)
75
+ end
148
76
 
149
- If you cannot keep to this restriction, simply return the string "$$cannot_compute$$" and the user will be asked to provide a new prompt.
150
- PROMPT
77
+ private
151
78
 
152
- @messages << { role: "user", content: full_prompt }
79
+ def continue_conversation(prompt)
80
+ @messages << { role: "user", content: prompt }
153
81
 
154
- response = openapi_client.chat(
82
+ response = openai_client.chat(
155
83
  parameters: {
156
- model: "gpt-4-turbo-preview",
84
+ model: @config["model"] || "gpt-4-turbo-preview",
157
85
  messages: @messages,
158
86
  temperature: 0.6,
159
87
  }
data/lib/gpterm.rb CHANGED
@@ -24,36 +24,66 @@ class GPTerm
24
24
  name = @options[:preset_prompt][0]
25
25
  prompt = @options[:preset_prompt][1]
26
26
  AppConfig.add_preset(@config, name, prompt)
27
- puts "Preset prompt '#{name}' saved with prompt '#{prompt}'".colorize(:green)
28
- exit
27
+ exit_with_message("Preset prompt '#{name}' saved with prompt '#{prompt}'", :green)
29
28
  elsif @options[:prompt]
30
- start_prompt(@options[:prompt])
29
+ start_conversation(@options[:prompt])
31
30
  end
32
31
  end
33
32
 
34
33
  private
35
34
 
36
- def execute_command(command)
35
+ def execute_shell_command(command)
37
36
  stdout, stderr, status = Open3.capture3(command)
38
37
  [stdout, stderr, status.exitstatus]
39
38
  end
40
39
 
41
- def start_prompt(prompt)
40
+ def exit_with_message(message, color = nil)
41
+ if color
42
+ puts message.colorize(color)
43
+ else
44
+ puts message
45
+ end
46
+
47
+ exit
48
+ end
49
+
50
+ # Ensures the user enters "y" or "n"
51
+ def get_yes_or_no
52
+ input = STDIN.gets.chomp.downcase
53
+ while ['y', 'n'].include?(input) == false
54
+ puts 'Please enter "y/Y" or "n/N":'.colorize(:yellow)
55
+ input = STDIN.gets.chomp.downcase
56
+ end
57
+ input
58
+ end
59
+
60
+ # Ensures the user enters a non-empty value
61
+ def get_non_empty_input
62
+ input = STDIN.gets.chomp.strip
63
+ while input.length == 0
64
+ puts 'Please enter a non-empty value:'.colorize(:yellow)
65
+ input = STDIN.gets.chomp.strip
66
+ end
67
+ input
68
+ end
69
+
70
+ def start_conversation(prompt)
42
71
  message = @client.first_prompt(prompt)
43
72
 
44
73
  if message.downcase == '$$cannot_compute$$'
45
- puts 'Sorry, a command could not be generated for that prompt. Try another.'.colorize(:red)
46
- exit
74
+ exit_with_message('Sorry, a command could not be generated for that prompt. Try another.', :red)
47
75
  end
48
76
 
49
77
  if message.downcase == '$$no_gathering_needed$$'
50
78
  puts 'No information gathering needed'.colorize(:magenta)
51
79
  output = "No information gathering was needed."
80
+ elsif message.downcase == '$$cannot_compute$$'
81
+ exit_with_message('Sorry, a command could not be generated for that prompt. Try another.', :red)
52
82
  else
53
83
  puts 'Information gathering command:'.colorize(:magenta)
54
84
  puts message.gsub(/^/, "#{" $".colorize(:blue)} ")
55
- puts 'Do you want to execute this command? (Y/n)'.colorize(:yellow)
56
- continue = STDIN.gets.chomp
85
+ puts 'Do you want to execute this command? (Y/n then hit return)'.colorize(:yellow)
86
+ continue = get_yes_or_no
57
87
 
58
88
  unless continue.downcase == 'y'
59
89
  exit
@@ -68,19 +98,19 @@ class GPTerm
68
98
  end
69
99
  end
70
100
 
71
- output = offer_more_information(output)
101
+ output = @client.offer_information_prompt(output, :shell_output_response)
72
102
 
73
103
  while output.downcase != '$$no_more_information_needed$$'
74
104
  puts "You have been asked to provide more information with this command:".colorize(:magenta)
75
105
  puts output.gsub(/^/, "#{" >".colorize(:blue)} ")
76
106
  puts "What is your response? (Type 'skip' to skip this step and force the final command to be generated)".colorize(:yellow)
77
107
 
78
- response = STDIN.gets.chomp
108
+ response = get_non_empty_input
79
109
 
80
110
  if response.downcase == 'skip'
81
111
  output = '$$no_more_information_needed$$'
82
112
  else
83
- output = offer_more_information(response)
113
+ output = @client.offer_information_prompt(response, :question_response)
84
114
  end
85
115
  end
86
116
 
@@ -88,12 +118,16 @@ class GPTerm
88
118
 
89
119
  message = @client.final_prompt(output)
90
120
 
121
+ if message.downcase == '$$cannot_compute$$'
122
+ exit_with_message('Sorry, a command could not be generated for that prompt. Try another.', :red)
123
+ end
124
+
91
125
  puts 'Generated command to accomplish your goal:'.colorize(:magenta)
92
126
  puts message.gsub(/^/, "#{" $".colorize(:green)} ")
93
127
 
94
- puts 'Do you want to execute this command? (Y/n)'.colorize(:yellow)
128
+ puts 'Do you want to execute this command? (Y/n then hit return)'.colorize(:yellow)
95
129
 
96
- continue = STDIN.gets.chomp
130
+ continue = get_yes_or_no
97
131
 
98
132
  unless continue.downcase == 'y'
99
133
  exit
@@ -102,12 +136,11 @@ class GPTerm
102
136
  commands = message.split("\n")
103
137
 
104
138
  commands.each do |command|
105
- stdout, stderr, exit_status = execute_command(command)
139
+ stdout, stderr, exit_status = execute_shell_command(command)
106
140
  if exit_status != 0
107
141
  puts "#{command} failed with the following output:".colorize(:red)
108
142
  puts "#{stderr.gsub(/^/, " ")}".colorize(:red) if stderr.length > 0
109
- puts " Exit status: #{exit_status}".colorize(:red)
110
- exit
143
+ exit_with_message(" Exit status: #{exit_status}", :red)
111
144
  end
112
145
  puts stdout if stdout.length > 0
113
146
  # I'm doing this here because git for some reason always returns the output of a push to stderr,
@@ -123,20 +156,29 @@ class GPTerm
123
156
  new_config = {}
124
157
  puts "Before we get started, we need to configure the application. All the info you provide will be saved in #{AppConfig::CONFIG_FILE}.".colorize(:magenta)
125
158
 
126
- puts "Enter your OpenAI API key's \"SECRET KEY\" value: ".colorize(:yellow)
127
- new_config['openapi_key'] = STDIN.gets.chomp
159
+ puts "Enter your OpenAI API key's \"SECRET KEY\" value then hit return: ".colorize(:yellow)
160
+ new_config['openapi_key'] = get_non_empty_input
128
161
 
129
162
  puts "Your PATH environment variable is: #{ENV['PATH']}".colorize(:magenta)
130
- puts 'Are you happy for your PATH to be sent to OpenAI to help with command generation? (Y/n) '.colorize(:yellow)
163
+ puts 'Are you happy for your PATH to be sent to OpenAI to help with command generation? (Y/n then hit return) '.colorize(:yellow)
164
+
165
+ input = get_yes_or_no
131
166
 
132
- if STDIN.gets.chomp.downcase == 'y'
167
+ if input == 'y'
133
168
  new_config['send_path'] = true
134
169
  else
135
170
  new_config['send_path'] = false
136
171
  end
137
172
 
173
+ default_model = 'gpt-4-turbo-preview'
174
+
175
+ puts "The default model is #{default_model}. If you would like to change it please enter the name of your preferred model:".colorize(:yellow)
176
+ new_config['model'] = STDIN.gets.chomp.strip || default_model
177
+
138
178
  AppConfig.save_config(new_config)
139
179
 
180
+ puts "Configuration saved to #{AppConfig::CONFIG_FILE}".colorize(:green)
181
+
140
182
  new_config
141
183
  else
142
184
  AppConfig.load_config
@@ -163,14 +205,12 @@ class GPTerm
163
205
  opts.banner = "gpterm config [--openapi_key <value>|--send_path <true|false>]"
164
206
  opts.on("--openapi_key VALUE", "Set the OpenAI API key") do |v|
165
207
  AppConfig.add_openapi_key(@config, v)
166
- puts "OpenAI API key saved"
167
- exit
208
+ exit_with_message("OpenAI API key saved")
168
209
  end
169
210
  opts.on("--send_path", "Send the PATH environment variable to OpenAI") do
170
211
  @config['send_path'] = true
171
212
  AppConfig.save_config(@config)
172
- puts "Your PATH environment variable will be sent to OpenAI to help with command generation"
173
- exit
213
+ exit_with_message("Your PATH environment variable will be sent to OpenAI to help with command generation")
174
214
  end
175
215
  end
176
216
  }
@@ -196,19 +236,14 @@ class GPTerm
196
236
  subcommands[command][:option_parser].parse!
197
237
  subcommands[command][:argument_parser].call(ARGV) if subcommands[command][:argument_parser]
198
238
  elsif command == 'help'
199
- puts main
200
- exit
239
+ exit_with_message(main)
201
240
  elsif command
202
241
  options[:prompt] = command
203
242
  else
204
243
  puts 'Enter a prompt to generate text from:'.colorize(:yellow)
205
- options[:prompt] = STDIN.gets.chomp
244
+ options[:prompt] = get_non_empty_input
206
245
  end
207
246
 
208
247
  options
209
248
  end
210
-
211
- def offer_more_information(output)
212
- output = @client.offer_information_prompt(output)
213
- end
214
249
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: gpterm
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.4.2
4
+ version: 0.6.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Dan Hough
@@ -24,6 +24,20 @@ dependencies:
24
24
  - - "<"
25
25
  - !ruby/object:Gem::Version
26
26
  version: '7'
27
+ - !ruby/object:Gem::Dependency
28
+ name: colorize
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
27
41
  description: gpterm is a powerful, flexible and dangerous command-line tool designed
28
42
  to help you generate commands for your terminal using OpenAI's Chat Completions
29
43
  email:
@@ -36,6 +50,7 @@ files:
36
50
  - LICENSE
37
51
  - README.md
38
52
  - bin/gpterm
53
+ - config/prompts.yml
39
54
  - lib/client.rb
40
55
  - lib/config.rb
41
56
  - lib/gpterm.rb
@@ -53,7 +68,7 @@ required_ruby_version: !ruby/object:Gem::Requirement
53
68
  requirements:
54
69
  - - ">="
55
70
  - !ruby/object:Gem::Version
56
- version: '0'
71
+ version: 2.6.0
57
72
  required_rubygems_version: !ruby/object:Gem::Requirement
58
73
  requirements:
59
74
  - - ">="