/kafka-sync-proxy

Make http calls to Kafka that waits for responses.

Primary LanguageGoMIT LicenseMIT

Kafka Synchronous Proxy

Go Reference Docker

A web service to make traditional rest api calls which publishes a messsage into a topic & wait for the response in a response channel.

How does it work?

The following steps are followed

  • Create a random string to be passed as key (currently done using uuidv4)
  • Prepare a kafka message containing the key, payload as the value and headers as is.
  • Produce it to the requestTopic
  • Consume from the responseTopic until a message with the exact key is received
  • If nothing is received in REQ_TIMEOUT seconds then the circuit breaks and HTTP 408 Request Timeout is sent back. Note: Currently, REQ_TIMEOUT is set to 5s.
sequenceDiagram
    http-client->>+kafka-sync-proxy: POST /v1 
    note left of kafka-sync-proxy: requestTopic<br>responseTopic<br>payload<br>headers  
    note right of kafka-sync-proxy: requestTopic<br>key [coorelationId]<br>payload<br>headers  
    kafka-sync-proxy-->>kafka: produce

    kafka-->>microservice:consumes
    note right of microservice: business logic
    microservice-->>kafka: produces

    kafka-->>kafka-sync-proxy: consume 
    note right of kafka-sync-proxy: responseTopic<br>key [coorelationId]<br>response payload<br>response headers  
    note left of kafka-sync-proxy: application/json<br>{respConse, headers}  
    kafka-sync-proxy->>-http-client: HTTP 200 OK


    note left of kafka-sync-proxy: after REQ_TIMEOUT seconds have passed  
    kafka-sync-proxy->>http-client: HTTP 408 Request Timeout
Loading

Installation

Go

Install from pkg.go.dev

go install github.com/sarkarshuvojit/kafka-sync-proxy

Docker

Run docker image.

docker pull sarkarshuvojit/kafka-sync-proxy

Usage/Examples

Running

It can be run as a standalone as well as inside a docker container.

standalone

After installing from pkg.go.dev

$ kafka-sync-proxy

 ┌───────────────────────────────────────────────────┐
 │                   Fiber v2.43.0                   │
 │               http://127.0.0.1:8420               │
 │       (bound on host 0.0.0.0 and port 8420)       │
 │                                                   │
 │ Handlers ............. 1  Processes ........... 1 │
 │ Prefork ....... Disabled  PID ............ 536624 │
 └───────────────────────────────────────────────────┘

docker

docker run -p 8420:8420 -d sarkarshuvojit/kafka-sync-proxy

docker-compose

Use it with docker-compose

version: '2'
services:
  kafka-sync-proxy:
    container_name: kafka-sync
    image: sarkarshuvojit/kafka-sync-proxy
    ports:
      - 8420:8420
    networks:
      - common-network

networks:
  common-network:

Calling the API

There is only route hosted at /v1/ which only listens to POST

Let's say you have a microservice listening to com.org.events.testInitiated which processes the message and responds to the com.org.events.testCompleted you can possibly test it by the following curl request.

$ curl --location --request POST 'http://localhost:8420/v1/' \
--header 'Content-Type: application/json' \
--data-raw '{
    "requestTopic": "com.org.events.testInitiated",
    "responseTopic": "com.org.events.testCompleted",
    "payload": {
        "message": "testing the proxy",
        "nested": {
            "child": {
                "subChild": 42
            }
        }
    },
    "headers": {
        "customAuthHeader": "ggg"
    },
    "brokers": [
        "localhost:29092"
    ]
}'

Should respond with:

HTTP 200 OK

{
   "headers" : {
      "customAuthHeader" : "ggg",
      "newHeaderKey" : "header val"
   },
   "response" : {
      "message" : "testing the proxy",
      "nested" : {
         "child" : {
            "subChild" : 42
         }
      }
   }
}

Note: This system requires that the response sends the message with the same key that was sent.

License

MIT