Add support for running N parallel devenv instances under separate compose
projects sharing Postgres, MinIO, mailer, and LDAP. Each instance has its
own main container, Valkey, source checkout, tmux session, and host port
range offset by 10000 (3449 -> 13449 -> 23449, etc.).
./manage.sh run-devenv-agentic --n-instances N reconciles the running set
to exactly {ws0..ws(N-1)}: missing instances are created (workspace sync
from the live repo via git ls-files + per-instance env-file generation
under docker/devenv/instances/ + detached tmux startup), surplus instances
are stopped highest-first via compose down (never -v), already-running
instances are left untouched. ws0 binds the live repo at PWD; ws1+ are
scratch clones under ~/.penpot/penpot_workspaces/.
Backend workers (enable-backend-worker) are gated on PENPOT_BACKEND_WORKER
in backend/scripts/_env; ws1+ overlays disable them so async-task
notifications stay bound to a single Valkey Pub/Sub instance.
Compose helpers wrap docker compose with env -i so per-instance overlay
--env-file actually overrides defaults.env -- without the strip, the shell
env from sourcing defaults.env at startup would shadow the overlay (Compose
gives shell precedence over --env-file).
Other:
- Drop network aliases (- main, - redis); use container_name for
cross-container DNS so multiple instances on the shared network don't
fight over the same DNS name.
- Pin volume names via name: (PENPOT_*_VOLUME) so volumes survive project
renames; ws0 keeps the pre-existing physical names (penpotdev_*).
- Remove cross-project depends_on from main.yml (postgres/minio-setup now
live in penpotdev-infra); manage.sh ensure-infra-up docker-waits on the
minio-setup one-shot.
- Strict arg parsing in run-devenv / run-devenv-agentic; --n-instances 0
rejected.
- Remove unused Host-matched server block from the Caddyfile.
Memory mem:devenv/core and developer docs updated.
Co-authored-by: Codex <codex@openai.com>
14 KiB
title, desc
| title | desc |
|---|---|
| 3.03. Dev environment | Dive into Penpot's development environment. Learn about self-hosting, configuration, developer tools, architecture, and more. See the Penpot Technical Guide! |
Development environment
System requirements
You need to have docker and docker-compose V2 installed on your system
in order to correctly set up the development environment.
You can look here for complete instructions.
Optionally, to improve performance, you can also increase the maximum number of user files able to be watched for changes with inotify:
echo fs.inotify.max_user_watches=524288 | sudo tee -a /etc/sysctl.conf && sudo sysctl -p
Getting Started
The interactive development environment requires some familiarity of tmux.
To start it, clone penpot repository, and execute:
./manage.sh pull-devenv
./manage.sh run-devenv
This will do the following:
- Pull the latest devenv image from dockerhub.
- Start all the containers in the background.
- Attach the terminal to the devenv container and execute the tmux session.
- The tmux session automatically starts all the necessary services.
This is an incomplete list of devenv related subcommands found on manage.sh script:
./manage.sh build-devenv-local # builds the local devenv docker image
./manage.sh start-devenv # brings up the shared infra + ws0 in background
./manage.sh run-devenv # ws0 with non-agentic tmux, attached (legacy alias)
./manage.sh run-devenv-agentic # ws0 (default) with MCP + Serena enabled; see below
./manage.sh attach-devenv # re-attaches to the tmux session of a running instance
./manage.sh stop-devenv # stops infra and all running parallel instances
./manage.sh drop-devenv # removes containers (data volumes preserved)
Parallel workspaces
The devenv runs as separate compose projects: shared infra (penpotdev-infra:
Postgres, MinIO, mailer, LDAP) plus one penpotdev-wsN project per runtime
instance. ws0 binds the live repo; ws1..wsN-1 are disposable clones under
~/.penpot/penpot_workspaces/ seeded from the current working tree on each
startup.
./manage.sh run-devenv-agentic --n-instances 3
is a desired-state reconciler: it brings the running set to exactly
{ws0, ws1, ws2}. Missing instances are created; surplus instances
(highest-numbered first) are stopped; instances already at their target index
are left alone. Stopping never removes data volumes or workspace directories.
Host ports are offset by 10000 × N:
| Service | ws0 | ws1 | ws2 |
|---|---|---|---|
| Penpot UI (HTTPS) | https://localhost:3449 |
https://localhost:13449 |
https://localhost:23449 |
| MCP HTTP stream | http://localhost:4401/mcp |
http://localhost:14401/mcp |
http://localhost:24401/mcp |
| Serena MCP | http://localhost:14281 |
http://localhost:24281 |
http://localhost:34281 |
Container-internal ports stay fixed. Target a specific instance with
--instance ws1 on attach-devenv / run-devenv-shell. run-devenv-agentic
accepts --no-mcp, --no-serena, and --serena-context CTX.
Shared state and workers
All instances share one Penpot database and one MinIO bucket; users, teams,
files, and MCP tokens are visible from every instance. Per-instance Valkey
keeps WebSocket Pub/Sub channels isolated. Background workers
(enable-backend-worker) run only on ws0 — ws1+ overlays disable it so async
task notifications stay bound to a single Pub/Sub. Trade-off: async tasks
triggered from a ws1+ tab execute (on ws0) but their completion notifications
never reach the originating tab.
Upgrading from a pre-parallel devenv
The devenv compose configuration has been split into two files and reorganized into separate compose projects per runtime instance:
docker/devenv/docker-compose.infra.yml(Postgres, MinIO, mailer, LDAP) runs under the compose projectpenpotdev-infra.docker/devenv/docker-compose.main.yml(one main container + its Valkey) runs once per runtime instance underpenpotdev-ws0,penpotdev-ws1, ….- Both projects join the external Docker network
penpot_shared, created idempotently bymanage.sh. - Per-instance configuration lives in
docker/devenv/defaults.env(ws0 baseline) plus generated overlays underdocker/devenv/instances/.
If you had the devenv running on the previous single-project (penpotdev)
layout, leftover containers and the auto-generated penpotdev_default
network must be removed before bringing the new ws0 instance up. The named
data volumes (penpotdev_postgres_data_pg16, penpotdev_minio_data,
penpotdev_user_data, penpotdev_valkey_data) are pinned by explicit
name: entries in the new compose files and are preserved through the
transition — your Postgres DB, MinIO objects, and home cache survive.
One-time cleanup, then bring up ws0:
# Stop and remove the old single-project containers (data volumes stay).
docker stop penpot-devenv-main penpot-devenv-valkey 2>/dev/null
docker rm penpotdev-postgres-1 penpotdev-minio-1 penpotdev-minio-setup-1 \
penpotdev-mailer-1 penpotdev-ldap-1 \
penpot-devenv-main penpot-devenv-valkey 2>/dev/null
# Remove the orphaned auto-generated network.
docker network rm penpotdev_default 2>/dev/null
# Bring up infra + ws0 under the new project layout.
./manage.sh run-devenv-agentic --n-instances 1
After the cleanup, normal ./manage.sh start-devenv / run-devenv /
run-devenv-agentic commands work against the new layout. The legacy
penpotdev compose project is no longer used.
Having the container running and tmux opened inside the container, you are free to execute commands and open as many shells as you want.
You can create a new shell just pressing the Ctr+b c shortcut. And Ctrl+b w for switch between windows, Ctrl+b & for kill the current window.
For more info: https://tmuxcheatsheet.com/
It may take a minute or so, but once all of the services have started, you can connect to penpot by browsing to http://localhost:3449 .
Frontend
The frontend build process is located on the tmux window 0 and window 1. On window 0 we have the gulp process responsible for watching and building styles, fonts, icon-spreads and templates.
On window 1 we can find the shadow-cljs process that is responsible for watching and building frontend clojurescript code.
In addition to the watch process you probably want to be able to open a REPL
process on the frontend application. In order to do this you can split the
window (Ctrl+b ") and execute:
cd penpot/frontend
npx shadow-cljs cljs-repl main
In order to have the REPL working you need to have an active browser session
with the penpot application opened (otherwise, you will get the error
No application has connected to the REPL server.).
Finally, in case you want to connect to the REPL from your IDE, you can set it
up to use nREPL with the port 3447 and the host localhost (you can see the
port in the startup message of the shadow-cljs process in window 1). You
will also need to call (shadow/repl :main) in the REPL to start the connection,
as explained here.
Storybook
The storybook local server is started on tmux window 2 and will listen
for changes in the styles, components or stories defined in the folders
under the design system namespace: app.main.ui.ds.
You can open the broser on http://localhost:6006/ to see it.
For more information about storybook check:
https://help.penpot.app/technical-guide/developer/ui/#storybook
Exporter
The exporter build process is located in the window 3 and in the same way as frontend application, it is built and watched using shadow-cljs.
The main difference is that exporter will be executed in a nodejs, on the server side instead of browser.
The window is split into two slices. The top slice shows the build process and on the bottom slice has a shell ready to execute the generated bundle.
You can start the exporter process executing:
node target/app.js
This process does not start automatically.
Backend
The backend related process is located in the tmux window 4, and
you can go directly to it using ctrl+b 4 shortcut.
By default the backend will be started in a non-interactive mode for convenience
but you can press Ctrl+c to exit and execute the following to start the repl:
./scripts/repl
On the REPL you have these helper functions:
(start): start all the environment(stop): stops the environment(restart): stops, reload and start again.
And many other that are defined in the dev/user.clj file.
If an exception is raised or an error occurs when code is reloaded, just use
(repl/refresh-all) to finish loading the code correctly and then use
(restart) again.
MCP Server
To set up the MCP server local development environment it's needed some additional steps.
Activate the MCP features variables
Create or modify the file frontend/resources/public/js/config.js and add (or modify) the penpotFlags to add the following:
var penpotFlags = "enable-mcp enable-access-tokens"
This will enable the MCP in the workspace and in the user settings profile.
Start the DEVENV
Start as usual the development environment
./manage.sh start-devenv
Once the TMUX is showing, create a new tmux tab (Ctrl+b c). And in the new tab run:
cd mcp
pnpm run bootstrap:multi-user
This will start the MCP server and the multi-user plugin that will be loaded automaticaly by Penpot.
There is a NGINX proxy that makes a proxy-pass from outside the docker container so you don't need to remember the ports it's using.
Configure the MCP in your tool
You can use the instructions in /mcp/#remote-mcp-in-5-steps to setup the server.
Warning: by default Cursor won't support HTTPS with a self-signed certificate. In order to work around this issue please use the port 3450 that uses an standard http protocol
An example of your cursor configuration can be:
{
"mcpServers": {
"penpot-devenv": {
"url": "http://localhost:3450/mcp/stream?userToken=TOKEN",
"type": "http"
}
}
}
To test email sending, the devenv includes MailCatcher, a SMTP server that is used for develop. It does not send any mail outbounds. Instead, it stores them in memory and allows to browse them via a web interface similar to a webmail client. Simply navigate to:
Create user
You can register a new user manually, or create new users automatically with this script. From your tmux instance, run:
cd penpot/backend/scripts
python3 manage.py create-profile
You can also skip tutorial and walkthrough steps:
python3 manage.py create-profile --skip-tutorial --skip-walkthrough
python3 manage.py create-profile -n "Jane Doe" -e jane@example.com -p secretpassword --skip-tutorial --skip-walkthrough
Feature Flags
Frontend flags via config.js
You can enable or disable feature flags on the frontend by creating (or editing) a
config.js file at frontend/resources/public/js/config.js. This file is
gitignored, so it has to be created manually. Your local flags won't affect other developers.
Set the penpotFlags variable with a space-separated list of flags:
var penpotFlags = "enable-mcp enable-webhooks enable-access-tokens";
Each flag entry uses the format enable-<flag> or disable-<flag>. They are
merged on top of the built-in defaults, so you only need to list the flags you want
to change.
Some examples of commonly used flags:
enable-access-tokens— enables the Access Tokens section under profile settings.enable-mcp— enables the MCP server configuration section.enable-webhooks— enables webhooks configuration.enable-login-with-ldap— enables LDAP login.
The full list of available flags can be found in common/src/app/common/flags.cljc.
After creating or modifying this file, reload the browser (no need to restart anything).
Backend flags via PENPOT_FLAGS
Backend feature flags are controlled through the PENPOT_FLAGS environment
variable using the same enable-<flag> / disable-<flag> format. You can set
this in the docker/devenv/docker-compose.yaml file under the main service
environment section:
environment:
- PENPOT_FLAGS=enable-access-tokens enable-mcp
This requires restarting the backend to take effect.
Note
: Some features (e.g., access tokens, webhooks) need both frontend and backend flags enabled to work end-to-end. The frontend flag enables the UI, while the backend flag enables the corresponding API endpoints.
Team Feature Flags
To test a Feature Flag, you can enable or disable them by team through the dbg page:
- Create a new team or navigate to an existing team in Penpot.
- Copy the
team-idfrom the URL (e.g.,?team-id=1234bd95-69dd-805c-8005-c015415436ae). If no team is selected, the default profile team will be used. - Go to http://localhost:3449/dbg.
- Open the Feature Flag panel, enter the
team-idand thefeaturename in either the enable or disable section, and clickSubmit.