// ======================================================================== // $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(); } }