1 /*
2  * Copyright (C) 2017 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 android.content.pm;
18 
19 import android.annotation.NonNull;
20 import android.annotation.Nullable;
21 import android.annotation.SystemApi;
22 import android.content.Intent;
23 import android.os.Bundle;
24 import android.os.Parcel;
25 import android.os.Parcelable;
26 
27 import java.security.MessageDigest;
28 import java.security.NoSuchAlgorithmException;
29 import java.security.SecureRandom;
30 import java.util.ArrayList;
31 import java.util.Arrays;
32 import java.util.Collections;
33 import java.util.List;
34 import java.util.Locale;
35 import java.util.Random;
36 
37 /**
38  * Describes an externally resolvable instant application. There are three states that this class
39  * can represent: <p/>
40  * <ul>
41  *     <li>
42  *         The first, usable only for non http/s intents, implies that the resolver cannot
43  *         immediately resolve this intent and would prefer that resolution be deferred to the
44  *         instant app installer. Represent this state with {@link #InstantAppResolveInfo(Bundle)}.
45  *         If the {@link android.content.Intent} has the scheme set to http/s and a set of digest
46  *         prefixes were passed into one of the resolve methods in
47  *         {@link android.app.InstantAppResolverService}, this state cannot be used.
48  *     </li>
49  *     <li>
50  *         The second represents a partial match and is constructed with any of the other
51  *         constructors. By setting one or more of the {@link Nullable}arguments to null, you
52  *         communicate to the resolver in response to
53  *         {@link android.app.InstantAppResolverService#onGetInstantAppResolveInfo(Intent, int[],
54  *                String, InstantAppResolverService.InstantAppResolutionCallback)}
55  *         that you need a 2nd round of resolution to complete the request.
56  *     </li>
57  *     <li>
58  *         The third represents a complete match and is constructed with all @Nullable parameters
59  *         populated.
60  *     </li>
61  * </ul>
62  * @hide
63  */
64 @SystemApi
65 public final class InstantAppResolveInfo implements Parcelable {
66     /** Algorithm that will be used to generate the domain digest */
67     private static final String SHA_ALGORITHM = "SHA-256";
68 
69     private static final byte[] EMPTY_DIGEST = new byte[0];
70 
71     private final InstantAppDigest mDigest;
72     private final String mPackageName;
73     /** The filters used to match domain */
74     private final List<InstantAppIntentFilter> mFilters;
75     /** The version code of the app that this class resolves to */
76     private final long mVersionCode;
77     /** Data about the app that should be passed along to the Instant App installer on resolve */
78     private final Bundle mExtras;
79     /**
80      * A flag that indicates that the resolver is aware that an app may match, but would prefer
81      * that the installer get the sanitized intent to decide.
82      */
83     private final boolean mShouldLetInstallerDecide;
84 
85     /** Constructor for intent-based InstantApp resolution results. */
InstantAppResolveInfo(@onNull InstantAppDigest digest, @Nullable String packageName, @Nullable List<InstantAppIntentFilter> filters, int versionCode)86     public InstantAppResolveInfo(@NonNull InstantAppDigest digest, @Nullable String packageName,
87             @Nullable List<InstantAppIntentFilter> filters, int versionCode) {
88         this(digest, packageName, filters, (long) versionCode, null /* extras */);
89     }
90 
91     /** Constructor for intent-based InstantApp resolution results with extras. */
InstantAppResolveInfo(@onNull InstantAppDigest digest, @Nullable String packageName, @Nullable List<InstantAppIntentFilter> filters, long versionCode, @Nullable Bundle extras)92     public InstantAppResolveInfo(@NonNull InstantAppDigest digest, @Nullable String packageName,
93             @Nullable List<InstantAppIntentFilter> filters, long versionCode,
94             @Nullable Bundle extras) {
95         this(digest, packageName, filters, versionCode, extras, false);
96     }
97 
98     /** Constructor for intent-based InstantApp resolution results by hostname. */
InstantAppResolveInfo(@onNull String hostName, @Nullable String packageName, @Nullable List<InstantAppIntentFilter> filters)99     public InstantAppResolveInfo(@NonNull String hostName, @Nullable String packageName,
100             @Nullable List<InstantAppIntentFilter> filters) {
101         this(new InstantAppDigest(hostName), packageName, filters, -1 /*versionCode*/,
102                 null /* extras */);
103     }
104 
105     /**
106      * Constructor that indicates that resolution could be delegated to the installer when the
107      * sanitized intent contains enough information to resolve completely.
108      */
InstantAppResolveInfo(@ullable Bundle extras)109     public InstantAppResolveInfo(@Nullable Bundle extras) {
110         this(InstantAppDigest.UNDEFINED, null, null, -1, extras, true);
111     }
112 
InstantAppResolveInfo(@onNull InstantAppDigest digest, @Nullable String packageName, @Nullable List<InstantAppIntentFilter> filters, long versionCode, @Nullable Bundle extras, boolean shouldLetInstallerDecide)113     private InstantAppResolveInfo(@NonNull InstantAppDigest digest, @Nullable String packageName,
114             @Nullable List<InstantAppIntentFilter> filters, long versionCode,
115             @Nullable Bundle extras, boolean shouldLetInstallerDecide) {
116         // validate arguments
117         if ((packageName == null && (filters != null && filters.size() != 0))
118                 || (packageName != null && (filters == null || filters.size() == 0))) {
119             throw new IllegalArgumentException();
120         }
121         mDigest = digest;
122         if (filters != null) {
123             mFilters = new ArrayList<>(filters.size());
124             mFilters.addAll(filters);
125         } else {
126             mFilters = null;
127         }
128         mPackageName = packageName;
129         mVersionCode = versionCode;
130         mExtras = extras;
131         mShouldLetInstallerDecide = shouldLetInstallerDecide;
132     }
133 
InstantAppResolveInfo(Parcel in)134     InstantAppResolveInfo(Parcel in) {
135         mShouldLetInstallerDecide = in.readBoolean();
136         mExtras = in.readBundle();
137         if (mShouldLetInstallerDecide) {
138             mDigest = InstantAppDigest.UNDEFINED;
139             mPackageName = null;
140             mFilters = Collections.emptyList();
141             mVersionCode = -1;
142         } else {
143             mDigest = in.readParcelable(null /*loader*/, android.content.pm.InstantAppResolveInfo.InstantAppDigest.class);
144             mPackageName = in.readString();
145             mFilters = new ArrayList<>();
146             in.readTypedList(mFilters, InstantAppIntentFilter.CREATOR);
147             mVersionCode = in.readLong();
148         }
149     }
150 
151     /**
152      * Returns true if the resolver is aware that an app may match, but would prefer
153      * that the installer get the sanitized intent to decide. This should not be true for
154      * resolutions that include a host and will be ignored in such cases.
155      */
shouldLetInstallerDecide()156     public boolean shouldLetInstallerDecide() {
157         return mShouldLetInstallerDecide;
158     }
159 
getDigestBytes()160     public byte[] getDigestBytes() {
161         return mDigest.mDigestBytes.length > 0 ? mDigest.getDigestBytes()[0] : EMPTY_DIGEST;
162     }
163 
getDigestPrefix()164     public int getDigestPrefix() {
165         return mDigest.getDigestPrefix()[0];
166     }
167 
getPackageName()168     public String getPackageName() {
169         return mPackageName;
170     }
171 
getIntentFilters()172     public List<InstantAppIntentFilter> getIntentFilters() {
173         return mFilters;
174     }
175 
176     /**
177      * @deprecated Use {@link #getLongVersionCode} instead.
178      */
179     @Deprecated
getVersionCode()180     public int getVersionCode() {
181         return (int) (mVersionCode & 0xffffffff);
182     }
183 
getLongVersionCode()184     public long getLongVersionCode() {
185         return mVersionCode;
186     }
187 
188     @Nullable
getExtras()189     public Bundle getExtras() {
190         return mExtras;
191     }
192 
193     @Override
describeContents()194     public int describeContents() {
195         return 0;
196     }
197 
198     @Override
writeToParcel(Parcel out, int flags)199     public void writeToParcel(Parcel out, int flags) {
200         out.writeBoolean(mShouldLetInstallerDecide);
201         out.writeBundle(mExtras);
202         if (mShouldLetInstallerDecide) {
203             return;
204         }
205         out.writeParcelable(mDigest, flags);
206         out.writeString(mPackageName);
207         out.writeTypedList(mFilters);
208         out.writeLong(mVersionCode);
209     }
210 
211     public static final @android.annotation.NonNull Parcelable.Creator<InstantAppResolveInfo> CREATOR
212             = new Parcelable.Creator<InstantAppResolveInfo>() {
213         public InstantAppResolveInfo createFromParcel(Parcel in) {
214             return new InstantAppResolveInfo(in);
215         }
216 
217         public InstantAppResolveInfo[] newArray(int size) {
218             return new InstantAppResolveInfo[size];
219         }
220     };
221 
222     /**
223      * Helper class to generate and store each of the digests and prefixes
224      * sent to the Instant App Resolver.
225      * <p>
226      * Since intent filters may want to handle multiple hosts within a
227      * domain [eg “*.google.com”], the resolver is presented with multiple
228      * hash prefixes. For example, "a.b.c.d.e" generates digests for
229      * "d.e", "c.d.e", "b.c.d.e" and "a.b.c.d.e".
230      *
231      * @hide
232      */
233     @SystemApi
234     public static final class InstantAppDigest implements Parcelable {
235         static final int DIGEST_MASK = 0xfffff000;
236 
237         /**
238          * A special instance that represents and undefined digest used for cases that a host was
239          * not provided or is irrelevant to the response.
240          */
241         public static final InstantAppDigest UNDEFINED =
242                 new InstantAppDigest(new byte[][]{}, new int[]{});
243 
244         private static Random sRandom = null;
245         static {
246             try {
247                 sRandom = SecureRandom.getInstance("SHA1PRNG");
248             } catch (NoSuchAlgorithmException e) {
249                 // oh well
250                 sRandom = new Random();
251             }
252         }
253         /** Full digest of the domain hashes */
254         private final byte[][] mDigestBytes;
255         /** The first 5 bytes of the domain hashes */
256         private final int[] mDigestPrefix;
257         /** The first 5 bytes of the domain hashes interspersed with random data */
258         private int[] mDigestPrefixSecure;
259 
InstantAppDigest(@onNull String hostName)260         public InstantAppDigest(@NonNull String hostName) {
261             this(hostName, -1 /*maxDigests*/);
262         }
263 
264         /** @hide */
InstantAppDigest(@onNull String hostName, int maxDigests)265         public InstantAppDigest(@NonNull String hostName, int maxDigests) {
266             if (hostName == null) {
267                 throw new IllegalArgumentException();
268             }
269             mDigestBytes = generateDigest(hostName.toLowerCase(Locale.ENGLISH), maxDigests);
270             mDigestPrefix = new int[mDigestBytes.length];
271             for (int i = 0; i < mDigestBytes.length; i++) {
272                 mDigestPrefix[i] =
273                         ((mDigestBytes[i][0] & 0xFF) << 24
274                                 | (mDigestBytes[i][1] & 0xFF) << 16
275                                 | (mDigestBytes[i][2] & 0xFF) << 8
276                                 | (mDigestBytes[i][3] & 0xFF) << 0)
277                         & DIGEST_MASK;
278             }
279         }
280 
InstantAppDigest(byte[][] digestBytes, int[] prefix)281         private InstantAppDigest(byte[][] digestBytes, int[] prefix) {
282             this.mDigestPrefix = prefix;
283             this.mDigestBytes = digestBytes;
284         }
285 
generateDigest(String hostName, int maxDigests)286         private static byte[][] generateDigest(String hostName, int maxDigests) {
287             ArrayList<byte[]> digests = new ArrayList<>();
288             try {
289                 final MessageDigest digest = MessageDigest.getInstance(SHA_ALGORITHM);
290                 if (maxDigests <= 0) {
291                     final byte[] hostBytes = hostName.getBytes();
292                     digests.add(digest.digest(hostBytes));
293                 } else {
294                     int prevDot = hostName.lastIndexOf('.');
295                     prevDot = hostName.lastIndexOf('.', prevDot - 1);
296                     // shortcut for short URLs
297                     if (prevDot < 0) {
298                         digests.add(digest.digest(hostName.getBytes()));
299                     } else {
300                         byte[] hostBytes =
301                                 hostName.substring(prevDot + 1, hostName.length()).getBytes();
302                         digests.add(digest.digest(hostBytes));
303                         int digestCount = 1;
304                         while (prevDot >= 0 && digestCount < maxDigests) {
305                             prevDot = hostName.lastIndexOf('.', prevDot - 1);
306                             hostBytes =
307                                     hostName.substring(prevDot + 1, hostName.length()).getBytes();
308                             digests.add(digest.digest(hostBytes));
309                             digestCount++;
310                         }
311                     }
312                 }
313             } catch (NoSuchAlgorithmException e) {
314                 throw new IllegalStateException("could not find digest algorithm");
315             }
316             return digests.toArray(new byte[digests.size()][]);
317         }
318 
InstantAppDigest(Parcel in)319         InstantAppDigest(Parcel in) {
320             final int digestCount = in.readInt();
321             if (digestCount == -1) {
322                 mDigestBytes = null;
323             } else {
324                 mDigestBytes = new byte[digestCount][];
325                 for (int i = 0; i < digestCount; i++) {
326                     mDigestBytes[i] = in.createByteArray();
327                 }
328             }
329             mDigestPrefix = in.createIntArray();
330             mDigestPrefixSecure = in.createIntArray();
331         }
332 
getDigestBytes()333         public byte[][] getDigestBytes() {
334             return mDigestBytes;
335         }
336 
getDigestPrefix()337         public int[] getDigestPrefix() {
338             return mDigestPrefix;
339         }
340 
341         /**
342          * Returns a digest prefix with additional random prefixes interspersed.
343          * @hide
344          */
getDigestPrefixSecure()345         public int[] getDigestPrefixSecure() {
346             if (this == InstantAppResolveInfo.InstantAppDigest.UNDEFINED) {
347                 return getDigestPrefix();
348             } else if (mDigestPrefixSecure == null) {
349                 // let's generate some random data to intersperse throughout the set of prefixes
350                 final int realSize = getDigestPrefix().length;
351                 final int manufacturedSize = realSize + 10 + sRandom.nextInt(10);
352                 mDigestPrefixSecure = Arrays.copyOf(getDigestPrefix(), manufacturedSize);
353                 for (int i = realSize; i < manufacturedSize; i++) {
354                     mDigestPrefixSecure[i] = sRandom.nextInt() & DIGEST_MASK;
355                 }
356                 Arrays.sort(mDigestPrefixSecure);
357             }
358             return mDigestPrefixSecure;
359         }
360 
361         @Override
describeContents()362         public int describeContents() {
363             return 0;
364         }
365 
366         @Override
writeToParcel(Parcel out, int flags)367         public void writeToParcel(Parcel out, int flags) {
368             final boolean isUndefined = this == UNDEFINED;
369             out.writeBoolean(isUndefined);
370             if (isUndefined) {
371                 return;
372             }
373             if (mDigestBytes == null) {
374                 out.writeInt(-1);
375             } else {
376                 out.writeInt(mDigestBytes.length);
377                 for (int i = 0; i < mDigestBytes.length; i++) {
378                     out.writeByteArray(mDigestBytes[i]);
379                 }
380             }
381             out.writeIntArray(mDigestPrefix);
382             out.writeIntArray(mDigestPrefixSecure);
383         }
384 
385         @SuppressWarnings("hiding")
386         public static final @android.annotation.NonNull Parcelable.Creator<InstantAppDigest> CREATOR =
387                 new Parcelable.Creator<InstantAppDigest>() {
388             @Override
389             public InstantAppDigest createFromParcel(Parcel in) {
390                 if (in.readBoolean() /* is undefined */) {
391                     return UNDEFINED;
392                 }
393                 return new InstantAppDigest(in);
394             }
395             @Override
396             public InstantAppDigest[] newArray(int size) {
397                 return new InstantAppDigest[size];
398             }
399         };
400     }
401 }
402