Deploy and Upgrade Systems
Overview
Five complementary tools provide deployment and upgrade capabilities:
- upgrade.php - Web-based upgrade system for client installations (recommended)
- publish_upgrade.php - Package creation tool for distributing updates (core + themes + plugins) — lives in the Server Manager plugin
- publish_theme.php - Individual theme/plugin publishing — lives in the Server Manager plugin
- install.sh - Universal installer for Docker and bare-metal deployments
- deploy.sh - Git-based deployment for development environments (not recommended for production)
/includes/DeploymentHelper.php) for shared validation, rollback, and theme/plugin preservation. Tools 2 and 3 require the Server Manager plugin to be active.For Docker and bare-metal deployments, see Installation Guide.
Docker Shared Base Image
Docker site images build FROM joinery-base:VERSION rather than FROM ubuntu:24.04. The base image contains Ubuntu + Apache + PHP 8.3 + PostgreSQL + Composer + cron and is shared across all site containers on a host. Per-site images only layer the site code, config, and VirtualHost on top.
Two-step build on a Docker host:
# 1. One-time per host — build the shared base image (~5-10 minutes, ~2.3 GB).
./install.sh build-base
# 2. Create sites normally — each site image builds in seconds and is ~500 MB.
./install.sh site mysite mysite.com 8080install.sh site refuses to run if joinery-base:VERSION is missing and tells you to run build-base first.
BASE_IMAGE_VERSION is a constant at the top of install.sh. Bump it and run build-base again whenever the system stack changes:
- Ubuntu base version changes
- PHP major/minor version changes
- New apt packages or PHP extensions added to
do_server_setup - Any other change to
Dockerfile.base
install.sh do_server_setup hash differs from the hash baked into the base image (stored as the joinery.install_sh_hash label). That's the signal to bump BASE_IMAGE_VERSION and rebuild the base.Two-tier Apache: real client IP
A docker-prod request crosses two Apache instances: host Apache terminates TLS and reverse-proxies to 127.0.0.1:{container_port}; container Apache runs PHP. Without help, $_SERVER['REMOTE_ADDR'] inside the container is always 172.17.0.1 (the docker bridge gateway), which silently breaks IP-based features (rate limiting, API key IP restriction, analytics, audit logs).
The contract:
- Host proxy (written by
install.shandmanage_domain.sh) setsRequestHeader set X-Forwarded-For %{REMOTE_ADDR}s— explicitset(not append) so the container receives a single trustworthy value. - Container Apache loads
mod_remoteipwithRemoteIPInternalProxy 172.17.0.0/16, rewritingREMOTE_ADDRfrom theX-Forwarded-Forheader before PHP runs. This is baked intoDockerfile.template(since v3.5). - Access logs use
%ainstead of%hso they show the rewritten address, not the bridge gateway.
CF-Connecting-IP.Upgrade-flow split (important)
This is the behavioural change most likely to trip up an operator who remembers the pre-shared-base model:
- Code / theme / plugin changes (PHP files under
public_html/, migrations, settings) — deliver via the existing publish/upgrade pipeline (publish_upgrade.php+upgrade.php). No base image work required. Nothing changes here. - System stack changes (new apt package, new PHP extension, Ubuntu bump, PHP bump, anything in
do_server_setup) — now require base rebuild + container rebuild, not justupgrade.php.upgrade.phprefreshes the application layer only; it cannot modify a running container's system packages. Operators must:
BASE_IMAGE_VERSION in install.sh
2. Run ./install.sh build-base on the host
3. Rebuild each site container (see migration steps in specs/implemented/docker_shared_base_image.md)Distribution Architecture
Updates are distributed as separate archives:
- Core archive (
joinery-core-X.XX.upg.zip) - Main application without themes/plugins - Theme archives (
theme-THEMENAME-X.XX.upg.zip) - Individual themes - Plugin archives (
plugin-PLUGINNAME-X.XX.upg.zip) - Individual plugins
- Independent versioning of themes and plugins
- Selective updates (update core without touching themes)
- Smaller download sizes for incremental updates
- Third-party theme/plugin distribution
Quick Reference
install.sh
Location: /var/www/html/joinerytest/maintenance_scripts/install_tools/install.sh
Universal installer for Docker and bare-metal deployments. Supports --themes flag to download published themes/plugins from the upgrade server after site creation (extensions whose manifests have included_in_publish: true).
Full documentation: Installation Guide
deploy.sh
Location: /var/www/html/joinerytest/maintenance_scripts/install_tools/deploy.sh
> Note: This script is functional but not recommended for production. Use upgrade.php for production deployments. deploy.sh is suitable for development environments where git-based deployment is convenient.
# Basic deployment
./deploy.sh joinerytest
# Verbose mode (recommended)
./deploy.sh joinerytest --verbose
# Disable auto-rollback for debugging
./deploy.sh joinerytest --norollback
# Manual rollback
./deploy.sh joinerytest --rollbackFeatures:
- Git-based deployment from repository
- Pre-deployment validation (PHP syntax, plugin loading, bootstrap tests)
- Automatic rollback on failure (trap-based)
- Preserves extensions marked
receives_upgrades: false - Composer integration and database migrations
upgrade.php
Location: /utils/upgrade.php
Web Usage:
# Check for upgrades
https://yoursite.com/utils/upgrade?serve-upgrade=1
# Perform upgrade (verbose)
https://yoursite.com/utils/upgrade?verbose=1CLI Usage:
# Basic upgrade
php /var/www/html/joinerytest/public_html/utils/upgrade.php
# Verbose mode
php /var/www/html/joinerytest/public_html/utils/upgrade.php --verboseFeatures:
- Downloads packages from upgrade server (configured via
upgrade_sourcesetting) - Downloads core, themes, and plugins as separate archives
- Pre-deployment validation via DeploymentHelper
- Preserves extensions marked
receives_upgrades: false - Enhanced rollback (preserves failed deployments with timestamps)
- Database migrations and composer integration
- Graceful handling of missing archives — if a theme or plugin archive returns 404, the upgrade warns and skips it instead of aborting. The core upgrade and all other themes/plugins proceed normally. A summary of skipped items is shown at the end.
plg_plugins) and attempts an archive fetch for each. Plugins published by the source succeed; plugins not in the source's catalog 404 at the upgrade endpoint (they were never packaged because they have included_in_publish: false — see Extension Distribution Flags below) and are skipped via the warning path above. Uninstalling a plugin removes its row, so an uninstalled plugin is not re-downloaded on subsequent upgrades — the operator's removal sticks. Conversely, a new upstream plugin won't auto-appear on existing sites; the operator gets it via the admin Plugins page (install a plugin already on disk) or a plugin upload.The two distribution flags on the plugin's manifest govern the distribution pipeline: included_in_publish controls what publish_upgrade.php packages (publisher-side), while receives_upgrades controls what DeploymentHelper preserves across a deploy swap and what _reconcile_upgradable_assets.sh re-downloads on container boot (customer-side). The upgrade-time refresh loop itself no longer filters by either flag — it just tries everything installed and lets the endpoint's response be the source of truth for whether a given plugin is in the publisher's catalog.
Download Flow:
- Fetches available upgrade info from upgrade server
- Downloads core archive (
joinery-core-X.XX.upg.zip) - Downloads each published theme archive (
theme-THEMENAME-X.XX.upg.zip) — themes the source published withincluded_in_publish: true - Downloads an archive (
plugin-PLUGINNAME-X.XX.upg.zip) for each plugin with a row inplg_plugins - If any theme/plugin archive is unavailable (404), logs a warning and continues
- Extracts and validates all archives
- Performs deployment with rollback protection
On any node detail page (/admin/server_manager/node_detail?mgn_id=N), the Updates tab exposes:
- Apply Update — single-site action, queues one
apply_updatejob for that node. - Upgrade All Sites on This Host — fans out to every enabled, non-deleted node sharing the same
mgn_host. Queues one independentapply_updatejob per sibling (so a per-site failure doesn't affect the others), then redirects to the Jobs page. To skip a specific site in the bulk run, disable it (mgn_enabled = false) via its node detail page first.
publish_upgrade.php
Location: plugins/server_manager/includes/publish_upgrade.php
Access: Requires the Server Manager plugin to be active. Superadmin only (permission level 10).
Preferred usage: Use the Publish Upgrade form on the Server Manager dashboard (/admin/server_manager). Enter release notes and submit — the plugin creates a job that builds all archives.
CLI usage:
php plugins/server_manager/includes/publish_upgrade.php "release notes here"
# Auto-detects the next version number> Note: The legacy location utils/publish_upgrade.php still exists for backward compatibility during the Phase 1 transition. It will be removed in a future release once all remote nodes have been upgraded.
Features:
- Creates separate archives for core, themes, and plugins
- Core archive excludes theme/ and plugins/ directories
- Each theme/plugin with
included_in_publish: truegets its own versioned archive - Prevents overwriting existing versions
- Automatic cleanup on failure
- Registers upgrade in stg_upgrades table
static_files/
├── joinery-core-3.26.upg.zip # Core application
├── theme-falcon-3.26.upg.zip # Falcon theme
├── theme-default-3.26.upg.zip # Default theme
├── plugin-bookings-3.26.upg.zip # Bookings plugin
├── plugin-controld-3.26.upg.zip # ControlD plugin
└── ...publish_theme.php
Location: plugins/server_manager/includes/publish_theme.php
Access: Requires the Server Manager plugin to be active. Superadmin only (permission level 10).
# Publish a single theme
https://yoursite.com/admin/server_manager/publish_theme?type=theme&name=falcon&version=1.0.0
# Publish a single plugin
https://yoursite.com/admin/server_manager/publish_theme?type=plugin&name=bookings&version=2.1.0
# List available themes (used by marketplace and upgrade.php)
https://yoursite.com/admin/server_manager/publish_theme?list=themes> Note: The legacy location utils/publish_theme.php still exists for backward compatibility during the Phase 1 transition.
Features:
- Publishes individual themes or plugins independently of core
- Allows different versioning for themes/plugins vs core
- Useful for third-party theme/plugin distribution
- Validates theme.json/plugin.json exists before packaging
- Serves catalog listings for the marketplace and
upgrade.php
How It Works
Deployment Flow
1. Download/extract to staging directory
2. DeploymentHelper validates:
- PHP syntax on all files
- Plugin class loading
- Bootstrap/core components
3. DeploymentHelper preserves extensions marked `receives_upgrades: false`
4. Backup current installation to public_html_last/
5. Deploy staged files to public_html/
6. Run database migrations (update_database.php)
7. Run composer_install_if_needed.php
8. Fix permissions (www-data:user1, 775)
If ANY step fails → Automatic rollbackDirectory Structure
/var/www/html/{site}/
├── public_html/ # Current live installation
├── public_html_last/ # Backup (for rollback)
├── public_html_stage/ # Staging area for validation
├── public_html_failed_*/ # Preserved failed deployments (timestamped)
├── static_files/ # Published upgrade packages
│ ├── joinery-core-X.XX.upg.zip
│ ├── theme-THEMENAME-X.XX.upg.zip
│ └── plugin-PLUGINNAME-X.XX.upg.zip
└── uploads/upgrades/ # Downloaded packages (client sites)Archive Naming Convention:
joinery-core-X.XX.upg.zip- Core application (no themes/plugins)theme-{name}-X.XX.upg.zip- Individual theme archiveplugin-{name}-X.XX.upg.zip- Individual plugin archive
Extension Distribution Flags
Themes and plugins carry two independent boolean flags in their manifests
(theme.json / plugin.json) that control distribution. Both default to true
when missing, and they govern different sides of the pipeline:
Example manifest (theme.json or plugin.json):
{
"name": "controld",
"version": "2.1.0",
"description": "ControlD DNS management plugin",
"receives_upgrades": true,
"included_in_publish": true
}receives_upgrades— customer-side, deploy preservation. Iftrue, the
false, the live copy is preserved across the swap and _reconcile_upgradable_assets.sh
will not re-download it. Mirrored to the database column
thm_receives_upgrades / plg_receives_upgrades so the admin UI can
toggle it; uploaded extensions are auto-set to false so a deploy doesn't
wipe them.
included_in_publish— publisher-side, packaging filter. Iftrue,
publish_upgrade.php packages this extension into the upgrade archive and
publish_theme.php's catalog endpoint advertises it. If false, it is
skipped. Manifest-only — there is no DB column and no admin UI for this
flag, since it has no meaning on a customer site.If a manifest is missing, it's auto-generated with both flags true.
update_database Behavior
Advisory Lock
update_database.php uses a PostgreSQL advisory lock (pg_try_advisory_lock(99999)) to prevent concurrent runs. If a second process tries to run while one is already in progress, it exits immediately with "already running." The lock is released automatically when the database connection closes.
Halt on Migration Failure
Migrations stop on the first failure — subsequent migrations are skipped. Fix the failing migration and re-run update_database.php to continue.
Migration test Semantics
Each migration has an optional test SQL query that returns a row with a count column. The runner interprets it as:
count > 0→ migration is skipped (already applied)count = 0→ migration runs
// Insert a row — skip if it already exists
$migration['test'] = "SELECT count(1) as count FROM emt_email_templates WHERE emt_name = 'my_template'";
$migration['migration_sql'] = "INSERT INTO emt_email_templates (emt_name, emt_body) VALUES ('my_template', '...')";> Note: do not use migrations to seed stg_settings rows. Setting names and defaults are declarative — see "Declarative Settings (no migration)" below.
Drop-table migrations require inverted logic. If you test for the table's presence the same way, the migration is skipped while the table still exists — the opposite of what you want. Use a CASE expression to flip the sense:
// Drop a table — run while table is present, skip once it's gone
$migration['test'] = "SELECT CASE WHEN EXISTS(
SELECT 1 FROM pg_tables WHERE tablename = 'old_table' AND schemaname = 'public'
) THEN 0 ELSE 1 END as count";
$migration['migration_sql'] = 'DROP TABLE IF EXISTS public.old_table CASCADE;';The CASE returns 0 while the table exists (→ run) and 1 once it has been dropped (→ skip). The DROP TABLE IF EXISTS makes the migration idempotent — safe to run even if the table is already gone.
Declarative Settings (no migration)
Setting names and defaults are declared, not migrated. Every update_database run reseeds them via Setting::seed_declared(), which uses INSERT ... ON CONFLICT (stg_name) DO NOTHING — existing rows are never overwritten, only missing ones are filled in.
- New core setting → add an entry to
public_html/settings.jsonwith a sensible default. Reseeded automatically on existing sites; included injoinery-install.sqlfor fresh installs. - New plugin-owned setting → add an entry to the plugin's
plugin.jsonundersettings. Seeded byPluginManager::syncSettings()when the plugin is activated. - Changing the default value of an existing setting → edit
settings.json(orplugin.json). Existing sites keep whatever value they have (ON CONFLICT DO NOTHING). If you also need to correct a wrong value on existing sites, add an UPDATE migration with a tight WHERE clause (e.g.WHERE stg_value = '<old default>'so admin overrides aren't trampled).
stg_settings migrations are deprecated. They duplicate what seed_declared already does, drift from the declarative source, and clutter migration history. UPDATE/DELETE migrations against stg_settings remain a valid tool — only INSERT-only seed migrations are off-limits.The same principle applies to core admin/profile menu rows (declared in public_html/admin_menus.json) and plugin menu rows (declared in plugin.json under adminMenu / profileMenu).
Plugin Tables Excluded
update_database.php always runs with include_plugins => false. Plugin tables are managed through the plugin activation workflow (PluginManager::activate() calls DatabaseUpdater::runPluginTablesOnly()), not through the core updater. This is intentional — core can't know about plugins at compile time.
DeploymentHelper API
Validation:
DeploymentHelper::validatePHPSyntax($directory, $verbose)
DeploymentHelper::testPluginLoading($stage_dir, $verbose)
DeploymentHelper::testBootstrap($stage_dir, $verbose)Theme/Plugin Preservation:
DeploymentHelper::preserveExtensionsAcrossDeploy($stage_dir, $backup_dir, $verbose)Rollback:
DeploymentHelper::performRollback($target_site, $preserve_failed, $verbose)All methods return structured arrays with success status, errors, and detailed results.
Common Issues
Permission Errors:
sudo chown -R www-data:user1 /var/www/html/joinerytest/public_html
sudo chmod -R 775 /var/www/html/joinerytest/public_htmlValidation Failures:
- Failed deployment preserved in
public_html_failed_*directory - Inspect for syntax errors or missing dependencies
- Fix and redeploy
- Check manifest has
"receives_upgrades": false - Restore from public_html_last/ if needed
- Check public_html_last/ exists
- Manually restore from backup
- Fix permissions after restore
Configuration
Required settings (in /config/Globalvars_site.php or stg_settings):
| Setting | Description |
|---|---|
baseDir | Base directory (e.g., /var/www/html/) |
site_template | Site directory name (e.g., joinerytest) |
system_version | Current version (e.g., 3.25) |
upgrade_source | URL of upgrade server to download from (e.g., https://joinerytest.site) |
composerAutoLoad | Composer vendor path |
upgrade_source setting specifies where a site downloads upgrades from.Marketplace
The marketplace admin page lets superadmins browse themes and plugins available on the upgrade server and install them with one click.
Admin Page: Server Manager > Marketplace (permission level 8)
Files: plugins/server_manager/views/admin/marketplace.php, plugins/server_manager/logic/admin_marketplace_logic.php
> Note: The old URL /admin/admin_marketplace redirects to /admin/server_manager/marketplace.
How It Works
- Fetches catalog from the upgrade server (
publish_theme.php?list=themesand?list=plugins) - Compares with locally installed themes/plugins
- Shows a card grid with install buttons for items not yet installed
- Install downloads the tar.gz archive and extracts it via
AbstractExtensionManager::installFromTarGz() - After install, files are on disk and synced to the database — user must activate separately via Themes or Plugins admin page
Prerequisites
upgrade_sourcesetting must be configured (URL of the upgrade server)- The upgrade server must have the Server Manager plugin active
Overwrite Protection
- Extensions with
receives_upgrades: true(or those without a manifest) can be reinstalled/replaced from the marketplace - Extensions with
receives_upgrades: falseare protected — the marketplace refuses to overwrite them
Catalog Endpoint Fields
The publish_theme.php catalog endpoints (?list=themes, ?list=plugins) include:
name— display name (unchanged for backward compatibility)directory_name— filesystem directory name (used for matching and downloads)display_name,version,description,author,is_system,included_in_publish
Related Documentation
- CLAUDE.md - System architecture and development guidelines
- Plugin Developer Guide - Plugin development
- Server Manager - Server management, publishing, and backup targets
- Specifications:
/specs/implemented/upgrade_system.md - Feature parity analysis
- /specs/implemented/fix_publish_upgrade_system.md - Publish upgrade fixes
- /specs/implemented/theme_plugin_distribution_refactor.md - Separate archive distribution
- /specs/implemented/server_manager_publish_upgrade.md - Moving publish/upgrade into server_manager plugin
- /specs/implemented/upgrade_graceful_theme_download.md - Graceful handling of missing archivesLast Updated: 2026-04-22