package org.openslx.filetransfer.util; import java.util.Iterator; import java.util.List; import org.openslx.filetransfer.FileRange; public class FileChunk { /** * Length in bytes of binary sha1 representation */ public static final int SHA1_LENGTH = 20; public static final int CHUNK_SIZE_MIB = 16; public static final int CHUNK_SIZE = CHUNK_SIZE_MIB * ( 1024 * 1024 ); public final FileRange range; private int failCount = 0; protected byte[] sha1sum; protected ChunkStatus status = ChunkStatus.MISSING; private boolean writtenToDisk = false; public FileChunk( long startOffset, long endOffset, byte[] sha1sum ) { this.range = new FileRange( startOffset, endOffset ); if ( sha1sum == null || sha1sum.length != SHA1_LENGTH ) { this.sha1sum = null; } else { this.sha1sum = sha1sum; } } synchronized void setSha1Sum( byte[] sha1sum ) { if ( this.sha1sum != null || sha1sum == null || sha1sum.length != SHA1_LENGTH ) return; this.sha1sum = sha1sum; if ( this.status == ChunkStatus.COMPLETE ) { this.status = ChunkStatus.HASHING; } } /** * Signal that transferring this chunk seems to have failed (checksum * mismatch). * * @return Number of times the transfer failed now */ synchronized int incFailed() { return ++failCount; } public int getChunkIndex() { return (int) ( range.startOffset / CHUNK_SIZE ); } @Override public String toString() { return "[Chunk " + getChunkIndex() + " (" + status + "), fails: " + failCount + "]"; } public synchronized byte[] getSha1Sum() { return sha1sum; } public synchronized ChunkStatus getStatus() { return status; } /** * Whether the chunk of data this chunk refers to has been written to * disk and is assumed to be valid/up to date. */ public synchronized boolean isWrittenToDisk() { return writtenToDisk; } synchronized void setStatus( ChunkStatus status ) { if ( status != null ) { if ( status == ChunkStatus.COMPLETE ) { this.writtenToDisk = true; } else if ( status == ChunkStatus.MISSING ) { this.writtenToDisk = false; } this.status = status; } } // public static int fileSizeToChunkCount( long fileSize ) { return (int) ( ( fileSize + CHUNK_SIZE - 1 ) / CHUNK_SIZE ); } public static void createChunkList( List list, long fileSize, List sha1Sums ) { if ( fileSize < 0 ) throw new IllegalArgumentException( "fileSize cannot be negative" ); if ( !list.isEmpty() ) throw new IllegalArgumentException( "Passed list is not empty" ); long offset = 0; Iterator hashIt = null; if ( sha1Sums != null ) { hashIt = sha1Sums.iterator(); } while ( offset < fileSize ) { long end = offset + CHUNK_SIZE; if ( end > fileSize ) end = fileSize; byte[] hash = null; if ( hashIt != null && hashIt.hasNext() ) { hash = hashIt.next(); } list.add( new FileChunk( offset, end, hash ) ); offset = end; } } public int getFailCount() { return failCount; } }