/*
Copyright 2007 Creare Inc.
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 com.rbnb.utility;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.util.Enumeration;
import java.util.Hashtable;
/******************************************************************************
* Parse a URL into its components.
*
* Components in the URL include:
*
*
* @author John P. Wilson
*
* @version 09/28/2006
*/
/*
* Copyright 2006 Creare Inc.
* All Rights Reserved
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 09/28/2006 JPW RBNB servlet no longer supports special meaning for the
* "msg" munge. Therefore, I have removed "msg" as an
* RBNB munge. As part of this, I removed the method
* getCompleteMessageMunge() and the "message" member
* variable.
* 09/15/2006 JPW Switch over to using KeyValueHash to parse the munge
* 05/10/2006 JPW Created
*
*/
public class ParseURL {
// A copy of the original, full URL
private String url;
// Request protocol, such as "http" or "ftp"
private String protocol = null;
// The request string, stripped of any protocol and munge options
private String request = null;
// The original, complete munge string
private String munge = null;
// RBNB munge options
private boolean hasRBNBMunge = false; //EMF 5/12/06
private Double time = null;
private Double duration = null;
private String reference = null;
private String fetch = null;
private String byteorder = null;
private String datatype = null;
private Integer mux = null;
private Integer blocksize = null;
private String mime = null;
// Hashtable of non-RBNB munge options
private Hashtable nonRBNBMunge = null;
/**************************************************************************
* Default constructor.
*
*
* @author John P. Wilson
*
* @version 05/10/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/10/2006 JPW Created
*
*/
public ParseURL() {
this(null,false);
}
/**************************************************************************
* Constructor. Call parse() to parse the given URL into its components.
*
*
* @author John P. Wilson
*
* @param urlStrI The URL to parse.
* @param bDebugI Print debug?
*
* @version 05/10/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/10/2006 JPW Created
*
*/
public ParseURL(String urlStrI, boolean bDebugI) {
parse(urlStrI,bDebugI);
}
/**************************************************************************
* Parse the given URL into its components.
*
* This method parses a munged URL such as:
*
* http://jpw.creare.com:80/RBNB/TestSource?r=newest&t=1.5
*
* In this case:
* "http" is the protocol
* "jpw.creare.com:80/RBNB/TestSource" is the request
* "newest" is the reference
* "1.5" is the time
*
* @author John P. Wilson
*
* @param urlStrI The URL to parse.
* @param bDebugI Print debug?
*
* @version 09/15/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 09/15/2006 JPW Switch over to using KeyValueHash to parse the munge
* 05/10/2006 JPW Created
*
*/
public void parse(String urlStrI, boolean bDebugI) {
// Reset all member variables
resetMemberData();
if ( (urlStrI == null) || (urlStrI.trim().equals("")) ) {
if (bDebugI) {
System.err.println("Unable to parse URL: empty string");
}
return;
}
url = new String(urlStrI);
if (bDebugI) {
System.err.println("URL = \"" + urlStrI + "\"");
}
////////////////////////////////////////////
// See if a protocol is specified in the URL
////////////////////////////////////////////
int colonIndex = urlStrI.indexOf(':');
String requestAndMungeStr = urlStrI;
if (colonIndex >= 0) {
protocol = urlStrI.substring(0,colonIndex);
if ( (protocol != null) && (protocol.trim().equals("")) ) {
protocol = null;
}
if ( (protocol != null) && (bDebugI) ) {
System.err.println("Protocol = \"" + protocol + "\"");
} else if (bDebugI) {
System.err.println("Protocol = null");
}
requestAndMungeStr = urlStrI.substring(colonIndex + 1);
}
////////////////////////////////////////////////
// Strip off leading '/' from requestAndMungeStr
////////////////////////////////////////////////
while ( (requestAndMungeStr != null) &&
(!requestAndMungeStr.equals("")) &&
(requestAndMungeStr.charAt(0) == '/') )
{
requestAndMungeStr = requestAndMungeStr.substring(1);
}
///////////////////////////////////////////////////////////
// Separate the request from the munge; munge portion could
// begin with either '?' or '@'
///////////////////////////////////////////////////////////
request = null;
munge = null;
int mungeStartCharIndex = requestAndMungeStr.indexOf('@');
if (mungeStartCharIndex < 0) {
mungeStartCharIndex = requestAndMungeStr.indexOf('?');
}
if (mungeStartCharIndex >= 0) {
request =
requestAndMungeStr.substring(0,mungeStartCharIndex);
if ( (request != null) && (request.trim().equals("")) ) {
request = null;
}
munge =
requestAndMungeStr.substring(mungeStartCharIndex+1);
if ( (munge != null) && (munge.trim().equals("")) ) {
munge = null;
}
} else { //EMF 5/11/06: no munge, all request
request = requestAndMungeStr;
munge=null;
}
if (bDebugI) {
if (request == null) {
System.err.println("Request = null");
} else {
System.err.println("Request = \"" + request + "\"");
}
if (munge == null) {
System.err.println("Munge = null");
} else {
System.err.println("Munge = \"" + munge + "\"");
}
}
//////////////////////////
// Parse the munge options
//////////////////////////
if (munge == null) {
// We're all done
return;
}
// JPW 09/15/2006: Use KeyValueHash to parse the munge
char[] terminatorChars = {'&'};
KeyValueHash kvh = new KeyValueHash(munge,terminatorChars);
Hashtable fullHashtable = kvh.getHash();
if (kvh.get("time") != null) {
// Store time as a Double object
try {
time = new Double(kvh.get("time"));
if (bDebugI) {
System.err.println("time = " + time);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
time = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("time");
}
if (kvh.get("t") != null) {
// Store time as a Double object
try {
time = new Double(kvh.get("t"));
if (bDebugI) {
System.err.println("time = " + time);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
time = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("t");
}
if (kvh.get("duration") != null) {
// Store duration as a Double object
try {
duration = new Double(kvh.get("duration"));
if (bDebugI) {
System.err.println("duration = " + duration);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
duration = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("duration");
}
if (kvh.get("d") != null) {
// Store duration as a Double object
try {
duration = new Double(kvh.get("d"));
if (bDebugI) {
System.err.println("duration = " + duration);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
duration = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("d");
}
if (kvh.get("reference") != null) {
reference = kvh.get("reference");
if (bDebugI) {
System.err.println("reference = " + reference);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("reference");
}
if (kvh.get("r") != null) {
reference = kvh.get("r");
if (bDebugI) {
System.err.println("reference = " + reference);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("r");
}
if (kvh.get("fetch") != null) {
fetch = kvh.get("fetch");
if (bDebugI) {
System.err.println("fetch = " + fetch);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("fetch");
}
if (kvh.get("f") != null) {
fetch = kvh.get("f");
if (bDebugI) {
System.err.println("fetch = " + fetch);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("f");
}
if (kvh.get("byteorder") != null) {
byteorder = kvh.get("byteorder");
if (bDebugI) {
System.err.println("byteorder = " + byteorder);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("byteorder");
}
if (kvh.get("bo") != null) {
byteorder = kvh.get("bo");
if (bDebugI) {
System.err.println("byteorder = " + byteorder);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("bo");
}
if (kvh.get("datatype") != null) {
datatype = kvh.get("datatype");
if (bDebugI) {
System.err.println("datatype = " + datatype);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("datatype");
}
if (kvh.get("dt") != null) {
datatype = kvh.get("dt");
if (bDebugI) {
System.err.println("datatype = " + datatype);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("dt");
}
if (kvh.get("mux") != null) {
// Store mux as an Integer object
try {
mux = new Integer(kvh.get("mux"));
if (bDebugI) {
System.err.println("mux = " + mux);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
mux = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("mux");
}
if (kvh.get("x") != null) {
// Store mux as an Integer object
try {
mux = new Integer(kvh.get("x"));
if (bDebugI) {
System.err.println("mux = " + mux);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
mux = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("x");
}
if (kvh.get("blocksize") != null) {
// Store blocksize as an Integer object
try {
blocksize = new Integer(kvh.get("blocksize"));
if (bDebugI) {
System.err.println("blocksize = " + blocksize);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
blocksize = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("blocksize");
}
if (kvh.get("bs") != null) {
// Store blocksize as an Integer object
try {
blocksize = new Integer(kvh.get("bs"));
if (bDebugI) {
System.err.println("blocksize = " + blocksize);
}
hasRBNBMunge=true;
} catch (NumberFormatException e) {
// Nothing to do
blocksize = null;
}
// Remove this entry from the hashtable
fullHashtable.remove("bs");
}
if (kvh.get("mime") != null) {
mime = kvh.get("mime");
if (bDebugI) {
System.err.println("mime = " + mime);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("mime");
}
if (kvh.get("m") != null) {
mime = kvh.get("m");
if (bDebugI) {
System.err.println("mime = " + mime);
}
hasRBNBMunge=true;
// Remove this entry from the hashtable
fullHashtable.remove("m");
}
// What remains in fullHashtable must be the non-RBNB munges
if ( (fullHashtable != null) && (!fullHashtable.isEmpty()) ) {
nonRBNBMunge = fullHashtable;
if (bDebugI) {
for (Enumeration e=nonRBNBMunge.keys(); e.hasMoreElements();) {
String key = (String)e.nextElement();
String value = (String)nonRBNBMunge.get(key);
System.err.println(
"Non-RBNB munge: key = \"" +
key +
"\", value = \"" +
value +
"\"");
}
}
}
}
/**************************************************************************
* Reset member data to default values.
*
*
* @author John P. Wilson
*
* @version 05/10/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/10/2006 JPW Created
*
*/
public void resetMemberData() {
url = null;
protocol = null;
request = null;
munge = null;
time = null;
duration = null;
reference = null;
fetch = null;
byteorder = null;
datatype = null;
mux = null;
blocksize = null;
mime = null;
nonRBNBMunge = null;
}
/**************************************************************************
* Get the original URL as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getURL() {
return url;
}
/**************************************************************************
* Get URL protocol as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getProtocol() {
return protocol;
}
/**************************************************************************
* Get URL request as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getRequest() {
return request;
}
/**************************************************************************
* Get the URL munge as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getMunge() {
return munge;
}
/**************************************************************************
* Indicates whether any RBNB munge exists
*
*
* @author Eric M. Friets
*
* @version 05/12/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/12/2006 EMF Created
*
*/
public boolean isRBNBMunge() {
return hasRBNBMunge;
}
/**************************************************************************
* Get time munge option as a Double object
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public Double getTime() {
return time;
}
/**************************************************************************
* Get duration munge option as a Double object
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public Double getDuration() {
return duration;
}
/**************************************************************************
* Set a new duration for the munge.
*
*
* @author John P. Wilson
*
* @version 06/02/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 06/02/2006 JPW Created
*
*/
public void setDuration(double durationI) {
if (durationI >= 0.0) {
duration = new Double(durationI);
}
}
/**************************************************************************
* Get reference munge option as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getReference() {
return reference;
}
/**************************************************************************
* Get fetch munge option as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getFetch() {
return fetch;
}
/**************************************************************************
* Get byteorder munge option as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getByteorder() {
return byteorder;
}
/**************************************************************************
* Get datatype munge option as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getDatatype() {
return datatype;
}
/**************************************************************************
* Get mux munge option as an Integer object
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public Integer getMux() {
return mux;
}
/**************************************************************************
* Get blocksize munge option as an Integer object
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public Integer getBlocksize() {
return blocksize;
}
/**************************************************************************
* Get mime munge option as a String
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getMime() {
return mime;
}
/**************************************************************************
* Get Hashtable nonRBNBMunge
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public Hashtable getNonRBNBMunge() {
return nonRBNBMunge;
}
/**************************************************************************
* Get all non-RBNB munge options concatenated together, using '&' as
* the separator character.
*
*
* @author John P. Wilson
*
* @version 05/11/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 05/11/2006 JPW Created
*
*/
public String getNonRBNBMungeStr() {
if (nonRBNBMunge == null) {
return null;
}
StringBuffer mungeStrBuf = null;
for (Enumeration e = nonRBNBMunge.keys(); e.hasMoreElements(); ) {
String nextKey = (String)e.nextElement();
String nextValue = (String)nonRBNBMunge.get(nextKey);
if (mungeStrBuf == null) {
mungeStrBuf = new StringBuffer(nextKey + "=" + nextValue);
} else {
mungeStrBuf.append("&" + nextKey + "=" + nextValue);
}
}
return mungeStrBuf.toString();
}
/**************************************************************************
* Default version of this method; include all munge components.
*
*
* @author John P. Wilson
*
* @version 06/02/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 06/02/2006 JPW Created
*
*/
public String createNewRBNBMunge() {
return
createNewRBNBMunge(
false,
false,
false,
false,
false,
false,
false,
false,
false,
false);
}
/**************************************************************************
* Create a new munge string containing the desired RBNB munge components.
*
* The user can selectively remove munge components as specified by the
* boolean arguments. If the user has not chosen to remove any munge
* component, then the original, complete munge string is returned.
*
* @author John P. Wilson
*
* @param bRemoveTimeI Remove time from returned munge?
* @param bRemoveDurationI Remove duration from returned munge?
* @param bRemoveReferenceI Remove reference from returned munge?
* @param bRemoveFetchI Remove fetch from returned munge?
* @param bRemoveByteorderI Remove byteorder from returned munge?
* @param bRemoveDatatypeI Remove datatype from returned munge?
* @param bRemoveMuxI Remove mux from returned munge?
* @param bRemoveBlocksizeI Remove blocksize from returned munge?
* @param bRemoveMimeI Remove mime from returned munge?
* @param bRemoveNonRBNBMungeI Remove non RBNB munges from returned munge?
*
* @version 09/28/2006
*/
/*
*
* Date By Description
* MM/DD/YYYY
* ---------- -- -----------
* 09/28/2006 JPW Change "bRemoveMessageI" to "bRemoveNonRBNBMungeI"
* 06/02/2006 JPW Created
*
*/
public String createNewRBNBMunge(
boolean bRemoveTimeI,
boolean bRemoveDurationI,
boolean bRemoveReferenceI,
boolean bRemoveFetchI,
boolean bRemoveByteorderI,
boolean bRemoveDatatypeI,
boolean bRemoveMuxI,
boolean bRemoveBlocksizeI,
boolean bRemoveMimeI,
boolean bRemoveNonRBNBMungeI)
{
// JPW 09/28/2006: If user has not chosen to remove anything, then
// simply return the original, complete munge string.
if ( (!bRemoveTimeI) &&
(!bRemoveDurationI) &&
(!bRemoveReferenceI) &&
(!bRemoveFetchI) &&
(!bRemoveByteorderI) &&
(!bRemoveDatatypeI) &&
(!bRemoveMuxI) &&
(!bRemoveBlocksizeI) &&
(!bRemoveMimeI) &&
(!bRemoveNonRBNBMungeI) )
{
return munge;
}
StringBuffer newMunge = new StringBuffer();
if ( (getTime() != null) && (!bRemoveTimeI) ) {
newMunge.append("&t=" + getTime().toString());
}
if ( (getDuration() != null) && (!bRemoveDurationI) ) {
newMunge.append("&d=" + getDuration().toString());
}
if ( (getReference() != null) && (!bRemoveReferenceI) ) {
newMunge.append("&r=" + getReference());
}
if ( (getFetch() != null) && (!bRemoveFetchI) ) {
newMunge.append("&f=" + getFetch());
}
if ( (getByteorder() != null) && (!bRemoveByteorderI) ) {
newMunge.append("&bo=" + getByteorder());
}
if ( (getDatatype() != null) && (!bRemoveDatatypeI) ) {
newMunge.append("&dt=" + getDatatype());
}
if ( (getMux() != null) && (!bRemoveMuxI) ) {
newMunge.append("&x=" + getMux().toString());
}
if ( (getBlocksize() != null) && (!bRemoveBlocksizeI) ) {
newMunge.append("&bs=" + getBlocksize().toString());
}
if ( (getMime() != null) && (!bRemoveMimeI) ) {
newMunge.append("&m=" + getMime());
}
// JPW 09/28/2006: Return the non RBNB munges as a series of key/value
// pairs; don't return them encoded in the "msg"
// munge (RBNB servlet doesn't do anything special
// with the "msg" munge now).
if ( (getNonRBNBMungeStr() != null) && (!bRemoveNonRBNBMungeI) ) {
newMunge.append("&" + getNonRBNBMungeStr());
}
if (newMunge.length() == 0) {
return null;
}
// If there is a leading '&', remove it
if (newMunge.charAt(0) == '&') {
newMunge = newMunge.deleteCharAt(0);
}
return newMunge.toString();
}
}
class KeyValueHash {
Hashtable hash = new Hashtable();
public KeyValueHash(byte[] userData) {
// JPW 09/15/2006: These aren't used
//ByteArrayInputStream byteArray = new ByteArrayInputStream(userData);
//DataInputStream inStream = new DataInputStream(byteArray);
// JPW 09/15/2006: Add new constructor
this(new String(userData), null);
}
public KeyValueHash(String inString) {
this(inString,null);
}
// JPW 09/15/2006: Add new constructor that takes a String and an optional
// set of terminating characters
public KeyValueHash(String inString, char[] terminatorCharsI) {
// The default terminating characters
char[] terminatorChars = {',','\n','\r'};
if ( (terminatorCharsI != null) && (terminatorCharsI.length > 0) ) {
terminatorChars = terminatorCharsI;
}
int lt = -1; // location of the terminator from the previous iteration
int nt=nextTerminator(inString,lt,terminatorChars);
while (nt!=-1) {
int ne=inString.indexOf('=',lt+1); // next equals sign
if (ne>lt+1 && ne hash.put(inString.substring(lt+1,ne),inString.substring(ne+1,nt));
//System.out.println(inString.substring(lt+1,ne)+" "+inString.substring(ne+1,nt));
}
lt=nt;
while ((nt=nextTerminator(inString,lt,terminatorChars))==lt+1) //ignore multiple adjacent separators
lt=nt;
}
} //end KeyValueHash constructor
public String get(String key) {
// JPW 09/15/2006: Add some checks for null
if (key == null) {
return null;
}
Object value = hash.get(key);
if (value == null) {
return null;
}
return (String)value;
}
public Hashtable getHash() {
return hash;
}
private int nextTerminator(String s, int n, char[] t) {
// JPW 09/15/2006: The terminator characters are provided as a parameter
// char[] t = {',','\n','\r'}; //terminator characters
int min=-1;
boolean foundOne=false;
if (n>=s.length()) return -1; //at end of string
for (int i=0;i int j=s.indexOf(t[i],n+1);
if (j>=0) {
if (foundOne) {
if (j }
else {
foundOne=true;
min=j;
}
}
}
if (foundOne) return min;
else return s.length();
}
} //end class KeyValueHash