September 24th, 2020

Optimizing bandwidth in the Azure Cosmos DB .NET SDK

Matias Quaranta
Software Engineer

When we create, update, and even delete an item in our Azure Cosmos containers, the actual body of the item is always included as part of the service response along with other useful information in the headers, like the request charge. This body content consumes bandwidth, and if we think about it, we are using almost the same bandwidth to send the operation than to receive its response. When the volume of operations it high, this accounts for quite a big chunk of data over the wire, which we might not be using at all. Wouldn’t it be great if there was a way to tell the service not to include it in the response?

 

What is included in a normal operation response?

Write operations include create, replace, upsert, and delete item operations. In most of them (except delete), you send a payload with the item content you want to write and when the operation is completed successfully on the Azure Cosmos backend, you receive a response that contains the original item content you sent, a status code indicating the success, and a set of known headers (request charge, session token, etc.). The information in the headers, which tends to be more important in these operations, often uses the smallest bandwidth.

 

Do we really need the response content?

Most of the time, we don’t really do much with the response content, because it’s what we just defined as input when we made the operation call or we plainly don’t need it (do we really need the content of the item we just deleted?). Let’s take a look at two code examples to make it more clear.

https://gist.github.com/ealsur/85c27d994602b95b497ef24ed8f8a41b

In this create operation, extract important information like the request units, ETag, or Session information from the headers, but the “Resource” property of the response really has the same information as the original “item” variable and we probably are not using it.

https://gist.github.com/ealsur/ee178f458380ad78a9c9fb50a1f1d2e8

In this replace operation, we receive the content of the replaced item again on the “Resource” property, that might be hardly used, but that’s consuming bandwidth nevertheless.

In applications where the item size is large or where the volume of operations is high, this accounts for quite a relevant chunk of network bandwidth, that we could be using to actually send more operations! Wouldn’t it be great if there was a way to tell the backend not to send the item body back?

 

Optimizing our bandwidth

The Azure Cosmos DB .NET V3 SDK has a feature that you can use, per operation, to avoid the item body in the response. This feature is part of the ItemRequestOptions and is called EnableContentResponseOnWrite.

When setting this property to false, the response will no longer include the content mapped to the Resource property and won’t consume the bandwidth for it.

Using our previous examples:

https://gist.github.com/ealsur/bd0262bf4805bd088f9f10031716a7d2

Now the write operation’s response, won’t include the item content, thus reducing the network bandwidth usage.

https://gist.github.com/ealsur/8566693b4c4dbfd3c011b1fc2af13539

The replace operation won’t include the unnecessary item content either.

 

Does this work when Bulk mode is enabled?

If you remember our previous posts about the Bulk mode in V3 .NET SDK, you know that Bulk is all about optimizing throughput and operational volume. In this case, reducing the network usage is a great step in that same direction (the faster we obtain those responses, the faster we can send new operations!). And the answer is, yes, EnableContentResponseOnWrite also when Bulk mode is enabled, so make sure you are taking advantage of it if you are not using those response contents!

 

Is there a global configuration?

Yes! If you want to use the setting across all the operations with the SDK, you can also define it at the client creation level on the CosmosClientOptions:

https://gist.github.com/ealsur/769d8b0d6594c3642d3db31fa6c325fe

Is this currently supported in other SDK languages?

While this post focuses on .NET, you can also see the feature implemented in other SDK languages.

In Java V4 SDK, the CosmosClientBuilder has contentResponseOnWriteEnabled, which you can set to false, to achieve the same behavior:

https://gist.github.com/ealsur/a5eb1ac1b63100e0d336f26290e6737f

Author

Matias Quaranta
Software Engineer

Software Engineer at Cosmos DB

6 comments

Discussion is closed. Login to edit/delete existing comments.

Newest
Newest
Popular
Oldest
  • Boris Tarasov

    Does this reduce the cost of an operation?

  • Keiji Kamebuchi

    Hi!

    I feel so cool about this option! Thank you for this article! And I have feedback about the EnableContentResponseOnWrite.

    In .NET SDK, I would like to set default behavior for content response on write. For example, I want to default behavior is false about EnableContentResponseOnWrite option, and specific operation require response. (Maybe support that, currently?)

    In Java SDK, I would like to set EnableContentResponseOnWrite option when specific operation behavior for content response on write.

    Thanks.

  • Rich Mercer

    And I imagine this would improve speed as well by decreasing the time taken to send the response without the body? My app doesn’t ever read the response body so can see this being a quick win to boost performance by omitting it on upserts.

    • Matias QuarantaMicrosoft employee Author

      That is correct. The client doesn’t spend time downloading the response’s body, so it would complete faster.

Feedback