Skip to content

Latest commit

 

History

History
267 lines (222 loc) · 15.6 KB

MIGRATION.md

File metadata and controls

267 lines (222 loc) · 15.6 KB

Migrating to the cloudant-java-sdk library

This document is to assist in migrating from the java-cloudant (coordinates: com.cloudant:cloudant-client) to the newly supported cloudant-java-sdk (coordinates: com.ibm.cloud:cloudant).

Initializing the client connection

There are several ways to create a client connection in cloudant-java-sdk:

  1. Environment variables
  2. External configuration file
  3. Programmatically

See the README for code examples on using environment variables.

Other differences

  1. In cloudant-java-sdk all operations are performed from the scope of the client instance and not associated with any sub-scope like Database. There is no need to instantiate a Database object to interact with documents - the database name is included as part of document operations. For example, in the case of updating a document you would first call getDocument to fetch and then putDocument to update, there is no need to getDatabase.
  2. In cloudant-java-sdk a user-supplied POJO is not required to represent a document. The default document representation is the Document model class. It is still possible to use POJOs in preference instead of the Document model, see examples in the section POJO usage in the cloudant-java-sdk library.
  3. Sending and receiving byte responses is available for operations that accept user-defined documents or return user-defined documents, document projections or map/reduce data. See the Raw IO section of cloudant-java-sdk README or the Bypass the document model and use the asStream methods section for more details.
  4. There is no built-in pagination support for views. Examples coming soon.
  5. Replay interceptors are replaced by the automatic retries feature for failed requests.
  6. Error handling is not transferable from java-cloudant to cloudant-java-sdk. For more information go to the Error handling section in our API docs.
  7. Custom HTTP client configurations in java-cloudant can be set differently in cloudant-java-sdk. For more information go to the Configuring the HTTP client section in the IBM Cloud SDK Common README.

Troubleshooting

  1. Authentication errors occur during service instantiation. For example, the code Cloudant service = Cloudant.newInstance("EXAMPLE"); will fail with Authentication information was not properly configured. if required environment variables prefixed with EXAMPLE are not set.
  2. Server errors occur when running a request against the service. We suggest to check server errors with getServerInformation which is the new alternative of metaInformation().

POJO usage in the new library

Since the new library supports models, this guide will demonstrate three ways to migrate your POJOs from java-cloudant.

Let's start with a simple POJO:

public class Pojo {
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return name;
    }
}

The code block below uses the POJO in a simple update implementation - reading the document into an object, performing an update and then writing the updated document in the database.

//... set up the service client and the database
Pojo p = db.find(Pojo.class, "example_id");
System.out.println(p); // the value of the Pojo's toString method

p.setName("new_name");
System.out.println(p.getName()); // will be new_name

Response response = db.update(p); // the same object is used for the update, it will set the name parameter to new_name

1. Use the Document model class, storing user properties in the Map

//set up the service client
GetDocumentOptions documentOptions =
        new GetDocumentOptions.Builder()
                .db("example_db")
                .docId("example_id")
                .build();

Document doc = service.getDocument(documentOptions)
        .execute()
        .getResult();

System.out.println(doc); // will be a JSON

// Set the JSON properties to the Pojo
Pojo p = new Pojo();
p.setName((String) doc.getProperties().get("name"));
System.out.println(p); // will be a Pojo with the same name as the JSON

p.setName("new_name");
System.out.println(p.getName()); // new_name

doc.put("name", p.getName()); // add your modifications to the Document object

PutDocumentOptions putDocumentOptions =
        new PutDocumentOptions.Builder()
                .db("example_db")
                .docId("example_id")
                .document(doc)
                .build();

DocumentResult response =
        service.putDocument(putDocumentOptions)
                .execute()
                .getResult();

2. Convert the Document model into a POJO (and vice versa)

//set up the service client
GetDocumentOptions documentOptions =
        new GetDocumentOptions.Builder()
                .db("example_db")
                .docId("example_id")
                .build();

Document doc = service.getDocument(documentOptions)
        .execute()
        .getResult();

System.out.println(doc); // will be a JSON

// Serialize the JSON to Pojo
Pojo p = YourJsonSerializer.fromJson(doc.toString(), Pojo.class);
System.out.println(p); // the value of the Pojo's toString method

p.setName("new_name");
System.out.println(p.getName()); // will be new_name

// Deserialize the Pojo back to the Document model
doc.setProperties(YourJsonSerializer.fromJson(YourJsonSerializer.toJson(p), Map.class));

PutDocumentOptions putDocumentOptions =
                new PutDocumentOptions.Builder()
                        .db("example_db")
                        .docId("example_id")
                        .document(doc)
                        .build();

DocumentResult response =
        service.putDocument(putDocumentOptions)
                .execute()
                .getResult();

3. Bypass the Document model and use the AsStream methods

//set up the service client
GetDocumentOptions documentOptions =
        new GetDocumentOptions.Builder()
                .db("example_db")
                .docId("example_id")
                .build();

Pojo p = new Pojo()
try(InputStream is = service.getDocumentAsStream(documentOptions).execute().getResult()){
    p = YourSeriliazer.fromJson(is, Old.Pojo.class);
    System.out.println(p); // the value of the Pojo's toString method
} catch (RuntimeException re){
    // ...
}
p.setName("new_name");
System.out.println(p.getName()); // will be new_name

try (InputStream is = new ByteArrayInputStream(YourSeriliazer.toJson(p).getBytes())) {
    PutDocumentOptions putDocumentOptions =
            new PutDocumentOptions.Builder()
                    .db("example_db")
                    .docId("example_id")
                    .body(is)
                    .build();

    DocumentResult response =
            service.putDocument(putDocumentOptions)
                    .execute()
                    .getResult();
} catch (IOException e) {
    // ...
}

Request mapping

Here's a list of the top 5 most frequently used java-cloudant operations and the cloudant-java-sdk equivalent API operation documentation link:

java-cloudant method cloudant-java-sdk API method documentation link
db.find() getDocument, getDocumentAsStream
db.getViewRequestBuilder().newRequest().build() postView, postViewAsStream
db.query() postFind, postFindAsStream
db.contains() headDocument
db.update() putDocument

A table with the whole list of operations is provided at the end of this guide.

The cloudant-java-sdk library is generated from a more complete API spec and provides a significant number of operations that do not exist in java-cloudant. See the IBM Cloud API Documentation to review request parameter and body options, code examples, and additional details for every endpoint.

Known Issues

There's an outline of known issues in the cloudant-java-sdk repository.

Reference table

The table below contains a list of java-cloudant functions and the cloudant-java-sdk equivalent API operation documentation link. The cloudant-java-sdk operation documentation link will contain the new function in a code sample e.g. getServerInformation link will contain a code example with getServerInformation().

Note: There are many API operations included in the new cloudant-java-sdk that are not available in the java-cloudant library. The API documentation contains the full list of operations.

java-cloudant operation cloudant-java-sdk method reference
metaInformation() getServerInformation
getActiveTasks() getActiveTasks
getAllDbs() getAllDbs
getMembership() getMembershipInformation
replicator().remove() deleteReplicationDocument
replicator().find() getReplicationDocument
replication().trigger()/replicator().save() putReplicationDocument
schedulerDocs() getSchedulerDocs
schedulerDoc() getSchedulerDocument
schedulerJobs() getSchedulerJobs
uuids() getUUids
deleteDB() deleteDatabase
db.info() getDatabaseInformation
db.save()/db.post() postDocument
createDB()/database() with create=true/createPartitionedDb(dbName) putDatabase
db.getAllDocsRequestBuilder().build() postAllDocs, postAllDocsAsStream
db.bulk() postBulkDocs
db.changes() postChanges, postChangesAsStream
db.getDesignDocumentManager().remove() deleteDesignDocument
db.getDesignDocumentManager().get() getDesignDocument
db.getDesignDocumentManager().put() putDesignDocument
db.search() postSearch, postSearchAsStream
db.getViewRequestBuilder().newRequest().build() postView, postViewAsStream
db.getDesignDocumentManager().list() with a filter postDesignDocs
db.query() postFind, postFindAsStream
db.listIndexes() getIndexesInformation
db.createIndex() postIndex
db.deleteIndex() deleteIndex
db.remove() with an _id with _local path deleteLocalDocument
db.find() with _local path getLocalDocument
db.save() with _local path putLocalDocument
db.partitionInfo() getPartitionInformation
db.getAllDocsRequestBuilder().partition().build() postPartitionAllDocs, postPartitionAllDocsAsStream
db.search() postPartitionSearch, postPartitionSearchAsStream
db.getViewRequestBuilder().newRequest().partition().build() postPartitionView, postPartitionViewAsStream
db.query() using partitionKey method arg postPartitionFind, postPartitionFindAsStream
db.getPermissions() getSecurity
db.setPermissions(userNameorApikey, permissions) putSecurity
db.getShards() getShardsInformation
db.getShard() getDocumentShardsInfo
db.remove() deleteDocument
db.find() getDocument, getDocumentAsStream
db.contains() headDocument
db.update() putDocument
db.removeAttachment() deleteAttachment
db.getAttachment() getAttachment
db.saveAttachment() putAttachment
generateApiKey() postApiKeys
db.setPermissions() putCloudantSecurityConfiguration