/*
* Copyright (C) 2009 eXo Platform SAS.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.exoplatform.services.jcr.impl.dataflow.serialization;
import org.exoplatform.services.jcr.dataflow.serialization.ObjectReader;
import org.exoplatform.services.jcr.dataflow.serialization.SerializationConstants;
import org.exoplatform.services.jcr.dataflow.serialization.UnknownClassIdException;
import org.exoplatform.services.jcr.impl.dataflow.AbstractPersistedValueData;
import org.exoplatform.services.jcr.impl.dataflow.persistent.ByteArrayPersistedValueData;
import org.exoplatform.services.jcr.impl.dataflow.persistent.StreamPersistedValueData;
import org.exoplatform.services.jcr.impl.util.io.FileCleaner;
import org.exoplatform.services.jcr.impl.util.io.PrivilegedFileHelper;
import org.exoplatform.services.jcr.impl.util.io.SpoolFile;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
/**
* Created by The eXo Platform SAS. <br/>
* Date:
*
* @author <a href="karpenko.sergiy@gmail.com">Karpenko Sergiy</a>
* @version $Id: PersistedValueDataReader.java 2597 2010-06-15 09:42:31Z tolusha $
*/
public class PersistedValueDataReader
{
/**
* FileCleaner used to construct TransientValueData.
*/
private final FileCleaner fileCleaner;
/**
* MaxBufferSize used to construct TransientValueData.
*/
private final int maxBufferSize;
/**
* ReadedSpoolFile holder,
*/
private final ReaderSpoolFileHolder holder;
/**
* Constructor.
*
* @param fileCleaner
* FileCleaner
* @param maxBufferSize
* int
* @param holder
* ReaderSpoolFileHolder
*/
public PersistedValueDataReader(FileCleaner fileCleaner, int maxBufferSize, ReaderSpoolFileHolder holder)
{
this.fileCleaner = fileCleaner;
this.maxBufferSize = maxBufferSize;
this.holder = holder;
}
/**
* Read and set PersistedValueData object data.
*
* @param in
* ObjectReader.
* @return PersistedValueData object.
* @throws UnknownClassIdException
* If read Class ID is not expected or do not exist.
* @throws IOException
* If an I/O error has occurred.
*/
public AbstractPersistedValueData read(ObjectReader in) throws UnknownClassIdException, IOException
{
File tempDirectory = new File(SerializationConstants.TEMP_DIR);
PrivilegedFileHelper.mkdirs(tempDirectory);
// read id
int key;
if ((key = in.readInt()) != SerializationConstants.PERSISTED_VALUE_DATA)
throw new UnknownClassIdException("There is unexpected class [" + key + "]");
int orderNumber = in.readInt();
boolean isByteArray = in.readBoolean();
if (isByteArray)
{
byte[] data = new byte[in.readInt()];
in.readFully(data);
return new ByteArrayPersistedValueData(orderNumber, data);
}
else
{
// read file id - used for reread data optimization
String id = in.readString();
// read file length
long length = in.readLong();
SerializationSpoolFile sf = holder.get(id);
if (sf == null)
{
sf = new SerializationSpoolFile(tempDirectory, id, holder);
// TODO optimize writeToFile - use channels or streams
writeToFile(in, sf, length);
holder.put(id, sf);
return new StreamPersistedValueData(orderNumber, sf);
}
else
{
sf.acquire(this); // TODO workaround for AsyncReplication test
try
{
AbstractPersistedValueData vd = new StreamPersistedValueData(orderNumber, sf);
// skip data in input stream
if (in.skip(length) != length)
throw new IOException("Content isn't skipped correctly.");
return vd;
}
finally
{
sf.release(this);
}
}
}
}
private void writeToFile(ObjectReader src, SpoolFile dest, long length) throws IOException
{
// write data to file
FileOutputStream sfout = PrivilegedFileHelper.fileOutputStream(dest);
int bSize = SerializationConstants.INTERNAL_BUFFER_SIZE;
try
{
byte[] buff = new byte[bSize];
for (; length >= bSize; length -= bSize)
{
src.readFully(buff);
sfout.write(buff);
}
if (length > 0)
{
buff = new byte[(int)length];
src.readFully(buff);
sfout.write(buff);
}
}
finally
{
sfout.close();
}
}
}