230 lines
7.8 KiB
Java
230 lines
7.8 KiB
Java
/*
|
|
* 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<Object>() {
|
|
@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<Object>() {
|
|
@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<Object> mFlushCallable = new Callable<Object>() {
|
|
@Override
|
|
public Object call() throws Exception {
|
|
mJsonWriter.flush();
|
|
return null;
|
|
}
|
|
};
|
|
|
|
private ScheduledFuture<Object> 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<Object>() {
|
|
@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;
|
|
}
|
|
}
|