diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index dea71b9..1e82231 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -11,6 +11,7 @@
+
+
+
+
+
@@ -78,12 +83,29 @@
android:resource="@xml/file_paths" />
+
+
+
+
+
+
+
+
+
+
diff --git a/app/src/main/aidl/net/typeblog/shelter/services/IFileShuttleService.aidl b/app/src/main/aidl/net/typeblog/shelter/services/IFileShuttleService.aidl
new file mode 100644
index 0000000..2236803
--- /dev/null
+++ b/app/src/main/aidl/net/typeblog/shelter/services/IFileShuttleService.aidl
@@ -0,0 +1,11 @@
+// IFileShuttleService.aidl
+package net.typeblog.shelter.services;
+
+import android.os.ParcelFileDescriptor;
+
+interface IFileShuttleService {
+ void ping();
+ List loadFiles(String path);
+ Map loadFileMeta(String path);
+ ParcelFileDescriptor openFile(String path, String mode);
+}
diff --git a/app/src/main/aidl/net/typeblog/shelter/services/IFileShuttleServiceCallback.aidl b/app/src/main/aidl/net/typeblog/shelter/services/IFileShuttleServiceCallback.aidl
new file mode 100644
index 0000000..ff33f53
--- /dev/null
+++ b/app/src/main/aidl/net/typeblog/shelter/services/IFileShuttleServiceCallback.aidl
@@ -0,0 +1,8 @@
+// IFileShuttleServiceCallback.aidl
+package net.typeblog.shelter.services;
+
+import net.typeblog.shelter.services.IFileShuttleService;
+
+interface IFileShuttleServiceCallback {
+ void callback(in IFileShuttleService service);
+}
diff --git a/app/src/main/java/net/typeblog/shelter/ShelterApplication.java b/app/src/main/java/net/typeblog/shelter/ShelterApplication.java
index 3ec8979..d2fb177 100644
--- a/app/src/main/java/net/typeblog/shelter/ShelterApplication.java
+++ b/app/src/main/java/net/typeblog/shelter/ShelterApplication.java
@@ -5,11 +5,13 @@ import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
+import net.typeblog.shelter.services.FileShuttleService;
import net.typeblog.shelter.services.ShelterService;
import net.typeblog.shelter.util.LocalStorageManager;
public class ShelterApplication extends Application {
private ServiceConnection mShelterServiceConnection = null;
+ private ServiceConnection mFileShuttleServiceConnection = null;
@Override
public void onCreate() {
@@ -25,6 +27,13 @@ public class ShelterApplication extends Application {
mShelterServiceConnection = conn;
}
+ public void bindFileShuttleService(ServiceConnection conn) {
+ unbindFileShuttleService();;
+ Intent intent = new Intent(getApplicationContext(), FileShuttleService.class);
+ bindService(intent, conn, Context.BIND_AUTO_CREATE);
+ mFileShuttleServiceConnection = conn;
+ }
+
public void unbindShelterService() {
if (mShelterServiceConnection != null) {
try {
@@ -38,4 +47,16 @@ public class ShelterApplication extends Application {
mShelterServiceConnection = null;
}
+
+ public void unbindFileShuttleService() {
+ if (mFileShuttleServiceConnection != null) {
+ try {
+ unbindService(mFileShuttleServiceConnection);
+ } catch (Exception e) {
+ // ...
+ }
+ }
+
+ mFileShuttleServiceConnection = null;
+ }
}
diff --git a/app/src/main/java/net/typeblog/shelter/services/FileShuttleService.java b/app/src/main/java/net/typeblog/shelter/services/FileShuttleService.java
new file mode 100644
index 0000000..db027fa
--- /dev/null
+++ b/app/src/main/java/net/typeblog/shelter/services/FileShuttleService.java
@@ -0,0 +1,117 @@
+package net.typeblog.shelter.services;
+
+import android.app.Service;
+import android.content.Intent;
+import android.os.Environment;
+import android.os.Handler;
+import android.os.IBinder;
+import android.os.Looper;
+import android.os.ParcelFileDescriptor;
+import android.provider.DocumentsContract;
+import android.support.annotation.Nullable;
+import android.webkit.MimeTypeMap;
+
+import net.typeblog.shelter.ShelterApplication;
+import net.typeblog.shelter.util.CrossProfileDocumentsProvider;
+
+import java.io.File;
+import java.io.FileNotFoundException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+// A service to forward file information across the profile boundary
+public class FileShuttleService extends Service {
+ public static final long TIMEOUT = 10000;
+ // Periodic task to stop the service when idle.
+ // This service does not need to persist.
+ private Runnable mSuicideTask = this::suicide;
+ private Handler mHandler = new Handler(Looper.getMainLooper());
+ private IFileShuttleService.Stub mStub = new IFileShuttleService.Stub() {
+ @Override
+ public void ping() {
+ // Dummy method
+ resetSuicideTask();
+ }
+
+ @Override
+ public List