We require a CLA from all contributors. See CONTRIBUTING.md for the details. The most important one is that signing is fully electronic and can be done in seconds. The list of GitHub users who have already signed the CLA is at CONTRIBUTORS.txt.
We use bors to make sure master is always green. Common commands are
bors r+
to merge a PR,bors r=username
to merge a PR on behalf of the user without r+ permissionsbors delegate+
to grant the author of PR r+ right for this PR.
Don't forget to say "Thank you!" when merging pull requests! :)
After pull request accepting you need:
- add the corresponding milestone for the pull request to make search "when this change was released" easier
- add project to inform QA that these changes should be tested. We don't usually add a project to pull request if it doesn't affect users
- mark the pull request by special labels (feature, fix, performance and internal) if you consider that the corresponding changes should be mentioned in changelog. If pull request doesn't have any of these labels it will be ignored while changelog generation See more about releases in the corresponding section.
- mark the pull request by to be documented label if the corresponding change should be mentioned in documentation or affects existing documentation
Each non-stalled pull-request should be assigned to a reviewer, who should make sure that PR moves forward. However, anybody with r+ can accept any PR, if they are confident that the PR is in a good state.
# Substitute 4.1 for latest version https://gradle.org/install/#install
# Repeat the command twice, as only the second iteration is idempotent =/
./gradlew wrapper --gradle-version 4.1 --distribution-type all
./gradlew wrapper --gradle-version 4.1 --distribution-type all
Note --distribution-type all
.
Each new major platform release brings not only a lot of new features but usually also a bunch of new incompatibilities (both source and binary ones). Since we want to support at least the latest stable major platform release and new EAPs, we have to compile the plugin with different platform versions to avoid any runtime errors.
To solve the problem with incompatibilities, we use conditional compilation based on Gradle source sets. See CONTRIBUTING.md for more details about concrete source structure.
The following explanation uses old
, current
and new
platform terms that mean:
old
- number of the oldest supported platform version that should be droppedcurrent
- number of the latest major stable platform releasenew
- number of new platform that should be supported
For example, at the moment of writing, 201
is old
, 202
is current
and 203
is new
.
See build_number_ranges for more info about platform versions.
Step by step instruction on how to support new platform version:
- Drop all code related to the
old
platform version, i.e. dropgradle-%old%.properties
and all%module_name%/src/%old%
directories in each module - Move code from
%module_name%/src/%current%
directories into common source set, i.e.%module_name%/src/%current%/main
into%module_name%/src/main
and%module_name%/src/%current%/test
into%module_name%/src/test
. Also, simplify the moved code if possible. - Add support for
new
platform, i.e. addgradle-%new%.properties
with all necessary properties and make it compile. It may be required to extract some code into platform-specific source sets to make plugin compile with each supported platform. See Tips and tricks section for the most common examples of how to do it - Fix tests
- Update log path in run configurations like
runIDEA
andrunCLion
. ForrunIDEA
configuration it requires:- drop
idea-%old%.log
item - add
idea-%new%.log
item with$PROJECT_DIR$/plugin/build/idea-sandbox-%new%/system/log/idea.log
path
- drop
- Update CI workflows to use new platform version instead of old one.
It usually requires just to update
platform-version
list in all workflows where we build the plugin. At the moment of writing they arecheck.yml
,rust-nightly.yml
andrust-release.yml
- Fix
BACKCOMPAT: %old%
comments
A non-exhaustive list of tips how you can adapt your code for several platforms:
- if you need to execute specific code for each platform in gradle build scripts (
build.gradle.kts
orsettings.gradle.kts
), just useplatformVersion
property andif
/when
conditions. Note, inbuild.gradle.kts
value of this property is already retrieved intoplatformVersion
variable - if you need to have different sources for each platform:
- check that you actually need to have specific code for each platform.
There is quite often a deprecated way to make necessary action.
If it's your case, don't forget to suppress the deprecation warning and add
// BACKCOMPAT: %current%
comment to mark this code and fix the deprecation in the future - extract different code into a function and place it into
compatUtils.kt
file in each platform-specific source set. It usually works well when you need to call specific public code to make the same things in each platform - if code that you need to call is not public (for example, it uses some protected methods of parent class), use the inheritance mechanism.
Extract
AwesomeClassBase
from yourAwesomeClass
, inheritAwesomeClass
fromAwesomeClassBase
, moveAwesomeClassBase
into platform specific source sets and move all platform specific code intoAwesomeClassBase
as protected methods. - sometimes, signatures of some methods can be specified while platform evolution.
For example,
protected abstract void foo(Bar<Baz> bar)
can be converted intoprotected abstract void foo(Bar<? extends Baz> bar)
and you have to override this method in your implementation. It introduces source incompatibility (although it doesn't break binary compatibility). The simplest way to make it compile for each platform is to introduce platform-specifictypealias
, i.e.typealias PlaformBar = Bar<Baz>
forcurrent
platform andtypealias PlaformBar = Bar<out Baz>
fornew
one and use it in signature of overridden method. Also, this approach works well when some class you depend on was moved into another package. - if creation of platform-specific source set is too heavy for your case, there is a way how you can choose specific code in runtime.
Just create the corresponding condition using
com.intellij.openapi.application.ApplicationInfo.getBuild
. Usually, it looks like
Of course, code should be compilable with all supported platforms to use this approach.val BUILD_%new% = BuildNumber.fromString("%new%")!! if (ApplicationInfo.getInstance().build < BUILD_%new%) { // code for current platform } else { // code for new platform }
- sometimes you want to disable some tests temporarily to find out why they don't work later.
The simplest way to do it is to mark them with
org.rust.IgnoreInPlatform
annotation. Under the hood, it uses the previous approach withApplicationInfo
.
- check that you actually need to have specific code for each platform.
There is quite often a deprecated way to make necessary action.
If it's your case, don't forget to suppress the deprecation warning and add
- if you need to register different items in
%xml_name%.xml
for each platform:- create
platform-%xml_name%.xml
in%module_name%/src/%current%/main/resources/META-INF
and%module_name%/src/%new%/main/resources/META-INF
- put platform specific definitions into these xml files
- include platform specific xml into
%module_name%/src/main/resources/META-INF/%xml_name%.xml
, i.e. add the following code
<idea-plugin xmlns:xi="http://www.w3.org/2001/XInclude"> <xi:include href="/META-INF/platform-%xml_name%.xml" xpointer="xpointer(/idea-plugin/*)"/> </idea-plugin>
- create
While supporting new IDE version we should check all UI components manually because we don't have UI tests yet. Especially it's important if it's major IDE update because major platform updates can bring a lot of changes.
- New rust project
- Import rust project
- Rust preferences (in Languages & Frameworks)
- Cargo toolbar
- Run configuration
- Debugger settings (in Build, Execution, Deployment > Debugger > Data Views > Rust, CLion only)
- Notifications (see
MissingToolchainNotificationProvider
) Run Cargo Command
actionImplement Members
refactoringIntroduce Variable
refactoringExtract Function
refactoringAuto Import
quick fix & optionsUnresolved Reference
inspection options
Nightly and beta are released automatically by GitHub workflows. Stable is generally released every two weeks.
One week before release we create release branch with release-%release_version%
name from the master
branch.
release_version
value is the same as the corresponding milestone version.
Release branches are used to build beta and stable plugin builds.
Most of release actions can be done automatically via GitHub workflow.
You can trigger them from GitHub UI.
Just open Action
tab, choose a necessary workflow and launch it via Run workflow
button.
Alternatively, there is scripts/release-actions.py
script to trigger events from your console.
Syntax: python release-actions.py release_command --token github_token
.
Also, you can provide IR_GITHUB_TOKEN
environment variable to provide github token.
It allows you to omit --token
option.
See instruction
how to create personal github token. Note, it should have repo
scope.
Note, we use pipenv to manage python version, dependencies and virtual environment. See instruction how to install it. To run any command, just execute the following:
cd scripts
pipenv install # to install dependencies
pipenv run python release-actions.py release_command --token github_token # to run script in virtual environment
Available commands:
release-branch
- creates new release branchrelease-%release_version%
frommaster
one where%release_version%
is the same aspatchVersion
property ingradle.properties
. After that it increasespatchVersion
by one, commits changes and pushes them to master. Note, the corresponding workflow is triggered on schedule to create release branch one week before release, so you don't usually need to trigger it manually.nightly-release
- builds the plugin frommaster
branch and publishes it intonightly
channel on Marketplace. Note, the corresponding workflow is triggered on schedule, so you don't usually need to trigger it manually.beta-release
- builds the plugin from release branch and publishes it intobeta
channel on Marketplace. Note, the corresponding workflow is triggered on schedule, so you don't usually need to trigger it manually.stable-release
- updates changelog link inplugin/src/main/resources/META-INF/plugin.xml
, commits changes, pushes intomaster
and cherry-picks the corresponding changes into release branch. After that, it builds the plugin from release branch and publishes it intostable
channel on Marketplace.
Note, each command may provide additional options. Add --help
option to get actual option list.
Release notes live in intellij-rust.github.io.
To write notes, run ./changelog.py
. It goes thorough all pull requests from the corresponding milestone and
creates a template with default info about merged PRs in _posts
.
The initial version of each post depends on special tags that PR can be labeled.
At this moment, changelog.py
supports feature
, performance
, fix
and internal
tags.
Note, PR can be marked with any subset of these tags.
Transform generated text to user-friendly one, add necessary links/gifs.
Don't forget to mention every contributor using by [@username]
syntax.