A Sublime Text plugin to interact with HTTP API endpoints and perform text manipulation with their responses. AssistantAI is bundled with OpenAI to interact with ChatGPT or Codex, Gitea to create Issues directly from the selected code, and Python prompts to automatically add Docstrings and more. The user can extend it's features by defining Server endpoints and custom Prompts in Sublime Settings JSON files.
AssistantAI provides general purpose implementation of an HTTP API client that allows perform text manipulation with remote API responses, as a result to requests based on selected text and optional user input. Can be extended by the user or by other packages, providing servers, endpoints, prompt templates and credentials placeholders, defined in Sublime Settings files, in JSON.
A common use case is to consume API of, for instance, Generative AIs like ChatGPT or OpenAI Codex, to complete, edit, comment or explain selected code chunks. Another example is to consume GitHub, Gitea or similar VCS server's API to add Issues from the selected code, without leaving Sublime Text.
Selected code, file path, name, file size and symbols may be sent to the remote third party API servers for processing (i.e.: OpenAI, see below), so make sure you have the necessary permissions to do so.
The plugin is available in Package Control.
Installing the plugin via Package Control is the recommended option:
- Invoke Command Palette (
Cmd
+Shift
+P
) - Search for "Package Control: Install Package"
- Search for "AssistantAI" and install it
Once installed, check the settings menu in Settings
> Package Settings
> AssistantAI
, which includes:
Settings
: General AssistantAI plugin settings. A place to add your custom prompts and servers.Key Bindings
: Example Key Bindings for invoking AssistantAI prompt selection panel and more.
Also bundled with the plugin you can configure:
OpenAI Server Settings
: Definition for ChatGPT completion API endpoints and general prompts.Gitea Server Settings
: Definition for Gitea Issue creation endpoint.Python Prompts
: Definition of Python-specific prompts (i.e.: to add a Doctring to selected code using OpenAI).
And for when you have at least one configured Server (see below):
- The
AssistantAI
command in the command palette.
AssistantAI is configured, and extended, using Sublime JSON settings files. The plugin provides several files for each service or syntax specific definitions (i.e.: one for OpenAI, another for Gitea, for Python, and more to come), where each file follows the same structure:
credentials
: Empty by default. You should overwrite this entry with your credentials for that server as described below.prompts
: Empty by default. Overwrite it to add (or import and refine) your custom prompts.servers
: Empty by default. Overwrite it to add (or import and refine) your custom servers.default_prompts
: Prompts provided by the package. You should not overwrite this entry.default_servers
: Servers provided by the package. You should not overwrite this entry.
Your custom settings files may include credentials
to enable the server, thus making all prompts depending on it available for when you invoke AssistantAI command. If you add custom prompts or servers, then prompts
and servers
may be present too in your custom files.
What should be never present in your custom files are
default_prompts
anddefault_servers
. If you want to create a server based on a default one, you can use aimport
as described in advanced topics or in the examples in the settings files.
The files are named following this convention:
assitant_ai.sublime-settings
for the general settings.assitant_ai_{NAME}.sublime-settings
for the service or syntax specific settings files.
Currently, at the very least, you should configure the API KEY of OpenAI, or API TOKEN of Gitea, to play with the plugin.
Everything can be configured with Sublime JSON settings with a very high degree of flexibility.
An example of key bindings file is provided to bind keyboard shortcuts to AssistantAI commands, so you can quickly get the available prompt list based on the current context or directly invoke a prompt.
There are two commands you can invoke with a Key Binding:
-
AssistnatAI Prompt
: identified asassistant_ai_prompt
. This command is also available in the Command Palette. When invoked without arguments shows a list of prompts that can be used, given the configured servers, selected text, buffer syntax, etc (i.e.: the context). To directly invoke a prompt, you can pass arguments:pid
: Prompt IDeid
: Endpoint ID...
: Any required input argument required by the prompt.
-
AssistnatAI Dump
: identified asassistant_ai_dump
. Used for debugging. Dumps all loaded settings and displays in a new buffer, so you can inspect what is being loaded. When you start playing withimport
for servers and prompts, you may need to know what has been processed by AsistantAI.
If no prompts are available, AssistantAI
command does nothing else than show a status bar message warning.
Once installed and configured, invoke AssistantAI
from the command palette, select a prompt from the presented list and follow it's flow.
Prompts are offered based on the context (configured servers, selected text, buffer syntax, available content, etc.), and as per specifications provided by the prompt it self.
Without any configured server, there will be no prompts available.
The general flow is simple. For instance:
- Select a text region (i.e.: A code function, or Markdown section).
- Command Palette >
AssistantAI
(or the keyboard shortcut if you set up one). - Select a prompt.
- If the prompt needs additional inputs, those will be requested using the UI.
- When several servers with valid endpoints qualifies for the selected prompt, a list of endpoints is presented.
Once AssistantAI have a prompt, with all needed inputs and the target endpoint is selected, it builds an HTTP payload based on that and makes the network request.
The response is parsed based on the endpoint specification and its specified command is executed.
The prompt may specify another action instead of text manipulation (replace, append, prepend, insert), such as show the response in an Output Panel, or a new Buffer.
As an example, consider this flow using the bundled OpenAI server endpoints definition (requires credentials) plugin while editing a Python file:
- Select a python function
- Ask AssistantAI for the prompt "Add python docstring"
- Since only one endpoint qualifies, a request is made to OpenAI without further inputs
- The python docstring is added to the selected function as returned by ChatGPT
The plugin implements the parser of the settings files where the servers, prompts and credentials are configured (by this package, another package, or the user). Manages the filtering of the prompts that are available given a context, and takes care of the API requests, building payloads, parsing responses, and performing the text manipulation.
Besides the plugin implementation, AssistantAI includes the following definitions of servers and prompts.
Servers are definitions of network resources that AssistantAI can consume using HTTP requests. Includes the URL, timeout and a description.
Each server must specify one or more endpoints which includes what are the possible request payloads structures, declarative instructions on how to build it based on the prompt user inputs (i.e.: selected text, or additional inputs the user is prompted for).
AssistantAI reads the configuration seeking for servers definitions, and considers a server to be available when all required credentials are configured by the user (i.e.: API TOKENS).
AssisntantAI includes currently the following server and endpoints definitions.
OpenAI server definition that allows consuming the API with three end points:
- Edits
- Completions
- Chat completions (the one powering ChatGPT)
Please note that OpenAI is a paid service, and you will need to have an API key and sufficient credit to use the plugin.
Once you have your API key, go to Settings
> Package Settings
> AssistantAI
> OpenAI Server Settings
and add the credential:
{
"credentials": {
// you must specify the Server ID ('openai' in this case) which this credential is for
"openai": {
// OpenAI requires only one credential named 'api_key'
"api_key": "sk-..."
// Other servers may require more tokens, identification, etc.
// For custom TLS: you can specify `verify` for a custom CA and `cert` for a Client Cert
// "verify": "~/.tls/my_ca.pem"
// "cert": "~/.tls/my_cert.pem" // must include unencrypted key
}
}
}
Gitea is an open source replacement of GitHub, less sophisticated and much lightweight, which you can deploy in your own infrastructure. It provides an API to interact with it, allows among many other workflows, to retrieve repository information, create Issues and PRs.
This Server definition includes two endpoints:
- Get repositories
- Create an Issue to a specific repository
Since you may have access to several Gitea instances (i.e.: your personal, and the one in your workplace), you can define your custom servers
by importing the default one:
"servers": [
{
"id": "gitea_personal",
"import": "gitea", // this is provided in 'default_servers'
"name": "Gitea Kanutron",
"url": "https://CUSTOM_URL:443", // the default points to https://localhost:3000
},
{
"id": "gitea_work",
"import": "gitea",
"name": "Gitea Work",
"url": "https://CUSTOM_URL:443",
},
],
And provide credentials for each of them as identified by it's id
:
{
"credentials": {
"gitea_personal": {
"token": "...",
},
"gitea_work": {
"token": "...",
"verify": "~/.gitea/secrets/rootca.pem",
"cert": "~/.gitea/secrets/user.pem",
},
}
}
Prompts are request templates that the user must fill with the variables when editing using Sublime Text. It's a straight forward process.
The required variables are quite flexible and typically includes text
, representing the selected text, if any.
Once the user invokes a prompt, it must resolve all required inputs if not yet solved automatically.
A request is then build and send to the available endpoint, or the one selected by the user if more than one is available.
For instance, if only text
is required by the prompt, the user must have selected text for the prompt to be usable.
If the prompt is limited to syntax
(i.e.: Python), the current buffer must be from that required syntax.
Available prompts out of the box with AssistantAI includes the following.
A set of generic prompts are provided if OpenAI server is enabled (i.e.: credentials are configured):
- Continue text from end of selection
- Ask to make a change on selection
- Ask to make a change on selection using Chat end point
These prompts requires at least one server endpoint that accepts text
as input, or text
+ instruction
.
A set of prompts commonly used while editing python code are bundled in AssistantAI.
- Add docstring: selecting a python function and invoking this prompt, a request will result in the python function being populated with a
docstring
. - Add comments: will comment line by line the selected python code.
- Explain selected text: will open the output panel with a verbose explanation of the selected code.
Those prompts requires at least one server that accepts text
as input. If you have OpenAI server enabled, you will be able to send this prompts there.
There is an example prompt in the settings files that is commented. It's an example of what you can achieve by fine-tuning the prompts.
{
// imported or created prompts specifications
"prompts": [
{
// Importing from another prompt
"import": "python_add_comments",
// Replacing name, icon and description
"name": "Add funny comments",
"icon": "🤣",
"description": "Add funny comments to selected ${syntax} code.",
// The generated text now adds an instruction to get funny
"vars": {
"text": [
"Comment the lines to the following ${syntax} code following theses rules:",
"* Return only the edited ${syntax} code.",
"* Do not alter the code.",
"* All comments must be written in a funny style, addressed to my future me.",
"",
"${text}",
],
},
},
]
}
There are two prompts (featuring prompt chaining) provided by the Gitea specifications:
- Get repo data: it's set as
visible
:false
since it is used to collect a list of available repositories. - Post Issue: this prompt requires an input that is the full name of a repository where an issue will be created. In order to select the repository, the previous prompt is invoked first. This demonstrates how prompts can be chained.
The main usage is to select a text and invoke AssitantAI to create a Gitea Issue. Then, a list of repositories is provided. After selecting the target repository, a Title and a Body are requested to the user and the issue is created.
Since the API from Gitea is similar to GitHub, GitLab and similar services, implementing the same flow for other providers should be a rather easy task now. Contributions welcome!
You can add your own servers using JSON settings (i.e.: Settings
> ... > AssistantAI
> Settings
), but the intended use is to install other Sublime Text plugins that provides complex and reusable server and prompt specifications.
Better documentation should be written to support potential contributors.
This plugin uses 3 types of specifications.
- Servers, with nested Endpoints
- Credentials
- Prompts, with nested PromptInputs
A server is a JSON specification that includes the URL, the needed headers and required credentials keys, and a set of endpoints.
{
"id": "openai",
"name": "OpenAI",
"url": "https://api.openai.com:443",
"timeout": 60,
"required_credentials": ["api_key"],
"headers": {
"Authorization": "Bearer ${api_key}",
"Content-Type": "application/json",
"cache-control": "no-cache",
},
"endpoints": { ... }
},
If the server specifies a required credential (like api_key
in this case), and this credential is not configured by the user, the server will be not available. Any prompt that explicitly requires endpoints of this server will be unavailable.
Headers to be sent to the server may include the credentials configured by the user. They will be expanded by Sublime Text when creating the HTTP request.
Specification of the request and expected response. It is included in the endpoints
key of a server specification.
Each server may provide one or more endpoints.
request
key provides the JSON object to be built by AssistantAI to send the request to the server's endpoint.
response
specifies two keys:
error
: for the key where any error will be retrievedoutput
: the path (forward slashes/
as a separator) where to retrieve the text
{
"chat_completions": {
"name": "Chat Completions",
"method": "POST",
"resource": "/v1/chat/completions",
"required_vars": ["text"],
"valid_params": {
"model": "string",
"messages": "string",
...
"user": "string",
},
"request": {
"model": "gpt-3.5-turbo",
"messages": [
{
"role": "user",
"content": "${text}",
}
],
},
"response": {
"error": "error",
"output": "choices/0/message/content",
},
}
}
They are key-value pairs used by server endpoints. Configured on each plugin settings file.
If you install AssistantAI and intent to use OpenAI API, you will have to setup this:
{
"credentials": {
"server_id": {
"api_key": "sk-..."
}
},
}
The key api_key
is later used by the server with specified id
as a variable that will be expanded in a HTTP header.
Each server specification will require the user to set up the credentials in its user-defined settings in order to enable the server.
A prompt is a configured set of inputs and variables needed to build a request based on user text selection and additional inputs.
Prompts are context aware, accounting for selected text, available pre- and post-text, syntax of the current buffer, and configured server endpoints.
If a prompt declares that needs an endpoint, that prompt will be shown only if that endpoint is available.
Similarly if a prompt declares is valid for python
syntax, will be shown only when editing a python file.
When a prompt requires text
input, and no selection is made, that prompt will not be shown.
{
"id": "continue_selected_text",
"name": "Continue selected text",
"description": "Given a selected text, continue writing from there.",
"required_inputs": ["text"],
"required_endpoints": [
"openai/completions",
"openai/chat_completions"
],
"params": {
"temperature": 1.0,
"max_tokens": 1800,
},
"command": "append",
}
Once a prompt is executed and a response is obtained, a command is executed as per the prompt specification.
Command can be;
replace
the entire selectionappend
at the end of selectionprepend
at the beginning of selectioninsert
replacing a placeholderoutput
to a new output (bottom) panelcreate
a new buffer with the response
If you want to contribute, feel free to open an Issue or send your PR.
There are four types of valuable contributions:
- Share your experience and creativity by opening Issues with bug reports and feature requests.
- Send PR to improve or fix the code. Ideally, as a response of an open issue.
- Add Servers, Prompts providing a specific and working
assistant_ai_{NAME}.sublime-settings
file. Ideally using a PR. - Developing a Sublime Text plugin (or updating your current one) that provides a
assistant_ai_{NAME}.sublime-settings
.
For an example assistant_ai_{NAME}.sublime-settings
check the OpenAI settings and Gitea settings files that includes prompts and server endpoints specifications consuming almost the entire AssitantAI implemented API.
The code is pretty much tidy now. But there are some missing features like:
- Proper documentation for AssistantAI plugin developers
- Testing Sublime Text versions other than
4143 macOS
- Implementing super cool prompts for Markdown, Java, Rust, ...
- Support multi-selection
- Provide current git repo information in the prompt context
- Add support to Todoist API to create tasks based on selected texts
- Add support to GitHub
- Add support to GitLab
- Add support to Dalai
- Add support to 🤗 Hugginface
- Implementing other plugins such as Gitea and similar to interact with their APIs. (DONE)
- Import statement for Server specifications (DONE)
- Improving Quick Panel inputs (DONE)
- JSON schema for Server and Prompts (DISMISSED)
This software is released under MIT license.
This is my first Sublime Text plugin and probably is full of bugs.
This plugin is complex to setup. Once properly done, the usage is straight forward though. A good documentation is key for increasing adoption.
My twitter accounts is @kanutron, and although I'm not super active there, I receive push notification on DM.
I been learning to code Sublime Text plugins by:
- Reading the code from CodexAI plugin from OpenAI
- Reading the code from https://github.com/yaroslavyaroslav/OpenAI-sublime-text/tree/master
- Interacting with ChatGPT (using this plugin!)
- Reading the docs from Sublime API reference and Package Control docs.