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

com.redislabs.riot.transfer.FlowThread Maven / Gradle / Ivy

There is a newer version: 1.11.2
Show newest version
package com.redislabs.riot.transfer;

import lombok.Builder;
import lombok.Getter;
import lombok.extern.slf4j.Slf4j;
import org.springframework.batch.item.ExecutionContext;

import java.util.List;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;

@Slf4j
public class FlowThread implements Runnable {

    public final static String CONTEXT_PARTITION = "partition";
    public final static String CONTEXT_PARTITIONS = "partitions";

    private final int threadId;
    private final int threads;
    @Getter
    private final Flow flow;
    private final Batcher batcher;
    private final Long flushRate;
    private long readCount;
    private long writeCount;
    private final ScheduledExecutorService scheduler = Executors.newSingleThreadScheduledExecutor();
    private ScheduledFuture flushFuture;
    private boolean running;
    private boolean stopped;

    @Builder
    public FlowThread(int threadId, int threads, Flow flow, Batcher batcher, Long flushRate) {
        super();
        this.threadId = threadId;
        this.threads = threads;
        this.flow = flow;
        this.batcher = batcher;
        this.flushRate = flushRate;
    }

    @Override
    public void run() {
        ExecutionContext executionContext = new ExecutionContext();
        executionContext.putInt(CONTEXT_PARTITION, threadId);
        executionContext.putInt(CONTEXT_PARTITIONS, threads);
        try {
            flow.open(executionContext);
        } catch (Exception e) {
            log.error("Could not open {}", flow.getName(), e);
            return;
        }
        if (flushRate != null) {
            flushFuture = scheduler.scheduleAtFixedRate(new Flusher(), flushRate, flushRate, TimeUnit.MILLISECONDS);
        }
        running = true;
        try {
            List items;
            while ((items = batcher.next()) != null && !stopped) {
                write(items);
            }
        } catch (Exception e) {
            log.error("Error during {}", flow.getName(), e);
        } finally {
            if (flushFuture != null) {
                flushFuture.cancel(true);
            }
            scheduler.shutdown();
            try {
                flow.close();
            } catch (Exception e) {
                log.error("Could not close {}", flow.getName(), e);
            }
            this.running = false;
            log.debug("Flow {} thread {} finished", flow.getName(), threadId);
        }

    }

    public Metrics progress() {
        return Metrics.builder().reads(readCount).writes(writeCount).runningThreads(running ? 1 : 0).build();
    }

    public void stop() {
        stopped = true;
    }

    private void write(List items) throws Exception {
        readCount += items.size();
        flow.getWriter().write(items);
        writeCount += items.size();
    }

    private class Flusher implements Runnable {

        @Override
        public void run() {
            List items = batcher.flush();
            if (!items.isEmpty()) {
                try {
                    FlowThread.this.write(items);
                } catch (Exception e) {
                    log.error("Could not flush {}", flow.getName(), e);
                }
            }
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy