Deployment configs as code. GitLab CI variables as code. Reproducible deploys every time.
Project description
DeployLane
Deployment configs as code. GitLab CI variables as code. Reproducible deploys every time.
DeployLane is a CLI for teams deploying containerized apps to Linux servers with Docker Compose and GitLab CI. It replaces scattered shell scripts and manually managed CI variables with a single, version-controlled workflow.
pip install deploylane
dlane login --host https://gitlab.example.com
dlane scaffold my-service
dlane deploy push my-service --yes
Is this for you?
DeployLane fits your stack if:
- ✅ You deploy Docker Compose apps to Linux servers (VPS, bare metal, cloud VMs)
- ✅ You use GitLab CI to build and push images
- ✅ You manage CI variables manually in the GitLab UI and it's becoming a mess
- ✅ You want zero-downtime blue-green deploys without Kubernetes complexity
It's probably not for you if you're on Kubernetes, AWS ECS, or a fully managed PaaS.
The problem it solves
Most teams start with a deploy.sh script that "just works." Over time it becomes:
- Different versions on different servers
- CI variables scattered across projects with no history
- "Works on staging, broken on prod" because someone changed something manually
- New team member spends a day figuring out how deployment actually works
DeployLane makes the entire deployment setup version-controlled and reproducible.
How it works
Your workstation GitLab CI Server
──────────────── ───────────── ──────
dlane scaffold
deploy.yml
vars.yml → dlane vars apply → GitLab CI variables ← Pipeline uses these
compose/ ↓
nginx/ deploy.sh (CI calls this)
ci/
← dlane deploy pull ← docker-compose.yml (pull before you push)
deploy.sh
.env (runtime state, never overwritten)
nginx.conf
→ dlane deploy push → docker-compose.yml (pull-first protected)
deploy.sh
→ dlane deploy install → nginx.conf (one-time setup)
sudoers
install.sh
dlane ci push → opens MR in GitLab with .gitlab-ci.yml
dlane deploy status → shows running containers and active color per target
dlane scaffoldgenerates all server-side files from a singledeploy.ymldlane vars applypushes your localvars.ymlto GitLab CI variablesdlane deploy pullfetches the current server state before making changesdlane deploy pushsyncsdocker-compose.ymlanddeploy.shto the server (blocked if server version differs)- GitLab CI builds the image and calls
deploy.sh— that's it
Installation
pip install deploylane
Requires Python 3.10+. Works on macOS and Linux.
Quick start (5 minutes)
1. Login
dlane login --host https://gitlab.example.com
# optionally name your profile and set a registry host
dlane login --profile prod --host https://gitlab.example.com --registry-host registry.example.com
2. Create a workspace
A workspace tracks multiple services together.
mkdir ops && cd ops
dlane init
dlane add gateway --gitlab-project acme/gateway --strategy bluegreen
dlane add api --gitlab-project acme/api --strategy plain
3. Scaffold a service
dlane scaffold gateway
This generates .workspace/gateway/.deploylane/ with:
deploy.yml— targets, ports, strategy (edit this to match your server)vars.yml— CI variable definitions (edit values, thenvars apply)compose/,nginx/,scripts/deploy.sh,ci/.gitlab-ci.yml
4. Configure your server details
Edit .workspace/gateway/.deploylane/deploy.yml:
version: 1
strategy: bluegreen
project: acme/gateway
app:
name: gateway
registry_host: registry.acme.com
targets:
prod:
host: 10.0.0.1
user: deploy
deploy_dir: /home/deploy/gateway
ports:
blue: 8080
green: 8081
5. Push CI variables
dlane vars plan gateway # preview what would change
dlane vars apply gateway # push to GitLab
6. Push files to server & install (once per server)
dlane deploy pull gateway # fetch server state first (if server already exists)
dlane deploy push gateway --yes # docker-compose.yml + deploy.sh (pull-first protected)
dlane deploy install gateway --yes # nginx + sudoers + runs install.sh (one-time)
7. Add the CI pipeline
dlane ci push gateway # opens an MR in acme/gateway with the generated .gitlab-ci.yml
Or copy .workspace/gateway/.deploylane/ci/.gitlab-ci.yml manually to your repo.
GitLab CI will now build, push, and deploy automatically on every push.
Deployment strategies
| Strategy | How it works | When to use |
|---|---|---|
bluegreen |
Two containers, nginx switches traffic — zero downtime | Production |
plain |
Single container, docker compose up -d |
Staging, internal tools |
CI variable management
Stop clicking through the GitLab UI. Manage variables as code:
dlane vars get gateway # pull current GitLab variables → vars.yml
dlane vars diff gateway # see what's different between local and GitLab
dlane vars plan gateway # preview creates / updates / orphans
dlane vars apply gateway # push local vars.yml → GitLab
dlane vars prune gateway # remove GitLab variables not in vars.yml
vars.yml is version-controlled. Every variable change is a git commit.
Variable naming convention:
| Variable | Purpose |
|---|---|
{TARGET}_HOST |
Server IP — e.g. PROD_HOST, STAGING_HOST |
{TARGET}_USER |
SSH user |
{TARGET}_DEPLOY_DIR |
Deploy directory on server |
{TARGET}_SSH_KEY |
Base64-encoded private key |
REGISTRY_HOST |
Docker registry hostname |
REGISTRY_USER |
Registry login user |
REGISTRY_PASS |
Registry login password |
Multi-service workspaces
dlane list # show all services and their status
dlane list --tag api # filter by tag
dlane deploy push --all --yes # push all services at once
dlane vars apply --all # apply variables for all services
Local overrides
Keep production secrets out of git with .local.env files:
# .workspace/gateway/.deploylane/env/prod.local.env (gitignored)
APP_PORT=9090 # override a port without touching the base env
DEBUG=false # environment-specific toggle
Local env is merged on top of the base env before pushing. Use this for machine-specific or temporary overrides — keep secrets and connection strings in GitLab CI variables via vars.yml.
Deploy safety
deploy push protects against accidental overwrites:
- compose — checks server version before pushing. If server differs from local, push is blocked:
run 'dlane deploy pull' first - deploy.sh — same pull-first check: blocked if server version differs from local
- .env — never overwritten if it already exists on the server (contains runtime state like
ACTIVE_COLOR) --force— bypasses compose and deploy.sh checks when you know what you're doing
dlane deploy pull gateway # fetch compose + deploy.sh + env + nginx from all targets
dlane deploy push gateway --yes # protected push (blocked if server differs)
dlane deploy push gateway --yes --force # skip version checks
dlane deploy status gateway # show running containers and active color
Workspace profiles
Each workspace can be pinned to a specific profile in workspace.yml:
default_profile: prod # always uses this profile, regardless of active global profile
All vars and ci commands use the workspace profile automatically. A warning is shown if it differs from the globally active profile.
CI variable management — masked variables
GitLab never returns masked variable values via the API (by design). This affects vars get:
- Non-masked variables — full round-trip:
get→ edit →apply✅ - Masked variables — write-only:
getfetches the key and metadata but not the value
When you run vars get, masked variables with no existing value in vars.yml are flagged:
⚠ 2 masked variable(s) have no value — fill in vars.yml before running 'vars apply'
- PROD_SSH_KEY
- REGISTRY_PASS
If vars.yml already exists, previously filled values are preserved. vars apply skips any masked variable with an empty value rather than pushing an empty string.
Recommendation: treat vars.yml as the source of truth. Don't rely on vars get to recover masked values — fill them in once and keep the file.
Security
- GitLab token stored at
~/.config/deploylane/config.tomlwith0600permissions - Masked variables shown as
<masked>in all output deploy pushrequires explicit--yesto apply changesdeploy pushis blocked by default if server state differs from local — rundeploy pullfirstdeploy installrequires explicit--yesand prompts for sudo password interactively.local.envfiles are gitignored automatically.envon the server is never overwritten — it holds runtime state (ACTIVE_COLOR, etc.)
Add to your .gitignore:
.workspace/
Command reference
Auth
| Command | Description |
|---|---|
dlane login |
Store credentials locally and verify against the API |
dlane logout |
Remove stored credentials |
dlane whoami |
Show current user for the active profile |
dlane status |
Show active profile and login status |
dlane profile list |
List stored profiles |
dlane profile use <name> |
Switch active profile |
Workspace
| Command | Description |
|---|---|
dlane init |
Create workspace in current directory |
dlane add <name> |
Add a service to the workspace |
dlane update <name> |
Update service fields |
dlane remove <name> |
Remove a service |
dlane list |
List services with status |
dlane scaffold <name> |
Generate / refresh deployment files |
Variables
| Command | Description |
|---|---|
dlane vars get <name> |
Fetch GitLab variables → vars.yml |
dlane vars plan <name> |
Preview what would change |
dlane vars diff <name> |
Diff local vs GitLab |
dlane vars apply <name> |
Push vars.yml → GitLab |
dlane vars prune <name> |
Delete GitLab vars not in vars.yml (shows list + confirms before deleting) |
Deploy
| Command | Description |
|---|---|
dlane deploy diff <name> |
Show what would change on the server without pushing |
dlane deploy pull <name> |
Fetch compose, deploy.sh, env, nginx from server (all targets) |
dlane deploy push <name> |
Push compose + deploy.sh to server (pull-first protected, never overwrites .env) |
dlane deploy install <name> |
One-time server setup (nginx + sudoers + install.sh) |
dlane deploy status <name> |
Show running containers and active color per target |
dlane deploy history <name> |
Show deployment history |
CI
| Command | Description |
|---|---|
dlane ci lint <name> |
Validate local .gitlab-ci.yml via GitLab lint API |
dlane ci pull <name> |
Fetch .gitlab-ci.yml from GitLab repo → local |
dlane ci push <name> |
Lint + push local .gitlab-ci.yml to GitLab via MR |
Tools
| Command | Description |
|---|---|
dlane --version |
Show installed version |
dlane gitlab list |
Browse GitLab projects |
Platform support
| Platform | vars | ci | deploy |
|---|---|---|---|
| GitLab | ✅ Full | ✅ Full (MR flow) | ✅ Full |
| GitHub Actions | 🧪 Experimental | ❌ Not yet | ✅ Full (SSH-based) |
GitHub Actions variable/secret management is available via pip install deploylane[github] but is not yet covered by the full command set (ci pull/push is GitLab-only for now).
Development
git clone https://github.com/ahmetatakan/deploylane
cd deploylane
pip install -e ".[dev]"
python -m build
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 deploylane-0.3.5.tar.gz.
File metadata
- Download URL: deploylane-0.3.5.tar.gz
- Upload date:
- Size: 65.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.10
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d8fce9a3c07b1c358a8c05493a0dd4a0b98c0a16c8bf22c20e4f019e2948b16e
|
|
| MD5 |
0847124fa80d2e3f30734139177de20e
|
|
| BLAKE2b-256 |
d8932065ac6a76cb0b5d3c0128b410fef4920bde8f1180f6a0dba52e9053a53e
|
File details
Details for the file deploylane-0.3.5-py3-none-any.whl.
File metadata
- Download URL: deploylane-0.3.5-py3-none-any.whl
- Upload date:
- Size: 70.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.10
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
258d41e8bceb517781e6c0f6870f72cd918af92ae84588c79650bc9d0b5c0e60
|
|
| MD5 |
4662351b009aae7b239e9068f7c1431d
|
|
| BLAKE2b-256 |
250f51cd2d3a4ecd57cc1177c3c20f3e2c6a8b741cbdf3a0f38b23c48af64b3c
|