1 /*
2  * Copyright (C) 2016 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 package android.signature.cts.intent;
17 
18 import android.content.pm.ApplicationInfo;
19 import android.content.pm.PackageManager;
20 import android.signature.cts.ApiDocumentParser;
21 import android.signature.cts.JDiffClassDescription.JDiffField;
22 import android.signature.cts.VirtualPath;
23 import android.util.Log;
24 
25 import androidx.test.InstrumentationRegistry;
26 import androidx.test.runner.AndroidJUnit4;
27 
28 import com.android.compatibility.common.util.DynamicConfigDeviceSide;
29 
30 import java.io.IOException;
31 import org.junit.Assert;
32 import org.junit.Before;
33 import org.junit.Test;
34 import org.junit.runner.RunWith;
35 
36 import java.io.BufferedReader;
37 import java.io.File;
38 import java.io.FileInputStream;
39 import java.io.InputStreamReader;
40 import java.util.HashSet;
41 import java.util.List;
42 import java.util.Set;
43 
44 /**
45  * Validate that the android intents used by APKs on this device are part of the
46  * platform.
47  */
48 @RunWith(AndroidJUnit4.class)
49 public class IntentTest {
50 
51     private static final String CURRENT_API_RESOURCE = "current.api.gz";
52 
53     private static final String SYSTEM_CURRENT_API_RESOURCE = "system-current.api.gz";
54 
55     private static final String SYSTEM_REMOVED_API_RESOURCE = "system-removed.api.gz";
56 
57     private static final String TAG = IntentTest.class.getSimpleName();
58 
59     private static final File SIGNATURE_TEST_PACKAGES =
60             new File("/data/local/tmp/signature-test-packages");
61     private static final String ANDROID_INTENT_PREFIX = "android.intent.action";
62     private static final String ACTION_LINE_PREFIX = "          Action: ";
63     private static final String MODULE_NAME = "CtsIntentSignatureTestCases";
64 
65     private PackageManager mPackageManager;
66     private Set<String> intentWhitelist;
67 
68     @Before
setupPackageManager()69     public void setupPackageManager() throws Exception {
70       mPackageManager = InstrumentationRegistry.getContext().getPackageManager();
71       intentWhitelist = getIntentAllowlist();
72     }
73 
74     @Test
shouldNotFindUnexpectedIntents()75     public void shouldNotFindUnexpectedIntents() throws Exception {
76         Set<String> platformIntents = lookupPlatformIntents();
77         platformIntents.addAll(intentWhitelist);
78 
79         Set<String> allInvalidIntents = new HashSet<>();
80 
81         Set<String> errors = new HashSet<>();
82         List<ApplicationInfo> packages =
83             mPackageManager.getInstalledApplications(PackageManager.GET_META_DATA);
84         for (ApplicationInfo appInfo : packages) {
85             if (!isSystemApp(appInfo) && !isUpdatedSystemApp(appInfo)) {
86                 // Only examine system apps
87                 continue;
88             }
89             Set<String> invalidIntents = new HashSet<>();
90             Set<String> activeIntents = lookupActiveIntents(appInfo.packageName);
91 
92             for (String activeIntent : activeIntents) {
93               String intent = activeIntent.trim();
94               if (!platformIntents.contains(intent) &&
95                     intent.startsWith(ANDROID_INTENT_PREFIX)) {
96                   invalidIntents.add(activeIntent);
97                   allInvalidIntents.add(activeIntent);
98               }
99             }
100 
101             String error = String.format("Package: %s Invalid Intent: %s",
102                   appInfo.packageName, invalidIntents);
103             if (!invalidIntents.isEmpty()) {
104                 errors.add(error);
105             }
106         }
107 
108         // Log the allowlist line to make it easy to update.
109         for (String intent : allInvalidIntents) {
110            Log.d(TAG, String.format("whitelist.add(\"%s\");", intent));
111         }
112 
113         Assert.assertTrue(errors.toString(), errors.isEmpty());
114     }
115 
lookupPlatformIntents()116     private Set<String> lookupPlatformIntents() throws IOException {
117         Set<String> intents = new HashSet<>();
118         intents.addAll(parse(CURRENT_API_RESOURCE));
119         intents.addAll(parse(SYSTEM_CURRENT_API_RESOURCE));
120         intents.addAll(parse(SYSTEM_REMOVED_API_RESOURCE));
121         return intents;
122     }
123 
parse(String apiResourceName)124     private static Set<String> parse(String apiResourceName) throws IOException {
125 
126         Set<String> androidIntents = new HashSet<>();
127 
128         ApiDocumentParser apiDocumentParser = new ApiDocumentParser(TAG);
129 
130         VirtualPath.ResourcePath virtualPath =
131                 VirtualPath.get(IntentTest.class.getClassLoader(), apiResourceName);
132         apiDocumentParser.parseAsStream(virtualPath).forEach(
133                 classDescription -> {
134                     for (JDiffField diffField : classDescription.getFieldList()) {
135                         String fieldValue = diffField.getValueString();
136                         if (fieldValue != null) {
137                             fieldValue = fieldValue.replace("\"", "");
138                             if (fieldValue.startsWith(ANDROID_INTENT_PREFIX)) {
139                                 androidIntents.add(fieldValue);
140                             }
141                         }
142                     }
143                 });
144 
145         return androidIntents;
146     }
147 
isSystemApp(ApplicationInfo applicationInfo)148     private static boolean isSystemApp(ApplicationInfo applicationInfo) {
149         return (applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0;
150     }
151 
isUpdatedSystemApp(ApplicationInfo applicationInfo)152     private static boolean isUpdatedSystemApp(ApplicationInfo applicationInfo) {
153         return (applicationInfo.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0;
154     }
155 
lookupActiveIntents(String packageName)156     private static Set<String> lookupActiveIntents(String packageName) {
157         HashSet<String> activeIntents = new HashSet<>();
158         File dumpsysPackage = new File(SIGNATURE_TEST_PACKAGES, packageName + ".txt");
159         if (!dumpsysPackage.exists() || dumpsysPackage.length() == 0) {
160           throw new RuntimeException("Missing package info: " + dumpsysPackage.getAbsolutePath());
161         }
162         try (
163             BufferedReader in = new BufferedReader(
164                   new InputStreamReader(new FileInputStream(dumpsysPackage)))) {
165             String line;
166             while ((line = in.readLine()) != null) {
167                 if (line.startsWith(ACTION_LINE_PREFIX)) {
168                     String intent = line.substring(
169                           ACTION_LINE_PREFIX.length(), line.length() - 1);
170                     activeIntents.add(intent.replace("\"", ""));
171                 }
172             }
173             return activeIntents;
174         } catch (Exception e) {
175           throw new RuntimeException("While retrieving dumpsys", e);
176         }
177     }
178 
getIntentAllowlist()179     private static Set<String> getIntentAllowlist() throws Exception {
180         Set<String> whitelist = new HashSet<>();
181 
182         DynamicConfigDeviceSide dcds = new DynamicConfigDeviceSide(MODULE_NAME);
183         List<String> intentWhitelist = dcds.getValues("intent_whitelist");
184 
185         // Log the allowlist Intent
186         for (String intent : intentWhitelist) {
187            Log.d(TAG, String.format("whitelist add: %s", intent));
188            whitelist.add(intent);
189         }
190 
191         return whitelist;
192     }
193 }
194