Save manipulation details

This commit is contained in:
Jonas Lochmann 2019-08-05 00:00:00 +00:00
parent f94f801663
commit 23ff30a1ed
No known key found for this signature in database
GPG key ID: 8B8C9AEE10FA5B36
9 changed files with 109 additions and 7 deletions

View file

@ -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
} }

View file

@ -339,6 +339,9 @@ const definitions = {
}, },
"accessibilityService": { "accessibilityService": {
"type": "boolean" "type": "boolean"
},
"ignoreHadManipulationFlags": {
"type": "number"
} }
}, },
"additionalProperties": false, "additionalProperties": false,

View file

@ -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

View file

@ -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 })
})
}

View file

@ -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,

View file

@ -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
} }
} }

View file

@ -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
} }

View file

@ -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,

View file

@ -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