2022-02-24 13:15:58 +00:00
|
|
|
import csv
|
2022-03-04 12:32:37 +00:00
|
|
|
import logging
|
2022-05-13 08:22:32 +00:00
|
|
|
from distutils.util import strtobool
|
2022-02-22 10:11:53 +00:00
|
|
|
from io import StringIO
|
2022-02-24 13:15:58 +00:00
|
|
|
|
|
|
|
import flask
|
|
|
|
import flask_weasyprint
|
|
|
|
from flask import Blueprint, g, make_response, request, url_for
|
2021-12-28 11:36:02 +00:00
|
|
|
from flask.views import View
|
2022-02-03 09:50:36 +00:00
|
|
|
from flask_login import current_user, login_required
|
2022-03-08 09:05:32 +00:00
|
|
|
from werkzeug.exceptions import NotFound
|
2021-12-28 11:36:02 +00:00
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
from ereuse_devicehub import messages
|
2022-03-08 09:05:32 +00:00
|
|
|
from ereuse_devicehub.db import db
|
2022-02-24 13:15:58 +00:00
|
|
|
from ereuse_devicehub.inventory.forms import (
|
2022-06-02 11:46:10 +00:00
|
|
|
AdvancedSearchForm,
|
2022-02-25 11:27:14 +00:00
|
|
|
AllocateForm,
|
|
|
|
DataWipeForm,
|
2022-05-30 14:33:01 +00:00
|
|
|
EditTransferForm,
|
2022-03-07 12:07:26 +00:00
|
|
|
FilterForm,
|
2022-02-25 11:27:14 +00:00
|
|
|
LotForm,
|
|
|
|
NewActionForm,
|
|
|
|
NewDeviceForm,
|
2022-06-03 18:01:13 +00:00
|
|
|
NotesForm,
|
2022-02-25 11:27:14 +00:00
|
|
|
TagDeviceForm,
|
2022-02-25 11:34:02 +00:00
|
|
|
TradeDocumentForm,
|
|
|
|
TradeForm,
|
2022-05-27 14:32:22 +00:00
|
|
|
TransferForm,
|
2022-02-25 11:57:13 +00:00
|
|
|
UploadSnapshotForm,
|
2022-02-25 11:56:01 +00:00
|
|
|
)
|
2022-03-31 14:12:35 +00:00
|
|
|
from ereuse_devicehub.labels.forms import PrintLabelsForm
|
2022-05-18 09:01:58 +00:00
|
|
|
from ereuse_devicehub.parser.models import SnapshotsLog
|
|
|
|
from ereuse_devicehub.resources.action.models import Trade
|
2022-02-25 11:57:13 +00:00
|
|
|
from ereuse_devicehub.resources.device.models import Computer, DataStorage, Device
|
|
|
|
from ereuse_devicehub.resources.documents.device_row import ActionRow, DeviceRow
|
2022-02-24 13:15:58 +00:00
|
|
|
from ereuse_devicehub.resources.hash_reports import insert_hash
|
2021-12-29 12:44:28 +00:00
|
|
|
from ereuse_devicehub.resources.lot.models import Lot
|
2022-01-11 12:13:57 +00:00
|
|
|
from ereuse_devicehub.resources.tag.model import Tag
|
2022-05-16 15:52:31 +00:00
|
|
|
from ereuse_devicehub.views import GenericMixin
|
2021-12-28 11:36:02 +00:00
|
|
|
|
2022-03-15 13:33:27 +00:00
|
|
|
devices = Blueprint('inventory', __name__, url_prefix='/inventory')
|
2021-12-28 11:36:02 +00:00
|
|
|
|
2022-03-04 12:32:37 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2021-12-28 11:36:02 +00:00
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class DeviceListMixin(GenericMixin):
|
2022-03-08 16:43:15 +00:00
|
|
|
template_name = 'inventory/device_list.html'
|
|
|
|
|
2022-05-13 08:22:32 +00:00
|
|
|
def get_context(self, lot_id, only_unassigned=True):
|
2022-04-29 11:10:44 +00:00
|
|
|
super().get_context()
|
|
|
|
lots = self.context['lots']
|
2022-05-13 08:22:32 +00:00
|
|
|
form_filter = FilterForm(lots, lot_id, only_unassigned=only_unassigned)
|
2022-05-09 11:43:02 +00:00
|
|
|
devices = form_filter.search()
|
2021-12-30 20:40:44 +00:00
|
|
|
lot = None
|
2022-05-30 14:33:01 +00:00
|
|
|
form_transfer = ''
|
2022-06-06 11:37:58 +00:00
|
|
|
form_delivery = ''
|
|
|
|
form_receiver = ''
|
2022-01-25 11:53:36 +00:00
|
|
|
|
2022-01-25 09:30:46 +00:00
|
|
|
if lot_id:
|
|
|
|
lot = lots.filter(Lot.id == lot_id).one()
|
2022-05-30 14:33:01 +00:00
|
|
|
if not lot.is_temporary and lot.transfer:
|
2022-05-30 15:35:27 +00:00
|
|
|
form_transfer = EditTransferForm(lot_id=lot.id)
|
2022-06-03 18:01:13 +00:00
|
|
|
form_delivery = NotesForm(lot_id=lot.id, type='Delivery')
|
|
|
|
form_receiver = NotesForm(lot_id=lot.id, type='Receiver')
|
2022-02-04 12:22:47 +00:00
|
|
|
|
2022-05-12 12:23:01 +00:00
|
|
|
form_new_action = NewActionForm(lot=lot_id)
|
2022-04-29 11:10:44 +00:00
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'devices': devices,
|
|
|
|
'form_tag_device': TagDeviceForm(),
|
|
|
|
'form_new_action': form_new_action,
|
2022-05-12 12:23:01 +00:00
|
|
|
'form_new_allocate': AllocateForm(lot=lot_id),
|
|
|
|
'form_new_datawipe': DataWipeForm(lot=lot_id),
|
2022-05-30 14:33:01 +00:00
|
|
|
'form_transfer': form_transfer,
|
2022-06-03 18:01:13 +00:00
|
|
|
'form_delivery': form_delivery,
|
|
|
|
'form_receiver': form_receiver,
|
2022-04-29 11:10:44 +00:00
|
|
|
'form_filter': form_filter,
|
|
|
|
'form_print_labels': PrintLabelsForm(),
|
|
|
|
'lot': lot,
|
2022-04-13 07:44:30 +00:00
|
|
|
'tags': self.get_user_tags(),
|
|
|
|
'list_devices': self.get_selected_devices(form_new_action),
|
2022-05-13 08:22:32 +00:00
|
|
|
'unassigned_devices': only_unassigned,
|
2022-04-29 11:10:44 +00:00
|
|
|
}
|
|
|
|
)
|
2022-01-10 14:53:11 +00:00
|
|
|
|
2022-02-01 12:40:06 +00:00
|
|
|
return self.context
|
|
|
|
|
2022-04-13 07:44:30 +00:00
|
|
|
def get_user_tags(self):
|
|
|
|
return (
|
|
|
|
Tag.query.filter(Tag.owner_id == current_user.id)
|
|
|
|
.filter(Tag.device_id.is_(None))
|
|
|
|
.order_by(Tag.id.asc())
|
|
|
|
)
|
|
|
|
|
|
|
|
def get_selected_devices(self, action_form):
|
|
|
|
"""Retrieve selected devices (when action form is submited)"""
|
|
|
|
action_devices = action_form.devices.data
|
|
|
|
if action_devices:
|
|
|
|
return [int(x) for x in action_devices.split(",")]
|
|
|
|
return []
|
|
|
|
|
2022-02-01 12:40:06 +00:00
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class DeviceListView(DeviceListMixin):
|
2022-02-01 12:40:06 +00:00
|
|
|
def dispatch_request(self, lot_id=None):
|
2022-05-13 08:22:32 +00:00
|
|
|
only_unassigned = request.args.get(
|
|
|
|
'only_unassigned', default=True, type=strtobool
|
|
|
|
)
|
|
|
|
self.get_context(lot_id, only_unassigned)
|
2022-02-01 12:40:06 +00:00
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2021-12-30 16:23:34 +00:00
|
|
|
|
|
|
|
|
2022-06-03 09:34:49 +00:00
|
|
|
class AdvancedSearchView(DeviceListMixin):
|
2022-06-02 11:46:10 +00:00
|
|
|
methods = ['GET', 'POST']
|
|
|
|
template_name = 'inventory/search.html'
|
|
|
|
title = "Advanced Search"
|
|
|
|
|
|
|
|
def dispatch_request(self):
|
|
|
|
query = request.args.get('q', '')
|
|
|
|
self.get_context(None)
|
|
|
|
form = AdvancedSearchForm(q=query)
|
|
|
|
self.context.update({'devices': form.devices, 'advanced_form': form})
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class DeviceDetailView(GenericMixin):
|
2022-01-13 15:08:55 +00:00
|
|
|
decorators = [login_required]
|
2022-02-02 11:21:30 +00:00
|
|
|
template_name = 'inventory/device_detail.html'
|
2022-01-13 15:08:55 +00:00
|
|
|
|
|
|
|
def dispatch_request(self, id):
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
2022-02-24 13:15:58 +00:00
|
|
|
device = (
|
|
|
|
Device.query.filter(Device.owner_id == current_user.id)
|
|
|
|
.filter(Device.devicehub_id == id)
|
|
|
|
.one()
|
|
|
|
)
|
2022-01-13 15:08:55 +00:00
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'device': device,
|
|
|
|
'page_title': 'Device {}'.format(device.devicehub_id),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2022-01-13 15:08:55 +00:00
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class LotCreateView(GenericMixin):
|
2021-12-30 20:35:54 +00:00
|
|
|
methods = ['GET', 'POST']
|
|
|
|
decorators = [login_required]
|
|
|
|
template_name = 'inventory/lot.html'
|
|
|
|
title = "Add a new lot"
|
|
|
|
|
2022-01-12 16:09:20 +00:00
|
|
|
def dispatch_request(self):
|
|
|
|
form = LotForm()
|
|
|
|
if form.validate_on_submit():
|
|
|
|
form.save()
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=form.id)
|
2022-01-12 16:09:20 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'form': form,
|
|
|
|
'title': self.title,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2022-01-12 16:09:20 +00:00
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class LotUpdateView(GenericMixin):
|
2022-01-12 16:09:20 +00:00
|
|
|
methods = ['GET', 'POST']
|
|
|
|
decorators = [login_required]
|
|
|
|
template_name = 'inventory/lot.html'
|
|
|
|
title = "Edit a new lot"
|
|
|
|
|
|
|
|
def dispatch_request(self, id):
|
2021-12-30 20:35:54 +00:00
|
|
|
form = LotForm(id=id)
|
|
|
|
if form.validate_on_submit():
|
2022-01-05 10:10:32 +00:00
|
|
|
form.save()
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=id)
|
2021-12-30 12:53:28 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'form': form,
|
|
|
|
'title': self.title,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2021-12-30 20:35:54 +00:00
|
|
|
|
2021-12-30 12:53:28 +00:00
|
|
|
|
2022-01-03 11:28:42 +00:00
|
|
|
class LotDeleteView(View):
|
|
|
|
methods = ['GET']
|
|
|
|
decorators = [login_required]
|
|
|
|
template_name = 'inventory/device_list.html'
|
|
|
|
|
|
|
|
def dispatch_request(self, id):
|
|
|
|
form = LotForm(id=id)
|
2022-03-07 09:20:18 +00:00
|
|
|
if form.instance.trade:
|
|
|
|
msg = "Sorry, the lot cannot be deleted because have a trade action "
|
2022-03-04 17:30:14 +00:00
|
|
|
messages.error(msg)
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=id)
|
2022-03-04 17:30:14 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-01-03 11:28:42 +00:00
|
|
|
form.remove()
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.devicelist')
|
2022-01-03 11:28:42 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class UploadSnapshotView(GenericMixin):
|
2022-01-17 13:17:52 +00:00
|
|
|
methods = ['GET', 'POST']
|
|
|
|
decorators = [login_required]
|
|
|
|
template_name = 'inventory/upload_snapshot.html'
|
|
|
|
|
2022-03-08 16:43:15 +00:00
|
|
|
def dispatch_request(self, lot_id=None):
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
2022-01-17 13:17:52 +00:00
|
|
|
form = UploadSnapshotForm()
|
2022-04-29 11:10:44 +00:00
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'page_title': 'Upload Snapshot',
|
|
|
|
'form': form,
|
|
|
|
'lot_id': lot_id,
|
|
|
|
}
|
|
|
|
)
|
2022-01-17 13:17:52 +00:00
|
|
|
if form.validate_on_submit():
|
2022-04-12 08:25:45 +00:00
|
|
|
snapshot, devices = form.save(commit=False)
|
2022-03-08 16:43:15 +00:00
|
|
|
if lot_id:
|
2022-04-29 11:10:44 +00:00
|
|
|
lots = self.context['lots']
|
2022-03-08 16:43:15 +00:00
|
|
|
lot = lots.filter(Lot.id == lot_id).one()
|
2022-04-12 08:25:45 +00:00
|
|
|
for dev in devices:
|
|
|
|
lot.devices.add(dev)
|
2022-03-08 16:43:15 +00:00
|
|
|
db.session.add(lot)
|
|
|
|
db.session.commit()
|
2022-01-17 13:17:52 +00:00
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2022-01-17 13:17:52 +00:00
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class DeviceCreateView(GenericMixin):
|
2022-01-19 12:40:40 +00:00
|
|
|
methods = ['GET', 'POST']
|
|
|
|
decorators = [login_required]
|
2022-02-02 11:21:30 +00:00
|
|
|
template_name = 'inventory/device_create.html'
|
2022-01-19 12:40:40 +00:00
|
|
|
|
2022-03-08 16:43:15 +00:00
|
|
|
def dispatch_request(self, lot_id=None):
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
2022-01-19 12:40:40 +00:00
|
|
|
form = NewDeviceForm()
|
2022-04-29 11:10:44 +00:00
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'page_title': 'New Device',
|
|
|
|
'form': form,
|
|
|
|
'lot_id': lot_id,
|
|
|
|
}
|
|
|
|
)
|
2022-01-19 12:40:40 +00:00
|
|
|
if form.validate_on_submit():
|
2022-03-08 16:43:15 +00:00
|
|
|
snapshot = form.save(commit=False)
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.devicelist')
|
2022-03-08 16:43:15 +00:00
|
|
|
if lot_id:
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=lot_id)
|
2022-04-29 11:10:44 +00:00
|
|
|
lots = self.context['lots']
|
2022-03-08 16:43:15 +00:00
|
|
|
lot = lots.filter(Lot.id == lot_id).one()
|
|
|
|
lot.devices.add(snapshot.device)
|
|
|
|
db.session.add(lot)
|
|
|
|
|
|
|
|
db.session.commit()
|
2022-03-08 16:49:56 +00:00
|
|
|
messages.success('Device "{}" created successfully!'.format(form.type.data))
|
2022-01-19 12:40:40 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2022-01-19 12:40:40 +00:00
|
|
|
|
|
|
|
|
2022-02-02 11:21:30 +00:00
|
|
|
class TagLinkDeviceView(View):
|
2022-01-25 11:53:36 +00:00
|
|
|
methods = ['POST']
|
|
|
|
decorators = [login_required]
|
2022-02-02 11:21:30 +00:00
|
|
|
# template_name = 'inventory/device_list.html'
|
2022-01-25 11:53:36 +00:00
|
|
|
|
|
|
|
def dispatch_request(self):
|
|
|
|
form = TagDeviceForm()
|
|
|
|
if form.validate_on_submit():
|
|
|
|
form.save()
|
|
|
|
|
|
|
|
return flask.redirect(request.referrer)
|
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class TagUnlinkDeviceView(GenericMixin):
|
2022-01-25 13:39:15 +00:00
|
|
|
methods = ['POST', 'GET']
|
2022-01-25 11:53:36 +00:00
|
|
|
decorators = [login_required]
|
2022-02-02 11:21:30 +00:00
|
|
|
template_name = 'inventory/tag_unlink_device.html'
|
2022-01-25 11:53:36 +00:00
|
|
|
|
2022-01-25 13:39:15 +00:00
|
|
|
def dispatch_request(self, id):
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
2022-01-25 13:39:15 +00:00
|
|
|
form = TagDeviceForm(delete=True, device=id)
|
2022-01-25 11:53:36 +00:00
|
|
|
if form.validate_on_submit():
|
|
|
|
form.remove()
|
|
|
|
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.devicelist')
|
2022-01-26 11:29:03 +00:00
|
|
|
return flask.redirect(next_url)
|
2022-01-25 11:53:36 +00:00
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
self.context.update(
|
|
|
|
{
|
|
|
|
'form': form,
|
|
|
|
'referrer': request.referrer,
|
|
|
|
}
|
2022-02-24 13:15:58 +00:00
|
|
|
)
|
2022-01-19 12:40:40 +00:00
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
return flask.render_template(self.template_name, **self.context)
|
|
|
|
|
2022-01-19 12:40:40 +00:00
|
|
|
|
2022-01-05 15:00:13 +00:00
|
|
|
class NewActionView(View):
|
|
|
|
methods = ['POST']
|
|
|
|
decorators = [login_required]
|
2022-02-08 10:52:30 +00:00
|
|
|
form_class = NewActionForm
|
2022-01-05 15:00:13 +00:00
|
|
|
|
|
|
|
def dispatch_request(self):
|
2022-02-08 10:52:30 +00:00
|
|
|
self.form = self.form_class()
|
2022-04-21 12:02:16 +00:00
|
|
|
next_url = self.get_next_url()
|
2022-02-04 11:10:27 +00:00
|
|
|
|
2022-02-01 12:40:06 +00:00
|
|
|
if self.form.validate_on_submit():
|
2022-02-24 13:15:58 +00:00
|
|
|
self.form.save()
|
|
|
|
messages.success(
|
|
|
|
'Action "{}" created successfully!'.format(self.form.type.data)
|
|
|
|
)
|
2022-02-08 10:49:13 +00:00
|
|
|
next_url = self.get_next_url()
|
2022-01-05 15:00:13 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-21 12:02:16 +00:00
|
|
|
messages.error('Action {} error!'.format(self.form.type.data))
|
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-02-08 10:49:13 +00:00
|
|
|
def get_next_url(self):
|
2022-02-04 11:10:27 +00:00
|
|
|
lot_id = self.form.lot.data
|
2022-02-08 10:49:13 +00:00
|
|
|
|
2022-02-04 11:10:27 +00:00
|
|
|
if lot_id:
|
2022-03-15 13:33:27 +00:00
|
|
|
return url_for('inventory.lotdevicelist', lot_id=lot_id)
|
2022-02-04 11:10:27 +00:00
|
|
|
|
2022-03-15 13:33:27 +00:00
|
|
|
return url_for('inventory.devicelist')
|
2022-01-05 15:00:13 +00:00
|
|
|
|
2022-01-03 11:28:42 +00:00
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class NewAllocateView(DeviceListMixin, NewActionView):
|
2022-02-01 12:40:06 +00:00
|
|
|
methods = ['POST']
|
2022-02-08 10:52:30 +00:00
|
|
|
form_class = AllocateForm
|
2022-02-01 12:40:06 +00:00
|
|
|
|
2022-02-07 13:01:56 +00:00
|
|
|
def dispatch_request(self):
|
2022-02-08 10:52:30 +00:00
|
|
|
self.form = self.form_class()
|
2022-02-04 11:10:27 +00:00
|
|
|
|
2022-02-03 09:14:10 +00:00
|
|
|
if self.form.validate_on_submit():
|
2022-02-24 13:15:58 +00:00
|
|
|
self.form.save()
|
|
|
|
messages.success(
|
|
|
|
'Action "{}" created successfully!'.format(self.form.type.data)
|
|
|
|
)
|
2022-02-03 09:14:10 +00:00
|
|
|
|
2022-02-08 10:49:13 +00:00
|
|
|
next_url = self.get_next_url()
|
2022-02-03 09:14:10 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-21 12:02:16 +00:00
|
|
|
messages.error('Action {} error!'.format(self.form.type.data))
|
2022-04-29 15:53:59 +00:00
|
|
|
for k, v in self.form.errors.items():
|
|
|
|
value = ';'.join(v)
|
2022-05-12 08:33:04 +00:00
|
|
|
key = self.form[k].label.text
|
|
|
|
messages.error('Action Error {key}: {value}!'.format(key=key, value=value))
|
2022-04-21 12:02:16 +00:00
|
|
|
next_url = self.get_next_url()
|
|
|
|
return flask.redirect(next_url)
|
2022-02-01 12:40:06 +00:00
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class NewDataWipeView(DeviceListMixin, NewActionView):
|
2022-02-07 13:01:56 +00:00
|
|
|
methods = ['POST']
|
2022-02-09 10:27:08 +00:00
|
|
|
form_class = DataWipeForm
|
2022-02-07 13:01:56 +00:00
|
|
|
|
|
|
|
def dispatch_request(self):
|
2022-02-09 10:27:08 +00:00
|
|
|
self.form = self.form_class()
|
|
|
|
|
|
|
|
if self.form.validate_on_submit():
|
2022-02-24 13:15:58 +00:00
|
|
|
self.form.save()
|
|
|
|
messages.success(
|
|
|
|
'Action "{}" created successfully!'.format(self.form.type.data)
|
|
|
|
)
|
2022-02-09 10:27:08 +00:00
|
|
|
|
|
|
|
next_url = self.get_next_url()
|
|
|
|
return flask.redirect(next_url)
|
2022-02-07 13:01:56 +00:00
|
|
|
|
2022-04-21 12:02:16 +00:00
|
|
|
messages.error('Action {} error!'.format(self.form.type.data))
|
|
|
|
next_url = self.get_next_url()
|
|
|
|
return flask.redirect(next_url)
|
2022-02-07 13:01:56 +00:00
|
|
|
|
|
|
|
|
2022-05-16 15:52:31 +00:00
|
|
|
class NewTradeView(DeviceListMixin, NewActionView):
|
2022-02-09 13:06:28 +00:00
|
|
|
methods = ['POST']
|
|
|
|
form_class = TradeForm
|
|
|
|
|
|
|
|
def dispatch_request(self):
|
|
|
|
self.form = self.form_class()
|
|
|
|
|
|
|
|
if self.form.validate_on_submit():
|
2022-02-24 13:15:58 +00:00
|
|
|
self.form.save()
|
|
|
|
messages.success(
|
|
|
|
'Action "{}" created successfully!'.format(self.form.type.data)
|
|
|
|
)
|
2022-02-09 13:06:28 +00:00
|
|
|
|
|
|
|
next_url = self.get_next_url()
|
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-21 12:02:16 +00:00
|
|
|
messages.error('Action {} error!'.format(self.form.type.data))
|
|
|
|
next_url = self.get_next_url()
|
|
|
|
return flask.redirect(next_url)
|
2022-02-09 13:06:28 +00:00
|
|
|
|
|
|
|
|
2022-02-21 12:03:03 +00:00
|
|
|
class NewTradeDocumentView(View):
|
|
|
|
methods = ['POST', 'GET']
|
|
|
|
decorators = [login_required]
|
|
|
|
template_name = 'inventory/trade_document.html'
|
|
|
|
form_class = TradeDocumentForm
|
|
|
|
title = "Add new document"
|
|
|
|
|
|
|
|
def dispatch_request(self, lot_id):
|
|
|
|
self.form = self.form_class(lot=lot_id)
|
2022-04-29 11:10:44 +00:00
|
|
|
self.get_context()
|
2022-02-21 12:03:03 +00:00
|
|
|
|
|
|
|
if self.form.validate_on_submit():
|
|
|
|
self.form.save()
|
|
|
|
messages.success('Document created successfully!')
|
2022-03-15 13:33:27 +00:00
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=lot_id)
|
2022-02-21 12:03:03 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-04-29 11:10:44 +00:00
|
|
|
self.context.update({'form': self.form, 'title': self.title})
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2022-02-21 12:03:03 +00:00
|
|
|
|
|
|
|
|
2022-06-01 13:01:02 +00:00
|
|
|
class NewTransferView(GenericMixin):
|
2022-05-27 14:32:22 +00:00
|
|
|
methods = ['POST', 'GET']
|
|
|
|
template_name = 'inventory/new_transfer.html'
|
|
|
|
form_class = TransferForm
|
|
|
|
title = "Add new transfer"
|
|
|
|
|
|
|
|
def dispatch_request(self, lot_id, type_id):
|
|
|
|
self.form = self.form_class(lot_id=lot_id, type=type_id)
|
|
|
|
self.get_context()
|
|
|
|
|
|
|
|
if self.form.validate_on_submit():
|
|
|
|
self.form.save()
|
|
|
|
new_lot_id = lot_id
|
|
|
|
if self.form.newlot.id:
|
2022-05-30 15:35:27 +00:00
|
|
|
new_lot_id = "{}".format(self.form.newlot.id)
|
2022-05-30 14:33:01 +00:00
|
|
|
Lot.query.filter(Lot.id == new_lot_id).one()
|
2022-05-27 14:32:22 +00:00
|
|
|
messages.success('Transfer created successfully!')
|
2022-06-01 11:15:04 +00:00
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=str(new_lot_id))
|
2022-05-27 14:32:22 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
|
|
|
self.context.update({'form': self.form, 'title': self.title})
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
|
|
|
|
|
|
|
|
2022-06-01 13:01:02 +00:00
|
|
|
class EditTransferView(GenericMixin):
|
2022-05-30 14:33:01 +00:00
|
|
|
methods = ['POST']
|
|
|
|
form_class = EditTransferForm
|
|
|
|
|
|
|
|
def dispatch_request(self, lot_id):
|
|
|
|
self.get_context()
|
|
|
|
form = self.form_class(request.form, lot_id=lot_id)
|
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=lot_id)
|
|
|
|
|
|
|
|
if form.validate_on_submit():
|
|
|
|
form.save()
|
|
|
|
messages.success('Transfer updated successfully!')
|
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
|
|
|
messages.error('Transfer updated error!')
|
2022-06-06 11:37:58 +00:00
|
|
|
for k, v in form.errors.items():
|
|
|
|
value = ';'.join(v)
|
|
|
|
key = form[k].label.text
|
|
|
|
messages.error('Error {key}: {value}!'.format(key=key, value=value))
|
2022-05-30 14:33:01 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
|
|
|
|
2022-02-22 10:11:53 +00:00
|
|
|
class ExportsView(View):
|
|
|
|
methods = ['GET']
|
|
|
|
decorators = [login_required]
|
|
|
|
|
2022-02-24 13:15:58 +00:00
|
|
|
def dispatch_request(self, export_id):
|
2022-02-22 10:11:53 +00:00
|
|
|
export_ids = {
|
2022-02-24 13:15:58 +00:00
|
|
|
'metrics': self.metrics,
|
|
|
|
'devices': self.devices_list,
|
|
|
|
'certificates': self.erasure,
|
2022-02-22 10:11:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if export_id not in export_ids:
|
|
|
|
return NotFound()
|
|
|
|
return export_ids[export_id]()
|
|
|
|
|
2022-02-24 13:15:58 +00:00
|
|
|
def find_devices(self):
|
|
|
|
args = request.args.get('ids')
|
|
|
|
ids = args.split(',') if args else []
|
|
|
|
query = Device.query.filter(Device.owner == g.user)
|
|
|
|
return query.filter(Device.devicehub_id.in_(ids))
|
|
|
|
|
|
|
|
def response_csv(self, data, name):
|
2022-02-22 10:11:53 +00:00
|
|
|
bfile = data.getvalue().encode('utf-8')
|
2022-02-24 13:15:58 +00:00
|
|
|
# insert proof
|
|
|
|
insert_hash(bfile)
|
2022-02-22 10:11:53 +00:00
|
|
|
output = make_response(bfile)
|
2022-02-24 13:15:58 +00:00
|
|
|
output.headers['Content-Disposition'] = 'attachment; filename={}'.format(name)
|
2022-02-22 10:11:53 +00:00
|
|
|
output.headers['Content-type'] = 'text/csv'
|
|
|
|
return output
|
|
|
|
|
2022-02-24 13:15:58 +00:00
|
|
|
def devices_list(self):
|
|
|
|
"""Get device query and put information in csv format."""
|
|
|
|
data = StringIO()
|
|
|
|
cw = csv.writer(data, delimiter=';', lineterminator="\n", quotechar='"')
|
|
|
|
first = True
|
|
|
|
|
|
|
|
for device in self.find_devices():
|
|
|
|
d = DeviceRow(device, {})
|
|
|
|
if first:
|
|
|
|
cw.writerow(d.keys())
|
|
|
|
first = False
|
|
|
|
cw.writerow(d.values())
|
|
|
|
|
|
|
|
return self.response_csv(data, "export.csv")
|
|
|
|
|
|
|
|
def metrics(self):
|
|
|
|
"""Get device query and put information in csv format."""
|
|
|
|
data = StringIO()
|
|
|
|
cw = csv.writer(data, delimiter=';', lineterminator="\n", quotechar='"')
|
|
|
|
first = True
|
|
|
|
devs_id = []
|
|
|
|
# Get the allocate info
|
|
|
|
for device in self.find_devices():
|
|
|
|
devs_id.append(device.id)
|
|
|
|
for allocate in device.get_metrics():
|
|
|
|
d = ActionRow(allocate)
|
|
|
|
if first:
|
|
|
|
cw.writerow(d.keys())
|
|
|
|
first = False
|
|
|
|
cw.writerow(d.values())
|
|
|
|
|
|
|
|
# Get the trade info
|
|
|
|
query_trade = Trade.query.filter(
|
|
|
|
Trade.devices.any(Device.id.in_(devs_id))
|
|
|
|
).all()
|
|
|
|
|
|
|
|
lot_id = request.args.get('lot')
|
|
|
|
if lot_id and not query_trade:
|
|
|
|
lot = Lot.query.filter_by(id=lot_id).one()
|
|
|
|
if hasattr(lot, "trade") and lot.trade:
|
|
|
|
if g.user in [lot.trade.user_from, lot.trade.user_to]:
|
|
|
|
query_trade = [lot.trade]
|
|
|
|
|
|
|
|
for trade in query_trade:
|
|
|
|
data_rows = trade.get_metrics()
|
|
|
|
for row in data_rows:
|
|
|
|
d = ActionRow(row)
|
|
|
|
if first:
|
|
|
|
cw.writerow(d.keys())
|
|
|
|
first = False
|
|
|
|
cw.writerow(d.values())
|
|
|
|
|
|
|
|
return self.response_csv(data, "actions_export.csv")
|
|
|
|
|
|
|
|
def erasure(self):
|
|
|
|
template = self.build_erasure_certificate()
|
|
|
|
res = flask_weasyprint.render_pdf(
|
|
|
|
flask_weasyprint.HTML(string=template),
|
|
|
|
download_filename='erasure-certificate.pdf',
|
|
|
|
)
|
|
|
|
insert_hash(res.data)
|
|
|
|
return res
|
|
|
|
|
|
|
|
def build_erasure_certificate(self):
|
|
|
|
erasures = []
|
|
|
|
for device in self.find_devices():
|
|
|
|
if isinstance(device, Computer):
|
|
|
|
for privacy in device.privacy:
|
|
|
|
erasures.append(privacy)
|
|
|
|
elif isinstance(device, DataStorage):
|
|
|
|
if device.privacy:
|
|
|
|
erasures.append(device.privacy)
|
|
|
|
|
|
|
|
params = {
|
|
|
|
'title': 'Erasure Certificate',
|
|
|
|
'erasures': tuple(erasures),
|
|
|
|
'url_pdf': '',
|
|
|
|
}
|
|
|
|
return flask.render_template('inventory/erasure.html', **params)
|
|
|
|
|
2022-02-22 10:11:53 +00:00
|
|
|
|
2022-05-18 09:01:58 +00:00
|
|
|
class SnapshotListView(GenericMixin):
|
2022-05-11 14:59:36 +00:00
|
|
|
template_name = 'inventory/snapshots_list.html'
|
|
|
|
|
2022-05-11 15:03:31 +00:00
|
|
|
def dispatch_request(self):
|
2022-05-11 14:59:36 +00:00
|
|
|
self.get_context()
|
2022-05-18 09:01:58 +00:00
|
|
|
self.context['page_title'] = "Snapshots Logs"
|
2022-05-19 11:48:52 +00:00
|
|
|
self.context['snapshots_log'] = self.get_snapshots_log()
|
2022-05-18 09:01:58 +00:00
|
|
|
|
2022-05-11 14:59:36 +00:00
|
|
|
return flask.render_template(self.template_name, **self.context)
|
2022-04-13 16:31:39 +00:00
|
|
|
|
2022-05-19 11:48:52 +00:00
|
|
|
def get_snapshots_log(self):
|
|
|
|
snapshots_log = SnapshotsLog.query.filter(
|
|
|
|
SnapshotsLog.owner == g.user
|
|
|
|
).order_by(SnapshotsLog.created.desc())
|
|
|
|
logs = {}
|
|
|
|
for snap in snapshots_log:
|
2022-06-15 11:39:06 +00:00
|
|
|
try:
|
|
|
|
system_uuid = snap.snapshot.device.system_uuid or ''
|
|
|
|
except AttributeError:
|
|
|
|
system_uuid = ''
|
|
|
|
|
2022-05-19 11:48:52 +00:00
|
|
|
if snap.snapshot_uuid not in logs:
|
|
|
|
logs[snap.snapshot_uuid] = {
|
|
|
|
'sid': snap.sid,
|
|
|
|
'snapshot_uuid': snap.snapshot_uuid,
|
|
|
|
'version': snap.version,
|
2022-05-19 16:16:47 +00:00
|
|
|
'device': snap.get_device(),
|
2022-06-15 11:39:06 +00:00
|
|
|
'system_uuid': system_uuid,
|
2022-05-19 11:48:52 +00:00
|
|
|
'status': snap.get_status(),
|
|
|
|
'severity': snap.severity,
|
|
|
|
'created': snap.created,
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
|
|
|
|
if snap.created > logs[snap.snapshot_uuid]['created']:
|
|
|
|
logs[snap.snapshot_uuid]['created'] = snap.created
|
|
|
|
|
|
|
|
if snap.severity > logs[snap.snapshot_uuid]['severity']:
|
|
|
|
logs[snap.snapshot_uuid]['severity'] = snap.severity
|
|
|
|
logs[snap.snapshot_uuid]['status'] = snap.get_status()
|
|
|
|
|
|
|
|
result = sorted(logs.values(), key=lambda d: d['created'])
|
|
|
|
result.reverse()
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
2022-04-13 16:31:39 +00:00
|
|
|
|
2022-05-19 16:16:47 +00:00
|
|
|
class SnapshotDetailView(GenericMixin):
|
|
|
|
template_name = 'inventory/snapshot_detail.html'
|
|
|
|
|
|
|
|
def dispatch_request(self, snapshot_uuid):
|
|
|
|
self.snapshot_uuid = snapshot_uuid
|
|
|
|
self.get_context()
|
|
|
|
self.context['page_title'] = "Snapshot Detail"
|
|
|
|
self.context['snapshots_log'] = self.get_snapshots_log()
|
|
|
|
self.context['snapshot_uuid'] = snapshot_uuid
|
|
|
|
self.context['snapshot_sid'] = ''
|
|
|
|
if self.context['snapshots_log'].count():
|
|
|
|
self.context['snapshot_sid'] = self.context['snapshots_log'][0].sid
|
|
|
|
|
|
|
|
return flask.render_template(self.template_name, **self.context)
|
|
|
|
|
|
|
|
def get_snapshots_log(self):
|
|
|
|
return (
|
|
|
|
SnapshotsLog.query.filter(SnapshotsLog.owner == g.user)
|
|
|
|
.filter(SnapshotsLog.snapshot_uuid == self.snapshot_uuid)
|
|
|
|
.order_by(SnapshotsLog.created.desc())
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-06-03 18:01:13 +00:00
|
|
|
class DeliveryNoteView(GenericMixin):
|
|
|
|
methods = ['POST']
|
|
|
|
form_class = NotesForm
|
|
|
|
|
|
|
|
def dispatch_request(self, lot_id):
|
|
|
|
self.get_context()
|
|
|
|
form = self.form_class(request.form, lot_id=lot_id, type='Delivery')
|
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=lot_id)
|
|
|
|
|
|
|
|
if form.validate_on_submit():
|
|
|
|
form.save()
|
2022-06-06 11:37:58 +00:00
|
|
|
messages.success('Delivery Note updated successfully!')
|
2022-06-03 18:01:13 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-06-06 11:37:58 +00:00
|
|
|
messages.error('Delivery Note updated error!')
|
|
|
|
for k, v in form.errors.items():
|
|
|
|
value = ';'.join(v)
|
|
|
|
key = form[k].label.text
|
|
|
|
messages.error('Error {key}: {value}!'.format(key=key, value=value))
|
2022-06-03 18:01:13 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
|
|
|
|
|
|
|
class ReceiverNoteView(GenericMixin):
|
|
|
|
methods = ['POST']
|
|
|
|
form_class = NotesForm
|
|
|
|
|
|
|
|
def dispatch_request(self, lot_id):
|
|
|
|
self.get_context()
|
|
|
|
form = self.form_class(request.form, lot_id=lot_id, type='Receiver')
|
|
|
|
next_url = url_for('inventory.lotdevicelist', lot_id=lot_id)
|
|
|
|
|
|
|
|
if form.validate_on_submit():
|
|
|
|
form.save()
|
2022-06-06 11:37:58 +00:00
|
|
|
messages.success('Receiver Note updated successfully!')
|
2022-06-03 18:01:13 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
2022-06-06 11:37:58 +00:00
|
|
|
messages.error('Receiver Note updated error!')
|
|
|
|
for k, v in form.errors.items():
|
|
|
|
value = ';'.join(v)
|
|
|
|
key = form[k].label.text
|
|
|
|
messages.error('Error {key}: {value}!'.format(key=key, value=value))
|
2022-06-03 18:01:13 +00:00
|
|
|
return flask.redirect(next_url)
|
|
|
|
|
|
|
|
|
2022-01-05 15:00:13 +00:00
|
|
|
devices.add_url_rule('/action/add/', view_func=NewActionView.as_view('action_add'))
|
2022-02-09 13:06:28 +00:00
|
|
|
devices.add_url_rule('/action/trade/add/', view_func=NewTradeView.as_view('trade_add'))
|
2022-02-24 13:15:58 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/action/allocate/add/', view_func=NewAllocateView.as_view('allocate_add')
|
|
|
|
)
|
|
|
|
devices.add_url_rule(
|
|
|
|
'/action/datawipe/add/', view_func=NewDataWipeView.as_view('datawipe_add')
|
|
|
|
)
|
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:lot_id>/trade-document/add/',
|
|
|
|
view_func=NewTradeDocumentView.as_view('trade_document_add'),
|
|
|
|
)
|
2021-12-29 09:13:34 +00:00
|
|
|
devices.add_url_rule('/device/', view_func=DeviceListView.as_view('devicelist'))
|
2022-06-02 11:46:10 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/search/', view_func=AdvancedSearchView.as_view('advanced_search')
|
|
|
|
)
|
2022-02-24 13:15:58 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/device/<string:id>/', view_func=DeviceDetailView.as_view('device_details')
|
|
|
|
)
|
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:lot_id>/device/', view_func=DeviceListView.as_view('lotdevicelist')
|
|
|
|
)
|
2022-01-12 16:09:20 +00:00
|
|
|
devices.add_url_rule('/lot/add/', view_func=LotCreateView.as_view('lot_add'))
|
2022-02-24 13:15:58 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:id>/del/', view_func=LotDeleteView.as_view('lot_del')
|
|
|
|
)
|
2022-01-12 16:09:20 +00:00
|
|
|
devices.add_url_rule('/lot/<string:id>/', view_func=LotUpdateView.as_view('lot_edit'))
|
2022-02-24 13:15:58 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/upload-snapshot/', view_func=UploadSnapshotView.as_view('upload_snapshot')
|
|
|
|
)
|
2022-03-08 16:43:15 +00:00
|
|
|
devices.add_url_rule(
|
2022-03-08 16:49:56 +00:00
|
|
|
'/lot/<string:lot_id>/upload-snapshot/',
|
|
|
|
view_func=UploadSnapshotView.as_view('lot_upload_snapshot'),
|
2022-03-08 16:43:15 +00:00
|
|
|
)
|
2022-02-02 11:21:30 +00:00
|
|
|
devices.add_url_rule('/device/add/', view_func=DeviceCreateView.as_view('device_add'))
|
2022-03-08 16:43:15 +00:00
|
|
|
devices.add_url_rule(
|
2022-03-08 16:49:56 +00:00
|
|
|
'/lot/<string:lot_id>/device/add/',
|
|
|
|
view_func=DeviceCreateView.as_view('lot_device_add'),
|
2022-03-08 16:43:15 +00:00
|
|
|
)
|
2022-02-24 13:15:58 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/tag/devices/add/', view_func=TagLinkDeviceView.as_view('tag_devices_add')
|
|
|
|
)
|
|
|
|
devices.add_url_rule(
|
|
|
|
'/tag/devices/<int:id>/del/',
|
|
|
|
view_func=TagUnlinkDeviceView.as_view('tag_devices_del'),
|
|
|
|
)
|
|
|
|
devices.add_url_rule(
|
|
|
|
'/export/<string:export_id>/', view_func=ExportsView.as_view('export')
|
|
|
|
)
|
2022-05-11 14:59:36 +00:00
|
|
|
devices.add_url_rule('/snapshots/', view_func=SnapshotListView.as_view('snapshotslist'))
|
2022-05-19 16:16:47 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/snapshots/<string:snapshot_uuid>/',
|
|
|
|
view_func=SnapshotDetailView.as_view('snapshot_detail'),
|
|
|
|
)
|
2022-05-27 14:32:22 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:lot_id>/transfer/<string:type_id>/',
|
|
|
|
view_func=NewTransferView.as_view('new_transfer'),
|
|
|
|
)
|
2022-05-30 14:33:01 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:lot_id>/transfer/',
|
|
|
|
view_func=EditTransferView.as_view('edit_transfer'),
|
|
|
|
)
|
2022-06-03 18:01:13 +00:00
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:lot_id>/deliverynote/',
|
2022-06-06 11:37:58 +00:00
|
|
|
view_func=DeliveryNoteView.as_view('delivery_note'),
|
2022-06-03 18:01:13 +00:00
|
|
|
)
|
|
|
|
devices.add_url_rule(
|
|
|
|
'/lot/<string:lot_id>/receivernote/',
|
2022-06-06 11:37:58 +00:00
|
|
|
view_func=ReceiverNoteView.as_view('receiver_note'),
|
2022-06-03 18:01:13 +00:00
|
|
|
)
|