This document describes the steps needed to update a service contained in this SDK project.
It is a good practice to keep the SDK code for each service updated so that it is in sync with the most recent production version of its API definition. So, when a service's API definition is changed, the SDK code for the service should be updated (re-generated) in each SDK project in which it exists. This could be a change such as (a) editorial changes made to various descriptions, (b) the addition of a new parameter to an existing operation, or (c) the addition of one or more new operations.
- If you are an IBMer, make sure that your Annual Open Source Training is current.
- Make sure that your internal github.ibm.com id is linked
to your external github.com id. The id linking step will also result in an invitation to join the
github.com/IBM
organization. Accept that invitation. - If you do not yet have "push" access to the SDK project, contact the project maintainer to request push access (you must be a member of the github.com/IBM org).
- Make sure that you have installed the tools required to build the project.
- To update a service, make sure the following additional tools are installed:
- Clone or fork the repo. If you have push access (see above), you can clone the repo directly (no fork).
Example:
git clone [email protected]:IBM/platform-services-java-sdk.git
- If you do not have push access, then you'll need to first create a fork and then clone your fork in your
local sandbox environment.
Example:
git clone [email protected]:my-git-id/platform-services-java-sdk.git
- Make sure that your local sandbox is in sync with the remote and then build/test the project. If you're
using a fork, you'd need to first make sure that your fork is in sync with the primary repo.
Example:
cd <project-root>
git checkout main
git pull
mvn clean install -DskipITs # builds/tests the project but skips integration tests
- Make sure that the integration tests and working examples run clean for your service. See Running Integration Tests/Examples for details.
Before proceeding to make any changes, make sure the above steps complete cleanly. This is your "baseline".
Prior to re-generating the SDK code for your service, be sure to validate the updated version of the API definition
using the IBM OpenAPI Validator.
Example:
lint-openapi example-service.yaml
This command will display a list of errors and warnings found in the API definition as well as a summary at the end. It's not required that you fix all errors and warnings before trying to use the SDK generator, but this step should identify any critical errors that will need to be fixed prior to the generation step.
Video: Getting Started With The OpenAPI Validator
After validating the API definition, you're ready to generate new SDK code for your service. However, before you do that, you should probably create a new feature branch in which to deliver your updates:
cd <project-root>
git checkout -b update-example-service
Next, run the IBM OpenAPI SDK Generator to process your API definition and generate new service and unit test code for the service:
cd <project-root>
openapi-sdkgen.sh generate -g ibm-java -i example-service.json -o ./modules
#
# Note the "-o ./modules" option above. You must specify "./modules" as the
# output directory so that the Java source files are written to the proper locations.
#
The generated service and unit test code is written to the service's module directory within the SDK project
(e.g. ./modules/example-service
).
Video: Getting Started With The SDK Generator
Next, it is recommended that you inspect the differences between the previous and new generated code to
get an overall view of the changes caused by the re-generation step. The changes that you see in the
generated SDK code should align with the API definition changes that have occurred since you last
generated the SDK code.
Example:
git diff # alternative: use the "source control" view within vscode
When the Java SDK code for a service is re-generated, it is not uncommon for some generated classes (especially model classes) to see changes only in their copyright statements, with no other changes to their generated source.
These types of changes are insignificant, and they can (and to some degree should), be reverted. Reverting these changes will help to avoid cluttering your PRs and git change history, and in some respects the copyright year should not be changed if no other changes are being made to the file. Unfortunately, it's difficult for the generator to operate this way so it simply re-generates all classes associated with a service and if this re-generation is occurring in a different year than the previous generation, then you'll see the change to the copyright statement.
To automatically revert these changes, just run the fix-copyrights.sh
script
in the project root directory:
cd <project-root>
scripts/fix-copyrights.sh
The script will examine each file with uncommitted changes, and for those files with changes
only in their copyright statement, a git checkout
is performed to
discard the changes and revert the files to their pervious version.
Next, run the unit tests. You can run the unit tests for all the services like this:
cd <project-root>
mvn clean install -DskipITs
or you can run the unit tests for your particular service like this:
cd <project-root>/modules/example-service
mvn clean install -DskipITs
The unit tests should run clean. If not, then any test failures should be diagnosed and resolved before proceeding.
After ensuring that your service's unit tests run clean, the next step would be to modify your service's integration tests and working examples code to reflect the updated version of your API definition. See Updating Integration Tests/Examples for more information on this topic.
Even if no changes are needed (perhaps only very minor updates were made to the generated SDK code), at a mininum you should make sure that the integration tests and examples run clean after you re-generate the service and unit test code.
For instructions on running the integration tests and examples code, see Running Integration Tests/Examples.
After completing the previous steps to update the service, unit test, integration test, and working examples
code, commit your changes.
Example:
cd <project-root>
git add .
git commit -s -m "feat(Example Service): re-gen service after recent API changes"
git push
Note: be sure to sign off on your commits (git commit -s
option) as that is a required PR check within the
github.com/IBM org.
Finally, open a pull request (PR) and tag the project maintainer for approval.
To run the integration tests and working examples for a particular service, follow these steps. We'll use the mythical "Example Service" within the examples below, but you can make the necessary adjustments for your own service.
- Make sure you have the required .env file in your project root directory. Each service's integration
test and working examples code assumes that external configuration properties (service URL, IAM ApiKey, etc.)
are stored in a .env file located in the project's root directory. The name of the file can be found in
the integration test and examples code.
Example:
/**
* This method provides our config filename to the base class.
*/
@Override
public String getConfigFilename() {
return "../../example_service_v1.env";
}
The precise set of configuration properties required by each service will vary somewhat among the services, but there are a minimal set of properties that are commonly required by every service. The integration tests and examples code for certain services might require additional service-specific configuration properties as well. Typically these are documented in the working examples code.
- Make sure that you have built/unit-tested the project successfully before trying to run the integration tests and/or examples:
cd <project-root>
mvn clean install -DskipITs
- To run the integration tests for a service, follow these steps:
# cd to your service's module directory
cd <project-root>/modules/example-service
# Run unit and integration tests:
mvn verify
You should see 100% clean test results from the mvn
command above, with no tests being skipped.
- To run the examples code for a service, follow these steps:
# The "examples" module contains the examples classes for all the services:
cd <project-root>/modules/examples
# Make sure the module is built/unit tested:
mvn clean install
# Locate the examples class for your service (e.g. com.ibm.cloud.mysdk.myservice.v1.MyServiceExamples)
# hint: run "find src/main/java -name \*Examples.java"
# Run the "main" method in your service's examples class:
mvn exec:java -Dexec.mainClass="com.ibm.cloud.mysdk.myservice.v1.MyServiceExamples"
You should see 100% clean results from the mvn
command above, with no tests being skipped.
Certain types of API changes will require that the integration tests and examples code are also updated along with the re-generated SDK service and unit test code. For example, perhaps a new operation was introduced or a new parameter was added to an existing operation and you'd like to incorporate it in the integration tests and examples.
Keep in mind that the integration tests are used to verify that the generated SDK code interacts correctly with the service implementation, so any non-trivial changes made to the API definition (and hence the generated service code) should probably result in updates to the integration tests. At a minimum, the integration tests for a service should include a testcase for EACH operation.
While modifying the integration tests, also consider if you should make any changes to the service's working examples code. We want the working examples to provide a good example for users to follow when writing their own application code which uses your service, so consider whether or not the examples code should be updated to reflect the changes made to the API.
The integration tests and examples code for each service were initially generated by the SDK generator, then (most likely) manual changes were made so that the tests and examples run cleanly using realistic values for various parameters and properties. The amount of manual changes required will vary from one service to the next, but usually depends on the degree to which your API definition:
- Includes good, realistic example values for operation parameters, request bodies, and responses.
- Includes links that capture any inter-operation dependencies (e.g. the
create_cloud
operation'sid
response property's value should be used as theget_cloud
operation'scloud_id
path parameter).
Regardless, it is likely that the integration tests and examples code have some manual changes which will need to be retained as you apply updates to them to reflect the current changes being made to the API.
Therefore, it is not recommended that you simply re-generate the integration tests and examples code
such that the existing files are overwritten. Instead, we recommend that you generate new integration tests
and examples off to the side, then manually copy fragments from the newly-generated files to the existing
files located in the SDK project.
Example:
openapi-sdkgen.sh generate -g ibm-java -i example-service.json --genITs --genExamples -o /tmp/code/modules
The newly-generated integration tests and examples would be found in /tmp/code/modules/example-service
and
/tmp/code/modules/examples
. You could then copy fragments from there as needed to modify the corresponding files
in the SDK project. This is not ideal, but you can minimize the amount of manual changes by improving your API definition
as mentioned above.