Ways of Working
Contents
Git branching model
We have adopted the following branching strategy - see Git Branching Model
Issues
Where possible, a pull request should correlate to a single GitHub issue. An issue should relate to a single functional or non-functional change - changes to alter/improve other pieces of functionality should be addressed in a separate issue in order to keep reviews atomic. The reasoning behind code changes should be documented in the GitHub issue. All resolved issues should be included in the next GitHub milestone, this enables releases to be linked to the included issues. If a code change requires users to make changes in order for them to adopt it, then the issue should be labelled 'migration-required', and a comment should be added similar to:
### Migration Steps
[Description of what needs to be done to adopt the code change with examples]
Workflow
- Assign yourself to the issue
- Create a new branch off develop using pattern
PAL-[issue number]-[issue-title]
- Commit your changes prefixing your commit title with
PAL-[issue-number] [commit title]
- Check your changes
- Create a pull request to merge your branch into develop
- The pull request will be reviewed and following any changes and approval your branch will be merged into develop
- Delete the branch
Pull Requests
Pull requests will undergo an in depth review by a Palisade committer to check the code changes are compliant with our coding style. This is a community so please be respectful of other members - offer encouragement, support and suggestions. As described in our git branching model - please raise pull requests to merge your changes into our develop branch. Please agree to the GCHQ OSS Contributor License Agreement before submitting a pull request.
Coding style
Please ensure your coding style is consistent with rest of the project and follow coding standards and best practices.
In particular please ensure you have adhered to the following:
- Strive for small, easy to read, and easy to understand classes and methods
- Separate out related classes into packages and avoid highly coupled classes and modules
- Checkstyle is run as part of 'mvn package' so you should ensure your code is compliant
- The project will not build if there are issues which cause these plugins to fail
- SonarQube and Hadolint are run by the CI/CD pipeline so you should ensure your code is compliant
- The pipeline will not complete successfully if there are issues which cause these checks to fail
- Classes and methods should comply with the single responsibility principal
- Avoid magic numbers and string literals
- Avoid coupling services, if necessary duplicate code so individual services can be updated without causing cascading issues
- Look after your streams - if you open one make sure you close it too
- This should apply to all volatile resource usage
- Don't swallow exceptions - ensure they are logged or rethrown
- Give credit for other peoples work
- Update the NOTICES file for changes to the dependencies
- Consider the scope of dependencies - restrict them when possible using the appropriate maven scope
- Don't expose private logic in classes through public methods
- Field access should be controlled via getters and setters
- Make use of the core Java API - don't reinvent the wheel
- Make use of generic typing
- Make use of appropriate object-oriented design patterns
- Use Loggers instead of System.out.print and Throwable.printStackTrace
- Ensure that the
toString()
,equals()
andhashCode()
methods are implemented where appropriate - Generated code should be tagged with a
@Generated
annotation such that it is excluded from coverage metrics
Javadoc
Ensure your java code has sufficient Javadoc explaining what the section of code does and the intended use of it. Javadoc should be used in addition to clean readable code, it is not an excuse to write lazy code. Method bodies should also be accompanied by inline comments where appropriate.
In particular, the following are required:
- public classes (not required for test classes unless an explanation of the testing is required)
- public methods (not required if the functionality is obvious from the method name e.g. getters and setters)
- public constants (not required if the constant is obvious from the name)
Any other access modifiers may have Javadoc or standard Java comments.
Tests
- All new code should be unit tested
- Where this is not possible the code should be invoked and the functionality should be tested in a component or contract test
- In a small number of cases this will not be possible - instead steps to verify the code should be thoroughly documented
- Tests should cover edge cases and exception cases as well as normal expected behavior
- Keep each test decoupled and don't rely on tests running in a given order - don't save state between tests
- For a given code change, aim to improve the tests' code coverage
- Unit test classes should test a single class and be named [testClass]Test
- Component test classes should test a single 'feature' and and be named [feature]Test
- Contract test classes should test the external communications made, whether network, file IO, or otherwise
- Tests should use JUnit 5.x and AssertJ
- Component tests will likely use a Spring test annotation such as
@WebMvcTest
- Contract tests will likely use a Testcontainer such as
KafkaTestcontainer
- Component tests will likely use a Spring test annotation such as
- Tests should be readable and self documenting where possible
- AssertJ's
.as(..)
should be added to all assertions to aid in documenting in this way
- AssertJ's
We suggest the following pattern:
@Test void testSomeThingDoesSomeBehaviour { // Given [Setup your test here] // When [Invoke the test method] // Then [assert the method did what was expected] }