Skip to content

Commit

Permalink
[strimzi#434] feat: added create topic endpoint
Browse files Browse the repository at this point in the history
Signed-off-by: ilkerkocatepe <[email protected]>
  • Loading branch information
ilkerkocatepe committed Sep 7, 2024
1 parent bc04d53 commit b97247a
Show file tree
Hide file tree
Showing 6 changed files with 136 additions and 0 deletions.
25 changes: 25 additions & 0 deletions src/main/java/io/strimzi/kafka/bridge/KafkaBridgeAdmin.java
Original file line number Diff line number Diff line change
Expand Up @@ -9,13 +9,15 @@
import org.apache.kafka.clients.admin.AdminClient;
import org.apache.kafka.clients.admin.Config;
import org.apache.kafka.clients.admin.ListOffsetsResult;
import org.apache.kafka.clients.admin.NewTopic;
import org.apache.kafka.clients.admin.OffsetSpec;
import org.apache.kafka.clients.admin.TopicDescription;
import org.apache.kafka.common.TopicPartition;
import org.apache.kafka.common.config.ConfigResource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Properties;
Expand Down Expand Up @@ -84,6 +86,29 @@ public CompletionStage<Set<String>> listTopics() {
return promise;
}

/**
* Creates a topic with given name
*
* @param topicName topic name to create
* @return a CompletionStage Void
*/
public CompletionStage<Void> createTopic(String topicName) {
log.trace("Create topic thread {}", Thread.currentThread());
log.info("Create topic {}", topicName);
CompletableFuture<Void> promise = new CompletableFuture<>();
this.adminClient.createTopics(Collections.singletonList(new NewTopic(topicName, 2, (short) 1)))
.all()
.whenComplete((topic, exception) -> {
log.trace("Create topic callback thread {}", Thread.currentThread());
if (exception == null) {
promise.complete(topic);
} else {
promise.completeExceptionally(exception);
}
});
return promise;
}

/**
* Returns the description of the specified topics.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,10 @@ public void handle(RoutingContext routingContext, Handler<HttpBridgeEndpoint> ha
doGetTopic(routingContext);
break;

case CREATE_TOPIC:
doCreateTopic(routingContext);
break;

case LIST_PARTITIONS:
doListPartitions(routingContext);
break;
Expand Down Expand Up @@ -170,6 +174,32 @@ public void doGetTopic(RoutingContext routingContext) {
});
}

/**
* Create a topic described in the HTTP request
*
* @param routingContext the routing context
*/
public void doCreateTopic(RoutingContext routingContext) {
String topicName = routingContext.pathParam("topicname");

this.kafkaBridgeAdmin.createTopic(topicName)
.whenComplete(((topic, exception) -> {
log.trace("Create topic handler thread {}", Thread.currentThread());
if (exception == null) {
ArrayNode root = JsonUtils.createArrayNode();
HttpUtils.sendResponse(routingContext, HttpResponseStatus.OK.code(),
BridgeContentType.KAFKA_JSON, JsonUtils.jsonToBytes(root));
} else {
HttpBridgeError error = new HttpBridgeError(
HttpResponseStatus.INTERNAL_SERVER_ERROR.code(),
exception.getMessage()
);
HttpUtils.sendResponse(routingContext, HttpResponseStatus.INTERNAL_SERVER_ERROR.code(),
BridgeContentType.KAFKA_JSON, JsonUtils.jsonToBytes(error.toJson()));
}
}));
}

/**
* Get partitions information related to the topic in the HTTP request
*
Expand Down
14 changes: 14 additions & 0 deletions src/main/java/io/strimzi/kafka/bridge/http/HttpBridge.java
Original file line number Diff line number Diff line change
Expand Up @@ -159,6 +159,7 @@ public void start(Promise<Void> startPromise) {
routerBuilder.operation(this.SEEK_TO_END.getOperationId().toString()).handler(this.SEEK_TO_END);
routerBuilder.operation(this.LIST_TOPICS.getOperationId().toString()).handler(this.LIST_TOPICS);
routerBuilder.operation(this.GET_TOPIC.getOperationId().toString()).handler(this.GET_TOPIC);
routerBuilder.operation(this.CREATE_TOPIC.getOperationId().toString()).handler(this.CREATE_TOPIC);
routerBuilder.operation(this.LIST_PARTITIONS.getOperationId().toString()).handler(this.LIST_PARTITIONS);
routerBuilder.operation(this.GET_PARTITION.getOperationId().toString()).handler(this.GET_PARTITION);
routerBuilder.operation(this.GET_OFFSETS.getOperationId().toString()).handler(this.GET_OFFSETS);
Expand Down Expand Up @@ -378,6 +379,11 @@ private void getTopic(RoutingContext routingContext) {
processAdminClient(routingContext);
}

private void createTopic(RoutingContext routingContext) {
this.httpBridgeContext.setOpenApiOperation(HttpOpenApiOperations.CREATE_TOPIC);
processAdminClient(routingContext);
}

private void listPartitions(RoutingContext routingContext) {
this.httpBridgeContext.setOpenApiOperation(HttpOpenApiOperations.LIST_PARTITIONS);
processAdminClient(routingContext);
Expand Down Expand Up @@ -721,6 +727,14 @@ public void process(RoutingContext routingContext) {
}
};

final HttpOpenApiOperation CREATE_TOPIC = new HttpOpenApiOperation(HttpOpenApiOperations.CREATE_TOPIC) {

@Override
public void process(RoutingContext routingContext) {
createTopic(routingContext);
}
};

final HttpOpenApiOperation LIST_PARTITIONS = new HttpOpenApiOperation(HttpOpenApiOperations.LIST_PARTITIONS) {

@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ public enum HttpOpenApiOperations {
LIST_TOPICS("listTopics"),
/** get information for a specific topic */
GET_TOPIC("getTopic"),
/** creates a topic with specified name */
CREATE_TOPIC("createTopic"),
/** list partitions for a specific topic */
LIST_PARTITIONS("listPartitions"),
/** get partition information for a specific topic */
Expand Down
50 changes: 50 additions & 0 deletions src/main/resources/openapi.json
Original file line number Diff line number Diff line change
Expand Up @@ -748,6 +748,56 @@
}
]
},
"/create-topic/{topicname}": {
"post": {
"tags": [
"Topics"
],
"description": "Creates a topic with given topic name.",
"operationId": "createTopic",
"responses": {
"200": {
"description": "Topic metadata",
"content": {
"application/vnd.kafka.v2+json": {
"schema": {
"$ref": "#/components/schemas/TopicMetadata"
}
}
}
},
"404": {
"description": "The specified topic was not found.",
"content": {
"application/vnd.kafka.v2+json": {
"schema": {
"$ref": "#/components/schemas/Error"
},
"examples": {
"response": {
"value": {
"error_code": 404,
"message": "The specified topic was not found."
}
}
}
}
}
}
}
},
"parameters": [
{
"name": "topicname",
"in": "path",
"description": "Name of the topic to send records to or retrieve metadata from.",
"required": true,
"schema": {
"type": "string"
}
}
]
},
"/consumers/{groupid}/instances/{name}/records": {
"get": {
"tags": [
Expand Down
15 changes: 15 additions & 0 deletions src/test/java/io/strimzi/kafka/bridge/http/AdminClientIT.java
Original file line number Diff line number Diff line change
Expand Up @@ -215,4 +215,19 @@ void setupTopic(VertxTestContext context, String topic, int partitions, int coun
});
producer.get(TEST_TIMEOUT, TimeUnit.SECONDS);
}

@Test
void createTopicTest(VertxTestContext context) {
baseService()
.postRequest("/create-topic/" + topic)
.as(BodyCodec.jsonArray())
.send(ar -> {
context.verify(() -> {
assertThat(ar.succeeded(), is(true));
HttpResponse<JsonArray> response = ar.result();
assertThat(response.statusCode(), is(HttpResponseStatus.OK.code()));
});
context.completeNow();
});
}
}

0 comments on commit b97247a

Please sign in to comment.