mirror of
https://codeberg.org/timelimit/timelimit-server.git
synced 2025-10-03 09:49:32 +02:00
Save manipulation details
This commit is contained in:
parent
f94f801663
commit
23ff30a1ed
9 changed files with 109 additions and 7 deletions
|
@ -15,6 +15,7 @@
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { DeviceHadManipulationFlags } from '../database/device'
|
||||||
import { assertIdWithinFamily } from '../util/token'
|
import { assertIdWithinFamily } from '../util/token'
|
||||||
import { ParentAction } from './basetypes'
|
import { ParentAction } from './basetypes'
|
||||||
|
|
||||||
|
@ -29,11 +30,12 @@ export class IgnoreManipulationAction extends ParentAction {
|
||||||
readonly ignoreAccessibilityServiceManipulation: boolean
|
readonly ignoreAccessibilityServiceManipulation: boolean
|
||||||
readonly ignoreDidReboot: boolean
|
readonly ignoreDidReboot: boolean
|
||||||
readonly ignoreHadManipulation: boolean
|
readonly ignoreHadManipulation: boolean
|
||||||
|
readonly ignoreHadManipulationFlags: number
|
||||||
|
|
||||||
constructor ({
|
constructor ({
|
||||||
deviceId, ignoreDeviceAdminManipulation, ignoreDeviceAdminManipulationAttempt,
|
deviceId, ignoreDeviceAdminManipulation, ignoreDeviceAdminManipulationAttempt,
|
||||||
ignoreAppDowngrade, ignoreNotificationAccessManipulation, ignoreUsageStatsAccessManipulation,
|
ignoreAppDowngrade, ignoreNotificationAccessManipulation, ignoreUsageStatsAccessManipulation,
|
||||||
ignoreOverlayPermissionManipulation, ignoreAccessibilityServiceManipulation, ignoreDidReboot, ignoreHadManipulation
|
ignoreOverlayPermissionManipulation, ignoreAccessibilityServiceManipulation, ignoreDidReboot, ignoreHadManipulation, ignoreHadManipulationFlags
|
||||||
}: {
|
}: {
|
||||||
deviceId: string
|
deviceId: string
|
||||||
ignoreDeviceAdminManipulation: boolean
|
ignoreDeviceAdminManipulation: boolean
|
||||||
|
@ -45,11 +47,19 @@ export class IgnoreManipulationAction extends ParentAction {
|
||||||
ignoreAccessibilityServiceManipulation: boolean
|
ignoreAccessibilityServiceManipulation: boolean
|
||||||
ignoreDidReboot: boolean
|
ignoreDidReboot: boolean
|
||||||
ignoreHadManipulation: boolean
|
ignoreHadManipulation: boolean
|
||||||
|
ignoreHadManipulationFlags: number
|
||||||
}) {
|
}) {
|
||||||
super()
|
super()
|
||||||
|
|
||||||
assertIdWithinFamily(deviceId)
|
assertIdWithinFamily(deviceId)
|
||||||
|
|
||||||
|
if (
|
||||||
|
(!Number.isSafeInteger(ignoreHadManipulationFlags)) ||
|
||||||
|
ignoreHadManipulationFlags < 0 || ignoreHadManipulationFlags > DeviceHadManipulationFlags.ALL
|
||||||
|
) {
|
||||||
|
throw new Error('invalid ignoreHadManipulationFlags')
|
||||||
|
}
|
||||||
|
|
||||||
this.deviceId = deviceId
|
this.deviceId = deviceId
|
||||||
this.ignoreDeviceAdminManipulation = ignoreDeviceAdminManipulation
|
this.ignoreDeviceAdminManipulation = ignoreDeviceAdminManipulation
|
||||||
this.ignoreDeviceAdminManipulationAttempt = ignoreDeviceAdminManipulationAttempt
|
this.ignoreDeviceAdminManipulationAttempt = ignoreDeviceAdminManipulationAttempt
|
||||||
|
@ -60,6 +70,7 @@ export class IgnoreManipulationAction extends ParentAction {
|
||||||
this.ignoreAccessibilityServiceManipulation = ignoreAccessibilityServiceManipulation
|
this.ignoreAccessibilityServiceManipulation = ignoreAccessibilityServiceManipulation
|
||||||
this.ignoreDidReboot = ignoreDidReboot
|
this.ignoreDidReboot = ignoreDidReboot
|
||||||
this.ignoreHadManipulation = ignoreHadManipulation
|
this.ignoreHadManipulation = ignoreHadManipulation
|
||||||
|
this.ignoreHadManipulationFlags = ignoreHadManipulationFlags
|
||||||
}
|
}
|
||||||
|
|
||||||
serialize = (): SerializedIgnoreManipulationAction => ({
|
serialize = (): SerializedIgnoreManipulationAction => ({
|
||||||
|
@ -72,10 +83,11 @@ export class IgnoreManipulationAction extends ParentAction {
|
||||||
overlay: this.ignoreOverlayPermissionManipulation,
|
overlay: this.ignoreOverlayPermissionManipulation,
|
||||||
accessibilityService: this.ignoreAccessibilityServiceManipulation,
|
accessibilityService: this.ignoreAccessibilityServiceManipulation,
|
||||||
usageStats: this.ignoreUsageStatsAccessManipulation,
|
usageStats: this.ignoreUsageStatsAccessManipulation,
|
||||||
hadManipulation: this.ignoreHadManipulation
|
hadManipulation: this.ignoreHadManipulation,
|
||||||
|
ignoreHadManipulationFlags: this.ignoreHadManipulationFlags
|
||||||
})
|
})
|
||||||
|
|
||||||
static parse = ({ deviceId, admin, adminA, downgrade, notification, usageStats, overlay, accessibilityService, reboot, hadManipulation }: SerializedIgnoreManipulationAction) => (
|
static parse = ({ deviceId, admin, adminA, downgrade, notification, usageStats, overlay, accessibilityService, reboot, hadManipulation, ignoreHadManipulationFlags }: SerializedIgnoreManipulationAction) => (
|
||||||
new IgnoreManipulationAction({
|
new IgnoreManipulationAction({
|
||||||
deviceId,
|
deviceId,
|
||||||
ignoreDeviceAdminManipulation: admin,
|
ignoreDeviceAdminManipulation: admin,
|
||||||
|
@ -86,7 +98,8 @@ export class IgnoreManipulationAction extends ParentAction {
|
||||||
ignoreOverlayPermissionManipulation: !!overlay,
|
ignoreOverlayPermissionManipulation: !!overlay,
|
||||||
ignoreAccessibilityServiceManipulation: !!accessibilityService,
|
ignoreAccessibilityServiceManipulation: !!accessibilityService,
|
||||||
ignoreDidReboot: !!reboot,
|
ignoreDidReboot: !!reboot,
|
||||||
ignoreHadManipulation: hadManipulation
|
ignoreHadManipulation: hadManipulation,
|
||||||
|
ignoreHadManipulationFlags: ignoreHadManipulationFlags || 0
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -104,4 +117,5 @@ export interface SerializedIgnoreManipulationAction {
|
||||||
reboot?: boolean
|
reboot?: boolean
|
||||||
overlay?: boolean
|
overlay?: boolean
|
||||||
accessibilityService?: boolean
|
accessibilityService?: boolean
|
||||||
|
ignoreHadManipulationFlags?: number
|
||||||
}
|
}
|
||||||
|
|
|
@ -339,6 +339,9 @@ const definitions = {
|
||||||
},
|
},
|
||||||
"accessibilityService": {
|
"accessibilityService": {
|
||||||
"type": "boolean"
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"ignoreHadManipulationFlags": {
|
||||||
|
"type": "number"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"additionalProperties": false,
|
"additionalProperties": false,
|
||||||
|
|
|
@ -22,6 +22,16 @@ import { RuntimePermissionStatus, runtimePermissionStatusValues } from '../model
|
||||||
import { authTokenColumn, booleanColumn, createEnumColumn, familyIdColumn, idWithinFamilyColumn, labelColumn, optionalIdWithinFamilyColumn, timestampColumn, versionColumn } from './columns'
|
import { authTokenColumn, booleanColumn, createEnumColumn, familyIdColumn, idWithinFamilyColumn, labelColumn, optionalIdWithinFamilyColumn, timestampColumn, versionColumn } from './columns'
|
||||||
import { SequelizeAttributes } from './types'
|
import { SequelizeAttributes } from './types'
|
||||||
|
|
||||||
|
export const DeviceHadManipulationFlags = {
|
||||||
|
ProtectionLevel: 1,
|
||||||
|
UsageStatsAccess: 2,
|
||||||
|
NotificationAccess: 4,
|
||||||
|
AppVersion: 8,
|
||||||
|
OverlayPermission: 16,
|
||||||
|
AccessibiityService: 32,
|
||||||
|
ALL: 1 | 2 | 4 | 8 | 16 | 32
|
||||||
|
}
|
||||||
|
|
||||||
export interface DeviceAttributesVersion1 {
|
export interface DeviceAttributesVersion1 {
|
||||||
familyId: string
|
familyId: string
|
||||||
deviceId: string
|
deviceId: string
|
||||||
|
@ -88,10 +98,14 @@ export interface DeviceAttributesVersion10 {
|
||||||
isQorLater: boolean
|
isQorLater: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface DeviceAttributesVersion11 {
|
||||||
|
hadManipulationFlags: number
|
||||||
|
}
|
||||||
|
|
||||||
export type DeviceAttributes = DeviceAttributesVersion1 & DeviceAttributesVersion2 &
|
export type DeviceAttributes = DeviceAttributesVersion1 & DeviceAttributesVersion2 &
|
||||||
DeviceAttributesVersion3 & DeviceAttributesVersion4 & DeviceAttributesVersion5 &
|
DeviceAttributesVersion3 & DeviceAttributesVersion4 & DeviceAttributesVersion5 &
|
||||||
DeviceAttributesVersion6 & DeviceAttributesVersion7 & DeviceAttributesVersion8 &
|
DeviceAttributesVersion6 & DeviceAttributesVersion7 & DeviceAttributesVersion8 &
|
||||||
DeviceAttributesVersion9 & DeviceAttributesVersion10
|
DeviceAttributesVersion9 & DeviceAttributesVersion10 & DeviceAttributesVersion11
|
||||||
|
|
||||||
export type DeviceModel = Sequelize.Model & DeviceAttributes
|
export type DeviceModel = Sequelize.Model & DeviceAttributes
|
||||||
export type DeviceModelStatic = typeof Sequelize.Model & {
|
export type DeviceModelStatic = typeof Sequelize.Model & {
|
||||||
|
@ -233,6 +247,18 @@ export const attributesVersion10: SequelizeAttributes<DeviceAttributesVersion10>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const attributesVersion11: SequelizeAttributes<DeviceAttributesVersion11> = {
|
||||||
|
hadManipulationFlags: {
|
||||||
|
type: Sequelize.INTEGER,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: 0,
|
||||||
|
validate: {
|
||||||
|
min: 0,
|
||||||
|
max: DeviceHadManipulationFlags.ALL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const attributes: SequelizeAttributes<DeviceAttributes> = {
|
export const attributes: SequelizeAttributes<DeviceAttributes> = {
|
||||||
...attributesVersion1,
|
...attributesVersion1,
|
||||||
...attributesVersion2,
|
...attributesVersion2,
|
||||||
|
@ -243,7 +269,8 @@ export const attributes: SequelizeAttributes<DeviceAttributes> = {
|
||||||
...attributesVersion7,
|
...attributesVersion7,
|
||||||
...attributesVersion8,
|
...attributesVersion8,
|
||||||
...attributesVersion9,
|
...attributesVersion9,
|
||||||
...attributesVersion10
|
...attributesVersion10,
|
||||||
|
...attributesVersion11
|
||||||
}
|
}
|
||||||
|
|
||||||
export const createDeviceModel = (sequelize: Sequelize.Sequelize): DeviceModelStatic => sequelize.define('Device', attributes) as DeviceModelStatic
|
export const createDeviceModel = (sequelize: Sequelize.Sequelize): DeviceModelStatic => sequelize.define('Device', attributes) as DeviceModelStatic
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
/*
|
||||||
|
* server component for the TimeLimit App
|
||||||
|
* Copyright (C) 2019 Jonas Lochmann
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, version 3 of the License.
|
||||||
|
*
|
||||||
|
* This program 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 Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { QueryInterface, Sequelize, Transaction } from 'sequelize'
|
||||||
|
import { attributesVersion11 as deviceAttributes } from '../../device'
|
||||||
|
|
||||||
|
export async function up (queryInterface: QueryInterface, sequelize: Sequelize) {
|
||||||
|
await sequelize.transaction({
|
||||||
|
type: Transaction.TYPES.EXCLUSIVE
|
||||||
|
}, async (transaction) => {
|
||||||
|
await queryInterface.addColumn('Devices', 'hadManipulationFlags', {
|
||||||
|
...deviceAttributes.hadManipulationFlags
|
||||||
|
}, {
|
||||||
|
transaction
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function down (queryInterface: QueryInterface, sequelize: Sequelize) {
|
||||||
|
await sequelize.transaction({
|
||||||
|
type: Transaction.TYPES.EXCLUSIVE
|
||||||
|
}, async (transaction) => {
|
||||||
|
await queryInterface.removeColumn('Devices', 'hadManipulationFlags', { transaction })
|
||||||
|
})
|
||||||
|
}
|
|
@ -48,6 +48,7 @@ export const prepareDeviceEntry = ({ familyId, userId, deviceAuthToken, deviceId
|
||||||
triedDisablingDeviceAdmin: false,
|
triedDisablingDeviceAdmin: false,
|
||||||
didReboot: false,
|
didReboot: false,
|
||||||
hadManipulation: false,
|
hadManipulation: false,
|
||||||
|
hadManipulationFlags: 0,
|
||||||
lastConnectivity: '0',
|
lastConnectivity: '0',
|
||||||
notSeenForLongTime: false,
|
notSeenForLongTime: false,
|
||||||
didDeviceReportUninstall: false,
|
didDeviceReportUninstall: false,
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { UpdateDeviceStatusAction } from '../../../../action'
|
import { UpdateDeviceStatusAction } from '../../../../action'
|
||||||
import { hasDeviceManipulation } from '../../../../database/device'
|
import { DeviceHadManipulationFlags, hasDeviceManipulation } from '../../../../database/device'
|
||||||
import { newPermissionStatusValues } from '../../../../model/newpermissionstatus'
|
import { newPermissionStatusValues } from '../../../../model/newpermissionstatus'
|
||||||
import { protetionLevels } from '../../../../model/protectionlevel'
|
import { protetionLevels } from '../../../../model/protectionlevel'
|
||||||
import { runtimePermissionStatusValues } from '../../../../model/runtimepermissionstatus'
|
import { runtimePermissionStatusValues } from '../../../../model/runtimepermissionstatus'
|
||||||
|
@ -56,6 +56,7 @@ export async function dispatchUpdateDeviceStatus ({ deviceId, action, cache }: {
|
||||||
|
|
||||||
if (hasChanged && (deviceEntry.currentProtectionLevel !== deviceEntry.highestProtectionLevel)) {
|
if (hasChanged && (deviceEntry.currentProtectionLevel !== deviceEntry.highestProtectionLevel)) {
|
||||||
deviceEntry.hadManipulation = true
|
deviceEntry.hadManipulation = true
|
||||||
|
deviceEntry.hadManipulationFlags |= DeviceHadManipulationFlags.ProtectionLevel
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -72,6 +73,7 @@ export async function dispatchUpdateDeviceStatus ({ deviceId, action, cache }: {
|
||||||
|
|
||||||
if (hasChanged && (deviceEntry.currentUsageStatsPermission !== deviceEntry.highestUsageStatsPermission)) {
|
if (hasChanged && (deviceEntry.currentUsageStatsPermission !== deviceEntry.highestUsageStatsPermission)) {
|
||||||
deviceEntry.hadManipulation = true
|
deviceEntry.hadManipulation = true
|
||||||
|
deviceEntry.hadManipulationFlags |= DeviceHadManipulationFlags.UsageStatsAccess
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,6 +90,7 @@ export async function dispatchUpdateDeviceStatus ({ deviceId, action, cache }: {
|
||||||
|
|
||||||
if (hasChanged && (deviceEntry.currentNotificationAccessPermission !== deviceEntry.highestNotificationAccessPermission)) {
|
if (hasChanged && (deviceEntry.currentNotificationAccessPermission !== deviceEntry.highestNotificationAccessPermission)) {
|
||||||
deviceEntry.hadManipulation = true
|
deviceEntry.hadManipulation = true
|
||||||
|
deviceEntry.hadManipulationFlags |= DeviceHadManipulationFlags.NotificationAccess
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,6 +107,7 @@ export async function dispatchUpdateDeviceStatus ({ deviceId, action, cache }: {
|
||||||
|
|
||||||
if (hasChanged && (deviceEntry.currentOverlayPermission !== deviceEntry.highestOverlayPermission)) {
|
if (hasChanged && (deviceEntry.currentOverlayPermission !== deviceEntry.highestOverlayPermission)) {
|
||||||
deviceEntry.hadManipulation = true
|
deviceEntry.hadManipulation = true
|
||||||
|
deviceEntry.hadManipulationFlags |= DeviceHadManipulationFlags.OverlayPermission
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -118,6 +122,7 @@ export async function dispatchUpdateDeviceStatus ({ deviceId, action, cache }: {
|
||||||
|
|
||||||
if (hasChanged && (deviceEntry.asEnabled !== deviceEntry.wasAsEnabled)) {
|
if (hasChanged && (deviceEntry.asEnabled !== deviceEntry.wasAsEnabled)) {
|
||||||
deviceEntry.hadManipulation = true
|
deviceEntry.hadManipulation = true
|
||||||
|
deviceEntry.hadManipulationFlags |= DeviceHadManipulationFlags.AccessibiityService
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -133,6 +138,7 @@ export async function dispatchUpdateDeviceStatus ({ deviceId, action, cache }: {
|
||||||
|
|
||||||
if (hasChanged && (deviceEntry.currentAppVersion !== deviceEntry.highestAppVersion)) {
|
if (hasChanged && (deviceEntry.currentAppVersion !== deviceEntry.highestAppVersion)) {
|
||||||
deviceEntry.hadManipulation = true
|
deviceEntry.hadManipulation = true
|
||||||
|
deviceEntry.hadManipulationFlags |= DeviceHadManipulationFlags.AppVersion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -70,6 +70,16 @@ export async function dispatchIgnoreManipulation ({ action, cache }: {
|
||||||
deviceEntry.hadManipulation = false
|
deviceEntry.hadManipulation = false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (action.ignoreHadManipulationFlags !== 0) {
|
||||||
|
const newFlags = deviceEntry.hadManipulationFlags & (~action.ignoreHadManipulationFlags)
|
||||||
|
|
||||||
|
deviceEntry.hadManipulationFlags = newFlags
|
||||||
|
|
||||||
|
if (newFlags === 0) {
|
||||||
|
deviceEntry.hadManipulation = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
await deviceEntry.save({ transaction: cache.transaction })
|
await deviceEntry.save({ transaction: cache.transaction })
|
||||||
cache.invalidiateDeviceList = true
|
cache.invalidiateDeviceList = true
|
||||||
}
|
}
|
||||||
|
|
|
@ -89,6 +89,7 @@ export const generateServerDataStatus = async ({ database, clientStatus, familyI
|
||||||
tDisablingAdmin: item.triedDisablingDeviceAdmin,
|
tDisablingAdmin: item.triedDisablingDeviceAdmin,
|
||||||
reboot: item.didReboot,
|
reboot: item.didReboot,
|
||||||
hadManipulation: item.hadManipulation,
|
hadManipulation: item.hadManipulation,
|
||||||
|
hadManipulationFlags: item.hadManipulationFlags,
|
||||||
reportUninstall: item.didDeviceReportUninstall,
|
reportUninstall: item.didDeviceReportUninstall,
|
||||||
isUserKeptSignedIn: item.isUserKeptSignedIn,
|
isUserKeptSignedIn: item.isUserKeptSignedIn,
|
||||||
showDeviceConnected: item.showDeviceConnected,
|
showDeviceConnected: item.showDeviceConnected,
|
||||||
|
|
|
@ -77,6 +77,7 @@ export interface ServerDeviceData {
|
||||||
tDisablingAdmin: boolean
|
tDisablingAdmin: boolean
|
||||||
reboot: boolean
|
reboot: boolean
|
||||||
hadManipulation: boolean
|
hadManipulation: boolean
|
||||||
|
hadManipulationFlags: number
|
||||||
reportUninstall: boolean
|
reportUninstall: boolean
|
||||||
isUserKeptSignedIn: boolean
|
isUserKeptSignedIn: boolean
|
||||||
showDeviceConnected: boolean
|
showDeviceConnected: boolean
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue