fedilab-Android-App/app/src/main/java/app/fedilab/android/services/LiveNotificationService.java

483 lines
26 KiB
Java
Raw Normal View History

2019-05-18 11:10:30 +02:00
package app.fedilab.android.services;
/* Copyright 2017 Thomas Schneider
*
2019-05-18 11:10:30 +02:00
* This file is a part of Fedilab
*
* This program is free software; you can redistribute it and/or modify it under the terms of the
* GNU General Public License as published by the Free Software Foundation; either version 3 of the
* License, or (at your option) any later version.
*
2019-05-18 11:10:30 +02:00
* Fedilab is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
* Public License for more details.
*
2019-05-18 11:10:30 +02:00
* You should have received a copy of the GNU General Public License along with Fedilab; if not,
* see <http://www.gnu.org/licenses>. */
2017-12-17 14:20:04 +01:00
2018-10-15 19:08:30 +02:00
import android.app.AlarmManager;
import android.app.PendingIntent;
2018-01-03 15:25:35 +01:00
import android.app.Service;
import android.content.Context;
import android.content.Intent;
2018-10-16 18:33:32 +02:00
import android.content.IntentFilter;
import android.content.SharedPreferences;
import android.database.sqlite.SQLiteDatabase;
2017-11-30 18:18:58 +01:00
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
2018-10-13 16:25:08 +02:00
import android.net.Uri;
2018-11-27 18:18:05 +01:00
import android.os.Build;
2017-11-30 18:18:58 +01:00
import android.os.Bundle;
2017-12-02 14:17:06 +01:00
import android.os.Handler;
import android.os.IBinder;
2017-12-02 14:17:06 +01:00
import android.os.Looper;
2018-10-15 19:08:30 +02:00
import android.os.SystemClock;
import android.preference.PreferenceManager;
2019-08-24 17:08:33 +02:00
2019-06-11 19:38:26 +02:00
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
2017-12-02 11:02:25 +01:00
import com.bumptech.glide.Glide;
import com.bumptech.glide.load.DataSource;
import com.bumptech.glide.load.engine.GlideException;
import com.bumptech.glide.request.RequestListener;
import com.bumptech.glide.request.target.SimpleTarget;
import com.bumptech.glide.request.target.Target;
import com.bumptech.glide.request.transition.Transition;
2018-12-08 15:43:12 +01:00
import com.koushikdutta.async.ByteBufferList;
import com.koushikdutta.async.DataEmitter;
import com.koushikdutta.async.callback.CompletedCallback;
2018-12-08 15:43:12 +01:00
import com.koushikdutta.async.callback.DataCallback;
2018-10-13 16:25:08 +02:00
import com.koushikdutta.async.http.AsyncHttpClient;
import com.koushikdutta.async.http.AsyncHttpRequest;
import com.koushikdutta.async.http.Headers;
import com.koushikdutta.async.http.WebSocket;
2017-11-30 18:18:58 +01:00
import org.json.JSONException;
import org.json.JSONObject;
2018-11-27 18:18:05 +01:00
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.util.HashMap;
2018-12-19 10:34:04 +01:00
import java.util.Iterator;
import java.util.List;
2018-12-19 10:34:04 +01:00
import java.util.Map;
import java.util.Objects;
2017-11-30 18:18:58 +01:00
2019-05-18 11:10:30 +02:00
import app.fedilab.android.client.API;
import app.fedilab.android.client.Entities.Account;
import app.fedilab.android.client.Entities.Notification;
import app.fedilab.android.client.TLSSocketFactory;
import app.fedilab.android.helper.Helper;
import app.fedilab.android.sqlite.AccountDAO;
import app.fedilab.android.sqlite.Sqlite;
import app.fedilab.android.R;
import app.fedilab.android.activities.MainActivity;
2017-11-30 18:18:58 +01:00
/**
2017-11-30 07:16:18 +01:00
* Created by Thomas on 29/11/2017.
* Manage service for streaming api and new notifications
*/
2018-10-16 18:33:32 +02:00
public class LiveNotificationService extends Service implements NetworkStateReceiver.NetworkStateReceiverListener {
2018-10-30 13:46:24 +01:00
static {
Helper.installProvider();
}
protected Account account;
boolean backgroundProcess;
2018-12-19 10:34:04 +01:00
private static HashMap<String, Thread> threads = new HashMap<>();
private static HashMap<String, Boolean> canStartStream = new HashMap<>();
2018-10-16 18:33:32 +02:00
private NetworkStateReceiver networkStateReceiver;
2018-12-08 15:43:12 +01:00
private static HashMap<String, WebSocket> webSocketFutures = new HashMap<>();
public void onCreate() {
super.onCreate();
2018-10-16 18:33:32 +02:00
networkStateReceiver = new NetworkStateReceiver();
networkStateReceiver.addListener(this);
registerReceiver(networkStateReceiver, new IntentFilter(android.net.ConnectivityManager.CONNECTIVITY_ACTION));
2018-10-16 18:33:32 +02:00
}
2018-12-19 10:34:04 +01:00
private void startStream(Account account){
SharedPreferences sharedpreferences = getSharedPreferences(Helper.APP_PREFS, Context.MODE_PRIVATE);
backgroundProcess = sharedpreferences.getBoolean(Helper.SET_KEEP_BACKGROUND_PROCESS, true);
2018-10-15 19:08:30 +02:00
boolean liveNotifications = sharedpreferences.getBoolean(Helper.SET_LIVE_NOTIFICATIONS, true);
SQLiteDatabase db = Sqlite.getInstance(getApplicationContext(), Sqlite.DB_NAME, null, Sqlite.DB_VERSION).open();
if( liveNotifications ){
2018-12-19 10:34:04 +01:00
if( account == null){
2019-01-13 14:31:32 +01:00
2018-12-19 10:34:04 +01:00
Iterator it = threads.entrySet().iterator();
Thread thread;
while (it.hasNext()) {
Map.Entry pair = (Map.Entry)it.next();
if( pair.getValue() == null || !((Thread)pair.getValue()).isAlive()) {
if ((pair.getValue()) != null)
((Thread) pair.getValue()).interrupt();
}
}
2019-01-29 14:08:02 +01:00
List<Account> accountStreams = new AccountDAO(getApplicationContext(), db).getAllAccountCrossAction();
2018-12-19 10:34:04 +01:00
if (accountStreams != null) {
for (final Account accountStream : accountStreams) {
if( accountStream.getSocial() == null || accountStream.getSocial().equals("MASTODON") || accountStream.getSocial().equals("PLEROMA")) {
2019-01-13 14:31:32 +01:00
thread = new Thread() {
@Override
public void run() {
2019-08-24 17:08:33 +02:00
2019-01-13 14:31:32 +01:00
taks(accountStream);
}
};
thread.start();
threads.put(accountStream.getAcct() + "@" + accountStream.getInstance(), thread);
}
2018-12-19 10:34:04 +01:00
}
}
}else {
String key = account.getAcct() + "@" + account.getInstance();
2018-12-23 10:51:20 +01:00
if(webSocketFutures.containsKey(key)){
if (webSocketFutures.get(key) != null && Objects.requireNonNull(webSocketFutures.get(key)).isOpen()) {
2018-12-23 10:51:20 +01:00
try {
Objects.requireNonNull(webSocketFutures.get(key)).close();
2019-08-24 17:08:33 +02:00
2018-12-23 10:51:20 +01:00
}catch (Exception ignored){}
}
}
2018-12-19 10:34:04 +01:00
if(threads.containsKey(key)){
if (threads.get(key) != null && !Objects.requireNonNull(threads.get(key)).isAlive()) {
Objects.requireNonNull(threads.get(key)).interrupt();
2018-12-19 10:34:04 +01:00
}
}
Thread thread = new Thread() {
@Override
public void run() {
taks(account);
2018-10-15 19:08:30 +02:00
}
};
thread.start();
2018-12-19 10:34:04 +01:00
threads.put(account.getAcct()+"@"+account.getInstance(), thread);
2018-12-19 10:34:04 +01:00
}
2018-10-15 19:08:30 +02:00
}
}
2018-12-19 18:10:13 +01:00
2018-01-02 14:47:13 +01:00
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
2018-01-03 15:25:35 +01:00
if( intent == null || intent.getBooleanExtra("stop", false) ) {
stopSelf();
}
if( backgroundProcess)
return START_STICKY;
else
return START_NOT_STICKY;
2018-01-03 15:25:35 +01:00
}
2018-10-16 18:33:32 +02:00
@Override
public void onDestroy() {
super.onDestroy();
networkStateReceiver.removeListener(this);
unregisterReceiver(networkStateReceiver);
2018-10-16 18:33:32 +02:00
}
2018-01-03 15:25:35 +01:00
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
2018-10-15 19:08:30 +02:00
@Override
public void onTaskRemoved(Intent rootIntent){
if(backgroundProcess){
restart();
}
super.onTaskRemoved(rootIntent);
}
private void restart(){
Intent restartServiceIntent = new Intent(LiveNotificationService.this, LiveNotificationService.class);
2018-10-15 19:08:30 +02:00
restartServiceIntent.setPackage(getPackageName());
PendingIntent restartServicePendingIntent = PendingIntent.getService(getApplicationContext(), 1, restartServiceIntent, PendingIntent.FLAG_ONE_SHOT);
AlarmManager alarmService = (AlarmManager) getApplicationContext().getSystemService(Context.ALARM_SERVICE);
2018-12-08 15:43:12 +01:00
assert alarmService != null;
2018-10-15 19:08:30 +02:00
alarmService.set(
AlarmManager.ELAPSED_REALTIME,
SystemClock.elapsedRealtime() + 1000,
restartServicePendingIntent);
}
2018-10-13 16:25:08 +02:00
private void taks(Account account) {
if (account != null) {
Headers headers = new Headers();
headers.add("Authorization", "Bearer " + account.getToken());
headers.add("Connection", "Keep-Alive");
headers.add("method", "GET");
headers.add("scheme", "https");
String notif_url = "user:notification";
if( account.getSocial().toUpperCase().equals("PLEROMA"))
notif_url = "user";
String urlKey = "wss://" + account.getInstance() + "/api/v1/streaming/?stream="+notif_url+"&access_token=" + account.getToken();
Uri url = Uri.parse(urlKey);
2018-10-13 16:25:08 +02:00
AsyncHttpRequest.setDefaultHeaders(headers, url);
if( webSocketFutures.containsKey(urlKey) ){
try {
2018-12-08 15:43:12 +01:00
if( webSocketFutures.get(urlKey) != null && webSocketFutures.get(urlKey).isOpen())
Objects.requireNonNull(webSocketFutures.get(urlKey)).close();
2018-12-19 10:34:04 +01:00
} catch (Exception ignored) {}
}
2018-11-27 18:18:05 +01:00
if( Build.VERSION.SDK_INT <= Build.VERSION_CODES.KITKAT ) {
try {
2019-05-23 07:46:34 +02:00
AsyncHttpClient.getDefaultInstance().getSSLSocketMiddleware().setSSLContext(new TLSSocketFactory(account.getInstance()).getSSLContext());
2018-11-27 18:18:05 +01:00
AsyncHttpClient.getDefaultInstance().getSSLSocketMiddleware().setConnectAllAddresses(true);
2018-12-19 18:30:54 +01:00
} catch (KeyManagementException e) {
e.printStackTrace();
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
2018-11-27 18:18:05 +01:00
}
}
AsyncHttpClient.getDefaultInstance().websocket("wss://" + account.getInstance() + "/api/v1/streaming/?stream="+notif_url+"&access_token=" + account.getToken(), "wss", new AsyncHttpClient.WebSocketConnectCallback() {
@Override
public void onCompleted(Exception ex, WebSocket webSocket) {
2018-12-08 15:43:12 +01:00
webSocketFutures.put(account.getAcct()+"@"+account.getInstance(), webSocket);
if (ex != null) {
if( !canStartStream.containsKey(account.getAcct()+"@"+account.getInstance()) || canStartStream.get(account.getAcct()+"@"+account.getInstance())) {
canStartStream.put(account.getAcct()+"@"+account.getInstance(),false);
2019-08-19 12:44:09 +02:00
2019-08-20 09:38:58 +02:00
Handler handler = new Handler(Looper.getMainLooper());
handler.postDelayed(new Runnable() {
@Override
public void run() {
startStream(account);
canStartStream.put(account.getAcct()+"@"+account.getInstance(),true);
}
2019-08-24 17:08:33 +02:00
}, 60000 );
}
return;
2018-10-13 18:52:56 +02:00
}
webSocket.setStringCallback(new WebSocket.StringCallback() {
public void onStringAvailable(String s) {
try {
JSONObject eventJson = new JSONObject(s);
onRetrieveStreaming(account, eventJson);
} catch (JSONException ignored) {}
}
});
webSocket.setClosedCallback(new CompletedCallback() {
@Override
public void onCompleted(Exception ex) {
2018-12-08 15:43:12 +01:00
try {
2018-12-19 10:34:04 +01:00
if (ex != null) {
2018-12-08 15:43:12 +01:00
webSocket.close();
2018-12-19 10:34:04 +01:00
}
2018-12-08 15:43:12 +01:00
} finally {
if( webSocketFutures != null && webSocketFutures.containsKey(urlKey))
webSocketFutures.remove(urlKey);
taks(account);
if( networkStateReceiver != null && networkStateReceiver.listeners.size() == 0){
networkStateReceiver.addListener(LiveNotificationService.this);
registerReceiver(networkStateReceiver, new IntentFilter(android.net.ConnectivityManager.CONNECTIVITY_ACTION));
}
}
2018-12-08 15:43:12 +01:00
}
});
webSocket.setDataCallback(new DataCallback() {
public void onDataAvailable(DataEmitter emitter, ByteBufferList byteBufferList) {
// note that this data has been read
byteBufferList.recycle();
}
});
}
});
2018-12-08 15:43:12 +01:00
2018-10-13 18:52:56 +02:00
2017-11-30 18:18:58 +01:00
}
}
2018-10-13 16:25:08 +02:00
private void onRetrieveStreaming(Account account, JSONObject response) {
2017-11-30 18:18:58 +01:00
if( response == null )
return;
final Notification notification;
2018-12-19 10:34:04 +01:00
String dataId;
2017-11-30 18:18:58 +01:00
Bundle b = new Bundle();
2018-10-12 18:12:10 +02:00
boolean canSendBroadCast = true;
Helper.EventStreaming event;
2018-11-17 14:31:03 +01:00
final SharedPreferences sharedpreferences = getSharedPreferences(Helper.APP_PREFS, Context.MODE_PRIVATE);
String userId = sharedpreferences.getString(Helper.PREF_KEY_ID, null);
2018-10-13 16:25:08 +02:00
try {
2018-10-13 17:41:20 +02:00
switch (response.get("event").toString()) {
case "notification":
event = Helper.EventStreaming.NOTIFICATION;
notification = API.parseNotificationResponse(getApplicationContext(), new JSONObject(response.get("payload").toString()));
b.putParcelable("data", notification);
boolean liveNotifications = sharedpreferences.getBoolean(Helper.SET_LIVE_NOTIFICATIONS, true);
boolean canNotify = Helper.canNotify(getApplicationContext());
boolean notify = sharedpreferences.getBoolean(Helper.SET_NOTIFY, true);
String targeted_account = null;
Helper.NotifType notifType = Helper.NotifType.MENTION;
2018-10-15 19:08:30 +02:00
boolean activityRunning = PreferenceManager.getDefaultSharedPreferences(this).getBoolean("isMainActivityRunning", false);
if ((userId == null || !userId.equals(account.getId()) || !activityRunning) && liveNotifications && canNotify && notify) {
2018-10-13 17:41:20 +02:00
boolean notif_follow = sharedpreferences.getBoolean(Helper.SET_NOTIF_FOLLOW, true);
boolean notif_add = sharedpreferences.getBoolean(Helper.SET_NOTIF_ADD, true);
boolean notif_mention = sharedpreferences.getBoolean(Helper.SET_NOTIF_MENTION, true);
boolean notif_share = sharedpreferences.getBoolean(Helper.SET_NOTIF_SHARE, true);
boolean notif_poll = sharedpreferences.getBoolean(Helper.SET_NOTIF_POLL, true);
boolean somethingToPush = (notif_follow || notif_add || notif_mention || notif_share || notif_poll);
2018-10-13 17:41:20 +02:00
String title = null;
if (somethingToPush && notification != null) {
switch (notification.getType()) {
case "mention":
notifType = Helper.NotifType.MENTION;
if (notif_mention) {
if (notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0)
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true), getString(R.string.notif_mention));
else
title = String.format("@%s %s", notification.getAccount().getAcct(), getString(R.string.notif_mention));
} else {
canSendBroadCast = false;
}
break;
case "reblog":
notifType = Helper.NotifType.BOOST;
if (notif_share) {
if (notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0)
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true), getString(R.string.notif_reblog));
else
title = String.format("@%s %s", notification.getAccount().getAcct(), getString(R.string.notif_reblog));
} else {
canSendBroadCast = false;
}
break;
case "favourite":
notifType = Helper.NotifType.FAV;
if (notif_add) {
if (notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0)
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true), getString(R.string.notif_favourite));
else
title = String.format("@%s %s", notification.getAccount().getAcct(), getString(R.string.notif_favourite));
} else {
canSendBroadCast = false;
}
break;
case "follow":
notifType = Helper.NotifType.FOLLLOW;
if (notif_follow) {
if (notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0)
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true), getString(R.string.notif_follow));
else
title = String.format("@%s %s", notification.getAccount().getAcct(), getString(R.string.notif_follow));
targeted_account = notification.getAccount().getId();
} else {
canSendBroadCast = false;
}
break;
case "poll":
notifType = Helper.NotifType.POLL;
if (notif_poll) {
if (notification.getAccount().getId() != null && notification.getAccount().getId().equals(userId))
title = getString(R.string.notif_poll_self);
else
title = getString(R.string.notif_poll);
} else {
canSendBroadCast = false;
}
break;
2018-10-13 17:41:20 +02:00
default:
}
//Some others notification
final Intent intent = new Intent(getApplicationContext(), MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_NEW_TASK);
2019-05-18 11:10:30 +02:00
intent.putExtra(Helper.INTENT_ACTION, Helper.NOTIFICATION_INTENT);
intent.putExtra(Helper.PREF_KEY_ID, account.getId());
2019-06-05 14:35:42 +02:00
intent.putExtra(Helper.PREF_INSTANCE, account.getInstance());
2018-10-24 15:19:33 +02:00
if (targeted_account != null) {
2019-05-18 11:10:30 +02:00
intent.putExtra(Helper.INTENT_TARGETED_ACCOUNT, targeted_account);
2018-10-24 15:19:33 +02:00
}
final String finalTitle = title;
Handler mainHandler = new Handler(Looper.getMainLooper());
Helper.NotifType finalNotifType = notifType;
Runnable myRunnable = new Runnable() {
@Override
public void run() {
if (finalTitle != null) {
Glide.with(getApplicationContext())
.asBitmap()
.load(notification.getAccount().getAvatar())
.listener(new RequestListener<Bitmap>() {
@Override
public boolean onResourceReady(Bitmap resource, Object model, Target<Bitmap> target, DataSource dataSource, boolean isFirstResource) {
return false;
}
2018-10-13 17:41:20 +02:00
@Override
public boolean onLoadFailed(@Nullable GlideException e, Object model, Target target, boolean isFirstResource) {
2019-08-01 18:15:11 +02:00
Helper.notify_user(getApplicationContext(),account, intent, BitmapFactory.decodeResource(getResources(),
R.drawable.mastodonlogo), finalNotifType, finalTitle, "@" + account.getAcct() + "@" + account.getInstance());
String lastNotif = sharedpreferences.getString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), null);
2019-08-08 12:10:29 +02:00
if (lastNotif == null || notification.getId().compareTo(lastNotif) >= 1) {
SharedPreferences.Editor editor = sharedpreferences.edit();
editor.putString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), notification.getId());
editor.apply();
2018-10-13 16:25:08 +02:00
}
return false;
}
})
.into(new SimpleTarget<Bitmap>() {
@Override
public void onResourceReady(@NonNull Bitmap resource, Transition<? super Bitmap> transition) {
2019-08-01 17:41:59 +02:00
2019-08-01 18:15:11 +02:00
Helper.notify_user(getApplicationContext(), account,intent, resource, finalNotifType, finalTitle, "@" + account.getAcct() + "@" + account.getInstance());
String lastNotif = sharedpreferences.getString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), null);
2019-08-08 12:10:29 +02:00
if (lastNotif == null || notification.getId().compareTo(lastNotif) >= 1) {
SharedPreferences.Editor editor = sharedpreferences.edit();
editor.putString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), notification.getId());
editor.apply();
2018-10-13 17:41:20 +02:00
}
}
});
2018-10-13 16:25:08 +02:00
}
}
};
mainHandler.post(myRunnable);
2018-10-13 16:25:08 +02:00
}
2017-11-30 18:18:58 +01:00
}
2018-12-23 10:43:45 +01:00
if( canSendBroadCast) {
if (account != null)
b.putString("userIdService", account.getId());
Intent intentBC = new Intent(Helper.RECEIVE_DATA);
intentBC.putExtra("eventStreaming", event);
intentBC.putExtras(b);
b.putParcelable("data", notification);
LocalBroadcastManager.getInstance(getApplicationContext()).sendBroadcast(intentBC);
}
2018-10-13 17:41:20 +02:00
break;
case "delete":
event = Helper.EventStreaming.DELETE;
try {
dataId = response.getString("id");
b.putString("dataId", dataId);
2018-12-19 08:40:37 +01:00
} catch (JSONException ignored) { }
2018-10-13 17:41:20 +02:00
break;
2017-11-30 18:18:58 +01:00
}
2018-12-19 10:34:04 +01:00
} catch (Exception ignored) { }
2017-11-30 18:18:58 +01:00
}
2018-10-16 18:33:32 +02:00
@Override
public void networkAvailable() {
2018-12-19 10:34:04 +01:00
startStream(null);
2018-10-16 18:33:32 +02:00
}
@Override
public void networkUnavailable() {
2018-12-19 10:34:04 +01:00
Iterator it = threads.entrySet().iterator();
while (it.hasNext()) {
Map.Entry pair = (Map.Entry) it.next();
if (pair.getValue() == null || !((Thread) pair.getValue()).isAlive()) {
if ((pair.getValue()) != null)
((Thread) pair.getValue()).interrupt();
}
it.remove();
}
2018-10-16 18:33:32 +02:00
}
}