Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit bc75177c authored by Brad Ebinger's avatar Brad Ebinger Committed by Automerger Merge Worker
Browse files

Merge "Add the ability to get Accept-Contact header params" am: 71a05ba2 am:...

Merge "Add the ability to get Accept-Contact header params" am: 71a05ba2 am: 0c5b3222 am: 1c5ed5e3

Original change: https://android-review.googlesource.com/c/platform/frameworks/base/+/1690090

Change-Id: Ibb8e787a5acdc66049e175954376141efab9e75d
parents 6e729ee3 1c5ed5e3
Loading
Loading
Loading
Loading
+156 −37
Original line number Original line Diff line number Diff line
@@ -17,6 +17,7 @@
package com.android.internal.telephony;
package com.android.internal.telephony;


import android.net.Uri;
import android.net.Uri;
import android.util.ArraySet;
import android.util.Log;
import android.util.Log;
import android.util.Pair;
import android.util.Pair;


@@ -24,6 +25,8 @@ import java.util.ArrayList;
import java.util.Arrays;
import java.util.Arrays;
import java.util.Collections;
import java.util.Collections;
import java.util.List;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;


/**
/**
 * Utility methods for parsing parts of {@link android.telephony.ims.SipMessage}s.
 * Utility methods for parsing parts of {@link android.telephony.ims.SipMessage}s.
@@ -70,6 +73,24 @@ public class SipMessageParsingUtils {
    // compact form of the call-id header key
    // compact form of the call-id header key
    private static final String CALL_ID_SIP_HEADER_KEY_COMPACT = "i";
    private static final String CALL_ID_SIP_HEADER_KEY_COMPACT = "i";


    // from header key
    private static final String FROM_HEADER_KEY = "from";
    // compact form of the from header key
    private static final String FROM_HEADER_KEY_COMPACT = "f";

    // to header key
    private static final String TO_HEADER_KEY = "to";
    // compact form of the to header key
    private static final String TO_HEADER_KEY_COMPACT = "t";

    // The tag parameter found in both the from and to headers
    private static final String TAG_PARAM_KEY = "tag";

    // accept-contact header key
    private static final String ACCEPT_CONTACT_HEADER_KEY = "accept-contact";
    // compact form of the accept-contact header key
    private static final String ACCEPT_CONTACT_HEADER_KEY_COMPACT = "a";

    /**
    /**
     * @return true if the SIP message start line is considered a request (based on known request
     * @return true if the SIP message start line is considered a request (based on known request
     * methods).
     * methods).
@@ -104,13 +125,25 @@ public class SipMessageParsingUtils {
            // branch param YY1.
            // branch param YY1.
            String[] subHeaders = header.second.split(SUBHEADER_VALUE_SEPARATOR);
            String[] subHeaders = header.second.split(SUBHEADER_VALUE_SEPARATOR);
            for (String subHeader : subHeaders) {
            for (String subHeader : subHeaders) {
                // Search for ;branch=z9hG4bKXXXXXX and return parameter value
                String paramValue = getParameterValue(subHeader, BRANCH_PARAM_KEY);
                String[] params = subHeader.split(PARAM_SEPARATOR);
                if (paramValue == null) continue;
                return paramValue;
            }
        }
        return null;
    }

    /**
     * Search a header's value for a specific parameter.
     * @param headerValue The header key's value.
     * @param parameterKey The parameter key we are looking for.
     * @return The value associated with the specified parameter key or {@link null} if that key is
     * not found.
     */
    private static String getParameterValue(String headerValue, String parameterKey) {
        String[] params = headerValue.split(PARAM_SEPARATOR);
        if (params.length < 2) {
        if (params.length < 2) {
                    // This param doesn't include a branch param, move to next param.
            return null;
                    Log.w(TAG, "getTransactionId: via detected without branch param:"
                            + subHeader);
                    continue;
        }
        }
        // by spec, each param can only appear once in a header.
        // by spec, each param can only appear once in a header.
        for (String param : params) {
        for (String param : params) {
@@ -121,20 +154,15 @@ public class SipMessageParsingUtils {
            }
            }
            if (pair.length > 2) {
            if (pair.length > 2) {
                Log.w(TAG,
                Log.w(TAG,
                                "getTransactionId: unexpected parameter" + Arrays.toString(pair));
                        "getParameterValue: unexpected parameter" + Arrays.toString(pair));
            }
            }
            // Trim whitespace in parameter
            // Trim whitespace in parameter
            pair[0] = pair[0].trim();
            pair[0] = pair[0].trim();
            pair[1] = pair[1].trim();
            pair[1] = pair[1].trim();
                    if (BRANCH_PARAM_KEY.equalsIgnoreCase(pair[0])) {
            if (parameterKey.equalsIgnoreCase(pair[0])) {
                        // There can be multiple "Via" headers in the SIP message, however we want
                        // to return the first once found, as this corresponds with the transaction
                        // that is relevant here.
                return pair[1];
                return pair[1];
            }
            }
        }
        }
            }
        }
        return null;
        return null;
    }
    }


@@ -143,23 +171,105 @@ public class SipMessageParsingUtils {
     * @param headerString The string containing the headers of the SIP message.
     * @param headerString The string containing the headers of the SIP message.
     */
     */
    public static String getCallId(String headerString) {
    public static String getCallId(String headerString) {
        // search for the call-Id header, there should only be one in the header.
        // search for the call-Id header, there should only be one in the headers.
        List<Pair<String, String>> headers = parseHeaders(headerString, true,
        List<Pair<String, String>> headers = parseHeaders(headerString, true,
                CALL_ID_SIP_HEADER_KEY, CALL_ID_SIP_HEADER_KEY_COMPACT);
                CALL_ID_SIP_HEADER_KEY, CALL_ID_SIP_HEADER_KEY_COMPACT);
        return !headers.isEmpty() ? headers.get(0).second : null;
        return !headers.isEmpty() ? headers.get(0).second : null;
    }
    }


    /**
     * @return Return the from header's tag parameter or {@code null} if it doesn't exist.
     */
    public static String getFromTag(String headerString) {
        // search for the from header, there should only be one in the headers.
        List<Pair<String, String>> headers = parseHeaders(headerString, true,
                FROM_HEADER_KEY, FROM_HEADER_KEY_COMPACT);
        if (headers.isEmpty()) {
            return null;
        }
        // There should only be one from header in the SIP message
        return getParameterValue(headers.get(0).second, TAG_PARAM_KEY);
    }

    /**
     * @return Return the to header's tag parameter or {@code null} if it doesn't exist.
     */
    public static String getToTag(String headerString) {
        // search for the to header, there should only be one in the headers.
        List<Pair<String, String>> headers = parseHeaders(headerString, true,
                TO_HEADER_KEY, TO_HEADER_KEY_COMPACT);
        if (headers.isEmpty()) {
            return null;
        }
        // There should only be one from header in the SIP message
        return getParameterValue(headers.get(0).second, TAG_PARAM_KEY);
    }

    /**
    /**
     * Validate that the start line is correct and split into its three segments.
     * Validate that the start line is correct and split into its three segments.
     * @param startLine The start line to verify and split.
     * @param startLine The start line to verify and split.
     * @return The split start line, which will always have three segments.
     * @return The split start line, which will always have three segments.
     */
     */
    public static String[] splitStartLineAndVerify(String startLine) {
    public static String[] splitStartLineAndVerify(String startLine) {
        String[] splitLine = startLine.split(" ");
        String[] splitLine = startLine.split(" ", 3);
        if (isStartLineMalformed(splitLine)) return null;
        if (isStartLineMalformed(splitLine)) return null;
        return splitLine;
        return splitLine;
    }
    }



    /**
     * @return All feature tags starting with "+" in the Accept-Contact header.
     */
    public static Set<String> getAcceptContactFeatureTags(String headerString) {
        List<Pair<String, String>> headers = SipMessageParsingUtils.parseHeaders(headerString,
                false, ACCEPT_CONTACT_HEADER_KEY, ACCEPT_CONTACT_HEADER_KEY_COMPACT);
        if (headerString.isEmpty()) {
            return Collections.emptySet();
        }
        Set<String> featureTags = new ArraySet<>();
        for (Pair<String, String> header : headers) {
            String[] splitParams = header.second.split(PARAM_SEPARATOR);
            if (splitParams.length < 2) {
                continue;
            }
            // Start at 1 here, since the first entry is the header value and not params.
            // We only care about IMS feature tags here, so filter tags with a "+"
            Set<String> fts = Arrays.asList(splitParams).subList(1, splitParams.length).stream()
                    .map(String::trim).filter(p -> p.startsWith("+")).collect(Collectors.toSet());
            for (String ft : fts) {
                String[] paramKeyValue = ft.split(PARAM_KEY_VALUE_SEPARATOR, 2);
                if (paramKeyValue.length < 2) {
                    featureTags.add(ft);
                    continue;
                }
                // Splits keys like +a="b,c" into +a="b" and +a="c"
                String[] splitValue = splitParamValue(paramKeyValue[1]);
                for (String value : splitValue) {
                    featureTags.add(paramKeyValue[0] + PARAM_KEY_VALUE_SEPARATOR + value);
                }
            }
        }
        return featureTags;
    }

    /**
     * Takes a string such as "\"a,b,c,d\"" and splits it by "," into a String array of
     * [\"a\", \"b\", \"c\", \"d\"]
     */
    private static String[] splitParamValue(String paramValue) {
        if (!paramValue.startsWith("\"") && !paramValue.endsWith("\"")) {
            return new String[] {paramValue};
        }
        // Remove quotes on outside
        paramValue = paramValue.substring(1, paramValue.length() - 1);
        String[] splitValues = paramValue.split(",");
        for (int i = 0; i < splitValues.length; i++) {
            // Encapsulate each split value in its own quotations.
            splitValues[i] = "\"" + splitValues[i] + "\"";
        }
        return splitValues;
    }

    private static boolean isStartLineMalformed(String[] startLine) {
    private static boolean isStartLineMalformed(String[] startLine) {
        if (startLine == null || startLine.length == 0)  {
        if (startLine == null || startLine.length == 0)  {
            return true;
            return true;
@@ -172,18 +282,27 @@ public class SipMessageParsingUtils {


    private static boolean verifySipRequest(String[] request) {
    private static boolean verifySipRequest(String[] request) {
        // Request-Line  =  Method SP Request-URI SP SIP-Version CRLF
        // Request-Line  =  Method SP Request-URI SP SIP-Version CRLF
        boolean verified = request[2].contains(SIP_VERSION_2);
        if (!request[2].contains(SIP_VERSION_2)) return false;
        verified &= (Uri.parse(request[1]).getScheme() != null);
        boolean verified;
        try {
            verified = (Uri.parse(request[1]).getScheme() != null);
        } catch (NumberFormatException e) {
            return false;
        }
        verified &= Arrays.stream(SIP_REQUEST_METHODS).anyMatch(s -> request[0].contains(s));
        verified &= Arrays.stream(SIP_REQUEST_METHODS).anyMatch(s -> request[0].contains(s));
        return verified;
        return verified;
    }
    }


    private static boolean verifySipResponse(String[] response) {
    private static boolean verifySipResponse(String[] response) {
        // Status-Line = SIP-Version SP Status-Code SP Reason-Phrase CRLF
        // Status-Line = SIP-Version SP Status-Code SP Reason-Phrase CRLF
        boolean verified = response[0].contains(SIP_VERSION_2);
        if (!response[0].contains(SIP_VERSION_2)) return false;
        int statusCode = Integer.parseInt(response[1]);
        int statusCode;
        verified &= (statusCode >= 100  && statusCode < 700);
        try {
        return verified;
            statusCode = Integer.parseInt(response[1]);
        } catch (NumberFormatException e) {
            return false;
        }
        return (statusCode >= 100  && statusCode < 700);
    }
    }


    /**
    /**
+1 −1
Original line number Original line Diff line number Diff line
@@ -103,7 +103,7 @@ public class SipDelegateManager {


    /**
    /**
     * The feature tag associated with the outgoing message does not match any known feature tags
     * The feature tag associated with the outgoing message does not match any known feature tags
     * and this message can not be sent.
     * or it matches a denied tag and this message can not be sent.
     */
     */
    public static final int MESSAGE_FAILURE_REASON_INVALID_FEATURE_TAG = 6;
    public static final int MESSAGE_FAILURE_REASON_INVALID_FEATURE_TAG = 6;