Git-like version control for your LLM prompts.
Save prompt versions, view history, compare diffs, and roll back — all from Python, with zero external dependencies.
LLM prompts change constantly. Tracking what changed, when, and which version performed better quickly becomes a mess.
promptvero solves this:
- Every change is automatically versioned — v1, v2, v3...
- Roll back to any version instantly
- Pin a specific version as your main (canonical) prompt
- See exactly what changed between two versions
- Zero dependencies, zero configuration
pip install promptverofrom promptvero import Prompt
p = Prompt("system")
p.save("You are a helpful assistant")
p.save("You are an expert analyst. Be concise and data-driven.")
for entry in p.log():
print(entry["version"], entry["timestamp"], entry["is_main"])
# Pin the first version as your stable main prompt
p.set_main("v1")
# Always get the pinned version, regardless of how many new versions exist
content = p.get_main()
# See what changed between two versions
print(p.changes("v1", "v2"))The pv command lets you use promptvero directly from the terminal — no Python script needed. Useful for quickly saving files, inspecting version history, or integrating into shell scripts and CI/CD pipelines.
pv list # list all saved prompts
pv save <name> <file> # save a file as a new version
pv log <name> # show version history
pv show <name> [version] # show prompt content
pv changes <name> [v1] [v2] # show what changed between versions
pv set-main <name> <version> # mark a version as main
pv get-main <name> # print the main version content
pv delete <name> <version> # delete a specific version
pv delete-prompt <name> # delete a prompt and all its versionsUse --base-dir to specify a custom storage directory:
pv --base-dir ./prompts log my-prompt| Parameter | Type | Description |
|---|---|---|
name |
str |
Unique identifier for this prompt. |
base_dir |
str |
Root directory for storage. Defaults to .promptvero. |
| Parameter | Type | Description |
|---|---|---|
content |
str |
The prompt text to persist. |
Returns: Version string (e.g. "v1").
| Parameter | Type | Description |
|---|---|---|
filepath |
str |
Path to the file to read. Any plain text file is accepted. |
Returns: Version string.
Raises: PromptNotFoundError if the file does not exist.
| Parameter | Type | Description |
|---|---|---|
version |
str | None |
Version to retrieve (e.g. "v2"). If None, returns the latest. |
Returns: Prompt text.
Returns the full version history of this prompt, ordered from oldest to newest.
Returns: List of dicts with keys version (str), timestamp (str), and is_main (bool).
Marks a specific version as the main (canonical) version. The main version does not change automatically when new versions are saved — only when set_main() is called again.
| Parameter | Type | Description |
|---|---|---|
version |
str |
Version to mark as main (e.g. "v2"). |
Raises: VersionNotFoundError if the version does not exist.
Returns the full content of the version marked as main. Use this to always retrieve your stable, pinned prompt regardless of how many new versions have been saved since.
Returns: Prompt text of the main version.
Raises: PromptNotFoundError if no main version has been set.
Permanently deletes this prompt and all its versions. Removes all version files, history, and the main pointer.
Raises: PromptNotFoundError if the prompt does not exist.
Permanently deletes a specific version. Removes the version file and its history entry. If the deleted version was marked as main, the main pointer is also cleared.
| Parameter | Type | Description |
|---|---|---|
version |
str |
Version string to delete (e.g. "v2"). |
Raises: VersionNotFoundError if the version does not exist.
Returns the raw line-by-line difference between two versions as a dict. Use this when you want to process the result programmatically. For a human-readable report, use changes() instead.
| Parameter | Type | Description |
|---|---|---|
v1 |
str |
Base version (e.g. "v1"). |
v2 |
str |
Target version (e.g. "v2"). |
Returns: Dict with keys added, removed, unchanged — each a list of strings.
Returns a human-readable change report between two versions, showing which lines were added and which were removed. If no arguments are given, compares the last two saved versions. For raw diff data, use diff() instead.
| Parameter | Type | Description |
|---|---|---|
v1 |
str | None |
Base version. If None, uses second-to-last. |
v2 |
str | None |
Target version. If None, uses latest. |
Returns: Formatted string showing added and removed lines.
Returns the full content of a version with a formatted header. Displays a [main] tag if the version is the pinned main.
| Parameter | Type | Description |
|---|---|---|
version |
str | None |
Version to display. If None, shows latest. |
Returns: Formatted string with header and prompt content.
Returns the names of all saved prompts in the given base directory.
Returns: Sorted list of prompt name strings.
Every Prompt writes to a .promptvero/ folder in your working directory.
.promptvero/
system/
v1.txt
v2.txt
history.json
main.json
history.json format:
[
{
"version": "v1",
"timestamp": "2024-01-01T10:00:00"
},
{
"version": "v2",
"timestamp": "2024-01-01T11:00:00"
}
]main.json format:
{
"version": "v1"
}Version numbering is automatic: each save() call increments the counter. Versions are never overwritten — only explicitly deleted via delete_version().
MIT