net.lightbody.bmp.proxy.jetty.http.InclusiveByteRange.java Source code

Java tutorial

Introduction

Here is the source code for net.lightbody.bmp.proxy.jetty.http.InclusiveByteRange.java

Source

// ========================================================================
// $Id: InclusiveByteRange.java,v 1.11 2005/08/13 00:01:24 gregwilkins Exp $
// Copyright 2002-2004 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// Licensed 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 net.lightbody.bmp.proxy.jetty.http;

import net.lightbody.bmp.proxy.jetty.log.LogFactory;
import net.lightbody.bmp.proxy.jetty.util.LazyList;
import net.lightbody.bmp.proxy.jetty.util.LogSupport;
import org.apache.commons.logging.Log;

import java.util.Enumeration;
import java.util.List;
import java.util.StringTokenizer;

/* ------------------------------------------------------------ */
/** Byte range inclusive of end points.
 * <PRE>
 * 
 *   parses the following types of byte ranges:
 * 
 *       bytes=100-499
 *       bytes=-300
 *       bytes=100-
 *       bytes=1-2,2-3,6-,-2
 *
 *   given an entity length, converts range to string
 * 
 *       bytes 100-499/500
 * 
 * </PRE>
 * 
 * Based on RFC2616 3.12, 14.16, 14.35.1, 14.35.2
 * @version $version$
 * @author Helmut Hissen
 */
public class InclusiveByteRange {
    private static Log log = LogFactory.getLog(InclusiveByteRange.class);

    long first = 0;
    long last = 0;

    public InclusiveByteRange(long first, long last) {
        this.first = first;
        this.last = last;
    }

    public long getFirst() {
        return first;
    }

    public long getLast() {
        return last;
    }

    /* ------------------------------------------------------------ */
    /** 
     * @param headers Enumeration of Range header fields.
     * @param size Size of the resource.
     * @return LazyList of satisfiable ranges
     */
    public static List satisfiableRanges(Enumeration headers, long size) {
        Object satRanges = null;

        // walk through all Range headers
        headers: while (headers.hasMoreElements()) {
            String header = (String) headers.nextElement();
            StringTokenizer tok = new StringTokenizer(header, "=,", false);
            String t = null;
            try {
                // read all byte ranges for this header 
                while (tok.hasMoreTokens()) {
                    t = tok.nextToken().trim();

                    long first = -1;
                    long last = -1;
                    int d = t.indexOf('-');
                    if (d < 0 || t.indexOf("-", d + 1) >= 0) {
                        if ("bytes".equals(t))
                            continue;
                        log.warn("Bad range format: " + t);
                        continue headers;
                    } else if (d == 0) {
                        if (d + 1 < t.length())
                            last = Long.parseLong(t.substring(d + 1).trim());
                        else {
                            log.warn("Bad range format: " + t);
                            continue headers;
                        }
                    } else if (d + 1 < t.length()) {
                        first = Long.parseLong(t.substring(0, d).trim());
                        last = Long.parseLong(t.substring(d + 1).trim());
                    } else
                        first = Long.parseLong(t.substring(0, d).trim());

                    if (first == -1 && last == -1)
                        continue headers;

                    if (first != -1 && last != -1 && (first > last))
                        continue headers;

                    if (first < size) {
                        InclusiveByteRange range = new InclusiveByteRange(first, last);
                        satRanges = LazyList.add(satRanges, range);
                    }
                }
            } catch (Exception e) {
                log.warn("Bad range format: " + t);
                LogSupport.ignore(log, e);
            }
        }
        return LazyList.getList(satRanges, true);
    }

    /* ------------------------------------------------------------ */
    public long getFirst(long size) {
        if (first < 0) {
            long tf = size - last;
            if (tf < 0)
                tf = 0;
            return tf;
        }
        return first;
    }

    /* ------------------------------------------------------------ */
    public long getLast(long size) {
        if (first < 0)
            return size - 1;

        if (last < 0 || last >= size)
            return size - 1;
        return last;
    }

    /* ------------------------------------------------------------ */
    public long getSize(long size) {
        return getLast(size) - getFirst(size) + 1;
    }

    /* ------------------------------------------------------------ */
    public String toHeaderRangeString(long size) {
        StringBuffer sb = new StringBuffer(40);
        sb.append("bytes ");
        sb.append(getFirst(size));
        sb.append('-');
        sb.append(getLast(size));
        sb.append("/");
        sb.append(size);
        return sb.toString();
    }

    /* ------------------------------------------------------------ */
    public static String to416HeaderRangeString(long size) {
        StringBuffer sb = new StringBuffer(40);
        sb.append("bytes */");
        sb.append(size);
        return sb.toString();
    }

    /* ------------------------------------------------------------ */
    public String toString() {
        StringBuffer sb = new StringBuffer(60);
        sb.append(Long.toString(first));
        sb.append(":");
        sb.append(Long.toString(last));
        return sb.toString();
    }

}