1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
|
/*
* Copyright (C) 2009 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.webkit;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.content.Context;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.database.ContentObserver;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteException;
import android.database.sqlite.SQLiteStatement;
import android.os.Handler;
import android.preference.PreferenceManager;
import android.provider.Settings;
import android.util.Log;
import java.io.File;
import java.util.HashSet;
/**
* Donut-specific hack to keep Gears permissions in sync with the
* system location setting.
*/
class GearsPermissionsManager {
// The application context.
Context mContext;
// The path to gears.so.
private String mGearsPath;
// The Gears permissions database directory.
private final static String GEARS_DATABASE_DIR = "gears";
// The Gears permissions database file name.
private final static String GEARS_DATABASE_FILE = "permissions.db";
// The Gears location permissions table.
private final static String GEARS_LOCATION_ACCESS_TABLE_NAME =
"LocationAccess";
// The Gears storage access permissions table.
private final static String GEARS_STORAGE_ACCESS_TABLE_NAME = "Access";
// The Gears permissions db schema version table.
private final static String GEARS_SCHEMA_VERSION_TABLE_NAME =
"VersionInfo";
// The shared pref name.
private static final String LAST_KNOWN_LOCATION_SETTING =
"lastKnownLocationSystemSetting";
// The Browser package name.
private static final String BROWSER_PACKAGE_NAME = "com.android.browser";
// The Secure Settings observer that will be notified when the system
// location setting changes.
private SecureSettingsObserver mSettingsObserver;
// The Google URLs whitelisted for Gears location access.
private static HashSet<String> sGearsWhiteList;
static {
sGearsWhiteList = new HashSet<String>();
// NOTE: DO NOT ADD A "/" AT THE END!
sGearsWhiteList.add("http://www.google.com");
sGearsWhiteList.add("http://www.google.co.uk");
}
private static final String LOGTAG = "webcore";
static final boolean DEBUG = false;
static final boolean LOGV_ENABLED = DEBUG;
GearsPermissionsManager(Context context, String gearsPath) {
mContext = context;
mGearsPath = gearsPath;
}
public void doCheckAndStartObserver() {
// Are we running in the browser?
if (!BROWSER_PACKAGE_NAME.equals(mContext.getPackageName())) {
return;
}
// Do the check.
checkGearsPermissions();
// Install the observer.
mSettingsObserver = new SecureSettingsObserver();
mSettingsObserver.observe();
}
private void checkGearsPermissions() {
// Get the current system settings.
int setting = Settings.Secure.getInt(mContext.getContentResolver(),
Settings.Secure.USE_LOCATION_FOR_SERVICES, -1);
// Check if we need to set the Gears permissions.
if (setting != -1 && locationSystemSettingChanged(setting)) {
setGearsPermissionForGoogleDomains(setting);
}
}
private boolean locationSystemSettingChanged(int newSetting) {
SharedPreferences prefs =
PreferenceManager.getDefaultSharedPreferences(mContext);
int oldSetting = 0;
oldSetting = prefs.getInt(LAST_KNOWN_LOCATION_SETTING, oldSetting);
if (oldSetting == newSetting) {
return false;
}
Editor ed = prefs.edit();
ed.putInt(LAST_KNOWN_LOCATION_SETTING, newSetting);
ed.commit();
return true;
}
private void setGearsPermissionForGoogleDomains(int systemPermission) {
// Transform the system permission into a Gears permission
int gearsPermission = (systemPermission == 1 ? 1 : 2);
// Build the path to the Gears library.
File file = new File(mGearsPath).getParentFile();
if (file == null) {
return;
}
// Build the Gears database file name.
file = new File(file.getAbsolutePath() + File.separator
+ GEARS_DATABASE_DIR + File.separator + GEARS_DATABASE_FILE);
// Remember whether or not we need to create the LocationAccess table.
boolean needToCreateTables = !file.exists();
// Try opening the Gears database.
SQLiteDatabase permissions;
try {
permissions = SQLiteDatabase.openOrCreateDatabase(file, null);
} catch (SQLiteException e) {
if (LOGV_ENABLED) {
Log.v(LOGTAG, "Could not open Gears permission DB: "
+ e.getMessage());
}
// Just bail out.
return;
}
// We now have a database open. Begin a transaction.
permissions.beginTransaction();
try {
if (needToCreateTables) {
// Create the tables. Note that this creates the
// Gears tables for the permissions DB schema version 2.
// The Gears schema upgrade process will take care of the rest.
// First, the storage access table.
SQLiteStatement statement = permissions.compileStatement(
"CREATE TABLE IF NOT EXISTS "
+ GEARS_STORAGE_ACCESS_TABLE_NAME
+ " (Name TEXT UNIQUE, Value)");
statement.execute();
// Next the location access table.
statement = permissions.compileStatement(
"CREATE TABLE IF NOT EXISTS "
+ GEARS_LOCATION_ACCESS_TABLE_NAME
+ " (Name TEXT UNIQUE, Value)");
statement.execute();
// Finally, the schema version table.
statement = permissions.compileStatement(
"CREATE TABLE IF NOT EXISTS "
+ GEARS_SCHEMA_VERSION_TABLE_NAME
+ " (Name TEXT UNIQUE, Value)");
statement.execute();
// Set the schema version to 2.
ContentValues schema = new ContentValues();
schema.put("Name", "Version");
schema.put("Value", 2);
permissions.insert(GEARS_SCHEMA_VERSION_TABLE_NAME, null,
schema);
}
ContentValues permissionValues = new ContentValues();
for (String url : sGearsWhiteList) {
permissionValues.put("Name", url);
permissionValues.put("Value", gearsPermission);
permissions.replace(GEARS_LOCATION_ACCESS_TABLE_NAME, null,
permissionValues);
permissionValues.clear();
}
// Commit the transaction.
permissions.setTransactionSuccessful();
} catch (SQLiteException e) {
if (LOGV_ENABLED) {
Log.v(LOGTAG, "Could not set the Gears permissions: "
+ e.getMessage());
}
} finally {
permissions.endTransaction();
permissions.close();
}
}
class SecureSettingsObserver extends ContentObserver {
SecureSettingsObserver() {
super(new Handler());
}
void observe() {
ContentResolver resolver = mContext.getContentResolver();
resolver.registerContentObserver(Settings.Secure.getUriFor(
Settings.Secure.USE_LOCATION_FOR_SERVICES), false, this);
}
@Override
public void onChange(boolean selfChange) {
checkGearsPermissions();
}
}
}
|