Deployment
This document describes how @xmachines/xmachines-js packages are built, versioned, and published to npm. All releases are automated through semantic-release running inside a GitLab CI pipeline. No manual npm publish commands are required for normal releases.
Deployment Targets
All @xmachines/* packages are published to the public npm registry (registry.npmjs.org). Every package uses "publishConfig": { "access": "public" } so scoped packages are accessible without an npm org subscription. See the Published Packages table for the full list.
| Target | Config File | Purpose |
|---|---|---|
| npm registry | .releaserc.json (per-package @semantic-release/npm plugin entries) | Publish all public packages |
| GitLab Releases | .releaserc.json (@semantic-release/gitlab plugin) | Attach tarball artifacts to the GitLab release tag |
| GitLab CI | .gitlab-ci.yml | Trigger builds, tests, and releases on push / MR / tag |
The root package.json is marked "private": true and is never published to npm.
Published Packages
The following packages are published on every release (derived from .releaserc.json):
| Package | Directory |
|---|---|
@xmachines/shared | packages/shared |
@xmachines/play | packages/play |
@xmachines/play-actor | packages/play-actor |
@xmachines/play-signals | packages/play-signals |
@xmachines/play-router | packages/play-router |
@xmachines/play-dom | packages/play-dom |
@xmachines/play-dom-router | packages/play-dom-router |
@xmachines/play-react | packages/play-react |
@xmachines/play-react-router | packages/play-react-router |
@xmachines/play-solid | packages/play-solid |
@xmachines/play-solid-router | packages/play-solid-router |
@xmachines/play-svelte | packages/play-svelte |
@xmachines/play-sveltekit-router | packages/play-sveltekit-router |
@xmachines/play-svelte-spa-router | packages/play-svelte-spa-router |
@xmachines/play-tanstack-react-router | packages/play-tanstack-react-router |
@xmachines/play-tanstack-solid-router | packages/play-tanstack-solid-router |
@xmachines/play-vue | packages/play-vue |
@xmachines/play-vue-router | packages/play-vue-router |
@xmachines/play-xstate | packages/play-xstate |
@xmachines/docs | packages/docs |
Example demo packages (under packages/*/examples/demo) are not published to npm ("npmPublish": false) but are still packed into release tarballs and attached as GitLab release assets for testing.
Each published package includes only the directories listed in its files field: dist, README.md, and LICENSE (or equivalent for packages without a compilation step).
Build Pipeline
CI Pipeline Stages
Pipeline runs are triggered by .gitlab-ci.yml under the following conditions:
| Trigger | Pipeline runs |
|---|---|
Push to main / master / beta / pre/rc | Full pipeline including release |
| Merge request | Full pipeline (release job skipped or manual) |
Tag push ($CI_COMMIT_TAG) | Full pipeline |
The pipeline includes two reusable CI components:
to-be-continuous/node/gitlab-ci-node@5.1.2— handles install, lint, build, test, and auditto-be-continuous/semantic-release/gitlab-ci-semrel@4.1.0— runssemantic-releaseon eligible branches
The node-build job runs with:
npm run build # tsc --build (TypeScript composite build)npm test:coverage # vitest run --coverage (with JUnit + Cobertura reporters)Coverage is extracted from the node-build job via the regex:
/All files[^|]*\|[^|]*\s+([\d\.]+)/JUnit results are published as reports/junit.xml and Cobertura coverage as reports/coverage/cobertura-coverage.xml.
Release Job Detail
The semantic-release CI job is configured with:
semantic-release: id_tokens: NPM_ID_TOKEN: aud: "npm:registry.npmjs.org" artifacts: when: always expire_in: 1 week paths: - dist/releases/*.tgzThe job uses a GitLab OIDC id token (NPM_ID_TOKEN) scoped to npm:registry.npmjs.org rather than a static NPM_TOKEN.
The semantic-release command is called with --no-ci to allow it to run inside CI without triggering its own CI-environment mode.
Release Smoke Test Job
The release-pack-smoke job is defined in .gitlab-ci.yml and runs on the same trigger rules as other CI jobs (automatically on main pushes and tags, manually on MRs):
npm run buildnode scripts/release-pack-smoke.mjsscripts/release-pack-smoke.mjs:
- Reads
.releaserc.jsonto find all@semantic-release/npmplugin entries wherenpmPublishis notfalse. - For each publishable package: runs
npm pack --jsonin the package directory. - Creates a temporary directory, runs
npm init -y, and installs the local tarball with--ignore-scripts. - Asserts the install succeeds — any missing files, broken exports, or pack-time errors surface here.
- Cleans up temp directories and tarballs after each check.
This smoke test runs before the actual publish so packaging problems are caught in CI rather than on npm.
Semantic Release Process
Branch Model
Releases are driven by commit history on protected branches:
| Branch | Channel | Tag format | Behaviour |
|---|---|---|---|
main | stable (default) | v1.2.3 | Full production release |
beta | beta | v1.2.3-beta.N | Beta pre-release |
pre/rc | pre/rc | v1.2.3-rc.N | Release candidate pre-release |
Tag format is v${version} (e.g., v1.0.0-beta.46).
Prepare Phase
Before any package is published, semantic-release runs the prepareCmd from .releaserc.json:
npm install --prefix . @npmcli/arboristnpm run apply-patchesnode scripts/set-workspace-versions.mjs ${nextRelease.version}npm run buildnpm run typedoc -w @xmachines/docs -- --gitRevision v${nextRelease.version}npm run format -w @xmachines/docsStep by step:
- Install Arborist — ensures the workspace introspection tool is available.
- Apply patches — runs
patch-packageto apply any patches in thepatches/directory. - Sync versions —
scripts/set-workspace-versions.mjssets the same release version across everypackage.jsonin the workspace (root + all packages), including cross-package workspace dependency ranges. - Build — runs
tsc --buildto produce compileddist/output in all packages. - Generate API docs — runs TypeDoc to regenerate
packages/docs/api/at the release git revision. - Format docs — runs
oxfmton the docs package to ensure consistent formatting.
Publish Phase
Each publishable package is published in sequence via individual @semantic-release/npm plugin entries. Each entry specifies:
pkgRoot— the package directory relative to the workspace roottarballDir—dist/releases/where the packed.tgzis saved
Example (from .releaserc.json):
["@semantic-release/npm", { "pkgRoot": "packages/play", "tarballDir": "dist/releases" }]Post-Release Phase
After all packages are published:
-
@semantic-release/gitcommits updated files back to the repository:CHANGELOG.mdpackage.jsonandpackage-lock.json(root)packages/*/{package.json,examples/*/package.json}(all workspace packages)packages/docs/api/**(generated API documentation)
Commit message format:
chore(release): <version><release notes>[skip ci] -
@semantic-release/gitlabcreates a GitLab Release entry attached to the version tag, with alldist/releases/*.tgztarballs as downloadable assets labelled “Release tarballs”. -
@semantic-release/changelogupdatesCHANGELOG.mdwith generated release notes.
Version Synchronisation
All packages in the monorepo share a single version at all times. This is enforced during release by scripts/set-workspace-versions.mjs, which:
- Sets
versionin every workspacepackage.json(including the root) to the new release version. - Rewrites
dependencies,devDependencies,peerDependencies, andoptionalDependenciesranges for any@xmachines/*workspace package to the exact new version (no^or~prefix — a strict pinned version is written).
This means packages always depend on the exact same version of sibling packages that was released together.
Credentials and Protected Variables
| Variable | Source | Purpose |
|---|---|---|
NPM_ID_TOKEN | GitLab CI OIDC (auto-generated per job) | Authenticates semantic-release to npm registry via OIDC token exchange |
GL_TOKEN / GITLAB_TOKEN | GitLab CI protected variable | Authenticates @semantic-release/gitlab for release creation and git push back |
CI_JOB_TOKEN | GitLab CI built-in | Used by the to-be-continuous components for GitLab API calls |
No static npm token is committed to the repository. The NPM_ID_TOKEN is a short-lived OIDC token generated for each CI job with audience npm:registry.npmjs.org.
To publish packages, a maintainer must:
- Have push access to a release-eligible branch (
main,beta,pre/rc). - Ensure the GitLab project CI/CD variables include the required tokens.
Artifacts
Release tarballs are stored as GitLab CI artifacts for 1 week after each pipeline:
dist/releases/*.tgzThe same tarballs are attached to the GitLab Release entry as permanent downloadable assets. Tarballs are not committed to git (the dist/ directory is gitignored).
The API documentation (packages/docs/api/) is committed to git by the post-release git commit and is the only generated artifact tracked in version control.
Rollback Procedure
There is no automated rollback mechanism. To revert a release:
-
npm deprecate (preferred) — deprecate the bad version on npm so it is hidden from installs:
Terminal window npm deprecate @xmachines/play@<bad-version> "Deprecated: use <good-version>"Repeat for each affected package.
-
npm unpublish — unpublish the bad version if it was published within 72 hours and no dependents exist:
Terminal window npm unpublish @xmachines/play@<bad-version> -
Patch release — push a
fix:commit tomain(or the relevant branch) to trigger a patch version release via semantic-release. This is the preferred long-term approach for production issues. -
GitLab Release — delete or edit the GitLab Release entry via the GitLab UI or API to remove broken tarballs. This does not affect what is on npm.
Local Release Dry-Run (Manual)
To preview what semantic-release would do without publishing:
# Install dependencies firstnpm ci
# Run semantic-release in dry-run modenpx semantic-release --dry-run --no-ciTo test package packaging locally (the same check run by release-pack-smoke in CI):
npm run buildnode scripts/release-pack-smoke.mjsThis packs every publishable package, installs each tarball in a temporary project, and verifies the install succeeds.
To manually version and pack a single package for local inspection:
# From the repo rootnpm run build -w packages/playnpm pack --workspaces --workspace=packages/playDo not run npm publish manually. All publishing must go through semantic-release in CI to keep CHANGELOG.md, git tags, and npm versions in sync.
Monitoring
The project does not include application runtime monitoring (no Sentry, Datadog, or OpenTelemetry) — it is a library monorepo, not a deployed service.
Post-release health checks:
- npm package availability — verify new versions appear on
https://www.npmjs.com/package/@xmachines/<pkg>after the pipeline completes. - GitLab pipeline status — monitor the
semantic-releasejob in the GitLab CI UI for the release commit. - GitLab Releases page — confirm the release tag and tarball assets appear at
https://gitlab.com/xmachin-es/xmachines-js/-/releases. packages/docs/api/commit — confirm the post-release git commit updated API docs on the target branch.