Minne - A Graph-Powered Personal Knowledge Base
Minne (Swedish for "memory") is a personal knowledge management system and save-for-later application for capturing, organizing, and accessing your information. Inspired by the Zettelkasten method, it uses a graph database to automatically create connections between your notes without manual linking overhead.
The "Why" Behind Minne
For a while I've been fascinated by Zettelkasten-style PKM systems. While tools like Logseq and Obsidian are excellent, I found the manual linking process to be a hindrance for me. I also wanted a centralized storage and easy access across devices.
While developing Minne, I discovered KaraKeep (formerly Hoarder), which is an excellent application in a similar space – you probably want to check it out! However, if you're specifically interested in a PKM that leverages a SurrealDB as its backend, utilizing both vector and graph retrieval, offers the possibility to chat with your knowledge resource, and provides a blend of manual and AI-driven organization, then Minne might be for you.
Core Philosophy & Features
Minne is designed to make it incredibly easy to save snippets of text, URLs, and other content(coming if there is demand). Simply send content along with a category tag. Minne then ingests this, leveraging AI to create relevant nodes and relationships within its graph database, alongside your manual categorization. This graph backend, powered by SurrealDB, allows for discoverable connections between your pieces of knowledge.
You can converse with your knowledge base through an LLM-powered chat interface (via OpenAI API). For those who like to see the bigger picture, Minne also includes an experimental feature to visually explore your knowledge graph.
The application is built for speed and efficiency using Rust with a Server-Side Rendered (SSR) frontend (HTMX and minimal JavaScript). It's fully responsive, offering a complete mobile interface for reading, editing, and managing your content, including the graph database itself. PWA (Progressive Web App) support means you can "install" Minne to your device for a native-like experience. For quick capture on the go, especially on iOS, a dedicated Shortcut makes sending content to your Minne instance a breeze.
Minne is open source (AGPL), self-hostable, and can be deployed flexibly: via Nix, Docker Compose, pre-built binaries, or by building from source. It can run as a single main binary or as separate server and worker processes for optimized resource allocation.
Tech Stack
- Backend: Rust
- Frontend: Server-Side Rendering (SSR) with HTMX, Axum, Minijinja, and plain JavaScript for interactivity.
- Database: SurrealDB (as a graph database)
- AI Integration: OpenAI API (for chat and content processing)
- Web Content Processing: Relies on a Chromium instance for robust webpage fetching/rendering.
Prerequisites
- For Docker/Nix: Docker or Nix installed. These methods handle SurrealDB and Chromium dependencies.
- For Binaries/Source:
- A running SurrealDB instance.
- Chromium (or a compatible Chrome browser) installed and accessible in your
PATH. - Git (if cloning and building from source).
- Rust toolchain (if building from source).
Getting Started
You have several options to get Minne up and running:
1. Nix (Recommended for ease of dependency management)
If you have Nix installed, you can run Minne directly:
nix run 'github:perstarkse/minne#main'
This command will fetch Minne and its dependencies (including Chromium) and run the main (combined server/worker) application.
2. Docker Compose (Recommended for containerized environments)
This is a great way to manage Minne and its SurrealDB dependency together.
-
Clone the repository (or just save the
docker-compose.ymlbelow). -
Create a
docker-compose.ymlfile:version: '3.8' services: minne: image: ghcr.io/perstarkse/minne:latest # Pulls the latest pre-built image # Or, to build from local source: # build: . container_name: minne_app ports: - "3000:3000" # Exposes Minne on port 3000 environment: # These are examples, ensure they match your SurrealDB setup below # and your actual OpenAI key. SURREALDB_ADDRESS: "ws://surrealdb:8000" SURREALDB_USERNAME: "root_user" # Default from SurrealDB service below SURREALDB_PASSWORD: "root_password" # Default from SurrealDB service below SURREALDB_DATABASE: "minne_db" SURREALDB_NAMESPACE: "minne_ns" OPENAI_API_KEY: "your_openai_api_key_here" # IMPORTANT: Replace with your actual key DATA_DIR: "/data" # Data directory inside the container RUST_LOG: "minne=info,tower_http=info" # Example logging level volumes: - ./minne_data:/data # Persists Minne's data (e.g., scraped content) on the host depends_on: - surrealdb networks: - minne-net # Waits for SurrealDB to be ready before starting Minne command: > sh -c " echo 'Waiting for SurrealDB to start...' && # Adjust sleep time if SurrealDB takes longer to initialize in your environment until nc -z surrealdb 8000; do echo 'Waiting for SurrealDB...'; sleep 2; done && echo 'SurrealDB is up, starting Minne application...' && /usr/local/bin/main " # For separate server/worker: # command: /usr/local/bin/server # or /usr/local/bin/worker surrealdb: image: surrealdb/surrealdb:latest container_name: minne_surrealdb ports: # Exposes SurrealDB on port 8000 (primarily for direct access/debugging if needed, # not strictly required for Minne if only accessed internally by the minne service) - "127.0.0.1:8000:8000" # Bind to localhost only for SurrealDB by default volumes: # Persists SurrealDB data on the host in a 'surreal_database' folder - ./surreal_database:/database command: > start --log info # Consider 'debug' for troubleshooting --user root_user --pass root_password file:/database/minne_v1.db # Using file-based storage for simplicity networks: - minne-net volumes: minne_data: {} # Defines a named volume for Minne data (can be managed by Docker) surreal_database: {} # Defines a named volume for SurrealDB data networks: minne-net: driver: bridge -
Create a
.envfile in the same directory as yourdocker-compose.yml(recommended for sensitive data):OPENAI_API_KEY="your_openai_api_key_here" # You can override other environment variables here if needed # e.g., if you want to expose SurrealDB differently or use different credentials. # SURREALDB_USERNAME_MINNE="custom_user" # If changing Minne's access credentials # SURREALDB_PASSWORD_MINNE="custom_pass"(If using a
.envfile, ensure variables indocker-compose.yml'senvironmentsection reference them like${OPENAI_API_KEY}or are directly set if not sensitive and common across setups) -
Run:
docker compose up -dMinne will be accessible at
http://localhost:3000.
3. Pre-built Binaries (GitHub Releases)
Binaries for Windows, macOS, and Linux (combined main version) are available on the GitHub Releases page.
- Download the appropriate binary for your system.
- You will need to provide and run SurrealDB and have Chromium installed and accessible in your PATH separately.
- Set the required Configuration environment variables or use a
config.yaml. - Run the executable.
4. Build from Source
- Clone the repository:
git clone https://github.com/perstarkse/minne.git cd minne - You will need to provide and run SurrealDB and have Chromium installed and accessible in your PATH separately.
- Set the required Configuration environment variables or use a
config.yaml. - Build and run:
- For the combined
mainbinary:cargo run --release --bin main - For the
serverbinary:cargo run --release --bin server - For the
workerbinary (if you want to run it separately):cargo run --release --bin worker
target/release/. - For the combined
Configuration
Minne can be configured using environment variables or a config.yaml file placed in the working directory where you run the application. Environment variables take precedence over config.yaml.
Required Configuration:
SURREALDB_ADDRESS: WebSocket address of your SurrealDB instance (e.g.,ws://127.0.0.1:8000orws://surrealdb:8000for Docker).SURREALDB_USERNAME: Username for SurrealDB (e.g.,root_user).SURREALDB_PASSWORD: Password for SurrealDB (e.g.,root_password).SURREALDB_DATABASE: Database name in SurrealDB (e.g.,minne_db).SURREALDB_NAMESPACE: Namespace in SurrealDB (e.g.,minne_ns).OPENAI_API_KEY: Your API key for OpenAI (e.g.,sk-YourActualOpenAIKeyGoesHere).DATA_DIR: Directory to store local data like fetched webpage content (e.g.,./minne_app_data).
Optional Configuration:
RUST_LOG: Controls logging level (e.g.,minne=info,tower_http=debug).HTTP_PORT: Port for the Minne server to listen on (Default:3000).CHROME_ADDRESS: Address of a remote Chrome DevTools Protocol endpoint (e.g.,http://localhost:9222, if not using local Chromium managed by Minne/Docker/Nix).
Example config.yaml:
surrealdb_address: "ws://127.0.0.1:8000"
surrealdb_username: "root_user"
surrealdb_password: "root_password"
surrealdb_database: "minne_db"
surrealdb_namespace: "minne_ns"
openai_api_key: "sk-YourActualOpenAIKeyGoesHere"
data_dir: "./minne_app_data"
# rust_log: "info"
# http_port: 3000
Application Architecture (Binaries)
Minne offers flexibility in deployment:
main: A combined binary running both server (API, web UI) and worker (background tasks) in one process. Ideal for simpler setups.server: Runs only the server component.worker: Runs only the worker component, suitable for deployment on a machine with more resources for intensive tasks.
This modularity allows scaling and resource optimization. The main binary or the Docker Compose setup (using main) is sufficient for most users.
Usage
Once Minne is running:
- Access the web interface at
http://localhost:3000(or your configured port). - On iOS, consider setting up the Minne iOS Shortcut for effortless content sending. Add the shortcut, replace the [insert_url] and the [insert_api_key] snippets.
- Start adding notes, URLs and explore your growing knowledge graph.
- Engage with the chat interface to query your saved content.
- Try the experimental visual graph explorer to see connections.
Roadmap
I've developed Minne primarily for my own use, but having been in the selfhosted space for a long time, and using the efforts by others, I thought I'd share with the community. Feature requests are welcome. The roadmap as of now is:
- Handle uploaded images wisely.
- An updated explorer of the graph database, and potentially the vector space.
- A TUI frontend which opens your system default editor for improved writing and document management.
Contributing
Contributions are welcome! Whether it's bug reports, feature suggestions, documentation improvements, or code contributions, please feel free to open an issue or submit a pull request.
License
Minne is licensed under the GNU Affero General Public License v3.0 (AGPL-3.0). See the LICENSE file for details. This means if you run a modified version of Minne as a network service, you must also offer the source code of that modified version to its users.