Skip to content

Latest commit

 

History

History
494 lines (339 loc) · 38.9 KB

monorepo.md

File metadata and controls

494 lines (339 loc) · 38.9 KB

Jetpack Monorepo Overview

Welcome to the Jetpack Monorepo! This document will give you some idea of the layout, and what is required for your project to fit in with our tooling.

Table of contents

Layout

Projects are divided into WordPress plugins, Composer packages, JS packages, and GitHub Actions.

  • WordPress plugins live in subdirectories of projects/plugins/. The directory name should probably match the WordPress plugin name, with a leading "jetpack-" removed if applicable.
  • Composer packages live in subdirectories of projects/packages/. The directory name should probably match the package name with the leading "Automattic/jetpack-" removed.
  • JS packages live in subdirectories of projects/js-packages/. The directory name should probably match the package name with the leading "Automattic/jetpack-" removed.
  • GitHub Actions live in subdirectories of projects/github-actions/. The directory name should match the action name with the leading "Automattic/action-" removed.

Tooling that's applicable to the monorepo as a whole, including tooling for generically handling projects, lives in tools/.

WordPress, being a part of the Docker environment, gets installed into the directory tools/docker/wordpress, with non-monorepo plugins stored in tools/docker/wordpress/wp-content/plugins.

Documentation that's applicable to the monorepo as a whole lives in docs/.

All GitHub Actions configuration for the monorepo, including CI, lives in .github. We should strive to make things here generic rather than specific to any one project.

  • Actual actions live in .github/actions/. If it doesn't have an action.yml file, it shouldn't be in there.
  • Pattern matchers (not associated with an action) go in .github/matchers/.
  • Other files specific to actions, including scripts used with run:, go in .github/files/.

Compatibility

All projects should be compatible with PHP versions WordPress supports. That's currently PHP 7.0 to 8.3.

First Time

First time working with the monorepo? We got you covered.

For the first time only:

  • From the root of the repo, run pnpm install && pnpm jetpack cli link (if you want the jetpack CLI tool installed globally) or pnpm install (if you don't).
  • That’s it. You won’t need to do that again unless you nuke your node_modules directory.

Once you’ve done that, it’s easy: run jetpack (or pnpm jetpack) while anywhere in the Jetpack repo. To explore on your own, run jetpack --help to see the available commands.

Jetpack Generate Wizard

Starting a new project? Great! Let the Jetpack Generate Wizard help jumpstart the files you need. To get started:

  • Make sure you're checked out to the branch you want.
  • Use the CLI command jetpack generate to start the process.
  • The wizard will walk you through the steps of starting a new package, plugin, or Github action.

Accepted Arguments

The wizard accepts a few arguments to speed things up:

  • [project type] - Accepted values: package, plugin, github-action
  • --name, --n - The name of your project (no spaces)

Example: jetpack generate plugin --name my_cool_plugin will generate plugin files for a plugin called my_cool_plugin under ../jetpack/projects/plugins

What's Included

The Jetpack Generate Wizard includes the following for each project:

All Projects:

  • composer.json
  • package.json
  • readme.md
  • license.txt
  • .gitignore

Packages

  • bootstrap.php
  • .gitkeep
  • .gitattributes
  • phpunit.xml.dist

Plugins

  • bootstrap.php
  • .gitkeep
  • .gitattributes
  • phpunit.xml.dist
  • readme.txt
  • A main plugin.php (plugin_name.php), with filled in header

Github Actions

  • action.yml

Next Steps

  • The wizard should auto-generate common information
  • Check things over to make sure it looks correct
  • If your project requires a build step, add steps to composer.json and package.json
  • Create a mirror repo if necessary. See Mirror repositories.

Project structure

We use composer.json to hold metadata about projects. Much of our generic tooling reads this metadata to customize handling of the project. Metadata keys used are:

  • .name: Generally "Automattic/jetpack-something". Used to report names in various places. For Composer packages, this must, of course, match the name on Packagist.
  • .version: If present, updated by tools/project-version.sh. This should not be included on Composer packages that will be served through Packagist.
  • .repositories: If you include a repository entry referencing monorepo packages, it must have .options.monorepo set to true. This allows the build tooling to recognize and remove it.
  • .scripts.build-development: If your project has a general build step, this must run the necessary commands. See Building for details.
  • .scripts.build-production: If your project requires a production-specific build step, this must run the necessary commands. See Building for details.
  • .scripts.test-coverage: If the package contains any tests, this must run the necessary commands to generate a coverage report. See Code coverage for details.
    • .scripts.skip-test-coverage: Run before .scripts.test-coverage in CI. If it exits with code 3, the test run will be skipped.
  • .scripts.test-e2e: If the package contains any E2E tests, this must run the necessary commands. See E2E tests for details.
  • .scripts.test-js: If the package contains any JavaScript tests, this must run the necessary commands. See JavaScript tests for details.
    • .scripts.skip-test-js: Run before .scripts.test-js in CI. If it exits with code 3, the test run will be skipped.
  • .scripts.test-php: If the package contains any PHPUnit tests, this must run the necessary commands. See PHP tests for details.
    • .scripts.skip-test-php: Run before .scripts.test-php in CI. If it exits with code 3, the test run will be skipped.
  • .extra.autorelease: Set truthy to enable automatic creation of a GitHub release for tagged versions. See Mirror repositories > Auto-release for details.
  • .extra.autotagger: Set truthy to enable automatic release-version tagging in the mirror repo. See Mirror repositories > Autotagger for details.
  • .extra.changelogger: Configuration object for Changelogger. See its documentation for details.
  • .extra.changelogger-default-type: Certain of our tools automatically create Changelogger change entries. This is the value to use for --type when doing so. Default type is changed.
  • .extra.dependencies.build: This optional array specifies the "slugs" of any within-monorepo build dependencies that can't otherwise be inferred. The "slug" consists of the two levels of directory under projects/, e.g. plugins/jetpack or packages/changelogger.
  • .extra.dependencies.test: This optional array specifies the "slugs" of any within-monorepo testing dependencies that can't otherwise be inferred. The "slug" consists of the two levels of directory under projects/, e.g. plugins/jetpack or packages/changelogger. See Testing for details.
  • .extra.dependencies.test-only: This optional array specifies the "slugs" of any within-monorepo dependencies that are only used for testing and/or static analysis and should be ignored otherwise when analyzing intra-monorepo dependencies.
  • .extra.dev-releases: Indicate that the plugin will have developer alpha releases. Instead of the mirror repositories showing "VER-alpha", they'll start at "VER-a.0" and you can use the -a flag to the release tooling to release "VER-a.1".
  • .extra.mirror-repo: This specifies the name of the GitHub mirror repo, i.e. the "Automattic/jetpack-something" in "https://github.com/Automattic/jetpack-something".
  • .extra.npmjs-autopublish: Set truthy to enable automatic publishing of tagged versions to npmjs.com. See Mirror repositories > Npmjs Auto-publisher for details.
  • .extra.release-branch-prefix: Our mirroring and release tooling considers any branch named like "prefix/branch-version" to be a release branch, and this specifies which prefix belongs to the project.
    • This may also be an array of multiple prefixes. In that case the first element in the array should be a prefix used only by this plugin, with any additional prefixes shared by multiple plugins coming after.
  • .extra.version-constants: When tools/project-version.sh is checking or updating versions, this specifies PHP constants to check or update. The value is an object matching constants to the file (relative to the package root) in which the constant is defined.
    • Note that constant definitions must be on a single line and use single quotes to be detected by the script. Like this:
      define( 'CONSTANT', 'version' );
    • Class constants may be specified by prefixing the constant name with "::", e.g. ::CONSTANT. In that case the definition must look like this:
      const CONSTANT = 'version';
  • .extra.wp-plugin-slug: This specifies the WordPress.org plugin slug, for use by scripts that deploy the plugin to WordPress.org.
    • .extra.beta-plugin-slug: This specifies the plugin slug for the Jetpack Beta Tester plugin, for cases where a plugin has not been published to WordPress.org but should still be offered by that plugin.
  • .extra.wp-svn-autopublish: Set truthy to enable automatic publishing of tagged versions to WordPress.org. See Mirror repositories > WordPress.org SVN Auto-publisher for details.

Our mirroring tooling also uses .gitattributes to specify built files to include in the mirror and unnecessary files to exclude.

Building

The Jetpack Monorepo includes GitHub actions to build all projects, and optionally to mirror them to mirror repos. The jetpack build command can be used to build locally.

A project must define .scripts.build-development and/or .scripts.build-production in composer.json to specify the commands needed to build. The build commands should assume that pnpm install and composer install have already been run, and must not run them again.

  • If you're building JavaScript bundles with Webpack and @automattic/jetpack-webpack-config (more information on setup in the README.md), note that your build-production command should set NODE_ENV=production and BABEL_ENV=production.
  • If you run into problems with Composer not recognizing the local git branch as being the right version, try setting COMPOSER_ROOT_VERSION=dev-trunk in the environment.
  • When building for the mirror repos, note that COMPOSER_MIRROR_PATH_REPOS=1 will be set in the environment and the list of repositories in composer.json may be altered. This is not normally done in development environments, even with jetpack build --production.
  • For a production build of a plugin, composer install is passed -o --no-dev --classmap-authoritative --prefer-dist. For development builds and for production builds of non-plugin projects, no options are passed. If you think you need something else, talk to us. Don't just have your build scripts run composer.

Testing

The Jetpack Monorepo includes GitHub actions to run a number of CI checks on all projects.

Tests for a project are only run for a PR if changes are made to the project or its dependencies. Dependencies may be specified as:

  • For Composer packages included in the monorepo, via .require and .require-dev in composer.json.
  • For JavaScript packages included in the monorepo, via .dependencies and .devDependencies in package.json.
  • For any other dependencies, via .extra.dependencies.test in composer.json.

The test environment will be set up with appropriate tools, including node, pnpm, php, phpdbg, and composer. Unless otherwise specified below, the versions of node and php will be those specified in .github/versions.sh. Other necessary tools may be pulled in via composer and pnpm.

All test commands must return a shell failure status when tests fail and a success status if tests pass or are skipped; usually your testing framework will already do this for you, but if you write custom shell scripts you'll need to make sure any failure is propagated.

If your project has multiple logical groups of tests, feel free to make use of GitHub Actions's grouping commands.

The following environment variables are available for all tests:

  • ARTIFACTS_DIR: If your tests generate any artifacts that might be useful for debugging, you may place them in the directory specified by this variable and they will be uploaded to GitHub after the test run. There's no need to be concerned about collisions with other projects' artifacts, a separate directory is used per project.
  • MONOREPO_BASE: Path to the monorepo. Useful if you're using things in tools/ from plugin tests.
  • WORDPRESS_DIR: Path to a copy of WordPress. Other than plugin tests, though, you should probably avoid using this.
  • WORDPRESS_DEVELOP_DIR: Path to a checkout of wordpress-develop. Other than plugin tests, though, you should probably avoid using this.
  • NODE_VERSION: The version of Node in use, as specified in .github/versions.sh.
  • PHP_VERSION: The version of PHP in use. Unless otherwise specified below, it will be the same as in .github/versions.sh.
  • TEST_SCRIPT: The test script being run.

Linting

We use eslint and phpcs to lint JavaScript and PHP code. Projects should comply with the coding standards enforced by these tools.

  • Projects may include .eslintrc.js to adjust eslint configuration as necessary, but try to keep to the spirit of it.

    Note we're using something of a hack to get eslint to read ignore rules from .gitignore and per-directory .eslintignore files. Any eslintrc that does root: true or an extends that extends from an eslintrc that includes the hack will have to do like

    const loadIgnorePatterns = require( 'jetpack-js-tools/load-eslint-ignore.js' );
    module.exports = {
    	// Whatever stuff, including `root: true` or `extends`.
    	ignorePatterns: loadIgnorePatterns( __dirname ),
    };
  • We're using a fork of phpcs and a custom filter that adds support for per-directory configuration (.phpcs.dir.xml) and use of .gitignore and .phpcsignore files. Again, try to keep to the spirit of things.

Static Analysis

We use Phan for PHP static analysis.1 Configuration for a project resides in the .phan/config.php within the project, which should generally build on top of the .phan/config.base.php from the monorepo root. A baseline file may also reside at .phan/baseline.php to allow for incremental fixing of errors.

Phan in the monorepo should be run locally via Jetpack's CLI tool as jetpack phan. Note that Phan soft-requires the PHP ast extension.

Installing the PHP ast extension on Linux

On most Linux distributions, you can install the PHP ast extension using your package manager:

For other Linux distributions, consult your package manager's documentation or consider compiling from source.

Mac users have reported having trouble installing the PHP ast extension. See the dropdown below for Mac-specific instructions.

Installing the PHP ast extension on Mac

This assumes you have PHP installed via Homebrew, e.g. you've done brew install php@8.2.

  1. First, check whether ast is already installed by running php --ri ast. If it prints something like this, you should already be good (unless you need a newer version; see Phan's README for version requirements):
    ast
    
    ast support => enabled
    extension version => 1.1.1
    AST version => Current version is 90. All versions (including experimental): {50, 60, 70, 80, 85, 90, 100}
    
  2. You may need to brew install pkg-config zlib to install some necessary dependencies.
  3. Update the list of available extensions: pecl channel-update pecl.php.net
  4. Build the extension: pecl install ast
    • If the build process fails due to mkdir errors with the pecl directory, you might try mkdir -p /opt/homebrew/lib/php/pecl and running the install again.
  5. You may also need to tell PHP where to find the newly-installed extension.
    1. Run pecl config-get ext_dir to find where pecl installs extensions.
    2. Run php -r 'echo ini_get( "extension_dir" ) . "\n";' to find where PHP currently expects extensions to live.
    3. If those are the same, great! If not, you have two options:
      • If PHP's current directory is empty, you could find your php.ini file (php --ini) and change extension_dir to pecl's location.
      • Or else, pecl probably added extension=ast.so to an ini file somewhere. You could change the ast.so value to be the full path inside pecl's directory.

Alternatives, if you can't install the ast extension, include running Phan with the --allow-polyfill-parser option (note this may cause false positives and cannot be used to update baseline files) or running Phan inside the Docker development environment.

PHP tests

If a project contains PHP tests (typically PHPUnit), it must define .scripts.test-php in composer.json to run the tests. The CI environment will run pnpm install and composer install beforehand, but if a build step is required before running tests the necessary commands for that should also be included in .scripts.test-php.

A MySQL database is available if needed; credentials may be found in ~/.my.cnf. Note that the host must be specified as 127.0.0.1, as when passed localhost PHP will try to connect via a Unix domain socket which is not available in the Actions environment.

Tests are run with a variety of supported PHP versions from 7.0 to 8.3. If you have tests that only need to be run once, run them when PHP_VERSION matches that in .github/versions.sh.

PHP tests for non-plugins

For all project types other than WordPress plugins, the necessary version of PHPUnit and/or any other tools should be pulled in via Composer.

We currently make use of the following packages in testing; it's encouraged to use these rather than introducing other tools that serve the same purpose.

  • yoast/phpunit-polyfills supplies polyfills for compatibility with PHPUnit 6.5 to 9.0, to support PHP 7.0 to 8.3.
    • Do not use Yoast\PHPUnitPolyfills\TestCases\TestCase or Yoast\PHPUnitPolyfills\TestCases\XTestCase. Just use the @before, @after, @beforeClass, and @afterClass annotations directly.
  • PHPUnit's built-in mocking is used for class mocks.
  • brain/monkey is used for mocking functions, and can also provide some functions for minimal WordPress compatibility.
  • automattic/wordbless is used to pull in WordPress for testing.
    • If using both Brain Monkey and WorDBless, note the following requirements:
      • You must require_once __DIR__ . '/../../vendor/antecedent/patchwork/Patchwork.php'; in bootstrap.php before WorDBless's setup, so Brain Monkey can mock WordPress functions.
      • Follow Brain Monkey's functions-setup.md instead of wordpress-setup.md; don't call Monkey\setUp() or try to use its WordPress-specific tools.

PHP tests for plugins

WordPress plugins may want to run within WordPress. All monorepo plugins are copied into place in a WordPress installation. Environment variable WORDPRESS_DIR points to this installation, and WORDPRESS_DEVELOP_DIR points a directory with WordPress's tests/phpunit/.

Tests will be run against the latest version of WordPress using the variety of supported PHP versions, and against the previous and trunk versions of WordPress using the PHP version in .github/versions.sh. The environment variable WP_BRANCH will be set to 'latest', 'previous', or 'trunk' accordingly. If you have tests that only need to be run once, run them when WP_BRANCH is 'latest'.

When implementing tests within a new plugin, you can follow the example set in the example bootstrap.php.

JavaScript tests

If a project contains JavaScript tests, it must define .scripts.test-js in composer.json to run the tests. The CI environment will run pnpm install beforehand, but if composer install or a build step is required before running tests the necessary commands for that should also be included in .scripts.test-js.

JavaScript tests should use jest, not mocha/chai/sinon. For React testing, use @testing-library/react rather than enzyme.

E2E tests

This is not implemented yet!

If a project contains end-to-end tests, it must define .scripts.test-e2e in composer.json to run the tests. If a build step is required before running tests, the necessary commands for that should also be included.

Code coverage

If a project contains PHP or JavaScript tests, it should also define .scripts.test-coverage in composer.json to run the tests in a mode that will generate code coverage output. The CI environment will run pnpm install and composer install beforehand, but if a build step is required before running tests the necessary commands for that should also be included in .scripts.test-coverage.

Output should be written to the path specified via the COVERAGE_DIR environment variable. Subdirectories of that path may be used as desired.

For PHP tests, you'll probably run PHPUnit as php -dpcov.directory=. "$(command -v phpunit)" --coverage-php "$COVERAGE_DIR/php.cov". If you have multiple runs (e.g. unit and integration), be sure to write the php.cov files to separate subdirectories of $COVERAGE_DIR.

For JS tests, you'll probably have a test script in package.json that runs jest with any needed options, and then a test-coverage script that does pnpm run test --coverage. If you have multiple runs (e.g. unit and integration), be sure each run writes to a different subdirectory of $COVERAGE_DIR.

There's no need to be concerned about collisions with other projects' coverage files, a separate directory is used per project. The coverage files are also automatically copied to ARTIFACTS_DIR.

Mirror repositories

Most projects in the monorepo should have a mirror repository holding a built version of the project, ready for deployment. Follow these steps to create the mirror repo and configure the monorepo tooling to push to it.

  1. Create the mirror repo on GitHub. It will most likely be named like "https://github.com/Automattic/jetpack-something".
    1. The repo's description should begin with [READ ONLY] and end with This repository is a mirror, for issue tracking and development head to: https://github.com/automattic/jetpack.
    2. The default branch should be trunk, matching the monorepo.
      • Note that you can't set the default branch until at least one branch is created in the repo.
    3. In the repo's settings, turn off wikis, issues, projects, and so on.
    4. Make sure that matticbot can push to the repo. You would do this here: https://github.com/Automattic/example-repository-name/settings/branches - creating a new branch protection rule where only Matticbot (and whoever needs access to push, for example Garage) can push to that repository.
    5. Make sure that Actions are enabled. The build process copies workflows from .github/files/mirror-.github into the mirror to do useful things like automatically close PRs with a reference back to the monorepo.
    6. Create any secrets needed (e.g. for Autotagger or Npmjs-Autopublisher). See PCYsg-xsv-p2#mirror-repo-secrets for details.
  2. For a PHP package (or a plugin listed in Packagist) you also need to go to packagist.org and create the package there. This requires pushing a first commit with a valid composer.json to the repository. That can be done by copying the new package's composer.json from the PR that introduced it.
    1. Be sure that automattic is added as a maintainer.
    2. If creating the package with your own account, make sure to link your Github account to Packagist so that you can sync the new package.
  3. If your project requires building, configure .scripts.build-production in your project's composer.json to run the necessary commands.
  4. If there are any files included in the monorepo that should not be included in the mirror, use .gitattributes to tag them with "production-exclude".
  5. If there are any built files in .gitignore that should be included in the mirror, use .gitattributes to tag them with "production-include".
  6. Set .extra.mirror-repo in your project's composer.json to the name of the repo.
    • When you push the PR making this change to composer.json, pay attention to the Build workflow. Download the "jetpack-build" artifact and make sure it contains your project, and that there are no extra or missing files.

Autotagger

If .extra.autotagger is set to a truthy value in the project's composer.json, a GitHub Action will be included in the mirror repo that will read the most recent version from the mirrored CHANGELOG.md in each push to trunk, and create the tag if that version has no prerelease or build suffix.

If .extra.autotagger is set to an object with a truthy value for major (i.e. if .extra.autotagger.major is truthy), the GitHub Action will additionally create or update a major-version tag as is common for GitHub Action repositories.

If .extra.autotagger is set to an object with falsey value for v (i.e. if .extra.autotagger.v is set and falsey), the tag will not be prefixed with "v".

Note that, for this to work, you'll need to create a secret API_TOKEN_GITHUB in the mirror repo. The value of the secret must be a GitHub access token. See PCYsg-xsv-p2#mirror-repo-secrets for details.

This is intended to work in combination with Changelogger: When any change files are present in the project, a -alpha version entry will be written to the changelog so the autotagging will not be triggered. To release a new plugin version, see: PCYsg-SU8-p2

Auto-release

If .extra.autorelease is set to a truthy value in the project's composer.json, a GitHub Action will be included in the mirror repo that will automatically create a GitHub release when a version tag is created. This works with Autotagger. Versions are recognized with and without a "v" prefix and with 2 to 4 components.

The body of the created release will be the entry from CHANGELOG.md for the tagged version. A zip file will be added to the release as an artifact. The zip file contains a single directory, which holds the output from git archive.

If .extra.autotagger is set to an object, the following are recognized:

  • .extra.autotagger.slug: Base name for the zip file, and the name of the base directory inside. If this is omitted, .extra.wp-plugin-slug will be used. If that is also not set, the portion of .name after the / will be used.
  • .extra.autotagger.titlefmt: Format for the release title. Must contain a single %s, which will be replaced with the version tagged. If omitted, the release title will simply be the version number.

Note the following will also be done by the build process:

  • An entry will be prepended to .gitattributes, setting export-ignore for /.git*. The file will be created if necessary. This prevents .github and other git files from being included in the zip.

Npmjs Auto-publisher

If .extra.npmjs-autopublish is set to a truthy value in the project's composer.json, a GitHub Action will be included in the mirror repo that will run npm publish when a version tag is created. This works with Autotagger. Versions must have a "v" prefix and have 3 components.

Note that, for this to work, you'll need to create a secret NPMJS_AUTOMATION_TOKEN in the mirror repo. The value of the secret must be an npmjs.com automation token for an account with the ability to publish the package. See PCYsg-xsv-p2#mirror-repo-secrets for details.

Note the following will also be done by the build process:

  • Entries will be prepended to .npmignore to ignore .github and composer.json during the NPM publish. This file will be created if not present.

Before you create the first release tag, you may want to check out the mirror and run npm publish --dry-run to ensure that only the files you want published will be published. If additional files need to be excluded, create an .npmignore.

WordPress.org SVN Auto-publisher

If .extra.wp-svn-autopublish is set to a truthy value in the project's composer.json, a GitHub Action will be included in the mirror repo that will automatically publish tags to WordPress.org's SVN when a version tag is created. This works with Autotagger. Versions are recognized with and without a "v" prefix, with 2 to 4 components, and with an optional prerelease suffix.

Note that, for this to work, you'll need to create secrets WPSVN_USERNAME and WPSVN_PASSWORD in the mirror repo. See PCYsg-xsv-p2#mirror-repo-secrets for details. Also note that .extra.wp-plugin-slug must be set in the project's composer.json or the action will fail.

The action will update the plugin's trunk to the tagged source and will create a tag in SVN for the tagged version. If the tagged version does not have a prerelease component, the "Stable tag" field in the tag's readme.txt will be updated too. The "Stable tag" in trunk will not be updated; this must be done manually.

Plugin release tooling

If you have set .extra.mirror-repo, .extra.release-branch-prefix, and .extra.wp-plugin-slug in your plugin's composer.json, we have tooling to make releasing to WordPress.org easier.

  • tools/create-release-branch.sh will help you create the correctly named release branch, and will automatically update version numbers and versions of monorepo packages for you. The GitHub Action will then mirror this branch to your plugin's mirror repo.
  • tools/deploy-to-svn.sh will prepare a temporary directory with the content of the mirror repo branch that is ready to be pushed to WordPress.org SVN.
  • tools/revert-release.sh will prepare a temporary directory that updates the "Stable version" tag in readme.txt to the previous version, in case an emergency rollback is required.

Jetpack Changelogger

The Jetpack Changelogger tool helps in managing a changelog for a project by having each PR drop a specially-formatted "change file" into a changelog directory, which the tool can then process for a release.

As implemented by the Jetpack Monorepo, any PR that touches the Jetpack plugin itself, or anything else in the /projects directory will need to add a specially-formatted file to the project's specified changelog directory; there is a command mentioned below that can help create the change file.

What does the change file look like? It’s a text file with a header-and-body format, like HTTP or email. A change file might look like this:

Significance: patch
Type: compat

Block Editor: update all blocks to be fully compatible with WordPress 5.7.

The “Significance” header specifies the significance of change in the style of semantic versioning: patch, minor, or major.

The “Type” header categorizes the change in the changelog. In Jetpack, for example, our changelog divides changes into “Major Enhancements”, “Enhancements”, “Improved compatibility”, and “Bugfixes”.

The body is separated from the headers by a blank line, and is the text that actually goes into the changelog. This should follow our recommendations for writing a good changelog entry. Feel free to (sparingly) use Markdown in the body text.

Using the Jetpack Changelogger

The changelogger tool can be used via Jetpack's CLI tool. You may use the following command to generate changelog entries for each project that needs one:

jetpack changelog add

Does it matter what the change file is named? Starting the file name with . should not be used. Also consider avoiding names that have extensions like .php or .js to avoid confusing other tools.

What if a change is so trivial that it doesn’t need a changelog entry? The change file is still required. If you specify the significance as “patch”, changelogger will allow the body section to be empty so as to not generate an entry in the changelog. In this case, use the “Comment” header instead, for example:

Significance: patch
Type: compat
Comment: Update composer.lock, no need for a changelog entry

Adding the first PR to a project after a release? If a PR is the first to Jetpack after a release, version numbers may need to be bumped. This also applies to the first semantic versioning “minor” or “major” change to any projects that use semantic versioning.

The “Linting / Changelogger validity” GitHub Actions check will help in making sure that all these version numbers are in sync with the version inferred from the changelog and change files. You can also check this locally with tools/changelogger-validate-all.sh.

Within a single project, changlogger’s version next command can tell you the next version, and the monorepo script tools/project-version.sh can be used to check and update the version numbers.

New Projects

To begin,

  • For Automatticians, drop us a line in #jetpack-crew to discuss your needs, just to be sure we don't have something already. For others, it would probably be best to open an issue to discuss it.
  • Use the jetpack generate command to create a skeleton project.
  • Create your project based on the skeleton and submit a PR as usual.

Once we're sure that the project will be created and what its name will be, someone (you or the Crew team) does the following:

  • Create a GitHub repo in the Automattic repo to be the mirror repo for this project. The new repo follows the mirror repo guidelines.

Creating a new Composer Package

In addition to the above, after creating the mirror repo,

  • Add a composer.json file to the repo, with some basic information about the package. This file is used by Packagist to generate the package page.
  • Create a new Packagist package on packagist.org under the Automattic org. Add automattic as a maintainer.

Creating a new plugin

In addition to the above, after creating the mirror repo,

  • Add a first version of a composer.json file to the mirror repo.
  • Add the plugin to Packagist, just like for Composer packages above, for folks who want to consume it through Composer.
  • Add an entry for the new plugin in the Beta server settings. Find extra details on this process in the Jetpack Beta Builder repository. More information: PCYsg-gDE-p2

Importing an existing repo

To move development of an existing (public) repo into the Jetpack monorepo, you might do something like this.

Preparation in the original repo:

  • Set up PHP_CodeSniffer with our ruleset and fix any lints identified.
  • Merge any PRs that are ready to merge.

In a checkout of the monorepo:

  • Use git remote add to add a new remote for the existing repo, e.g. git remote add existing-source-repo git@github.com:Automattic/existing-source-repo
  • git fetch existing-source-repo
  • Create a new (temporary) branch based on the existing source repo: git checkout -b existing-repo/prepare-source existing-source-repo/trunk
  • Move the files to where they should live in the monorepo, e.g. git mv -k * .* projects/plugins/new-plugin
    • You may need to do something like mkdir --parents ./projects/plugins/new-plugin for the move to work.
    • TODO: Consider whether git filter-repo might be better. See p9dueE-2on-p2#comment-5761
  • Commit git add --all && git commit -m "Prepare XXX for monorepo"
  • Create the branch for the actual import: git fetch origin && git checkout -b add/import-from-existing-repo origin/trunk
  • git merge --allow-unrelated-histories existing-repo/prepare-source. This will merge in the source plugin into the monorepo while maintaining all previous commits.
  • Create additional commits to clean up the new project: adjust tooling to use what the monorepo provides, remove unneeded tooling, set monorepo configuration in composer.json, etc.
  • Run linting and such. Commit anything necessary.
  • git push origin HEAD and create your PR. Add the "DO NOT MERGE" tag.
  • When it's time to merge the PR, go to the GitHub settings page and enable "Allow merge commits". Then go to the PR. There should be a caret dropdown next to "Squash and Merge" which you can use to select "Create a merge commit" instead.
  • Clean up:
    • Go back to the settings and turn "Allow merge commits" back off.
    • git branch -D existing-repo/prepare-source to delete the temporary branch.
    • If you want to move any open PRs from the old repo, check out the branches, git merge origin/trunk (and resolve any conflicts), push to origin, and recreate.
    • git remote remove existing-source-repo to remove the remote.
  • If you're going to reuse the old repo as the mirror, reconfigure it to match the mirror repo guidelines.

See p9dueE-2on-p2 for past uses of this process.

While a private repo could be imported similarly, you'd have a lot of auditing to do to make sure no old commit exposes any private information.

Footnotes

  1. In 2024 we evaluated Phan, Psalm, and PHPStan. Psalm was unable to produce a consistent baseline. PHPStan was confused about which constants were defined, and would have needed a bootstrapping file re-defining them all to work. Thus we settled on Phan. Details in pdWQjU-IH-p2.