1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.contacts.quickcontact; 18 19 import static android.util.Patterns.GOOD_IRI_CHAR; 20 21 import java.util.Locale; 22 import java.util.regex.Matcher; 23 import java.util.regex.Pattern; 24 25 /** 26 * Below is a partial copy of {@link android.net.WebAddress}. The original author doesn't 27 * believe this API is suitable for making public. He recommends we copy it. 28 * 29 * Web Address Parser 30 * 31 * This is called WebAddress, rather than URL or URI, because it 32 * attempts to parse the stuff that a user will actually type into a 33 * browser address widget. 34 * 35 * Unlike java.net.uri, this parser will not choke on URIs missing 36 * schemes. It will only throw a ParseException if the input is 37 * really hosed. 38 * 39 * If given an https scheme but no port, fills in port 40 * 41 */ 42 public class WebAddress { 43 44 private String mScheme; 45 private String mHost; 46 private int mPort; 47 private String mPath; 48 private String mAuthInfo; 49 50 static final int MATCH_GROUP_SCHEME = 1; 51 static final int MATCH_GROUP_AUTHORITY = 2; 52 static final int MATCH_GROUP_HOST = 3; 53 static final int MATCH_GROUP_PORT = 4; 54 static final int MATCH_GROUP_PATH = 5; 55 56 static Pattern sAddressPattern = Pattern.compile( 57 /* scheme */ "(?:(http|https|file)\\:\\/\\/)?" + 58 /* authority */ "(?:([-A-Za-z0-9$_.+!*'(),;?&=]+(?:\\:[-A-Za-z0-9$_.+!*'(),;?&=]+)?)@)?" + 59 /* host */ "([" + GOOD_IRI_CHAR + "%_-][" + GOOD_IRI_CHAR + "%_\\.-]*|\\[[0-9a-fA-F:\\.]+\\])?" + 60 /* port */ "(?:\\:([0-9]*))?" + 61 /* path */ "(\\/?[^#]*)?" + 62 /* anchor */ ".*", Pattern.CASE_INSENSITIVE); 63 64 /** parses given uriString. */ 65 public WebAddress(String address) throws ParseException { 66 if (address == null) { 67 throw new NullPointerException(); 68 } 69 70 mScheme = ""; 71 mHost = ""; 72 mPort = -1; 73 mPath = "/"; 74 mAuthInfo = ""; 75 76 Matcher m = sAddressPattern.matcher(address); 77 String t; 78 if (m.matches()) { 79 t = m.group(MATCH_GROUP_SCHEME); 80 if (t != null) mScheme = t.toLowerCase(Locale.ROOT); 81 t = m.group(MATCH_GROUP_AUTHORITY); 82 if (t != null) mAuthInfo = t; 83 t = m.group(MATCH_GROUP_HOST); 84 if (t != null) mHost = t; 85 t = m.group(MATCH_GROUP_PORT); 86 if (t != null && t.length() > 0) { 87 // The ':' character is not returned by the regex. 88 try { 89 mPort = Integer.parseInt(t); 90 } catch (NumberFormatException ex) { 91 throw new ParseException("Bad port"); 92 } 93 } 94 t = m.group(MATCH_GROUP_PATH); 95 if (t != null && t.length() > 0) { 96 /* handle busted myspace frontpage redirect with 97 missing initial "/" */ 98 if (t.charAt(0) == '/') { 99 mPath = t; 100 } else { 101 mPath = "/" + t; 102 } 103 } 104 105 } else { 106 // nothing found... outa here 107 throw new ParseException("Bad address"); 108 } 109 110 /* Get port from scheme or scheme from port, if necessary and 111 possible */ 112 if (mPort == 443 && mScheme.equals("")) { 113 mScheme = "https"; 114 } else if (mPort == -1) { 115 if (mScheme.equals("https")) 116 mPort = 443; 117 else 118 mPort = 80; // default 119 } 120 if (mScheme.equals("")) mScheme = "http"; 121 } 122 123 @Override 124 public String toString() { 125 String port = ""; 126 if ((mPort != 443 && mScheme.equals("https")) || 127 (mPort != 80 && mScheme.equals("http"))) { 128 port = ":" + Integer.toString(mPort); 129 } 130 String authInfo = ""; 131 if (mAuthInfo.length() > 0) { 132 authInfo = mAuthInfo + "@"; 133 } 134 135 return mScheme + "://" + authInfo + mHost + port + mPath; 136 } 137 138 public class ParseException extends Exception { 139 public String response; 140 141 ParseException(String response) { 142 this.response = response; 143 } 144 } 145 } 146