From 990dd93356460f0b393301be87da4a5b16a9457d Mon Sep 17 00:00:00 2001 From: Juan Pablo Crossley Date: Mon, 22 Nov 2021 15:47:12 +0100 Subject: [PATCH] Implemented a new strategy for Sink allowing users to provide an object id as string in a field and then it will be converted to ObjectId inside the connector --- .../BsonOidProvidedInKeyStrategy.java | 27 +++++++ .../BsonOidProvidedInValueStrategy.java | 27 +++++++ .../id/strategy/BsonOidProvidedStrategy.java | 77 +++++++++++++++++++ .../sink/processor/id/strategy/README.md | 20 +++++ 4 files changed, 151 insertions(+) create mode 100644 src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInKeyStrategy.java create mode 100644 src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInValueStrategy.java create mode 100644 src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedStrategy.java create mode 100644 src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/README.md diff --git a/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInKeyStrategy.java b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInKeyStrategy.java new file mode 100644 index 000000000..c965a0ad0 --- /dev/null +++ b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInKeyStrategy.java @@ -0,0 +1,27 @@ +/* + * Copyright 2008-present MongoDB, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * Original Work: Apache License, Version 2.0, Copyright 2017 Hans-Peter Grahsl. + */ + +package com.mongodb.kafka.connect.sink.processor.id.strategy; + +public class BsonOidProvidedInKeyStrategy extends BsonOidProvidedStrategy { + + public BsonOidProvidedInKeyStrategy() { + super(ProvidedIn.KEY); + } + +} diff --git a/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInValueStrategy.java b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInValueStrategy.java new file mode 100644 index 000000000..79aba06c6 --- /dev/null +++ b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedInValueStrategy.java @@ -0,0 +1,27 @@ +/* + * Copyright 2008-present MongoDB, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * Original Work: Apache License, Version 2.0, Copyright 2017 Hans-Peter Grahsl. + */ + +package com.mongodb.kafka.connect.sink.processor.id.strategy; + +public class BsonOidProvidedInValueStrategy extends BsonOidProvidedStrategy { + + public BsonOidProvidedInValueStrategy() { + super(ProvidedIn.VALUE); + } + +} diff --git a/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedStrategy.java b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedStrategy.java new file mode 100644 index 000000000..47c0d0e28 --- /dev/null +++ b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/BsonOidProvidedStrategy.java @@ -0,0 +1,77 @@ +/* + * Copyright 2008-present MongoDB, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * Original Work: Apache License, Version 2.0, Copyright 2017 Hans-Peter Grahsl. + */ + +package com.mongodb.kafka.connect.sink.processor.id.strategy; + +import static com.mongodb.kafka.connect.sink.MongoSinkTopicConfig.ID_FIELD; + +import java.util.Optional; + +import org.apache.kafka.connect.errors.DataException; +import org.apache.kafka.connect.sink.SinkRecord; + +import org.bson.BsonObjectId; +import org.bson.BsonString; +import org.bson.BsonDocument; +import org.bson.BsonNull; +import org.bson.BsonValue; +import org.bson.types.ObjectId; + +import com.mongodb.kafka.connect.sink.converter.SinkDocument; + +abstract class BsonOidProvidedStrategy implements IdStrategy { + + protected enum ProvidedIn { + KEY, + VALUE + } + + private ProvidedIn where; + + BsonOidProvidedStrategy(final ProvidedIn where) { + this.where = where; + } + + @Override + public BsonValue generateId(final SinkDocument doc, final SinkRecord orig) { + Optional optionalDoc = Optional.empty(); + if (where.equals(ProvidedIn.KEY)) { + optionalDoc = doc.getKeyDoc(); + } + + if (where.equals(ProvidedIn.VALUE)) { + optionalDoc = doc.getValueDoc(); + } + + BsonValue id = + optionalDoc + .map(d -> d.get(ID_FIELD)) + .orElseThrow( + () -> + new DataException( + "Provided id strategy is used but the document structure either contained" + + " no _id field or it was null")); + + if (id instanceof BsonNull) { + throw new DataException( + "Provided id strategy used but the document structure contained an _id of type BsonNull"); + } + return new BsonObjectId(new ObjectId((((BsonString)id).getValue()))); + } +} + diff --git a/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/README.md b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/README.md new file mode 100644 index 000000000..3befcfead --- /dev/null +++ b/src/main/java/com/mongodb/kafka/connect/sink/processor/id/strategy/README.md @@ -0,0 +1,20 @@ +# MongoDB Kafka BsonObjectId + +This implementaion allows the MongoDB Kafka sink connector to convert an existing _id that is string as ObjectId + +# Usage + +If the id is provided as part of the value you can use: +``` + "post.processor.chain": "com.mongodb.kafka.connect.sink.processor.DocumentIdAdder", + "document.id.strategy": "com.mongodb.kafka.connect.sink.processor.id.strategy.BsonOidProvidedInValueStrategy", + "document.id.strategy.overwrite.existing": true, +``` + +If the id is provided as part of the key you can use: +``` + "post.processor.chain": "com.mongodb.kafka.connect.sink.processor.DocumentIdAdder", + "document.id.strategy": "com.mongodb.kafka.connect.sink.processor.id.strategy.BsonOidProvidedInKeyStrategy", + "document.id.strategy.overwrite.existing": true, +``` +