Versioning And Updates
Release metadata, update flow, preflight checks, and deployment/update mechanics.
Current release: 1.4.13 (2026-05-12)
See changelog.md for full history.
How versioning works
version.jsonin the repository root is the authoritative version file.GET /api/versionreturns the installed version from that file.- The admin General settings route (
#/admin/settings/general) shows the installed version and offers a Check for Updates button, which fetchesversion.jsonfromUPDATE_VERSION_URLwhen set, otherwise from the public GitHubmainbranch, and compares. - Admin sessions also perform a silent background update check at most once per day via the notification center. Overlapping checks for the same admin are serialised server-side before fetching upstream metadata, and if a newer version is found, an in-app notification is created linking directly to
#/admin/settings/general. - When an update is available, an Update Now button opens a preflight checklist followed by a one-click updater that downloads, extracts, copies files, and runs database migrations automatically.
- The preflight now checks both writable directories and whether existing files are actually overwriteable by the PHP process. This matters on shared hosting, where PHP often cannot replace files owned by your account even if the directory itself appears writable.
- Release metadata may declare
minimum_update_from. When set, the updater blocks direct upgrades from older installs to newer releases and tells the admin which bridge version to install first. - The version metadata source can be overridden with
UPDATE_VERSION_URL. The updater package source can be overridden withUPDATE_REPO_ZIP_URLandUPDATE_REPO_PREFIX; otherwise it uses the public GitHub metadata to prefer the full release package for the latest version, falling back to the GitHub source zip if the package is unavailable. - When a new release is ready, prepare the
docs/changelog.mdUnreleasedsection with the relevant notes, then runmake release. That target bumpsversion.json, updatesdocs/version.md, updates the client version inpublic_html/index.php, converts the preparedUnreleasednotes into the new versioned changelog section, stages the current worktree, commits, and pushes. make deploynow runsphp bin/migrate.phpon the remote host aftercomposer install, so numbered DB migrations are applied during normal rsync deployments as well as in-app updates.
In-app updater
The updater (src/Core/UpdateController.php) performs the following steps:
- Preflight (
GET /api/update/preflight) — checks upgrade-path compatibility, ZipArchive extension, download capability, a PHP dependency update path, write permissions on key directories includingvendor/, overwriteability of existing files in those directories, temp directory, and disk space. Returns pass/fail per check with specific fix instructions. - Run (
POST /api/update/run) — downloads the full GitHub release package when available, otherwise falls back to the configured/source zip, extracts it, copies files over the installation (skipping.env,storage/,.git,install.lock,Makefile.local, anddocs/videos), updates PHP dependencies from packagedvendor/files or by runningcomposer install --no-dev --optimize-autoloader, runsschema.sql, applies new migrations tracked in theschema_migrationsDB table, and resets the opcode cache.
The full GitHub release package includes vendor/, so shared-hosting installs without Composer can still receive new PHP dependencies through the web updater as long as the PHP process can overwrite the application files. If a custom UPDATE_REPO_ZIP_URL points at a source archive without vendor/, Composer must be installed and executable by PHP.
Bridge Update Requirement
Version 1.4.9 is the updater bridge release. Installs older than 1.4.9 must update to 1.4.9 before updating to 1.4.10 or newer, because 1.4.9 adds support for full release packages and packaged vendor/ dependencies.
Current and future releases can enforce this by setting these fields in version.json:
{
"minimum_update_from": "1.4.9",
"minimum_update_reason": "Version 1.4.9 upgrades the updater so full release packages and packaged vendor dependencies can be installed safely."
}
This cannot be enforced retrospectively by updater code that is already installed on a pre-1.4.9 host, because that old updater does not know about bridge metadata. Those hosts should be manually directed to update to 1.4.9 first.
Database migrations are tracked in schema_migrations (auto-created on first update or CLI migrate run). Migration 001_initial.sql is always skipped as it is covered by schema.sql. New migrations are applied in filename order. The updater lock remains held until schema and migration work completes, preventing overlapping update runs from racing each other.