All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.debezium.server.pravega.PravegaChangeConsumer Maven / Gradle / Ivy

There is a newer version: 3.0.0.CR1
Show newest version
/*
 * Copyright Debezium Authors.
 *
 * Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
 */
package io.debezium.server.pravega;

import java.net.URI;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import jakarta.annotation.PostConstruct;
import jakarta.annotation.PreDestroy;
import jakarta.enterprise.context.Dependent;
import jakarta.inject.Named;

import org.eclipse.microprofile.config.ConfigProvider;
import org.eclipse.microprofile.config.inject.ConfigProperty;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import io.debezium.engine.ChangeEvent;
import io.debezium.engine.DebeziumEngine.ChangeConsumer;
import io.debezium.engine.DebeziumEngine.RecordCommitter;
import io.debezium.server.BaseChangeConsumer;
import io.pravega.client.ClientConfig;
import io.pravega.client.EventStreamClientFactory;
import io.pravega.client.stream.EventStreamWriter;
import io.pravega.client.stream.EventWriterConfig;
import io.pravega.client.stream.Transaction;
import io.pravega.client.stream.TransactionalEventStreamWriter;
import io.pravega.client.stream.TxnFailedException;
import io.pravega.client.stream.impl.ByteArraySerializer;

@Named("pravega")
@Dependent
public class PravegaChangeConsumer extends BaseChangeConsumer implements ChangeConsumer> {

    private static final Logger LOGGER = LoggerFactory.getLogger(PravegaChangeConsumer.class);

    private static final String PROP_PREFIX = "debezium.sink.pravega.";
    private static final String PROP_CONTROLLER = PROP_PREFIX + "controller.uri";
    private static final String PROP_SCOPE = PROP_PREFIX + "scope";
    private static final String PROP_TXN = PROP_PREFIX + "transaction";

    @ConfigProperty(name = PROP_CONTROLLER, defaultValue = "tcp://localhost:9090")
    URI controllerUri;

    private String scope;

    @ConfigProperty(name = PROP_TXN, defaultValue = "false")
    boolean txn;

    private ClientConfig clientConfig;
    private EventStreamClientFactory factory;
    private EventWriterConfig writerConfig;

    @PostConstruct
    void constructor() {
        scope = ConfigProvider.getConfig().getValue(PROP_SCOPE, String.class);
        clientConfig = ClientConfig.builder()
                .controllerURI(controllerUri)
                .build();
        LOGGER.debug("Creating client factory for scope {} with controller {}", scope, controllerUri);
        factory = EventStreamClientFactory.withScope(scope, clientConfig);
        writerConfig = EventWriterConfig.builder().build();
    }

    @PreDestroy
    void destructor() {
        LOGGER.debug("Closing client factory");
        factory.close();
    }

    @Override
    public void handleBatch(List> records, RecordCommitter> committer) throws InterruptedException {
        try (PravegaSink impl = (txn) ? new PravegaTxnSinkImpl() : new PravegaSinkImpl()) {
            impl.handleBatch(records, committer);
        }
        catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    class PravegaSinkImpl implements PravegaSink {
        private final Map> writers = new HashMap<>();

        @Override
        public void handleBatch(List> records, RecordCommitter> committer) throws InterruptedException {
            for (ChangeEvent changeEvent : records) {
                String streamName = streamNameMapper.map(changeEvent.destination());
                final EventStreamWriter writer = writers.computeIfAbsent(streamName, (stream) -> createWriter(stream));
                if (changeEvent.key() != null) {
                    writer.writeEvent(getString(changeEvent.key()), getBytes(changeEvent.value()));
                }
                else {
                    writer.writeEvent(getBytes(changeEvent.value()));
                }
                committer.markProcessed(changeEvent);
            }
            committer.markBatchFinished();
        }

        private EventStreamWriter createWriter(String stream) {
            LOGGER.debug("Creating writer for stream {}", stream);
            return factory.createEventWriter(stream, new ByteArraySerializer(), writerConfig);
        }

        @Override
        public void close() throws Exception {
            LOGGER.debug("Closing {} writer(s)", writers.size());
            writers.values().forEach(EventStreamWriter::close);
        }
    }

    class PravegaTxnSinkImpl implements PravegaSink {
        private final Map> writers = new HashMap<>();
        private final Map> txns = new HashMap<>();

        @Override
        public void handleBatch(List> records, RecordCommitter> committer) throws InterruptedException {
            for (ChangeEvent changeEvent : records) {
                String streamName = streamNameMapper.map(changeEvent.destination());
                final Transaction txn = txns.computeIfAbsent(streamName, (stream) -> createTxn(stream));
                try {
                    if (changeEvent.key() != null) {
                        txn.writeEvent(getString(changeEvent.key()), getBytes(changeEvent.value()));
                    }
                    else {
                        txn.writeEvent(getBytes(changeEvent.value()));
                    }
                }
                catch (TxnFailedException e) {
                    throw new RuntimeException(e);
                }
                committer.markProcessed(changeEvent);
            }
            txns.values().forEach(t -> {
                try {
                    t.commit();
                }
                catch (TxnFailedException e) {
                    throw new RuntimeException(e);
                }
            });
            committer.markBatchFinished();
            txns.clear();
        }

        private Transaction createTxn(String stream) {
            final TransactionalEventStreamWriter writer = writers.computeIfAbsent(stream, (s) -> createWriter(s));
            LOGGER.debug("Creating transaction for stream {}", stream);
            return writer.beginTxn();
        }

        private TransactionalEventStreamWriter createWriter(String stream) {
            LOGGER.debug("Creating writer for stream {}", stream);
            return factory.createTransactionalEventWriter(stream, new ByteArraySerializer(), writerConfig);
        }

        @Override
        public void close() throws Exception {
            LOGGER.debug("Closing {} writer(s)", writers.size());
            writers.values().forEach(TransactionalEventStreamWriter::close);
        }
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy