diff --git a/.github/workflows/continuous-integration.yml b/.github/workflows/continuous-integration.yml
index 0b803370d..90098dc68 100644
--- a/.github/workflows/continuous-integration.yml
+++ b/.github/workflows/continuous-integration.yml
@@ -56,8 +56,20 @@ jobs:
run: gcloud config set project ${{ secrets.FIREBASE_PROJECT_ID }}
# Run Instrumentation Tests in Firebase Test Lab
+ # dm3q Samasung galaxy s23 SDK 33
+ # e1q Samasung galaxy s24 SDK 34
+ # bluejay Google Pixel 6a SDK 32
+ # a51 Samsung Galaxy A51 SDK 31
+ # q2q Samsung Galaxy Z Fold3 SDK 30
+ # cactus Xiaomi Redmi 6A SDK 27
- name: Run Instrumentation Tests in Firebase Test Lab
- run: gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=oriole,version=31,locale=en,orientation=portrait
+ run: |
+ gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=e1q,version=34,locale=en,orientation=portrait
+ gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=dm3q,version=33,locale=en,orientation=portrait
+ gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=bluejay,version=32,locale=en,orientation=portrait
+ gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=a51,version=31,locale=en,orientation=portrait
+ gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=q2q,version=30,locale=en,orientation=portrait
+ gcloud firebase test android run --type instrumentation --app ${{ env.module_app }}/build/outputs/apk/debug/${{ env.module_app }}-debug.apk --test ${{ env.module_app }}/build/outputs/apk/androidTest/debug/${{ env.module_app }}-debug-androidTest.apk --device model=cactus,version=27,locale=en,orientation=portrait
##########################################
diff --git a/README.md b/README.md
index 3a03ee41c..e65264ad9 100644
--- a/README.md
+++ b/README.md
@@ -50,14 +50,14 @@ Now you can choose (from the server information) whether this inventory should c
### GLPI Android Inventory Agent
-|GLPI|9.1|9.2+|9.3+|9.4+|9.5+|10.0+|
-|:---|:---:|:---:|:---:|:---:|:---:|:---:|
-|**GLPI Android Inventory Agent**|1.0.0|1.0.0|1.0.0|1.0.0|1.1.0|1.3.0|
-|**Plugin GLPI Agent Config**|1.0|1.0|1.0|1.0|1.1.0|1.2.0|
+|GLPI|9.1|9.2+|9.3+|9.4+|9.5+|10.0+|10.0+|10.0+|
+|:---|:---:|:---:|:---:|:---:|:---:|:---:|:---:|:---:|
+|**GLPI Android Inventory Agent**|1.0.0|1.0.0|1.0.0|1.0.0|1.1.0|1.3.0|1.4.0|1.5.0|
+|**Plugin GLPI Agent Config**|1.0|1.0|1.0|1.0|1.1.0|1.2.0|1.2.1|1.2.1|
### Inventory Agent & Android Versions
-GLPI Android Inventory Agent is compatible with Android 4.1 and higher (to Android 13).
+GLPI Android Inventory Agent is compatible with Android 5 and higher (to Android 13).
## Installation
diff --git a/app/build.gradle b/app/build.gradle
index 1793a984e..4affd9652 100644
--- a/app/build.gradle
+++ b/app/build.gradle
@@ -2,14 +2,16 @@ apply plugin: 'com.android.application'
apply plugin: 'com.google.firebase.crashlytics'
android {
- compileSdkVersion 34
+ compileSdk 34
useLibrary 'org.apache.http.legacy'
defaultConfig {
applicationId "org.glpi.inventory.agent"
minSdkVersion 21
targetSdkVersion 34
+
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
+ multiDexEnabled true
}
testOptions {
@@ -28,19 +30,10 @@ android {
}
}
- dexOptions {
- preDexLibraries true
- maxProcessCount 8
- }
-
namespace 'org.glpi.inventory.agent'
lint {
abortOnError false
}
-
- defaultConfig {
- multiDexEnabled true
- }
}
dependencies {
@@ -63,8 +56,8 @@ dependencies {
androidTestImplementation 'tools.fastlane:screengrab:2.1.1'
- testImplementation 'org.mockito:mockito-core:2.18.3'
- androidTestImplementation 'org.mockito:mockito-android:2.18.3'
+ testImplementation 'org.mockito:mockito-android:5.5.0' // Use the latest version
+ androidTestImplementation 'org.mockito:mockito-android:5.5.0'
implementation 'androidx.test.espresso:espresso-idling-resource:3.6.1'
implementation fileTree(dir: 'libs', include: ['*.jar'])
diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index eaaae5d1a..372a35889 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -1,10 +1,11 @@
+ xmlns:tools="http://schemas.android.com/tools"
+ android:versionCode="39413"
+ android:versionName="">
+
@@ -30,9 +31,6 @@ android:versionName="">
-
-
-
@@ -56,12 +54,6 @@ android:versionName="">
-
-
-
-
-
-
@@ -72,21 +64,19 @@ android:versionName="">
-
-
-
-
-
+
+
-
-
@@ -96,4 +86,4 @@ android:versionName="">
-
+
\ No newline at end of file
diff --git a/app/src/main/java/org/glpi/inventory/agent/broadcast/BootStartAgent.java b/app/src/main/java/org/glpi/inventory/agent/broadcast/BootStartAgent.java
index 1315fa1a5..f173c667a 100644
--- a/app/src/main/java/org/glpi/inventory/agent/broadcast/BootStartAgent.java
+++ b/app/src/main/java/org/glpi/inventory/agent/broadcast/BootStartAgent.java
@@ -38,53 +38,16 @@
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
-import android.content.SharedPreferences;
-import android.os.Build;
-import android.preference.PreferenceManager;
-import android.widget.Toast;
-
-import org.flyve.inventory.InventoryLog;
-import org.glpi.inventory.agent.service.InventoryService;
import org.glpi.inventory.agent.ui.ActivityMain;
-import org.glpi.inventory.agent.utils.AgentLog;
public class BootStartAgent extends BroadcastReceiver {
- TimeAlarm alarm = new TimeAlarm();
-
- /**
- * It sets an alarm after the user has finished booting
- * @param context in which the receiver is running
- * @param intent being received
- */
@Override
public void onReceive(Context context, Intent intent) {
-
- String action = intent.getAction();
- if(action==null) {
- return;
+ if (Intent.ACTION_BOOT_COMPLETED.equals(intent.getAction())) {
+ Intent activityIntent = new Intent(context, ActivityMain.class);
+ activityIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
+ context.startActivity(activityIntent);
}
-
- SharedPreferences customSharedPreference = PreferenceManager.getDefaultSharedPreferences(context);
- if (customSharedPreference.getBoolean("boot", false)) {
- try {
-
- if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
- Intent myIntent = new Intent(context, InventoryService.class);
- context.startForegroundService(myIntent);
- } else {
- Intent myIntent = new Intent(context, ActivityMain.class);
- myIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- context.startActivity(myIntent);
- }
- }catch(Exception ex) {
- Toast.makeText(context, ex.getMessage(), Toast.LENGTH_LONG).show();
- }
- }
-
- if(customSharedPreference.getBoolean("autoStartInventory", false)){
- alarm.setAlarm(context);
- }
-
}
}
diff --git a/app/src/main/java/org/glpi/inventory/agent/broadcast/InventoryJobScheduler.java b/app/src/main/java/org/glpi/inventory/agent/broadcast/InventoryJobScheduler.java
new file mode 100644
index 000000000..4172d2295
--- /dev/null
+++ b/app/src/main/java/org/glpi/inventory/agent/broadcast/InventoryJobScheduler.java
@@ -0,0 +1,204 @@
+/**
+ * ---------------------------------------------------------------------
+ * GLPI Android Inventory Agent
+ * Copyright (C) 2019 Teclib.
+ *
+ * https://glpi-project.org
+ *
+ * Based on Flyve MDM Inventory Agent For Android
+ * Copyright © 2018 Teclib. All rights reserved.
+ *
+ * ---------------------------------------------------------------------
+ *
+ * LICENSE
+ *
+ * This file is part of GLPI Android Inventory Agent.
+ *
+ * GLPI Android Inventory Agent is a subproject of GLPI.
+ *
+ * GLPI Android Inventory Agent 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.
+ *
+ * GLPI Android Inventory Agent 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.
+ * ---------------------------------------------------------------------
+ * @copyright Copyright © 2019 Teclib. All rights reserved.
+ * @license GPLv3 https://www.gnu.org/licenses/gpl-3.0.html
+ * @link https://github.com/glpi-project/android-inventory-agent
+ * @link https://glpi-project.org/glpi-network/
+ * ---------------------------------------------------------------------
+ */
+
+package org.glpi.inventory.agent.broadcast;
+
+import android.annotation.SuppressLint;
+import android.app.Notification;
+import android.app.NotificationChannel;
+import android.app.NotificationManager;
+import android.app.PendingIntent;
+import android.app.job.JobParameters;
+import android.app.job.JobService;
+import android.content.Context;
+import android.content.Intent;
+import android.content.SharedPreferences;
+import android.graphics.Color;
+import android.net.Uri;
+import android.os.Build;
+import android.os.Handler;
+import android.os.HandlerThread;
+import android.preference.PreferenceManager;
+import android.provider.Settings;
+
+import androidx.core.app.NotificationCompat;
+
+import org.flyve.inventory.InventoryTask;
+import org.glpi.inventory.agent.R;
+import org.glpi.inventory.agent.schema.ServerSchema;
+import org.glpi.inventory.agent.ui.ActivityMain;
+import org.glpi.inventory.agent.utils.AgentLog;
+import org.glpi.inventory.agent.utils.Helpers;
+import org.glpi.inventory.agent.utils.HttpInventory;
+import org.glpi.inventory.agent.utils.LocalPreferences;
+
+import java.util.ArrayList;
+import java.util.Calendar;
+
+@SuppressLint("SpecifyJobSchedulerIdRange")
+public class InventoryJobScheduler extends JobService {
+
+ public static final int INVENTORY_JOB_ID = 4492015;
+ private static final String NOTIFICATION_CHANNEL_ID = "org.glpi.inventory.agent";
+
+ @Override
+ public boolean onStartJob(final JobParameters params) {
+ HandlerThread handlerThread = new HandlerThread("SomeOtherThread");
+ handlerThread.start();
+
+ Handler handler = new Handler(handlerThread.getLooper());
+ handler.post(new Runnable() {
+ @Override
+ public void run() {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Run task" + Calendar.getInstance().getTime());
+ doInventory();
+ jobFinished(params, true);
+ }
+ });
+
+ return true;
+ }
+
+ private void doInventory() {
+ Context context = getApplicationContext();
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Launch inventory from JobScheduler " + Calendar.getInstance().getTime());
+
+ // check if autoStartInventory is deactivated
+ SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
+ if (!sharedPreferences.getBoolean("autoStartInventory", false)) {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : The inventory will not be send, is deactivated");
+ return;
+ }
+
+ showPersistentNotification();
+
+ final InventoryTask inventory = new InventoryTask(context.getApplicationContext(), Helpers.getAgentDescription(context), true);
+ final HttpInventory httpInventory = new HttpInventory(context.getApplicationContext());
+ ArrayList serverArray = new LocalPreferences(context).loadServer();
+ if (!serverArray.isEmpty()) {
+ for (final String serverName : serverArray) {
+ final ServerSchema model = httpInventory.setServerModel(serverName);
+ inventory.setTag(model.getTag());
+ inventory.setAssetItemtype(model.getItemtype());
+ inventory.getXML(new InventoryTask.OnTaskCompleted() {
+ @Override
+ public void onTaskSuccess(String data) {
+ ServerSchema model = httpInventory.setServerModel(serverName);
+ if(!model.getSerial().trim().isEmpty()) {
+ data = data.replaceAll("(.*)","" + model.getSerial() + "");
+ }
+ httpInventory.sendInventory(data, model, new HttpInventory.OnTaskCompleted() {
+ @Override
+ public void onTaskSuccess(String data) {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Inventory Success");
+ Helpers.sendToNotificationBar(context.getApplicationContext(), context.getResources().getString(R.string.inventory_notification_sent));
+ //Helpers.sendAnonymousData(context.getApplicationContext(), inventory);
+ }
+
+ @Override
+ public void onTaskError(String error) {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Inventory error");
+ Helpers.sendToNotificationBar(context.getApplicationContext(), context.getResources().getString(R.string.inventory_notification_fail));
+ AgentLog.e(error);
+ }
+ });
+ }
+
+ @Override
+ public void onTaskError(Throwable error) {
+ AgentLog.e(error.getMessage());
+ Helpers.sendToNotificationBar(context, context.getResources().getString(R.string.inventory_notification_fail));
+ }
+ });
+ }
+ } else {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : " + context.getResources().getString(R.string.inventory_no_server));
+ }
+ }
+
+ @Override
+ public boolean onStopJob(final JobParameters params) {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : onStopJob() was called");
+ return true;
+ }
+
+ private void showPersistentNotification() {
+ NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
+
+ // create channel if needed
+ if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
+ NotificationChannel chan = null;
+ chan = new NotificationChannel(NOTIFICATION_CHANNEL_ID, getString(R.string.app_is_running), NotificationManager.IMPORTANCE_DEFAULT);
+ chan.setLightColor(Color.BLUE);
+ chan.setLockscreenVisibility(Notification.VISIBILITY_PRIVATE);
+ manager.createNotificationChannel(chan);
+ }
+
+ // create intent to redirect to notification to settings
+ Intent notificationIntent;
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
+ notificationIntent = new Intent(Settings.ACTION_APP_NOTIFICATION_SETTINGS);
+ notificationIntent.putExtra(Settings.EXTRA_APP_PACKAGE, getPackageName());
+ notificationIntent.putExtra(Settings.EXTRA_CHANNEL_ID, NOTIFICATION_CHANNEL_ID);
+ notificationIntent.putExtra("app_package", getPackageName());
+ notificationIntent.putExtra("app_uid", getApplicationInfo().uid);
+ notificationIntent.putExtra("android.provider.extra.APP_PACKAGE", getPackageName());
+ } else {
+ notificationIntent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
+ notificationIntent.setData(Uri.parse("package:" + getPackageName()));
+ }
+ PendingIntent notificationPendingIntent = PendingIntent.getActivity(getApplicationContext(), 0, notificationIntent, PendingIntent.FLAG_IMMUTABLE);
+
+
+ Intent appIntent = new Intent(this, ActivityMain.class);
+ appIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
+ PendingIntent appIntentRedirect = PendingIntent.getActivity(this, 0, appIntent, PendingIntent.FLAG_IMMUTABLE);
+
+ // create notification
+ NotificationCompat.Builder notification = new NotificationCompat.Builder(this, NOTIFICATION_CHANNEL_ID);
+ notification.setOngoing(false);
+ notification.setSmallIcon(R.drawable.ic_stat);
+ notification.setContentTitle(getString(R.string.app_is_running));
+ notification.setContentText(getString(R.string.agent_description));
+ notification.setCategory(Notification.CATEGORY_EVENT);
+ notification.setPriority(NotificationCompat.PRIORITY_DEFAULT);
+ notification.setStyle(new NotificationCompat.BigTextStyle().bigText(getString(R.string.app_is_running_extend)));
+ notification.setContentIntent(appIntentRedirect);
+ notification.addAction(R.drawable.ic_stat, getString(R.string.disable_notification), notificationPendingIntent);
+
+ // notify
+ manager.notify(1, notification.build());
+ }
+}
diff --git a/app/src/main/java/org/glpi/inventory/agent/broadcast/TimeAlarm.java b/app/src/main/java/org/glpi/inventory/agent/broadcast/TimeAlarm.java
deleted file mode 100644
index 360d92aba..000000000
--- a/app/src/main/java/org/glpi/inventory/agent/broadcast/TimeAlarm.java
+++ /dev/null
@@ -1,175 +0,0 @@
-/**
- * ---------------------------------------------------------------------
- * GLPI Android Inventory Agent
- * Copyright (C) 2019 Teclib.
- *
- * https://glpi-project.org
- *
- * Based on Flyve MDM Inventory Agent For Android
- * Copyright © 2018 Teclib. All rights reserved.
- *
- * ---------------------------------------------------------------------
- *
- * LICENSE
- *
- * This file is part of GLPI Android Inventory Agent.
- *
- * GLPI Android Inventory Agent is a subproject of GLPI.
- *
- * GLPI Android Inventory Agent 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.
- *
- * GLPI Android Inventory Agent 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.
- * ---------------------------------------------------------------------
- * @copyright Copyright © 2019 Teclib. All rights reserved.
- * @license GPLv3 https://www.gnu.org/licenses/gpl-3.0.html
- * @link https://github.com/glpi-project/android-inventory-agent
- * @link https://glpi-project.org/glpi-network/
- * ---------------------------------------------------------------------
- */
-
-package org.glpi.inventory.agent.broadcast;
-
-import android.app.AlarmManager;
-import android.app.PendingIntent;
-import android.content.BroadcastReceiver;
-import android.content.Context;
-import android.content.Intent;
-import android.content.SharedPreferences;
-import android.os.Build;
-import android.os.PowerManager;
-import android.preference.PreferenceManager;
-
-import androidx.annotation.RequiresApi;
-
-import org.flyve.inventory.InventoryTask;
-import org.glpi.inventory.agent.R;
-import org.glpi.inventory.agent.schema.ServerSchema;
-import org.glpi.inventory.agent.utils.AgentLog;
-import org.glpi.inventory.agent.utils.Helpers;
-import org.glpi.inventory.agent.utils.HttpInventory;
-import org.glpi.inventory.agent.utils.LocalPreferences;
-
-import java.util.ArrayList;
-
-public class TimeAlarm extends BroadcastReceiver {
-
- /**
- * If the success XML is created, it sends the inventory
- * @param context in which the receiver is running
- * @param intent being received
- */
- @Override
- public void onReceive(final Context context, Intent intent) {
-
- // check if is deactivated
- SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
- Boolean val = sharedPreferences.getBoolean("autoStartInventory", false);
- if (!val) {
- AgentLog.d("The inventory will not be send, is deactivated");
- return;
- }
-
- AgentLog.d("Launch inventory from alarm");
-
- PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
- PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "");
- wl.acquire();
-
- final InventoryTask inventory = new InventoryTask(context.getApplicationContext(), Helpers.getAgentDescription(context), true);
- final HttpInventory httpInventory = new HttpInventory(context.getApplicationContext());
- ArrayList serverArray = new LocalPreferences(context).loadServer();
- if (!serverArray.isEmpty()) {
- for (final String serverName : serverArray) {
- final ServerSchema model = httpInventory.setServerModel(serverName);
- inventory.setTag(model.getTag());
- inventory.setAssetItemtype(model.getItemtype());
- inventory.getXML(new InventoryTask.OnTaskCompleted() {
- @Override
- public void onTaskSuccess(String data) {
- ServerSchema model = httpInventory.setServerModel(serverName);
- if(!model.getSerial().trim().isEmpty()) {
- data = data.replaceAll("(.*)","" + model.getSerial() + "");
- }
- httpInventory.sendInventory(data, model, new HttpInventory.OnTaskCompleted() {
- @Override
- public void onTaskSuccess(String data) {
- Helpers.sendToNotificationBar(context.getApplicationContext(), context.getResources().getString(R.string.inventory_notification_sent));
- //Helpers.sendAnonymousData(context.getApplicationContext(), inventory);
- }
-
- @Override
- public void onTaskError(String error) {
- Helpers.sendToNotificationBar(context.getApplicationContext(), context.getResources().getString(R.string.inventory_notification_fail));
- AgentLog.e(error);
- }
- });
- }
-
- @Override
- public void onTaskError(Throwable error) {
- AgentLog.e(error.getMessage());
- Helpers.sendToNotificationBar(context, context.getResources().getString(R.string.inventory_notification_fail));
- }
- });
- }
- } else {
- AgentLog.e(context.getResources().getString(R.string.inventory_no_server));
- }
-
- wl.release();
- }
-
- /**
- * Schedules the alarm
- * @param context
- */
- @RequiresApi(api = Build.VERSION_CODES.M)
- public void setAlarm(Context context) {
-
- AgentLog.d("Set Alarm");
-
- AlarmManager am =(AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
- Intent i = new Intent(context, TimeAlarm.class);
- i.setAction("org.glpi.inventory.agent.ALARM");
- PendingIntent pi = PendingIntent.getBroadcast(context, 0, i, PendingIntent.FLAG_IMMUTABLE);
-
- SharedPreferences customSharedPreference = PreferenceManager.getDefaultSharedPreferences(context);
- String timeInventory = customSharedPreference.getString("timeInventory", "Week");
-
- int time = 60 * 1000;
-
- if (timeInventory.equals("Day")) {
- time = 24 * 60 * 60 * 1000;
- AgentLog.d("Alarm Daily");
- } else if(timeInventory.equals("Week")) {
- time = 7 * 24 * 60 * 60 * 1000;
- AgentLog.d("Alarm Weekly");
- } else if(timeInventory.equals("Month")) {
- time = 30 * 24 * 60 * 60 * 1000;
- AgentLog.d("Alarm Monthly");
- }
-
- try {
- am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), time, pi);
- } catch (NullPointerException ex) {
- AgentLog.e(ex.getMessage());
- }
- }
-
- /**
- * Removes the alarm with a matching argument
- * @param context
- */
- public void cancelAlarm(Context context) {
- Intent intent = new Intent(context, TimeAlarm.class);
- PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0);
- AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
- alarmManager.cancel(sender);
- }
-}
diff --git a/app/src/main/java/org/glpi/inventory/agent/core/home/Home.java b/app/src/main/java/org/glpi/inventory/agent/core/home/Home.java
index bf30bead0..81cfbc314 100644
--- a/app/src/main/java/org/glpi/inventory/agent/core/home/Home.java
+++ b/app/src/main/java/org/glpi/inventory/agent/core/home/Home.java
@@ -54,14 +54,12 @@ interface Presenter {
void showError(String message);
// Models
- void doBindService(Activity activity);
void setupList(Activity activity, ListView lst);
void clickItem(final Activity activity, HomeSchema homeSchema);
List getListItems();
}
interface Model {
- void doBindService(Activity activity);
void setupList(Activity activity, ListView lst);
void clickItem(final Activity activity, HomeSchema homeSchema);
List getListItems();
diff --git a/app/src/main/java/org/glpi/inventory/agent/core/home/HomeModel.java b/app/src/main/java/org/glpi/inventory/agent/core/home/HomeModel.java
index b4b89e3e8..d6635ce6c 100644
--- a/app/src/main/java/org/glpi/inventory/agent/core/home/HomeModel.java
+++ b/app/src/main/java/org/glpi/inventory/agent/core/home/HomeModel.java
@@ -36,22 +36,15 @@
package org.glpi.inventory.agent.core.home;
import android.app.Activity;
-import android.app.ActivityManager;
-import android.content.ComponentName;
-import android.content.Context;
import android.content.Intent;
-import android.os.Build;
-import android.util.Log;
import android.widget.ListView;
import org.glpi.inventory.agent.R;
import org.glpi.inventory.agent.adapter.HomeAdapter;
import org.glpi.inventory.agent.preference.GlobalParametersPreference;
import org.glpi.inventory.agent.preference.InventoryParametersPreference;
-import org.glpi.inventory.agent.service.InventoryService;
import org.glpi.inventory.agent.ui.ActivityInventoryReport;
import org.glpi.inventory.agent.ui.DialogListServers;
-import org.glpi.inventory.agent.utils.AgentLog;
import java.util.ArrayList;
import java.util.List;
@@ -65,53 +58,6 @@ public HomeModel(Home.Presenter presenter) {
this.presenter = presenter;
}
- public void doBindService(Activity activity) {
- // Establish a connection with the service. We use an explicit
- // class name because we want a specific service implementation that
- // we know will be running in our own process (and thus won't be
- // supporting component replacement by other applications).
- InventoryService inventoryService = new InventoryService();
- Intent mServiceIntent = new Intent(activity, inventoryService.getClass());
- ComponentName result;
-
- if (!isMyServiceRunning(activity, inventoryService.getClass())) {
- if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
- result = activity.startForegroundService(mServiceIntent);
- } else {
- result = activity.startService(mServiceIntent);
- }
-
- if (result != null) {
- AgentLog.log(this, " Agent started ", Log.INFO);
- } else {
- AgentLog.log(this, " Agent fail", Log.ERROR);
- }
- } else {
- AgentLog.log(this, " Agent already started ", Log.ERROR);
- }
- }
-
- /**
- * Check if the service is running
- * @param serviceClass Class
- * @return boolean
- */
- private boolean isMyServiceRunning(Activity activity, Class> serviceClass) {
- try {
- ActivityManager manager = (ActivityManager) activity.getSystemService(Context.ACTIVITY_SERVICE);
- for (ActivityManager.RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
- if (serviceClass.getName().equals(service.service.getClassName())) {
- return true;
- }
- }
- } catch (Exception ex) {
- AgentLog.e(ex.getMessage());
- }
-
- return false;
- }
-
-
@Override
public void setupList(Activity activity, ListView lst) {
arrHome = new ArrayList<>();
diff --git a/app/src/main/java/org/glpi/inventory/agent/core/home/HomePresenter.java b/app/src/main/java/org/glpi/inventory/agent/core/home/HomePresenter.java
index afc2b2d2d..f443a5f76 100644
--- a/app/src/main/java/org/glpi/inventory/agent/core/home/HomePresenter.java
+++ b/app/src/main/java/org/glpi/inventory/agent/core/home/HomePresenter.java
@@ -57,11 +57,6 @@ public void showError(String message) {
}
}
- @Override
- public void doBindService(Activity activity) {
- model.doBindService(activity);
- }
-
@Override
public void setupList(Activity activity, ListView lst) {
model.setupList(activity, lst);
diff --git a/app/src/main/java/org/glpi/inventory/agent/core/main/MainModel.java b/app/src/main/java/org/glpi/inventory/agent/core/main/MainModel.java
index d3001291e..b9e99aad9 100644
--- a/app/src/main/java/org/glpi/inventory/agent/core/main/MainModel.java
+++ b/app/src/main/java/org/glpi/inventory/agent/core/main/MainModel.java
@@ -46,6 +46,7 @@
import org.glpi.inventory.agent.R;
import org.glpi.inventory.agent.adapter.DrawerAdapter;
+import org.glpi.inventory.agent.ui.ActivityMain;
import org.glpi.inventory.agent.ui.FragmentAbout;
import org.glpi.inventory.agent.ui.FragmentHelp;
import org.glpi.inventory.agent.ui.FragmentHome;
@@ -70,30 +71,9 @@ public MainModel(Main.Presenter presenter) {
@Override
public void setupInventoryAlarm(Context context) {
- LocalStorage cache = new LocalStorage(context);
- if (cache.getDataBoolean("changeSchedule")) {
- Calendar calendar = Calendar.getInstance();
-
- SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
- String timeInventory = sharedPreferences.getString("timeInventory", "week");
-
- // week by default
- if (timeInventory.equalsIgnoreCase("week")) {
- calendar.add(Calendar.DATE, 7);
- }
-
- if (timeInventory.equalsIgnoreCase("day")) {
- calendar.add(Calendar.DATE, 1);
- }
-
- if (timeInventory.equalsIgnoreCase("month")) {
- calendar.add(Calendar.DATE, 30);
- }
-
- long dateTime = calendar.getTime().getTime();
-
- cache.setDataLong("data", dateTime);
- cache.setDataBoolean("changeSchedule", false);
+ if (context instanceof ActivityMain) {
+ ActivityMain activity = (ActivityMain) context;
+ activity.scheduleJob();
}
}
diff --git a/app/src/main/java/org/glpi/inventory/agent/preference/InventoryParametersPreference.java b/app/src/main/java/org/glpi/inventory/agent/preference/InventoryParametersPreference.java
index 72f771084..ca04e7f70 100644
--- a/app/src/main/java/org/glpi/inventory/agent/preference/InventoryParametersPreference.java
+++ b/app/src/main/java/org/glpi/inventory/agent/preference/InventoryParametersPreference.java
@@ -73,15 +73,16 @@ public void onClick(View v) {
@Override
public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String s) {
-
if ("timeInventory".equals(s)) {
Intent intent = new Intent("timeAlarmChanged");
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
- AgentLog.d("Preference "+ s +" changed -> " + sharedPreferences.getString(s, "Week"));
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Preference "+ s +" changed -> " + sharedPreferences.getString(s, "Week"));
}
if("autoStartInventory".equals(s)){
- AgentLog.d("Preference "+ s +" changed -> " + sharedPreferences.getBoolean(s, false));
+ Intent intent = new Intent("timeAlarmChanged");
+ LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Preference "+ s +" changed -> " + sharedPreferences.getBoolean(s, false));
}
}
}
diff --git a/app/src/main/java/org/glpi/inventory/agent/service/InventoryService.java b/app/src/main/java/org/glpi/inventory/agent/service/InventoryService.java
deleted file mode 100644
index c7e1e223a..000000000
--- a/app/src/main/java/org/glpi/inventory/agent/service/InventoryService.java
+++ /dev/null
@@ -1,313 +0,0 @@
-/**
- * ---------------------------------------------------------------------
- * GLPI Android Inventory Agent
- * Copyright (C) 2019 Teclib.
- *
- * https://glpi-project.org
- *
- * Based on Flyve MDM Inventory Agent For Android
- * Copyright © 2018 Teclib. All rights reserved.
- *
- * ---------------------------------------------------------------------
- *
- * LICENSE
- *
- * This file is part of GLPI Android Inventory Agent.
- *
- * GLPI Android Inventory Agent is a subproject of GLPI.
- *
- * GLPI Android Inventory Agent 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.
- *
- * GLPI Android Inventory Agent 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.
- * ---------------------------------------------------------------------
- * @copyright Copyright © 2019 Teclib. All rights reserved.
- * @license GPLv3 https://www.gnu.org/licenses/gpl-3.0.html
- * @link https://github.com/glpi-project/android-inventory-agent
- * @link https://glpi-project.org/glpi-network/
- * ---------------------------------------------------------------------
- */
-
-package org.glpi.inventory.agent.service;
-
-import android.app.Notification;
-import android.app.NotificationChannel;
-import android.app.NotificationManager;
-import android.app.PendingIntent;
-import android.app.Service;
-import android.content.Context;
-import android.content.Intent;
-import android.content.SharedPreferences;
-import android.graphics.Color;
-import android.os.Build;
-import android.os.Handler;
-import android.os.IBinder;
-import android.os.PowerManager;
-import android.preference.PreferenceManager;
-import androidx.annotation.Nullable;
-import androidx.annotation.RequiresApi;
-import androidx.core.app.NotificationCompat;
-import android.util.Log;
-import android.util.Xml;
-
-import org.flyve.inventory.InventoryTask;
-import org.glpi.inventory.agent.R;
-import org.glpi.inventory.agent.schema.ServerSchema;
-import org.glpi.inventory.agent.ui.ActivityMain;
-import org.glpi.inventory.agent.utils.AgentLog;
-import org.glpi.inventory.agent.utils.Helpers;
-import org.glpi.inventory.agent.utils.HttpInventory;
-import org.glpi.inventory.agent.utils.LocalPreferences;
-import org.glpi.inventory.agent.utils.LocalStorage;
-import org.glpi.inventory.agent.utils.Utils;
-import org.xmlpull.v1.XmlSerializer;
-
-import java.io.StringReader;
-import java.io.StringWriter;
-import java.util.ArrayList;
-import java.util.Calendar;
-import java.util.Date;
-import java.util.Timer;
-import java.util.TimerTask;
-import java.util.concurrent.TimeUnit;
-
-public class InventoryService extends Service {
-
- public static final String TIMER_RECEIVER = "org.glpi.inventory.service.timer";
-
- private Handler mHandler;
- Calendar calendar;
- long longDate;
- Date dateCurrent, dateDiff;
- LocalStorage cache;
-
- private Timer mTimer = null;
- public static final long NOTIFY_INTERVAL = 1000;
- Intent intent;
-
- @Nullable
- @Override
- public IBinder onBind(Intent intent) {
- return null;
- }
-
- @Override
- public void onCreate() {
- super.onCreate();
-
- mHandler = new Handler();
-
- cache = new LocalStorage(getApplicationContext());
- calendar = Calendar.getInstance();
-
- mTimer = new Timer();
- mTimer.scheduleAtFixedRate(new TimeDisplayTimerTask(), 5, NOTIFY_INTERVAL);
- intent = new Intent(TIMER_RECEIVER);
-
- if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
- startMyOwnForeground();
- }
-
- }
-
- @RequiresApi(api = Build.VERSION_CODES.O)
- private void startMyOwnForeground(){
- String NOTIFICATION_CHANNEL_ID = "org.glpi.inventory.agent";
- String channelName = getApplicationContext().getResources().getString(R.string.app_is_running);
- NotificationChannel chan = new NotificationChannel(NOTIFICATION_CHANNEL_ID, channelName, NotificationManager.IMPORTANCE_NONE);
- chan.setLightColor(Color.BLUE);
- chan.setLockscreenVisibility(Notification.VISIBILITY_PRIVATE);
- NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
- assert manager != null;
- manager.createNotificationChannel(chan);
-
- //create intent to redirect user to app on click
- Intent appIntent = new Intent(getApplicationContext(), ActivityMain.class);
- appIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
- PendingIntent appIntentRedirect = PendingIntent.getActivity(getApplicationContext(), 0, appIntent, PendingIntent.FLAG_MUTABLE);
-
-
- //create inent to invite user to disable notification
- Intent notificationIntent = new Intent();
- notificationIntent.setAction("android.settings.APP_NOTIFICATION_SETTINGS");
- notificationIntent.putExtra("app_package", getPackageName());
- notificationIntent.putExtra("app_uid", getApplicationInfo().uid);
- notificationIntent.putExtra("android.provider.extra.APP_PACKAGE", getPackageName());
- PendingIntent notificationIntentRedirect = PendingIntent.getActivity(getApplicationContext(), 0,
- notificationIntent, PendingIntent.FLAG_MUTABLE);
-
- //create notification
- NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(this, NOTIFICATION_CHANNEL_ID);
- Notification notification = notificationBuilder.setOngoing(true)
- .setSmallIcon(R.drawable.ic_stat)
- .setContentTitle(getApplicationContext().getResources().getString(R.string.app_is_running))
- .setPriority(NotificationManager.IMPORTANCE_MIN)
- .setCategory(Notification.CATEGORY_SERVICE)
- .setStyle(new NotificationCompat.BigTextStyle()
- .bigText(getApplicationContext().getResources().getString(R.string.app_is_running_extend)))
- .setContentIntent(appIntentRedirect)
- .addAction(R.drawable.ic_about, getApplicationContext().getResources().getString(R.string.disable_notification), notificationIntentRedirect)
- .build();
- startForeground(2, notification);
- }
-
- class TimeDisplayTimerTask extends TimerTask {
- @Override
- public void run() {
- mHandler.post(new Runnable() {
-
- @Override
- public void run() {
- calendar = Calendar.getInstance();
- longDate = calendar.getTime().getTime();
- Log.i("strDate", String.valueOf(longDate));
- twoDatesBetweenTime();
- }
-
- });
- }
- }
-
- private void twoDatesBetweenTime() {
-
- try {
- dateCurrent = new Date(longDate);
- } catch (Exception ex) {
- AgentLog.e(ex.getMessage());
- }
-
- try {
- dateDiff = new Date(cache.getDataLong("data"));
- } catch (Exception ex) {
- AgentLog.e(ex.getMessage());
- }
-
- try {
- long diff = dateDiff.getTime() - dateCurrent.getTime();
-
- long days = TimeUnit.MILLISECONDS.toDays(diff);
- diff -= TimeUnit.DAYS.toMillis(days);
-
- long hours = TimeUnit.MILLISECONDS.toHours(diff);
- diff -= TimeUnit.HOURS.toMillis(hours);
-
- long minutes = TimeUnit.MILLISECONDS.toMinutes(diff);
- diff -= TimeUnit.MINUTES.toMillis(minutes);
-
- long seconds = TimeUnit.MILLISECONDS.toSeconds(diff);
-
- if (seconds + hours + minutes + days > 0) {
- String strTesting;
- if (days != 0)
- strTesting = days + " days " + hours + ":" + minutes + ":" + seconds;
- else
- strTesting = hours + ":" + minutes + ":" + seconds;
- updateTime(strTesting);
- } else {
- sendInventory();
- setupInventorySchedule();
- }
- } catch (Exception e) {
- mTimer.cancel();
- mTimer.purge();
- }
-
- }
-
- private void setupInventorySchedule() {
- calendar = Calendar.getInstance();
-
- SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
- String timeInventory = sharedPreferences.getString("timeInventory", "week");
-
- // week by default
- if (timeInventory.equalsIgnoreCase("week")) {
- calendar.add(Calendar.DATE, 7);
- }
-
- if(timeInventory.equalsIgnoreCase("day")) {
- calendar.add(Calendar.DATE, 1);
- }
-
- if(timeInventory.equalsIgnoreCase("month")) {
- calendar.add(Calendar.DATE, 30);
- }
-
- long dateTime = calendar.getTime().getTime();
-
- cache = new LocalStorage(getApplicationContext());
- cache.setDataLong("data", dateTime);
- }
-
- private void sendInventory() {
- final Context context = getApplicationContext();
-
- SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
- Boolean autoStartInventory = sharedPreferences.getBoolean("autoStartInventory", false);
- if(!autoStartInventory) { return; }
-
- PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
- PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "android:inventory:agent");
- wl.acquire();
-
- final InventoryTask inventory = new InventoryTask(context.getApplicationContext(), Helpers.getAgentDescription(context), true);
- final HttpInventory httpInventory = new HttpInventory(context.getApplicationContext());
- ArrayList serverArray = new LocalPreferences(context).loadServer();
- if (!serverArray.isEmpty()) {
- for (String serverName : serverArray) {
- final ServerSchema model = httpInventory.setServerModel(serverName);
- inventory.setTag(model.getTag());
- inventory.setAssetItemtype(model.getItemtype());
- inventory.getXML(new InventoryTask.OnTaskCompleted() {
- @Override
- public void onTaskSuccess(String data) {
- if(!model.getSerial().trim().isEmpty()) {
- data = data.replaceAll("(.*)","" + model.getSerial() + "");
- }
- httpInventory.sendInventory(data, model, new HttpInventory.OnTaskCompleted() {
- @Override
- public void onTaskSuccess(String data) {
- Helpers.sendToNotificationBar(context.getApplicationContext(), context.getResources().getString(R.string.inventory_notification_sent));
- //Helpers.sendAnonymousData(context.getApplicationContext(), inventory);
- }
-
- @Override
- public void onTaskError(String error) {
- Helpers.sendToNotificationBar(context.getApplicationContext(), error);
- AgentLog.e(error);
- }
- });
- }
-
- @Override
- public void onTaskError(Throwable error) {
- AgentLog.e(error.getMessage());
- Helpers.sendToNotificationBar(context, context.getResources().getString(R.string.inventory_notification_fail));
- }
- });
- }
- } else {
- Helpers.sendToNotificationBar(context.getApplicationContext(), context.getResources().getString(R.string.no_servers_added));
- AgentLog.e(context.getResources().getString(R.string.no_servers_added));
- }
-
- wl.release();
- }
-
- @Override
- public void onDestroy() {
- super.onDestroy();
- Log.e("Service finish", "Finish");
- getApplicationContext().startService(new Intent(getApplicationContext(), InventoryService.class));
- }
-
- private void updateTime(String strTime) {
- intent.putExtra("time", strTime);
- sendBroadcast(intent);
- }
-}
diff --git a/app/src/main/java/org/glpi/inventory/agent/ui/ActivityMain.java b/app/src/main/java/org/glpi/inventory/agent/ui/ActivityMain.java
index a78c87e3d..4f921df09 100644
--- a/app/src/main/java/org/glpi/inventory/agent/ui/ActivityMain.java
+++ b/app/src/main/java/org/glpi/inventory/agent/ui/ActivityMain.java
@@ -36,7 +36,10 @@
package org.glpi.inventory.agent.ui;
import android.Manifest;
+import android.app.job.JobInfo;
+import android.app.job.JobScheduler;
import android.content.BroadcastReceiver;
+import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
@@ -45,7 +48,6 @@
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Bundle;
-import android.os.Environment;
import android.os.Parcelable;
import android.preference.PreferenceManager;
@@ -53,14 +55,13 @@
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;
import androidx.core.app.ActivityCompat;
+import androidx.core.content.ContextCompat;
import androidx.drawerlayout.widget.DrawerLayout;
import androidx.enterprise.feedback.KeyedAppState;
import androidx.enterprise.feedback.KeyedAppStatesReporter;
-import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
-import android.provider.Settings;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.AnimationUtils;
@@ -70,28 +71,25 @@
import com.google.android.material.floatingactionbutton.FloatingActionButton;
-import org.flyve.inventory.InventoryTask;
import org.glpi.inventory.agent.R;
-import org.glpi.inventory.agent.core.detailserver.DetailServer;
-import org.glpi.inventory.agent.core.detailserver.DetailServerPresenter;
+import org.glpi.inventory.agent.broadcast.InventoryJobScheduler;
import org.glpi.inventory.agent.core.main.Main;
import org.glpi.inventory.agent.core.main.MainPresenter;
import org.glpi.inventory.agent.preference.GlobalParametersPreference;
import org.glpi.inventory.agent.preference.InventoryParametersPreference;
-import org.glpi.inventory.agent.schema.ServerSchema;
-import org.glpi.inventory.agent.service.InventoryService;
import org.glpi.inventory.agent.utils.AgentLog;
import org.glpi.inventory.agent.utils.Helpers;
-import org.glpi.inventory.agent.utils.HttpInventory;
import org.glpi.inventory.agent.utils.LocalPreferences;
import org.glpi.inventory.agent.utils.LocalStorage;
import org.json.JSONException;
import org.json.JSONObject;
import java.util.ArrayList;
+import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Map;
+import java.util.concurrent.TimeUnit;
public class ActivityMain extends AppCompatActivity
implements Main.View, SharedPreferences.OnSharedPreferenceChangeListener {
@@ -107,18 +105,7 @@ public class ActivityMain extends AppCompatActivity
private Boolean isOpen;
private Animation fab_open, fab_close, fab_clock, fab_anticlock;
private TextView textview_settings, textview_scheduler;
-
- private BroadcastReceiver broadcastReceiver = new BroadcastReceiver() {
- @Override
- public void onReceive(Context context, Intent intent) {
- String strTime = intent.getStringExtra("time");
- if (sharedPreferences.getBoolean("autoStartInventory", false)) {
- toolbar.setSubtitle(strTime);
- } else {
- toolbar.setSubtitle("");
- }
- }
- };
+ InventoryJobScheduler alarm = new InventoryJobScheduler();
private BroadcastReceiver mMessageReceiver = new BroadcastReceiver() {
@Override
@@ -129,23 +116,45 @@ public void onReceive(Context context, Intent intent) {
};
private BroadcastReceiver appRestrictionChange = null;
- private KeyedAppStatesReporter appRestrictionChangeReporter = null;
@Override
protected void onStart() {
super.onStart();
- IntentFilter restrictionsFilter =
- new IntentFilter(Intent.ACTION_APPLICATION_RESTRICTIONS_CHANGED);
-
+ IntentFilter restrictionsFilter = new IntentFilter(Intent.ACTION_APPLICATION_RESTRICTIONS_CHANGED);
BroadcastReceiver appRestrictionChange = new BroadcastReceiver() {
@Override public void onReceive(Context context, Intent intent) {
resolveRestrictions();
}
};
- registerReceiver(appRestrictionChange, restrictionsFilter);
+ if (Build.VERSION.SDK_INT >= 34) {
+ registerReceiver(appRestrictionChange, restrictionsFilter, RECEIVER_EXPORTED);
+ } else {
+ registerReceiver(appRestrictionChange, restrictionsFilter);
+ }
+
+ if (sharedPreferences.getBoolean("autoStartInventory", false)) {
+ String label = getString(R.string.schedule_inventory_each);
+ String frequency = "weekly";
+ switch(sharedPreferences.getString("timeInventory", "Week")){
+ case "Day":
+ frequency = "daily";
+ break;
+ case "Month":
+ frequency = "monthly";
+ break;
+ default:
+ frequency = "weekly";
+ break;
+ }
+
+ toolbar.setSubtitle(label + ' ' + frequency);
+ } else {
+ toolbar.setSubtitle("Scheduled inventory not configured");
+ }
}
+
@Override
protected void onStop() {
super.onStop();
@@ -155,136 +164,52 @@ protected void onStop() {
}
}
- public static void enterpriseFeedback(Context context,
- String key,
- String message,
- String data,
- int severity) {
- KeyedAppStatesReporter keyedAppStatesReporter = KeyedAppStatesReporter.create(context);
- KeyedAppState keyedAppStateMessage = KeyedAppState.builder()
- .setSeverity(severity)
- .setKey(key)
- .setMessage(message)
- .setData(data)
- .build();
- List list = new ArrayList<>();
- list.add(keyedAppStateMessage);
- keyedAppStatesReporter.setStates(list);
+ @Override
+ protected void onResume() {
+ super.onResume();
+ resolveRestrictions();
}
- private void resolveRestrictions() {
- AgentLog.e("EMM - START resolve restrictions");
- RestrictionsManager myRestrictionsMgr = null;
- myRestrictionsMgr = (RestrictionsManager) getSystemService(Context.RESTRICTIONS_SERVICE);
- Bundle appRestrictions = myRestrictionsMgr.getApplicationRestrictions();
+ @Override
+ protected void onPause() {
+ super.onPause();
+ //unregisterReceiver(broadcastReceiver);
+ }
- SharedPreferences customSharedPreference = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
- SharedPreferences.Editor editor = customSharedPreference.edit();
+ private void checkPermissions() {
+ List permissionsList = new ArrayList<>(Arrays.asList(
+ Manifest.permission.READ_PHONE_STATE,
+ Manifest.permission.CAMERA
+ ));
- if (appRestrictions.containsKey("automatic_inventory")) {
- editor.putBoolean("autoStartInventory", appRestrictions.getBoolean("automatic_inventory"));
- enterpriseFeedback(getApplicationContext(), "automatic_inventory", "automatic_inventory option set successfully", appRestrictions.getBoolean("automatic_inventory") ? "true" : "false", KeyedAppState.SEVERITY_INFO);
- AgentLog.e("EMM - set automatic inventory to " + appRestrictions.getBoolean("automatic_inventory"));
- editor.apply();
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
+ permissionsList.add(Manifest.permission.POST_NOTIFICATIONS);
}
- if (appRestrictions.containsKey("frequency")) {
- editor.putString("timeInventory", appRestrictions.getString("frequency"));
- enterpriseFeedback(getApplicationContext(), "frequency", "frequency option set successfully", appRestrictions.getString("frequency"), KeyedAppState.SEVERITY_INFO);
- AgentLog.e("EMM - set frequency to " + appRestrictions.getString("frequency"));
- editor.apply();
- }
+ String[] permissions = permissionsList.toArray(new String[0]);
- if (appRestrictions.containsKey("auto_start_on_boot")) {
- editor.putBoolean("boot", appRestrictions.getBoolean("auto_start_on_boot"));
- enterpriseFeedback(getApplicationContext(), "auto_start_on_boot", "auto_start_on_boot option set successfully", appRestrictions.getBoolean("auto_start_on_boot") ? "true" : "false", KeyedAppState.SEVERITY_INFO);
- AgentLog.e("EMM - set auto start on boot to " + appRestrictions.getBoolean("auto_start_on_boot"));
- editor.apply();
- }
-
- Parcelable[] parcelables = appRestrictions.getParcelableArray("server_configuration_list");
- if (parcelables != null && parcelables.length > 0) {
- final Context context = getApplicationContext();
- for (int i = 0; i < parcelables.length; i++) {
- Bundle serverConfig = (Bundle) parcelables[i];
- JSONObject jsonServerConfig = new JSONObject();
- LocalPreferences preferences = new LocalPreferences(context);
-
- if (serverConfig.getString("server_url").isEmpty()) {
- enterpriseFeedback(getApplicationContext(), "server_url", "Error server URL is mandatory -> ", serverConfig.getString("server_url"), KeyedAppState.SEVERITY_ERROR);
- AgentLog.e("EMM - server url is mandatory");
- continue;
- }
-
- try {
- jsonServerConfig.put("address", serverConfig.getString("server_url"));
- jsonServerConfig.put("tag", serverConfig.getString("server_tag"));
- jsonServerConfig.put("login", serverConfig.getString("server_login"));
- jsonServerConfig.put("pass", serverConfig.getString("server_password"));
- jsonServerConfig.put("itemtype", serverConfig.getString("server_itemtype"));
- jsonServerConfig.put("serial", serverConfig.getString("server_custom_asset_serial"));
-
- AgentLog.e("EMM - Receive the following configuration '" + jsonServerConfig.toString());
-
- JSONObject local_server = preferences.loadJSONObject(serverConfig.getString("server_url"));
- AgentLog.e("EMM - Try to load '" + serverConfig.getString("server_url") + "' server if exist");
- AgentLog.e("EMM - Found '" + local_server.toString() + "'");
- AgentLog.e("EMM - Exist ? -> '" + !local_server.toString().equals("{}") + "'");
-
- if (local_server.toString().equals("{}")) {
- ArrayList serverArray = preferences.loadServer();
- serverArray.add(serverConfig.getString("server_url"));
- preferences.saveServer(serverArray);
- preferences.saveJSONObject(serverConfig.getString("server_url"), jsonServerConfig);
- enterpriseFeedback(getApplicationContext(), "server_url", "server_url added successfully", serverConfig.getString("server_url"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_tag", "server_tag added successfully", serverConfig.getString("server_tag"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_login", "server_login added successfully", serverConfig.getString("server_login"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_password", "server_password added successfully", "***", KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_itemtype", "server_itemtype added successfully", serverConfig.getString("server_itemtype"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_custom_asset_serial", "server_custom_asset_serial added successfully", serverConfig.getString("server_custom_asset_serial"), KeyedAppState.SEVERITY_INFO);
- AgentLog.e("EMM - Server added successfully");
- } else {
- preferences.deletePreferences(serverConfig.getString("server_url"));
- preferences.saveJSONObject(serverConfig.getString("server_url"), jsonServerConfig);
- enterpriseFeedback(getApplicationContext(), "server_url", "server_url updated successfully", serverConfig.getString("server_url"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_tag", "server_tag updated successfully", serverConfig.getString("server_tag"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_login", "server_login updated successfully", serverConfig.getString("server_login"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_password", "server_password updated successfully", "***", KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_itemtype", "server_itemtype updated successfully", serverConfig.getString("server_itemtype"), KeyedAppState.SEVERITY_INFO);
- enterpriseFeedback(getApplicationContext(), "server_custom_asset_serial", "server_custom_asset_serial updated successfully", serverConfig.getString("server_custom_asset_serial"), KeyedAppState.SEVERITY_INFO);
- AgentLog.e("EMM - Server updated successfully");
- }
-
- } catch (JSONException e) {
- enterpriseFeedback(getApplicationContext(), "server_url", "error while adding/updating server -> " + e.getMessage(), serverConfig.getString("server_url"), KeyedAppState.SEVERITY_ERROR);
- AgentLog.e("EMM - error while adding/updating server");
- AgentLog.e("EMM - " + e.getMessage());
- }
+ List permissionsToRequest = new ArrayList<>();
+ for (String permission : permissions) {
+ if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) {
+ permissionsToRequest.add(permission);
}
- } else {
- AgentLog.e("EMM - 'server_configuration_list' key is empty");
}
- AgentLog.e("EMM - END resolve restrictions");
- }
- @Override
- protected void onResume() {
- super.onResume();
- registerReceiver(broadcastReceiver,new IntentFilter(InventoryService.TIMER_RECEIVER));
- resolveRestrictions();
- }
-
- @Override
- protected void onPause() {
- super.onPause();
- unregisterReceiver(broadcastReceiver);
+ if (!permissionsToRequest.isEmpty()) {
+ ActivityCompat.requestPermissions(this, permissionsToRequest.toArray(new String[0]), 0);
+ }
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
- //Fabric.with(this, new Crashlytics());
+
+ checkPermissions();
+
setContentView(R.layout.activity_main);
+ if (ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
+ ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CAMERA}, 1);
+ }
ActivityCompat.requestPermissions(ActivityMain.this,
new String[]{
@@ -297,6 +222,8 @@ protected void onCreate(Bundle savedInstanceState) {
loadCategories();
}
+ scheduleJob();
+
// Menu list
ListView lst = findViewById(R.id.lst);
@@ -386,11 +313,6 @@ public void onClick(View view) {
ActivityMain.this.startActivity(miIntent);
}
});
-
- //app restriction change
- KeyedAppStatesReporter appRestrictionChangeReporter = KeyedAppStatesReporter.create(getApplicationContext());
-
-
}
private void disableFab(){
@@ -470,20 +392,195 @@ public void onBackPressed() {
}
@Override
- public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
+ public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
- switch (requestCode) {
- case 1: {
+ boolean allGranted = true;
+ for (int result : grantResults) {
+ if (result != PackageManager.PERMISSION_GRANTED) {
+ allGranted = false;
+ break;
+ }
+ }
- // If request is cancelled, the result arrays are empty.
- if (grantResults.length > 0
- && grantResults[0] == PackageManager.PERMISSION_GRANTED
- && grantResults[1] == PackageManager.PERMISSION_GRANTED) {
- } else {
- String message = getResources().getString(R.string.permission_error_result);
- Helpers.snackClose(ActivityMain.this, message, getString(R.string.permission_snack_ok), true);
+ if (!allGranted) {
+ String message = getResources().getString(R.string.permission_error_result);
+ Helpers.snackClose(ActivityMain.this, message, getString(R.string.permission_snack_ok), true);
+ }
+ }
+
+ public static void enterpriseFeedback(Context context,
+ String key,
+ String message,
+ String data,
+ int severity) {
+ KeyedAppStatesReporter keyedAppStatesReporter = KeyedAppStatesReporter.create(context);
+ KeyedAppState keyedAppStateMessage = KeyedAppState.builder()
+ .setSeverity(severity)
+ .setKey(key)
+ .setMessage(message)
+ .setData(data)
+ .build();
+ List list = new ArrayList<>();
+ list.add(keyedAppStateMessage);
+ keyedAppStatesReporter.setStates(list);
+ }
+
+ private void resolveRestrictions() {
+ AgentLog.e("EMM - START resolve restrictions");
+ RestrictionsManager myRestrictionsMgr = null;
+ myRestrictionsMgr = (RestrictionsManager) getSystemService(Context.RESTRICTIONS_SERVICE);
+ Bundle appRestrictions = myRestrictionsMgr.getApplicationRestrictions();
+
+ SharedPreferences customSharedPreference = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
+ SharedPreferences.Editor editor = customSharedPreference.edit();
+
+ if (appRestrictions.containsKey("automatic_inventory")) {
+ editor.putBoolean("autoStartInventory", appRestrictions.getBoolean("automatic_inventory"));
+ enterpriseFeedback(getApplicationContext(), "automatic_inventory", "automatic_inventory option set successfully", appRestrictions.getBoolean("automatic_inventory") ? "true" : "false", KeyedAppState.SEVERITY_INFO);
+ AgentLog.e("EMM - set automatic inventory to " + appRestrictions.getBoolean("automatic_inventory"));
+ editor.apply();
+ }
+
+ if (appRestrictions.containsKey("frequency")) {
+ editor.putString("timeInventory", appRestrictions.getString("frequency"));
+ enterpriseFeedback(getApplicationContext(), "frequency", "frequency option set successfully", appRestrictions.getString("frequency"), KeyedAppState.SEVERITY_INFO);
+ AgentLog.e("EMM - set frequency to " + appRestrictions.getString("frequency"));
+ editor.apply();
+ }
+
+ if (appRestrictions.containsKey("auto_start_on_boot")) {
+ editor.putBoolean("boot", appRestrictions.getBoolean("auto_start_on_boot"));
+ enterpriseFeedback(getApplicationContext(), "auto_start_on_boot", "auto_start_on_boot option set successfully", appRestrictions.getBoolean("auto_start_on_boot") ? "true" : "false", KeyedAppState.SEVERITY_INFO);
+ AgentLog.e("EMM - set auto start on boot to " + appRestrictions.getBoolean("auto_start_on_boot"));
+ editor.apply();
+ }
+
+ Parcelable[] parcelables = appRestrictions.getParcelableArray("server_configuration_list");
+ if (parcelables != null && parcelables.length > 0) {
+ final Context context = getApplicationContext();
+ for (int i = 0; i < parcelables.length; i++) {
+ Bundle serverConfig = (Bundle) parcelables[i];
+ JSONObject jsonServerConfig = new JSONObject();
+ LocalPreferences preferences = new LocalPreferences(context);
+
+ if (serverConfig.getString("server_url").isEmpty()) {
+ enterpriseFeedback(getApplicationContext(), "server_url", "Error server URL is mandatory -> ", serverConfig.getString("server_url"), KeyedAppState.SEVERITY_ERROR);
+ AgentLog.e("EMM - server url is mandatory");
+ continue;
+ }
+
+ try {
+ jsonServerConfig.put("address", serverConfig.getString("server_url"));
+ jsonServerConfig.put("tag", serverConfig.getString("server_tag"));
+ jsonServerConfig.put("login", serverConfig.getString("server_login"));
+ jsonServerConfig.put("pass", serverConfig.getString("server_password"));
+ jsonServerConfig.put("itemtype", serverConfig.getString("server_itemtype"));
+ jsonServerConfig.put("serial", serverConfig.getString("server_custom_asset_serial"));
+
+ AgentLog.e("EMM - Receive the following configuration '" + jsonServerConfig.toString());
+
+ JSONObject local_server = preferences.loadJSONObject(serverConfig.getString("server_url"));
+ AgentLog.e("EMM - Try to load '" + serverConfig.getString("server_url") + "' server if exist");
+ AgentLog.e("EMM - Found '" + local_server.toString() + "'");
+ AgentLog.e("EMM - Exist ? -> '" + !local_server.toString().equals("{}") + "'");
+
+ if (local_server.toString().equals("{}")) {
+ ArrayList serverArray = preferences.loadServer();
+ serverArray.add(serverConfig.getString("server_url"));
+ preferences.saveServer(serverArray);
+ preferences.saveJSONObject(serverConfig.getString("server_url"), jsonServerConfig);
+ enterpriseFeedback(getApplicationContext(), "server_url", "server_url added successfully", serverConfig.getString("server_url"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_tag", "server_tag added successfully", serverConfig.getString("server_tag"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_login", "server_login added successfully", serverConfig.getString("server_login"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_password", "server_password added successfully", "***", KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_itemtype", "server_itemtype added successfully", serverConfig.getString("server_itemtype"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_custom_asset_serial", "server_custom_asset_serial added successfully", serverConfig.getString("server_custom_asset_serial"), KeyedAppState.SEVERITY_INFO);
+ AgentLog.e("EMM - Server added successfully");
+ } else {
+ preferences.deletePreferences(serverConfig.getString("server_url"));
+ preferences.saveJSONObject(serverConfig.getString("server_url"), jsonServerConfig);
+ enterpriseFeedback(getApplicationContext(), "server_url", "server_url updated successfully", serverConfig.getString("server_url"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_tag", "server_tag updated successfully", serverConfig.getString("server_tag"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_login", "server_login updated successfully", serverConfig.getString("server_login"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_password", "server_password updated successfully", "***", KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_itemtype", "server_itemtype updated successfully", serverConfig.getString("server_itemtype"), KeyedAppState.SEVERITY_INFO);
+ enterpriseFeedback(getApplicationContext(), "server_custom_asset_serial", "server_custom_asset_serial updated successfully", serverConfig.getString("server_custom_asset_serial"), KeyedAppState.SEVERITY_INFO);
+ AgentLog.e("EMM - Server updated successfully");
+ }
+
+ } catch (JSONException e) {
+ enterpriseFeedback(getApplicationContext(), "server_url", "error while adding/updating server -> " + e.getMessage(), serverConfig.getString("server_url"), KeyedAppState.SEVERITY_ERROR);
+ AgentLog.e("EMM - error while adding/updating server");
+ AgentLog.e("EMM - " + e.getMessage());
}
}
+ } else {
+ AgentLog.e("EMM - 'server_configuration_list' key is empty");
}
+ AgentLog.e("EMM - END resolve restrictions");
+ }
+
+ public void scheduleJob() {
+ SharedPreferences customSharedPreference = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
+ String timeInventory = customSharedPreference.getString("timeInventory", "Week");
+
+ final JobScheduler jobScheduler = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
+ final ComponentName name = new ComponentName(this, InventoryJobScheduler.class);
+ final JobInfo jobInfo = getJobInfo(InventoryJobScheduler.INVENTORY_JOB_ID, timeInventory, name);
+
+ if(customSharedPreference.getBoolean("autoStartInventory", false)) {
+ if (!isAlreadyScheduled(jobScheduler, jobInfo)) {
+ if (jobScheduler.schedule(jobInfo) == JobScheduler.RESULT_SUCCESS) {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Scheduled job successfully!");
+ }
+ } else {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Scheduled job already initialized");
+ }
+ } else {
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : autoStartInventory disabled, cancel JobScheduler");
+ if (isAlreadyScheduled(jobScheduler, jobInfo)) {
+ jobScheduler.cancel(InventoryJobScheduler.INVENTORY_JOB_ID);
+ }
+ }
+ }
+
+ private boolean isAlreadyScheduled(JobScheduler jobScheduler, JobInfo jobInfo) {
+ boolean jobAlreadyScheduled = false;
+ for (JobInfo existingJob: jobScheduler.getAllPendingJobs()) {
+ // check if scheduler exist with same ID and same minimum latency
+ if (existingJob.getId() == jobInfo.getId()
+ && existingJob.getMinLatencyMillis() == jobInfo.getMinLatencyMillis()) {
+ jobAlreadyScheduled = true;
+ break;
+ }
+ }
+ return jobAlreadyScheduled;
+ }
+
+ private JobInfo getJobInfo(final int id, final String timeInventory, final ComponentName name) {
+
+ //default week
+ long interval = TimeUnit.DAYS.toMillis(7);
+ if (timeInventory.equals("Day")) {
+ interval = TimeUnit.DAYS.toMillis(1);
+ } else if (timeInventory.equals("Week")) {
+ interval = TimeUnit.DAYS.toMillis(7);
+ } else if (timeInventory.equals("Month")) {
+ interval = TimeUnit.DAYS.toMillis(30);
+ }
+
+ AgentLog.d("GLPI-AGENT-JOBSCHEDULER : Alarm scheduled each " + timeInventory);
+
+ JobInfo.Builder builder = new JobInfo.Builder(id, name)
+ .setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY)
+ .setPersisted(true);
+
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
+ builder.setMinimumLatency(interval);
+ } else {
+ builder.setPeriodic(interval);
+ }
+
+ return builder.build();
}
}
diff --git a/app/src/main/java/org/glpi/inventory/agent/ui/FragmentHome.java b/app/src/main/java/org/glpi/inventory/agent/ui/FragmentHome.java
index dc2da5c92..8ae5ecba2 100644
--- a/app/src/main/java/org/glpi/inventory/agent/ui/FragmentHome.java
+++ b/app/src/main/java/org/glpi/inventory/agent/ui/FragmentHome.java
@@ -67,8 +67,6 @@ public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle sa
presenter = new HomePresenter(this);
- presenter.doBindService(FragmentHome.this.getActivity());
-
Button btn_run = v.findViewById(R.id.btn_run_inventory);
btn_run.setOnClickListener(new View.OnClickListener() {
@Override
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index fcbc8140b..cec1fd94b 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -52,6 +52,7 @@
No server found
Crash report is disable, please enable to test it
Select the format to share
+ Inventory is scheduled
- XML
- JSON