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

io.milton.zsync.DataRange 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 io.milton.zsync;

import io.milton.common.BufferingOutputStream;
import io.milton.common.RangeUtils;
import io.milton.http.Range;
import java.io.IOException;
import java.io.InputStream;
import java.io.RandomAccessFile;


/**
 * An object consisting of a Range and a stream of bytes. If the number of bytes
 * is beyond a certain threshold, the data is stored in a temporary file;
 * otherwise, it is just stored in an array.
 *
 * @author Nick
 *
 */
public class DataRange {

    /**
     * Once more than
     * threshold bytes are written to data, the backing store is
     * switched from a byte array to a temporary file
     */
    public static final int threshold = 1024 * 1024;
    private final Range range;
    private final BufferingOutputStream data;

    /**
     * Sets the range equal to the specified Range and pulls a number of bytes
     * from the InputStream equal to the size of the range
     *
     * @param range A Range object specifying the target location of the byte
     * stream
     * @param instream A stream from which the data portion will be pulled
     * @throws IOException
     */
    public DataRange(Range range, InputStream instream) throws IOException {

        this.range = range;
        this.data = new BufferingOutputStream(threshold);

        long length = range.getFinish() - range.getStart();
        RangeUtils.sendBytes(instream, data, length);

        this.data.close();

    }

    /**
     * Sets the range to the specified Range and seeks to the beginning of the
     * range in the RandomAccessFile before copying bytes
     *
     * @param range
     * @param randAccess
     * @throws IOException
     */
    public DataRange(Range range, RandomAccessFile randAccess) throws IOException {

        this.range = range;
        this.data = new BufferingOutputStream(threshold);

        randAccess.seek(range.getStart());
        long length = range.getFinish() - range.getStart();

        byte[] buffer = new byte[2048];
        int bytesLeft = (int) length;
        int bytesRead = Math.min(bytesLeft, buffer.length);

        while (bytesLeft > 0) {
            randAccess.read(buffer, 0, bytesRead);
            data.write(buffer, 0, bytesRead);
            bytesLeft -= bytesRead;
            bytesRead = Math.min(bytesLeft, buffer.length);
        }

        data.close();

    }

    @Override
    public String toString() {
        return range.toString();
    }

    public Range getRange() {

        return range;
    }

    /**
     * Returns the stream of bytes. If a temporary file is used to store data,
     * it will be deleted when the returned stream is closed.
     *
     * @return The byte stream corresponding to
     * range
     */
    public InputStream getInputStream() {

        return data.getInputStream();
    }

    /**
     * Returns the length of the Range (and the number of bytes in the byte
     * stream)
     *
     * @return
     */
    public long getLength() {

        return range.getFinish() - range.getStart();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy