blob: 659696e0e2126be2e2ce6725145c543a0fef0417 [file] [log] [blame]
Joseph Wen6a34bb22015-02-25 14:00:39 -05001/*
2 * Copyright (C) 2015 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
17package com.android.statementservice;
18
19import android.app.Service;
20import android.content.Intent;
21import android.net.http.HttpResponseCache;
22import android.os.Bundle;
23import android.os.Handler;
24import android.os.HandlerThread;
25import android.os.IBinder;
26import android.os.Looper;
27import android.os.ResultReceiver;
28import android.util.Log;
29
30import com.android.statementservice.retriever.AbstractAsset;
31import com.android.statementservice.retriever.AbstractAssetMatcher;
32import com.android.statementservice.retriever.AbstractStatementRetriever;
33import com.android.statementservice.retriever.AbstractStatementRetriever.Result;
34import com.android.statementservice.retriever.AssociationServiceException;
35import com.android.statementservice.retriever.Relation;
36import com.android.statementservice.retriever.Statement;
37
38import org.json.JSONException;
39
40import java.io.File;
41import java.io.IOException;
42import java.util.ArrayList;
43import java.util.List;
44import java.util.concurrent.Callable;
45
46/**
47 * Handles com.android.statementservice.service.CHECK_ALL_ACTION intents.
48 */
49public final class DirectStatementService extends Service {
50 private static final String TAG = DirectStatementService.class.getSimpleName();
51
52 /**
53 * Returns true if every asset in {@code SOURCE_ASSET_DESCRIPTORS} is associated with {@code
54 * EXTRA_TARGET_ASSET_DESCRIPTOR} for {@code EXTRA_RELATION} relation.
55 *
56 * <p>Takes parameter {@code EXTRA_RELATION}, {@code SOURCE_ASSET_DESCRIPTORS}, {@code
57 * EXTRA_TARGET_ASSET_DESCRIPTOR}, and {@code EXTRA_RESULT_RECEIVER}.
58 */
59 public static final String CHECK_ALL_ACTION =
60 "com.android.statementservice.service.CHECK_ALL_ACTION";
61
62 /**
63 * Parameter for {@link #CHECK_ALL_ACTION}.
64 *
65 * <p>A relation string.
66 */
67 public static final String EXTRA_RELATION =
68 "com.android.statementservice.service.RELATION";
69
70 /**
71 * Parameter for {@link #CHECK_ALL_ACTION}.
72 *
73 * <p>An array of asset descriptors in JSON.
74 */
75 public static final String EXTRA_SOURCE_ASSET_DESCRIPTORS =
76 "com.android.statementservice.service.SOURCE_ASSET_DESCRIPTORS";
77
78 /**
79 * Parameter for {@link #CHECK_ALL_ACTION}.
80 *
81 * <p>An asset descriptor in JSON.
82 */
83 public static final String EXTRA_TARGET_ASSET_DESCRIPTOR =
84 "com.android.statementservice.service.TARGET_ASSET_DESCRIPTOR";
85
86 /**
87 * Parameter for {@link #CHECK_ALL_ACTION}.
88 *
89 * <p>A {@code ResultReceiver} instance that will be used to return the result. If the request
90 * failed, return {@link #RESULT_FAIL} and an empty {@link android.os.Bundle}. Otherwise, return
91 * {@link #RESULT_SUCCESS} and a {@link android.os.Bundle} with the result stored in {@link
92 * #IS_ASSOCIATED}.
93 */
94 public static final String EXTRA_RESULT_RECEIVER =
95 "com.android.statementservice.service.RESULT_RECEIVER";
96
97 /**
98 * A boolean bundle entry that stores the result of {@link #CHECK_ALL_ACTION}.
99 * This is set only if the service returns with {@code RESULT_SUCCESS}.
100 * {@code IS_ASSOCIATED} is true if and only if {@code FAILED_SOURCES} is empty.
101 */
102 public static final String IS_ASSOCIATED = "is_associated";
103
104 /**
105 * A String ArrayList bundle entry that stores sources that can't be verified.
106 */
107 public static final String FAILED_SOURCES = "failed_sources";
108
109 /**
110 * Returned by the service if the request is successfully processed. The caller should check
111 * the {@code IS_ASSOCIATED} field to determine if the association exists or not.
112 */
113 public static final int RESULT_SUCCESS = 0;
114
115 /**
116 * Returned by the service if the request failed. The request will fail if, for example, the
117 * input is not well formed, or the network is not available.
118 */
119 public static final int RESULT_FAIL = 1;
120
121 private static final long HTTP_CACHE_SIZE_IN_BYTES = 1 * 1024 * 1024; // 1 MBytes
122 private static final String CACHE_FILENAME = "request_cache";
123
124 private AbstractStatementRetriever mStatementRetriever;
125 private Handler mHandler;
126 private HandlerThread mThread;
127 private HttpResponseCache mHttpResponseCache;
128
129 @Override
130 public void onCreate() {
131 mThread = new HandlerThread("DirectStatementService thread",
132 android.os.Process.THREAD_PRIORITY_BACKGROUND);
133 mThread.start();
134 onCreate(AbstractStatementRetriever.createDirectRetriever(this), mThread.getLooper(),
135 getCacheDir());
136 }
137
138 /**
139 * Creates a DirectStatementService with the dependencies passed in for easy testing.
140 */
141 public void onCreate(AbstractStatementRetriever statementRetriever, Looper looper,
142 File cacheDir) {
143 super.onCreate();
144 mStatementRetriever = statementRetriever;
145 mHandler = new Handler(looper);
146
147 try {
148 File httpCacheDir = new File(cacheDir, CACHE_FILENAME);
149 mHttpResponseCache = HttpResponseCache.install(httpCacheDir, HTTP_CACHE_SIZE_IN_BYTES);
150 } catch (IOException e) {
151 Log.i(TAG, "HTTPS response cache installation failed:" + e);
152 }
153 }
154
155 @Override
156 public void onDestroy() {
157 super.onDestroy();
Hidehiko Tsuchiya17d67682018-01-18 13:50:55 +0900158 final HttpResponseCache responseCache = mHttpResponseCache;
159 mHandler.post(new Runnable() {
160 public void run() {
161 try {
162 if (responseCache != null) {
163 responseCache.delete();
164 }
165 } catch (IOException e) {
166 Log.i(TAG, "HTTP(S) response cache deletion failed:" + e);
167 }
168 Looper.myLooper().quit();
Joseph Wen6a34bb22015-02-25 14:00:39 -0500169 }
Hidehiko Tsuchiya17d67682018-01-18 13:50:55 +0900170 });
171 mHttpResponseCache = null;
Joseph Wen6a34bb22015-02-25 14:00:39 -0500172 }
173
174 @Override
175 public IBinder onBind(Intent intent) {
176 return null;
177 }
178
179 @Override
180 public int onStartCommand(Intent intent, int flags, int startId) {
181 super.onStartCommand(intent, flags, startId);
182
183 if (intent == null) {
184 Log.e(TAG, "onStartCommand called with null intent");
185 return START_STICKY;
186 }
187
188 if (intent.getAction().equals(CHECK_ALL_ACTION)) {
189
190 Bundle extras = intent.getExtras();
191 List<String> sources = extras.getStringArrayList(EXTRA_SOURCE_ASSET_DESCRIPTORS);
192 String target = extras.getString(EXTRA_TARGET_ASSET_DESCRIPTOR);
193 String relation = extras.getString(EXTRA_RELATION);
194 ResultReceiver resultReceiver = extras.getParcelable(EXTRA_RESULT_RECEIVER);
195
196 if (resultReceiver == null) {
197 Log.e(TAG, " Intent does not have extra " + EXTRA_RESULT_RECEIVER);
198 return START_STICKY;
199 }
200 if (sources == null) {
201 Log.e(TAG, " Intent does not have extra " + EXTRA_SOURCE_ASSET_DESCRIPTORS);
202 resultReceiver.send(RESULT_FAIL, Bundle.EMPTY);
203 return START_STICKY;
204 }
205 if (target == null) {
206 Log.e(TAG, " Intent does not have extra " + EXTRA_TARGET_ASSET_DESCRIPTOR);
207 resultReceiver.send(RESULT_FAIL, Bundle.EMPTY);
208 return START_STICKY;
209 }
210 if (relation == null) {
211 Log.e(TAG, " Intent does not have extra " + EXTRA_RELATION);
212 resultReceiver.send(RESULT_FAIL, Bundle.EMPTY);
213 return START_STICKY;
214 }
215
216 mHandler.post(new ExceptionLoggingFutureTask<Void>(
217 new IsAssociatedCallable(sources, target, relation, resultReceiver), TAG));
218 } else {
219 Log.e(TAG, "onStartCommand called with unsupported action: " + intent.getAction());
220 }
221 return START_STICKY;
222 }
223
224 private class IsAssociatedCallable implements Callable<Void> {
225
226 private List<String> mSources;
227 private String mTarget;
228 private String mRelation;
229 private ResultReceiver mResultReceiver;
230
231 public IsAssociatedCallable(List<String> sources, String target, String relation,
232 ResultReceiver resultReceiver) {
233 mSources = sources;
234 mTarget = target;
235 mRelation = relation;
236 mResultReceiver = resultReceiver;
237 }
238
239 private boolean verifyOneSource(AbstractAsset source, AbstractAssetMatcher target,
240 Relation relation) throws AssociationServiceException {
241 Result statements = mStatementRetriever.retrieveStatements(source);
242 for (Statement statement : statements.getStatements()) {
243 if (relation.matches(statement.getRelation())
244 && target.matches(statement.getTarget())) {
245 return true;
246 }
247 }
248 return false;
249 }
250
251 @Override
252 public Void call() {
253 Bundle result = new Bundle();
254 ArrayList<String> failedSources = new ArrayList<String>();
255 AbstractAssetMatcher target;
256 Relation relation;
257 try {
258 target = AbstractAssetMatcher.createMatcher(mTarget);
259 relation = Relation.create(mRelation);
260 } catch (AssociationServiceException | JSONException e) {
261 Log.e(TAG, "isAssociatedCallable failed with exception", e);
262 mResultReceiver.send(RESULT_FAIL, Bundle.EMPTY);
263 return null;
264 }
265
266 boolean allSourcesVerified = true;
267 for (String sourceString : mSources) {
268 AbstractAsset source;
269 try {
270 source = AbstractAsset.create(sourceString);
271 } catch (AssociationServiceException e) {
272 mResultReceiver.send(RESULT_FAIL, Bundle.EMPTY);
273 return null;
274 }
275
276 try {
277 if (!verifyOneSource(source, target, relation)) {
278 failedSources.add(source.toJson());
279 allSourcesVerified = false;
280 }
281 } catch (AssociationServiceException e) {
282 failedSources.add(source.toJson());
283 allSourcesVerified = false;
284 }
285 }
286
287 result.putBoolean(IS_ASSOCIATED, allSourcesVerified);
288 result.putStringArrayList(FAILED_SOURCES, failedSources);
289 mResultReceiver.send(RESULT_SUCCESS, result);
290 return null;
291 }
292 }
293}