devicehub-django/did/views.py

264 lines
8.5 KiB
Python
Raw Permalink Normal View History

2024-11-20 17:06:52 +00:00
import json
import logging
2024-11-19 20:18:29 +00:00
from django.http import JsonResponse, Http404
from django.views.generic.base import TemplateView
from device.models import Device
from evidence.parse import Build
2024-11-20 17:20:11 +00:00
from dpp.api_dlt import ALGORITHM
2024-11-20 17:06:52 +00:00
from dpp.models import Proof
2024-11-27 17:11:22 +00:00
from dpp.api_dlt import PROOF_TYPE
2024-12-04 17:25:16 +00:00
from did.template_credential import dpp_tmpl
2024-11-20 17:06:52 +00:00
logger = logging.getLogger('django')
2024-11-19 20:18:29 +00:00
class PublicDeviceWebView(TemplateView):
2024-11-20 17:06:52 +00:00
template_name = "device_did.html"
2024-11-19 20:18:29 +00:00
def get(self, request, *args, **kwargs):
self.pk = kwargs['pk']
2024-11-20 17:06:52 +00:00
chid = self.pk.split(":")[0]
proof = Proof.objects.filter(signature=self.pk).first()
if proof:
self.object = Device(id=chid, uuid=proof.uuid)
else:
self.object = Device(id=chid)
2024-11-19 20:18:29 +00:00
if not self.object.last_evidence:
raise Http404
if self.request.headers.get('Accept') == 'application/json':
return self.get_json_response()
return super().get(request, *args, **kwargs)
def get_context_data(self, **kwargs):
2024-11-20 17:06:52 +00:00
self.context = super().get_context_data(**kwargs)
2024-11-19 20:18:29 +00:00
self.object.initial()
2024-11-20 17:06:52 +00:00
roles = [("Operator", "Operator")]
role = "Operator"
if self.request.user.is_anonymous:
roles = []
role = None
self.context.update({
'object': self.object,
'role': role,
'roles': roles,
'path': self.request.path,
'last_dpp': "",
'before_dpp': "",
2024-11-19 20:18:29 +00:00
})
2024-11-20 17:06:52 +00:00
if not self.request.user.is_anonymous:
self.get_manuals()
return self.context
2024-11-19 20:18:29 +00:00
@property
def public_fields(self):
return {
'id': self.object.id,
'shortid': self.object.shortid,
'uuids': self.object.uuids,
'hids': self.object.hids,
'components': self.remove_serial_number_from(self.object.components),
}
@property
def authenticated_fields(self):
return {
'serial_number': self.object.serial_number,
'components': self.object.components,
}
def remove_serial_number_from(self, components):
for component in components:
if 'serial_number' in component:
del component['SerialNumber']
return components
def get_device_data(self):
data = self.public_fields
if self.request.user.is_authenticated:
data.update(self.authenticated_fields)
return data
def get_json_response(self):
device_data = self.get_result()
# device_data = self.get_device_data()
2024-11-28 15:37:38 +00:00
response = JsonResponse(device_data)
response["Access-Control-Allow-Origin"] = "*"
return response
2024-11-19 20:18:29 +00:00
2024-11-20 17:20:11 +00:00
def get_result(self):
2024-12-04 17:25:16 +00:00
if len(self.pk.split(":")) > 1:
return self.build_from_dpp()
else:
return self.build_from_chid()
def build_from_dpp(self):
2024-11-20 17:20:11 +00:00
data = {
'document': {},
'dpp': self.pk,
'algorithm': ALGORITHM,
2024-12-04 17:25:16 +00:00
'components': [],
2024-11-20 17:20:11 +00:00
'manufacturer DPP': '',
'device': {},
2024-11-20 17:20:11 +00:00
}
2024-12-04 17:25:16 +00:00
dev = Build(self.object.last_evidence.doc, None, check=True)
doc = dev.get_phid()
data['document'] = json.dumps(doc)
data['device'] = dev.device
data['components'] = dev.components
self.object.get_evidences()
last_dpp = Proof.objects.filter(
uuid__in=self.object.uuids, type=PROOF_TYPE['IssueDPP']
).order_by("-timestamp").first()
key = self.pk
if last_dpp:
key = last_dpp.signature
url = "https://{}/did/{}".format(
self.request.get_host(),
key
)
data['url_last'] = url
tmpl = dpp_tmpl.copy()
tmpl["credentialSubject"]["data"] = data
return tmpl
def build_from_chid(self):
2024-11-20 17:20:11 +00:00
dpps = []
2024-11-27 15:53:31 +00:00
self.object.initial()
for d in self.object.evidences:
d.get_doc()
dev = Build(d.doc, None, check=True)
doc = dev.get_phid()
ev = json.dumps(doc)
2024-11-29 14:48:48 +00:00
phid = dev.get_signature(doc)
2024-11-28 15:22:06 +00:00
dpp = "{}:{}".format(self.pk, phid)
2024-11-20 17:20:11 +00:00
rr = {
2024-11-28 15:22:06 +00:00
'dpp': dpp,
2024-11-27 15:53:31 +00:00
'document': ev,
2024-11-20 17:20:11 +00:00
'algorithm': ALGORITHM,
'manufacturer DPP': '',
'device': dev.device,
'components': dev.components
2024-11-20 17:20:11 +00:00
}
2024-12-04 17:25:16 +00:00
tmpl = dpp_tmpl.copy()
tmpl["credentialSubject"]["data"] = rr
dpps.append(tmpl)
2024-11-20 17:20:11 +00:00
return {
'@context': ['https://ereuse.org/dpp0.json'],
'data': dpps,
}
2024-11-20 17:06:52 +00:00
def get_manuals(self):
manuals = {
'ifixit': [],
'icecat': [],
'details': {},
'laer': [],
'energystar': {},
}
try:
params = {
"manufacturer": self.object.manufacturer,
"model": self.object.model,
}
self.params = json.dumps(params)
manuals['ifixit'] = self.request_manuals('ifixit')
manuals['icecat'] = self.request_manuals('icecat')
manuals['laer'] = self.request_manuals('laer')
manuals['energystar'] = self.request_manuals('energystar') or {}
if manuals['icecat']:
manuals['details'] = manuals['icecat'][0]
except Exception as err:
logger.error("Error: {}".format(err))
self.context['manuals'] = manuals
self.parse_energystar()
def parse_energystar(self):
if not self.context.get('manuals', {}).get('energystar'):
return
# Defined in:
# https://dev.socrata.com/foundry/data.energystar.gov/j7nq-iepp
energy_types = [
'functional_adder_allowances_kwh',
'tec_allowance_kwh',
'long_idle_watts',
'short_idle_watts',
'off_mode_watts',
'sleep_mode_watts',
'tec_of_model_kwh',
'tec_requirement_kwh',
'work_off_mode_watts',
'work_weighted_power_of_model_watts',
]
energy = {}
for field in energy_types:
energy[field] = []
for e in self.context['manuals']['energystar']:
for field in energy_types:
for k, v in e.items():
if not v:
continue
if field in k:
energy[field].append(v)
for k, v in energy.items():
if not v:
energy[k] = 0
continue
tt = sum([float(i) for i in v])
energy[k] = round(tt / len(v), 2)
self.context['manuals']['energystar'] = energy
def request_manuals(self, prefix):
#TODO reimplement manuals service
response = {
"laer": [{"metal": 40, "plastic_post_consumer": 27, "plastic_post_industry": 34}],
"energystar": [{
'functional_adder_allowances_kwh': 180,
"long_idle_watts": 240,
"short_idle_watts": 120,
"sleep_mode_watts": 30,
"off_mode_watts": 3,
"tec_allowance_kwh": 180,
"tec_of_model_kwh": 150,
"tec_requirement_kwh": 220,
"work_off_mode_watts": 70,
"work_weighted_power_of_model_watts": 240
}],
"ifixit": [
{
"image": "https://guide-images.cdn.ifixit.com/igi/156EpI4YdQeVfVPa.medium",
"url": "https://es.ifixit.com/Gu%C3%ADa/HP+ProBook+450+G4+Back+Panel+Replacement/171196?lang=en",
"title": "HP ProBook 450 G4 Back Panel Replacement"
},
{
"image": "https://guide-images.cdn.ifixit.com/igi/usTIqCKpuxVWC3Ix.140x105",
"url": "https://es.ifixit.com/Gu%C3%ADa/HP+ProBook+450+G4+Display+Assembly+Replacement/171101?lang=en",
"title": "Display Assembly Replacement"
}
],
"icecat": [
{
"logo": "https://images.icecat.biz/img/brand/thumb/1_cf8603f6de7b4c4d8ac4f5f0ef439a05.jpg",
"image": "https://guide-images.cdn.ifixit.com/igi/Q2nYjTIQfG6GaI5B.standard",
"pdf": "https://icecat.biz/rest/product-pdf?productId=32951710&lang=en",
"title": "HP ProBook 450 G3"
}
]
}
return response.get(prefix, {})