2018-04-27 17:16:43 +00:00
|
|
|
from flask import current_app as app
|
2018-04-30 17:58:19 +00:00
|
|
|
from marshmallow import ValidationError, post_load, validates_schema
|
2018-04-27 17:16:43 +00:00
|
|
|
from marshmallow.fields import Boolean, DateTime, Integer, Nested, String, TimeDelta, UUID
|
|
|
|
from marshmallow.validate import Length, Range
|
|
|
|
from marshmallow_enum import EnumField
|
|
|
|
|
|
|
|
from ereuse_devicehub.marshmallow import NestedOn
|
|
|
|
from ereuse_devicehub.resources.device.schemas import Component, Device
|
|
|
|
from ereuse_devicehub.resources.event.enums import Appearance, Bios, Functionality, Orientation, \
|
|
|
|
SoftwareType, StepTypes, TestHardDriveLength
|
|
|
|
from ereuse_devicehub.resources.models import STR_BIG_SIZE, STR_SIZE
|
|
|
|
from ereuse_devicehub.resources.schemas import Thing
|
|
|
|
from ereuse_devicehub.resources.user.schemas import User
|
|
|
|
from teal.marshmallow import Color, Version
|
|
|
|
from teal.resource import Schema
|
|
|
|
|
|
|
|
|
|
|
|
class Event(Thing):
|
|
|
|
id = Integer(dump_only=True)
|
|
|
|
title = String(default='',
|
|
|
|
validate=Length(STR_BIG_SIZE),
|
|
|
|
description='A name or title for the event. Used when searching for events.')
|
|
|
|
date = DateTime('iso', description='When this event happened. '
|
|
|
|
'Leave it blank if it is happening now. '
|
|
|
|
'This is used when creating events retroactively.')
|
|
|
|
secured = Boolean(default=False,
|
|
|
|
description='Can we ensure the info in this event is totally correct?'
|
|
|
|
'Devicehub will automatically set this too for some events,'
|
|
|
|
'for example in snapshots if it could detect the ids of the'
|
|
|
|
'hardware without margin of doubt.')
|
|
|
|
incidence = Boolean(default=False,
|
|
|
|
description='Was something wrong in this event?')
|
2018-05-13 13:13:12 +00:00
|
|
|
snapshot = NestedOn('Snapshot', dump_only=True)
|
2018-04-27 17:16:43 +00:00
|
|
|
description = String(default='', description='A comment about the event.')
|
2018-05-11 16:58:48 +00:00
|
|
|
components = NestedOn(Component, dump_only=True, many=True)
|
2018-04-27 17:16:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class EventWithOneDevice(Event):
|
2018-05-11 16:58:48 +00:00
|
|
|
device = NestedOn(Device, only='id')
|
2018-04-27 17:16:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class EventWithMultipleDevices(Event):
|
2018-05-13 13:13:12 +00:00
|
|
|
devices = NestedOn(Device, many=True, only='id')
|
2018-04-27 17:16:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Add(EventWithOneDevice):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class Remove(EventWithOneDevice):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class Allocate(EventWithMultipleDevices):
|
2018-05-11 16:58:48 +00:00
|
|
|
to = NestedOn(User,
|
|
|
|
description='The user the devices are allocated to.')
|
2018-04-27 17:16:43 +00:00
|
|
|
organization = String(validate=Length(STR_SIZE),
|
|
|
|
description='The organization where the user was when this happened.')
|
|
|
|
|
|
|
|
|
|
|
|
class Deallocate(EventWithMultipleDevices):
|
2018-05-11 16:58:48 +00:00
|
|
|
from_rel = Nested(User,
|
2018-04-27 17:16:43 +00:00
|
|
|
data_key='from',
|
|
|
|
description='The user where the devices are not allocated to anymore.')
|
|
|
|
organization = String(validate=Length(STR_SIZE),
|
|
|
|
description='The organization where the user was when this happened.')
|
|
|
|
|
|
|
|
|
|
|
|
class EraseBasic(EventWithOneDevice):
|
|
|
|
starting_time = DateTime(required=True, data_key='startingTime')
|
|
|
|
ending_time = DateTime(required=True, data_key='endingTime')
|
|
|
|
secure_random_steps = Integer(validate=Range(min=0), required=True,
|
|
|
|
data_key='secureRandomSteps')
|
|
|
|
success = Boolean(required=True)
|
|
|
|
clean_with_zeros = Boolean(required=True, data_key='cleanWithZeros')
|
|
|
|
|
|
|
|
|
|
|
|
class EraseSectors(EraseBasic):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class Step(Schema):
|
|
|
|
id = Integer(dump_only=True)
|
|
|
|
type = EnumField(StepTypes, required=True)
|
|
|
|
starting_time = DateTime(required=True, data_key='startingTime')
|
|
|
|
ending_time = DateTime(required=True, data_key='endingTime')
|
|
|
|
secure_random_steps = Integer(validate=Range(min=0),
|
|
|
|
required=True,
|
|
|
|
data_key='secureRandomSteps')
|
|
|
|
success = Boolean(required=True)
|
|
|
|
clean_with_zeros = Boolean(required=True, data_key='cleanWithZeros')
|
|
|
|
|
|
|
|
|
|
|
|
class Condition(Schema):
|
|
|
|
appearance = EnumField(Appearance,
|
|
|
|
required=True,
|
|
|
|
description='Grades the imperfections that aesthetically '
|
|
|
|
'affect the device, but not its usage.')
|
|
|
|
appearance_score = Integer(validate=Range(-3, 5), dump_only=True)
|
|
|
|
functionality = EnumField(Functionality,
|
|
|
|
required=True,
|
|
|
|
description='Grades the defects of a device that affect its usage.')
|
|
|
|
functionality_score = Integer(validate=Range(-3, 5),
|
|
|
|
dump_only=True,
|
|
|
|
data_key='functionalityScore')
|
|
|
|
labelling = Boolean(description='Sets if there are labels stuck that should be removed.')
|
|
|
|
bios = EnumField(Bios, description='How difficult it has been to set the bios to '
|
|
|
|
'boot from the network.')
|
|
|
|
general = Integer(dump_only=True,
|
|
|
|
validate=Range(0, 5),
|
|
|
|
description='The grade of the device.')
|
|
|
|
|
|
|
|
|
|
|
|
class Installation(Schema):
|
|
|
|
name = String(validate=Length(STR_BIG_SIZE),
|
|
|
|
required=True,
|
|
|
|
description='The name of the OS installed.')
|
|
|
|
elapsed = TimeDelta(precision=TimeDelta.SECONDS, required=True)
|
|
|
|
success = Boolean(required=True)
|
|
|
|
|
|
|
|
|
|
|
|
class Inventory(Schema):
|
|
|
|
elapsed = TimeDelta(precision=TimeDelta.SECONDS, required=True)
|
|
|
|
|
|
|
|
|
|
|
|
class Snapshot(EventWithOneDevice):
|
2018-05-13 13:13:12 +00:00
|
|
|
"""
|
|
|
|
The Snapshot updates the state of the device with information about
|
|
|
|
its components and events performed at them.
|
|
|
|
|
|
|
|
See docs for more info.
|
|
|
|
"""
|
2018-04-27 17:16:43 +00:00
|
|
|
device = NestedOn(Device) # todo and when dumping?
|
2018-05-13 13:13:12 +00:00
|
|
|
components = NestedOn(Component,
|
|
|
|
many=True,
|
|
|
|
description='A list of components that are inside of the device'
|
|
|
|
'at the moment of this Snapshot.'
|
|
|
|
'Order is preserved, so the component num 0 when'
|
|
|
|
'submitting is the component num 0 when returning it back.')
|
2018-04-27 17:16:43 +00:00
|
|
|
uuid = UUID(required=True)
|
|
|
|
version = Version(required=True, description='The version of the SnapshotSoftware.')
|
|
|
|
software = EnumField(SoftwareType,
|
|
|
|
required=True,
|
|
|
|
description='The software that generated this Snapshot.')
|
|
|
|
condition = Nested(Condition, required=True)
|
|
|
|
install = Nested(Installation)
|
|
|
|
elapsed = TimeDelta(precision=TimeDelta.SECONDS, required=True)
|
|
|
|
inventory = Nested(Inventory)
|
|
|
|
color = Color(description='Main color of the device.')
|
|
|
|
orientation = EnumField(Orientation, description='Is the device main stand wider or larger?')
|
2018-04-30 17:58:19 +00:00
|
|
|
force_creation = Boolean(data_key='forceCreation')
|
2018-05-13 13:13:12 +00:00
|
|
|
events = NestedOn(Event, many=True, dump_only=True)
|
2018-04-27 17:16:43 +00:00
|
|
|
|
|
|
|
@validates_schema
|
|
|
|
def validate_workbench_version(self, data: dict):
|
|
|
|
if data['software'] == SoftwareType.Workbench:
|
|
|
|
if data['version'] < app.config['MIN_WORKBENCH']:
|
|
|
|
raise ValidationError(
|
|
|
|
'Min. supported Workbench version is {}'.format(app.config['MIN_WORKBENCH']),
|
|
|
|
field_names=['version']
|
|
|
|
)
|
|
|
|
|
|
|
|
@validates_schema
|
|
|
|
def validate_components_only_workbench(self, data: dict):
|
|
|
|
if data['software'] != SoftwareType.Workbench:
|
|
|
|
if data['components'] is not None:
|
|
|
|
raise ValidationError('Only Workbench can add component info',
|
|
|
|
field_names=['components'])
|
|
|
|
|
2018-04-30 17:58:19 +00:00
|
|
|
@post_load
|
|
|
|
def normalize_nested(self, data: dict):
|
|
|
|
data.update(data.pop('condition'))
|
|
|
|
data['condition'] = data.pop('general', None)
|
|
|
|
data.update({'install_' + key: value for key, value in data.pop('install', {})})
|
|
|
|
data['inventory_elapsed'] = data.get('inventory', {}).pop('elapsed', None)
|
|
|
|
return data
|
|
|
|
|
2018-04-27 17:16:43 +00:00
|
|
|
|
|
|
|
class Test(EventWithOneDevice):
|
|
|
|
elapsed = TimeDelta(precision=TimeDelta.SECONDS, required=True)
|
|
|
|
success = Boolean(required=True)
|
|
|
|
|
|
|
|
|
|
|
|
class TestHardDrive(Test):
|
|
|
|
length = EnumField(TestHardDriveLength, required=True)
|
|
|
|
status = String(validate=Length(max=STR_SIZE), required=True)
|
|
|
|
lifetime = TimeDelta(precision=TimeDelta.DAYS, required=True)
|
|
|
|
first_error = Integer()
|
|
|
|
|
|
|
|
|
|
|
|
class StressTest(Test):
|
|
|
|
pass
|