Home | History | Annotate | Download | only in quickcontact
      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