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

org.apache.camel.converter.stream.FileInputStreamCache Maven / Gradle / Ivy

The newest version!
/*
 * 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.converter.stream;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Files;
import java.nio.file.StandardOpenOption;
import java.security.GeneralSecurityException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

import javax.crypto.CipherInputStream;
import javax.crypto.CipherOutputStream;

import org.apache.camel.Exchange;
import org.apache.camel.ExchangePropertyKey;
import org.apache.camel.RuntimeCamelException;
import org.apache.camel.StreamCache;
import org.apache.camel.spi.StreamCachingStrategy;
import org.apache.camel.spi.Synchronization;
import org.apache.camel.spi.UnitOfWork;
import org.apache.camel.support.SynchronizationAdapter;
import org.apache.camel.util.FileUtil;
import org.apache.camel.util.IOHelper;
import org.apache.camel.util.ObjectHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * A {@link StreamCache} for {@link File}s.
 * 

* Important: All the classes from the Camel release that implements {@link StreamCache} is NOT intended for end * users to create as instances, but they are part of Camels * stream-caching functionality. */ public final class FileInputStreamCache extends InputStream implements StreamCache { private final Lock lock = new ReentrantLock(); private InputStream stream; private final long length; private final FileInputStreamCache.TempFileManager tempFileManager; private final File file; private final CipherPair ciphers; /** Only for testing purposes. */ public FileInputStreamCache(File file) { this(new TempFileManager(file, true)); } FileInputStreamCache(TempFileManager closer) { this.file = closer.getTempFile(); this.stream = null; this.ciphers = closer.getCiphers(); this.length = file.length(); this.tempFileManager = closer; this.tempFileManager.add(this); } @Override public void close() { if (stream != null) { IOHelper.close(stream); } } @Override public void reset() { lock.lock(); try { // reset by closing and creating a new stream based on the file close(); // reset by creating a new stream based on the file stream = null; if (!file.exists()) { throw new RuntimeCamelException("Cannot reset stream from file " + file); } } finally { lock.unlock(); } } @Override public void writeTo(OutputStream os) throws IOException { if (stream == null && ciphers == null) { Files.copy(file.toPath(), os); } else { IOHelper.copy(getInputStream(), os); } } @Override public StreamCache copy(Exchange exchange) throws IOException { tempFileManager.addExchange(exchange); return new FileInputStreamCache(tempFileManager); } @Override public boolean inMemory() { return false; } @Override public long length() { return length; } @Override public long position() { return -1; } @Override public int available() throws IOException { return getInputStream().available(); } @Override public int read() throws IOException { return getInputStream().read(); } @Override public int read(byte[] b) throws IOException { return getInputStream().read(b); } @Override public int read(byte[] b, int off, int len) throws IOException { return getInputStream().read(b, off, len); } @Override public long skip(long n) throws IOException { return getInputStream().skip(n); } @Override public byte[] readAllBytes() throws IOException { return getInputStream().readAllBytes(); } @Override public byte[] readNBytes(int len) throws IOException { return getInputStream().readNBytes(len); } @Override public int readNBytes(byte[] b, int off, int len) throws IOException { return getInputStream().readNBytes(b, off, len); } @Override public long transferTo(OutputStream out) throws IOException { return getInputStream().transferTo(out); } private InputStream getInputStream() throws IOException { if (stream == null) { stream = createInputStream(file); } return stream; } private InputStream createInputStream(File file) throws IOException { InputStream in = new BufferedInputStream(Files.newInputStream(file.toPath(), StandardOpenOption.READ)); if (ciphers != null) { in = new CipherInputStream(in, ciphers.createDecryptor()) { boolean closed; @Override public void close() throws IOException { if (!closed) { super.close(); closed = true; } } }; } return in; } /** * Manages the temporary file for the file input stream caches. * * Collects all FileInputStreamCache instances of the temporary file. Counts the number of exchanges which have a * FileInputStreamCache instance of the temporary file. Deletes the temporary file, if all exchanges are done. * * @see CachedOutputStream */ static class TempFileManager { private static final Logger LOG = LoggerFactory.getLogger(TempFileManager.class); /** * Indicator whether the file input stream caches are closed on completion of the exchanges. */ private final boolean closedOnCompletion; private final Lock lock = new ReentrantLock(); private final AtomicInteger exchangeCounter = new AtomicInteger(); private File tempFile; private OutputStream outputStream; // file output stream private CipherPair ciphers; // there can be several input streams, for example in the multi-cast, or wiretap parallel processing private List fileInputStreamCaches; /** Only for testing. */ private TempFileManager(File file, boolean closedOnCompletion) { this(closedOnCompletion); this.tempFile = file; } TempFileManager(boolean closedOnCompletion) { this.closedOnCompletion = closedOnCompletion; } /** * Adds a FileInputStreamCache instance to the closer. *

* Must be synchronized, because can be accessed by several threads. */ void add(FileInputStreamCache fileInputStreamCache) { lock.lock(); try { if (fileInputStreamCaches == null) { fileInputStreamCaches = new ArrayList<>(3); } fileInputStreamCaches.add(fileInputStreamCache); } finally { lock.unlock(); } } void addExchange(Exchange exchange) { if (closedOnCompletion) { exchangeCounter.incrementAndGet(); // add on completion so we can cleanup after the exchange is done such as deleting temporary files Synchronization onCompletion = new SynchronizationAdapter() { @Override public void onDone(Exchange exchange) { int actualExchanges = exchangeCounter.decrementAndGet(); if (actualExchanges == 0) { // only one exchange (one thread) left, therefore we must not synchronize the following lines of code try { closeFileInputStreams(); if (outputStream != null) { outputStream.close(); } } catch (Exception e) { LOG.warn("Error closing streams. This exception will be ignored.", e); } try { cleanUpTempFile(); } catch (Exception e) { LOG.warn("Error deleting temporary cache file: {}. This exception will be ignored.", tempFile, e); } } } @Override public String toString() { return "OnCompletion[CachedOutputStream]"; } }; UnitOfWork streamCacheUnitOfWork = exchange.getProperty(ExchangePropertyKey.STREAM_CACHE_UNIT_OF_WORK, UnitOfWork.class); if (streamCacheUnitOfWork != null && streamCacheUnitOfWork.getRoute() != null) { // The stream cache must sometimes not be closed when the exchange is deleted. This is for example the // case in the splitter and multi-cast case with AggregationStrategy where the result of the sub-routes // are aggregated later in the main route. Here, the cached streams of the sub-routes must be closed with // the Unit of Work of the main route. // streamCacheUnitOfWork.getRoute() != null means that the unit of work is still active and the done method // was not yet called: It can happen that streamCacheUnitOfWork.getRoute() == null in the split or // multi-cast case when there is a timeout on the main route and an exchange of the sub-route is added after // the timeout. This we have to avoid because the stream cache would never be closed then. streamCacheUnitOfWork.addSynchronization(onCompletion); } else { // add on completion so we can cleanup after the exchange is done such as deleting temporary files exchange.getExchangeExtension().addOnCompletion(onCompletion); } } } OutputStream createOutputStream(StreamCachingStrategy strategy) throws IOException { // should only be called once if (tempFile != null) { throw new IllegalStateException("The method 'createOutputStream' can only be called once!"); } if (closedOnCompletion && exchangeCounter.get() == 0) { // exchange was already stopped -> in this case the tempFile would never be deleted. // This can happen when in the splitter or Multi-cast case with parallel processing, the CachedOutputStream is created when the main unit of work // is still active, but has a timeout and after the timeout which stops the unit of work the FileOutputStream is created. // We only can throw here an Exception and inform the user that the processing took longer than the set timeout. String error = "Cannot create a FileOutputStream for Stream Caching, because this FileOutputStream would never be removed from the file system." + " This situation can happen with a Splitter or Multi Cast in parallel processing if there is a timeout set on the Splitter or Multi Cast, " + " and the processing in a sub-branch takes longer than the timeout. Consider to increase the timeout."; LOG.error(error); throw new IOException(error); } tempFile = FileUtil.createTempFile("cos", ".tmp", strategy.getSpoolDirectory()); LOG.trace("Creating temporary stream cache file: {}", tempFile); OutputStream out = new BufferedOutputStream( Files.newOutputStream(tempFile.toPath(), StandardOpenOption.CREATE, StandardOpenOption.WRITE)); if (ObjectHelper.isNotEmpty(strategy.getSpoolCipher())) { try { if (ciphers == null) { ciphers = new CipherPair(strategy.getSpoolCipher()); } } catch (GeneralSecurityException e) { throw new IOException(e.getMessage(), e); } out = new CipherOutputStream(out, ciphers.getEncryptor()) { boolean closed; @Override public void close() throws IOException { if (!closed) { super.close(); closed = true; } } }; } outputStream = out; return out; } FileInputStreamCache newStreamCache() { return new FileInputStreamCache(this); } void closeFileInputStreams() { if (fileInputStreamCaches != null) { for (FileInputStreamCache fileInputStreamCache : fileInputStreamCaches) { fileInputStreamCache.close(); } fileInputStreamCaches.clear(); } } void cleanUpTempFile() { // cleanup temporary file try { if (tempFile != null) { FileUtil.deleteFile(tempFile); tempFile = null; } } catch (Exception e) { LOG.warn("Error deleting temporary cache file: {}. This exception will be ignored.", tempFile, e); } } File getTempFile() { return tempFile; } CipherPair getCiphers() { return ciphers; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy