Thank you for your interest in contributing to the Model Context Protocol Java SDK! This document outlines how to contribute to this project.
The following software is required to work on the codebase:
Java 17or aboveDockernpx
- Fork the repository
- Clone your fork:
git clone https://github.com/YOUR-USERNAME/java-sdk.git
cd java-sdk- Build from source:
./mvnw clean install -DskipTests # skip the tests
./mvnw test # run testsPlease create an issue in the repository if you discover a bug or would like to propose an enhancement. Bug reports should have a reproducer in the form of a code sample or a repository attached that the maintainers or contributors can work with to address the problem.
- Create a new branch:
git checkout -b feature/your-feature-name- Make your changes
- Validate your changes:
./mvnw clean test- Simple + Minimal: It is much easier to add things to the codebase than it is to remove them. To maintain simplicity, we keep a high bar for adding new concepts and primitives as each addition requires maintenance and compatibility consideration.
- Concrete: Code changes need to be based on specific usage and implementation challenges and not on speculative ideas. Most importantly, the SDK is meant to implement the MCP specification.
- For non-trivial changes, please clarify with the maintainers in an issue whether you can contribute the change and the desired scope of the change.
- For trivial changes (for example a couple of lines or documentation changes) there is no need to open an issue first.
- Push your changes to your fork.
- Submit a pull request to the main repository.
- Follow the pull request template.
- Wait for review.
- For any follow-up work, please add new commits instead of force-pushing. This will allow the reviewer to focus on incremental changes instead of having to restart the review process.
Records in McpSchema are serialized directly to the MCP JSON wire format. Follow these rules whenever you add a field to an existing record to keep the protocol forward- and backward-compatible.
- Add new components only at the end of the record's component list. Never reorder or rename existing components.
- Annotate every component with
@JsonProperty("fieldName")even when the Java name already matches. This survives local renames via refactoring tools. - Use boxed types (
Boolean,Integer,Long,Double) so the field can be absent on the wire without a special sentinel. - Default to
null, not an empty collection or neutral value, so the@JsonInclude(NON_NULL)rule omits the field for clients that don't know about it yet. - Keep existing constructors as source-compatible overloads that delegate to the new canonical constructor and pass
nullfor the new component. Do not remove them in the same release that adds the field. - Do not put
@JsonCreatoron the canonical constructor unless strictly necessary. Jackson auto-detects record canonical constructors; adding@JsonCreatorpins deserialization to that exact parameter order forever. - Do not convert
nullto a default value in the canonical constructor. Null carries "absent" semantics and must be preserved through the serialization round-trip. - Add three tests per new field (put them in the relevant test class in
mcp-test):- Deserialize JSON without the field → succeeds, field is
null. - Serialize an instance with the field unset (
null) → the key is absent from output. - Deserialize JSON with an extra unknown field → succeeds.
- Deserialize JSON without the field → succeeds, field is
- An inner
Buildersubclass can be used. This improves the developer experience since frequently not all fields are required.
Suppose ToolAnnotations gains an optional audience field:
// Before
@JsonInclude(JsonInclude.Include.NON_NULL)
@JsonIgnoreProperties(ignoreUnknown = true)
public record ToolAnnotations(
@JsonProperty("title") String title,
@JsonProperty("readOnlyHint") Boolean readOnlyHint,
@JsonProperty("destructiveHint") Boolean destructiveHint,
@JsonProperty("idempotentHint") Boolean idempotentHint,
@JsonProperty("openWorldHint") Boolean openWorldHint) { ... }
// After — new component appended at the end
@JsonInclude(JsonInclude.Include.NON_NULL)
@JsonIgnoreProperties(ignoreUnknown = true)
public record ToolAnnotations(
@JsonProperty("title") String title,
@JsonProperty("readOnlyHint") Boolean readOnlyHint,
@JsonProperty("destructiveHint") Boolean destructiveHint,
@JsonProperty("idempotentHint") Boolean idempotentHint,
@JsonProperty("openWorldHint") Boolean openWorldHint,
@JsonProperty("audience") List<String> audience) { // new — added at end
// Keep the old constructor so existing callers still compile
public ToolAnnotations(String title, Boolean readOnlyHint,
Boolean destructiveHint, Boolean idempotentHint, Boolean openWorldHint) {
this(title, readOnlyHint, destructiveHint, idempotentHint, openWorldHint, null);
}
}Tests to add:
@Test
void toolAnnotationsDeserializesWithoutAudience() throws IOException {
ToolAnnotations a = mapper.readValue("""
{"title":"My tool","readOnlyHint":true}""", ToolAnnotations.class);
assertThat(a.audience()).isNull();
}
@Test
void toolAnnotationsOmitsNullAudience() throws IOException {
String json = mapper.writeValueAsString(new ToolAnnotations("t", null, null, null, null));
assertThat(json).doesNotContain("audience");
}
@Test
void toolAnnotationsToleratesUnknownFields() throws IOException {
ToolAnnotations a = mapper.readValue("""
{"title":"t","futureField":42}""", ToolAnnotations.class);
assertThat(a.title()).isEqualTo("t");
}This project follows a Code of Conduct. Please review it in CODE_OF_CONDUCT.md.
If you have questions, please create a discussion in the repository.
By contributing, you agree that your contributions will be licensed under the MIT License.
Please review our Security Policy for reporting security issues.