Skip to content

Latest commit

 

History

History
304 lines (257 loc) · 14.2 KB

update_service.md

File metadata and controls

304 lines (257 loc) · 14.2 KB

How to update a service

This document describes the steps needed to update a service contained in this SDK project.

Table of Contents

Overview

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.

Prerequisites

  1. If you are an IBMer, make sure that your Annual Open Source Training is current.
  2. 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.
  3. 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).
  4. Make sure that you have installed the tools required to build the project.
  5. To update a service, make sure the following additional tools are installed:

Initial project setup

  1. 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
  1. 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
  1. 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
  1. 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".

Steps to update a service

1. Validate the API definition

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

2. Create feature branch

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

3. Re-generate the SDK code

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

4. Inspect new generated SDK code

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

Revert insignificant changes

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.

5. Run unit tests

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.

6. Modify integration tests and examples

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.

7. Open PR with your changes

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.

Appendix

Running Integration Tests/Examples

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.

  1. 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.

  1. 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
  1. 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.

  1. 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.

Updating Integration Tests/Examples

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:

  1. Includes good, realistic example values for operation parameters, request bodies, and responses.
  2. Includes links that capture any inter-operation dependencies (e.g. the create_cloud operation's id response property's value should be used as the get_cloud operation's cloud_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.

References