A local-first privacy layer for Large Language Models.
Project description
PromptMask
A local-first privacy layer for Large Language Models.
Cloud AI is smart but sacrifices privacy.
Local AI keeps your secret but is dumb.
What if we can combine the advantages of both sides?
PromptMask ensures your private data never leaves your machines. It redacts and un-redacts sensitive data locally, so that only anonymized data is sent to third-party AI services.
Table of Contents
- Table of Contents
- How It Works
- Quickstart
- Configuration
- Advanced Usage: PromptMask Class
- Web Server: WebUI & API
- Development & Contribution
- License
How It Works
The core principle is to use a trusted (local) model as a "privacy filter" for a powerful, remote model. The process is fully automated.
Quickstart
Prerequisites
Ensure you have a local LLM running with an OpenAI-compatible API endpoint. Ollama is a popular and straightforward option. By default, PromptMask will attempt to connect to http://localhost:11434/v1.
Other options to run a local OpenAI-compatible LLM API include llama.cpp and vLLM.
For General Users: local OpenAI-compatible API Gateway
You can point any existing tool or application at the local gateway. It's the seamless way to add PromptMask layer without coding in Python.
-
Install promptmask-web via pip:
pip install "promptmask[web]"
-
Run the web server:
promptmask-web
The gateway is now running at
http://localhost:8000. -
Use the gateway endpoint: Simply replace the official OpenAI API base URL with the local gateway's URL in your tool of choice.
For example, using
curl:curl http://localhost:8000/gateway/v1/chat/completions \ -H "Authorization: Bearer $YOUR_OPENAI_API_KEY" \ -H "Content-Type: application/json" \ -d '{ "model": "gpt-99-ultra", "messages": [ { "role": "user", "content": "My name is Ho Shih-Chieh and my appointment ID is Y1a2e87. I booked a dental appointment on Oct 26, but I have to cancel for a meeting. Please help me write a cancellation request email in French." } ] }'
Your sensitive data (
Ho Shih Chieh,Y1a2e87) will be redacted before being sent to OpenAI, and then restored in the final response.If you are using other cloud AI providers, such as Google Gemini, you need to add
web.upstream_oai_api_baseto your config file (more detail on configuration section)[web] upstream_oai_api_base = "https://generativelanguage.googleapis.com/v1beta/openai"
For Python Developers: OpenAIMasked
The OpenAIMasked class is a drop-in replacement for the standard openai.OpenAI client.
-
Install the base package:
pip install promptmask
-
Mask the OpenAI client in your code: The adapter automatically handles masking/unmasking for standard and streaming requests.
Simply replace the standard
openai.OpenAIclient as follows:# from openai import OpenAI from promptmask import OpenAIMasked # client = OpenAI() client = OpenAIMasked()
Full example:
from promptmask import OpenAIMasked # This client has the same interface as openai.OpenAI, but with automatic privacy redaction. client = OpenAIMasked(base_url="https://api.cloud-ai-service.example.com/v1") # reads OPENAI_API_KEY from environment variables by default. # --- Standard Request --- response = client.chat.completions.create( model="gpt-100-pro", messages=[ {"role": "user", "content": "My user ID is johndoe and my phone number is 4567890. Please help me write an application letter."} ] ) # The response content is automatically unmasked. print(response.choices[0].message.content) # --- Streaming Request --- stream = client.chat.completions.create( model="gpt-101-turbo-mini", stream=True, messages=[ {"role": "user", "content": "My patient, Jensen Huang (Patient ID: P123456789), is taking metformin and is experiencing nausea. What are the common side effects and management strategies?"} ] ) # The stream chunks are unmasked on-the-fly. for chunk in stream: print(chunk.choices[0].delta.content or "", end="")
Configuration
To customize, create a promptmask.config.user.toml file in your working directory. For example, to change the categories of data to mask:
# promptmask.config.user.toml
[llm_api]
# Specify a particular local model to use for masking
model = "qwen2.5:7b"
# Define what data is considered sensitive.
[sensitive]
# Override the default one
include = "personal ID and passwords"
Check promptmask.config.default.toml for a full config file example.
Environment variables can also be used to override specific settings:
LOCALAI_API_BASE: The Base URL for your local LLM's API (e.g.,http://192.168.1.234:11434/v1).LOCALAI_API_KEY: The API key for your local LLM, if required.
Configuration Priority Hierarchy
PromptMask is configured through a hierarchy of sources, from highest to lowest priority:
LOCALAI_API_BASEandLOCALAI_API_KEYenvironment variables.- A
dictpassed directly to thePromptMaskconstructor (configparameter). - A path to a TOML file (
config_fileparameter). - A
promptmask.config.user.tomlfile in the current working directory. - The packaged
promptmask.config.default.toml.
Advanced Usage: PromptMask Class
For more granular control, you can import the PromptMask class directly to perform masking and unmasking as separate steps.
import asyncio # PromptMask also runs syncrounously
from promptmask import PromptMask
async def main():
masker = PromptMask()
original_text = "Please process the visa application for Jensen Huang, passport number A12345678."
# 1. Mask the text
masked_text, mask_map = await masker.async_mask_str(original_text)
print(f"Masked Text: {masked_text}")
# Expected output (may vary): Masked Text: Please process the visa application for ${PERSON_NAME}, passport number ${PASSPORT_NUMBER}.
print(f"Mask Map: {mask_map}")
# Expected output: Mask Map: {"Jensen Huang": "${PERSON_NAME}", "A12345678": "${PASSPORT_NUMBER}"}
# (Imagine sending masked_text to a remote API and getting a response)
remote_response_text = "The visa application for ${PERSON_NAME} with passport ${PASSPORT_NUMBER} is now under review."
# 2. Unmask the response
unmasked_response = masker.unmask_str(remote_response_text, mask_map)
print(f"Unmasked Response: {unmasked_response}")
# Expected output: Unmasked Response: The visa application for Jensen Huang with passport A12345678 is now under review.
if __name__ == "__main__":
asyncio.run(main())
Web Server: WebUI & API
When you run promptmask-web with the installed promptmask[web], a full-featured web service is launched at http://localhost:8000. It includes:
- A simple Web UI
- to try out features including masking/unmasking and the gateway
- A Configuration Manager to view and hot-reload settings.
- Interactive API documentation (via Swagger UI) at
http://localhost:8000/docs- API Gateway at
/gateway/v1/chat/completionsto take care of your privacy seamlessly. - Direct Masking/Unmasking API Endpoints (details on API documentation).
- Edit configuration via
/configendpoint.
- API Gateway at
Development & Contribution
Contributions are welcome.
- Clone the repository:
git clone https://github.com/cxumol/promptmask.git cd promptmask
- Install in editable mode with development dependencies:
pip install -e ".[dev,web]"
- Run tests:
pytest
- Lint and format code (optional):
rufffor linting and formatting, if you want.ruff check . ruff format .
Please open an issue or submit a pull request for any bugs or feature proposals.
License
PromptMask is distributed under the MIT License. See LICENSE for more information.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file promptmask-0.1.0.tar.gz.
File metadata
- Download URL: promptmask-0.1.0.tar.gz
- Upload date:
- Size: 21.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c969fa8eb79efa42b4517ea290afab649ab970e05cacd5a8f50cd15d45a807b1
|
|
| MD5 |
8d6ec45d566cb512afb75f669bce7656
|
|
| BLAKE2b-256 |
7d14ed1dcd76d840cd1b37737319207d8dfae543235577767ff4fff80594abf8
|
File details
Details for the file promptmask-0.1.0-py3-none-any.whl.
File metadata
- Download URL: promptmask-0.1.0-py3-none-any.whl
- Upload date:
- Size: 26.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
05528233774c3503c6442e483e3d2d16cbc070d86a4cda2ca72c5b4f1606eeb6
|
|
| MD5 |
0a603d667bcfc8e8f005fe06172bb30d
|
|
| BLAKE2b-256 |
8dbab9e85ce4ee779ec2042f42cb5822daad1b3ee29dce00b4a769a1970f2aa0
|