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

org.apache.camel.component.stream.StreamConsumer Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.
 */
package org.apache.camel.component.stream;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutorService;
import java.util.stream.Stream;

import org.apache.camel.Exchange;
import org.apache.camel.Processor;
import org.apache.camel.support.DefaultConsumer;
import org.apache.camel.support.service.ServiceHelper;
import org.apache.camel.util.FileUtil;
import org.apache.camel.util.IOHelper;
import org.apache.camel.util.StringHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Consumer that can read from streams
 */
public class StreamConsumer extends DefaultConsumer implements Runnable {

    private static final Logger LOG = LoggerFactory.getLogger(StreamConsumer.class);

    private static final String TYPES = "in,file,http";
    private static final String INVALID_URI = "Invalid uri, valid form: 'stream:{" + TYPES + "}'";
    private static final List TYPES_LIST = Arrays.asList(TYPES.split(","));
    private ExecutorService executor;
    private FileWatcherStrategy fileWatcher;
    private volatile boolean watchFileChanged;
    private volatile InputStream inputStream = System.in;
    private volatile InputStream inputStreamToClose;
    private volatile URLConnection urlConnectionToClose;
    private volatile File file;
    private StreamEndpoint endpoint;
    private String uri;
    private volatile boolean initialPromptDone;
    private final List lines = new CopyOnWriteArrayList<>();

    public StreamConsumer(StreamEndpoint endpoint, Processor processor, String uri) throws Exception {
        super(endpoint, processor);
        this.endpoint = endpoint;
        this.uri = uri;
        validateUri(uri);
    }

    @Override
    protected void doStart() throws Exception {
        super.doStart();

        // use file watch service if we read from file
        if (endpoint.isFileWatcher()) {
            String dir = new File(endpoint.getFileName()).getParent();
            fileWatcher = new FileWatcherStrategy(dir, file -> {
                String onlyName = file.getName();
                String target = FileUtil.stripPath(endpoint.getFileName());
                LOG.trace("File changed: {}", onlyName);
                if (onlyName.equals(target)) {
                    // file is changed
                    watchFileChanged = true;
                }
            });
            fileWatcher.setCamelContext(getEndpoint().getCamelContext());
        }
        ServiceHelper.startService(fileWatcher);

        // if we scan the stream we are lenient and can wait for the stream to be available later
        if (!endpoint.isScanStream()) {
            initializeStreamLineMode();
        }

        executor = endpoint.getCamelContext().getExecutorServiceManager().newSingleThreadExecutor(this,
                endpoint.getEndpointUri());
        executor.execute(this);

        if (endpoint.getGroupLines() < 0) {
            throw new IllegalArgumentException(
                    "Option groupLines must be 0 or positive number, was " + endpoint.getGroupLines());
        }
    }

    @Override
    public void doStop() throws Exception {
        if (executor != null) {
            endpoint.getCamelContext().getExecutorServiceManager().shutdownNow(executor);
            executor = null;
        }
        ServiceHelper.stopAndShutdownService(fileWatcher);
        lines.clear();

        IOHelper.close(inputStreamToClose);
        if (urlConnectionToClose != null) {
            closeURLConnection(urlConnectionToClose);
            urlConnectionToClose = null;
        }
        super.doStop();
    }

    @Override
    public void run() {
        try {
            if (endpoint.isReadLine()) {
                readFromStreamLineMode();
            } else {
                readFromStreamRawMode();
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        } catch (Exception e) {
            getExceptionHandler().handleException(e);
        }
    }

    private BufferedReader initializeStreamLineMode() throws Exception {
        // close old stream, before obtaining a new stream
        IOHelper.close(inputStreamToClose);
        if (urlConnectionToClose != null) {
            closeURLConnection(urlConnectionToClose);
        }

        if ("in".equals(uri)) {
            inputStream = System.in;
            inputStreamToClose = null;
        } else if ("file".equals(uri)) {
            inputStream = resolveStreamFromFile();
            inputStreamToClose = inputStream;
        } else if ("http".equals(uri)) {
            inputStream = resolveStreamFromUrl();
            inputStreamToClose = inputStream;
        }

        if (inputStream != null) {
            if ("http".equals(uri)) {
                // read as-is
                return IOHelper.buffered(new InputStreamReader(inputStream));
            } else {
                Charset charset = endpoint.getCharset();
                return IOHelper.buffered(new InputStreamReader(inputStream, charset));
            }
        } else {
            return null;
        }
    }

    private InputStream initializeStreamRawMode() throws Exception {
        // close old stream, before obtaining a new stream
        IOHelper.close(inputStreamToClose);
        if (urlConnectionToClose != null) {
            closeURLConnection(urlConnectionToClose);
        }

        if ("in".equals(uri)) {
            inputStream = System.in;
            // do not close regular inputStream as it may be System.in etc.
            inputStreamToClose = null;
        } else if ("file".equals(uri)) {
            inputStream = resolveStreamFromFile();
            inputStreamToClose = inputStream;
        } else if ("http".equals(uri)) {
            inputStream = resolveStreamFromUrl();
            inputStreamToClose = inputStream;
        }

        return inputStream;
    }

    private void readFromStreamRawMode() throws Exception {
        long index = 0;
        InputStream is = initializeStreamRawMode();

        if (endpoint.isScanStream()) {
            // repeat scanning from stream
            while (isRunAllowed()) {

                byte[] data = null;
                try {
                    data = is.readAllBytes();
                } catch (IOException e) {
                    // ignore
                }
                boolean eos = data == null || data.length == 0;

                if (isRunAllowed() && endpoint.isRetry()) {
                    boolean reOpen = true;
                    if (endpoint.isFileWatcher()) {
                        reOpen = watchFileChanged;
                    }
                    if (reOpen) {
                        LOG.debug("File: {} changed/rollover, re-reading file from beginning", file);
                        is = initializeStreamRawMode();
                        // we have re-initialized the stream so lower changed flag
                        if (endpoint.isFileWatcher()) {
                            watchFileChanged = false;
                        }
                    } else {
                        LOG.trace("File: {} not changed since last read", file);
                    }
                }

                // sleep only if there is no input
                if (eos) {
                    try {
                        Thread.sleep(endpoint.getScanStreamDelay());
                    } catch (InterruptedException e) {
                        Thread.currentThread().interrupt();
                        break;
                    }
                }
            }
        } else {
            // regular read stream once until end of stream
            boolean eos = false;
            byte[] data = null;
            while (!eos && isRunAllowed()) {
                if (endpoint.getPromptMessage() != null) {
                    doPromptMessage();
                }

                try {
                    data = is.readAllBytes();
                } catch (IOException e) {
                    // ignore
                }
                eos = data == null || data.length == 0;
                if (!eos) {
                    processRaw(data, index);
                }
            }
        }
    }

    private void readFromStreamLineMode() throws Exception {
        long index = 0;
        String line;
        BufferedReader br = initializeStreamLineMode();

        if (endpoint.isScanStream()) {
            // repeat scanning from stream
            while (isRunAllowed()) {
                if (br != null) {
                    line = br.readLine();
                    LOG.trace("Read line: {}", line);
                } else {
                    line = null;
                }
                boolean eos = line == null;
                if (!eos && isRunAllowed()) {
                    index = processLine(line, false, index);
                } else if (eos && isRunAllowed() && endpoint.isRetry()) {
                    boolean reOpen = true;
                    if (endpoint.isFileWatcher()) {
                        reOpen = watchFileChanged;
                    }
                    if (reOpen) {
                        LOG.debug("File: {} changed/rollover, re-reading file from beginning", file);
                        br = initializeStreamLineMode();
                        // we have re-initialized the stream so lower changed flag
                        if (endpoint.isFileWatcher()) {
                            watchFileChanged = false;
                        }
                    } else {
                        LOG.trace("File: {} not changed since last read", file);
                    }
                }

                // sleep only if there is no input
                if (eos) {
                    try {
                        Thread.sleep(endpoint.getScanStreamDelay());
                    } catch (InterruptedException e) {
                        Thread.currentThread().interrupt();
                        break;
                    }
                }
            }
        } else {
            // regular read stream once until end of stream
            boolean eos = false;
            String line2 = null;
            while (!eos && isRunAllowed()) {
                if (endpoint.getPromptMessage() != null) {
                    doPromptMessage();
                }

                if (line2 == null) {
                    line = br.readLine();
                } else {
                    line = line2;
                }
                LOG.trace("Read line: {}", line);

                eos = line == null;
                if (!eos && isRunAllowed()) {
                    // read ahead if there is more data
                    line2 = readAhead(br);
                    boolean last = line2 == null;
                    index = processLine(line, last, index);
                }
            }
            // EOL so trigger any
            processLine(null, true, index);
        }
        // important: do not close the reader as it will close the standard system.in etc.
    }

    /**
     * Strategy method for processing the line
     */
    protected long processLine(String line, boolean last, long index) throws Exception {
        lock.lock();
        try {
            if (endpoint.getGroupLines() > 0) {
                // remember line
                if (line != null) {
                    lines.add(line);
                }

                // should we flush lines?
                if (!lines.isEmpty() && (lines.size() >= endpoint.getGroupLines() || last)) {
                    // spit out lines as we hit the size, or it was the last
                    List copy = new ArrayList<>(lines);
                    Object body = endpoint.getGroupStrategy().groupLines(copy);
                    // remember to inc index when we create an exchange
                    Exchange exchange = createExchange(body, index++, last);

                    // clear lines
                    lines.clear();

                    getProcessor().process(exchange);
                }
            } else if (line != null) {
                // single line
                // remember to inc index when we create an exchange
                Exchange exchange = createExchange(line, index++, last);
                getProcessor().process(exchange);
            }

            return index;
        } finally {
            lock.unlock();
        }
    }

    /**
     * Strategy method for processing the data
     */
    protected long processRaw(byte[] body, long index) throws Exception {
        lock.lock();
        try {
            Exchange exchange = createExchange(body, index++, true);
            getProcessor().process(exchange);
            return index;
        } finally {
            lock.unlock();
        }
    }

    /**
     * Strategy method for prompting the prompt message
     */
    protected void doPromptMessage() {
        long delay = 0;

        if (!initialPromptDone && endpoint.getInitialPromptDelay() > 0) {
            initialPromptDone = true;
            delay = endpoint.getInitialPromptDelay();
        } else if (endpoint.getPromptDelay() > 0) {
            delay = endpoint.getPromptDelay();
        }

        if (delay > 0) {
            try {
                Thread.sleep(delay);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        }
        if (inputStream == System.in) {
            System.out.print(endpoint.getPromptMessage());
        }
    }

    private String readAhead(BufferedReader br) throws IOException {
        if (uri.equals("in")) {
            // do not read ahead with reading from system in
            return null;
        } else {
            return br.readLine();
        }
    }

    private InputStream resolveStreamFromFile() throws IOException {
        String fileName = endpoint.getFileName();
        StringHelper.notEmpty(fileName, "fileName");

        FileInputStream fileStream;

        file = new File(fileName);

        if (LOG.isDebugEnabled()) {
            LOG.debug("File to be scanned: {}, path: {}", file.getName(), file.getAbsolutePath());
        }

        if (file.canRead()) {
            fileStream = new FileInputStream(file);
        } else if (endpoint.isScanStream()) {
            // if we scan the stream then it may not be available and we should return null
            fileStream = null;
        } else {
            throw new IllegalArgumentException(INVALID_URI);
        }

        return fileStream;
    }

    /**
     * From a comma-separated list of headers in the format of "FIELD=VALUE" or "FIELD:VALUE", split on the commas and
     * split on the separator to create a stream of Map.Entry values while filtering out invalid combinations
     *
     * @param  headerList A string containing a comma-separated list of headers
     * @return            A Stream of Map.Entry items which can then be added as headers to a URLConnection
     */
    Stream> parseHeaders(String headerList) {
        return Arrays.asList(headerList.split(","))
                .stream()
                .map(s -> s.split("[=:]"))
                .filter(h -> h.length == 2)
                .map(h -> Map.entry(h[0].trim(), h[1].trim()));
    }

    private InputStream resolveStreamFromUrl() throws IOException {
        String url = endpoint.getHttpUrl();
        StringHelper.notEmpty(url, "httpUrl");

        urlConnectionToClose = new URL(url).openConnection();
        urlConnectionToClose.setUseCaches(false);
        String headers = endpoint.getHttpHeaders();
        if (headers != null) {
            parseHeaders(headers)
                    .forEach(e -> urlConnectionToClose.setRequestProperty(e.getKey(), e.getValue()));
        }

        InputStream is;

        try {
            is = urlConnectionToClose.getInputStream();
        } catch (IOException e) {
            // close the http connection to avoid
            // leaking gaps in case of an exception
            if (urlConnectionToClose instanceof HttpURLConnection) {
                ((HttpURLConnection) urlConnectionToClose).disconnect();
            }
            throw e;
        }

        return is;
    }

    private void validateUri(String uri) throws IllegalArgumentException {
        String[] s = uri.split(":");
        if (s.length < 2) {
            throw new IllegalArgumentException(INVALID_URI);
        }
        String[] t = s[1].split("\\?");

        if (t.length < 1) {
            throw new IllegalArgumentException(INVALID_URI);
        }

        this.uri = t[0].trim();
        if (this.uri.startsWith("//")) {
            this.uri = this.uri.substring(2);
        }

        if (!TYPES_LIST.contains(this.uri)) {
            throw new IllegalArgumentException(INVALID_URI);
        }
    }

    protected Exchange createExchange(Object body, long index, boolean last) {
        Exchange exchange = createExchange(true);
        exchange.getIn().setBody(body);
        exchange.getIn().setHeader(StreamConstants.STREAM_INDEX, index);
        exchange.getIn().setHeader(StreamConstants.STREAM_COMPLETE, last);
        return exchange;
    }

    private static void closeURLConnection(URLConnection con) {
        if (con instanceof HttpURLConnection) {
            try {
                ((HttpURLConnection) con).disconnect();
            } catch (Exception e) {
                // ignore
            }
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy