When contributing to this repository, please first discuss the change you wish to make via GitHub issues, Discord, or any other method with the owners of this repository before making a change.
These are the most important things to know before contributing (also explained in more detail later in this document):
- Commit messages must adhere to Conventional Commits.
- Branch names must be formatted correctly. The format is
{category}/GH-{issue number}/{lowercase-description}
. Category must match a category used in our Commitlint config. - We use Checkstyle in our build workflow to validate coding style. It is recommended to import the config/checkstyle/checkstyle.xml or config/intellij-code-style.xml file into your IDE, so that formatting rules are respected.
- Branches are kept up to date by rebasing, not by merging.
- For non-technical changes, adding a changelog entry is required.
- Keep your pull request (PR) as small as possible, this makes reviewing easier.
- Commits serve a clear purpose and have a fitting commit message.
- Branches are kept up to date by rebasing (updating a branch by merging makes for a confusing Git history).
- PRs are merged by merging the commits on top of the target branch (which is
develop
). - Remember to add your changes in
CHANGELOG.md
. If your changes are merely technical, it's not necessary to update the changelog as it's not relevant for users.
Commit messages must adhere to Conventional Commits. We use Commitlint to validate commit messages.
We use the conventional configuration for Commitlint.
It is recommended to install the Conventional Commit plugin to make it easier to write commit messages.
Because we use merge commits when merging a PR, branch names will be part of the history of the repository. That is why branch names must follow a certain standard.
The format is {category}/GH-{issue number}/{lowercase-description}
and a branch name can be maximum 50 characters of
length.
Category must match a category used in our Commitlint config.
Valid examples are:
fix/GH-123/add-branch-linting
docs/GH-123/cleanup
If you want to contribute to the translations of this project, please use Crowdin.
This project adheres to Semantic Versioning.
The changelog is kept in CHANGELOG.md
.
Keeping a readable, relevant and user-friendly changelog is essential for our end users to stay up to date with the project.
Please refrain from using technical terminology or adding entries for technical changes that are (generally) not relevant to the end-user.
The format is based on Keep a Changelog.
This project uses Gitflow.
Documentation must be kept up to date when adding or changing functionality.
Javadoc is available after every release on https://refinedmods.com/javadoc/refinedstorage2/.
Public APIs must be annotated with an @API
annotation
from API Guardian.
We use Checkstyle in our build workflow to validate coding style.
It is recommended to import the config/checkstyle/checkstyle.xml or config/intellij-code-style.xml file into your IDE, so that formatting rules are respected.
Moreover, the CheckStyle-IDEA plugin can be used to check if there are no style violations.
The ADRs of this project can be found under doc/architecture/decision.
When adding functionality or fixing a bug, it is important to add tests. Tests are important, if not more important, than the implementation code.
That means that they need to be first class citizens in the codebase, and must be readable at all times.
They ensure that there are no regressions, act as general documentation for the codebase, and ensure that the project can evolve over time.
To avoid brittle tests, tests need to validate behavior. A test cannot rely on the internal code structure, so most use of mocking should be avoided.
Tests in the API modules are regular unit tests. Don't see a "unit" here as a code unit, but as a unit of behavior.
Don't isolate code by using mocking, even in a unit test.
These tests don't rely on, nor know about, Minecraft.
Additionally, tests in the refinedstorage-network
module use the refinedstorage-network-test
JUnit plugin to
easily set up networks for testing.
To test the entire chain from Minecraft to the API modules, we use integration tests. These tests are located in the
test source set of the refinedstorage-platform-neoforge
module.
We write these integration tests as Minecraft gametests.
Our SonarQube quality gate requires a minimum test coverage percentage of 80%. This an aggregated percentage over all the API modules, with an exclusion for the platform modules.
The
refinedstorage-platform-*
modules are excluded because they contain a lot of Minecraft-specific code and are harder to test.
We also use Pitest mutation testing.
For the API modules, our build workflow requires a minimum test coverage percentage of 80% and a minimum mutation coverage percentage of 90%.
The release process is automated and follows Gitflow.
Before running the "Draft release" workflow to start the release process make sure CHANGELOG.md
contains all the
unreleased changes.
To determine the version number to be released, the workflow will ask you which release type this is (major, minor,
patch).
The latest version from CHANGELOG.md
will be used as a base, and that will be incremented
depending on the release type.
CHANGELOG.md
will be updated by this workflow, you can review this in the resulting release PR.
If you merge the release PR, the "Publish release" workflow will automatically publish the release. An additional PR
will be created to merge the changes in CHANGELOG.md
back into develop
.
The hotfix process is semi-automated and follows Gitflow:
- Create a hotfix branch off
main
. - Commit your changes on this branch.
- Update
CHANGELOG.md
(with version number and release date) manually on this branch. - Push the branch and create a PR for it, merging into
main
.
The "Publish release" workflow will take care of the rest.
We have a few GitHub workflows:
- Build (PRs, pushes to
develop
andmain
) - Draft release (manual trigger)
- Publish release (merging a PR to
main
) - Validate changelog (PRs)
- To validate if
CHANGELOG.md
is valid and updated. - Not every pull request needs a changelog change, so the
skip-changelog
label can be added to the pull request to ignore this.
- To validate if
- Validate commit messages (PRs)
- Validates whether the commits on a pull request respect Conventional Commits.
- We use the conventional configuration.
- Validate branch names (PRs)
- Issue for unsupported version (issues)
- Posts a message on a GitHub issue if the issue is about an unsupported version.
- Lock resolved issues and PRs (every night)
The build workflow triggers when a pull request is updated or when a commit is pushed to develop
or main
.
The build workflow takes care of the following:
-
Running a Gradle build, running our tests in the process and generating an aggregated code coverage report for the API modules.
-
Running Minecraft gametests.
-
Analyzing the code on SonarQube.
Because of limitations with SonarQube, pull requests originating from a fork aren't analyzed on SonarQube.
-
Code style validation with Checkstyle.
-
Mutation and line coverage test with Pitest.
-
Uploading the artifacts on the action.
The draft release workflow is a manual workflow which will create a release branch from develop
.
To determine the version number to be released, it will extract the latest version number from CHANGELOG.md
and
increment it depending on the release type selected.
This workflow takes care of the following:
- Creating the release branch.
- Updating the changelog on this release branch.
- Creating a pull request merging the release branch into
main
.
The "publish release" workflow is triggered when a release or hotfix PR is merged to main
. Usually, this will be the
PR created earlier in the "Draft release" workflow.
The workflow takes care of the following:
- Extracting the version number from the release or hotfix branch name that is merged in the PR.
- Extracting the changelog entry for this version number.
- Running a build.
- Publishing on GitHub packages and CreeperHost Maven.
- Publishing Javadoc on GitHub pages.
- Deploying on GitHub releases.
- Announcing the release on Discord and Twitter.
- Creating a PR that merges
main
back intodevelop
to get the changes toCHANGELOG.md
andbuild.gradle
intodevelop
from the draft release workflow.
Refined Storage is split up into various modules.
Most modules aren't dependent on Minecraft or a mod loader. Only modules that start
with refinedstorage-platform-*
have dependencies on Minecraft.
Name | Use in addons | Description |
---|---|---|
refinedstorage-core-api |
✔️ | Contains some utility classes and enums. |
refinedstorage-grid-api |
✔️ | Contains Grid related functionality. |
refinedstorage-network-api |
✔️ | Contains storage network related functionality. |
refinedstorage-network |
❌ | Contains implementations of refinedstorage-network-api . |
refinedstorage-network-test |
✔️ | JUnit extension which helps with setting up a network and a network node for testing. |
refinedstorage-query-parser |
✔️ | A query parser, contains a lexer and parser. Only used for Grid query parsing. |
refinedstorage-resource-api |
✔️ | Contains API for handling resources. |
refinedstorage-storage-api |
✔️ | Contains storage related functionality. |
refinedstorage-platform-api |
✔️ | Implements the various Refined Storage API modules for use in Minecraft. |
refinedstorage-platform-fabric |
❌ | The platform module for Fabric. This module contains Fabric specific code. |
refinedstorage-platform-forge |
❌ | The platform module for Forge. This module contains Forge specific code and the integration tests. |
refinedstorage-platform-common |
❌ | Common mod code. Most gameplay code is in here. |