org.apache.myfaces.trinidadinternal.config.upload.UploadedFileImpl 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.myfaces.trinidadinternal.config.upload;
import java.io.BufferedInputStream;
import java.io.EOFException;
import java.io.InputStream;
import java.io.IOException;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.apache.myfaces.trinidad.model.ExtendedUploadedFile;
import org.apache.myfaces.trinidad.model.UploadedFile;
/**
* UploadedFileImpl defines a single file that has been uploaded
* to the server. UploadedFileImpl is not thread-safe.
*
* @version $Name: $ ($Revision: adfrt/faces/adf-faces-impl/src/main/java/oracle/adfinternal/view/faces/webapp/UploadedFileImpl.java#0 $) $Date: 10-nov-2005.18:49:03 $
*/
public class UploadedFileImpl extends ExtendedUploadedFile
{
UploadedFileImpl()
{
}
/**
* Returns the filename reported from the client.
*/
public String getFilename()
{
return _filename;
}
/**
* Returns the MIME type of the file.
*/
public String getContentType()
{
return _contentType;
}
/**
* Returns the total length (in bytes) of the file.
*/
public long getLength()
{
return _length;
}
public Object getOpaqueData()
{
return null;
}
@Override
public Map getProperties()
{
return _properties;
}
/**
* Returns an InputStream that can be used to read the file.
* This method can be called repeatedly.
*/
public InputStream getInputStream() throws IOException
{
if (_buffers != null)
{
return new BufferIS(_buffers, _sizeOfLastBuffer);
}
if (_file == null)
{
if (_tempFilename != null)
{
_file = new File(_tempFilename);
}
}
if (_file != null)
{
return new BufferedInputStream(new FileInputStream(_file));
}
return null;
}
/**
* Writes the entire contents of the file to an OutputStream.
*/
private void _writeFile(OutputStream out) throws IOException
{
if (_buffers != null)
{
int size = _buffers.size();
for (int i = 0; i < size; i++)
{
byte[] buffer = _buffers.get(i);
int bytes;
if (i == (size - 1))
bytes = _sizeOfLastBuffer;
else
bytes = buffer.length;
out.write(buffer, 0, bytes);
}
}
else if (_file != null)
{
FileInputStream in = new FileInputStream(_file);
try
{
byte[] buffer = new byte[_DISK_BUFFER_SIZE];
while (true)
{
int bytes = in.read(buffer);
if (bytes < 0)
break;
out.write(buffer, 0, bytes);
}
}
finally
{
in.close();
}
}
}
/**
* Disposes of all resources used to store this file.
*/
public void dispose()
{
_length = 0;
_buffers = null;
if (_file != null)
{
_file.delete();
_file = null;
}
}
/**
* Loads the file from a MultipartFormItem.
*/
public void loadFile(
UploadedFile file,
long remainingMemory,
long remainingDiskSpace,
String directory) throws IOException
{
_filename = file.getFilename();
_contentType = file.getContentType();
if (file instanceof ExtendedUploadedFile)
_properties = ((ExtendedUploadedFile) file).getProperties();
InputStream in = file.getInputStream();
// First step: try to read it into memory. Instead of trying
// to build up one big buffer, we create a list of buffers
// that are each at most 4K. This is vastly faster to create
// and insignificantly slower to read.
while (_length < remainingMemory)
{
// Read another block of buffered memory. We'll read up to 4K
// at a time, but no more than the remaining amount of memory plus
// one byte. Why plus one? Well, we want to know when we've
// exceeded the memory size - not just reached it.
int size = Math.min(((int) (remainingMemory - _length)) + 1,
_MEMORY_BUFFER_SIZE);
byte[] buffer = new byte[size];
int bytes = _fillBuffer(in, buffer, size);
_sizeOfLastBuffer = bytes;
_length = _length + bytes;
if (_buffers == null)
_buffers = new ArrayList();
_buffers.add(buffer);
// If we're done, bail right here.
if (bytes < size)
return;
}
// If we didn't have enough space to read the file into memory,
// and we also don't have enough space available on disk, then
// punt right here and now.
if (_length > remainingDiskSpace)
{
_buffers = null;
_length = 0;
throw new EOFException("Per-request disk space limits exceeded.");
}
OutputStream out = _createOutputStream(directory);
try
{
try
{
// First, copy the file from memory to the file
if (_length > 0)
{
_writeFile(out);
}
// Free the buffers, since we're
_buffers = null;
// Now, write directly to the file.
while (_length < remainingDiskSpace)
{
byte[] buffer = new byte[_DISK_BUFFER_SIZE];
int bytes = _fillBuffer(in, buffer, _DISK_BUFFER_SIZE);
out.write(buffer, 0, bytes);
_length = _length + bytes;
if (bytes < _DISK_BUFFER_SIZE)
break;
}
}
finally
{
out.close();
// If we read too much - then drop the file, and bail.
if (_length > remainingDiskSpace)
{
_file.delete();
_file = null;
_length = 0;
throw new EOFException("Per-request disk space limits exceeded.");
}
}
}
catch (Throwable e)
{
// If there were any errors make sure the file is cleaned up
if (_file != null)
_file.delete();
}
}
@Override
public int hashCode()
{
if (_filename == null)
return 0;
return _filename.hashCode();
}
@Override
public boolean equals(Object o)
{
// UploadedFiles are only equal to themselves.
return (this == o);
}
//
// Create the OutputStream.
//
private OutputStream _createOutputStream(String directory) throws IOException
{
File tempDir;
if (directory == null)
tempDir = null;
else
tempDir = new File(directory);
// Create our temporary file.
_file = File.createTempFile("uix", null, tempDir);
// HA does not replicate File objects so we store the path so it can
// be loaded in the replicated server. Replication is only done
// for file which are written to temp file. In-memory files are not
// replicated.
_tempFilename = _file.getCanonicalPath();
// Even though we're supposed to clean up these files ourselves,
// make sure they get deleted even if an exception terminates
// our code.
// Remove the deleteOnExit() call because in a HA environment
// a server can be shutdown and we don't want the file to be deleted
// until it's truly no longer needed.
//_file.deleteOnExit();
// No need for additional buffering of the output - we always
// buffer the writes - so _don't_ add a BufferedOutputStream.
return new FileOutputStream(_file);
}
//
// If true, we're either in memory (or empty); if false,
// we're on disk.
//
boolean __isInMemory()
{
return (_buffers != null);
}
//
// Fill a byte buffer from an InputStream.
//
private int _fillBuffer(
InputStream in,
byte[] buffer,
int size) throws IOException
{
int offset = 0;
// Read until the buffer is full, or the InputStream runs out.
do
{
int bytes = in.read(buffer, offset, size - offset);
if (bytes < 0)
break;
offset = offset + bytes;
}
while (offset < size);
return offset;
}
//
// InputStream implementation that reads from an in-memory buffer.
//
static private class BufferIS extends InputStream
{
public BufferIS(List bufferList, int sizeOfLastBuffer)
{
_bufferList = bufferList;
_sizeOfLastBuffer = sizeOfLastBuffer;
}
// Read a single byte.
@Override
public int read()
{
int bufferIndex = _bufferIndex;
if (bufferIndex < 0)
return -1;
byte[] buffer = _getBuffer(bufferIndex);
byte currByte = buffer[_bufferPos];
int bufferSize = _getBufferSize(buffer, bufferIndex);
_advanceTo(_bufferPos + 1, bufferSize);
return currByte & 0xff;
}
// Read into a buffer.
@Override
public int read(byte b[], int off, int len)
{
int bufferIndex = _bufferIndex;
if (bufferIndex < 0)
return -1;
byte[] buffer = _getBuffer(bufferIndex);
int bufferSize = _getBufferSize(buffer, bufferIndex);
// Read as many bytes are available in the current buffer block,
// up to "len" bytes. If "len" is larger, then in theory
// we could loop and read multiple blocks, but that'd complicate
// our logic here without actually simplifying the developer's
// logic any (they can't assume an InputStream behaves that way).
int bufferPos = _bufferPos;
int bytes = Math.min(len, bufferSize - bufferPos);
System.arraycopy(buffer, bufferPos, b, off, bytes);
_advanceTo(bufferPos + bytes, bufferSize);
return bytes;
}
//
// Returns the number of bytes that will be made avaialable
// in the next call to read(byte[], int, int);
//
@Override
public int available()
{
int bufferIndex = _bufferIndex;
if (bufferIndex < 0)
return 0;
byte[] buffer = _getBuffer(bufferIndex);
int bufferSize = _getBufferSize(buffer, bufferIndex);
return bufferSize - _bufferPos;
}
// Advance the current buffer to a certain position.
private void _advanceTo(int pos, int bufferSize)
{
if (pos >= bufferSize)
{
_bufferPos = 0;
_bufferIndex = _bufferIndex + 1;
if (_bufferIndex >= _bufferList.size())
_bufferIndex = -1;
}
else
{
_bufferPos = pos;
}
}
// Return the size of a given buffer. (The last buffer
// may not be completely full.)
private int _getBufferSize(byte[] buffer, int bufferIndex)
{
if (bufferIndex == _bufferList.size() - 1)
return _sizeOfLastBuffer;
return buffer.length;
}
// Return a buffer.
private byte[] _getBuffer(int bufferIndex)
{
return _bufferList.get(bufferIndex);
}
// ArrayList of all the buffers
private List _bufferList;
// The number of bytes in the last buffer (which may not be full)
private int _sizeOfLastBuffer;
// Current index into _bufferList
private int _bufferIndex;
// Current position in _bufferList[_bufferIndex]
private int _bufferPos;
}
private String _filename;
private String _tempFilename;
private String _contentType;
private Map _properties = new HashMap();
// Total length fo the content, whether in memory or on disk
transient private long _length;
// File where the content is stored (or null if the content
// is in memory)
transient private File _file;
// ArrayList of all the in-memory buffers (or null if it's in
// a File)
transient private ArrayList _buffers;
// The number of bytes in the last buffer (which may not be full)
transient private int _sizeOfLastBuffer;
// Buffer sizes. The memory buffer size can be very small
// because of our list-of-small-buffers technique; for disk I/O,
// use a larger buffer.
static private final int _MEMORY_BUFFER_SIZE = 2048;
static private final int _DISK_BUFFER_SIZE = 8192;
static private final long serialVersionUID = 3895593282556327786L;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy