diff --git a/backend/kernelCI_app/serializers.py b/backend/kernelCI_app/serializers.py index ffa6a71..1cd9d7a 100644 --- a/backend/kernelCI_app/serializers.py +++ b/backend/kernelCI_app/serializers.py @@ -67,3 +67,13 @@ def get_test_status(self, obj): "null": obj.null_tests, "total": obj.total_tests } + + +class TreeDetailsSerializer(serializers.Serializer): + id = serializers.CharField() + architecture = serializers.CharField() + config_name = serializers.CharField() + valid = serializers.BooleanField() + start_time = serializers.CharField() + duration = serializers.CharField() + compiler = serializers.CharField() diff --git a/backend/kernelCI_app/urls.py b/backend/kernelCI_app/urls.py index 7084c79..0ee011c 100644 --- a/backend/kernelCI_app/urls.py +++ b/backend/kernelCI_app/urls.py @@ -3,5 +3,6 @@ urlpatterns = [ - path('tree/', views.TreeView.as_view(), name='tree') + path('tree/', views.TreeView.as_view(), name='tree'), + path('tree/', views.TreeDetails.as_view(), name='treeDetails') ] diff --git a/backend/kernelCI_app/views.py b/backend/kernelCI_app/views.py index 82f3544..7002e0c 100644 --- a/backend/kernelCI_app/views.py +++ b/backend/kernelCI_app/views.py @@ -1,8 +1,8 @@ from django.http import JsonResponse from django.views import View -from kernelCI_app.models import Checkouts -from kernelCI_app.serializers import TreeSerializer +from kernelCI_app.models import Checkouts, Builds +from kernelCI_app.serializers import TreeSerializer, TreeDetailsSerializer from kernelCI_app.utils import get_visible_record_identifiers @@ -46,3 +46,57 @@ def get(self, _): serializer = TreeSerializer(checkouts, many=True) resp = JsonResponse(serializer.data, safe=False) return resp + + +class TreeDetails(View): + + def create_default_status(self): + return {"valid": 0, "invalid": 0, "null": 0} + + def create_summary(self, builds_dict): + status_map = {True: 'valid', False: 'invalid', None: 'null'} + + build_summ = self.create_default_status() + config_summ = {} + arch_summ = {} + + for build in builds_dict: + status_key = status_map[build['valid']] + build_summ[status_key] += 1 + + if config := build['config_name']: + status = config_summ.get(config) + if not status: + status = self.create_default_status() + config_summ[config] = status + status[status_key] += 1 + + if arch := build['architecture']: + status = arch_summ.setdefault(arch, self.create_default_status()) + status[status_key] += 1 + compiler = build['compiler'] + if compiler and compiler not in status.setdefault('compilers', []): + status['compilers'].append(compiler) + + return {"builds": build_summ, "configs": config_summ, "architectures": arch_summ} + + def get(self, request, commit_hash): + builds = Builds.objects.raw( + """ + SELECT + builds.id, builds.architecture, builds.config_name, + builds.compiler, builds.valid, + builds.start_time, builds.duration + FROM + builds + INNER JOIN + checkouts ON checkouts.id = builds.checkout_id + WHERE checkouts.git_commit_hash = %s; + """, + [commit_hash] + ) + + data = TreeDetailsSerializer(builds, many=True).data + summary = self.create_summary(data) + resp = {"builds": data, "summary": summary} + return JsonResponse(resp, safe=False)