add progress bar

This commit is contained in:
rmanach 2023-09-22 16:42:34 +02:00
parent b9e87bd0ad
commit 1d9c4e0164
9 changed files with 244 additions and 28 deletions

View File

@ -1,4 +1,4 @@
# Generated by Django 4.2.5 on 2023-09-20 17:23 # Generated by Django 4.2.5 on 2023-09-22 14:39
from django.conf import settings from django.conf import settings
from django.db import migrations, models from django.db import migrations, models
@ -46,6 +46,8 @@ class Migration(migrations.Migration):
), ),
("created_at", models.DateTimeField(auto_now_add=True)), ("created_at", models.DateTimeField(auto_now_add=True)),
("updated_at", models.DateTimeField(auto_now=True)), ("updated_at", models.DateTimeField(auto_now=True)),
("error", models.TextField(null=True)),
("task_id", models.UUIDField(null=True)),
( (
"user", "user",
models.ForeignKey( models.ForeignKey(

View File

@ -1,4 +1,5 @@
from enum import Enum from enum import Enum
from typing import Any
from django.contrib.auth.models import User from django.contrib.auth.models import User
@ -38,7 +39,13 @@ class Deployment(models.Model):
) )
created_at = models.DateTimeField(auto_now_add=True) created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True) updated_at = models.DateTimeField(auto_now=True)
error = models.TextField(null=True, blank=False)
task_id = models.UUIDField(null=True)
user = models.ForeignKey(User, on_delete=models.CASCADE) user = models.ForeignKey(User, on_delete=models.CASCADE)
def __str__(self): def __str__(self):
return f"{self.name} | {self.type} | {self.status}" return f"{self.name} | {self.type} | {self.status}"
def __init__(self, *args: Any, **kwargs: Any):
self.progress = None
super().__init__(*args, **kwargs)

View File

@ -1,6 +1,8 @@
var start = function (url) { var start = function (url) {
var es = new ReconnectingEventSource(url); var es = new ReconnectingEventSource(url);
console.log("url: " + url);
es.onopen = function () { es.onopen = function () {
console.log('connected'); console.log('connected');
}; };
@ -36,6 +38,5 @@ var start = function (url) {
window.location.reload(); window.location.reload();
} }
} }
}, false); }, false);
}; };

View File

@ -0,0 +1,37 @@
var start = function (url) {
var es = new ReconnectingEventSource(url);
console.log("url: " + url);
es.onopen = function () {
console.log('connected');
};
es.addEventListener('stream-error', function (e) {
es.close();
message = JSON.parse(e.data);
console.log('stream error: ' + message.condition + ': ' + message.text);
}, false);
es.onerror = function (e) {
console.log('connection error');
};
es.addEventListener('message', function (e) {
message = JSON.parse(e.data);
console.log("id: " + message.id);
console.log("status: " + message.status);
console.log("progress: " + message.progress);
var progress = document.getElementById("deployment-progress");
progress.style["width"] = message.progress+"%";
if (message.status == "SUCCESS") {
setTimeout(() => window.location.reload(), 1000);
}
if (message.status == "FAILED") {
window.location.reload();
}
}, false);
};

View File

@ -2,31 +2,135 @@ import time
import random import random
from uuid import UUID from uuid import UUID
from celery import shared_task from celery import Task, shared_task
from django_eventstream import send_event from django_eventstream import send_event
from deployment.models import Deployment, Type, Status from deployment.models import Deployment, Type, Status
@shared_task class DeploymentTask(Task):
def deploy(deployment_id: UUID): def try_exec(self, stop: int):
deploy = Deployment.objects.get(id=deployment_id) if stop == 0:
stop = 1
deploy.status = Status.RUNNING.name for i in range(1, stop):
deploy.save() time.sleep(5 * stop - 2 * i)
match deploy.type: if random.randint(0, 10) == 10:
case Type.SLIM.name: raise Exception("unable to perform the deployment")
time.sleep(10)
case Type.MEDIUM.name:
time.sleep(60)
case Type.LARGE.name:
time.sleep(120)
deploy.status = ( yield i
Status.FAILED.name if random.randint(0, 10) % 2 != 0 else Status.SUCCESS.name
)
deploy.save()
send_event("deployment", "message", {"id": deploy.id, "status": deploy.status}) def run_slim(self):
print("event sent") progress = 95
self.update_state(meta={"progress": progress})
time.sleep(5)
status = Status.SUCCESS.name
progress = 100
self.update_state(meta={"progress": progress})
self.deploy.status = status
send_event(
f"deployment-{self.deploy.id}",
"message",
{
"id": self.deploy.id,
"status": self.deploy.status,
"progress": progress,
},
)
def run_medium(self):
progress = 62
self.update_state(meta={"progress": progress})
time.sleep(10)
if random.randint(0, 10) % 2 != 0:
status = Status.FAILED.name
self.deploy.error = "arg.. no chance"
else:
status = Status.SUCCESS.name
progress = 100
self.update_state(meta={"progress": progress})
self.deploy.status = status
send_event(
f"deployment-{self.deploy.id}",
"message",
{
"id": self.deploy.id,
"status": self.deploy.status,
"progress": progress,
},
)
def run_large(self):
progress = 0
try:
for i in self.try_exec(6):
progress = i * 20
self.update_state(meta={"progress": progress})
if progress != 100:
send_event(
f"deployment-{self.deploy.id}",
"message",
{
"id": self.deploy.id,
"status": self.deploy.status
if i != 5
else Status.SUCCESS.name,
"progress": progress,
},
)
except Exception as e:
self.deploy.status = Status.FAILED.name
self.deploy.error = e
else:
self.deploy.status = Status.SUCCESS.name
send_event(
f"deployment-{self.deploy.id}",
"message",
{
"id": self.deploy.id,
"status": self.deploy.status,
"progress": progress,
},
)
@property
def progress(self):
return self.request.get("meta", {}).get("progress", 0)
def launch(self, deployment_id: UUID):
progress = 0
self.update_state(meta={"progress": progress})
self.deploy = Deployment.objects.get(id=deployment_id)
self.deploy.task_id = self.request.id
self.deploy.status = Status.RUNNING.name
self.deploy.save()
match self.deploy.type:
case Type.SLIM.name:
self.run_slim()
case Type.MEDIUM.name:
self.run_medium()
case Type.LARGE.name:
self.run_large()
self.deploy.task_id = None
self.deploy.save()
send_event(
"deployment",
"message",
{"id": self.deploy.id, "status": self.deploy.status},
)
@shared_task(base=DeploymentTask, bind=True, ignore_result=True)
def deploy(self, deployment_id: UUID):
self.launch(deployment_id)

View File

@ -14,4 +14,9 @@ urlpatterns = [
{"channels": ["deployment"]}, {"channels": ["deployment"]},
name="deployment-events", name="deployment-events",
), ),
path(
"<deployment_id>/events/",
include(django_eventstream.urls),
{"format-channels": ["deployment-{deployment_id}"]},
),
] ]

View File

@ -1,5 +1,6 @@
from uuid import uuid4 from uuid import uuid4
from celery.result import AsyncResult
from django.core.paginator import Paginator from django.core.paginator import Paginator
from django.http import ( from django.http import (
HttpResponseRedirect, HttpResponseRedirect,
@ -37,6 +38,10 @@ def deploy(request, deployment_id):
deployment = get_object_or_404(Deployment, id=deployment_id) deployment = get_object_or_404(Deployment, id=deployment_id)
if request.method == "POST": if request.method == "POST":
# override previous errors
if deployment.error:
deployment.error = None
deployment.status = Status.PENDING.name deployment.status = Status.PENDING.name
deployment.save() deployment.save()
launch_deploy.delay(deployment_id) launch_deploy.delay(deployment_id)
@ -49,17 +54,29 @@ def deploy(request, deployment_id):
def details(request, deployment_id): def details(request, deployment_id):
deployment = get_object_or_404(Deployment, id=deployment_id) deployment = get_object_or_404(Deployment, id=deployment_id)
if deployment.status == Status.RUNNING.name:
# retrieve the progression in the backend task
res = AsyncResult(str(deployment.task_id))
deployment.progress = res.info.get("progress", 0)
if request.method == "POST": if request.method == "POST":
if deployment.status == Status.RUNNING.name: if deployment.status == Status.RUNNING.name:
return HttpResponseBadRequest("deployment is running") return HttpResponseBadRequest("deployment is running")
if deployment.status == Status.PENDING.name:
return HttpResponseBadRequest("deployment is pending")
try: try:
deployment.delete() deployment.delete()
except Exception as e: except Exception as e:
return HttpResponseServerError(e) return HttpResponseServerError(e)
return HttpResponseRedirect("/deployment") return HttpResponseRedirect("/deployment")
return render(request, "deployment/details.html", {"deployment": deployment}) return render(
request,
"deployment/details.html",
{"deployment": deployment, "url": f"{deployment.id}/events/"},
)
def create(request): def create(request):

View File

@ -4,7 +4,7 @@ from celery import Celery
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "mumui.settings") os.environ.setdefault("DJANGO_SETTINGS_MODULE", "mumui.settings")
app = Celery("mumui", broker="redis://redis:6379/0") app = Celery("mumui", broker="redis://redis:6379/0", backend="redis://redis:6379/0")
app.config_from_object("django.conf:settings", namespace="CELERY") app.config_from_object("django.conf:settings", namespace="CELERY")
app.autodiscover_tasks() app.autodiscover_tasks()

View File

@ -1,7 +1,27 @@
{% extends 'base.html' %} {% extends 'base.html' %}
{% load static %}
{% block title %} Deployment details: {{ deployment.name }} {% endblock %} {% block title %} Deployment details: {{ deployment.name }} {% endblock %}
{% if deployment.status == "RUNNING" %}
{% block bodyattr %}
{% if deployment.status == "RUNNING" %}
onload="start('{{ url|safe }}');"
{% endif %}
{% endblock %}
{% block headscript %}
{% if deployment.status == "RUNNING" %}
<script src="{% static 'django_eventstream/json2.js' %}"></script>
<script src="{% static 'django_eventstream/eventsource.min.js' %}"></script>
<script src="{% static 'django_eventstream/reconnecting-eventsource.js' %}"></script>
{% endif %}
{% endblock %}
{% endif %}
{% block content %} {% block content %}
<div class="container-fluid"> <div class="container-fluid">
<div class="row justify-content-md-center"> <div class="row justify-content-md-center">
@ -19,6 +39,13 @@
<input type="text" readonly class="form-control-plaintext" id="status" value="{{ deployment.status }}"> <input type="text" readonly class="form-control-plaintext" id="status" value="{{ deployment.status }}">
<label for="status">Status</label> <label for="status">Status</label>
</div> </div>
{% if deployment.status == "RUNNING" %}
<div class="form-floating mb-3">
<div class="progress" role="progressbar" aria-label="deployment-progress" aria-valuenow="{{ deployment.progress }}" aria-valuemin="0" aria-valuemax="100">
<div id="deployment-progress" class="progress-bar progress-bar-striped progress-bar-animated" style="width: {{ deployment.progress }}%"></div>
</div>
</div>
{% endif %}
<div class="form-floating mb-3"> <div class="form-floating mb-3">
<input type="text" readonly class="form-control-plaintext" id="created_at" value="{{ deployment.created_at }}"> <input type="text" readonly class="form-control-plaintext" id="created_at" value="{{ deployment.created_at }}">
<label for="created-at">Created at</label> <label for="created-at">Created at</label>
@ -27,6 +54,18 @@
<input type="text" readonly class="form-control-plaintext" id="updated_at" value="{{ deployment.updated_at }}"> <input type="text" readonly class="form-control-plaintext" id="updated_at" value="{{ deployment.updated_at }}">
<label for="updated-at">Updated at</label> <label for="updated-at">Updated at</label>
</div> </div>
{% if deployment.status == "RUNNING" %}
<div class="form-floating mb-3">
<input type="text" readonly class="form-control-plaintext" id="updated_at" value="{{ deployment.task_id }}">
<label for="updated-at">Task UUID</label>
</div>
{% endif %}
{% if deployment.status == "FAILED" %}
<div class="form-floating mb-3">
<input type="text" readonly class="form-control-plaintext" id="updated_at" value="{{ deployment.error }}">
<label for="updated-at">Error</label>
</div>
{% endif %}
<form action="" method="post"> <form action="" method="post">
{% csrf_token %} {% csrf_token %}
<button type="button" onclick="goBack()" class="btn btn-secondary">Back</button> <button type="button" onclick="goBack()" class="btn btn-secondary">Back</button>
@ -43,9 +82,13 @@
{% endblock %} {% endblock %}
{% block script %} {% block script %}
<script> {% if deployment.status == "RUNNING" %}
function goBack() { <script src="{% static 'deployment/js/event_source_details.js' %}" />
window.location=document.referrer; </script>
} {% endif %}
</script> <script>
function goBack() {
window.location=document.referrer;
}
</script>
{% endblock %} {% endblock %}