package nodomain.freeyourgadget.gadgetbridge.service; import android.app.NotificationManager; import android.app.Service; import android.content.BroadcastReceiver; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.SharedPreferences; import android.database.Cursor; import android.net.Uri; import android.os.IBinder; import android.preference.PreferenceManager; import android.provider.ContactsContract; import android.support.v4.content.LocalBroadcastManager; import android.widget.Toast; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.ArrayList; import java.util.UUID; import nodomain.freeyourgadget.gadgetbridge.R; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.model.Alarm; import nodomain.freeyourgadget.gadgetbridge.model.ServiceCommand; import nodomain.freeyourgadget.gadgetbridge.util.GB; import static nodomain.freeyourgadget.gadgetbridge.model.DeviceService.*; public class DeviceCommunicationService extends Service { private static final Logger LOG = LoggerFactory.getLogger(DeviceCommunicationService.class); private boolean mStarted = false; private GBDevice mGBDevice = null; private DeviceSupport mDeviceSupport; private BroadcastReceiver mReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (action.equals(GBDevice.ACTION_DEVICE_CHANGED)) { GBDevice device = intent.getParcelableExtra(GBDevice.EXTRA_DEVICE); if (mGBDevice.equals(device)) { mGBDevice = device; boolean enableReceivers = mDeviceSupport != null && (mDeviceSupport.useAutoConnect() || mGBDevice.isInitialized()); GB.setReceiversEnableState(enableReceivers, context); GB.updateNotification(mGBDevice.getName() + " " + mGBDevice.getStateString(), context); } else { LOG.error("Got ACTION_DEVICE_CHANGED from unexpected device: " + mGBDevice); } } } }; @Override public void onCreate() { LOG.debug("DeviceCommunicationService is being created"); super.onCreate(); LocalBroadcastManager.getInstance(this).registerReceiver(mReceiver, new IntentFilter(GBDevice.ACTION_DEVICE_CHANGED)); } @Override public int onStartCommand(Intent intent, int flags, int startId) { if (intent == null) { LOG.info("no intent"); return START_NOT_STICKY; } String action = intent.getAction(); boolean pair = intent.getBooleanExtra(EXTRA_PERFORM_PAIR, false); if (action == null) { LOG.info("no action"); return START_NOT_STICKY; } LOG.debug("Service startcommand: " + action); if (!action.equals(ACTION_START) && !action.equals(ACTION_CONNECT)) { if (!mStarted) { // using the service before issuing ACTION_START LOG.info("Must start service with " + ACTION_START + " or " + ACTION_CONNECT + " before using it: " + action); return START_NOT_STICKY; } if (mDeviceSupport == null || (!isInitialized() && !mDeviceSupport.useAutoConnect())) { // trying to send notification without valid Bluetooth connection if (mGBDevice != null) { // at least send back the current device state mGBDevice.sendDeviceUpdateIntent(this); } return START_STICKY; } } // when we get past this, we should have valid mDeviceSupport and mGBDevice instances switch (action) { case ACTION_START: start(); break; case ACTION_CONNECT: start(); // ensure started String btDeviceAddress = intent.getStringExtra(EXTRA_DEVICE_ADDRESS); SharedPreferences sharedPrefs = PreferenceManager.getDefaultSharedPreferences(this); if (sharedPrefs != null) { // may be null in test cases if (btDeviceAddress == null) { btDeviceAddress = sharedPrefs.getString("last_device_address", null); } else { sharedPrefs.edit().putString("last_device_address", btDeviceAddress).apply(); } } if (btDeviceAddress != null && !isConnecting() && !isConnected()) { if (mDeviceSupport != null) { mDeviceSupport.dispose(); mDeviceSupport = null; } try { DeviceSupportFactory factory = new DeviceSupportFactory(this); mDeviceSupport = factory.createDeviceSupport(btDeviceAddress); if (mDeviceSupport != null) { mGBDevice = mDeviceSupport.getDevice(); if (pair) { mDeviceSupport.pair(); } else { mDeviceSupport.connect(); } } } catch (Exception e) { GB.toast(this, getString(R.string.cannot_connect, e.getMessage()), Toast.LENGTH_SHORT, GB.ERROR); mDeviceSupport = null; mGBDevice = null; } } else if (mGBDevice != null) { // send an update at least mGBDevice.sendDeviceUpdateIntent(this); } break; case ACTION_REQUEST_DEVICEINFO: mGBDevice.sendDeviceUpdateIntent(this); break; case ACTION_NOTIFICATION_GENERIC: { String title = intent.getStringExtra(EXTRA_NOTIFICATION_TITLE); String body = intent.getStringExtra(EXTRA_NOTIFICATION_BODY); mDeviceSupport.onGenericNotification(title, body); break; } case ACTION_NOTIFICATION_SMS: { String sender = intent.getStringExtra(EXTRA_NOTIFICATION_SENDER); String body = intent.getStringExtra(EXTRA_NOTIFICATION_BODY); String senderName = getContactDisplayNameByNumber(sender); mDeviceSupport.onSMS(senderName, body); break; } case ACTION_NOTIFICATION_EMAIL: { String sender = intent.getStringExtra(EXTRA_NOTIFICATION_SENDER); String subject = intent.getStringExtra(EXTRA_NOTIFICATION_SUBJECT); String body = intent.getStringExtra(EXTRA_NOTIFICATION_BODY); mDeviceSupport.onEmail(sender, subject, body); break; } case ACTION_REBOOT: { mDeviceSupport.onReboot(); break; } case ACTION_FETCH_ACTIVITY_DATA: { mDeviceSupport.onFetchActivityData(); break; } case ACTION_DISCONNECT: { mDeviceSupport.dispose(); mDeviceSupport = null; break; } case ACTION_FIND_DEVICE: { boolean start = intent.getBooleanExtra(EXTRA_FIND_START, false); mDeviceSupport.onFindDevice(start); break; } case ACTION_CALLSTATE: ServiceCommand command = (ServiceCommand) intent.getSerializableExtra(EXTRA_CALL_COMMAND); String phoneNumber = intent.getStringExtra(EXTRA_CALL_PHONENUMBER); String callerName = null; if (phoneNumber != null) { callerName = getContactDisplayNameByNumber(phoneNumber); } mDeviceSupport.onSetCallState(phoneNumber, callerName, command); break; case ACTION_SETTIME: mDeviceSupport.onSetTime(); break; case ACTION_SETMUSICINFO: String artist = intent.getStringExtra(EXTRA_MUSIC_ARTIST); String album = intent.getStringExtra(EXTRA_MUSIC_ALBUM); String track = intent.getStringExtra(EXTRA_MUSIC_TRACK); mDeviceSupport.onSetMusicInfo(artist, album, track); break; case ACTION_REQUEST_APPINFO: mDeviceSupport.onAppInfoReq(); break; case ACTION_REQUEST_SCREENSHOT: mDeviceSupport.onScreenshotReq(); break; case ACTION_STARTAPP: { UUID uuid = (UUID) intent.getSerializableExtra(EXTRA_APP_UUID); mDeviceSupport.onAppStart(uuid); break; } case ACTION_DELETEAPP: { UUID uuid = (UUID) intent.getSerializableExtra(EXTRA_APP_UUID); mDeviceSupport.onAppDelete(uuid); break; } case ACTION_INSTALL: Uri uri = intent.getParcelableExtra(EXTRA_URI); if (uri != null) { LOG.info("will try to install app/fw"); mDeviceSupport.onInstallApp(uri); } break; case ACTION_SET_ALARMS: ArrayList alarms = intent.getParcelableArrayListExtra(EXTRA_ALARMS); mDeviceSupport.onSetAlarms(alarms); break; } return START_STICKY; } private void start() { if (!mStarted) { startForeground(GB.NOTIFICATION_ID, GB.createNotification(getString(R.string.gadgetbridge_running), this)); mStarted = true; } } private boolean isConnected() { return mGBDevice != null && mGBDevice.isConnected(); } private boolean isConnecting() { return mGBDevice != null && mGBDevice.isConnecting(); } private boolean isInitialized() { return mGBDevice != null && mGBDevice.isInitialized(); } @Override public void onDestroy() { LOG.debug("DeviceCommunicationService is being destroyed"); super.onDestroy(); LocalBroadcastManager.getInstance(this).unregisterReceiver(mReceiver); GB.setReceiversEnableState(false, this); // disable BroadcastReceivers if (mDeviceSupport != null) { mDeviceSupport.dispose(); } NotificationManager nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); nm.cancel(GB.NOTIFICATION_ID); // need to do this because the updated notification wont be cancelled when service stops } @Override public IBinder onBind(Intent intent) { return null; } private String getContactDisplayNameByNumber(String number) { Uri uri = Uri.withAppendedPath(ContactsContract.PhoneLookup.CONTENT_FILTER_URI, Uri.encode(number)); String name = number; if (number == null || number.equals("")) { return name; } ContentResolver contentResolver = getContentResolver(); Cursor contactLookup = contentResolver.query(uri, null, null, null, null); try { if (contactLookup != null && contactLookup.getCount() > 0) { contactLookup.moveToNext(); name = contactLookup.getString(contactLookup.getColumnIndex(ContactsContract.Data.DISPLAY_NAME)); } } finally { if (contactLookup != null) { contactLookup.close(); } } return name; } }