mirror of
https://codeberg.org/timelimit/timelimit-server.git
synced 2025-10-06 03:50:05 +02:00
Add support for parent blocked times
This commit is contained in:
parent
32d278bdd5
commit
1969fe4042
15 changed files with 363 additions and 8 deletions
|
@ -34,6 +34,7 @@ export { RemoveCategoryAppsAction } from './removecategoryapps'
|
||||||
export { RemoveInstalledAppsAction } from './removeinstalledapps'
|
export { RemoveInstalledAppsAction } from './removeinstalledapps'
|
||||||
export { RemoveUserAction } from './removeuser'
|
export { RemoveUserAction } from './removeuser'
|
||||||
export { RenameChildAction } from './renamechild'
|
export { RenameChildAction } from './renamechild'
|
||||||
|
export { ResetParentBlockedTimesAction } from './resetparentblockedtimes'
|
||||||
export { SetCategoryExtraTimeAction } from './setcategoryextratime'
|
export { SetCategoryExtraTimeAction } from './setcategoryextratime'
|
||||||
export { SetCategoryForUnassignedAppsAction } from './setcategoryforunassignedapps'
|
export { SetCategoryForUnassignedAppsAction } from './setcategoryforunassignedapps'
|
||||||
export { SetChildPasswordAction } from './setchildpassword'
|
export { SetChildPasswordAction } from './setchildpassword'
|
||||||
|
@ -59,5 +60,6 @@ export { UpdateDeviceNameAction } from './updatedevicename'
|
||||||
export { UpdateDeviceStatusAction } from './updatedevicestatus'
|
export { UpdateDeviceStatusAction } from './updatedevicestatus'
|
||||||
export { UpdateEnableActivityLevelBlockingAction } from './updateenableactivitylevelblocking'
|
export { UpdateEnableActivityLevelBlockingAction } from './updateenableactivitylevelblocking'
|
||||||
export { UpdateNetworkTimeVerificationAction } from './updatenetworktimeverification'
|
export { UpdateNetworkTimeVerificationAction } from './updatenetworktimeverification'
|
||||||
|
export { UpdateParentBlockedTimesAction } from './updateparentblockedtimes'
|
||||||
export { UpdateParentNotificationFlagsAction } from './updateparentnotificationflags'
|
export { UpdateParentNotificationFlagsAction } from './updateparentnotificationflags'
|
||||||
export { UpdateTimelimitRuleAction } from './updatetimelimitrule'
|
export { UpdateTimelimitRuleAction } from './updatetimelimitrule'
|
||||||
|
|
49
src/action/resetparentblockedtimes.ts
Normal file
49
src/action/resetparentblockedtimes.ts
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* 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 { assertIdWithinFamily } from '../util/token'
|
||||||
|
import { ParentAction } from './basetypes'
|
||||||
|
|
||||||
|
export class ResetParentBlockedTimesAction extends ParentAction {
|
||||||
|
readonly parentId: string
|
||||||
|
|
||||||
|
constructor ({ parentId }: {
|
||||||
|
parentId: string
|
||||||
|
}) {
|
||||||
|
super()
|
||||||
|
|
||||||
|
assertIdWithinFamily(parentId)
|
||||||
|
|
||||||
|
this.parentId = parentId
|
||||||
|
}
|
||||||
|
|
||||||
|
serialize = (): SerializedResetParentBlockedTimesAction => ({
|
||||||
|
type: 'RESET_PARENT_BLOCKED_TIMES',
|
||||||
|
parentId: this.parentId
|
||||||
|
})
|
||||||
|
|
||||||
|
static parse = ({ parentId }: SerializedResetParentBlockedTimesAction) => (
|
||||||
|
new ResetParentBlockedTimesAction({
|
||||||
|
parentId
|
||||||
|
})
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SerializedResetParentBlockedTimesAction {
|
||||||
|
type: 'RESET_PARENT_BLOCKED_TIMES'
|
||||||
|
parentId: string
|
||||||
|
}
|
|
@ -28,6 +28,7 @@ import { IncrementCategoryExtraTimeAction, SerializedIncrementCategoryExtraTimeA
|
||||||
import { RemoveCategoryAppsAction, SerializedRemoveCategoryAppsAction } from '../removecategoryapps'
|
import { RemoveCategoryAppsAction, SerializedRemoveCategoryAppsAction } from '../removecategoryapps'
|
||||||
import { RemoveUserAction, SerializedRemoveUserAction } from '../removeuser'
|
import { RemoveUserAction, SerializedRemoveUserAction } from '../removeuser'
|
||||||
import { RenameChildAction, SerializedRenameChildAction } from '../renamechild'
|
import { RenameChildAction, SerializedRenameChildAction } from '../renamechild'
|
||||||
|
import { ResetParentBlockedTimesAction, SerializedResetParentBlockedTimesAction } from '../resetparentblockedtimes'
|
||||||
import { SerializedSetCategoryExtraTimeAction, SetCategoryExtraTimeAction } from '../setcategoryextratime'
|
import { SerializedSetCategoryExtraTimeAction, SetCategoryExtraTimeAction } from '../setcategoryextratime'
|
||||||
import { SerializedSetCategoryForUnassignedAppsAction, SetCategoryForUnassignedAppsAction } from '../setcategoryforunassignedapps'
|
import { SerializedSetCategoryForUnassignedAppsAction, SetCategoryForUnassignedAppsAction } from '../setcategoryforunassignedapps'
|
||||||
import { SerializedSetChildPasswordAction, SetChildPasswordAction } from '../setchildpassword'
|
import { SerializedSetChildPasswordAction, SetChildPasswordAction } from '../setchildpassword'
|
||||||
|
@ -49,6 +50,7 @@ import { SerializedUpdateCategoryTitleAction, UpdateCategoryTitleAction } from '
|
||||||
import { SerializedUpdateDeviceNameAction, UpdateDeviceNameAction } from '../updatedevicename'
|
import { SerializedUpdateDeviceNameAction, UpdateDeviceNameAction } from '../updatedevicename'
|
||||||
import { SerializedUpdateEnableActivityLevelBlockingAction, UpdateEnableActivityLevelBlockingAction } from '../updateenableactivitylevelblocking'
|
import { SerializedUpdateEnableActivityLevelBlockingAction, UpdateEnableActivityLevelBlockingAction } from '../updateenableactivitylevelblocking'
|
||||||
import { SerialiizedUpdateNetworkTimeVerificationAction, UpdateNetworkTimeVerificationAction } from '../updatenetworktimeverification'
|
import { SerialiizedUpdateNetworkTimeVerificationAction, UpdateNetworkTimeVerificationAction } from '../updatenetworktimeverification'
|
||||||
|
import { SerializedUpdateParentBlockedTimesAction, UpdateParentBlockedTimesAction } from '../updateparentblockedtimes'
|
||||||
import { SerializedUpdateParentNotificationFlagsAction, UpdateParentNotificationFlagsAction } from '../updateparentnotificationflags'
|
import { SerializedUpdateParentNotificationFlagsAction, UpdateParentNotificationFlagsAction } from '../updateparentnotificationflags'
|
||||||
import { SerializedUpdateTimelimitRuleAction, UpdateTimelimitRuleAction } from '../updatetimelimitrule'
|
import { SerializedUpdateTimelimitRuleAction, UpdateTimelimitRuleAction } from '../updatetimelimitrule'
|
||||||
|
|
||||||
|
@ -65,6 +67,7 @@ export type SerializedParentAction =
|
||||||
SerializedRemoveCategoryAppsAction |
|
SerializedRemoveCategoryAppsAction |
|
||||||
SerializedRemoveUserAction |
|
SerializedRemoveUserAction |
|
||||||
SerializedRenameChildAction |
|
SerializedRenameChildAction |
|
||||||
|
SerializedResetParentBlockedTimesAction |
|
||||||
SerializedSetCategoryForUnassignedAppsAction |
|
SerializedSetCategoryForUnassignedAppsAction |
|
||||||
SerializedSetChildPasswordAction |
|
SerializedSetChildPasswordAction |
|
||||||
SerializedSetConsiderRebootManipulationAction |
|
SerializedSetConsiderRebootManipulationAction |
|
||||||
|
@ -86,6 +89,7 @@ export type SerializedParentAction =
|
||||||
SerializedUpdateDeviceNameAction |
|
SerializedUpdateDeviceNameAction |
|
||||||
SerializedUpdateEnableActivityLevelBlockingAction |
|
SerializedUpdateEnableActivityLevelBlockingAction |
|
||||||
SerialiizedUpdateNetworkTimeVerificationAction |
|
SerialiizedUpdateNetworkTimeVerificationAction |
|
||||||
|
SerializedUpdateParentBlockedTimesAction |
|
||||||
SerializedUpdateParentNotificationFlagsAction |
|
SerializedUpdateParentNotificationFlagsAction |
|
||||||
SerializedUpdateTimelimitRuleAction
|
SerializedUpdateTimelimitRuleAction
|
||||||
|
|
||||||
|
@ -114,6 +118,8 @@ export const parseParentAction = (action: SerializedParentAction): ParentAction
|
||||||
return RemoveUserAction.parse(action)
|
return RemoveUserAction.parse(action)
|
||||||
} else if (action.type === 'RENAME_CHILD') {
|
} else if (action.type === 'RENAME_CHILD') {
|
||||||
return RenameChildAction.parse(action)
|
return RenameChildAction.parse(action)
|
||||||
|
} else if (action.type === 'RESET_PARENT_BLOCKED_TIMES') {
|
||||||
|
return ResetParentBlockedTimesAction.parse(action)
|
||||||
} else if (action.type === 'SET_CATEGORY_EXTRA_TIME') {
|
} else if (action.type === 'SET_CATEGORY_EXTRA_TIME') {
|
||||||
return SetCategoryExtraTimeAction.parse(action)
|
return SetCategoryExtraTimeAction.parse(action)
|
||||||
} else if (action.type === 'SET_CATEGORY_FOR_UNASSIGNED_APPS') {
|
} else if (action.type === 'SET_CATEGORY_FOR_UNASSIGNED_APPS') {
|
||||||
|
@ -156,6 +162,8 @@ export const parseParentAction = (action: SerializedParentAction): ParentAction
|
||||||
return UpdateEnableActivityLevelBlockingAction.parse(action)
|
return UpdateEnableActivityLevelBlockingAction.parse(action)
|
||||||
} else if (action.type === 'UPDATE_NETWORK_TIME_VERIFICATION') {
|
} else if (action.type === 'UPDATE_NETWORK_TIME_VERIFICATION') {
|
||||||
return UpdateNetworkTimeVerificationAction.parse(action)
|
return UpdateNetworkTimeVerificationAction.parse(action)
|
||||||
|
} else if (action.type === 'UPDATE_PARENT_BLOCKED_TIMES') {
|
||||||
|
return UpdateParentBlockedTimesAction.parse(action)
|
||||||
} else if (action.type === 'UPDATE_PARENT_NOTIFICATION_FLAGS') {
|
} else if (action.type === 'UPDATE_PARENT_NOTIFICATION_FLAGS') {
|
||||||
return UpdateParentNotificationFlagsAction.parse(action)
|
return UpdateParentNotificationFlagsAction.parse(action)
|
||||||
} else if (action.type === 'UPDATE_TIMELIMIT_RULE') {
|
} else if (action.type === 'UPDATE_TIMELIMIT_RULE') {
|
||||||
|
|
74
src/action/updateparentblockedtimes.ts
Normal file
74
src/action/updateparentblockedtimes.ts
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
/*
|
||||||
|
* 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 { validateAndParseBitmask } from '../util/bitmask'
|
||||||
|
import { assertIdWithinFamily } from '../util/token'
|
||||||
|
import { ParentAction } from './basetypes'
|
||||||
|
|
||||||
|
export class UpdateParentBlockedTimesAction extends ParentAction {
|
||||||
|
readonly parentId: string
|
||||||
|
readonly blockedTimes: string
|
||||||
|
|
||||||
|
constructor ({ parentId, blockedTimes }: {
|
||||||
|
parentId: string
|
||||||
|
blockedTimes: string
|
||||||
|
}) {
|
||||||
|
super()
|
||||||
|
|
||||||
|
assertIdWithinFamily(parentId)
|
||||||
|
|
||||||
|
{
|
||||||
|
const parsedBlockedTimes = validateAndParseBitmask(blockedTimes, 60 * 24 * 7 /* number of minutes per week */)
|
||||||
|
|
||||||
|
for (let day = 0; day < 7; day++) {
|
||||||
|
let blockedMinutes = 0
|
||||||
|
|
||||||
|
for (let minute = 0; minute < 60 * 24 /* 1 day */; minute++) {
|
||||||
|
if (parsedBlockedTimes[day * 60 * 24 + minute]) {
|
||||||
|
blockedMinutes++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (blockedMinutes > 60 * 18 /* 18 hours */) {
|
||||||
|
throw new Error('too much blocked minutes per day')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.parentId = parentId
|
||||||
|
this.blockedTimes = blockedTimes
|
||||||
|
}
|
||||||
|
|
||||||
|
serialize = (): SerializedUpdateParentBlockedTimesAction => ({
|
||||||
|
type: 'UPDATE_PARENT_BLOCKED_TIMES',
|
||||||
|
parentId: this.parentId,
|
||||||
|
times: this.blockedTimes
|
||||||
|
})
|
||||||
|
|
||||||
|
static parse = ({ parentId, times }: SerializedUpdateParentBlockedTimesAction) => (
|
||||||
|
new UpdateParentBlockedTimesAction({
|
||||||
|
parentId,
|
||||||
|
blockedTimes: times
|
||||||
|
})
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SerializedUpdateParentBlockedTimesAction {
|
||||||
|
type: 'UPDATE_PARENT_BLOCKED_TIMES'
|
||||||
|
parentId: string
|
||||||
|
times: string
|
||||||
|
}
|
|
@ -450,6 +450,25 @@ const definitions = {
|
||||||
"type"
|
"type"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"SerializedResetParentBlockedTimesAction": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"RESET_PARENT_BLOCKED_TIMES"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"parentId": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": false,
|
||||||
|
"required": [
|
||||||
|
"parentId",
|
||||||
|
"type"
|
||||||
|
]
|
||||||
|
},
|
||||||
"SerializedSetCategoryExtraTimeAction": {
|
"SerializedSetCategoryExtraTimeAction": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
|
@ -942,6 +961,29 @@ const definitions = {
|
||||||
"type"
|
"type"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"SerializedUpdateParentBlockedTimesAction": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"UPDATE_PARENT_BLOCKED_TIMES"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"parentId": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"times": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": false,
|
||||||
|
"required": [
|
||||||
|
"parentId",
|
||||||
|
"times",
|
||||||
|
"type"
|
||||||
|
]
|
||||||
|
},
|
||||||
"SerializedUpdateParentNotificationFlagsAction": {
|
"SerializedUpdateParentNotificationFlagsAction": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
|
@ -1535,6 +1577,9 @@ export const isSerializedParentAction: (value: object) => value is SerializedPar
|
||||||
{
|
{
|
||||||
"$ref": "#/definitions/SerializedRenameChildAction"
|
"$ref": "#/definitions/SerializedRenameChildAction"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/SerializedResetParentBlockedTimesAction"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"$ref": "#/definitions/SerializedSetCategoryExtraTimeAction"
|
"$ref": "#/definitions/SerializedSetCategoryExtraTimeAction"
|
||||||
},
|
},
|
||||||
|
@ -1598,6 +1643,9 @@ export const isSerializedParentAction: (value: object) => value is SerializedPar
|
||||||
{
|
{
|
||||||
"$ref": "#/definitions/SerialiizedUpdateNetworkTimeVerificationAction"
|
"$ref": "#/definitions/SerialiizedUpdateNetworkTimeVerificationAction"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/SerializedUpdateParentBlockedTimesAction"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"$ref": "#/definitions/SerializedUpdateParentNotificationFlagsAction"
|
"$ref": "#/definitions/SerializedUpdateParentNotificationFlagsAction"
|
||||||
},
|
},
|
||||||
|
|
|
@ -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 { attributesVersion5 as userAttributes } from '../../user'
|
||||||
|
|
||||||
|
export async function up (queryInterface: QueryInterface, sequelize: Sequelize) {
|
||||||
|
await sequelize.transaction({
|
||||||
|
type: Transaction.TYPES.EXCLUSIVE
|
||||||
|
}, async (transaction) => {
|
||||||
|
await queryInterface.addColumn('Users', 'blockedTimes', {
|
||||||
|
...userAttributes.blockedTimes
|
||||||
|
}, {
|
||||||
|
transaction
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function down (queryInterface: QueryInterface, sequelize: Sequelize) {
|
||||||
|
await sequelize.transaction({
|
||||||
|
type: Transaction.TYPES.EXCLUSIVE
|
||||||
|
}, async (transaction) => {
|
||||||
|
await queryInterface.removeColumn('Users', 'blockedTimes', { transaction })
|
||||||
|
})
|
||||||
|
}
|
|
@ -16,6 +16,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import * as Sequelize from 'sequelize'
|
import * as Sequelize from 'sequelize'
|
||||||
|
import { serializedBitmaskRegex } from '../util/bitmask'
|
||||||
import { optionalPasswordRegex, optionalSaltRegex } from '../util/password'
|
import { optionalPasswordRegex, optionalSaltRegex } from '../util/password'
|
||||||
import { booleanColumn, createEnumColumn, familyIdColumn, idWithinFamilyColumn, labelColumn, optionalIdWithinFamilyColumn, timestampColumn } from './columns'
|
import { booleanColumn, createEnumColumn, familyIdColumn, idWithinFamilyColumn, labelColumn, optionalIdWithinFamilyColumn, timestampColumn } from './columns'
|
||||||
import { SequelizeAttributes } from './types'
|
import { SequelizeAttributes } from './types'
|
||||||
|
@ -49,8 +50,12 @@ export interface UserAttributesVersion4 {
|
||||||
mailNotificationFlags: number
|
mailNotificationFlags: number
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface UserAttributesVersion5 {
|
||||||
|
blockedTimes: string
|
||||||
|
}
|
||||||
|
|
||||||
export type UserAttributes = UserAttributesVersion1 & UserAttributesVersion2 &
|
export type UserAttributes = UserAttributesVersion1 & UserAttributesVersion2 &
|
||||||
UserAttributesVersion3 & UserAttributesVersion4
|
UserAttributesVersion3 & UserAttributesVersion4 & UserAttributesVersion5
|
||||||
|
|
||||||
export type UserModel = Sequelize.Model & UserAttributes
|
export type UserModel = Sequelize.Model & UserAttributes
|
||||||
export type UserModelStatic = typeof Sequelize.Model & {
|
export type UserModelStatic = typeof Sequelize.Model & {
|
||||||
|
@ -123,11 +128,23 @@ export const attributesVersion4: SequelizeAttributes<UserAttributesVersion4> = {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const attributesVersion5: SequelizeAttributes<UserAttributesVersion5> = {
|
||||||
|
blockedTimes: {
|
||||||
|
type: Sequelize.TEXT,
|
||||||
|
allowNull: false,
|
||||||
|
defaultValue: '',
|
||||||
|
validate: {
|
||||||
|
is: serializedBitmaskRegex
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const attributes: SequelizeAttributes<UserAttributes> = {
|
export const attributes: SequelizeAttributes<UserAttributes> = {
|
||||||
...attributesVersion1,
|
...attributesVersion1,
|
||||||
...attributesVersion2,
|
...attributesVersion2,
|
||||||
...attributesVersion3,
|
...attributesVersion3,
|
||||||
...attributesVersion4
|
...attributesVersion4,
|
||||||
|
...attributesVersion5
|
||||||
}
|
}
|
||||||
|
|
||||||
export const createUserModel = (sequelize: Sequelize.Sequelize): UserModelStatic => sequelize.define('User', attributes) as UserModelStatic
|
export const createUserModel = (sequelize: Sequelize.Sequelize): UserModelStatic => sequelize.define('User', attributes) as UserModelStatic
|
||||||
|
|
|
@ -81,7 +81,8 @@ export const createFamily = async ({ database, mailAuthToken, firstParentDevice,
|
||||||
currentDevice: '',
|
currentDevice: '',
|
||||||
categoryForNotAssignedApps: '',
|
categoryForNotAssignedApps: '',
|
||||||
relaxPrimaryDeviceRule: false,
|
relaxPrimaryDeviceRule: false,
|
||||||
mailNotificationFlags: 1 // enable warning notifications
|
mailNotificationFlags: 1, // enable warning notifications
|
||||||
|
blockedTimes: ''
|
||||||
}, { transaction })
|
}, { transaction })
|
||||||
|
|
||||||
// add parent device
|
// add parent device
|
||||||
|
|
|
@ -36,7 +36,8 @@ export async function dispatchAddUser ({ action, cache }: {
|
||||||
currentDevice: '',
|
currentDevice: '',
|
||||||
categoryForNotAssignedApps: '',
|
categoryForNotAssignedApps: '',
|
||||||
relaxPrimaryDeviceRule: false,
|
relaxPrimaryDeviceRule: false,
|
||||||
mailNotificationFlags: 0
|
mailNotificationFlags: 0,
|
||||||
|
blockedTimes: ''
|
||||||
}, { transaction: cache.transaction })
|
}, { transaction: cache.transaction })
|
||||||
|
|
||||||
cache.invalidiateUserList = true
|
cache.invalidiateUserList = true
|
||||||
|
|
|
@ -29,6 +29,7 @@ import {
|
||||||
RemoveCategoryAppsAction,
|
RemoveCategoryAppsAction,
|
||||||
RemoveUserAction,
|
RemoveUserAction,
|
||||||
RenameChildAction,
|
RenameChildAction,
|
||||||
|
ResetParentBlockedTimesAction,
|
||||||
SetCategoryExtraTimeAction,
|
SetCategoryExtraTimeAction,
|
||||||
SetCategoryForUnassignedAppsAction,
|
SetCategoryForUnassignedAppsAction,
|
||||||
SetChildPasswordAction,
|
SetChildPasswordAction,
|
||||||
|
@ -50,6 +51,7 @@ import {
|
||||||
UpdateDeviceNameAction,
|
UpdateDeviceNameAction,
|
||||||
UpdateEnableActivityLevelBlockingAction,
|
UpdateEnableActivityLevelBlockingAction,
|
||||||
UpdateNetworkTimeVerificationAction,
|
UpdateNetworkTimeVerificationAction,
|
||||||
|
UpdateParentBlockedTimesAction,
|
||||||
UpdateParentNotificationFlagsAction,
|
UpdateParentNotificationFlagsAction,
|
||||||
UpdateTimelimitRuleAction
|
UpdateTimelimitRuleAction
|
||||||
} from '../../../../action'
|
} from '../../../../action'
|
||||||
|
@ -66,6 +68,7 @@ import { dispatchIncrementCategoryExtraTime } from './incrementcategoryextratime
|
||||||
import { dispatchRemoveCategoryApps } from './removecategoryapps'
|
import { dispatchRemoveCategoryApps } from './removecategoryapps'
|
||||||
import { dispatchRemoveUser } from './removeuser'
|
import { dispatchRemoveUser } from './removeuser'
|
||||||
import { dispatchRenameChild } from './renamechild'
|
import { dispatchRenameChild } from './renamechild'
|
||||||
|
import { dispatchResetParentBlockedTimes } from './resetparentblockedtimes'
|
||||||
import { dispatchSetCategoryExtraTime } from './setcategoryextratime'
|
import { dispatchSetCategoryExtraTime } from './setcategoryextratime'
|
||||||
import { dispatchSetCategoryForUnassignedApps } from './setcategoryforunassignedapps'
|
import { dispatchSetCategoryForUnassignedApps } from './setcategoryforunassignedapps'
|
||||||
import { dispatchSetChildPassword } from './setchildpassword'
|
import { dispatchSetChildPassword } from './setchildpassword'
|
||||||
|
@ -87,6 +90,7 @@ import { dispatchUpdateCategoryTitle } from './updatecategorytitle'
|
||||||
import { dispatchUpdateDeviceName } from './updatedevicename'
|
import { dispatchUpdateDeviceName } from './updatedevicename'
|
||||||
import { dispatchUpdateEnableActivityLevelBlocking } from './updateenableactivitylevelblocking'
|
import { dispatchUpdateEnableActivityLevelBlocking } from './updateenableactivitylevelblocking'
|
||||||
import { dispatchUpdateNetworkTimeVerification } from './updatenetworktimeverification'
|
import { dispatchUpdateNetworkTimeVerification } from './updatenetworktimeverification'
|
||||||
|
import { dispatchUpdateParentBlockedTimes } from './updateparentblockedtimes'
|
||||||
import { dispatchUpdateParentNotificationFlags } from './updateparentnotificationflags'
|
import { dispatchUpdateParentNotificationFlags } from './updateparentnotificationflags'
|
||||||
import { dispatchUpdateTimelimitRule } from './updatetimelimitrule'
|
import { dispatchUpdateTimelimitRule } from './updatetimelimitrule'
|
||||||
|
|
||||||
|
@ -166,6 +170,10 @@ export const dispatchParentAction = async ({ action, cache, parentUserId, source
|
||||||
await dispatchIgnoreManipulation({ action, cache })
|
await dispatchIgnoreManipulation({ action, cache })
|
||||||
} else if (action instanceof UpdateCategoryTimeWarningsAction) {
|
} else if (action instanceof UpdateCategoryTimeWarningsAction) {
|
||||||
await dispatchUpdateCategoryTimeWarnings({ action, cache })
|
await dispatchUpdateCategoryTimeWarnings({ action, cache })
|
||||||
|
} else if (action instanceof ResetParentBlockedTimesAction) {
|
||||||
|
await dispatchResetParentBlockedTimes({ action, cache })
|
||||||
|
} else if (action instanceof UpdateParentBlockedTimesAction) {
|
||||||
|
await dispatchUpdateParentBlockedTimes({ action, cache })
|
||||||
} else {
|
} else {
|
||||||
throw new Error('unsupported action type')
|
throw new Error('unsupported action type')
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* 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 { ResetParentBlockedTimesAction } from '../../../../action'
|
||||||
|
import { Cache } from '../cache'
|
||||||
|
|
||||||
|
export async function dispatchResetParentBlockedTimes ({ action, cache }: {
|
||||||
|
action: ResetParentBlockedTimesAction
|
||||||
|
cache: Cache
|
||||||
|
}) {
|
||||||
|
const [affectedRows] = await cache.database.user.update({
|
||||||
|
blockedTimes: ''
|
||||||
|
}, {
|
||||||
|
where: {
|
||||||
|
familyId: cache.familyId,
|
||||||
|
userId: action.parentId,
|
||||||
|
type: 'parent'
|
||||||
|
},
|
||||||
|
transaction: cache.transaction
|
||||||
|
})
|
||||||
|
|
||||||
|
if (affectedRows === 0) {
|
||||||
|
throw new Error('invalid parent user id provided')
|
||||||
|
}
|
||||||
|
|
||||||
|
cache.invalidiateUserList = true
|
||||||
|
cache.areChangesImportant = true
|
||||||
|
}
|
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* 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 { UpdateParentBlockedTimesAction } from '../../../../action'
|
||||||
|
import { Cache } from '../cache'
|
||||||
|
|
||||||
|
export async function dispatchUpdateParentBlockedTimes ({ action, cache }: {
|
||||||
|
action: UpdateParentBlockedTimesAction
|
||||||
|
cache: Cache
|
||||||
|
}) {
|
||||||
|
const [affectedRows] = await cache.database.user.update({
|
||||||
|
blockedTimes: action.blockedTimes
|
||||||
|
}, {
|
||||||
|
where: {
|
||||||
|
familyId: cache.familyId,
|
||||||
|
userId: action.parentId,
|
||||||
|
type: 'parent'
|
||||||
|
},
|
||||||
|
transaction: cache.transaction
|
||||||
|
})
|
||||||
|
|
||||||
|
if (affectedRows === 0) {
|
||||||
|
throw new Error('invalid parent user id provided')
|
||||||
|
}
|
||||||
|
|
||||||
|
cache.invalidiateUserList = true
|
||||||
|
cache.areChangesImportant = true
|
||||||
|
}
|
|
@ -123,7 +123,8 @@ export const generateServerDataStatus = async ({ database, clientStatus, familyI
|
||||||
'currentDevice',
|
'currentDevice',
|
||||||
'categoryForNotAssignedApps',
|
'categoryForNotAssignedApps',
|
||||||
'relaxPrimaryDeviceRule',
|
'relaxPrimaryDeviceRule',
|
||||||
'mailNotificationFlags'
|
'mailNotificationFlags',
|
||||||
|
'blockedTimes'
|
||||||
],
|
],
|
||||||
transaction
|
transaction
|
||||||
})).map((item) => ({
|
})).map((item) => ({
|
||||||
|
@ -138,7 +139,8 @@ export const generateServerDataStatus = async ({ database, clientStatus, familyI
|
||||||
currentDevice: item.currentDevice,
|
currentDevice: item.currentDevice,
|
||||||
categoryForNotAssignedApps: item.categoryForNotAssignedApps,
|
categoryForNotAssignedApps: item.categoryForNotAssignedApps,
|
||||||
relaxPrimaryDeviceRule: item.relaxPrimaryDeviceRule,
|
relaxPrimaryDeviceRule: item.relaxPrimaryDeviceRule,
|
||||||
mailNotificationFlags: item.mailNotificationFlags
|
mailNotificationFlags: item.mailNotificationFlags,
|
||||||
|
blockedTimes: item.blockedTimes
|
||||||
}))
|
}))
|
||||||
|
|
||||||
result.users = {
|
result.users = {
|
||||||
|
@ -155,7 +157,8 @@ export const generateServerDataStatus = async ({ database, clientStatus, familyI
|
||||||
currentDevice: item.currentDevice,
|
currentDevice: item.currentDevice,
|
||||||
categoryForNotAssignedApps: item.categoryForNotAssignedApps,
|
categoryForNotAssignedApps: item.categoryForNotAssignedApps,
|
||||||
relaxPrimaryDevice: item.relaxPrimaryDeviceRule,
|
relaxPrimaryDevice: item.relaxPrimaryDeviceRule,
|
||||||
mailNotificationFlags: item.mailNotificationFlags
|
mailNotificationFlags: item.mailNotificationFlags,
|
||||||
|
blockedTimes: item.blockedTimes
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,6 +57,7 @@ export interface ServerUserEntry {
|
||||||
categoryForNotAssignedApps: string
|
categoryForNotAssignedApps: string
|
||||||
relaxPrimaryDevice: boolean
|
relaxPrimaryDevice: boolean
|
||||||
mailNotificationFlags: number
|
mailNotificationFlags: number
|
||||||
|
blockedTimes: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ServerDeviceData {
|
export interface ServerDeviceData {
|
||||||
|
|
|
@ -15,7 +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 { split } from 'lodash'
|
import { range, split } from 'lodash'
|
||||||
|
|
||||||
export const serializedBitmaskRegex = /^(\d*,\d*(,\d*,\d*)*)?$/
|
export const serializedBitmaskRegex = /^(\d*,\d*(,\d*,\d*)*)?$/
|
||||||
|
|
||||||
|
@ -52,3 +52,23 @@ export const validateBitmask = (bitmask: string, maxLength: number) => {
|
||||||
previousValue = item
|
previousValue = item
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const validateAndParseBitmask = (bitmask: string, maxLength: number) => {
|
||||||
|
validateBitmask(bitmask, maxLength)
|
||||||
|
|
||||||
|
const result = range(0, maxLength).map((_) => false)
|
||||||
|
|
||||||
|
const splitpoints = split(bitmask, ',').map((item) => parseInt(item, 10))
|
||||||
|
|
||||||
|
let i = 0
|
||||||
|
while (i < splitpoints.length) {
|
||||||
|
const start = splitpoints[i++]
|
||||||
|
const end = splitpoints[i++]
|
||||||
|
|
||||||
|
for (let j = start; j < end; j++) {
|
||||||
|
result[j] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue