/* * Copyright (C) 2012 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 com.android.inputmethod.research; import android.util.JsonWriter; import android.util.Log; import com.android.inputmethod.latin.define.ProductionFlag; import java.io.BufferedWriter; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.io.OutputStream; import java.io.OutputStreamWriter; import java.util.concurrent.Callable; import java.util.concurrent.Executors; import java.util.concurrent.RejectedExecutionException; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; /** * Logs the use of the LatinIME keyboard. * * This class logs operations on the IME keyboard, including what the user has typed. * Data is stored locally in a file in app-specific storage. * * This functionality is off by default. See {@link ProductionFlag#IS_EXPERIMENTAL}. */ public class ResearchLog { private static final String TAG = ResearchLog.class.getSimpleName(); private static final boolean DEBUG = false && ProductionFlag.IS_EXPERIMENTAL_DEBUG; private static final long FLUSH_DELAY_IN_MS = 1000 * 5; private static final int ABORT_TIMEOUT_IN_MS = 1000 * 4; /* package */ final ScheduledExecutorService mExecutor; /* package */ final File mFile; private JsonWriter mJsonWriter = NULL_JSON_WRITER; // true if at least one byte of data has been written out to the log file. This must be // remembered because JsonWriter requires that calls matching calls to beginObject and // endObject, as well as beginArray and endArray, and the file is opened lazily, only when // it is certain that data will be written. Alternatively, the matching call exceptions // could be caught, but this might suppress other errors. private boolean mHasWrittenData = false; private static final JsonWriter NULL_JSON_WRITER = new JsonWriter( new OutputStreamWriter(new NullOutputStream())); private static class NullOutputStream extends OutputStream { /** {@inheritDoc} */ @Override public void write(byte[] buffer, int offset, int count) { // nop } /** {@inheritDoc} */ @Override public void write(byte[] buffer) { // nop } @Override public void write(int oneByte) { } } public ResearchLog(final File outputFile) { if (outputFile == null) { throw new IllegalArgumentException(); } mExecutor = Executors.newSingleThreadScheduledExecutor(); mFile = outputFile; } public synchronized void close(final Runnable onClosed) { mExecutor.submit(new Callable() { @Override public Object call() throws Exception { try { if (mHasWrittenData) { mJsonWriter.endArray(); mJsonWriter.flush(); mJsonWriter.close(); if (DEBUG) { Log.d(TAG, "wrote log to " + mFile); } mHasWrittenData = false; } else { if (DEBUG) { Log.d(TAG, "close() called, but no data, not outputting"); } } } catch (Exception e) { Log.d(TAG, "error when closing ResearchLog:"); e.printStackTrace(); } finally { if (mFile.exists()) { mFile.setWritable(false, false); } if (onClosed != null) { onClosed.run(); } } return null; } }); removeAnyScheduledFlush(); mExecutor.shutdown(); } private boolean mIsAbortSuccessful; public synchronized void abort() { mExecutor.submit(new Callable() { @Override public Object call() throws Exception { try { if (mHasWrittenData) { mJsonWriter.endArray(); mJsonWriter.close(); mHasWrittenData = false; } } finally { mIsAbortSuccessful = mFile.delete(); } return null; } }); removeAnyScheduledFlush(); mExecutor.shutdown(); } public boolean blockingAbort() throws InterruptedException { abort(); mExecutor.awaitTermination(ABORT_TIMEOUT_IN_MS, TimeUnit.MILLISECONDS); return mIsAbortSuccessful; } public void awaitTermination(int delay, TimeUnit timeUnit) throws InterruptedException { mExecutor.awaitTermination(delay, timeUnit); } /* package */ synchronized void flush() { removeAnyScheduledFlush(); mExecutor.submit(mFlushCallable); } private final Callable mFlushCallable = new Callable() { @Override public Object call() throws Exception { mJsonWriter.flush(); return null; } }; private ScheduledFuture mFlushFuture; private void removeAnyScheduledFlush() { if (mFlushFuture != null) { mFlushFuture.cancel(false); mFlushFuture = null; } } private void scheduleFlush() { removeAnyScheduledFlush(); mFlushFuture = mExecutor.schedule(mFlushCallable, FLUSH_DELAY_IN_MS, TimeUnit.MILLISECONDS); } public synchronized void publish(final LogUnit logUnit, final boolean isIncludingPrivateData) { try { mExecutor.submit(new Callable() { @Override public Object call() throws Exception { logUnit.publishTo(ResearchLog.this, isIncludingPrivateData); scheduleFlush(); return null; } }); } catch (RejectedExecutionException e) { // TODO: Add code to record loss of data, and report. if (DEBUG) { Log.d(TAG, "ResearchLog.publish() rejecting scheduled execution"); } } } /** * Return a JsonWriter for this ResearchLog. It is initialized the first time this method is * called. The cached value is returned in future calls. */ public JsonWriter getValidJsonWriterLocked() { try { if (mJsonWriter == NULL_JSON_WRITER) { mJsonWriter = new JsonWriter(new BufferedWriter(new FileWriter(mFile))); mJsonWriter.beginArray(); mHasWrittenData = true; } } catch (IOException e) { e.printStackTrace(); Log.w(TAG, "Error in JsonWriter; disabling logging"); try { mJsonWriter.close(); } catch (IllegalStateException e1) { // Assume that this is just the json not being terminated properly. // Ignore } catch (IOException e1) { e1.printStackTrace(); } finally { mJsonWriter = NULL_JSON_WRITER; } } return mJsonWriter; } }