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
- build_dev_from_source.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: 1. BumpBASE_IMAGE_VERSIONininstall.sh2. Run./install.sh build-baseon the host 3. Rebuild each site container (see migration steps inspecs/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
build_dev_from_source.sh
Location: /var/www/html/joinerytest/maintenance_scripts/install_tools/build_dev_from_source.sh
> Note: This script is functional but not recommended for production. Use upgrade.php for production deployments. build_dev_from_source.sh is suitable for development environments where git-based deployment is convenient.
# Basic deployment
./build_dev_from_source.sh joinerytest
# Verbose mode (recommended)
./build_dev_from_source.sh joinerytest --verbose
# Disable auto-rollback for debugging
./build_dev_from_source.sh joinerytest --norollback
# Manual rollback
./build_dev_from_source.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
└── ...Component version integrity:
Each publish records a per-release snapshot of every published component's (version, tree_hash) in upg_upgrades.upg_component_state (JSON, keyed by themes/plugins). The snapshot of the most recent release row that carries one is the baseline for change detection; rows without a parseable snapshot are skipped, so an aborted publish never poisons the baseline. When no prior snapshot exists, the run re-baselines — records everything, bumps nothing.
The tree_hash is a deterministic, git-independent SHA-256 of the component's working tree (.git/ and .gitignore excluded). The manifest (theme.json / plugin.json) is hashed with its version member removed, so the hash measures content-minus-version.
Per component, before archiving, the publisher applies a four-way decision against the baseline entry:
- No baseline entry — first publish of this component: record as-is, no bump.
- Manifest version higher — author bumped deliberately: respect and record.
- Manifest version lower — record and archive as-is, with a warning line in the publish summary naming the component and both versions. Not aborted: a backward component version has no destructive effect at publish time, and any resulting
dependsviolation is caught fail-closed at activation. - Manifest version equal — compare hashes. Equal → unchanged, carry the entry forward. Differ → auto patch-bump the manifest's
version(targeted string edit, no reformatting) and archive under the new version.
VERSION file. Authors still bump minor/major for meaningful releases; auto patch-bump is the floor that keeps archive filenames honest when a content change ships without one.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 on-disk copy is replaced from the upgrade payload during a deploy swap. Iffalse, the live copy is preserved across the swap and_reconcile_upgradable_assets.shwill not re-download it. Mirrored to the database columnthm_receives_upgrades/plg_receives_upgradesso the admin UI can toggle it; uploaded extensions are auto-set tofalseso a deploy doesn't wipe them.included_in_publish— publisher-side, packaging filter. Iftrue,publish_upgrade.phppackages this extension into the upgrade archive andpublish_theme.php's catalog endpoint advertises it. Iffalse, 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.
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.
Agent File Regeneration
After migrations and plugin sync, update_database regenerates DB-managed agent files (CLAUDE.md, GEMINI.md, etc.) from the agf_agent_files table — the table is the source of truth, the on-disk files are generated output. Only rows previously written to disk (agf_last_written_time IS NOT NULL) are regenerated, so a never-written customer baseline row stays dormant until the customer opts in.
A drift guard protects out-of-band edits: if a target file on disk was changed since it was last written (its sha256 no longer matches agf_last_written_hash), the row is skipped with a warning rather than overwritten. Resolve a skipped row from /admin/admin_agent_files — writing from there prompts for confirmation and backs the on-disk copy up as <filename>.old before overwriting. See specs/implemented/agent_files_management.md for the full design.
Agent File Upgrades (Customer Baseline)
The default_agents_template.md file ships inside the upgrade tarball. When update_database runs it compares the template's normalized SHA-256 against each Customer baseline row's agf_template_baseline_hash:
| Row state | Result |
|---|---|
| Baseline hash is null (pre-feature install) | Skipped — no surprise updates |
| Hash matches template | Already up to date, no action |
| Row content matches new template | Admin hand-applied the update; baseline hash bumped silently |
| Row content unchanged from its baseline | Auto-upgraded in place; new content and hash written; regeneration step picks it up |
| Row edited and template changed | A candidate row is created (or rolled forward if one already exists) |
/admin/admin_agent_files with a "Candidate for #N" badge and an inline panel:> An updated agent template is available. [Compare] [Switch to new version]
- Compare opens a read-only side-by-side view of the current content vs the candidate.
- Switch to new version moves target filenames from the active row to the candidate, archives the previously-active row (name prefixed
Archived —, target filenames cleared), and writes the new content to disk.
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
The Joinery vhost bakes a RewriteCond %{HTTP:CF-Visitor} !"scheme":"https" guard into the redirect rule, so it cannot loop in any CF SSL mode (or with no CF at all). The admin Settings page also surfaces a yellow warning banner whenever the platform detects it's being served from Flexible-mode CF — so the misconfig is visible without needing to read logs.
Apache Vhost
Every site installed by install.sh has the same Apache vhost shape, regardless of whether it sits behind Cloudflare, behind another CDN, or is exposed directly to the public internet, and regardless of whether the origin has its own TLS certificate.
Shape. Defined by template files in maintenance_scripts/install_tools/ (single source of truth per deployment mode) — default_proxy_vhost.conf for Docker reverse-proxy sites, default_virtualhost.conf for bare-metal sites. install.sh write_universal_vhost substitutes the placeholders ({{DOMAIN_NAME}}, {{SITE_NAME}}, {{PORT}}, {{SERVER_IP}}) and writes the result to /etc/apache2/sites-available/${sitename}.conf.
- Port 80 — proxies/serves traffic, plus a
RewriteRulethat redirects to HTTPS. The redirect carries aCF-Visitorguard so it cannot loop under CF Flexible mode and is a no-op when no CF is in front. - Port 443 — wrapped in
<IfFile /etc/letsencrypt/live/${domain}/fullchain.pem>. Apache evaluates<IfFile>at config-parse time: if the cert exists, the:443vhost activates; if not, Apache silently skips the block. Sites with no origin cert just serve port 80, and whatever's in front (Cloudflare etc.) handles TLS at the edge.
install.sh runs provision_origin_cert once during install:- Domain resolves to this server → Let's Encrypt HTTP-01 challenge via
certbot --apache --no-redirect. - Domain resolves elsewhere (Cloudflare, other CDN) and a matching DNS-API credentials file exists at
/etc/letsencrypt/<provider>.ini→ LE DNS-01 with the matching certbot plugin. Plugin map:
*.ns.cloudflare.com | certbot-dns-cloudflare | /etc/letsencrypt/cloudflare.ini |
| awsdns-* | certbot-dns-route53 | /etc/letsencrypt/route53.ini |
| ns[1-5].linode.com | certbot-dns-linode | /etc/letsencrypt/linode.ini |
| ns[1-3].digitalocean.com| certbot-dns-digitalocean | /etc/letsencrypt/digitalocean.ini |Each plugin reads its credential file in its own standard format; certbot's docs cover the schemas.
- Neither path produces a cert → install proceeds without origin SSL. The
:443vhost stays dormant via the<IfFile>guard; CF or another front-end handles TLS.
detect_dns_provider in install.sh — add one case clause mapping the provider's NS-record signature to its tag, and document the plugin package + credential file format in the table above.Enabling origin SSL later (e.g. after dropping a CF API token in place to switch a CF zone to Full strict): sudo /var/www/html/<site>/maintenance_scripts/sysadmin_tools/setup_ssl.sh <domain>. The script re-enters the decision tree; the :443 vhost begins serving on the next Apache reload because the <IfFile> guard sees the new cert.
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://dev.getjoinery.com) |
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 archives
Last Updated: 2026-05-16