Markpact to minimalny runtime, który pozwala trzymać cały projekt w jednym README.md.
Runtime ignoruje zwykły Markdown, a wykonuje wyłącznie codeblocki markpact:*.
Markpact to narzędzie, które zamienia plik README.md w wykonywalny kontrakt projektu. Zamiast utrzymywać osobno dokumentację i kod źródłowy, wszystko znajduje się w jednym miejscu.
| Funkcja | Opis |
|---|---|
| Executable README | Uruchom cały projekt z jednego pliku README.md |
| LLM Generation | Wygeneruj projekt z opisu tekstowego: markpact -p "REST API" |
| Multi-language | Python, Node.js, Go, Rust, PHP, TypeScript, React |
| Publishing | Publikuj do PyPI, npm, Docker Hub jedną komendą |
| Docker Sandbox | Uruchom w izolowanym kontenerze: --docker |
| HTTP Testing | Definiuj testy HTTP w markpact:test http |
| Auto-fix | Automatyczne naprawianie błędów runtime |
- Deweloperzy – szybkie prototypowanie i uruchamianie projektów
- DevOps – CI/CD z README jako single source of truth
- Edukatorzy – interaktywne tutoriale z wykonywalnym kodem
- LLM/AI – generowanie i modyfikacja projektów przez AI
git clone https://github.com/wronai/markpact.git
# Instalacja
pip install markpact[llm]
# Konfiguracja LLM (wybierz jeden)
markpact config --provider ollama # lokalny
markpact config --provider openrouter --api-key sk-or-v1-xxx # chmura
# Generuj i uruchom jedną komendą!
markpact -p "REST API do zarządzania zadaniami z SQLite" -o todo/README.md --run
markpact -p "URL shortener with FastAPI and SQLite" -o url-test/README.md --run
# Lub z gotowego przykładu
markpact -e todo-api -o todo/README.md --runWygeneruj kompletny projekt z opisu tekstowego:
# Lista 16 gotowych przykładów
markpact --list-examples
# Generuj z promptu
markpact -p "URL shortener z FastAPI i SQLite" -o url/README.md
# Generuj i uruchom natychmiast (one-liner)
markpact -p "Chat WebSocket z FastAPI" -o chat/README.md --run
# Uruchom w izolowanym Docker
markpact -p "Blog API z komentarzami" -o blog/README.md --run --dockerObsługiwane providery: Ollama (lokalny), OpenRouter, OpenAI, Anthropic, Groq
Szczegóły: docs/generator.md
Publikuj artefakty bezpośrednio z README:
# PyPI
markpact README.md --publish --bump patch
# npm
markpact README.md --publish --registry npm
# Docker Hub
markpact README.md --publish --registry docker
# GitHub Container Registry
markpact README.md --publish --registry ghcrObsługiwane rejestry: PyPI, npm, Docker Hub, GitHub Packages, GHCR
Konwertuj notebooki do formatu markpact:
# Lista obsługiwanych formatów
markpact --list-notebook-formats
# Konwersja Jupyter Notebook
markpact --from-notebook notebook.ipynb -o project/README.md
# Konwersja i uruchomienie
markpact --from-notebook notebook.ipynb -o project/README.md --run
# Podgląd konwersji
markpact --from-notebook notebook.ipynb --convert-onlyObsługiwane formaty:
| Format | Rozszerzenie | Opis |
|---|---|---|
| Jupyter Notebook | .ipynb |
Python, R, Julia |
| R Markdown | .Rmd |
R z markdown |
| Quarto | .qmd |
Wielojęzyczny |
| Databricks | .dib |
Python, Scala, R |
| Zeppelin | .zpln |
Python, Scala, SQL |
| Przykład | Opis | Uruchomienie |
|---|---|---|
| FastAPI Todo | REST API z bazą danych | markpact examples/fastapi-todo/README.md |
| Flask Blog | Aplikacja webowa z szablonami | markpact examples/flask-blog/README.md |
| CLI Tool | Narzędzie linii poleceń | markpact examples/cli-tool/README.md |
| Streamlit Dashboard | Dashboard danych | markpact examples/streamlit-dashboard/README.md |
| Kivy Mobile | Aplikacja mobilna | markpact examples/kivy-mobile/README.md |
| Electron Desktop | Aplikacja desktopowa | markpact examples/electron-desktop/README.md |
| Markdown Converter | Konwersja zwykłego MD | markpact examples/markdown-converter/sample.md --convert |
| Go HTTP API | REST API w Go | markpact examples/go-http-api/README.md |
| Node Express API | REST API w Node.js | markpact examples/node-express-api/README.md |
| Static Frontend | Statyczny HTML/CSS/JS | markpact examples/static-frontend/README.md |
| Python Typer CLI | CLI w Python (Typer) | markpact examples/python-typer-cli/README.md |
| Rust Axum API | REST API w Rust | markpact examples/rust-axum-api/README.md |
| PHP CLI | CLI w PHP | markpact examples/php-cli/README.md |
| React TypeScript SPA | SPA React + TS | markpact examples/react-typescript-spa/README.md |
| TypeScript Node API | REST API w TS (Node) | markpact examples/typescript-node-api/README.md |
| PyPI Publish | Publikacja do PyPI | markpact examples/pypi-publish/README.md --publish |
| npm Publish | Publikacja do npm | markpact examples/npm-publish/README.md --publish |
| Docker Publish | Publikacja do Docker | markpact examples/docker-publish/README.md --publish |
| Notebook Converter | Konwersja .ipynb do markpact | markpact --from-notebook examples/notebook-converter/sample.ipynb --convert-only |
Uruchom automatyczne testy wszystkich przykładów:
# Dry-run (tylko parsowanie)
./scripts/test_examples.sh
# Pełne uruchomienie
./scripts/test_examples.sh --run
# Verbose output
./scripts/test_examples.sh --verboseMarkpact może automatycznie konwertować zwykłe pliki Markdown (bez tagów markpact:*) do formatu wykonywalnego:
# Podgląd konwersji
markpact README.md --convert-only
# Konwersja i uruchomienie
markpact README.md --convert
# Auto-detekcja (konwertuj jeśli brak markpact blocks)
markpact README.md --auto
# Zapisz skonwertowany plik
markpact README.md --convert-only --save-converted output.mdKonwerter analizuje code blocks i na podstawie heurystyk wykrywa:
- Zależności →
markpact:deps(pakiety Python/Node) - Pliki źródłowe →
markpact:file(importy, klasy, funkcje) - Komendy →
markpact:run(python, uvicorn, npm, etc.)
- Jedno README jako źródło prawdy
- Możliwość uruchomienia projektu bez ręcznego tworzenia struktury plików
- Automatyzacja Bootstrap tworzy pliki w sandboxie, instaluje zależności i uruchamia komendę startową.
markpact:bootstrap <lang>Dokładnie jeden bootstrap na README. Odpowiada za parsowanie codeblocków i uruchomienie.markpact:deps <scope>Lista zależności dla danego scope (np.python).markpact:file <lang> path=...Zapisuje plik do sandboxu pod ścieżkąpath=....markpact:run <lang>Jedna komenda uruchomieniowa wykonywana w sandboxie.
#!/usr/bin/env python3
"""MARKPACT v0.1 – Executable Markdown Runtime"""
import os, re, subprocess, sys
from pathlib import Path
README = Path(sys.argv[1] if len(sys.argv) > 1 else "README.md")
SANDBOX = Path(os.environ.get("MARKPACT_SANDBOX", "./sandbox"))
SANDBOX.mkdir(parents=True, exist_ok=True)
RE = re.compile(r"^```markpact:(?P<kind>\w+)(?:\s+(?P<meta>[^\n]+))?\n(?P<body>.*?)\n^```[ \t]*$", re.DOTALL | re.MULTILINE)
def run(cmd):
print(f"[markpact] RUN: {cmd}")
env = os.environ.copy()
venv = SANDBOX / ".venv" / "bin"
if venv.exists():
env.update(VIRTUAL_ENV=str(venv.parent), PATH=f"{venv}:{env.get('PATH','')}")
subprocess.check_call(cmd, shell=True, cwd=SANDBOX, env=env)
def main():
deps, run_cmd = [], None
for m in RE.finditer(README.read_text()):
kind, meta, body = m.group("kind"), (m.group("meta") or "").strip(), m.group("body").strip()
if kind == "file":
p = re.search(r"\bpath=(\S+)", meta)
if not p: raise ValueError(f"markpact:file requires path=..., got {meta!r}")
f = SANDBOX / p[1]
f.parent.mkdir(parents=True, exist_ok=True)
f.write_text(body)
print(f"[markpact] wrote {f}")
elif kind == "deps" and meta == "python":
deps.extend(line.strip() for line in body.splitlines() if line.strip())
elif kind == "run":
run_cmd = body
if deps:
venv_pip = SANDBOX / ".venv" / "bin" / "pip"
if os.environ.get("MARKPACT_NO_VENV") != "1" and not venv_pip.exists():
run(f"{sys.executable} -m venv .venv")
(SANDBOX / "requirements.txt").write_text("\n".join(deps))
run(f"{'.venv/bin/pip' if venv_pip.exists() else 'pip'} install -r requirements.txt")
if run_cmd:
run(run_cmd)
else:
print("[markpact] No run command defined")
if __name__ == "__main__":
main()
pip install markpactUżycie:
markpact README.md # uruchom projekt
markpact README.md --dry-run # podgląd bez wykonywania
markpact README.md -s ./my-sandbox # własny katalog sandboxgit clone https://github.com/wronai/markpact.git
cd markpact
make install # lub: pip install -e .-
Ekstrakcja bootstrapu do pliku
Ten wariant jest odporny na przypadek, gdy w samym bootstrapie występują znaki ``` (np. w regexie):
sed -n '/^```markpact:bootstrap/,/^```[[:space:]]*$/p' README.md | sed '1d;$d' > markpact.py
-
Uruchomienie
python3 markpact.py
-
Konfiguracja (env vars)
MARKPACT_PORT=8001 MARKPACT_SANDBOX=./.markpact-sandbox python3 markpact.py
MARKPACT_SANDBOXZmienia katalog sandboxu (domyślnie./sandbox).MARKPACT_NO_VENV=1Wyłącza tworzenie.venvw sandboxie (przydatne, jeśli CI/Conda zarządza środowiskiem).- Port zajęty (
[Errno 98] address already in use) UstawMARKPACT_PORTna inny port lub zatrzymaj proces, który używa8000.
- Python
Bootstrap zbiera
markpact:deps python, zapisujerequirements.txtw sandboxie i instaluje zależności.
- Wejście
python3 markpact.py [README.md] - Kolejność
Bootstrap parsuje wszystkie codeblocki, zapisuje pliki i dopiero na końcu uruchamia
markpact:run.
-
Nagłówek codeblocka
```markpact:<kind> <lang> <meta>Minimalnie wymagane jest
markpact:<kind>.langjest opcjonalny i pełni rolę informacyjną (bootstrap może go ignorować). -
Metadane Dla
markpact:filewymagane jestpath=.... Metadane mogą zawierać dodatkowe tokeny (np. w przyszłościmode=...,chmod=...).
-
Rekomendacja Uruchamiaj bootstrap w czystym środowisku (np. job CI) i ustaw sandbox na katalog roboczy joba.
-
Przykład (shell)
export MARKPACT_SANDBOX=./.markpact-sandbox export MARKPACT_PORT=8001 python3 markpact.py README.md
-
Wskazówki
- Deterministyczność
Pinuj wersje w
markpact:deps(np.fastapi==...). - Bezpieczeństwo
Traktuj
markpact:runjak skrypt uruchomieniowy repo: w CI odpalaj tylko zaufane README. - Cache
Jeśli CI wspiera cache, cache’uj katalog
MARKPACT_SANDBOX/.venv.
- Deterministyczność
Pinuj wersje w
- Zasada
LLM może generować/edytować projekt poprzez modyfikacje README (codeblocki
markpact:file,markpact:deps,markpact:run). - Oczekiwania
markpact:filezawsze zawiera pełną zawartość pliku.- Każda zmiana zależności idzie przez
markpact:deps. - Jedna komenda startowa w
markpact:run.
- Bootstrap jako pierwszy fenced codeblock w README
- Każdy plik w osobnym
markpact:file - Zależności tylko w
markpact:deps - Jedna komenda startowa w
markpact:run - Ekstrakcja bootstrapu
Nie używaj zakresu
/,/```/(bomoże wystąpić w treści, np. w regexie). Używaj `^$` na końcu.
# Markpact LLM Configuration
MARKPACT_MODEL="openrouter/nvidia/nemotron-3-nano-30b-a3b:free"
MARKPACT_API_BASE="https://openrouter.ai/api/v1"
MARKPACT_API_KEY="sk-or-v1-xxxxx"
MARKPACT_TEMPERATURE="0.7"
MARKPACT_MAX_TOKENS="4096"markpact config --provider ollama
markpact config --model ollama/qwen2.5-coder:14b
markpact -p "REST API dla książek"markpact config --provider openrouter --api-key sk-or-v1-xxxxx
markpact config --model openrouter/nvidia/nemotron-3-nano-30b-a3b:free
markpact -p "REST API dla książek"markpact:deps python
fastapi
uvicorn
markpact:file python path=app/main.py
from fastapi import FastAPI
app = FastAPI()
@app.get("/")
def root():
return {"message": "Hello from Executable Markdown"}
markpact:run python
uvicorn app.main:app --host 0.0.0.0 --port ${MARKPACT_PORT:-8088}

