Compare commits
7 Commits
1938470b43
...
cbf7f3eeea
| Author | SHA1 | Date | |
|---|---|---|---|
| cbf7f3eeea | |||
| 935ded253b | |||
| f95229b421 | |||
| fef6a08ea6 | |||
| 67900cd09a | |||
| 1e71f67c59 | |||
| 003238da4c |
@@ -49,6 +49,10 @@
|
|||||||
<action android:name="android.provider.Telephony.SMS_RECEIVED" />
|
<action android:name="android.provider.Telephony.SMS_RECEIVED" />
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
</receiver>
|
</receiver>
|
||||||
|
<receiver android:name=".broadcast.HeartbeatAlarmReceiver"
|
||||||
|
android:enabled="true"
|
||||||
|
android:exported="true">
|
||||||
|
</receiver>
|
||||||
<receiver android:name=".broadcast.PhoneCallReceiver"
|
<receiver android:name=".broadcast.PhoneCallReceiver"
|
||||||
android:enabled="true"
|
android:enabled="true"
|
||||||
android:exported="true">
|
android:exported="true">
|
||||||
|
|||||||
@@ -1,20 +1,10 @@
|
|||||||
package com.nbee.echolink;
|
package com.nbee.echolink;
|
||||||
|
|
||||||
import android.app.Application;
|
import android.app.Application;
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import androidx.work.ExistingPeriodicWorkPolicy;
|
|
||||||
import androidx.work.PeriodicWorkRequest;
|
|
||||||
import androidx.work.WorkManager;
|
|
||||||
|
|
||||||
import com.nbee.echolink.utils.HeartbeatWorker;
|
|
||||||
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
public class EchoLink extends Application {
|
public class EchoLink extends Application {
|
||||||
private static final int EXECUTION_INTERVAL = 10; // 示例:15分钟
|
|
||||||
@Override
|
@Override
|
||||||
public void onCreate() {
|
public void onCreate() {
|
||||||
super.onCreate();
|
super.onCreate();
|
||||||
@@ -24,17 +14,6 @@ public class EchoLink extends Application {
|
|||||||
// 打印日志文件的路径
|
// 打印日志文件的路径
|
||||||
Timber.d("Log file path: " + this.getFilesDir() + "/logs/echoLink.log");
|
Timber.d("Log file path: " + this.getFilesDir() + "/logs/echoLink.log");
|
||||||
// 安排定时任务
|
// 安排定时任务
|
||||||
doScheduledTask(this);
|
// doScheduledTask(this);
|
||||||
}
|
|
||||||
private void doScheduledTask(Context context){
|
|
||||||
Timber.d("start do scheduled task");
|
|
||||||
PeriodicWorkRequest heartbeatWorkRequest =
|
|
||||||
new PeriodicWorkRequest.Builder(HeartbeatWorker.class, EXECUTION_INTERVAL, TimeUnit.MINUTES)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
// 安排周期性工作
|
|
||||||
WorkManager.getInstance(context).enqueueUniquePeriodicWork("HeartbeatWork",
|
|
||||||
ExistingPeriodicWorkPolicy.KEEP,
|
|
||||||
heartbeatWorkRequest);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,22 +1,17 @@
|
|||||||
package com.nbee.echolink;
|
package com.nbee.echolink;
|
||||||
|
|
||||||
import static java.lang.Thread.*;
|
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.os.AsyncTask;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.os.PowerManager;
|
import android.os.PowerManager;
|
||||||
import android.view.View;
|
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
|
|
||||||
import androidx.appcompat.app.AppCompatActivity;
|
import androidx.appcompat.app.AppCompatActivity;
|
||||||
import androidx.work.ExistingPeriodicWorkPolicy;
|
|
||||||
import androidx.work.PeriodicWorkRequest;
|
|
||||||
import androidx.work.WorkManager;
|
|
||||||
|
|
||||||
|
import com.nbee.echolink.async.BatteryOptimizationTask;
|
||||||
import com.nbee.echolink.service.MonitorService;
|
import com.nbee.echolink.service.MonitorService;
|
||||||
import com.nbee.echolink.utils.BatteryOptimizationUtil;
|
import com.nbee.echolink.utils.BatteryOptimizationUtil;
|
||||||
import com.nbee.echolink.utils.HeartbeatWorker;
|
|
||||||
import com.nbee.echolink.utils.PermissionUtils;
|
import com.nbee.echolink.utils.PermissionUtils;
|
||||||
import com.nbee.echolink.utils.ShellUtils;
|
import com.nbee.echolink.utils.ShellUtils;
|
||||||
|
|
||||||
@@ -24,7 +19,6 @@ import java.io.BufferedReader;
|
|||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileReader;
|
import java.io.FileReader;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
@@ -40,16 +34,13 @@ public class MainActivity extends AppCompatActivity {
|
|||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
setContentView(R.layout.activity_main);
|
setContentView(R.layout.activity_main);
|
||||||
|
|
||||||
isRoot = shellUtils.checkRootPermission();
|
|
||||||
if (!isRoot){
|
|
||||||
Timber.d("未获取root权限: "+ isRoot);
|
|
||||||
}
|
|
||||||
logTextView = findViewById(R.id.logTextView);
|
logTextView = findViewById(R.id.logTextView);
|
||||||
updateLog("初始化日志...");
|
updateLog("初始化日志...");
|
||||||
|
|
||||||
requestIgnoreBattery();
|
requestIgnoreBatteryAsync();
|
||||||
permissionUtils = new PermissionUtils(this);
|
permissionUtils = new PermissionUtils(this);
|
||||||
permissionUtils.checkPermissions();
|
permissionUtils.checkPermissions();
|
||||||
|
|
||||||
// 启动MonitorService服务
|
// 启动MonitorService服务
|
||||||
|
|
||||||
Intent serviceIntent = new Intent(this, MonitorService.class);
|
Intent serviceIntent = new Intent(this, MonitorService.class);
|
||||||
@@ -65,19 +56,26 @@ public class MainActivity extends AppCompatActivity {
|
|||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
String logContent = readLogFile();
|
readLogFileAsync();
|
||||||
updateLog(logContent);
|
|
||||||
}
|
}
|
||||||
private void requestIgnoreBattery(){
|
|
||||||
Context context = getApplicationContext();
|
|
||||||
PowerManager powerManager = (PowerManager) getSystemService(POWER_SERVICE);
|
private void readLogFileAsync() {
|
||||||
boolean hasIgnored = powerManager.isIgnoringBatteryOptimizations(context.getPackageName());
|
new AsyncTask<Void, Void, String>() {
|
||||||
if(!hasIgnored){
|
@Override
|
||||||
BatteryOptimizationUtil.requestIgnoreBatteryOptimization(context);
|
protected String doInBackground(Void... voids) {
|
||||||
Timber.d("不在在优化电池白名单内");
|
return readLogFile();
|
||||||
} else {
|
}
|
||||||
return;
|
|
||||||
}
|
@Override
|
||||||
|
protected void onPostExecute(String logContent) {
|
||||||
|
updateLog(logContent);
|
||||||
|
}
|
||||||
|
}.execute();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void requestIgnoreBatteryAsync() {
|
||||||
|
new BatteryOptimizationTask(this).execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void updateLog(String newLog) {
|
private void updateLog(String newLog) {
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
package com.nbee.echolink.async;
|
||||||
|
|
||||||
|
import static android.content.Context.POWER_SERVICE;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.os.AsyncTask;
|
||||||
|
import android.os.PowerManager;
|
||||||
|
|
||||||
|
import com.nbee.echolink.utils.BatteryOptimizationUtil;
|
||||||
|
|
||||||
|
import timber.log.Timber;
|
||||||
|
|
||||||
|
public class BatteryOptimizationTask extends AsyncTask<Void, Void, Void> {
|
||||||
|
private Context context;
|
||||||
|
|
||||||
|
public BatteryOptimizationTask(Context context) {
|
||||||
|
this.context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Void doInBackground(Void... voids) {
|
||||||
|
PowerManager powerManager = (PowerManager) context.getSystemService(POWER_SERVICE);
|
||||||
|
boolean hasIgnored = powerManager.isIgnoringBatteryOptimizations(context.getPackageName());
|
||||||
|
if (!hasIgnored) {
|
||||||
|
BatteryOptimizationUtil.requestIgnoreBatteryOptimization(context);
|
||||||
|
Timber.d("请求忽略电池优化");
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,17 +1,15 @@
|
|||||||
package com.nbee.echolink.utils;
|
package com.nbee.echolink.broadcast;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import android.content.BroadcastReceiver;
|
||||||
import androidx.work.Worker;
|
|
||||||
import androidx.work.WorkerParameters;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.Handler;
|
import android.content.Intent;
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import com.google.gson.Gson;
|
import com.google.gson.Gson;
|
||||||
import com.google.gson.JsonSyntaxException;
|
import com.google.gson.JsonSyntaxException;
|
||||||
import com.nbee.echolink.R;
|
import com.nbee.echolink.R;
|
||||||
import com.nbee.echolink.model.DeviceInfo;
|
import com.nbee.echolink.model.DeviceInfo;
|
||||||
import com.nbee.echolink.response.ApiResponse;
|
import com.nbee.echolink.response.ApiResponse;
|
||||||
|
import com.nbee.echolink.utils.DeviceInfoUtils;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
@@ -21,24 +19,16 @@ import okhttp3.Request;
|
|||||||
import okhttp3.RequestBody;
|
import okhttp3.RequestBody;
|
||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
public class HeartbeatWorker extends Worker {
|
public class HeartbeatAlarmReceiver extends BroadcastReceiver {
|
||||||
private final Context context;
|
private Context context;
|
||||||
private String accessToken;
|
private String accessToken;
|
||||||
private String SN;
|
private String SN;
|
||||||
|
|
||||||
|
|
||||||
public HeartbeatWorker(@NonNull Context context, @NonNull WorkerParameters workerParams) {
|
|
||||||
super(context, workerParams);
|
|
||||||
this.context = context; // 保存上下文引用
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
@Override
|
@Override
|
||||||
public Result doWork() {
|
public void onReceive(Context context1, Intent intent) {
|
||||||
Context context = getApplicationContext();
|
// 在这里执行发送心跳的逻辑
|
||||||
// 在这里实现您的心跳发送逻辑
|
context = context1;
|
||||||
|
// 例如,你可以启动一个服务或者直接在这里执行网络请求
|
||||||
sendHeartbeatSignal();
|
sendHeartbeatSignal();
|
||||||
return Result.success();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendHeartbeatSignal() {
|
private void sendHeartbeatSignal() {
|
||||||
@@ -81,7 +71,6 @@ public class HeartbeatWorker extends Worker {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private <T>Request buildRequest(String apiUrl){
|
private <T>Request buildRequest(String apiUrl){
|
||||||
SN = context.getResources().getString(R.string.SN);
|
SN = context.getResources().getString(R.string.SN);
|
||||||
accessToken = context.getResources().getString(R.string.access_token);
|
accessToken = context.getResources().getString(R.string.access_token);
|
||||||
@@ -1,9 +1,12 @@
|
|||||||
package com.nbee.echolink.service;
|
package com.nbee.echolink.service;
|
||||||
|
|
||||||
|
import android.app.AlarmManager;
|
||||||
import android.app.Notification;
|
import android.app.Notification;
|
||||||
import android.app.NotificationChannel;
|
import android.app.NotificationChannel;
|
||||||
import android.app.NotificationManager;
|
import android.app.NotificationManager;
|
||||||
|
import android.app.PendingIntent;
|
||||||
import android.app.Service;
|
import android.app.Service;
|
||||||
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.os.IBinder;
|
import android.os.IBinder;
|
||||||
@@ -12,14 +15,25 @@ import android.util.Log;
|
|||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
import androidx.core.app.NotificationCompat;
|
import androidx.core.app.NotificationCompat;
|
||||||
|
|
||||||
|
import com.google.gson.Gson;
|
||||||
|
import com.google.gson.JsonSyntaxException;
|
||||||
import com.nbee.echolink.R;
|
import com.nbee.echolink.R;
|
||||||
|
import com.nbee.echolink.broadcast.HeartbeatAlarmReceiver;
|
||||||
import com.nbee.echolink.model.CallInfo;
|
import com.nbee.echolink.model.CallInfo;
|
||||||
|
import com.nbee.echolink.model.DeviceInfo;
|
||||||
import com.nbee.echolink.model.SMSInfo;
|
import com.nbee.echolink.model.SMSInfo;
|
||||||
|
import com.nbee.echolink.response.ApiResponse;
|
||||||
|
import com.nbee.echolink.utils.DeviceInfoUtils;
|
||||||
import com.nbee.echolink.utils.NetworkUtil;
|
import com.nbee.echolink.utils.NetworkUtil;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
import java.text.SimpleDateFormat;
|
import java.text.SimpleDateFormat;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
|
||||||
|
import okhttp3.MediaType;
|
||||||
|
import okhttp3.OkHttpClient;
|
||||||
|
import okhttp3.Request;
|
||||||
|
import okhttp3.RequestBody;
|
||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
public class MonitorService extends Service {
|
public class MonitorService extends Service {
|
||||||
@@ -27,18 +41,17 @@ public class MonitorService extends Service {
|
|||||||
private static final String TAG = "MonitorService";
|
private static final String TAG = "MonitorService";
|
||||||
private String lastPhoneNumber = null;
|
private String lastPhoneNumber = null;
|
||||||
private long lastCallTime = 0;
|
private long lastCallTime = 0;
|
||||||
private int count;
|
private static final long HEARTBEAT_INTERVAL_MINUTES = 10; // 10分钟
|
||||||
@Override
|
@Override
|
||||||
public void onCreate() {
|
public void onCreate() {
|
||||||
super.onCreate();
|
super.onCreate();
|
||||||
Timber.d("MonitorService onCreate");
|
Timber.d("监控服务 onCreate");
|
||||||
startForegroundService();
|
startForegroundService();
|
||||||
networkUtil = new NetworkUtil(this);
|
networkUtil = new NetworkUtil(this);
|
||||||
|
scheduleHeartbeat(); // 启动心跳定时任务
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int onStartCommand(Intent intent, int flags, int startId) {
|
public int onStartCommand(Intent intent, int flags, int startId) {
|
||||||
Timber.d("MonitorService onStartCommand");
|
|
||||||
if (intent != null) {
|
if (intent != null) {
|
||||||
if (intent.hasExtra("incomingNumber")) {
|
if (intent.hasExtra("incomingNumber")) {
|
||||||
String incomingNumber = intent.getStringExtra("incomingNumber");
|
String incomingNumber = intent.getStringExtra("incomingNumber");
|
||||||
@@ -126,4 +139,20 @@ public class MonitorService extends Service {
|
|||||||
Timber.d("MonitorService onBind");
|
Timber.d("MonitorService onBind");
|
||||||
return null; // 不提供绑定服务的接口
|
return null; // 不提供绑定服务的接口
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void scheduleHeartbeat() {
|
||||||
|
AlarmManager alarmManager = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
|
||||||
|
Intent intent = new Intent(this, HeartbeatAlarmReceiver.class); // HeartbeatAlarmReceiver是一个BroadcastReceiver
|
||||||
|
PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0, intent, 0);
|
||||||
|
|
||||||
|
long intervalMillis = HEARTBEAT_INTERVAL_MINUTES * 60 * 1000; // 10分钟的毫秒数
|
||||||
|
|
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
||||||
|
alarmManager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + intervalMillis, pendingIntent);
|
||||||
|
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
|
||||||
|
alarmManager.setExact(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + intervalMillis, pendingIntent);
|
||||||
|
} else {
|
||||||
|
alarmManager.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + intervalMillis, pendingIntent);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user