Skip to content

Upgrade guide

Shubhang Balkundi edited this page Aug 11, 2021 · 5 revisions

Upgrade Guide for Ziggurat 4.0.0

Breaking changes

The stream handler function now accepts a map which contains both the message and the kafka metadata

{ :topic <string>   
  :partition <int>   
  :timesamp <long> }

The handler signature still remains the same as before (fn [m] ...) , but the argument m is a clojure map with 2 keys :metadata and :message. The :message is the byte value of the message that is present on kafka and the :metadata contains a clojure map with the kafka metadata.

{ :message <byte array> 
  :metadata { :topic "x-log" 
              :partition 1 
              :timestamp 1234567890}}

Upgrade Guide for Ziggurat 3.1.0

Configuration Changes

There were some breaking changes to kafka streams library being used by Ziggurat version 3.0. Ziggurat 3.0 has upgraded kafka streams to v2.1. This requires the user to follow certain steps while upgrading. These are explained below.

For upgrading Ziggurat to 3.1.0, per Apache Kafka Upgrade Guide and KIP-268, the user need to follow these steps

  • Add a new config upgrade-from for each of the top level config-map in :stream-router section of config.edn. This config can be added either in config.edn file in the project or as an environment variable, as explained below.
  • Do a rolling deploy of the application with the newly added configuration (above).
  • Remove the configs added above
  • Do a rolling deploy of the application again.

This can be understood with the help of an example. For the following stream-router configuration

:stream-router        {:topic-entity-1       {:application-id                 "application-1"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "first-topic"}}
                      {:topic-entity-2       {:application-id                 "application-2"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "second-topic"}}

For projects using Ziggurat version <= 2.7.2, ([tech.gojek/ziggurat "2.6.0"], for example), the new stream-router config should look like this

:stream-router        {:topic-entity-1       {:application-id                 "application-1"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "first-topic"
                                              :upgrade-from                   "0.11.0"}}
                      {:topic-entity-2       {:application-id                 "application-2"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "second-topic"
                                              :upgrade-from                   "0.11.0"}}

Or, if the user is not comfortable with modifying config.edn, equivalent environment variables can be added to the project environment.

ZIGGURAT_STREAM_ROUTER_TOPIC_ENTITY_1_UPGRADE_FROM=0.11.0
ZIGGURAT_STREAM_ROUTER_TOPIC_ENTITY_2_UPGRADE_FROM=0.11.0

For projects using Ziggurat version > 2.7.2, ([tech.gojek/ziggurat "2.12.1"], for example), the new stream-router config should look like

:stream-router        {:topic-entity-1       {:application-id                 "application-1"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "first-topic"
                                              :upgrade-from                   "1.1"}}
                      {:topic-entity-2       {:application-id                 "application-2"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "second-topic"
                                              :upgrade-from                   "1.1"}}

Or, an equivalent environment variable can be added for each of the entiteis in stream-router section. Like this

ZIGGURAT_STREAM_ROUTER_TOPIC_ENTITY_1_UPGRADE_FROM=1.1
ZIGGURAT_STREAM_ROUTER_TOPIC_ENTITY_2_UPGRADE_FROM=1.1

Using Middlewares

In the versions preceding 3.0, Ziggurat would only process messages which were serialized in protobuf format. This was a major limitation as users could not use any other formats like JSON or Avro.

In Ziggurat 3.0, the logic for deserialization has been extracted out as middlewares which can be used not only for deserializing a message in any given format, but can be plugged together to perform a set of tasks before a message is processed.

You can read more about them at Ziggurat Middlewares

As far as message processing is concerned, messages will be provided as byte arrays and the user has to explicitly use ziggurat.middleware.default/protobuf->hash to deserialize a message before processing it.

For example, in previous versions, for this stream-routes configuration,

:stream-router        {:topic-entity-1       {:application-id                 "application-1"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "first-topic"}}
                                              :channels {:geofence-channel {:worker-count [10 :int]
                                                         :retry            {:count [3 :int]}}}}
                      {:topic-entity-2       {:application-id                 "application-2"
                                              :bootstrap-servers              "localhost:9092"
                                              :origin-topic                   "second-topic"}}
                                                                                                                                                                             :enabled [true :bool]

a mapper function in a Ziggurat-based project would look like this

(init/main start-fn stop-fn {:topic-entity-1 {:handler-function  mapper-fn 
                                              :geofence-channel  channel-mapper-fn}
                             topic-entity-2  {:handler-function  mapper-fn-2}})

An upgrade to Ziggurat 3.0 would require the user to write a new mapper function which explicitly deserializes the message using the proto middleware (provided in Ziggurat be default) before passing it as an argument to the old mapper function.

(def middleware-based-mapper-fn
  (-> mapper-fn
      (ziggurat.middleware.default/protobuf->hash com.gojek.esb.booking.BookingLogMessage :topic-entity-1)))
      
(def middleware-based-mapper-fn-2
  (-> mapper-fn-2
      (ziggurat.middleware.default/protobuf->hash com.gojek.esb.booking.BookingLogMessage :topic-entity-2)))
      
(def middleware-based-channel-mapper-fn
  (-> channel-mapper-fn
      (ziggurat.middleware.default/protobuf->hash com.gojek.esb.booking.BookingLogMessage :topic-entity-1)))

(init/main start-fn stop-fn {:topic-entity-1 {:handler-function  middleware-based-mapper-fn 
                                              :geofence-channel  middleware-based-channel-mapper-fn}
                             topic-entity-2  {:handler-function  middleware-based-mapper-fn-2}})

A similar change will be required for all the handler-functions/channel-functions in stream-routes map which is passed to init/main.

Development is under way to provide a JSON middleware for deserializing JSON messages. It is expected to be available in 3.1.0-alpha.2.