JAVA FIRST. NO BLOAT. PURE POWER.
Get JClaw running in one command. The installer downloads the self-contained
jclaw-bundle.zip from the latest GitHub Release, verifies Java 25+ (the bundle's
only runtime dependency), extracts it to ~/.jclaw, and starts JClaw on
http://localhost:9000.
macOS & Linux
curl -fsSL https://raw.githubusercontent.com/tsukhani/jclaw/main/install.sh | shWindows (PowerShell)
irm https://raw.githubusercontent.com/tsukhani/jclaw/main/install.ps1 | iexOn Windows the bundle runs through Git Bash or WSL (the launcher is a POSIX shell script). The installer prefers Git Bash; if only WSL is present it launches there; if neither is found it installs and prints how to run it.
Once running, manage it with jclaw status, jclaw stop, jclaw restart.
Requirements: a Java 25+ runtime (Zulu or Temurin). Nothing else — the bundle bakes in the framework, app dependencies, precompiled classes, and the prebuilt SPA.
Configuration (optional environment variables):
| Variable | Default | Purpose |
|---|---|---|
JCLAW_HOME |
~/.jclaw |
Install directory |
JCLAW_VERSION |
latest |
Pin a release tag, e.g. v0.13.40 |
JCLAW_PORT |
9000 |
Port reported on launch |
JCLAW_NO_START |
— | Set to 1 to install without starting |
# Pin a version and install without auto-starting:
curl -fsSL https://raw.githubusercontent.com/tsukhani/jclaw/main/install.sh \
| JCLAW_VERSION=v0.13.40 JCLAW_NO_START=1 shJClaw is Abundent's AI-powered automation platform — built from scratch in pure Java on a customized Play Framework 1.x foundation. It draws ideas and feature designs from three predecessor projects:
- OpenClaw (Node.js/TypeScript) — agent orchestration, memory system, conversational AI patterns
- JavaClaw (Spring Boot) — job scheduling, background task processing, browser automation
- Hermes (Python) — cron/task scheduling parity, subagent delegation patterns
The implementation is entirely original — no code is shared with either project. JClaw is built on lean library primitives (OkHttp 5, db-scheduler, ProcessBuilder, virtual threads, JPA) with no Spring, no heavy framework bloat, no Python, and no Node.js runtime on the server. The result is a leaner, faster, more maintainable platform for building AI agents and automation workflows.
Web chat with memory-aware agents, tool execution, and markdown rendering.
- 🤖 Agent System — Conversational AI agents with memory and context
- ⚡ Job Scheduling — Persistent cron & scheduled tasks via db-scheduler, with automatic retries and crash recovery
- 🔧 Pure Java — No Python/JavaScript runtimes required
- 📦 Built-in Frontend — Nuxt 4 SPA (Vue 3 + TypeScript)
- 🔌 Plugin Architecture — Modular, extensible design
- 🧠 Memory & Context — Persistent conversations across sessions
- 🚀 Lightweight — Minimal resource footprint, fast startup
jclaw/
├── app/ # Application code
│ ├── controllers/ # HTTP controllers (Play 1.x pattern)
│ ├── models/ # JPA domain entities
│ ├── services/ # Business logic (incl. db-scheduler bridge)
│ ├── agents/ # AI agent implementations
│ ├── channels/ # Messaging channels (web, Telegram, Slack)
│ ├── llm/ # LLM provider drivers (OkHttp 5)
│ ├── tools/ # Agent tool implementations
│ ├── memory/ # Agent memory stores (JPA-backed)
│ ├── mcp/ # Model Context Protocol client
│ ├── slash/ # Slash-command handlers
│ ├── jobs/ # Play @Every jobs + db-scheduler handlers
│ ├── views/ # Groovy server templates
│ └── utils/ # Utility classes
├── conf/ # Play configuration
│ ├── application.conf # Main app config
│ ├── routes # URL routing
│ ├── play.plugins # Play plugin registration
│ └── log4j2.xml # Logging configuration
├── frontend/ # Nuxt 4 SPA (SPA-only; ssr: false)
│ ├── app.vue # Root component
│ ├── layouts/ # Page layouts
│ ├── pages/ # Nuxt file-based routes
│ ├── components/ # Reusable Vue components
│ ├── composables/ # Shared reactive state (useAuth, useEventBus, ...)
│ ├── middleware/ # Global route middleware (auth guard)
│ ├── public/ # Static assets
│ └── nuxt.config.ts # Nuxt configuration
├── lib/ # Custom JARs (if needed)
├── modules/ # Play modules (auto-managed)
├── public/ # Static web assets
├── test/ # Unit and integration tests
├── tmp/ # Play temp/runtime files
├── logs/ # Application logs
└── README.md # This file
- JDK 25+ (Zulu recommended)
That's the whole list. The Abundent Play 1.x fork,
app dependencies, precompiled classes, and the prebuilt SPA all ship inside
jclaw-bundle.zip, so a Java 25 runtime is the only thing the host needs to run
JClaw — see Quick Install. (Building from source instead
adds a dev toolchain — Node.js, pnpm, and the play CLI — which the
Dev Container installs for you.)
Tesseract OCR — required for text extraction from images, scanned PDFs,
and image-only PDFs via the documents tool. Apache Tika invokes the
tesseract binary as a subprocess; without it, image inputs return empty
text. A startup probe logs a WARN line at boot if tesseract is missing so
the missing capability is visible without trial and error.
# Debian / Ubuntu
sudo apt-get install tesseract-ocr
# macOS
brew install tesseract
# Windows
choco install tesseract
# or: winget install --id UB-Mannheim.TesseractOCRAdditional language packs install separately. The default is English
(eng); install tesseract-ocr-fra, tesseract-ocr-jpn, etc. for other
languages, then update ocr.tesseract.languages in conf/application.conf
(e.g. eng+fra+jpn).
Local Ollama — required only if you want to bind agents to the
ollama-local LLM provider for self-hosted inference. JClaw seeds
provider.ollama-local.baseUrl=http://localhost:11434/v1 at first boot,
so the provider is already listed in Settings without further wiring —
install Ollama on the host and pull a model to make it usable. A startup
probe logs INFO with the model count when the local server is reachable,
and WARN with an install hint when the server is reachable but broken;
a fresh install with no Ollama running stays silent (no spurious WARN
on every JVM start).
# Linux
curl https://ollama.com/install.sh | sh
# macOS
brew install ollama
# Windows — download the installer from https://ollama.com/downloadAfter installing, pull a model:
ollama pull qwen2.5Then open the Settings page, expand the ollama-local card, and either
run Discover Models against http://localhost:11434/v1 or paste the
JSON for the model you pulled into the models field. Bind an agent
to ollama-local from the Agent Edit page to start chatting against
your local model.
LM Studio — required only if you want to bind agents to the
lm-studio LLM provider. JClaw seeds
provider.lm-studio.baseUrl=http://localhost:1234/v1 at first boot
so the provider is already listed under "Local" in Settings. Same
boot-time probe as ollama-local: INFO when reachable, WARN with a
launch hint when reachable-but-broken, silent (DEBUG) when LM Studio
isn't running.
LM Studio is a desktop application — install it from https://lmstudio.ai (macOS DMG, Windows installer, or Linux AppImage). After launching, load a model in the My Models tab, then switch to the Server tab and click Start Server. The default port is 1234.
In JClaw Settings, expand the lm-studio card and either run
Discover Models against http://localhost:1234/v1 or paste the JSON
for the model you loaded into the models field. Bind an agent to
lm-studio from the Agent Edit page to use it.
git clone https://bitbucket.abundent.com/scm/jclaw/jclaw.git
cd jclawDependencies are automatically installed when you start with jclaw.sh.
The fastest way to start coding without installing any of the Prerequisites on your host machine is to use the included dev container. The .devcontainer/Dockerfile ships a pinned toolchain (Java 25, Python 3.14, Node 24, corepack, the Play fork at the version recorded in .play-version, tesseract-ocr) on top of Ubuntu 26.04 LTS — all the prerequisites listed above, already installed.
Just two things on your machine:
- Docker Desktop (macOS/Windows) or Docker Engine (Linux) — the dev container runs in a Docker container, so the Docker daemon needs to be running.
- An IDE that supports the Dev Containers spec — any of:
- Cursor (built-in support)
- VS Code with the Dev Containers extension
- JetBrains Gateway with the Dev Containers plugin
- GitHub Codespaces (cloud, no local Docker needed)
After cloning, open the project in your IDE and trigger the "Reopen in Container" command:
| IDE | How to launch |
|---|---|
| Cursor | Cmd/Ctrl+Shift+P → Dev Containers: Reopen in Container |
| VS Code | Click the blue corner icon (bottom-left) → Reopen in Container, or Cmd/Ctrl+Shift+P → same command |
| GitHub Codespaces | Push your branch to GitHub → click Code → Codespaces tab → Create codespace on main |
| JetBrains Gateway | New Connection → Dev Containers → point at the local jclaw folder |
What happens automatically once you click:
- Docker builds the image from
.devcontainer/Dockerfile(~5–10 min the first time, cached on subsequent rebuilds). - Your local jclaw directory is bind-mounted into the container at
/workspaces/jclaw. Edits you make inside the container persist on your host — the container is an environment, not a copy. - The IDE runs
postCreateCommand: ./jclaw.sh setupautomatically, which:- Validates all prerequisites (every check passes — they're baked into the image)
- Wires git hooks (
.githooks/pre-commit,.githooks/pre-push) - Validates the pinned pnpm version via corepack with integrity-hash verification
- Runs
pnpm installfor the frontend - Adds the canonical
githubremote (https://github.com/tsukhani/jclaw.git)
- Recommended VS Code/Cursor extensions install (Volar, Java Pack, ESLint, Stylelint, YAML).
- The IDE attaches to the container — your terminal, file explorer, and editor are now running inside it.
Everything works the same as it would on a native host. The container is a Linux dev box with the pre-installed toolchain:
./jclaw.sh --dev start # dev mode (Play autoreload + Nuxt HMR)
./jclaw.sh test # backend + frontend test suites
./jclaw.sh status # check what's running
./jclaw.sh stopPorts 9000 (backend) and 3000 (Nuxt) are forwarded to your host automatically. Open http://localhost:9000 and http://localhost:3000 in your host's browser while the dev server runs inside the container. The Nuxt port is configured to auto-open the browser when it boots; the backend port emits a notification.
The pre-commit hook (frontend lint-staged) and pre-push hook (full test suite) work inside the container without any extra setup. Two nuances:
- Signed commits —
/deployproduces signed commits and signed tags (commit -S,tag -s). Your host's GPG/SSH keys aren't visible inside the container by default. Two recovery options:- Easiest: do
/deployfrom your host shell (open a host terminal,cdinto the project, run the slash command). Code inside the container, deploy from outside. - More setup: add a
mountsblock to.devcontainer/devcontainer.jsonto bind-mount~/.sshand~/.gnupginto the container. Same end result, more configuration.
- Easiest: do
- File ownership — files written inside the container land on your host with UID 1000 (
ubuntuuser). On macOS this maps to your user automatically; on Linux you may see "owned by 1000" inls -lif your host UID differs. Usually harmless.
When the toolchain changes (e.g., a new Play version, a JDK bump, a base-image bump), you'll want a fresh build:
| IDE | How to rebuild |
|---|---|
| Cursor / VS Code | Cmd/Ctrl+Shift+P → Dev Containers: Rebuild Container |
| JetBrains Gateway | Container settings → Rebuild |
| CLI fallback | docker build -t jclaw-devcontainer:latest .devcontainer/ (manual, you'd then need to update the IDE config to use the rebuilt image) |
Most rebuilds reuse cached apt + JDK + Node layers and only re-download what changed (e.g., the Play release zip if PLAY_VERSION was bumped). Full cold rebuilds run ~5–10 min.
- "Docker not running" — start Docker Desktop /
sudo systemctl start docker. - First build hangs on apt-get — your network is slow or the Ubuntu mirror is rate-limiting. Retry; layers are cached so progress isn't lost.
- Postcreate fails on
./jclaw.sh setup— read the error; it'll point at the failing prereq. Open.devcontainer/Dockerfileto see what's installed; if a tool is missing, file an issue or patch the Dockerfile and rebuild. - Edits in the IDE don't appear on host — verify you opened the folder via "Reopen in Container," not by mounting a Docker volume. The bind-mount is what makes the edits round-trip.
docker rmito clean up —docker rmi jclaw-devcontainer:latest(or the container image name your IDE assigns) removes the cached image. The next "Reopen in Container" rebuilds from scratch.
# Start both backend and frontend in dev mode
./jclaw.sh --dev start
# Stop
./jclaw.sh --dev stop
# Check status
./jclaw.sh --dev status
# View logs (tails both backend and frontend logs)
./jclaw.sh --dev logsDefault ports: backend on :9000, frontend on :3000.
# Deploy to /tmp (creates /tmp/jclaw), build everything, and start
./jclaw.sh --deploy /tmp start
# Stop
./jclaw.sh --deploy /tmp stop
# View logs
./jclaw.sh --deploy /tmp logsThis packages the app with play dist, unzips to <dir>/jclaw/, installs dependencies, builds the frontend, and starts both services in production mode.
To start an existing deployment (without re-packaging):
# Start
./jclaw.sh start
# Stop
./jclaw.sh stop
# View logs
./jclaw.sh logsThe simplest way to run JClaw in production is with Docker Compose. The shipped docker-compose.yml pulls the prebuilt image from GHCR, publishes the app on :9000, and persists data/, logs/, workspace/, and skills/ to the host so config and conversations survive restarts.
# Start in the background
docker compose up -d
# Follow logs
docker compose logs -f
# Stop and remove the container
docker compose down
# Run on a custom port (default: 9000)
JCLAW_PORT=8080 docker compose up -dThat's it — no .env setup needed. On first boot the container's entrypoint generates a 64-character PLAY_SECRET (used to sign session cookies) and persists it to ./data/.play-secret. Subsequent restarts read the same file, so existing user sessions survive across docker compose down / up cycles. To rotate the secret, delete ./data/.play-secret and restart the container — all existing PLAY_SESSION cookies become invalid, which is the point.
If you'd rather pin the secret yourself (e.g. for multi-host deployments that need a shared cookie key, or rotation managed by your secret-store), drop a .env file alongside docker-compose.yml with PLAY_SECRET=<value> — Compose will forward it into the container and the entrypoint will defer to it instead of generating one.
You can also set JCLAW_PORT in .env alongside docker-compose.yml instead of passing it inline — Compose reads the same file for variable interpolation in the YAML and for the runtime environment of the jclaw service.
The container runs in production mode — the Nuxt SPA is already built into the image, so no local Node.js, pnpm, or Play toolchain is required on the host. Open http://localhost:9000 (or your custom port) once the container is healthy.
The container also exposes HTTPS on :9443 (with HTTP/3 over the same UDP port) using a self-signed TLS cert generated at certs/host.cert on first boot. HTTPS works as-is but browsers show a cert-warning interstitial, and Chrome refuses HTTP/3 entirely — QUIC requires the cert to be in the system trust store. To get browser-trusted HTTPS plus working HTTP/3, sign the cert with mkcert's local CA from your host. Install mkcert via your platform's package manager:
# macOS
brew install mkcert
# Debian / Ubuntu (22.04+)
sudo apt install mkcert libnss3-tools
# Fedora / RHEL
sudo dnf install mkcert nss-tools
# Arch
sudo pacman -S mkcert nss
# Windows
choco install mkcert
# or: scoop bucket add extras && scoop install mkcertFor older distros that don't package mkcert, grab the prebuilt binary from the mkcert releases page and install libnss3-tools (Debian/Ubuntu) or nss-tools (Fedora) separately so mkcert can register with Firefox.
Then trust the local CA, regenerate the cert, and restart the container:
sudo mkcert -install # adds mkcert's CA to the system trust store (and Firefox NSS if installed)
./jclaw.sh https # regenerates certs/host.cert + host.key, signed by the CA
docker compose restart jclaw # JVM reloads the new cert at bootSubsequent docker compose up -d calls reuse the existing cert — you only need to re-run ./jclaw.sh https after rotating mkcert's CA or deleting the certs/ directory. Run ./jclaw.sh no-https to delete the cert+key (the next start boots HTTP/1.1 only). conf/application.conf is never modified by either command.
Use --backend-port and --frontend-port with any jclaw.sh mode. The frontend reads the backend port via the JCLAW_BACKEND_PORT environment variable at startup — no files are modified.
# Dev mode with custom ports
./jclaw.sh --dev --backend-port 8080 --frontend-port 4000 start
# Production deploy with custom ports (creates /tmp/jclaw)
./jclaw.sh --deploy /tmp --backend-port 8080 --frontend-port 4000 start
# Bare start with custom backend port
./jclaw.sh --backend-port 8080 startRun the backend and frontend test suites together and print a consolidated pass/fail summary:
./jclaw.sh testThis runs play autotest (backend JUnit + functional tests) followed by pnpm test (frontend Vitest), streams each side's output live, and finishes with a two-line verdict like:
backend : PASSED (47 classes, 26s)
frontend : PASSED Tests 199 passed (199) (5s)
Full logs land in logs/test-backend.log and logs/test-frontend.log for post-mortem on failure. The command exits non-zero if either suite failed, so it's safe to wire into git hooks or CI.
An in-repo .githooks/pre-push hook runs ./jclaw.sh test before a push reaches the remote and caches the tested SHA in $GIT_DIR/jclaw-last-tested-sha, so the second push in a two-remote deploy flow (origin + github) reuses the result instead of re-running the suite. Enable once per clone:
git config core.hooksPath .githooksTo bypass for a one-off push (e.g. urgent hotfix, docs-only change): JCLAW_SKIP_TESTS=1 git push origin HEAD.
- Models: JPA entities with Play's model pattern
- Controllers: RESTful API endpoints
- Services: Business logic with dependency injection
- Agents: Conversational AI with memory/context persistence
- Jobs: Internal maintenance (cleanup, probes, boot checks) on Play's built-in
@Every/@OnApplicationStartjob system - Scheduling: User-facing Tasks — cron, scheduled, and immediate — run on db-scheduler, persisted in a
scheduled_taskstable with atomic row-claim, pluggable retries, and heartbeat-based dead-execution recovery
- Framework: Vue 3 + TypeScript + Nuxt 4 (SPA mode,
ssr: false) - UI components: shadcn-nuxt on Reka UI primitives, with
class-variance-authority+tailwind-mergefor variants - Styling: Tailwind CSS v4 (via
@tailwindcss/vite), Lucide + Heroicons icons, Inter variable font - State: Composables backed by
useState(no Pinia);@vueuse/coreutilities - Data & rendering:
@tanstack/vue-tablefor tables,marked+dompurifyfor safe Markdown,zodfor validation - API: Cookie-session authenticated
$fetchto the Play backend, proxied via Nitro in dev - Tooling: Vitest +
@nuxt/test-utils+ happy-dom, Playwright e2e, ESLint + Stylelint,vue-tsctypecheck, a11y viavue-axe/axe-core
- Java-First — Everything in Java. No Python, no Node for server-side logic.
- Minimal Dependencies — Only bring in what we absolutely need.
- Memory & Context — Agents remember. Context persists. Conversations flow.
- Async by Default — Jobs run in background. APIs are non-blocking.
- Modular Skills — Agents can automatically create, share, and chain skills. Skill primitives are reusable across agents and shareable with other JClaw users.
This is an internal Abundent project. For questions or contributions, reach out to the team.
This project is licensed under the MIT License.
Built with ☕ Java and ❤️ by the Abundent crew.

