1 /*
2  * Copyright (C) 2019 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.settings.deviceinfo.legal;
18 
19 import android.content.ContentProvider;
20 import android.content.ContentResolver;
21 import android.content.ContentValues;
22 import android.content.Context;
23 import android.content.SharedPreferences;
24 import android.content.pm.PackageInfo;
25 import android.content.pm.PackageManager;
26 import android.content.res.AssetManager;
27 import android.database.Cursor;
28 import android.net.Uri;
29 import android.os.ParcelFileDescriptor;
30 import android.util.Log;
31 
32 import androidx.annotation.VisibleForTesting;
33 import androidx.core.util.Preconditions;
34 
35 import java.io.File;
36 import java.io.IOException;
37 import java.io.InputStream;
38 import java.nio.file.Files;
39 import java.nio.file.StandardCopyOption;
40 import java.util.List;
41 import java.util.zip.GZIPInputStream;
42 
43 public class ModuleLicenseProvider extends ContentProvider {
44     private static final String TAG = "ModuleLicenseProvider";
45 
46     public static final String AUTHORITY = "com.android.settings.module_licenses";
47     static final String GZIPPED_LICENSE_FILE_NAME = "NOTICE.html.gz";
48     static final String LICENSE_FILE_NAME = "NOTICE.html";
49     static final String LICENSE_FILE_MIME_TYPE = "text/html";
50     static final String PREFS_NAME = "ModuleLicenseProvider";
51 
52     @Override
onCreate()53     public boolean onCreate() {
54         return true;
55     }
56 
57     @Override
query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)58     public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,
59             String sortOrder) {
60         throw new UnsupportedOperationException();
61     }
62 
63     @Override
getType(Uri uri)64     public String getType(Uri uri) {
65         checkUri(getModuleContext(), uri);
66         return LICENSE_FILE_MIME_TYPE;
67     }
68 
69     @Override
insert(Uri uri, ContentValues values)70     public Uri insert(Uri uri, ContentValues values) {
71         throw new UnsupportedOperationException();
72     }
73 
74     @Override
delete(Uri uri, String selection, String[] selectionArgs)75     public int delete(Uri uri, String selection, String[] selectionArgs) {
76         throw new UnsupportedOperationException();
77     }
78 
79     @Override
update(Uri uri, ContentValues values, String selection, String[] selectionArgs)80     public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
81         throw new UnsupportedOperationException();
82     }
83 
84     @Override
openFile(Uri uri, String mode)85     public ParcelFileDescriptor openFile(Uri uri, String mode) {
86         final Context context = getModuleContext();
87         checkUri(context, uri);
88         Preconditions.checkArgument("r".equals(mode), "Read is the only supported mode");
89 
90         try {
91             String packageName = uri.getPathSegments().get(0);
92             File cachedFile = getCachedHtmlFile(context, packageName);
93             if (isCachedHtmlFileOutdated(context, packageName)) {
94                 try (InputStream in = new GZIPInputStream(
95                         getPackageAssetManager(context.getPackageManager(), packageName)
96                                 .open(GZIPPED_LICENSE_FILE_NAME))) {
97                     File directory = getCachedFileDirectory(context, packageName);
98                     if (!directory.exists()) {
99                         directory.mkdir();
100                     }
101                     Files.copy(in, cachedFile.toPath(), StandardCopyOption.REPLACE_EXISTING);
102                 }
103                 // Now that the file is saved, write the package's version code to shared prefs
104                 SharedPreferences.Editor editor = getPrefs(context).edit();
105                 editor.putLong(
106                         packageName,
107                         getPackageInfo(context, packageName).getLongVersionCode())
108                                 .commit();
109             }
110             return ParcelFileDescriptor.open(cachedFile, ParcelFileDescriptor.MODE_READ_ONLY);
111         } catch (PackageManager.NameNotFoundException e) {
112             Log.wtf(TAG, "checkUri should have already caught this error", e);
113         } catch (IOException e) {
114             Log.e(TAG, "Could not open file descriptor", e);
115         }
116         return null;
117     }
118 
119     /**
120      * Returns true if the cached file for the given package is outdated. A cached file is
121      * outdated if one of the following are true:
122      * 1. the shared prefs does not contain a version code for this package
123      * 2. The version code does not match the package's version code
124      * 3. There is no file or the file is empty.
125      */
126     @VisibleForTesting
isCachedHtmlFileOutdated(Context context, String packageName)127     static boolean isCachedHtmlFileOutdated(Context context, String packageName)
128             throws PackageManager.NameNotFoundException {
129         SharedPreferences prefs = getPrefs(context);
130         File file = getCachedHtmlFile(context, packageName);
131         return !prefs.contains(packageName)
132                 || prefs.getLong(packageName, 0L)
133                         != getPackageInfo(context, packageName).getLongVersionCode()
134                 || !file.exists() || file.length() == 0;
135     }
136 
getPackageAssetManager(PackageManager packageManager, String packageName)137     static AssetManager getPackageAssetManager(PackageManager packageManager, String packageName)
138             throws PackageManager.NameNotFoundException {
139         return packageManager.getResourcesForApplication(
140                 packageManager.getPackageInfo(packageName, PackageManager.MATCH_APEX)
141                         .applicationInfo)
142                                 .getAssets();
143     }
144 
getUriForPackage(String packageName)145     static Uri getUriForPackage(String packageName) {
146         return new Uri.Builder()
147                 .scheme(ContentResolver.SCHEME_CONTENT)
148                 .authority(AUTHORITY)
149                 .appendPath(packageName)
150                 .appendPath(LICENSE_FILE_NAME)
151                 .build();
152     }
153 
checkUri(Context context, Uri uri)154     private static void checkUri(Context context, Uri uri) {
155         List<String> pathSegments = uri.getPathSegments();
156         // A URI is valid iff it:
157         // 1. is a content URI
158         // 2. uses the correct authority
159         // 3. has exactly 2 segments and the last one is NOTICE.html
160         // 4. (checked below) first path segment is the package name of a module
161         if (!ContentResolver.SCHEME_CONTENT.equals(uri.getScheme())
162                 || !AUTHORITY.equals(uri.getAuthority())
163                 || pathSegments == null
164                 || pathSegments.size() != 2
165                 || !LICENSE_FILE_NAME.equals(pathSegments.get(1))) {
166             throw new IllegalArgumentException(uri + "is not a valid URI");
167         }
168         // Grab the first path segment, which is the package name of the module and make sure that
169         // there's actually a module for that package. getModuleInfo will throw if it does not
170         // exist.
171         try {
172             context.getPackageManager().getModuleInfo(pathSegments.get(0), 0 /* flags */);
173         } catch (PackageManager.NameNotFoundException e) {
174             throw new IllegalArgumentException(uri + "is not a valid URI", e);
175         }
176     }
177 
getCachedFileDirectory(Context context, String packageName)178     private static File getCachedFileDirectory(Context context, String packageName) {
179         return new File(context.getCacheDir(), packageName);
180     }
181 
getCachedHtmlFile(Context context, String packageName)182     private static File getCachedHtmlFile(Context context, String packageName) {
183         return new File(context.getCacheDir() + "/" + packageName, LICENSE_FILE_NAME);
184     }
185 
getPackageInfo(Context context, String packageName)186     private static  PackageInfo getPackageInfo(Context context, String packageName)
187             throws PackageManager.NameNotFoundException {
188         return context.getPackageManager().getPackageInfo(packageName, PackageManager.MATCH_APEX);
189     }
190 
getPrefs(Context context)191     private static SharedPreferences getPrefs(Context context) {
192         return context.getSharedPreferences(PREFS_NAME, Context.MODE_PRIVATE);
193     }
194 
195     // Method to allow context injection for testing purposes.
196     @VisibleForTesting
getModuleContext()197     protected Context getModuleContext() {
198         return getContext();
199     }
200 }
201