Request missing permission when rescanning USB devices
This commit is contained in:
@@ -17,7 +17,7 @@
|
|||||||
|
|
||||||
package org.sufficientlysecure.keychain.securitytoken;
|
package org.sufficientlysecure.keychain.securitytoken;
|
||||||
|
|
||||||
import android.app.Activity;
|
import android.app.PendingIntent;
|
||||||
import android.content.BroadcastReceiver;
|
import android.content.BroadcastReceiver;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
@@ -25,30 +25,37 @@ import android.content.IntentFilter;
|
|||||||
import android.hardware.usb.UsbDevice;
|
import android.hardware.usb.UsbDevice;
|
||||||
import android.hardware.usb.UsbManager;
|
import android.hardware.usb.UsbManager;
|
||||||
|
|
||||||
|
import org.sufficientlysecure.keychain.BuildConfig;
|
||||||
|
import org.sufficientlysecure.keychain.securitytoken.SecurityTokenInfo.TokenType;
|
||||||
import org.sufficientlysecure.keychain.securitytoken.usb.UsbTransport;
|
import org.sufficientlysecure.keychain.securitytoken.usb.UsbTransport;
|
||||||
import org.sufficientlysecure.keychain.ui.UsbEventReceiverActivity;
|
|
||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
|
|
||||||
public class UsbConnectionDispatcher {
|
public class UsbConnectionDispatcher {
|
||||||
private Activity mActivity;
|
private static final String ACTION_USB_PERMISSION = "org.sufficientlysecure.keychain.ui.USB_PERMISSION";
|
||||||
|
|
||||||
private OnDiscoveredUsbDeviceListener mListener;
|
private Context context;
|
||||||
private UsbManager mUsbManager;
|
private OnDiscoveredUsbDeviceListener onDiscoveredUsbDeviceListener;
|
||||||
|
private UsbManager usbManager;
|
||||||
|
|
||||||
/**
|
public UsbConnectionDispatcher(Context context, OnDiscoveredUsbDeviceListener listener) {
|
||||||
* Receives broadcast when a supported USB device get permission.
|
this.context = context.getApplicationContext();
|
||||||
*/
|
this.onDiscoveredUsbDeviceListener = listener;
|
||||||
private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() {
|
this.usbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
|
||||||
|
}
|
||||||
|
|
||||||
|
private final BroadcastReceiver usbBroadcastReceiver = new BroadcastReceiver() {
|
||||||
@Override
|
@Override
|
||||||
public void onReceive(Context context, Intent intent) {
|
public void onReceive(Context context, Intent intent) {
|
||||||
String action = intent.getAction();
|
String action = intent.getAction();
|
||||||
|
if (action == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
switch (action) {
|
switch (action) {
|
||||||
case UsbEventReceiverActivity.ACTION_USB_PERMISSION: {
|
case ACTION_USB_PERMISSION: {
|
||||||
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
||||||
boolean permission = intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED,
|
boolean permission = intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false);
|
||||||
false);
|
|
||||||
if (permission) {
|
if (permission) {
|
||||||
Timber.d("Got permission for " + usbDevice.getDeviceName());
|
Timber.d("Got permission for " + usbDevice.getDeviceName());
|
||||||
sendUsbTransportDiscovered(usbDevice);
|
sendUsbTransportDiscovered(usbDevice);
|
||||||
@@ -59,21 +66,15 @@ public class UsbConnectionDispatcher {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
public UsbConnectionDispatcher(final Activity activity, final OnDiscoveredUsbDeviceListener listener) {
|
|
||||||
this.mActivity = activity;
|
|
||||||
this.mListener = listener;
|
|
||||||
this.mUsbManager = (UsbManager) activity.getSystemService(Context.USB_SERVICE);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onStart() {
|
public void onStart() {
|
||||||
final IntentFilter intentFilter = new IntentFilter();
|
final IntentFilter intentFilter = new IntentFilter();
|
||||||
intentFilter.addAction(UsbEventReceiverActivity.ACTION_USB_PERMISSION);
|
intentFilter.addAction(ACTION_USB_PERMISSION);
|
||||||
|
|
||||||
mActivity.registerReceiver(mUsbReceiver, intentFilter);
|
context.registerReceiver(usbBroadcastReceiver, intentFilter);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void onStop() {
|
public void onStop() {
|
||||||
mActivity.unregisterReceiver(mUsbReceiver);
|
context.unregisterReceiver(usbBroadcastReceiver);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -83,24 +84,48 @@ public class UsbConnectionDispatcher {
|
|||||||
// Note: we don't check devices VID/PID because
|
// Note: we don't check devices VID/PID because
|
||||||
// we check for permission instead.
|
// we check for permission instead.
|
||||||
// We should have permission only for matching devices
|
// We should have permission only for matching devices
|
||||||
for (UsbDevice usbDevice : mUsbManager.getDeviceList().values()) {
|
for (UsbDevice usbDevice : usbManager.getDeviceList().values()) {
|
||||||
if (mUsbManager.hasPermission(usbDevice)) {
|
Timber.d("Device: %s", usbDevice.toString());
|
||||||
|
if (usbManager.hasPermission(usbDevice)) {
|
||||||
|
Timber.d("Got permission!");
|
||||||
sendUsbTransportDiscovered(usbDevice);
|
sendUsbTransportDiscovered(usbDevice);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
TokenType tokenType = UsbTransport.getTokenTypeFromUsbDeviceInfo(
|
||||||
|
usbDevice.getVendorId(), usbDevice.getProductId(), null);
|
||||||
|
if (tokenType != null) {
|
||||||
|
Timber.d("Token type: %s", tokenType);
|
||||||
|
requestPermissionForUsbDevice(context, usbDevice);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
Timber.d("Unknown device type, doing nothing…");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendUsbTransportDiscovered(UsbDevice usbDevice) {
|
private void sendUsbTransportDiscovered(UsbDevice usbDevice) {
|
||||||
if (mListener == null) {
|
if (onDiscoveredUsbDeviceListener == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
UsbTransport usbTransport = UsbTransport.createUsbTransport(mActivity.getBaseContext(), usbDevice);
|
UsbTransport usbTransport = UsbTransport.createUsbTransport(context, usbDevice);
|
||||||
mListener.usbTransportDiscovered(usbTransport);
|
onDiscoveredUsbDeviceListener.usbTransportDiscovered(usbTransport);
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface OnDiscoveredUsbDeviceListener {
|
public interface OnDiscoveredUsbDeviceListener {
|
||||||
void usbTransportDiscovered(UsbTransport usbTransport);
|
void usbTransportDiscovered(UsbTransport usbTransport);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void requestPermissionForUsbDevice(Context context, UsbDevice usbDevice) {
|
||||||
|
UsbManager usbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
|
||||||
|
if (usbManager == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Intent answerBroadcastIntent = new Intent(ACTION_USB_PERMISSION);
|
||||||
|
answerBroadcastIntent.setPackage(BuildConfig.APPLICATION_ID);
|
||||||
|
PendingIntent answerPendingIntent = PendingIntent.getBroadcast(context, 0, answerBroadcastIntent, 0);
|
||||||
|
|
||||||
|
Timber.d("Requesting permission for " + usbDevice.getDeviceName());
|
||||||
|
usbManager.requestPermission(usbDevice, answerPendingIntent);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -210,9 +210,14 @@ public class UsbTransport implements Transport {
|
|||||||
@Nullable
|
@Nullable
|
||||||
@Override
|
@Override
|
||||||
public TokenType getTokenTypeIfAvailable() {
|
public TokenType getTokenTypeIfAvailable() {
|
||||||
switch (usbDevice.getVendorId()) {
|
return getTokenTypeFromUsbDeviceInfo(usbDevice.getVendorId(), usbDevice.getProductId(), usbConnection.getSerial());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public static TokenType getTokenTypeFromUsbDeviceInfo(int vendorId, int productId, String serialNo) {
|
||||||
|
switch (vendorId) {
|
||||||
case VENDOR_YUBICO: {
|
case VENDOR_YUBICO: {
|
||||||
switch (usbDevice.getProductId()) {
|
switch (productId) {
|
||||||
case PRODUCT_YUBIKEY_NEO_OTP_CCID:
|
case PRODUCT_YUBIKEY_NEO_OTP_CCID:
|
||||||
case PRODUCT_YUBIKEY_NEO_CCID:
|
case PRODUCT_YUBIKEY_NEO_CCID:
|
||||||
case PRODUCT_YUBIKEY_NEO_U2F_CCID:
|
case PRODUCT_YUBIKEY_NEO_U2F_CCID:
|
||||||
@@ -227,11 +232,10 @@ public class UsbTransport implements Transport {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case VENDOR_NITROKEY: {
|
case VENDOR_NITROKEY: {
|
||||||
switch (usbDevice.getProductId()) {
|
switch (productId) {
|
||||||
case PRODUCT_NITROKEY_PRO:
|
case PRODUCT_NITROKEY_PRO:
|
||||||
return TokenType.NITROKEY_PRO;
|
return TokenType.NITROKEY_PRO;
|
||||||
case PRODUCT_NITROKEY_START:
|
case PRODUCT_NITROKEY_START:
|
||||||
String serialNo = usbConnection.getSerial();
|
|
||||||
SecurityTokenInfo.Version gnukVersion = SecurityTokenInfo.parseGnukVersionString(serialNo);
|
SecurityTokenInfo.Version gnukVersion = SecurityTokenInfo.parseGnukVersionString(serialNo);
|
||||||
boolean versionGreaterEquals125 = gnukVersion != null
|
boolean versionGreaterEquals125 = gnukVersion != null
|
||||||
&& SecurityTokenInfo.Version.create("1.2.5").compareTo(gnukVersion) <= 0;
|
&& SecurityTokenInfo.Version.create("1.2.5").compareTo(gnukVersion) <= 0;
|
||||||
@@ -242,7 +246,6 @@ public class UsbTransport implements Transport {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case VENDOR_FSIJ: {
|
case VENDOR_FSIJ: {
|
||||||
String serialNo = usbConnection.getSerial();
|
|
||||||
SecurityTokenInfo.Version gnukVersion = SecurityTokenInfo.parseGnukVersionString(serialNo);
|
SecurityTokenInfo.Version gnukVersion = SecurityTokenInfo.parseGnukVersionString(serialNo);
|
||||||
boolean versionGreaterEquals125 = gnukVersion != null
|
boolean versionGreaterEquals125 = gnukVersion != null
|
||||||
&& SecurityTokenInfo.Version.create("1.2.5").compareTo(gnukVersion) <= 0;
|
&& SecurityTokenInfo.Version.create("1.2.5").compareTo(gnukVersion) <= 0;
|
||||||
@@ -253,8 +256,7 @@ public class UsbTransport implements Transport {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Timber.d("Unknown USB token. Vendor ID: " + usbDevice.getVendorId() + ", Product ID: " +
|
Timber.d("Unknown USB token. Vendor ID: %s, Product ID: %s", vendorId, productId);
|
||||||
usbDevice.getProductId());
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,43 +17,29 @@
|
|||||||
|
|
||||||
package org.sufficientlysecure.keychain.ui;
|
package org.sufficientlysecure.keychain.ui;
|
||||||
|
|
||||||
|
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.app.PendingIntent;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.hardware.usb.UsbDevice;
|
import android.hardware.usb.UsbDevice;
|
||||||
import android.hardware.usb.UsbManager;
|
import android.hardware.usb.UsbManager;
|
||||||
import android.os.Bundle;
|
|
||||||
|
|
||||||
import timber.log.Timber;
|
import org.sufficientlysecure.keychain.securitytoken.UsbConnectionDispatcher;
|
||||||
|
|
||||||
|
|
||||||
public class UsbEventReceiverActivity extends Activity {
|
public class UsbEventReceiverActivity extends Activity {
|
||||||
public static final String ACTION_USB_PERMISSION =
|
|
||||||
"org.sufficientlysecure.keychain.ui.USB_PERMISSION";
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
|
||||||
super.onCreate(savedInstanceState);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
final UsbManager usbManager = (UsbManager) getSystemService(Context.USB_SERVICE);
|
|
||||||
|
|
||||||
Intent intent = getIntent();
|
Intent intent = getIntent();
|
||||||
if (intent != null) {
|
if (intent != null) {
|
||||||
if (UsbManager.ACTION_USB_DEVICE_ATTACHED.equals(intent.getAction())) {
|
if (UsbManager.ACTION_USB_DEVICE_ATTACHED.equals(intent.getAction())) {
|
||||||
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
||||||
|
UsbConnectionDispatcher.requestPermissionForUsbDevice(getApplicationContext(), usbDevice);
|
||||||
Timber.d("Requesting permission for " + usbDevice.getDeviceName());
|
|
||||||
usbManager.requestPermission(usbDevice,
|
|
||||||
PendingIntent.getBroadcast(this, 0, new Intent(ACTION_USB_PERMISSION), 0));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close the activity
|
|
||||||
finish();
|
finish();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -29,7 +29,6 @@ import android.nfc.TagLostException;
|
|||||||
import android.os.AsyncTask;
|
import android.os.AsyncTask;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
|
|
||||||
import nordpol.android.OnDiscoveredTagListener;
|
|
||||||
import nordpol.android.TagDispatcher;
|
import nordpol.android.TagDispatcher;
|
||||||
import nordpol.android.TagDispatcherBuilder;
|
import nordpol.android.TagDispatcherBuilder;
|
||||||
import org.sufficientlysecure.keychain.R;
|
import org.sufficientlysecure.keychain.R;
|
||||||
@@ -56,8 +55,7 @@ import org.sufficientlysecure.keychain.util.Passphrase;
|
|||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
|
|
||||||
public abstract class BaseSecurityTokenActivity extends BaseActivity
|
public abstract class BaseSecurityTokenActivity extends BaseActivity {
|
||||||
implements OnDiscoveredTagListener, UsbConnectionDispatcher.OnDiscoveredUsbDeviceListener {
|
|
||||||
public static final int REQUEST_CODE_PIN = 1;
|
public static final int REQUEST_CODE_PIN = 1;
|
||||||
|
|
||||||
public static final String EXTRA_TAG_HANDLING_ENABLED = "tag_handling_enabled";
|
public static final String EXTRA_TAG_HANDLING_ENABLED = "tag_handling_enabled";
|
||||||
@@ -108,8 +106,7 @@ public abstract class BaseSecurityTokenActivity extends BaseActivity
|
|||||||
onSecurityTokenError(error);
|
onSecurityTokenError(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
private void nfcTagDiscovered(Tag tag) {
|
||||||
public void tagDiscovered(Tag tag) {
|
|
||||||
// Actual NFC operations are executed in doInBackground to not block the UI thread
|
// Actual NFC operations are executed in doInBackground to not block the UI thread
|
||||||
if (!mTagHandlingEnabled) {
|
if (!mTagHandlingEnabled) {
|
||||||
return;
|
return;
|
||||||
@@ -119,8 +116,7 @@ public abstract class BaseSecurityTokenActivity extends BaseActivity
|
|||||||
securityTokenDiscovered(nfcTransport);
|
securityTokenDiscovered(nfcTransport);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
private void usbTransportDiscovered(UsbTransport usbTransport) {
|
||||||
public void usbTransportDiscovered(UsbTransport usbTransport) {
|
|
||||||
// Actual USB operations are executed in doInBackground to not block the UI thread
|
// Actual USB operations are executed in doInBackground to not block the UI thread
|
||||||
if (!mTagHandlingEnabled) {
|
if (!mTagHandlingEnabled) {
|
||||||
return;
|
return;
|
||||||
@@ -184,14 +180,14 @@ public abstract class BaseSecurityTokenActivity extends BaseActivity
|
|||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
|
|
||||||
mNfcTagDispatcher = new TagDispatcherBuilder(this, this)
|
mNfcTagDispatcher = new TagDispatcherBuilder(this, this::nfcTagDiscovered)
|
||||||
.enableUnavailableNfcUserPrompt(false)
|
.enableUnavailableNfcUserPrompt(false)
|
||||||
.enableSounds(true)
|
.enableSounds(true)
|
||||||
.enableDispatchingOnUiThread(true)
|
.enableDispatchingOnUiThread(true)
|
||||||
.enableBroadcomWorkaround(false)
|
.enableBroadcomWorkaround(false)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
mUsbDispatcher = new UsbConnectionDispatcher(this, this);
|
mUsbDispatcher = new UsbConnectionDispatcher(this, this::usbTransportDiscovered);
|
||||||
|
|
||||||
// Check whether we're recreating a previously destroyed instance
|
// Check whether we're recreating a previously destroyed instance
|
||||||
if (savedInstanceState != null) {
|
if (savedInstanceState != null) {
|
||||||
|
|||||||
Reference in New Issue
Block a user