Contributing to Model Context Protocol Java SDK
May 4, 2026 · View on GitHub
Thank you for your interest in contributing to the Model Context Protocol Java SDK! This document outlines how to contribute to this project.
Prerequisites
The following software is required to work on the codebase:
Java 17or aboveDockernpx
Getting Started
- 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 tests
Reporting Issues
Please 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.
Making Changes
- Create a new branch:
git checkout -b feature/your-feature-name
- Make your changes
- Validate your changes:
./mvnw clean test
Change Proposal Guidelines
Principles of MCP
- 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.
Submitting Changes
- 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.
Evolving wire-serialized records
Records in McpSchema are serialized directly to the MCP JSON wire format. The rules differ depending on whether the field you are adding (or maintaining) is optional — Java code may legitimately leave it null and the wire may omit it — or spec-required by MCP. Follow Case A for optional fields and Case B for spec-required fields.
Case A — Optional fields
- 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_ABSENT)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. (For records that also have spec-required fields, the@JsonCreatorbelongs on a separate staticfromJsonfactory — see Case B, Rule 2.) - Do not convert
nullto a default value in the canonical constructor. Null carries "absent" semantics and must be preserved through the serialization round-trip. (Spec-required fields are the exception — see Case B, Rule 1.) - 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.
Case B — Spec-required fields
When the MCP specification marks a field as required, callers must not be able to construct a structurally invalid record, but the wire parser must still tolerate peers that fail to send it. Follow these rules in addition to the relevant Case A rules (annotation, naming, append-only).
- Reject
nullin the compact constructor. UseAssert.notNullfor required objects orAssert.hasTextfor requiredStringidentifiers (name,uri,uriTemplate,version). This throwsIllegalArgumentExceptionat construction time instead of producing a record that fails later in serialization or protocol handling. Overrides Case A Rule 7 for this field. - Add a
@JsonCreatorstaticfromJsonfactory alongside the canonical constructor. When a required field is absent from the wire, substitute a documented safe default (""for strings,[]for collections,{}for maps,0/0.0for numerics,INFOforLoggingLevel, etc.) and log atWARNnaming the field and the value used. The SDK must not halt the conversation because of a missing field. Place@JsonCreatoron thisfromJsonfactory, never on the canonical constructor (Case A Rule 6 still applies to the canonical constructor itself).- Exception:
JSONRPCResponse.JSONRPCErrorfails fast on missingcode/messagebecause a malformed JSON-RPC error envelope is unrecoverable.
- Exception:
- Provide a required-first builder factory
builder(req1, req2, …)and remove the corresponding setters from theBuilder. A no-argbuilder()factory must not exist on a record that has required fields. If one already exists for source compatibility, mark it@Deprecated. - Add tests per required field:
- Constructing the record with
nullfor the field throwsIllegalArgumentException. - Deserializing JSON without the field succeeds and yields the documented default.
- Deserializing JSON with an extra unknown field still succeeds (Case A Rule 8 also applies).
- Constructing the record with
Example
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");
}
Code of Conduct
This project follows a Code of Conduct. Please review it in CODE_OF_CONDUCT.md.
Questions
If you have questions, please create a discussion in the repository.
License
By contributing, you agree that your contributions will be licensed under the MIT License.
Security
Please review our Security Policy for reporting security issues.