From e6215434fb0233487258b619a131e40d6838aac6 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Tue, 26 Sep 2023 12:38:58 +0200 Subject: [PATCH 01/12] Fixed the file upload progress bar --- embark/static/scripts/fileUpload.js | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/embark/static/scripts/fileUpload.js b/embark/static/scripts/fileUpload.js index 06ff4ce7b..fe2fd3247 100644 --- a/embark/static/scripts/fileUpload.js +++ b/embark/static/scripts/fileUpload.js @@ -107,11 +107,7 @@ $(window).bind("load", function() { function saveFiles() { "use strict"; var progressBar = document.getElementById("progress-wrapper"); - if (progressBar.style.display == "none") { - progressBar.style.display = "block"; - } else { - progressBar.style.display = "none"; - } + progressBar.style.display = "block"; var fileData = document.getElementById('file-input').files; var formData = new FormData(); for (let index = 0; index < fileData.length; index++) { From 76d80b833f5c72d2033ded8ae247f952221388f9 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Wed, 27 Sep 2023 13:24:26 +0200 Subject: [PATCH 02/12] Fixed a crash when showing the logs, but the logs contain non-utf-8 bytes --- embark/dashboard/views.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/embark/dashboard/views.py b/embark/dashboard/views.py index 79fb45d90..7d9360de5 100644 --- a/embark/dashboard/views.py +++ b/embark/dashboard/views.py @@ -118,7 +118,7 @@ def show_log(request, analysis_id): log_file_path_ = f"{Path(firmware.path_to_logs).parent}/emba_run.log" logger.debug("Taking file at %s and render it", log_file_path_) try: - with open(log_file_path_, 'r', encoding='utf-8') as log_file_: + with open(log_file_path_, 'rb') as log_file_: return HttpResponse(content=log_file_, content_type="text/plain") except FileNotFoundError: return HttpResponseServerError(content="File is not yet available") From 74fc88cbe2d3bd9990a0d8d287fd8fd407d0fe84 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sat, 30 Sep 2023 16:27:52 +0200 Subject: [PATCH 03/12] Added a log viewer to trace the logs in real time --- embark/dashboard/urls.py | 3 +- embark/dashboard/views.py | 22 ++ embark/embark/consumers.py | 2 +- embark/embark/logviewer.py | 193 ++++++++++++++++++ embark/embark/routing.py | 6 +- embark/embark/tests/__init__.py | 0 embark/embark/tests/test_linecache.py | 44 ++++ embark/static/scripts/logViewer.js | 119 +++++++++++ embark/static/scripts/serviceDashboard.js | 6 + embark/templates/dashboard/logViewer.html | 53 +++++ .../templates/dashboard/reportDashboard.html | 2 +- 11 files changed, 445 insertions(+), 5 deletions(-) create mode 100644 embark/embark/logviewer.py create mode 100644 embark/embark/tests/__init__.py create mode 100644 embark/embark/tests/test_linecache.py create mode 100644 embark/static/scripts/logViewer.js create mode 100644 embark/templates/dashboard/logViewer.html diff --git a/embark/dashboard/urls.py b/embark/dashboard/urls.py index e2cbd14fa..119d770c2 100644 --- a/embark/dashboard/urls.py +++ b/embark/dashboard/urls.py @@ -10,5 +10,6 @@ path('dashboard/report/', views.report_dashboard, name='embark-ReportDashboard'), path('dashboard/individualReport/', views.individual_report_dashboard, name='embark-IndividualReportDashboard'), path('dashboard/stop/', views.stop_analysis, name='embark-stop-analysis'), - path('dashboard/log/', views.show_log, name='embark-show-log') + path('dashboard/log/', views.show_log, name='embark-show-log'), + path('dashboard/logviewer/', views.show_logviewer, name='embark-show-logviewer') ] diff --git a/embark/dashboard/views.py b/embark/dashboard/views.py index 79fb45d90..1dc4e9f29 100644 --- a/embark/dashboard/views.py +++ b/embark/dashboard/views.py @@ -122,3 +122,25 @@ def show_log(request, analysis_id): return HttpResponse(content=log_file_, content_type="text/plain") except FileNotFoundError: return HttpResponseServerError(content="File is not yet available") + +@require_http_methods(["GET"]) +@login_required(login_url='/' + settings.LOGIN_URL) +def show_logviewer(request, analysis_id): + """ + renders a log viewer to scroll through emba_run.log + + :params request: HTTP request + + :return: rendered emba_run.log + """ + + logger.info("showing log viewer for analyze_id: %s", analysis_id) + firmware = FirmwareAnalysis.objects.get(id=analysis_id) + # get the file path + log_file_path_ = f"{Path(firmware.path_to_logs).parent}/emba_run.log" + logger.debug("Taking file at %s and render it", log_file_path_) + try: + return render(request, 'dashboard/logViewer.html', {'analysis_id': analysis_id, 'username': request.user.username}) + + except FileNotFoundError: + return HttpResponseServerError(content="File is not yet available") diff --git a/embark/embark/consumers.py b/embark/embark/consumers.py index d72de08ad..a8703eaa8 100644 --- a/embark/embark/consumers.py +++ b/embark/embark/consumers.py @@ -11,7 +11,7 @@ # consumer class for synchronous/asynchronous websocket communication -class WSConsumer(AsyncWebsocketConsumer): +class ProgressConsumer(AsyncWebsocketConsumer): @database_sync_to_async def get_message(self): diff --git a/embark/embark/logviewer.py b/embark/embark/logviewer.py new file mode 100644 index 000000000..1b4820661 --- /dev/null +++ b/embark/embark/logviewer.py @@ -0,0 +1,193 @@ +import asyncio +import os +import base64 +import logging +import json + +from pathlib import Path +import sys +from watchdog.observers import Observer +from watchdog.events import FileSystemEventHandler +from channels.db import database_sync_to_async +from channels.generic.websocket import AsyncWebsocketConsumer + +from uploader.models import FirmwareAnalysis + +logger = logging.getLogger(__name__) + + +class LogConsumer(AsyncWebsocketConsumer): + def load_file_content(self): + logger.info( + 'Getting file content for analysis id "%s"; view: %s', + self.analysis_id, + self.file_view, + ) + + num_lines = self.line_cache.num_lines() + limit = min(num_lines, self.file_view.limit) + + # Ensure you can't read negative lines + offset = max(self.file_view.offset, 0) + # Ensure you can't read lines bigger than the last line + offset = min(num_lines - limit, offset) + + # Fix the offset, in case the user tried to read invalid lines + self.file_view.offset = offset + + content = self.line_cache.read_lines(offset, offset + limit - 1) + self.file_view.content = base64.b64encode(content).decode("ascii") + self.file_view.num_lines = num_lines + + @database_sync_to_async + def get_firmware(self, analysis_id: str) -> FirmwareAnalysis: + return FirmwareAnalysis.objects.get(id=analysis_id, user=self.scope["user"]) + + async def connect(self): + logger.info("WS - connect") + await self.accept() + logger.info("WS - connect - accept") + + self.analysis_id = self.scope["url_route"]["kwargs"]["analysis_id"] + + firmware = await self.get_firmware(self.analysis_id) + + log_file_path_ = f"{Path(firmware.path_to_logs).parent}/emba_run.log" + + if not os.path.isfile(log_file_path_): + await self.send_message({"error": "The log file does not exist, yet."}) + await self.close() + + self.line_cache = LineCache(log_file_path_) + + self.file_view = FileView() + + this = self + + class ModifyEventHandler(FileSystemEventHandler): + def on_modified(self, event): + asyncio.run(this.update_lines()) + + event_handler = ModifyEventHandler() + + self.observer = Observer() + self.observer.schedule(event_handler, log_file_path_) + self.observer.start() + + await self.update_lines() + + async def send_file_content(self) -> None: + self.load_file_content() + await self.send_message({"file_view": self.file_view.__dict__}) + + async def update_lines(self) -> None: + self.line_cache.refresh() + await self.send_file_content() + + async def receive(self, text_data: str = "", bytes_data=None) -> None: + logger.info("WS - receive") + try: + data = json.loads(text_data) + if data["action"] == "change_view": + logger.info("WS - action: change view") + logger.info(data["file_view"]) + self.file_view = FileView(**data["file_view"]) + await self.send_file_content() + else: + raise Exception("Unknown action") + except Exception as e: + logger.error(e) + await self.send_message({"error": "Unknown error"}) + + async def disconnect(self, code): + logger.info("WS - disconnected: %s", code) + if self.line_cache: + self.line_cache.close() + + if self.observer: + self.observer.stop() + + # send data to frontend + async def send_message(self, message: dict) -> None: + # logger.info(f"WS - send message: " + str(message)) + logger.info("WS - send message") + # Send message to WebSocket + await self.send(json.dumps(message, sort_keys=False)) + + +class FileView: + def __init__(self, offset=0, limit=30, content="", num_lines=0) -> None: + self.offset = offset + self.limit = limit + self.content = content + self.num_lines = num_lines + + +refresh_lines = 10 # This will define how many of the last lines will be refreshed + + +class LineCache: + def __init__(self, filepath: str) -> None: + self.line_beginnings = [0] + self.line_endings = [0] + self.filehandle = open(filepath, "rb") + self.refresh() + + def refresh(self) -> None: + refresh_from_line = min( + len(self.line_beginnings), refresh_lines + ) # Make sure to not go out of bounds + refresh_from_byte = self.line_beginnings[-refresh_from_line] + self.line_beginnings = self.line_beginnings[ + :-refresh_from_line + ] # Remove the line beginnings to be refreshed + self.line_endings = self.line_endings[ + :-refresh_from_line + ] # Remove the line endings to be refreshed + + logger.debug( + f"Start refreshing line cache from line {refresh_from_line} (start counting from end of the file)" + ) + logger.debug(f"Start refreshing line cache from byte {refresh_from_byte}") + + self.filehandle.seek(refresh_from_byte) + + while True: + line_beginning = self.filehandle.tell() + line = self.filehandle.readline() + + line_ending = self.filehandle.tell() + + if len(line) > 0 and line[-1:] == b'\n': + line_ending = line_ending - 1 + + self.line_beginnings.append(line_beginning) + self.line_endings.append(line_ending) + + if len(line) == 0 or line[-1:] != b'\n': + break + + def num_lines(self) -> int: + return len(self.line_beginnings) + + def read_lines(self, first_line: int, last_line: int) -> bytes: + num_lines = self.num_lines() + + if first_line > last_line: + raise Exception("The first line cannot be bigger than the last line") + + if first_line < 0: + raise IndexError("The first line cannot be below 0") + + if last_line >= num_lines: + raise IndexError("The first line cannot be equal or above the number of lines") + + first_byte = self.line_beginnings[num_lines - last_line - 1] + last_byte = self.line_endings[num_lines - first_line - 1] + self.filehandle.seek(first_byte) + output = self.filehandle.read(last_byte - first_byte) + + return output + + def close(self) -> None: + self.filehandle.close() diff --git a/embark/embark/routing.py b/embark/embark/routing.py index 2dffe2cf2..7cc3059c0 100644 --- a/embark/embark/routing.py +++ b/embark/embark/routing.py @@ -1,9 +1,11 @@ from django.urls import path # from django.urls import re_path # from django.conf.urls import url -from embark.consumers import WSConsumer +from embark.consumers import ProgressConsumer +from embark.logviewer import LogConsumer # url patterns for websocket communication -> equivalent to urls.py ws_urlpatterns = [ - path('ws/progress/', WSConsumer.as_asgi()) + path('ws/progress/', ProgressConsumer.as_asgi()), + path('ws/logs/', LogConsumer.as_asgi()) ] diff --git a/embark/embark/tests/__init__.py b/embark/embark/tests/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/embark/embark/tests/test_linecache.py b/embark/embark/tests/test_linecache.py new file mode 100644 index 000000000..2cdcf8748 --- /dev/null +++ b/embark/embark/tests/test_linecache.py @@ -0,0 +1,44 @@ +import unittest + +from embark.logviewer import LineCache + +class TestLineCache(unittest.TestCase): + + def test_default(self): + line_cache = LineCache('./test/logviewer/line_cache_test1.log') + + for i in range(0,2): + self.assertEqual(12, line_cache.num_lines(), 'Incorrect number of lines.') + self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') + self.assertEqual([0, 7, 11, 23, 31, 41, 50, 54, 58, 72, 86, 104], line_cache.line_beginnings, 'The line beginning cache is not valid.') + line_cache.refresh() + + self.assertEqual(b'10: ggggggggg\n11: hhhhhhhhhhhhh\n', line_cache.read_lines(0, 2), 'The line cache did not return the correct value.') + self.assertEqual(b'9: ffffffffff\n10: ggggggggg\n11: hhhhhhhhhhhhh', line_cache.read_lines(1, 3), 'The line cache did not return the correct value.') + + def test_no_newline_end(self): + line_cache = LineCache('./test/logviewer/line_cache_test_no_newline.log') + + for i in range(0,2): + self.assertEqual(11, line_cache.num_lines(), 'Incorrect number of lines.') + self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') + self.assertEqual([0, 7, 11, 23, 31, 41, 50, 54, 58, 72, 86], line_cache.line_beginnings, 'The line beginning cache is not valid.') + line_cache.refresh() + + self.assertEqual(b'9: ffffffffff\n10: ggggggggg\n11: hhhhhhhhhhhhh', line_cache.read_lines(0, 2), 'The line cache did not return the correct value.') + self.assertEqual(b'8: \n9: ffffffffff\n10: ggggggggg', line_cache.read_lines(1, 3), 'The line cache did not return the correct value.') + + def test_empty(self): + line_cache = LineCache('./test/logviewer/line_cache_test_empty.log') + + for i in range(0,2): + self.assertEqual(1, line_cache.num_lines(), 'Incorrect number of lines.') + self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') + self.assertEqual([0], line_cache.line_beginnings, 'The line beginning cache is not valid.') + line_cache.refresh() + + self.assertEqual(b'', line_cache.read_lines(0, 0), 'The line cache did not return the correct value.') + + +if __name__ == '__main__': + unittest.main() \ No newline at end of file diff --git a/embark/static/scripts/logViewer.js b/embark/static/scripts/logViewer.js new file mode 100644 index 000000000..aadfbadb6 --- /dev/null +++ b/embark/static/scripts/logViewer.js @@ -0,0 +1,119 @@ +// jshint unused:false +// ^ this should only be added AFTER successfull check (disables waring for global functions) + +window.addEventListener("load", onLoad, false); + +function onLoad() { + var loc = window.location; + var wsStart = 'ws://'; + var wsPort = ':8001'; + if (loc.protocol == 'https:') { + wsStart = 'wss://'; + wsPort = ':8000'; + } + var socket = new WebSocket( + wsStart + location.hostname + wsPort + '/ws/logs/' + analysis_id + ); + + var file_view = { + offset: 0, + limit: 50, + content: '' + }; + + var logArea = document.getElementById('logArea') + + socket.onopen = function (evt) { onOpen(evt) }; + socket.onclose = function (evt) { onClose(evt) }; + socket.onmessage = function (evt) { onMessage(evt) }; + socket.onerror = function (evt) { onError(evt) }; + + function onOpen(evt) { + console.log('open', evt); + } + + function onClose(evt) { + console.log('close', evt); + } + + function onMessage(evt) { + // There are two types of messages: + // 1. a chat participant message itself + // 2. a message with a number of connected chat participants + var message = JSON.parse(evt.data); + + if (message.file_view) { + fileContent = atob(message.file_view.content) + '\u00a0' // The nbsp is required in order to preserve trailing newlines + logArea.textContent = fileContent + file_view = message.file_view + } + } + + function onError(evt) { + console.log('error', evt); + } + + function requestUpdate() { + request_view = Object.assign({}, file_view) + request_view.content = '' + socket.send(JSON.stringify({ 'action': 'change_view', 'file_view': request_view })); + } + + window.LogControls = { + move_offset: function(lines) { + file_view.offset += lines + requestUpdate(); + }, + increase_view_size: function(lines) { + file_view.limit += lines + if (file_view.limit < 5) { + file_view.limit = 5 + } + if (file_view.num_lines != 0 && file_view.limit > file_view.num_lines) { + file_view.limit = file_view.num_lines + } + requestUpdate(); + } + } + + document.onkeyup = checkKey; + + function checkKey(e) { + if (e.shiftKey) { + if (e.keyCode == '38') { // Arrow Up + window.LogControls.increase_view_size(1); + e.preventDefault(); + } + else if (e.keyCode == '40') { // Arrow Down + window.LogControls.increase_view_size(-1); + e.preventDefault(); + } + else if (e.keyCode == '33') { // Page Up + window.LogControls.increase_view_size(5); + e.preventDefault(); + } + else if (e.keyCode == '34') { // Page Down + window.LogControls.increase_view_size(-5); + e.preventDefault(); + } + } + else { + if (e.keyCode == '38') { // Arrow Up + window.LogControls.move_offset(1); + e.preventDefault(); + } + else if (e.keyCode == '40') { // Arrow Down + window.LogControls.move_offset(-1); + e.preventDefault(); + } + else if (e.keyCode == '33') { // Page Up + window.LogControls.move_offset(5); + e.preventDefault(); + } + else if (e.keyCode == '34') { // Page Down + window.LogControls.move_offset(-5); + e.preventDefault(); + } + } + } +} \ No newline at end of file diff --git a/embark/static/scripts/serviceDashboard.js b/embark/static/scripts/serviceDashboard.js index 2eafeb29d..2433c2cfc 100644 --- a/embark/static/scripts/serviceDashboard.js +++ b/embark/static/scripts/serviceDashboard.js @@ -135,6 +135,12 @@ function add_container_to_running(status_dict) {
    + +
    +
    + +
    +
    0 % diff --git a/embark/templates/dashboard/logViewer.html b/embark/templates/dashboard/logViewer.html new file mode 100644 index 000000000..088a90d33 --- /dev/null +++ b/embark/templates/dashboard/logViewer.html @@ -0,0 +1,53 @@ +{% extends "base.html" %} +{% load static %} +{% block title %}EMBArk Logs{% endblock title %} +{% block navigation %}{% include "navigation.html" %}{% endblock navigation %} +{% block maincontent %} +
    +
    +

    Logs

    +
    +        
    +
    +
    + +
    +
    +
    +
    + +
    +
    +

    Controls

    +
    +
    + + + +
    +
    + + + +
    +
    +
    +
    + +{% endblock maincontent %} +{% block inlinejs %} + + +{% endblock inlinejs %} \ No newline at end of file diff --git a/embark/templates/dashboard/reportDashboard.html b/embark/templates/dashboard/reportDashboard.html index 5268538ab..16d87b476 100644 --- a/embark/templates/dashboard/reportDashboard.html +++ b/embark/templates/dashboard/reportDashboard.html @@ -80,7 +80,7 @@ {% endif %} {% endif %} -
    +
    From 0ae6ed573ba0abf264b3cd9c5fe4ddda10202203 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sat, 30 Sep 2023 16:31:26 +0200 Subject: [PATCH 04/12] Updated the gitignore to ignore uploaded filed --- .gitignore | 3 +- Pipfile | 1 + Pipfile.lock | 919 ++++++++++++---------- embark/dashboard/views.py | 3 +- embark/embark/logviewer.py | 61 +- embark/embark/tests/test_linecache.py | 9 +- embark/static/scripts/logViewer.js | 196 ++--- embark/templates/dashboard/logViewer.html | 16 +- 8 files changed, 643 insertions(+), 565 deletions(-) diff --git a/.gitignore b/.gitignore index 7c47be4ff..efc85077b 100644 --- a/.gitignore +++ b/.gitignore @@ -23,4 +23,5 @@ safe/ *.sql # migrations -**/migrations/*_initial.py \ No newline at end of file +**/migrations/*_initial.py +media/**/* \ No newline at end of file diff --git a/Pipfile b/Pipfile index 089159c54..705810ce7 100644 --- a/Pipfile +++ b/Pipfile @@ -21,6 +21,7 @@ django-bootstrap5 = "*" django-tables2 = "*" requests = "*" djangorestframework = "*" +watchdog = "*" [dev-packages] pycodestyle = "*" diff --git a/Pipfile.lock b/Pipfile.lock index 7d48620eb..7195a1bd6 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "7323bb76dbee072a96dc74e032a1dc12b2b9a0feab12555ad237833afa8a1083" + "sha256": "92aa5498317424c08dd5c8a4729782ca5d2fb5dd279d5de93ed41b49dc78152a" }, "pipfile-spec": 6, "requires": { @@ -18,11 +18,11 @@ "default": { "apscheduler": { "hashes": [ - "sha256:0293937d8f6051a0f493359440c1a1b93e882c57daf0197afeff0e727777b96e", - "sha256:e813ad5ada7aff36fb08cdda746b520531eaac7757832abc204868ba78e0c8f6" + "sha256:e6df071b27d9be898e486bc7940a7be50b4af2e9da7c08f0744a96d4bd4cef4a", + "sha256:fb91e8a768632a4756a585f79ec834e0e27aad5860bac7eaa523d9ccefd87661" ], - "markers": "python_full_version >= '3.6.0'", - "version": "==3.10.1" + "markers": "python_version >= '3.6'", + "version": "==3.10.4" }, "asgiref": { "hashes": [ @@ -34,11 +34,11 @@ }, "async-timeout": { "hashes": [ - "sha256:2163e1640ddb52b7a8c80d0a67a08587e5d245cc9c553a74a847056bc2976b15", - "sha256:8ca1e4fcf50d07413d66d1a5e416e42cfdf5851c981d679a09851a6853383b3c" + "sha256:4640d96be84d82d02ed59ea2b7105a0f7b33abe8703703cd0ab0bf87c427522f", + "sha256:7405140ff1230c310e51dc27b3145b9092d659ce68ff733fb0cefe3ee42be028" ], "markers": "python_full_version <= '3.11.2'", - "version": "==4.0.2" + "version": "==4.0.3" }, "attrs": { "hashes": [ @@ -67,77 +67,66 @@ "sha256:539cc1d13202e33ca466e88b2807e29f4c13049d6d87031a3c110744495cb082", "sha256:92d6037539857d8206b8f6ae472e8b77db8058fec5937a1ef3f54304089edbb9" ], - "markers": "python_full_version >= '3.6.0'", + "markers": "python_version >= '3.6'", "version": "==2023.7.22" }, "cffi": { "hashes": [ - "sha256:00a9ed42e88df81ffae7a8ab6d9356b371399b91dbdf0c3cb1e84c03a13aceb5", - "sha256:03425bdae262c76aad70202debd780501fabeaca237cdfddc008987c0e0f59ef", - "sha256:04ed324bda3cda42b9b695d51bb7d54b680b9719cfab04227cdd1e04e5de3104", - "sha256:0e2642fe3142e4cc4af0799748233ad6da94c62a8bec3a6648bf8ee68b1c7426", - "sha256:173379135477dc8cac4bc58f45db08ab45d228b3363adb7af79436135d028405", - "sha256:198caafb44239b60e252492445da556afafc7d1e3ab7a1fb3f0584ef6d742375", - "sha256:1e74c6b51a9ed6589199c787bf5f9875612ca4a8a0785fb2d4a84429badaf22a", - "sha256:2012c72d854c2d03e45d06ae57f40d78e5770d252f195b93f581acf3ba44496e", - "sha256:21157295583fe8943475029ed5abdcf71eb3911894724e360acff1d61c1d54bc", - "sha256:2470043b93ff09bf8fb1d46d1cb756ce6132c54826661a32d4e4d132e1977adf", - "sha256:285d29981935eb726a4399badae8f0ffdff4f5050eaa6d0cfc3f64b857b77185", - "sha256:30d78fbc8ebf9c92c9b7823ee18eb92f2e6ef79b45ac84db507f52fbe3ec4497", - "sha256:320dab6e7cb2eacdf0e658569d2575c4dad258c0fcc794f46215e1e39f90f2c3", - "sha256:33ab79603146aace82c2427da5ca6e58f2b3f2fb5da893ceac0c42218a40be35", - "sha256:3548db281cd7d2561c9ad9984681c95f7b0e38881201e157833a2342c30d5e8c", - "sha256:3799aecf2e17cf585d977b780ce79ff0dc9b78d799fc694221ce814c2c19db83", - "sha256:39d39875251ca8f612b6f33e6b1195af86d1b3e60086068be9cc053aa4376e21", - "sha256:3b926aa83d1edb5aa5b427b4053dc420ec295a08e40911296b9eb1b6170f6cca", - "sha256:3bcde07039e586f91b45c88f8583ea7cf7a0770df3a1649627bf598332cb6984", - "sha256:3d08afd128ddaa624a48cf2b859afef385b720bb4b43df214f85616922e6a5ac", - "sha256:3eb6971dcff08619f8d91607cfc726518b6fa2a9eba42856be181c6d0d9515fd", - "sha256:40f4774f5a9d4f5e344f31a32b5096977b5d48560c5592e2f3d2c4374bd543ee", - "sha256:4289fc34b2f5316fbb762d75362931e351941fa95fa18789191b33fc4cf9504a", - "sha256:470c103ae716238bbe698d67ad020e1db9d9dba34fa5a899b5e21577e6d52ed2", - "sha256:4f2c9f67e9821cad2e5f480bc8d83b8742896f1242dba247911072d4fa94c192", - "sha256:50a74364d85fd319352182ef59c5c790484a336f6db772c1a9231f1c3ed0cbd7", - "sha256:54a2db7b78338edd780e7ef7f9f6c442500fb0d41a5a4ea24fff1c929d5af585", - "sha256:5635bd9cb9731e6d4a1132a498dd34f764034a8ce60cef4f5319c0541159392f", - "sha256:59c0b02d0a6c384d453fece7566d1c7e6b7bae4fc5874ef2ef46d56776d61c9e", - "sha256:5d598b938678ebf3c67377cdd45e09d431369c3b1a5b331058c338e201f12b27", - "sha256:5df2768244d19ab7f60546d0c7c63ce1581f7af8b5de3eb3004b9b6fc8a9f84b", - "sha256:5ef34d190326c3b1f822a5b7a45f6c4535e2f47ed06fec77d3d799c450b2651e", - "sha256:6975a3fac6bc83c4a65c9f9fcab9e47019a11d3d2cf7f3c0d03431bf145a941e", - "sha256:6c9a799e985904922a4d207a94eae35c78ebae90e128f0c4e521ce339396be9d", - "sha256:70df4e3b545a17496c9b3f41f5115e69a4f2e77e94e1d2a8e1070bc0c38c8a3c", - "sha256:7473e861101c9e72452f9bf8acb984947aa1661a7704553a9f6e4baa5ba64415", - "sha256:8102eaf27e1e448db915d08afa8b41d6c7ca7a04b7d73af6514df10a3e74bd82", - "sha256:87c450779d0914f2861b8526e035c5e6da0a3199d8f1add1a665e1cbc6fc6d02", - "sha256:8b7ee99e510d7b66cdb6c593f21c043c248537a32e0bedf02e01e9553a172314", - "sha256:91fc98adde3d7881af9b59ed0294046f3806221863722ba7d8d120c575314325", - "sha256:94411f22c3985acaec6f83c6df553f2dbe17b698cc7f8ae751ff2237d96b9e3c", - "sha256:98d85c6a2bef81588d9227dde12db8a7f47f639f4a17c9ae08e773aa9c697bf3", - "sha256:9ad5db27f9cabae298d151c85cf2bad1d359a1b9c686a275df03385758e2f914", - "sha256:a0b71b1b8fbf2b96e41c4d990244165e2c9be83d54962a9a1d118fd8657d2045", - "sha256:a0f100c8912c114ff53e1202d0078b425bee3649ae34d7b070e9697f93c5d52d", - "sha256:a591fe9e525846e4d154205572a029f653ada1a78b93697f3b5a8f1f2bc055b9", - "sha256:a5c84c68147988265e60416b57fc83425a78058853509c1b0629c180094904a5", - "sha256:a66d3508133af6e8548451b25058d5812812ec3798c886bf38ed24a98216fab2", - "sha256:a8c4917bd7ad33e8eb21e9a5bbba979b49d9a97acb3a803092cbc1133e20343c", - "sha256:b3bbeb01c2b273cca1e1e0c5df57f12dce9a4dd331b4fa1635b8bec26350bde3", - "sha256:cba9d6b9a7d64d4bd46167096fc9d2f835e25d7e4c121fb2ddfc6528fb0413b2", - "sha256:cc4d65aeeaa04136a12677d3dd0b1c0c94dc43abac5860ab33cceb42b801c1e8", - "sha256:ce4bcc037df4fc5e3d184794f27bdaab018943698f4ca31630bc7f84a7b69c6d", - "sha256:cec7d9412a9102bdc577382c3929b337320c4c4c4849f2c5cdd14d7368c5562d", - "sha256:d400bfb9a37b1351253cb402671cea7e89bdecc294e8016a707f6d1d8ac934f9", - "sha256:d61f4695e6c866a23a21acab0509af1cdfd2c013cf256bbf5b6b5e2695827162", - "sha256:db0fbb9c62743ce59a9ff687eb5f4afbe77e5e8403d6697f7446e5f609976f76", - "sha256:dd86c085fae2efd48ac91dd7ccffcfc0571387fe1193d33b6394db7ef31fe2a4", - "sha256:e00b098126fd45523dd056d2efba6c5a63b71ffe9f2bbe1a4fe1716e1d0c331e", - "sha256:e229a521186c75c8ad9490854fd8bbdd9a0c9aa3a524326b55be83b54d4e0ad9", - "sha256:e263d77ee3dd201c3a142934a086a4450861778baaeeb45db4591ef65550b0a6", - "sha256:ed9cb427ba5504c1dc15ede7d516b84757c3e3d7868ccc85121d9310d27eed0b", - "sha256:fa6693661a4c91757f4412306191b6dc88c1703f780c8234035eac011922bc01", - "sha256:fcd131dd944808b5bdb38e6f5b53013c5aa4f334c5cad0c72742f6eba4b73db0" - ], - "version": "==1.15.1" + "sha256:0c9ef6ff37e974b73c25eecc13952c55bceed9112be2d9d938ded8e856138bcc", + "sha256:131fd094d1065b19540c3d72594260f118b231090295d8c34e19a7bbcf2e860a", + "sha256:1b8ebc27c014c59692bb2664c7d13ce7a6e9a629be20e54e7271fa696ff2b417", + "sha256:2c56b361916f390cd758a57f2e16233eb4f64bcbeee88a4881ea90fca14dc6ab", + "sha256:2d92b25dbf6cae33f65005baf472d2c245c050b1ce709cc4588cdcdd5495b520", + "sha256:31d13b0f99e0836b7ff893d37af07366ebc90b678b6664c955b54561fc36ef36", + "sha256:32c68ef735dbe5857c810328cb2481e24722a59a2003018885514d4c09af9743", + "sha256:3686dffb02459559c74dd3d81748269ffb0eb027c39a6fc99502de37d501faa8", + "sha256:582215a0e9adbe0e379761260553ba11c58943e4bbe9c36430c4ca6ac74b15ed", + "sha256:5b50bf3f55561dac5438f8e70bfcdfd74543fd60df5fa5f62d94e5867deca684", + "sha256:5bf44d66cdf9e893637896c7faa22298baebcd18d1ddb6d2626a6e39793a1d56", + "sha256:6602bc8dc6f3a9e02b6c22c4fc1e47aa50f8f8e6d3f78a5e16ac33ef5fefa324", + "sha256:673739cb539f8cdaa07d92d02efa93c9ccf87e345b9a0b556e3ecc666718468d", + "sha256:68678abf380b42ce21a5f2abde8efee05c114c2fdb2e9eef2efdb0257fba1235", + "sha256:68e7c44931cc171c54ccb702482e9fc723192e88d25a0e133edd7aff8fcd1f6e", + "sha256:6b3d6606d369fc1da4fd8c357d026317fbb9c9b75d36dc16e90e84c26854b088", + "sha256:748dcd1e3d3d7cd5443ef03ce8685043294ad6bd7c02a38d1bd367cfd968e000", + "sha256:7651c50c8c5ef7bdb41108b7b8c5a83013bfaa8a935590c5d74627c047a583c7", + "sha256:7b78010e7b97fef4bee1e896df8a4bbb6712b7f05b7ef630f9d1da00f6444d2e", + "sha256:7e61e3e4fa664a8588aa25c883eab612a188c725755afff6289454d6362b9673", + "sha256:80876338e19c951fdfed6198e70bc88f1c9758b94578d5a7c4c91a87af3cf31c", + "sha256:8895613bcc094d4a1b2dbe179d88d7fb4a15cee43c052e8885783fac397d91fe", + "sha256:88e2b3c14bdb32e440be531ade29d3c50a1a59cd4e51b1dd8b0865c54ea5d2e2", + "sha256:8f8e709127c6c77446a8c0a8c8bf3c8ee706a06cd44b1e827c3e6a2ee6b8c098", + "sha256:9cb4a35b3642fc5c005a6755a5d17c6c8b6bcb6981baf81cea8bfbc8903e8ba8", + "sha256:9f90389693731ff1f659e55c7d1640e2ec43ff725cc61b04b2f9c6d8d017df6a", + "sha256:a09582f178759ee8128d9270cd1344154fd473bb77d94ce0aeb2a93ebf0feaf0", + "sha256:a6a14b17d7e17fa0d207ac08642c8820f84f25ce17a442fd15e27ea18d67c59b", + "sha256:a72e8961a86d19bdb45851d8f1f08b041ea37d2bd8d4fd19903bc3083d80c896", + "sha256:abd808f9c129ba2beda4cfc53bde801e5bcf9d6e0f22f095e45327c038bfe68e", + "sha256:ac0f5edd2360eea2f1daa9e26a41db02dd4b0451b48f7c318e217ee092a213e9", + "sha256:b29ebffcf550f9da55bec9e02ad430c992a87e5f512cd63388abb76f1036d8d2", + "sha256:b2ca4e77f9f47c55c194982e10f058db063937845bb2b7a86c84a6cfe0aefa8b", + "sha256:b7be2d771cdba2942e13215c4e340bfd76398e9227ad10402a8767ab1865d2e6", + "sha256:b84834d0cf97e7d27dd5b7f3aca7b6e9263c56308ab9dc8aae9784abb774d404", + "sha256:b86851a328eedc692acf81fb05444bdf1891747c25af7529e39ddafaf68a4f3f", + "sha256:bcb3ef43e58665bbda2fb198698fcae6776483e0c4a631aa5647806c25e02cc0", + "sha256:c0f31130ebc2d37cdd8e44605fb5fa7ad59049298b3f745c74fa74c62fbfcfc4", + "sha256:c6a164aa47843fb1b01e941d385aab7215563bb8816d80ff3a363a9f8448a8dc", + "sha256:d8a9d3ebe49f084ad71f9269834ceccbf398253c9fac910c4fd7053ff1386936", + "sha256:db8e577c19c0fda0beb7e0d4e09e0ba74b1e4c092e0e40bfa12fe05b6f6d75ba", + "sha256:dc9b18bf40cc75f66f40a7379f6a9513244fe33c0e8aa72e2d56b0196a7ef872", + "sha256:e09f3ff613345df5e8c3667da1d918f9149bd623cd9070c983c013792a9a62eb", + "sha256:e4108df7fe9b707191e55f33efbcb2d81928e10cea45527879a4749cbe472614", + "sha256:e6024675e67af929088fda399b2094574609396b1decb609c55fa58b028a32a1", + "sha256:e70f54f1796669ef691ca07d046cd81a29cb4deb1e5f942003f401c0c4a2695d", + "sha256:e715596e683d2ce000574bae5d07bd522c781a822866c20495e52520564f0969", + "sha256:e760191dd42581e023a68b758769e2da259b5d52e3103c6060ddc02c9edb8d7b", + "sha256:ed86a35631f7bfbb28e108dd96773b9d5a6ce4811cf6ea468bb6a359b256b1e4", + "sha256:ee07e47c12890ef248766a6e55bd38ebfb2bb8edd4142d56db91b21ea68b7627", + "sha256:fa3a0128b152627161ce47201262d3140edb5a5c3da88d73a1b790a959126956", + "sha256:fcc8eb6d5902bb1cf6dc4f187ee3ea80a1eba0a89aba40a5cb20a5087d961357" + ], + "markers": "python_version >= '3.8'", + "version": "==1.16.0" }, "channels": { "hashes": [ @@ -145,6 +134,7 @@ "sha256:2253334ac76f67cba68c2072273f7e0e67dbdac77eeb7e318f511d2f9a53c5e4" ], "index": "pypi", + "markers": "python_version >= '3.7'", "version": "==4.0.0" }, "channels-redis": { @@ -153,88 +143,104 @@ "sha256:6bd4f75f4ab4a7db17cee495593ace886d7e914c66f8214a1f247ff6659c073a" ], "index": "pypi", + "markers": "python_version >= '3.7'", "version": "==4.1.0" }, "charset-normalizer": { "hashes": [ - "sha256:04e57ab9fbf9607b77f7d057974694b4f6b142da9ed4a199859d9d4d5c63fe96", - "sha256:09393e1b2a9461950b1c9a45d5fd251dc7c6f228acab64da1c9c0165d9c7765c", - "sha256:0b87549028f680ca955556e3bd57013ab47474c3124dc069faa0b6545b6c9710", - "sha256:1000fba1057b92a65daec275aec30586c3de2401ccdcd41f8a5c1e2c87078706", - "sha256:1249cbbf3d3b04902ff081ffbb33ce3377fa6e4c7356f759f3cd076cc138d020", - "sha256:1920d4ff15ce893210c1f0c0e9d19bfbecb7983c76b33f046c13a8ffbd570252", - "sha256:193cbc708ea3aca45e7221ae58f0fd63f933753a9bfb498a3b474878f12caaad", - "sha256:1a100c6d595a7f316f1b6f01d20815d916e75ff98c27a01ae817439ea7726329", - "sha256:1f30b48dd7fa1474554b0b0f3fdfdd4c13b5c737a3c6284d3cdc424ec0ffff3a", - "sha256:203f0c8871d5a7987be20c72442488a0b8cfd0f43b7973771640fc593f56321f", - "sha256:246de67b99b6851627d945db38147d1b209a899311b1305dd84916f2b88526c6", - "sha256:2dee8e57f052ef5353cf608e0b4c871aee320dd1b87d351c28764fc0ca55f9f4", - "sha256:2efb1bd13885392adfda4614c33d3b68dee4921fd0ac1d3988f8cbb7d589e72a", - "sha256:2f4ac36d8e2b4cc1aa71df3dd84ff8efbe3bfb97ac41242fbcfc053c67434f46", - "sha256:3170c9399da12c9dc66366e9d14da8bf7147e1e9d9ea566067bbce7bb74bd9c2", - "sha256:3b1613dd5aee995ec6d4c69f00378bbd07614702a315a2cf6c1d21461fe17c23", - "sha256:3bb3d25a8e6c0aedd251753a79ae98a093c7e7b471faa3aa9a93a81431987ace", - "sha256:3bb7fda7260735efe66d5107fb7e6af6a7c04c7fce9b2514e04b7a74b06bf5dd", - "sha256:41b25eaa7d15909cf3ac4c96088c1f266a9a93ec44f87f1d13d4a0e86c81b982", - "sha256:45de3f87179c1823e6d9e32156fb14c1927fcc9aba21433f088fdfb555b77c10", - "sha256:46fb8c61d794b78ec7134a715a3e564aafc8f6b5e338417cb19fe9f57a5a9bf2", - "sha256:48021783bdf96e3d6de03a6e39a1171ed5bd7e8bb93fc84cc649d11490f87cea", - "sha256:4957669ef390f0e6719db3613ab3a7631e68424604a7b448f079bee145da6e09", - "sha256:5e86d77b090dbddbe78867a0275cb4df08ea195e660f1f7f13435a4649e954e5", - "sha256:6339d047dab2780cc6220f46306628e04d9750f02f983ddb37439ca47ced7149", - "sha256:681eb3d7e02e3c3655d1b16059fbfb605ac464c834a0c629048a30fad2b27489", - "sha256:6c409c0deba34f147f77efaa67b8e4bb83d2f11c8806405f76397ae5b8c0d1c9", - "sha256:7095f6fbfaa55defb6b733cfeb14efaae7a29f0b59d8cf213be4e7ca0b857b80", - "sha256:70c610f6cbe4b9fce272c407dd9d07e33e6bf7b4aa1b7ffb6f6ded8e634e3592", - "sha256:72814c01533f51d68702802d74f77ea026b5ec52793c791e2da806a3844a46c3", - "sha256:7a4826ad2bd6b07ca615c74ab91f32f6c96d08f6fcc3902ceeedaec8cdc3bcd6", - "sha256:7c70087bfee18a42b4040bb9ec1ca15a08242cf5867c58726530bdf3945672ed", - "sha256:855eafa5d5a2034b4621c74925d89c5efef61418570e5ef9b37717d9c796419c", - "sha256:8700f06d0ce6f128de3ccdbc1acaea1ee264d2caa9ca05daaf492fde7c2a7200", - "sha256:89f1b185a01fe560bc8ae5f619e924407efca2191b56ce749ec84982fc59a32a", - "sha256:8b2c760cfc7042b27ebdb4a43a4453bd829a5742503599144d54a032c5dc7e9e", - "sha256:8c2f5e83493748286002f9369f3e6607c565a6a90425a3a1fef5ae32a36d749d", - "sha256:8e098148dd37b4ce3baca71fb394c81dc5d9c7728c95df695d2dca218edf40e6", - "sha256:94aea8eff76ee6d1cdacb07dd2123a68283cb5569e0250feab1240058f53b623", - "sha256:95eb302ff792e12aba9a8b8f8474ab229a83c103d74a750ec0bd1c1eea32e669", - "sha256:9bd9b3b31adcb054116447ea22caa61a285d92e94d710aa5ec97992ff5eb7cf3", - "sha256:9e608aafdb55eb9f255034709e20d5a83b6d60c054df0802fa9c9883d0a937aa", - "sha256:a103b3a7069b62f5d4890ae1b8f0597618f628b286b03d4bc9195230b154bfa9", - "sha256:a386ebe437176aab38c041de1260cd3ea459c6ce5263594399880bbc398225b2", - "sha256:a38856a971c602f98472050165cea2cdc97709240373041b69030be15047691f", - "sha256:a401b4598e5d3f4a9a811f3daf42ee2291790c7f9d74b18d75d6e21dda98a1a1", - "sha256:a7647ebdfb9682b7bb97e2a5e7cb6ae735b1c25008a70b906aecca294ee96cf4", - "sha256:aaf63899c94de41fe3cf934601b0f7ccb6b428c6e4eeb80da72c58eab077b19a", - "sha256:b0dac0ff919ba34d4df1b6131f59ce95b08b9065233446be7e459f95554c0dc8", - "sha256:baacc6aee0b2ef6f3d308e197b5d7a81c0e70b06beae1f1fcacffdbd124fe0e3", - "sha256:bf420121d4c8dce6b889f0e8e4ec0ca34b7f40186203f06a946fa0276ba54029", - "sha256:c04a46716adde8d927adb9457bbe39cf473e1e2c2f5d0a16ceb837e5d841ad4f", - "sha256:c0b21078a4b56965e2b12f247467b234734491897e99c1d51cee628da9786959", - "sha256:c1c76a1743432b4b60ab3358c937a3fe1341c828ae6194108a94c69028247f22", - "sha256:c4983bf937209c57240cff65906b18bb35e64ae872da6a0db937d7b4af845dd7", - "sha256:c4fb39a81950ec280984b3a44f5bd12819953dc5fa3a7e6fa7a80db5ee853952", - "sha256:c57921cda3a80d0f2b8aec7e25c8aa14479ea92b5b51b6876d975d925a2ea346", - "sha256:c8063cf17b19661471ecbdb3df1c84f24ad2e389e326ccaf89e3fb2484d8dd7e", - "sha256:ccd16eb18a849fd8dcb23e23380e2f0a354e8daa0c984b8a732d9cfaba3a776d", - "sha256:cd6dbe0238f7743d0efe563ab46294f54f9bc8f4b9bcf57c3c666cc5bc9d1299", - "sha256:d62e51710986674142526ab9f78663ca2b0726066ae26b78b22e0f5e571238dd", - "sha256:db901e2ac34c931d73054d9797383d0f8009991e723dab15109740a63e7f902a", - "sha256:e03b8895a6990c9ab2cdcd0f2fe44088ca1c65ae592b8f795c3294af00a461c3", - "sha256:e1c8a2f4c69e08e89632defbfabec2feb8a8d99edc9f89ce33c4b9e36ab63037", - "sha256:e4b749b9cc6ee664a3300bb3a273c1ca8068c46be705b6c31cf5d276f8628a94", - "sha256:e6a5bf2cba5ae1bb80b154ed68a3cfa2fa00fde979a7f50d6598d3e17d9ac20c", - "sha256:e857a2232ba53ae940d3456f7533ce6ca98b81917d47adc3c7fd55dad8fab858", - "sha256:ee4006268ed33370957f55bf2e6f4d263eaf4dc3cfc473d1d90baff6ed36ce4a", - "sha256:eef9df1eefada2c09a5e7a40991b9fc6ac6ef20b1372abd48d2794a316dc0449", - "sha256:f058f6963fd82eb143c692cecdc89e075fa0828db2e5b291070485390b2f1c9c", - "sha256:f25c229a6ba38a35ae6e25ca1264621cc25d4d38dca2942a7fce0b67a4efe918", - "sha256:f2a1d0fd4242bd8643ce6f98927cf9c04540af6efa92323e9d3124f57727bfc1", - "sha256:f7560358a6811e52e9c4d142d497f1a6e10103d3a6881f18d04dbce3729c0e2c", - "sha256:f779d3ad205f108d14e99bb3859aa7dd8e9c68874617c72354d7ecaec2a054ac", - "sha256:f87f746ee241d30d6ed93969de31e5ffd09a2961a051e60ae6bddde9ec3583aa" - ], - "markers": "python_version >= '3.7'", - "version": "==3.2.0" + "sha256:02673e456dc5ab13659f85196c534dc596d4ef260e4d86e856c3b2773ce09843", + "sha256:02af06682e3590ab952599fbadac535ede5d60d78848e555aa58d0c0abbde786", + "sha256:03680bb39035fbcffe828eae9c3f8afc0428c91d38e7d61aa992ef7a59fb120e", + "sha256:0570d21da019941634a531444364f2482e8db0b3425fcd5ac0c36565a64142c8", + "sha256:09c77f964f351a7369cc343911e0df63e762e42bac24cd7d18525961c81754f4", + "sha256:0d3d5b7db9ed8a2b11a774db2bbea7ba1884430a205dbd54a32d61d7c2a190fa", + "sha256:1063da2c85b95f2d1a430f1c33b55c9c17ffaf5e612e10aeaad641c55a9e2b9d", + "sha256:12ebea541c44fdc88ccb794a13fe861cc5e35d64ed689513a5c03d05b53b7c82", + "sha256:153e7b6e724761741e0974fc4dcd406d35ba70b92bfe3fedcb497226c93b9da7", + "sha256:15b26ddf78d57f1d143bdf32e820fd8935d36abe8a25eb9ec0b5a71c82eb3895", + "sha256:1872d01ac8c618a8da634e232f24793883d6e456a66593135aeafe3784b0848d", + "sha256:187d18082694a29005ba2944c882344b6748d5be69e3a89bf3cc9d878e548d5a", + "sha256:1b2919306936ac6efb3aed1fbf81039f7087ddadb3160882a57ee2ff74fd2382", + "sha256:232ac332403e37e4a03d209a3f92ed9071f7d3dbda70e2a5e9cff1c4ba9f0678", + "sha256:23e8565ab7ff33218530bc817922fae827420f143479b753104ab801145b1d5b", + "sha256:24817cb02cbef7cd499f7c9a2735286b4782bd47a5b3516a0e84c50eab44b98e", + "sha256:249c6470a2b60935bafd1d1d13cd613f8cd8388d53461c67397ee6a0f5dce741", + "sha256:24a91a981f185721542a0b7c92e9054b7ab4fea0508a795846bc5b0abf8118d4", + "sha256:2502dd2a736c879c0f0d3e2161e74d9907231e25d35794584b1ca5284e43f596", + "sha256:250c9eb0f4600361dd80d46112213dff2286231d92d3e52af1e5a6083d10cad9", + "sha256:278c296c6f96fa686d74eb449ea1697f3c03dc28b75f873b65b5201806346a69", + "sha256:2935ffc78db9645cb2086c2f8f4cfd23d9b73cc0dc80334bc30aac6f03f68f8c", + "sha256:2f4a0033ce9a76e391542c182f0d48d084855b5fcba5010f707c8e8c34663d77", + "sha256:30a85aed0b864ac88309b7d94be09f6046c834ef60762a8833b660139cfbad13", + "sha256:380c4bde80bce25c6e4f77b19386f5ec9db230df9f2f2ac1e5ad7af2caa70459", + "sha256:3ae38d325b512f63f8da31f826e6cb6c367336f95e418137286ba362925c877e", + "sha256:3b447982ad46348c02cb90d230b75ac34e9886273df3a93eec0539308a6296d7", + "sha256:3debd1150027933210c2fc321527c2299118aa929c2f5a0a80ab6953e3bd1908", + "sha256:4162918ef3098851fcd8a628bf9b6a98d10c380725df9e04caf5ca6dd48c847a", + "sha256:468d2a840567b13a590e67dd276c570f8de00ed767ecc611994c301d0f8c014f", + "sha256:4cc152c5dd831641e995764f9f0b6589519f6f5123258ccaca8c6d34572fefa8", + "sha256:542da1178c1c6af8873e143910e2269add130a299c9106eef2594e15dae5e482", + "sha256:557b21a44ceac6c6b9773bc65aa1b4cc3e248a5ad2f5b914b91579a32e22204d", + "sha256:5707a746c6083a3a74b46b3a631d78d129edab06195a92a8ece755aac25a3f3d", + "sha256:588245972aca710b5b68802c8cad9edaa98589b1b42ad2b53accd6910dad3545", + "sha256:5adf257bd58c1b8632046bbe43ee38c04e1038e9d37de9c57a94d6bd6ce5da34", + "sha256:619d1c96099be5823db34fe89e2582b336b5b074a7f47f819d6b3a57ff7bdb86", + "sha256:63563193aec44bce707e0c5ca64ff69fa72ed7cf34ce6e11d5127555756fd2f6", + "sha256:67b8cc9574bb518ec76dc8e705d4c39ae78bb96237cb533edac149352c1f39fe", + "sha256:6a685067d05e46641d5d1623d7c7fdf15a357546cbb2f71b0ebde91b175ffc3e", + "sha256:70f1d09c0d7748b73290b29219e854b3207aea922f839437870d8cc2168e31cc", + "sha256:750b446b2ffce1739e8578576092179160f6d26bd5e23eb1789c4d64d5af7dc7", + "sha256:7966951325782121e67c81299a031f4c115615e68046f79b85856b86ebffc4cd", + "sha256:7b8b8bf1189b3ba9b8de5c8db4d541b406611a71a955bbbd7385bbc45fcb786c", + "sha256:7f5d10bae5d78e4551b7be7a9b29643a95aded9d0f602aa2ba584f0388e7a557", + "sha256:805dfea4ca10411a5296bcc75638017215a93ffb584c9e344731eef0dcfb026a", + "sha256:81bf654678e575403736b85ba3a7867e31c2c30a69bc57fe88e3ace52fb17b89", + "sha256:82eb849f085624f6a607538ee7b83a6d8126df6d2f7d3b319cb837b289123078", + "sha256:85a32721ddde63c9df9ebb0d2045b9691d9750cb139c161c80e500d210f5e26e", + "sha256:86d1f65ac145e2c9ed71d8ffb1905e9bba3a91ae29ba55b4c46ae6fc31d7c0d4", + "sha256:86f63face3a527284f7bb8a9d4f78988e3c06823f7bea2bd6f0e0e9298ca0403", + "sha256:8eaf82f0eccd1505cf39a45a6bd0a8cf1c70dcfc30dba338207a969d91b965c0", + "sha256:93aa7eef6ee71c629b51ef873991d6911b906d7312c6e8e99790c0f33c576f89", + "sha256:96c2b49eb6a72c0e4991d62406e365d87067ca14c1a729a870d22354e6f68115", + "sha256:9cf3126b85822c4e53aa28c7ec9869b924d6fcfb76e77a45c44b83d91afd74f9", + "sha256:9fe359b2e3a7729010060fbca442ca225280c16e923b37db0e955ac2a2b72a05", + "sha256:a0ac5e7015a5920cfce654c06618ec40c33e12801711da6b4258af59a8eff00a", + "sha256:a3f93dab657839dfa61025056606600a11d0b696d79386f974e459a3fbc568ec", + "sha256:a4b71f4d1765639372a3b32d2638197f5cd5221b19531f9245fcc9ee62d38f56", + "sha256:aae32c93e0f64469f74ccc730a7cb21c7610af3a775157e50bbd38f816536b38", + "sha256:aaf7b34c5bc56b38c931a54f7952f1ff0ae77a2e82496583b247f7c969eb1479", + "sha256:abecce40dfebbfa6abf8e324e1860092eeca6f7375c8c4e655a8afb61af58f2c", + "sha256:abf0d9f45ea5fb95051c8bfe43cb40cda383772f7e5023a83cc481ca2604d74e", + "sha256:ac71b2977fb90c35d41c9453116e283fac47bb9096ad917b8819ca8b943abecd", + "sha256:ada214c6fa40f8d800e575de6b91a40d0548139e5dc457d2ebb61470abf50186", + "sha256:b09719a17a2301178fac4470d54b1680b18a5048b481cb8890e1ef820cb80455", + "sha256:b1121de0e9d6e6ca08289583d7491e7fcb18a439305b34a30b20d8215922d43c", + "sha256:b3b2316b25644b23b54a6f6401074cebcecd1244c0b8e80111c9a3f1c8e83d65", + "sha256:b3d9b48ee6e3967b7901c052b670c7dda6deb812c309439adaffdec55c6d7b78", + "sha256:b5bcf60a228acae568e9911f410f9d9e0d43197d030ae5799e20dca8df588287", + "sha256:b8f3307af845803fb0b060ab76cf6dd3a13adc15b6b451f54281d25911eb92df", + "sha256:c2af80fb58f0f24b3f3adcb9148e6203fa67dd3f61c4af146ecad033024dde43", + "sha256:c350354efb159b8767a6244c166f66e67506e06c8924ed74669b2c70bc8735b1", + "sha256:c5a74c359b2d47d26cdbbc7845e9662d6b08a1e915eb015d044729e92e7050b7", + "sha256:c71f16da1ed8949774ef79f4a0260d28b83b3a50c6576f8f4f0288d109777989", + "sha256:d47ecf253780c90ee181d4d871cd655a789da937454045b17b5798da9393901a", + "sha256:d7eff0f27edc5afa9e405f7165f85a6d782d308f3b6b9d96016c010597958e63", + "sha256:d97d85fa63f315a8bdaba2af9a6a686e0eceab77b3089af45133252618e70884", + "sha256:db756e48f9c5c607b5e33dd36b1d5872d0422e960145b08ab0ec7fd420e9d649", + "sha256:dc45229747b67ffc441b3de2f3ae5e62877a282ea828a5bdb67883c4ee4a8810", + "sha256:e0fc42822278451bc13a2e8626cf2218ba570f27856b536e00cfa53099724828", + "sha256:e39c7eb31e3f5b1f88caff88bcff1b7f8334975b46f6ac6e9fc725d829bc35d4", + "sha256:e46cd37076971c1040fc8c41273a8b3e2c624ce4f2be3f5dfcb7a430c1d3acc2", + "sha256:e5c1502d4ace69a179305abb3f0bb6141cbe4714bc9b31d427329a95acfc8bdd", + "sha256:edfe077ab09442d4ef3c52cb1f9dab89bff02f4524afc0acf2d46be17dc479f5", + "sha256:effe5406c9bd748a871dbcaf3ac69167c38d72db8c9baf3ff954c344f31c4cbe", + "sha256:f0d1e3732768fecb052d90d62b220af62ead5748ac51ef61e7b32c266cac9293", + "sha256:f5969baeaea61c97efa706b9b107dcba02784b1601c74ac84f2a532ea079403e", + "sha256:f8888e31e3a85943743f8fc15e71536bda1c81d5aa36d014a3c0c44481d7db6e", + "sha256:fc52b79d83a3fe3a360902d3f5d79073a993597d48114c29485e9431092905d8" + ], + "markers": "python_full_version >= '3.7.0'", + "version": "==3.3.0" }, "constantly": { "hashes": [ @@ -245,32 +251,32 @@ }, "cryptography": { "hashes": [ - "sha256:01f1d9e537f9a15b037d5d9ee442b8c22e3ae11ce65ea1f3316a41c78756b711", - "sha256:079347de771f9282fbfe0e0236c716686950c19dee1b76240ab09ce1624d76d7", - "sha256:182be4171f9332b6741ee818ec27daff9fb00349f706629f5cbf417bd50e66fd", - "sha256:192255f539d7a89f2102d07d7375b1e0a81f7478925b3bc2e0549ebf739dae0e", - "sha256:2a034bf7d9ca894720f2ec1d8b7b5832d7e363571828037f9e0c4f18c1b58a58", - "sha256:342f3767e25876751e14f8459ad85e77e660537ca0a066e10e75df9c9e9099f0", - "sha256:439c3cc4c0d42fa999b83ded80a9a1fb54d53c58d6e59234cfe97f241e6c781d", - "sha256:49c3222bb8f8e800aead2e376cbef687bc9e3cb9b58b29a261210456a7783d83", - "sha256:674b669d5daa64206c38e507808aae49904c988fa0a71c935e7006a3e1e83831", - "sha256:7a9a3bced53b7f09da251685224d6a260c3cb291768f54954e28f03ef14e3766", - "sha256:7af244b012711a26196450d34f483357e42aeddb04128885d95a69bd8b14b69b", - "sha256:7d230bf856164de164ecb615ccc14c7fc6de6906ddd5b491f3af90d3514c925c", - "sha256:84609ade00a6ec59a89729e87a503c6e36af98ddcd566d5f3be52e29ba993182", - "sha256:9a6673c1828db6270b76b22cc696f40cde9043eb90373da5c2f8f2158957f42f", - "sha256:9b6d717393dbae53d4e52684ef4f022444fc1cce3c48c38cb74fca29e1f08eaa", - "sha256:9c3fe6534d59d071ee82081ca3d71eed3210f76ebd0361798c74abc2bcf347d4", - "sha256:a719399b99377b218dac6cf547b6ec54e6ef20207b6165126a280b0ce97e0d2a", - "sha256:b332cba64d99a70c1e0836902720887fb4529ea49ea7f5462cf6640e095e11d2", - "sha256:d124682c7a23c9764e54ca9ab5b308b14b18eba02722b8659fb238546de83a76", - "sha256:d73f419a56d74fef257955f51b18d046f3506270a5fd2ac5febbfa259d6c0fa5", - "sha256:f0dc40e6f7aa37af01aba07277d3d64d5a03dc66d682097541ec4da03cc140ee", - "sha256:f14ad275364c8b4e525d018f6716537ae7b6d369c094805cae45300847e0894f", - "sha256:f772610fe364372de33d76edcd313636a25684edb94cee53fd790195f5989d14" + "sha256:004b6ccc95943f6a9ad3142cfabcc769d7ee38a3f60fb0dddbfb431f818c3a67", + "sha256:047c4603aeb4bbd8db2756e38f5b8bd7e94318c047cfe4efeb5d715e08b49311", + "sha256:0d9409894f495d465fe6fda92cb70e8323e9648af912d5b9141d616df40a87b8", + "sha256:23a25c09dfd0d9f28da2352503b23e086f8e78096b9fd585d1d14eca01613e13", + "sha256:2ed09183922d66c4ec5fdaa59b4d14e105c084dd0febd27452de8f6f74704143", + "sha256:35c00f637cd0b9d5b6c6bd11b6c3359194a8eba9c46d4e875a3660e3b400005f", + "sha256:37480760ae08065437e6573d14be973112c9e6dcaf5f11d00147ee74f37a3829", + "sha256:3b224890962a2d7b57cf5eeb16ccaafba6083f7b811829f00476309bce2fe0fd", + "sha256:5a0f09cefded00e648a127048119f77bc2b2ec61e736660b5789e638f43cc397", + "sha256:5b72205a360f3b6176485a333256b9bcd48700fc755fef51c8e7e67c4b63e3ac", + "sha256:7e53db173370dea832190870e975a1e09c86a879b613948f09eb49324218c14d", + "sha256:7febc3094125fc126a7f6fb1f420d0da639f3f32cb15c8ff0dc3997c4549f51a", + "sha256:80907d3faa55dc5434a16579952ac6da800935cd98d14dbd62f6f042c7f5e839", + "sha256:86defa8d248c3fa029da68ce61fe735432b047e32179883bdb1e79ed9bb8195e", + "sha256:8ac4f9ead4bbd0bc8ab2d318f97d85147167a488be0e08814a37eb2f439d5cf6", + "sha256:93530900d14c37a46ce3d6c9e6fd35dbe5f5601bf6b3a5c325c7bffc030344d9", + "sha256:9eeb77214afae972a00dee47382d2591abe77bdae166bda672fb1e24702a3860", + "sha256:b5f4dfe950ff0479f1f00eda09c18798d4f49b98f4e2006d644b3301682ebdca", + "sha256:c3391bd8e6de35f6f1140e50aaeb3e2b3d6a9012536ca23ab0d9c35ec18c8a91", + "sha256:c880eba5175f4307129784eca96f4e70b88e57aa3f680aeba3bab0e980b0f37d", + "sha256:cecfefa17042941f94ab54f769c8ce0fe14beff2694e9ac684176a2535bf9714", + "sha256:e40211b4923ba5a6dc9769eab704bdb3fbb58d56c5b336d30996c24fcf12aadb", + "sha256:efc8ad4e6fc4f1752ebfb58aefece8b4e3c4cae940b0994d43649bdfce8d0d4f" ], - "index": "pypi", - "version": "==41.0.2" + "markers": "python_version >= '3.7'", + "version": "==41.0.4" }, "daphne": { "hashes": [ @@ -278,15 +284,17 @@ "sha256:cce9afc8f49a4f15d4270b8cfb0e0fe811b770a5cc795474e97e4da287497666" ], "index": "pypi", + "markers": "python_version >= '3.7'", "version": "==4.0.0" }, "django": { "hashes": [ - "sha256:45a747e1c5b3d6df1b141b1481e193b033fd1fdbda3ff52677dc81afdaacbaed", - "sha256:f7c7852a5ac5a3da5a8d5b35cc6168f31b605971441798dac845f17ca8028039" + "sha256:5e5c1c9548ffb7796b4a8a4782e9a2e5a3df3615259fc1bfd3ebc73b646146c1", + "sha256:b6b2b5cae821077f137dc4dade696a1c2aa292f892eca28fa8d7bfdf2608ddd4" ], "index": "pypi", - "version": "==4.2.3" + "markers": "python_version >= '3.8'", + "version": "==4.2.5" }, "django-apscheduler": { "hashes": [ @@ -298,11 +306,12 @@ }, "django-bootstrap5": { "hashes": [ - "sha256:3a3748b1ce1a2a73c2eca851c2926e7b6e5a317815e4da7ced6b1807c4792b1a", - "sha256:d0b1997528b8a557968cf1ac7ff4c68659897779011a5a42816fb1ad9e57d6ae" + "sha256:21e1956a8a819370decc5d365ce4f4207761cb065afec5a7df8c4c8c49507f2e", + "sha256:ca1bb2f40175ed1c1725f52f249a574bf629b33b5a0af3bec0eab1de6d72836c" ], "index": "pypi", - "version": "==23.1" + "markers": "python_version >= '3.7'", + "version": "==23.3" }, "django-hashid-field": { "hashes": [ @@ -314,11 +323,11 @@ }, "django-tables2": { "hashes": [ - "sha256:e336fdf8899a8fab110550a40cad956064bd4054818e0b972c1893b3e2542168", - "sha256:f6c1623aac188d29aae9cf6b4de3211c96c525e49890654bec3359c181600eb9" + "sha256:04f23c1181d93716c67085a3c324b449180fd0c5162ef4619acb0b2d9a166133", + "sha256:479eed04007cc04bcf764a6fb7a5e3955d94b878ba7f3a4bd4edbd2f7769e08d" ], "index": "pypi", - "version": "==2.5.3" + "version": "==2.6.0" }, "djangorestframework": { "hashes": [ @@ -326,6 +335,7 @@ "sha256:eb63f58c9f218e1a7d064d17a70751f528ed4e1d35547fdade9aaf4cd103fd08" ], "index": "pypi", + "markers": "python_version >= '3.6'", "version": "==3.14.0" }, "hashids": { @@ -363,6 +373,7 @@ "sha256:8440ffe49c4ae81a8df57c1ae1eb4b6bfa7acb830099bfb3e305b383005cc128" ], "index": "pypi", + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", "version": "==1.3.5" }, "mod-wsgi-httpd": { @@ -380,105 +391,101 @@ }, "msgpack": { "hashes": [ - "sha256:06f5174b5f8ed0ed919da0e62cbd4ffde676a374aba4020034da05fab67b9164", - "sha256:0c05a4a96585525916b109bb85f8cb6511db1c6f5b9d9cbcbc940dc6b4be944b", - "sha256:137850656634abddfb88236008339fdaba3178f4751b28f270d2ebe77a563b6c", - "sha256:17358523b85973e5f242ad74aa4712b7ee560715562554aa2134d96e7aa4cbbf", - "sha256:18334484eafc2b1aa47a6d42427da7fa8f2ab3d60b674120bce7a895a0a85bdd", - "sha256:1835c84d65f46900920b3708f5ba829fb19b1096c1800ad60bae8418652a951d", - "sha256:1967f6129fc50a43bfe0951c35acbb729be89a55d849fab7686004da85103f1c", - "sha256:1ab2f3331cb1b54165976a9d976cb251a83183631c88076613c6c780f0d6e45a", - "sha256:1c0f7c47f0087ffda62961d425e4407961a7ffd2aa004c81b9c07d9269512f6e", - "sha256:20a97bf595a232c3ee6d57ddaadd5453d174a52594bf9c21d10407e2a2d9b3bd", - "sha256:20c784e66b613c7f16f632e7b5e8a1651aa5702463d61394671ba07b2fc9e025", - "sha256:266fa4202c0eb94d26822d9bfd7af25d1e2c088927fe8de9033d929dd5ba24c5", - "sha256:28592e20bbb1620848256ebc105fc420436af59515793ed27d5c77a217477705", - "sha256:288e32b47e67f7b171f86b030e527e302c91bd3f40fd9033483f2cacc37f327a", - "sha256:3055b0455e45810820db1f29d900bf39466df96ddca11dfa6d074fa47054376d", - "sha256:332360ff25469c346a1c5e47cbe2a725517919892eda5cfaffe6046656f0b7bb", - "sha256:362d9655cd369b08fda06b6657a303eb7172d5279997abe094512e919cf74b11", - "sha256:366c9a7b9057e1547f4ad51d8facad8b406bab69c7d72c0eb6f529cf76d4b85f", - "sha256:36961b0568c36027c76e2ae3ca1132e35123dcec0706c4b7992683cc26c1320c", - "sha256:379026812e49258016dd84ad79ac8446922234d498058ae1d415f04b522d5b2d", - "sha256:382b2c77589331f2cb80b67cc058c00f225e19827dbc818d700f61513ab47bea", - "sha256:476a8fe8fae289fdf273d6d2a6cb6e35b5a58541693e8f9f019bfe990a51e4ba", - "sha256:48296af57cdb1d885843afd73c4656be5c76c0c6328db3440c9601a98f303d87", - "sha256:4867aa2df9e2a5fa5f76d7d5565d25ec76e84c106b55509e78c1ede0f152659a", - "sha256:4c075728a1095efd0634a7dccb06204919a2f67d1893b6aa8e00497258bf926c", - "sha256:4f837b93669ce4336e24d08286c38761132bc7ab29782727f8557e1eb21b2080", - "sha256:4f8d8b3bf1ff2672567d6b5c725a1b347fe838b912772aa8ae2bf70338d5a198", - "sha256:525228efd79bb831cf6830a732e2e80bc1b05436b086d4264814b4b2955b2fa9", - "sha256:5494ea30d517a3576749cad32fa27f7585c65f5f38309c88c6d137877fa28a5a", - "sha256:55b56a24893105dc52c1253649b60f475f36b3aa0fc66115bffafb624d7cb30b", - "sha256:56a62ec00b636583e5cb6ad313bbed36bb7ead5fa3a3e38938503142c72cba4f", - "sha256:57e1f3528bd95cc44684beda696f74d3aaa8a5e58c816214b9046512240ef437", - "sha256:586d0d636f9a628ddc6a17bfd45aa5b5efaf1606d2b60fa5d87b8986326e933f", - "sha256:5cb47c21a8a65b165ce29f2bec852790cbc04936f502966768e4aae9fa763cb7", - "sha256:6c4c68d87497f66f96d50142a2b73b97972130d93677ce930718f68828b382e2", - "sha256:821c7e677cc6acf0fd3f7ac664c98803827ae6de594a9f99563e48c5a2f27eb0", - "sha256:916723458c25dfb77ff07f4c66aed34e47503b2eb3188b3adbec8d8aa6e00f48", - "sha256:9e6ca5d5699bcd89ae605c150aee83b5321f2115695e741b99618f4856c50898", - "sha256:9f5ae84c5c8a857ec44dc180a8b0cc08238e021f57abdf51a8182e915e6299f0", - "sha256:a2b031c2e9b9af485d5e3c4520f4220d74f4d222a5b8dc8c1a3ab9448ca79c57", - "sha256:a61215eac016f391129a013c9e46f3ab308db5f5ec9f25811e811f96962599a8", - "sha256:a740fa0e4087a734455f0fc3abf5e746004c9da72fbd541e9b113013c8dc3282", - "sha256:a9985b214f33311df47e274eb788a5893a761d025e2b92c723ba4c63936b69b1", - "sha256:ab31e908d8424d55601ad7075e471b7d0140d4d3dd3272daf39c5c19d936bd82", - "sha256:ac9dd47af78cae935901a9a500104e2dea2e253207c924cc95de149606dc43cc", - "sha256:addab7e2e1fcc04bd08e4eb631c2a90960c340e40dfc4a5e24d2ff0d5a3b3edb", - "sha256:b1d46dfe3832660f53b13b925d4e0fa1432b00f5f7210eb3ad3bb9a13c6204a6", - "sha256:b2de4c1c0538dcb7010902a2b97f4e00fc4ddf2c8cda9749af0e594d3b7fa3d7", - "sha256:b5ef2f015b95f912c2fcab19c36814963b5463f1fb9049846994b007962743e9", - "sha256:b72d0698f86e8d9ddf9442bdedec15b71df3598199ba33322d9711a19f08145c", - "sha256:bae7de2026cbfe3782c8b78b0db9cbfc5455e079f1937cb0ab8d133496ac55e1", - "sha256:bf22a83f973b50f9d38e55c6aade04c41ddda19b00c4ebc558930d78eecc64ed", - "sha256:c075544284eadc5cddc70f4757331d99dcbc16b2bbd4849d15f8aae4cf36d31c", - "sha256:c396e2cc213d12ce017b686e0f53497f94f8ba2b24799c25d913d46c08ec422c", - "sha256:cb5aaa8c17760909ec6cb15e744c3ebc2ca8918e727216e79607b7bbce9c8f77", - "sha256:cdc793c50be3f01106245a61b739328f7dccc2c648b501e237f0699fe1395b81", - "sha256:d25dd59bbbbb996eacf7be6b4ad082ed7eacc4e8f3d2df1ba43822da9bfa122a", - "sha256:e42b9594cc3bf4d838d67d6ed62b9e59e201862a25e9a157019e171fbe672dd3", - "sha256:e57916ef1bd0fee4f21c4600e9d1da352d8816b52a599c46460e93a6e9f17086", - "sha256:ed40e926fa2f297e8a653c954b732f125ef97bdd4c889f243182299de27e2aa9", - "sha256:ef8108f8dedf204bb7b42994abf93882da1159728a2d4c5e82012edd92c9da9f", - "sha256:f933bbda5a3ee63b8834179096923b094b76f0c7a73c1cfe8f07ad608c58844b", - "sha256:fe5c63197c55bce6385d9aee16c4d0641684628f63ace85f73571e65ad1c1e8d" + "sha256:04ad6069c86e531682f9e1e71b71c1c3937d6014a7c3e9edd2aa81ad58842862", + "sha256:0bfdd914e55e0d2c9e1526de210f6fe8ffe9705f2b1dfcc4aecc92a4cb4b533d", + "sha256:1dc93e8e4653bdb5910aed79f11e165c85732067614f180f70534f056da97db3", + "sha256:1e2d69948e4132813b8d1131f29f9101bc2c915f26089a6d632001a5c1349672", + "sha256:235a31ec7db685f5c82233bddf9858748b89b8119bf4538d514536c485c15fe0", + "sha256:27dcd6f46a21c18fa5e5deed92a43d4554e3df8d8ca5a47bf0615d6a5f39dbc9", + "sha256:28efb066cde83c479dfe5a48141a53bc7e5f13f785b92ddde336c716663039ee", + "sha256:3476fae43db72bd11f29a5147ae2f3cb22e2f1a91d575ef130d2bf49afd21c46", + "sha256:36e17c4592231a7dbd2ed09027823ab295d2791b3b1efb2aee874b10548b7524", + "sha256:384d779f0d6f1b110eae74cb0659d9aa6ff35aaf547b3955abf2ab4c901c4819", + "sha256:38949d30b11ae5f95c3c91917ee7a6b239f5ec276f271f28638dec9156f82cfc", + "sha256:3967e4ad1aa9da62fd53e346ed17d7b2e922cba5ab93bdd46febcac39be636fc", + "sha256:3e7bf4442b310ff154b7bb9d81eb2c016b7d597e364f97d72b1acc3817a0fdc1", + "sha256:3f0c8c6dfa6605ab8ff0611995ee30d4f9fcff89966cf562733b4008a3d60d82", + "sha256:484ae3240666ad34cfa31eea7b8c6cd2f1fdaae21d73ce2974211df099a95d81", + "sha256:4a7b4f35de6a304b5533c238bee86b670b75b03d31b7797929caa7a624b5dda6", + "sha256:4cb14ce54d9b857be9591ac364cb08dc2d6a5c4318c1182cb1d02274029d590d", + "sha256:4e71bc4416de195d6e9b4ee93ad3f2f6b2ce11d042b4d7a7ee00bbe0358bd0c2", + "sha256:52700dc63a4676669b341ba33520f4d6e43d3ca58d422e22ba66d1736b0a6e4c", + "sha256:572efc93db7a4d27e404501975ca6d2d9775705c2d922390d878fcf768d92c87", + "sha256:576eb384292b139821c41995523654ad82d1916da6a60cff129c715a6223ea84", + "sha256:5b0bf0effb196ed76b7ad883848143427a73c355ae8e569fa538365064188b8e", + "sha256:5b6ccc0c85916998d788b295765ea0e9cb9aac7e4a8ed71d12e7d8ac31c23c95", + "sha256:5ed82f5a7af3697b1c4786053736f24a0efd0a1b8a130d4c7bfee4b9ded0f08f", + "sha256:6d4c80667de2e36970ebf74f42d1088cc9ee7ef5f4e8c35eee1b40eafd33ca5b", + "sha256:730076207cb816138cf1af7f7237b208340a2c5e749707457d70705715c93b93", + "sha256:7687e22a31e976a0e7fc99c2f4d11ca45eff652a81eb8c8085e9609298916dcf", + "sha256:822ea70dc4018c7e6223f13affd1c5c30c0f5c12ac1f96cd8e9949acddb48a61", + "sha256:84b0daf226913133f899ea9b30618722d45feffa67e4fe867b0b5ae83a34060c", + "sha256:85765fdf4b27eb5086f05ac0491090fc76f4f2b28e09d9350c31aac25a5aaff8", + "sha256:8dd178c4c80706546702c59529ffc005681bd6dc2ea234c450661b205445a34d", + "sha256:8f5b234f567cf76ee489502ceb7165c2a5cecec081db2b37e35332b537f8157c", + "sha256:98bbd754a422a0b123c66a4c341de0474cad4a5c10c164ceed6ea090f3563db4", + "sha256:993584fc821c58d5993521bfdcd31a4adf025c7d745bbd4d12ccfecf695af5ba", + "sha256:a40821a89dc373d6427e2b44b572efc36a2778d3f543299e2f24eb1a5de65415", + "sha256:b291f0ee7961a597cbbcc77709374087fa2a9afe7bdb6a40dbbd9b127e79afee", + "sha256:b573a43ef7c368ba4ea06050a957c2a7550f729c31f11dd616d2ac4aba99888d", + "sha256:b610ff0f24e9f11c9ae653c67ff8cc03c075131401b3e5ef4b82570d1728f8a9", + "sha256:bdf38ba2d393c7911ae989c3bbba510ebbcdf4ecbdbfec36272abe350c454075", + "sha256:bfef2bb6ef068827bbd021017a107194956918ab43ce4d6dc945ffa13efbc25f", + "sha256:cab3db8bab4b7e635c1c97270d7a4b2a90c070b33cbc00c99ef3f9be03d3e1f7", + "sha256:cb70766519500281815dfd7a87d3a178acf7ce95390544b8c90587d76b227681", + "sha256:cca1b62fe70d761a282496b96a5e51c44c213e410a964bdffe0928e611368329", + "sha256:ccf9a39706b604d884d2cb1e27fe973bc55f2890c52f38df742bc1d79ab9f5e1", + "sha256:dc43f1ec66eb8440567186ae2f8c447d91e0372d793dfe8c222aec857b81a8cf", + "sha256:dd632777ff3beaaf629f1ab4396caf7ba0bdd075d948a69460d13d44357aca4c", + "sha256:e45ae4927759289c30ccba8d9fdce62bb414977ba158286b5ddaf8df2cddb5c5", + "sha256:e50ebce52f41370707f1e21a59514e3375e3edd6e1832f5e5235237db933c98b", + "sha256:ebbbba226f0a108a7366bf4b59bf0f30a12fd5e75100c630267d94d7f0ad20e5", + "sha256:ec79ff6159dffcc30853b2ad612ed572af86c92b5168aa3fc01a67b0fa40665e", + "sha256:f0936e08e0003f66bfd97e74ee530427707297b0d0361247e9b4f59ab78ddc8b", + "sha256:f26a07a6e877c76a88e3cecac8531908d980d3d5067ff69213653649ec0f60ad", + "sha256:f64e376cd20d3f030190e8c32e1c64582eba56ac6dc7d5b0b49a9d44021b52fd", + "sha256:f6ffbc252eb0d229aeb2f9ad051200668fc3a9aaa8994e49f0cb2ffe2b7867e7", + "sha256:f9a7c509542db4eceed3dcf21ee5267ab565a83555c9b88a8109dcecc4709002", + "sha256:ff1d0899f104f3921d94579a5638847f783c9b04f2d5f229392ca77fba5b82fc" ], "index": "pypi", - "version": "==1.0.5" + "markers": "python_version >= '3.8'", + "version": "==1.0.7" }, "mysqlclient": { "hashes": [ - "sha256:0d1cd3a5a4d28c222fa199002810e8146cffd821410b67851af4cc80aeccd97c", - "sha256:828757e419fb11dd6c5ed2576ec92c3efaa93a0f7c39e263586d1ee779c3d782", - "sha256:996924f3483fd36a34a5812210c69e71dea5a3d5978d01199b78b7f6d485c855", - "sha256:b355c8b5a7d58f2e909acdbb050858390ee1b0e13672ae759e5e784110022994", - "sha256:c1ed71bd6244993b526113cca3df66428609f90e4652f37eb51c33496d478b37", - "sha256:c812b67e90082a840efb82a8978369e6e69fc62ce1bda4ca8f3084a9d862308b", - "sha256:dea88c8d3f5a5d9293dfe7f087c16dd350ceb175f2f6631c9cf4caf3e19b7a96" + "sha256:004fe1d30d2c2ff8072f8ea513bcec235fd9b896f70dad369461d0ad7e570e98", + "sha256:04368445f9c487d8abb7a878e3d23e923e6072c04a6c320f9e0dc8a82efba14e", + "sha256:530ece9995a36cadb6211b9787f0c9e05cdab6702549bdb4236af5e9b535ed6a", + "sha256:5670679ff1be1cc3fef0fa81bf39f0cd70605ba121141050f02743eb878ac114", + "sha256:68837b6bb23170acffb43ae411e47533a560b6360c06dac39aa55700972c93b2", + "sha256:955dba905a7443ce4788c63fdb9f8d688316260cf60b20ff51ac3b1c77616ede", + "sha256:9c6b142836c7dba4f723bf9c93cc46b6e5081d65b2af807f400dda9eb85a16d0" ], "index": "pypi", - "version": "==2.1.1" + "markers": "python_version >= '3.8'", + "version": "==2.2.0" }, "psutil": { "hashes": [ - "sha256:149555f59a69b33f056ba1c4eb22bb7bf24332ce631c44a319cec09f876aaeff", - "sha256:16653106f3b59386ffe10e0bad3bb6299e169d5327d3f187614b1cb8f24cf2e1", - "sha256:3d7f9739eb435d4b1338944abe23f49584bde5395f27487d2ee25ad9a8774a62", - "sha256:3ff89f9b835100a825b14c2808a106b6fdcc4b15483141482a12c725e7f78549", - "sha256:54c0d3d8e0078b7666984e11b12b88af2db11d11249a8ac8920dd5ef68a66e08", - "sha256:54d5b184728298f2ca8567bf83c422b706200bcbbfafdc06718264f9393cfeb7", - "sha256:6001c809253a29599bc0dfd5179d9f8a5779f9dffea1da0f13c53ee568115e1e", - "sha256:68908971daf802203f3d37e78d3f8831b6d1014864d7a85937941bb35f09aefe", - "sha256:6b92c532979bafc2df23ddc785ed116fced1f492ad90a6830cf24f4d1ea27d24", - "sha256:852dd5d9f8a47169fe62fd4a971aa07859476c2ba22c2254d4a1baa4e10b95ad", - "sha256:9120cd39dca5c5e1c54b59a41d205023d436799b1c8c4d3ff71af18535728e94", - "sha256:c1ca331af862803a42677c120aff8a814a804e09832f166f226bfd22b56feee8", - "sha256:efeae04f9516907be44904cc7ce08defb6b665128992a56957abc9b61dca94b7", - "sha256:fd8522436a6ada7b4aad6638662966de0d61d241cb821239b2ae7013d41a43d4" + "sha256:104a5cc0e31baa2bcf67900be36acde157756b9c44017b86b2c049f11957887d", + "sha256:3c6f686f4225553615612f6d9bc21f1c0e305f75d7d8454f9b46e901778e7217", + "sha256:4aef137f3345082a3d3232187aeb4ac4ef959ba3d7c10c33dd73763fbc063da4", + "sha256:5410638e4df39c54d957fc51ce03048acd8e6d60abc0f5107af51e5fb566eb3c", + "sha256:5b9b8cb93f507e8dbaf22af6a2fd0ccbe8244bf30b1baad6b3954e935157ae3f", + "sha256:7a7dd9997128a0d928ed4fb2c2d57e5102bb6089027939f3b722f3a210f9a8da", + "sha256:89518112647f1276b03ca97b65cc7f64ca587b1eb0278383017c2a0dcc26cbe4", + "sha256:8c5f7c5a052d1d567db4ddd231a9d27a74e8e4a9c3f44b1032762bd7b9fdcd42", + "sha256:ab8ed1a1d77c95453db1ae00a3f9c50227ebd955437bcf2a574ba8adbf6a74d5", + "sha256:acf2aef9391710afded549ff602b5887d7a2349831ae4c26be7c807c0a39fac4", + "sha256:b258c0c1c9d145a1d5ceffab1134441c4c5113b2417fafff7315a917a026c3c9", + "sha256:be8929ce4313f9f8146caad4272f6abb8bf99fc6cf59344a3167ecd74f4f203f", + "sha256:c607bb3b57dc779d55e1554846352b4e358c10fff3abf3514a7a6601beebdb30", + "sha256:ea8518d152174e1249c4f2a1c89e3e6065941df2fa13a1ab45327716a23c2b48" ], "index": "pypi", - "version": "==5.9.4" + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", + "version": "==5.9.5" }, "pyasn1": { "hashes": [ @@ -516,22 +523,23 @@ "sha256:f5971a9226b701070a4bf2c38c89e5a3f0d64de8debda981d1db98583009122a" ], "index": "pypi", + "markers": "python_version >= '3.8'", "version": "==1.0.0" }, "pytz": { "hashes": [ - "sha256:1d8ce29db189191fb55338ee6d0387d82ab59f3d00eac103412d64e0ebd0c588", - "sha256:a151b3abb88eda1d4e34a9814df37de2a80e301e68ba0fd856fb9b46bfbbbffb" + "sha256:7b4fddbeb94a1eba4b557da24f19fdf9db575192544270a9101d8509f9f43d7b", + "sha256:ce42d816b81b68506614c11e8937d3aa9e41007ceb50bfdcb0749b921bf646c7" ], - "version": "==2023.3" + "version": "==2023.3.post1" }, "redis": { "hashes": [ - "sha256:585dc516b9eb042a619ef0a39c3d7d55fe81bdb4df09a52c9cdde0d07bf1aa7d", - "sha256:e2b03db868160ee4591de3cb90d40ebb50a90dd302138775937f6a42b7ed183c" + "sha256:0dab495cd5753069d3bc650a0dde8a8f9edde16fc5691b689a566eda58100d0f", + "sha256:ed4802971884ae19d640775ba3b03aa2e7bd5e8fb8dfaed2decce4d0fc48391f" ], "markers": "python_version >= '3.7'", - "version": "==4.6.0" + "version": "==5.0.1" }, "requests": { "hashes": [ @@ -539,6 +547,7 @@ "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1" ], "index": "pypi", + "markers": "python_version >= '3.7'", "version": "==2.31.0" }, "rx": { @@ -546,7 +555,7 @@ "sha256:922c5f4edb3aa1beaa47bf61d65d5380011ff6adcd527f26377d05cb73ed8ec8", "sha256:b657ca2b45aa485da2f7dcfd09fac2e554f7ac51ff3c2f8f2ff962ecd963d91c" ], - "index": "pypi", + "markers": "python_full_version >= '3.6.0'", "version": "==3.2.0" }, "service-identity": { @@ -558,11 +567,11 @@ }, "setuptools": { "hashes": [ - "sha256:11e52c67415a381d10d6b462ced9cfb97066179f0e871399e006c4ab101fc85f", - "sha256:baf1fdb41c6da4cd2eae722e135500da913332ab3f2f5c7d33af9b492acb5235" + "sha256:4ac1475276d2f1c48684874089fefcd83bd7162ddaafb81fac866ba0db282a87", + "sha256:b454a35605876da60632df1a60f736524eb73cc47bbc9f3f1ef1b644de74fd2a" ], - "markers": "python_version >= '3.7'", - "version": "==68.0.0" + "markers": "python_version >= '3.8'", + "version": "==68.2.2" }, "six": { "hashes": [ @@ -585,11 +594,11 @@ "tls" ], "hashes": [ - "sha256:32acbd40a94f5f46e7b42c109bfae2b302250945561783a8b7a059048f2d4d31", - "sha256:86c55f712cc5ab6f6d64e02503352464f0400f66d4f079096d744080afcccbd0" + "sha256:3c73360add17336a622c0d811c2a2ce29866b6e59b1125fd6509b17252098a24", + "sha256:b8bdba145de120ffb36c20e6e071cce984e89fba798611ed0704216fb7f884cd" ], "markers": "python_full_version >= '3.7.1'", - "version": "==22.10.0" + "version": "==23.8.0" }, "txaio": { "hashes": [ @@ -601,11 +610,11 @@ }, "typing-extensions": { "hashes": [ - "sha256:440d5dd3af93b060174bf433bccd69b0babc3b15b1a8dca43789fd7f61514b36", - "sha256:b75ddc264f0ba5615db7ba217daeb99701ad295353c45f9e95963337ceeeffb2" + "sha256:8f92fc8806f9a6b641eaa5318da32b44d401efaac0f6678c9bc448ba3605faa0", + "sha256:df8e4339e9cb77357558cbdbceca33c303714cf861d1eef15e1070055ae8b7ef" ], "markers": "python_version < '3.11'", - "version": "==4.7.1" + "version": "==4.8.0" }, "tzlocal": { "hashes": [ @@ -617,13 +626,47 @@ }, "urllib3": { "hashes": [ - "sha256:8d22f86aae8ef5e410d4f539fde9ce6b2113a001bb4d189e0aed70642d602b11", - "sha256:de7df1803967d2c2a98e4b11bb7d6bd9210474c46e8a0401514e3a42a75ebde4" + "sha256:13abf37382ea2ce6fb744d4dad67838eec857c9f4f57009891805e0b5e123594", + "sha256:ef16afa8ba34a1f989db38e1dbbe0c302e4289a47856990d0682e374563ce35e" + ], + "markers": "python_version >= '3.7'", + "version": "==2.0.5" + }, + "watchdog": { + "hashes": [ + "sha256:0e06ab8858a76e1219e68c7573dfeba9dd1c0219476c5a44d5333b01d7e1743a", + "sha256:13bbbb462ee42ec3c5723e1205be8ced776f05b100e4737518c67c8325cf6100", + "sha256:233b5817932685d39a7896b1090353fc8efc1ef99c9c054e46c8002561252fb8", + "sha256:25f70b4aa53bd743729c7475d7ec41093a580528b100e9a8c5b5efe8899592fc", + "sha256:2b57a1e730af3156d13b7fdddfc23dea6487fceca29fc75c5a868beed29177ae", + "sha256:336adfc6f5cc4e037d52db31194f7581ff744b67382eb6021c868322e32eef41", + "sha256:3aa7f6a12e831ddfe78cdd4f8996af9cf334fd6346531b16cec61c3b3c0d8da0", + "sha256:3ed7c71a9dccfe838c2f0b6314ed0d9b22e77d268c67e015450a29036a81f60f", + "sha256:4c9956d27be0bb08fc5f30d9d0179a855436e655f046d288e2bcc11adfae893c", + "sha256:4d98a320595da7a7c5a18fc48cb633c2e73cda78f93cac2ef42d42bf609a33f9", + "sha256:4f94069eb16657d2c6faada4624c39464f65c05606af50bb7902e036e3219be3", + "sha256:5113334cf8cf0ac8cd45e1f8309a603291b614191c9add34d33075727a967709", + "sha256:51f90f73b4697bac9c9a78394c3acbbd331ccd3655c11be1a15ae6fe289a8c83", + "sha256:5d9f3a10e02d7371cd929b5d8f11e87d4bad890212ed3901f9b4d68767bee759", + "sha256:7ade88d0d778b1b222adebcc0927428f883db07017618a5e684fd03b83342bd9", + "sha256:7c5f84b5194c24dd573fa6472685b2a27cc5a17fe5f7b6fd40345378ca6812e3", + "sha256:7e447d172af52ad204d19982739aa2346245cc5ba6f579d16dac4bfec226d2e7", + "sha256:8ae9cda41fa114e28faf86cb137d751a17ffd0316d1c34ccf2235e8a84365c7f", + "sha256:8f3ceecd20d71067c7fd4c9e832d4e22584318983cabc013dbf3f70ea95de346", + "sha256:9fac43a7466eb73e64a9940ac9ed6369baa39b3bf221ae23493a9ec4d0022674", + "sha256:a70a8dcde91be523c35b2bf96196edc5730edb347e374c7de7cd20c43ed95397", + "sha256:adfdeab2da79ea2f76f87eb42a3ab1966a5313e5a69a0213a3cc06ef692b0e96", + "sha256:ba07e92756c97e3aca0912b5cbc4e5ad802f4557212788e72a72a47ff376950d", + "sha256:c07253088265c363d1ddf4b3cdb808d59a0468ecd017770ed716991620b8f77a", + "sha256:c9d8c8ec7efb887333cf71e328e39cffbf771d8f8f95d308ea4125bf5f90ba64", + "sha256:d00e6be486affb5781468457b21a6cbe848c33ef43f9ea4a73b4882e5f188a44", + "sha256:d429c2430c93b7903914e4db9a966c7f2b068dd2ebdd2fa9b9ce094c7d459f33" ], + "index": "pypi", "markers": "python_version >= '3.7'", - "version": "==2.0.4" + "version": "==3.0.0" }, - "zope.interface": { + "zope-interface": { "hashes": [ "sha256:042f2381118b093714081fd82c98e3b189b68db38ee7d35b63c327c470ef8373", "sha256:0ec9653825f837fbddc4e4b603d90269b501486c11800d7c761eee7ce46d1bbb", @@ -663,11 +706,11 @@ "develop": { "astroid": { "hashes": [ - "sha256:389656ca57b6108f939cf5d2f9a2a825a3be50ba9d589670f393236e0a03b91c", - "sha256:903f024859b7c7687d7a7f3a3f73b17301f8e42dfd9cc9df9d4418172d3e2dbd" + "sha256:1aa149fc5c6589e3d0ece885b4491acd80af4f087baafa3fb5203b113e68cd3c", + "sha256:6c107453dffee9055899705de3c9ead36e74119cee151e5a9aaf7f0b0e020a6a" ], "markers": "python_full_version >= '3.7.2'", - "version": "==2.15.6" + "version": "==2.15.8" }, "attrs": { "hashes": [ @@ -682,16 +725,16 @@ "sha256:539cc1d13202e33ca466e88b2807e29f4c13049d6d87031a3c110744495cb082", "sha256:92d6037539857d8206b8f6ae472e8b77db8058fec5937a1ef3f54304089edbb9" ], - "markers": "python_full_version >= '3.6.0'", + "markers": "python_version >= '3.6'", "version": "==2023.7.22" }, "click": { "hashes": [ - "sha256:48ee849951919527a045bfe3bf7baa8a959c423134e1a5b98c05c20ba75a1cbd", - "sha256:fa244bb30b3b5ee2cae3da8f55c9e5e0c0e86093306301fb418eb9dc40fbded5" + "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28", + "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de" ], "markers": "python_version >= '3.7'", - "version": "==8.1.6" + "version": "==8.1.7" }, "colorama": { "hashes": [ @@ -717,11 +760,12 @@ }, "djlint": { "hashes": [ - "sha256:0817bbc69df234a4a6208685c6ff2d521f64a330eb1f7c0a052025233a21971c", - "sha256:bed6fa1c97aa88d4be23d9234de0c5eba71a908871d34eb9704f1df286a80335" + "sha256:60b4f4ca99fd83106603bdd466f35314fda33776f3a6e70ea9d674da9d0ad053", + "sha256:bdc26cc607dee8b46e262654eb0fbac7862c34d68172c8adc25a0b56fc7d8173" ], "index": "pypi", - "version": "==1.19.16" + "markers": "python_full_version >= '3.8.0' and python_full_version < '4.0.0'", + "version": "==1.34.0" }, "editorconfig": { "hashes": [ @@ -732,11 +776,11 @@ }, "exceptiongroup": { "hashes": [ - "sha256:12c3e887d6485d16943a309616de20ae5582633e0a2eda17f4e10fd61c1e8af5", - "sha256:e346e69d186172ca7cf029c8c1d16235aa0e04035e5750b4b95039e65204328f" + "sha256:097acd85d473d75af5bb98e41b61ff7fe35efe6675e4f9370ec6ec5126d160e9", + "sha256:343280667a4585d195ca1cf9cef84a4e178c4b6cf2274caef9859782b567d5e3" ], "markers": "python_version < '3.11'", - "version": "==1.1.2" + "version": "==1.1.3" }, "h11": { "hashes": [ @@ -784,6 +828,13 @@ ], "version": "==1.14.9" }, + "json5": { + "hashes": [ + "sha256:740c7f1b9e584a468dbb2939d8d458db3427f2c93ae2139d05f47e453eae964f", + "sha256:9ed66c3a6ca3510a976a9ef9b8c0787de24802724ab1860bc0153c7fdd589b02" + ], + "version": "==0.9.14" + }, "lazy-object-proxy": { "hashes": [ "sha256:09763491ce220c0299688940f8dc2c5d05fd1f45af1e42e636b2e8b2303e4382", @@ -844,35 +895,36 @@ }, "pathspec": { "hashes": [ - "sha256:2798de800fa92780e33acca925945e9a19a133b715067cf165b8866c15a31687", - "sha256:d8af70af76652554bd134c22b3e8a1cc46ed7d91edcdd721ef1a0c51a84a5293" + "sha256:1d6ed233af05e679efb96b1851550ea95bbb64b7c490b0f5aa52996c11e92a20", + "sha256:e0d8d0ac2f12da61956eb2306b69f9469b42f4deb0f3cb6ed47b9cce9996ced3" ], "markers": "python_version >= '3.7'", - "version": "==0.11.1" + "version": "==0.11.2" }, "platformdirs": { "hashes": [ - "sha256:1b42b450ad933e981d56e59f1b97495428c9bd60698baab9f3eb3d00d5822421", - "sha256:ad8291ae0ae5072f66c16945166cb11c63394c7a3ad1b1bc9828ca3162da8c2f" + "sha256:b45696dab2d7cc691a3226759c0d3b00c47c8b6e293d96f6436f733303f77f6d", + "sha256:d7c24979f292f916dc9cbf8648319032f551ea8c49a4c9bf2fb556a02070ec1d" ], "markers": "python_version >= '3.7'", - "version": "==3.9.1" + "version": "==3.10.0" }, "pycodestyle": { "hashes": [ - "sha256:347187bdb476329d98f695c213d7295a846d1152ff4fe9bacb8a9590b8ee7053", - "sha256:8a4eaf0d0495c7395bdab3589ac2db602797d76207242c17d470186815706610" + "sha256:259bcc17857d8a8b3b4a2327324b79e5f020a13c16074670f9c8c8f872ea76d0", + "sha256:5d1013ba8dc7895b548be5afb05740ca82454fd899971563d2ef625d090326f8" ], "index": "pypi", - "version": "==2.10.0" + "markers": "python_version >= '3.8'", + "version": "==2.11.0" }, "pylint": { "hashes": [ - "sha256:5dcf1d9e19f41f38e4e85d10f511e5b9c35e1aa74251bf95cdd8cb23584e2db1", - "sha256:7a1145fb08c251bdb5cca11739722ce64a63db479283d10ce718b2460e54123c" + "sha256:27a8d4c7ddc8c2f8c18aa0050148f89ffc09838142193fdbe98f172781a3ff87", + "sha256:f4fcac7ae74cfe36bc8451e931d8438e4a476c20314b1101c458ad0f05191fad" ], "markers": "python_full_version >= '3.7.2'", - "version": "==2.17.4" + "version": "==2.17.7" }, "pylint-django": { "hashes": [ @@ -900,7 +952,9 @@ }, "pyyaml": { "hashes": [ + "sha256:04ac92ad1925b2cff1db0cfebffb6ffc43457495c9b3c39d3fcae417d7125dc5", "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc", + "sha256:0d3304d8c0adc42be59c5f8a4d9e3d7379e6955ad754aa9d6ab7a398b59dd1df", "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741", "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206", "sha256:18aeb1bf9a78867dc38b259769503436b7c72f7a1f1f4c93ff9a17de54319b27", @@ -908,7 +962,10 @@ "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62", "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98", "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696", + "sha256:326c013efe8048858a6d312ddd31d56e468118ad4cdeda36c719bf5bb6192290", + "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9", "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d", + "sha256:49a183be227561de579b4a36efbb21b3eab9651dd81b1858589f796549873dd6", "sha256:4fb147e7a67ef577a588a0e2c17b6db51dda102c71de36f8549b6816a96e1867", "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47", "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486", @@ -916,9 +973,12 @@ "sha256:596106435fa6ad000c2991a98fa58eeb8656ef2325d7e158344fb33864ed87e3", "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007", "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938", + "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0", "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c", "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735", "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d", + "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28", + "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4", "sha256:9046c58c4395dff28dd494285c82ba00b546adfc7ef001486fbf0324bc174fba", "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8", "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5", @@ -933,7 +993,9 @@ "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43", "sha256:c8098ddcc2a85b61647b2590f825f3db38891662cfc2fc776415143f599bb859", "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673", + "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54", "sha256:d858aa552c999bc8a8d57426ed01e40bef403cd8ccdd0fc5f6f04a00414cac2a", + "sha256:e7d73685e87afe9f3b36c799222440d6cf362062f78be1013661b00c5c6f678b", "sha256:f003ed9ad21d6a4713f0a9b5a7a0a79e08dd0f221aff4525a2be4c346ee60aab", "sha256:f22ac1c3cac4dbc50079e965eba2c1058622631e526bd9afd45fedd49ba781fa", "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c", @@ -946,105 +1008,106 @@ }, "regex": { "hashes": [ - "sha256:052b670fafbe30966bbe5d025e90b2a491f85dfe5b2583a163b5e60a85a321ad", - "sha256:0653d012b3bf45f194e5e6a41df9258811ac8fc395579fa82958a8b76286bea4", - "sha256:0a069c8483466806ab94ea9068c34b200b8bfc66b6762f45a831c4baaa9e8cdd", - "sha256:0cf0da36a212978be2c2e2e2d04bdff46f850108fccc1851332bcae51c8907cc", - "sha256:131d4be09bea7ce2577f9623e415cab287a3c8e0624f778c1d955ec7c281bd4d", - "sha256:144486e029793a733e43b2e37df16a16df4ceb62102636ff3db6033994711066", - "sha256:1ddf14031a3882f684b8642cb74eea3af93a2be68893901b2b387c5fd92a03ec", - "sha256:1eba476b1b242620c266edf6325b443a2e22b633217a9835a52d8da2b5c051f9", - "sha256:20f61c9944f0be2dc2b75689ba409938c14876c19d02f7585af4460b6a21403e", - "sha256:22960019a842777a9fa5134c2364efaed5fbf9610ddc5c904bd3a400973b0eb8", - "sha256:22e7ebc231d28393dfdc19b185d97e14a0f178bedd78e85aad660e93b646604e", - "sha256:23cbb932cc53a86ebde0fb72e7e645f9a5eec1a5af7aa9ce333e46286caef783", - "sha256:29c04741b9ae13d1e94cf93fca257730b97ce6ea64cfe1eba11cf9ac4e85afb6", - "sha256:2bde29cc44fa81c0a0c8686992c3080b37c488df167a371500b2a43ce9f026d1", - "sha256:2cdc55ca07b4e70dda898d2ab7150ecf17c990076d3acd7a5f3b25cb23a69f1c", - "sha256:370f6e97d02bf2dd20d7468ce4f38e173a124e769762d00beadec3bc2f4b3bc4", - "sha256:395161bbdbd04a8333b9ff9763a05e9ceb4fe210e3c7690f5e68cedd3d65d8e1", - "sha256:44136355e2f5e06bf6b23d337a75386371ba742ffa771440b85bed367c1318d1", - "sha256:44a6c2f6374e0033873e9ed577a54a3602b4f609867794c1a3ebba65e4c93ee7", - "sha256:4919899577ba37f505aaebdf6e7dc812d55e8f097331312db7f1aab18767cce8", - "sha256:4b4b1fe58cd102d75ef0552cf17242705ce0759f9695334a56644ad2d83903fe", - "sha256:4bdd56ee719a8f751cf5a593476a441c4e56c9b64dc1f0f30902858c4ef8771d", - "sha256:4bf41b8b0a80708f7e0384519795e80dcb44d7199a35d52c15cc674d10b3081b", - "sha256:4cac3405d8dda8bc6ed499557625585544dd5cbf32072dcc72b5a176cb1271c8", - "sha256:4fe7fda2fe7c8890d454f2cbc91d6c01baf206fbc96d89a80241a02985118c0c", - "sha256:50921c140561d3db2ab9f5b11c5184846cde686bb5a9dc64cae442926e86f3af", - "sha256:5217c25229b6a85049416a5c1e6451e9060a1edcf988641e309dbe3ab26d3e49", - "sha256:5352bea8a8f84b89d45ccc503f390a6be77917932b1c98c4cdc3565137acc714", - "sha256:542e3e306d1669b25936b64917285cdffcd4f5c6f0247636fec037187bd93542", - "sha256:543883e3496c8b6d58bd036c99486c3c8387c2fc01f7a342b760c1ea3158a318", - "sha256:586b36ebda81e6c1a9c5a5d0bfdc236399ba6595e1397842fd4a45648c30f35e", - "sha256:597f899f4ed42a38df7b0e46714880fb4e19a25c2f66e5c908805466721760f5", - "sha256:5a260758454580f11dd8743fa98319bb046037dfab4f7828008909d0aa5292bc", - "sha256:5aefb84a301327ad115e9d346c8e2760009131d9d4b4c6b213648d02e2abe144", - "sha256:5e6a5567078b3eaed93558842346c9d678e116ab0135e22eb72db8325e90b453", - "sha256:5ff525698de226c0ca743bfa71fc6b378cda2ddcf0d22d7c37b1cc925c9650a5", - "sha256:61edbca89aa3f5ef7ecac8c23d975fe7261c12665f1d90a6b1af527bba86ce61", - "sha256:659175b2144d199560d99a8d13b2228b85e6019b6e09e556209dfb8c37b78a11", - "sha256:6a9a19bea8495bb419dc5d38c4519567781cd8d571c72efc6aa959473d10221a", - "sha256:6b30bddd61d2a3261f025ad0f9ee2586988c6a00c780a2fb0a92cea2aa702c54", - "sha256:6ffd55b5aedc6f25fd8d9f905c9376ca44fcf768673ffb9d160dd6f409bfda73", - "sha256:702d8fc6f25bbf412ee706bd73019da5e44a8400861dfff7ff31eb5b4a1276dc", - "sha256:74bcab50a13960f2a610cdcd066e25f1fd59e23b69637c92ad470784a51b1347", - "sha256:75f591b2055523fc02a4bbe598aa867df9e953255f0b7f7715d2a36a9c30065c", - "sha256:763b64853b0a8f4f9cfb41a76a4a85a9bcda7fdda5cb057016e7706fde928e66", - "sha256:76c598ca73ec73a2f568e2a72ba46c3b6c8690ad9a07092b18e48ceb936e9f0c", - "sha256:78d680ef3e4d405f36f0d6d1ea54e740366f061645930072d39bca16a10d8c93", - "sha256:7b280948d00bd3973c1998f92e22aa3ecb76682e3a4255f33e1020bd32adf443", - "sha256:7db345956ecce0c99b97b042b4ca7326feeec6b75facd8390af73b18e2650ffc", - "sha256:7dbdce0c534bbf52274b94768b3498abdf675a691fec5f751b6057b3030f34c1", - "sha256:7ef6b5942e6bfc5706301a18a62300c60db9af7f6368042227ccb7eeb22d0892", - "sha256:7f5a3ffc731494f1a57bd91c47dc483a1e10048131ffb52d901bfe2beb6102e8", - "sha256:8a45b6514861916c429e6059a55cf7db74670eaed2052a648e3e4d04f070e001", - "sha256:8ad241da7fac963d7573cc67a064c57c58766b62a9a20c452ca1f21050868dfa", - "sha256:8b0886885f7323beea6f552c28bff62cbe0983b9fbb94126531693ea6c5ebb90", - "sha256:8ca88da1bd78990b536c4a7765f719803eb4f8f9971cc22d6ca965c10a7f2c4c", - "sha256:8e0caeff18b96ea90fc0eb6e3bdb2b10ab5b01a95128dfeccb64a7238decf5f0", - "sha256:957403a978e10fb3ca42572a23e6f7badff39aa1ce2f4ade68ee452dc6807692", - "sha256:9af69f6746120998cd9c355e9c3c6aec7dff70d47247188feb4f829502be8ab4", - "sha256:9c94f7cc91ab16b36ba5ce476f1904c91d6c92441f01cd61a8e2729442d6fcf5", - "sha256:a37d51fa9a00d265cf73f3de3930fa9c41548177ba4f0faf76e61d512c774690", - "sha256:a3a98921da9a1bf8457aeee6a551948a83601689e5ecdd736894ea9bbec77e83", - "sha256:a3c1ebd4ed8e76e886507c9eddb1a891673686c813adf889b864a17fafcf6d66", - "sha256:a5f9505efd574d1e5b4a76ac9dd92a12acb2b309551e9aa874c13c11caefbe4f", - "sha256:a8ff454ef0bb061e37df03557afda9d785c905dab15584860f982e88be73015f", - "sha256:a9d0b68ac1743964755ae2d89772c7e6fb0118acd4d0b7464eaf3921c6b49dd4", - "sha256:aa62a07ac93b7cb6b7d0389d8ef57ffc321d78f60c037b19dfa78d6b17c928ee", - "sha256:ac741bf78b9bb432e2d314439275235f41656e189856b11fb4e774d9f7246d81", - "sha256:ae1e96785696b543394a4e3f15f3f225d44f3c55dafe3f206493031419fedf95", - "sha256:b683e5fd7f74fb66e89a1ed16076dbab3f8e9f34c18b1979ded614fe10cdc4d9", - "sha256:b7a8b43ee64ca8f4befa2bea4083f7c52c92864d8518244bfa6e88c751fa8fff", - "sha256:b8e38472739028e5f2c3a4aded0ab7eadc447f0d84f310c7a8bb697ec417229e", - "sha256:bfff48c7bd23c6e2aec6454aaf6edc44444b229e94743b34bdcdda2e35126cf5", - "sha256:c14b63c9d7bab795d17392c7c1f9aaabbffd4cf4387725a0ac69109fb3b550c6", - "sha256:c27cc1e4b197092e50ddbf0118c788d9977f3f8f35bfbbd3e76c1846a3443df7", - "sha256:c28d3309ebd6d6b2cf82969b5179bed5fefe6142c70f354ece94324fa11bf6a1", - "sha256:c670f4773f2f6f1957ff8a3962c7dd12e4be54d05839b216cb7fd70b5a1df394", - "sha256:ce6910b56b700bea7be82c54ddf2e0ed792a577dfaa4a76b9af07d550af435c6", - "sha256:d0213671691e341f6849bf33cd9fad21f7b1cb88b89e024f33370733fec58742", - "sha256:d03fe67b2325cb3f09be029fd5da8df9e6974f0cde2c2ac6a79d2634e791dd57", - "sha256:d0e5af9a9effb88535a472e19169e09ce750c3d442fb222254a276d77808620b", - "sha256:d243b36fbf3d73c25e48014961e83c19c9cc92530516ce3c43050ea6276a2ab7", - "sha256:d26166acf62f731f50bdd885b04b38828436d74e8e362bfcb8df221d868b5d9b", - "sha256:d403d781b0e06d2922435ce3b8d2376579f0c217ae491e273bab8d092727d244", - "sha256:d8716f82502997b3d0895d1c64c3b834181b1eaca28f3f6336a71777e437c2af", - "sha256:e4f781ffedd17b0b834c8731b75cce2639d5a8afe961c1e58ee7f1f20b3af185", - "sha256:e613a98ead2005c4ce037c7b061f2409a1a4e45099edb0ef3200ee26ed2a69a8", - "sha256:ef4163770525257876f10e8ece1cf25b71468316f61451ded1a6f44273eedeb5" + "sha256:0085da0f6c6393428bf0d9c08d8b1874d805bb55e17cb1dfa5ddb7cfb11140bf", + "sha256:06c57e14ac723b04458df5956cfb7e2d9caa6e9d353c0b4c7d5d54fcb1325c46", + "sha256:09b7f4c66aa9d1522b06e31a54f15581c37286237208df1345108fcf4e050c18", + "sha256:0c59122ceccb905a941fb23b087b8eafc5290bf983ebcb14d2301febcbe199c7", + "sha256:1005c60ed7037be0d9dea1f9c53cc42f836188227366370867222bda4c3c6bd7", + "sha256:14898830f0a0eb67cae2bbbc787c1a7d6e34ecc06fbd39d3af5fe29a4468e2c9", + "sha256:14dc6f2d88192a67d708341f3085df6a4f5a0c7b03dec08d763ca2cd86e9f559", + "sha256:1e7d84d64c84ad97bf06f3c8cb5e48941f135ace28f450d86af6b6512f1c9a71", + "sha256:2162ae2eb8b079622176a81b65d486ba50b888271302190870b8cc488587d280", + "sha256:22283c769a7b01c8ac355d5be0715bf6929b6267619505e289f792b01304d898", + "sha256:239c3c2a339d3b3ddd51c2daef10874410917cd2b998f043c13e2084cb191684", + "sha256:293352710172239bf579c90a9864d0df57340b6fd21272345222fb6371bf82b3", + "sha256:2ae54a338191e1356253e7883d9d19f8679b6143703086245fb14d1f20196be9", + "sha256:2e73e5243af12d9cd6a9d6a45a43570dbe2e5b1cdfc862f5ae2b031e44dd95a8", + "sha256:2e9216e0d2cdce7dbc9be48cb3eacb962740a09b011a116fd7af8c832ab116ca", + "sha256:3026cbcf11d79095a32d9a13bbc572a458727bd5b1ca332df4a79faecd45281c", + "sha256:3611576aff55918af2697410ff0293d6071b7e00f4b09e005d614686ac4cd57c", + "sha256:3ae646c35cb9f820491760ac62c25b6d6b496757fda2d51be429e0e7b67ae0ab", + "sha256:3b8e6ea6be6d64104d8e9afc34c151926f8182f84e7ac290a93925c0db004bfd", + "sha256:3d370ff652323c5307d9c8e4c62efd1956fb08051b0e9210212bc51168b4ff56", + "sha256:3f7454aa427b8ab9101f3787eb178057c5250478e39b99540cfc2b889c7d0586", + "sha256:40f029d73b10fac448c73d6eb33d57b34607f40116e9f6e9f0d32e9229b147d7", + "sha256:423adfa872b4908843ac3e7a30f957f5d5282944b81ca0a3b8a7ccbbfaa06103", + "sha256:4873ef92e03a4309b3ccd8281454801b291b689f6ad45ef8c3658b6fa761d7ac", + "sha256:48c640b99213643d141550326f34f0502fedb1798adb3c9eb79650b1ecb2f177", + "sha256:4ae594c66f4a7e1ea67232a0846649a7c94c188d6c071ac0210c3e86a5f92109", + "sha256:4b694430b3f00eb02c594ff5a16db30e054c1b9589a043fe9174584c6efa8033", + "sha256:51d8ea2a3a1a8fe4f67de21b8b93757005213e8ac3917567872f2865185fa7fb", + "sha256:54de2619f5ea58474f2ac211ceea6b615af2d7e4306220d4f3fe690c91988a61", + "sha256:551ad543fa19e94943c5b2cebc54c73353ffff08228ee5f3376bd27b3d5b9800", + "sha256:5543c055d8ec7801901e1193a51570643d6a6ab8751b1f7dd9af71af467538bb", + "sha256:5cd9cd7170459b9223c5e592ac036e0704bee765706445c353d96f2890e816c8", + "sha256:5ec4b3f0aebbbe2fc0134ee30a791af522a92ad9f164858805a77442d7d18570", + "sha256:67ecd894e56a0c6108ec5ab1d8fa8418ec0cff45844a855966b875d1039a2e34", + "sha256:6ab2ed84bf0137927846b37e882745a827458689eb969028af8032b1b3dac78e", + "sha256:704f63b774218207b8ccc6c47fcef5340741e5d839d11d606f70af93ee78e4d4", + "sha256:7098c524ba9f20717a56a8d551d2ed491ea89cbf37e540759ed3b776a4f8d6eb", + "sha256:7aed90a72fc3654fba9bc4b7f851571dcc368120432ad68b226bd593f3f6c0b7", + "sha256:7ce606c14bb195b0e5108544b540e2c5faed6843367e4ab3deb5c6aa5e681208", + "sha256:7eb95fe8222932c10d4436e7a6f7c99991e3fdd9f36c949eff16a69246dee2dc", + "sha256:80b80b889cb767cc47f31d2b2f3dec2db8126fbcd0cff31b3925b4dc6609dcdb", + "sha256:82cd0a69cd28f6cc3789cc6adeb1027f79526b1ab50b1f6062bbc3a0ccb2dbc3", + "sha256:83215147121e15d5f3a45d99abeed9cf1fe16869d5c233b08c56cdf75f43a504", + "sha256:88900f521c645f784260a8d346e12a1590f79e96403971241e64c3a265c8ecdb", + "sha256:91129ff1bb0619bc1f4ad19485718cc623a2dc433dff95baadbf89405c7f6b57", + "sha256:920974009fb37b20d32afcdf0227a2e707eb83fe418713f7a8b7de038b870d0b", + "sha256:9233ac249b354c54146e392e8a451e465dd2d967fc773690811d3a8c240ac601", + "sha256:941460db8fe3bd613db52f05259c9336f5a47ccae7d7def44cc277184030a116", + "sha256:942f8b1f3b223638b02df7df79140646c03938d488fbfb771824f3d05fc083a8", + "sha256:964b16dcc10c79a4a2be9f1273fcc2684a9eedb3906439720598029a797b46e6", + "sha256:9691a549c19c22d26a4f3b948071e93517bdf86e41b81d8c6ac8a964bb71e5a6", + "sha256:96979d753b1dc3b2169003e1854dc67bfc86edf93c01e84757927f810b8c3c93", + "sha256:987b9ac04d0b38ef4f89fbc035e84a7efad9cdd5f1e29024f9289182c8d99e09", + "sha256:988631b9d78b546e284478c2ec15c8a85960e262e247b35ca5eaf7ee22f6050a", + "sha256:9a96edd79661e93327cfeac4edec72a4046e14550a1d22aa0dd2e3ca52aec921", + "sha256:9b7408511fca48a82a119d78a77c2f5eb1b22fe88b0d2450ed0756d194fe7a9a", + "sha256:9dd6082f4e2aec9b6a0927202c85bc1b09dcab113f97265127c1dc20e2e32495", + "sha256:a2ad5add903eb7cdde2b7c64aaca405f3957ab34f16594d2b78d53b8b1a6a7d6", + "sha256:a8c65c17aed7e15a0c824cdc63a6b104dfc530f6fa8cb6ac51c437af52b481c7", + "sha256:aadf28046e77a72f30dcc1ab185639e8de7f4104b8cb5c6dfa5d8ed860e57236", + "sha256:b076da1ed19dc37788f6a934c60adf97bd02c7eea461b73730513921a85d4235", + "sha256:b2aeab3895d778155054abea5238d0eb9a72e9242bd4b43f42fd911ef9a13470", + "sha256:b82edc98d107cbc7357da7a5a695901b47d6eb0420e587256ba3ad24b80b7d0b", + "sha256:b8a0ccc8f2698f120e9e5742f4b38dc944c38744d4bdfc427616f3a163dd9de5", + "sha256:b993b6f524d1e274a5062488a43e3f9f8764ee9745ccd8e8193df743dbe5ee61", + "sha256:bb34d1605f96a245fc39790a117ac1bac8de84ab7691637b26ab2c5efb8f228c", + "sha256:bd3366aceedf274f765a3a4bc95d6cd97b130d1dda524d8f25225d14123c01db", + "sha256:c12f6f67495ea05c3d542d119d270007090bad5b843f642d418eb601ec0fa7be", + "sha256:c662a4cbdd6280ee56f841f14620787215a171c4e2d1744c9528bed8f5816c96", + "sha256:c884d1a59e69e03b93cf0dfee8794c63d7de0ee8f7ffb76e5f75be8131b6400a", + "sha256:ca339088839582d01654e6f83a637a4b8194d0960477b9769d2ff2cfa0fa36d2", + "sha256:cd2b6c5dfe0929b6c23dde9624483380b170b6e34ed79054ad131b20203a1a63", + "sha256:ce0f9fbe7d295f9922c0424a3637b88c6c472b75eafeaff6f910494a1fa719ef", + "sha256:cf0633e4a1b667bfe0bb10b5e53fe0d5f34a6243ea2530eb342491f1adf4f739", + "sha256:cf9273e96f3ee2ac89ffcb17627a78f78e7516b08f94dc435844ae72576a276e", + "sha256:d909b5a3fff619dc7e48b6b1bedc2f30ec43033ba7af32f936c10839e81b9217", + "sha256:d9b6627408021452dcd0d2cdf8da0534e19d93d070bfa8b6b4176f99711e7f90", + "sha256:de35342190deb7b866ad6ba5cbcccb2d22c0487ee0cbb251efef0843d705f0d4", + "sha256:e51c80c168074faa793685656c38eb7a06cbad7774c8cbc3ea05552d615393d8", + "sha256:e6bd1e9b95bc5614a7a9c9c44fde9539cba1c823b43a9f7bc11266446dd568e3", + "sha256:e7a9aaa5a1267125eef22cef3b63484c3241aaec6f48949b366d26c7250e0357", + "sha256:e951d1a8e9963ea51efd7f150450803e3b95db5939f994ad3d5edac2b6f6e2b4", + "sha256:e9941a4ada58f6218694f382e43fdd256e97615db9da135e77359da257a7168b", + "sha256:f0640913d2c1044d97e30d7c41728195fc37e54d190c5385eacb52115127b882", + "sha256:f0ccf3e01afeb412a1a9993049cb160d0352dba635bbca7762b2dc722aa5742a", + "sha256:f2181c20ef18747d5f4a7ea513e09ea03bdd50884a11ce46066bb90fe4213675", + "sha256:f2200e00b62568cfd920127782c61bc1c546062a879cdc741cfcc6976668dfcf", + "sha256:fcbdc5f2b0f1cd0f6a56cdb46fe41d2cce1e644e3b68832f3eeebc5fb0f7712e" ], "markers": "python_version >= '3.6'", - "version": "==2022.10.31" + "version": "==2023.8.8" }, "selenium": { "hashes": [ - "sha256:28430ac54a54fa59ad1f5392a1b89b169fe3ab2c2ccd1a9a10b6fe74f36cd6da", - "sha256:61cda3a304f82637162bc155cae7bf88fdb04c115fa2cb1c1c2e1358fcd19a9f" + "sha256:3c413a4f1b8af67824703195e3b1c19cfb1c3186c799efa035d55fd59d6dd59f", + "sha256:f0f9185c01ae249a321529c4e3aa0edc2a900642e61fdbb76988cd72d2762ece" ], "index": "pypi", - "version": "==4.8.3" + "markers": "python_version >= '3.8'", + "version": "==4.13.0" }, "six": { "hashes": [ @@ -1079,19 +1142,19 @@ }, "tomlkit": { "hashes": [ - "sha256:8c726c4c202bdb148667835f68d68780b9a003a9ec34167b6c673b38eff2a171", - "sha256:9330fc7faa1db67b541b28e62018c17d20be733177d290a13b24c62d1614e0c3" + "sha256:38e1ff8edb991273ec9f6181244a6a391ac30e9f5098e7535640ea6be97a7c86", + "sha256:712cbd236609acc6a3e2e97253dfc52d4c2082982a88f61b640ecf0817eab899" ], "markers": "python_version >= '3.7'", - "version": "==0.11.8" + "version": "==0.12.1" }, "tqdm": { "hashes": [ - "sha256:1871fb68a86b8fb3b59ca4cdd3dcccbc7e6d613eeed31f4c332531977b89beb5", - "sha256:c4f53a17fe37e132815abceec022631be8ffe1b9381c2e6e30aa70edc99e9671" + "sha256:d302b3c5b53d47bce91fea46679d9c3c6508cf6332229aa1e7d8653723793386", + "sha256:d88e651f9db8d8551a62556d3cff9e3034274ca5d66e93197cf2490e2dcb69c7" ], "markers": "python_version >= '3.7'", - "version": "==4.65.0" + "version": "==4.66.1" }, "trio": { "hashes": [ @@ -1103,27 +1166,27 @@ }, "trio-websocket": { "hashes": [ - "sha256:1a748604ad906a7dcab9a43c6eb5681e37de4793ba0847ef0bc9486933ed027b", - "sha256:a9937d48e8132ebf833019efde2a52ca82d223a30a7ea3e8d60a7d28f75a4e3a" + "sha256:18c11793647703c158b1f6e62de638acada927344d534e3c7628eedcb746839f", + "sha256:520d046b0d030cf970b8b2b2e00c4c2245b3807853ecd44214acd33d74581638" ], "markers": "python_version >= '3.7'", - "version": "==0.10.3" + "version": "==0.11.1" }, "typing-extensions": { "hashes": [ - "sha256:440d5dd3af93b060174bf433bccd69b0babc3b15b1a8dca43789fd7f61514b36", - "sha256:b75ddc264f0ba5615db7ba217daeb99701ad295353c45f9e95963337ceeeffb2" + "sha256:8f92fc8806f9a6b641eaa5318da32b44d401efaac0f6678c9bc448ba3605faa0", + "sha256:df8e4339e9cb77357558cbdbceca33c303714cf861d1eef15e1070055ae8b7ef" ], "markers": "python_version < '3.11'", - "version": "==4.7.1" + "version": "==4.8.0" }, "urllib3": { "hashes": [ - "sha256:8d22f86aae8ef5e410d4f539fde9ce6b2113a001bb4d189e0aed70642d602b11", - "sha256:de7df1803967d2c2a98e4b11bb7d6bd9210474c46e8a0401514e3a42a75ebde4" + "sha256:13abf37382ea2ce6fb744d4dad67838eec857c9f4f57009891805e0b5e123594", + "sha256:ef16afa8ba34a1f989db38e1dbbe0c302e4289a47856990d0682e374563ce35e" ], "markers": "python_version >= '3.7'", - "version": "==2.0.4" + "version": "==2.0.5" }, "wrapt": { "hashes": [ @@ -1211,7 +1274,7 @@ "sha256:ad565f26ecb92588a3e43bc3d96164de84cd9902482b130d0ddbaa9664a85065", "sha256:b9acddd652b585d75b20477888c56642fdade28bdfd3579aa24a4d2c037dd736" ], - "markers": "python_version >= '3.7'", + "markers": "python_full_version >= '3.7.0'", "version": "==1.2.0" } } diff --git a/embark/dashboard/views.py b/embark/dashboard/views.py index 1dc4e9f29..cfa2e9883 100644 --- a/embark/dashboard/views.py +++ b/embark/dashboard/views.py @@ -123,6 +123,7 @@ def show_log(request, analysis_id): except FileNotFoundError: return HttpResponseServerError(content="File is not yet available") + @require_http_methods(["GET"]) @login_required(login_url='/' + settings.LOGIN_URL) def show_logviewer(request, analysis_id): @@ -141,6 +142,6 @@ def show_logviewer(request, analysis_id): logger.debug("Taking file at %s and render it", log_file_path_) try: return render(request, 'dashboard/logViewer.html', {'analysis_id': analysis_id, 'username': request.user.username}) - + except FileNotFoundError: return HttpResponseServerError(content="File is not yet available") diff --git a/embark/embark/logviewer.py b/embark/embark/logviewer.py index 1b4820661..7a66cb187 100644 --- a/embark/embark/logviewer.py +++ b/embark/embark/logviewer.py @@ -5,7 +5,6 @@ import json from pathlib import Path -import sys from watchdog.observers import Observer from watchdog.events import FileSystemEventHandler from channels.db import database_sync_to_async @@ -17,6 +16,14 @@ class LogConsumer(AsyncWebsocketConsumer): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + + self.analysis_id = None + self.line_cache = None + self.file_view = None + self.observer = None + def load_file_content(self): logger.info( 'Getting file content for analysis id "%s"; view: %s', @@ -26,15 +33,15 @@ def load_file_content(self): num_lines = self.line_cache.num_lines() limit = min(num_lines, self.file_view.limit) - + # Ensure you can't read negative lines offset = max(self.file_view.offset, 0) # Ensure you can't read lines bigger than the last line offset = min(num_lines - limit, offset) - + # Fix the offset, in case the user tried to read invalid lines self.file_view.offset = offset - + content = self.line_cache.read_lines(offset, offset + limit - 1) self.file_view.content = base64.b64encode(content).decode("ascii") self.file_view.num_lines = num_lines @@ -53,11 +60,11 @@ async def connect(self): firmware = await self.get_firmware(self.analysis_id) log_file_path_ = f"{Path(firmware.path_to_logs).parent}/emba_run.log" - + if not os.path.isfile(log_file_path_): await self.send_message({"error": "The log file does not exist, yet."}) await self.close() - + self.line_cache = LineCache(log_file_path_) self.file_view = FileView() @@ -65,7 +72,7 @@ async def connect(self): this = self class ModifyEventHandler(FileSystemEventHandler): - def on_modified(self, event): + def on_modified(self, _event): asyncio.run(this.update_lines()) event_handler = ModifyEventHandler() @@ -94,9 +101,9 @@ async def receive(self, text_data: str = "", bytes_data=None) -> None: self.file_view = FileView(**data["file_view"]) await self.send_file_content() else: - raise Exception("Unknown action") - except Exception as e: - logger.error(e) + raise NotImplementedError("Unknown action") + except Exception as exception: + logger.error(exception) await self.send_message({"error": "Unknown error"}) async def disconnect(self, code): @@ -123,32 +130,32 @@ def __init__(self, offset=0, limit=30, content="", num_lines=0) -> None: self.num_lines = num_lines -refresh_lines = 10 # This will define how many of the last lines will be refreshed +REFRESH_LINES = 10 # This will define how many of the last lines will be refreshed class LineCache: def __init__(self, filepath: str) -> None: self.line_beginnings = [0] self.line_endings = [0] + # Intentionally not using with to save resources + # because we don't have to open the file as often + # pylint: disable-next=consider-using-with self.filehandle = open(filepath, "rb") self.refresh() def refresh(self) -> None: - refresh_from_line = min( - len(self.line_beginnings), refresh_lines - ) # Make sure to not go out of bounds + # Make sure to not go out of bounds + refresh_from_line = min(len(self.line_beginnings), REFRESH_LINES) refresh_from_byte = self.line_beginnings[-refresh_from_line] - self.line_beginnings = self.line_beginnings[ - :-refresh_from_line - ] # Remove the line beginnings to be refreshed - self.line_endings = self.line_endings[ - :-refresh_from_line - ] # Remove the line endings to be refreshed + + # Remove the line beginnings and endings to be refreshed + self.line_beginnings = self.line_beginnings[:-refresh_from_line] + self.line_endings = self.line_endings[:-refresh_from_line] logger.debug( - f"Start refreshing line cache from line {refresh_from_line} (start counting from end of the file)" + "Start refreshing line cache from line %s (start counting from end of the file)", refresh_from_line ) - logger.debug(f"Start refreshing line cache from byte {refresh_from_byte}") + logger.debug("Start refreshing line cache from byte %s", refresh_from_byte) self.filehandle.seek(refresh_from_byte) @@ -160,13 +167,13 @@ def refresh(self) -> None: if len(line) > 0 and line[-1:] == b'\n': line_ending = line_ending - 1 - + self.line_beginnings.append(line_beginning) self.line_endings.append(line_ending) if len(line) == 0 or line[-1:] != b'\n': break - + def num_lines(self) -> int: return len(self.line_beginnings) @@ -174,14 +181,14 @@ def read_lines(self, first_line: int, last_line: int) -> bytes: num_lines = self.num_lines() if first_line > last_line: - raise Exception("The first line cannot be bigger than the last line") - + raise IndexError("The first line cannot be bigger than the last line") + if first_line < 0: raise IndexError("The first line cannot be below 0") if last_line >= num_lines: raise IndexError("The first line cannot be equal or above the number of lines") - + first_byte = self.line_beginnings[num_lines - last_line - 1] last_byte = self.line_endings[num_lines - first_line - 1] self.filehandle.seek(first_byte) diff --git a/embark/embark/tests/test_linecache.py b/embark/embark/tests/test_linecache.py index 2cdcf8748..448f7a96b 100644 --- a/embark/embark/tests/test_linecache.py +++ b/embark/embark/tests/test_linecache.py @@ -2,12 +2,13 @@ from embark.logviewer import LineCache + class TestLineCache(unittest.TestCase): def test_default(self): line_cache = LineCache('./test/logviewer/line_cache_test1.log') - for i in range(0,2): + for _ in range(0, 2): self.assertEqual(12, line_cache.num_lines(), 'Incorrect number of lines.') self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') self.assertEqual([0, 7, 11, 23, 31, 41, 50, 54, 58, 72, 86, 104], line_cache.line_beginnings, 'The line beginning cache is not valid.') @@ -19,7 +20,7 @@ def test_default(self): def test_no_newline_end(self): line_cache = LineCache('./test/logviewer/line_cache_test_no_newline.log') - for i in range(0,2): + for _ in range(0, 2): self.assertEqual(11, line_cache.num_lines(), 'Incorrect number of lines.') self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') self.assertEqual([0, 7, 11, 23, 31, 41, 50, 54, 58, 72, 86], line_cache.line_beginnings, 'The line beginning cache is not valid.') @@ -31,7 +32,7 @@ def test_no_newline_end(self): def test_empty(self): line_cache = LineCache('./test/logviewer/line_cache_test_empty.log') - for i in range(0,2): + for _ in range(0, 2): self.assertEqual(1, line_cache.num_lines(), 'Incorrect number of lines.') self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') self.assertEqual([0], line_cache.line_beginnings, 'The line beginning cache is not valid.') @@ -41,4 +42,4 @@ def test_empty(self): if __name__ == '__main__': - unittest.main() \ No newline at end of file + unittest.main() diff --git a/embark/static/scripts/logViewer.js b/embark/static/scripts/logViewer.js index aadfbadb6..aeea233be 100644 --- a/embark/static/scripts/logViewer.js +++ b/embark/static/scripts/logViewer.js @@ -1,119 +1,123 @@ // jshint unused:false // ^ this should only be added AFTER successfull check (disables waring for global functions) -window.addEventListener("load", onLoad, false); +/* global analysis_id */ -function onLoad() { - var loc = window.location; - var wsStart = 'ws://'; - var wsPort = ':8001'; - if (loc.protocol == 'https:') { - wsStart = 'wss://'; - wsPort = ':8000'; - } - var socket = new WebSocket( - wsStart + location.hostname + wsPort + '/ws/logs/' + analysis_id - ); +window.addEventListener( + "load", + function () { + "use strict"; + var loc = window.location; + var wsStart = "ws://"; + var wsPort = ":8001"; + if (loc.protocol == "https:") { + wsStart = "wss://"; + wsPort = ":8000"; + } - var file_view = { + var socket = new WebSocket( + wsStart + location.hostname + wsPort + "/ws/logs/" + analysis_id + ); + + var file_view = { offset: 0, limit: 50, - content: '' - }; - - var logArea = document.getElementById('logArea') - - socket.onopen = function (evt) { onOpen(evt) }; - socket.onclose = function (evt) { onClose(evt) }; - socket.onmessage = function (evt) { onMessage(evt) }; - socket.onerror = function (evt) { onError(evt) }; + content: "", + }; - function onOpen(evt) { - console.log('open', evt); - } + var logArea = document.getElementById("logArea"); - function onClose(evt) { - console.log('close', evt); - } - - function onMessage(evt) { + function onMessage(evt) { // There are two types of messages: // 1. a chat participant message itself // 2. a message with a number of connected chat participants var message = JSON.parse(evt.data); if (message.file_view) { - fileContent = atob(message.file_view.content) + '\u00a0' // The nbsp is required in order to preserve trailing newlines - logArea.textContent = fileContent - file_view = message.file_view + var fileContent = atob(message.file_view.content) + "\u00a0"; // The nbsp is required in order to preserve trailing newlines + logArea.textContent = fileContent; + file_view = message.file_view; } - } + } - function onError(evt) { - console.log('error', evt); - } + function onError(evt) { + console.log("error", evt); + } - function requestUpdate() { - request_view = Object.assign({}, file_view) - request_view.content = '' - socket.send(JSON.stringify({ 'action': 'change_view', 'file_view': request_view })); - } + socket.onmessage = function (evt) { + onMessage(evt); + }; - window.LogControls = { - move_offset: function(lines) { - file_view.offset += lines - requestUpdate(); - }, - increase_view_size: function(lines) { - file_view.limit += lines - if (file_view.limit < 5) { - file_view.limit = 5 - } - if (file_view.num_lines != 0 && file_view.limit > file_view.num_lines) { - file_view.limit = file_view.num_lines - } - requestUpdate(); - } - } + socket.onerror = function (evt) { + onError(evt); + }; - document.onkeyup = checkKey; + function requestUpdate() { + var requestView = Object.assign({}, file_view); + requestView.content = ""; + socket.send( + JSON.stringify({ action: "change_view", file_view: requestView }) + ); + } - function checkKey(e) { + window.LogControls = { + move_offset: function (lines) { + file_view.offset += lines; + requestUpdate(); + }, + increase_view_size: function (lines) { + file_view.limit += lines; + if (file_view.limit < 5) { + file_view.limit = 5; + } + if (file_view.num_lines != 0 && file_view.limit > file_view.num_lines) { + file_view.limit = file_view.num_lines; + } + requestUpdate(); + }, + }; + + function checkKey(e) { if (e.shiftKey) { - if (e.keyCode == '38') { // Arrow Up - window.LogControls.increase_view_size(1); - e.preventDefault(); - } - else if (e.keyCode == '40') { // Arrow Down - window.LogControls.increase_view_size(-1); - e.preventDefault(); - } - else if (e.keyCode == '33') { // Page Up - window.LogControls.increase_view_size(5); - e.preventDefault(); - } - else if (e.keyCode == '34') { // Page Down - window.LogControls.increase_view_size(-5); - e.preventDefault(); - } + if (e.keyCode == "38") { + // Arrow Up + window.LogControls.increase_view_size(1); + e.preventDefault(); + } else if (e.keyCode == "40") { + // Arrow Down + window.LogControls.increase_view_size(-1); + e.preventDefault(); + } else if (e.keyCode == "33") { + // Page Up + window.LogControls.increase_view_size(5); + e.preventDefault(); + } else if (e.keyCode == "34") { + // Page Down + window.LogControls.increase_view_size(-5); + e.preventDefault(); + } + } else { + if (e.keyCode == "38") { + // Arrow Up + window.LogControls.move_offset(1); + e.preventDefault(); + } else if (e.keyCode == "40") { + // Arrow Down + window.LogControls.move_offset(-1); + e.preventDefault(); + } else if (e.keyCode == "33") { + // Page Up + window.LogControls.move_offset(5); + e.preventDefault(); + } else if (e.keyCode == "34") { + // Page Down + window.LogControls.move_offset(-5); + e.preventDefault(); + } } - else { - if (e.keyCode == '38') { // Arrow Up - window.LogControls.move_offset(1); - e.preventDefault(); - } - else if (e.keyCode == '40') { // Arrow Down - window.LogControls.move_offset(-1); - e.preventDefault(); - } - else if (e.keyCode == '33') { // Page Up - window.LogControls.move_offset(5); - e.preventDefault(); - } - else if (e.keyCode == '34') { // Page Down - window.LogControls.move_offset(-5); - e.preventDefault(); - } - } - } -} \ No newline at end of file + } + + document.onkeyup = checkKey; + }, + false +); diff --git a/embark/templates/dashboard/logViewer.html b/embark/templates/dashboard/logViewer.html index 088a90d33..9fed2d6cc 100644 --- a/embark/templates/dashboard/logViewer.html +++ b/embark/templates/dashboard/logViewer.html @@ -21,24 +21,24 @@

    Controls

    - -
    - -
    From e937c66c1f879deb48713c21cefb9bb6500bd165 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sun, 1 Oct 2023 17:20:54 +0200 Subject: [PATCH 05/12] Added buttons to jump to top or bottom of the file --- embark/static/scripts/logViewer.js | 6 +++++- embark/templates/dashboard/logViewer.html | 6 ++++++ 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/embark/static/scripts/logViewer.js b/embark/static/scripts/logViewer.js index aeea233be..365959d35 100644 --- a/embark/static/scripts/logViewer.js +++ b/embark/static/scripts/logViewer.js @@ -19,7 +19,7 @@ window.addEventListener( wsStart + location.hostname + wsPort + "/ws/logs/" + analysis_id ); - var file_view = { + window.file_view = { offset: 0, limit: 50, content: "", @@ -65,6 +65,10 @@ window.addEventListener( file_view.offset += lines; requestUpdate(); }, + set_offset: function (offset) { + file_view.offset = offset; + requestUpdate(); + }, increase_view_size: function (lines) { file_view.limit += lines; if (file_view.limit < 5) { diff --git a/embark/templates/dashboard/logViewer.html b/embark/templates/dashboard/logViewer.html index 9fed2d6cc..1d4a0dfee 100644 --- a/embark/templates/dashboard/logViewer.html +++ b/embark/templates/dashboard/logViewer.html @@ -27,6 +27,9 @@ + @@ -38,6 +41,9 @@ + From 82d0ce5514c07ff76772e490498d27ec72bdf8c3 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sun, 1 Oct 2023 11:57:07 +0200 Subject: [PATCH 06/12] Implemented ansi coloring for the log viewer --- embark/static/content/css/logviewer.css | 8 ++++ embark/static/scripts/logViewer.js | 47 ++++++++++++++--------- embark/templates/dashboard/logViewer.html | 2 + installer.sh | 2 + 4 files changed, 40 insertions(+), 19 deletions(-) create mode 100644 embark/static/content/css/logviewer.css diff --git a/embark/static/content/css/logviewer.css b/embark/static/content/css/logviewer.css new file mode 100644 index 000000000..16da84e85 --- /dev/null +++ b/embark/static/content/css/logviewer.css @@ -0,0 +1,8 @@ +#logArea, +#logArea * { + font-family: monospace; +} + +#logArea { + line-height: 1.19; +} \ No newline at end of file diff --git a/embark/static/scripts/logViewer.js b/embark/static/scripts/logViewer.js index 365959d35..c81a48abc 100644 --- a/embark/static/scripts/logViewer.js +++ b/embark/static/scripts/logViewer.js @@ -27,30 +27,39 @@ window.addEventListener( var logArea = document.getElementById("logArea"); - function onMessage(evt) { - // There are two types of messages: - // 1. a chat participant message itself - // 2. a message with a number of connected chat participants - var message = JSON.parse(evt.data); - - if (message.file_view) { - var fileContent = atob(message.file_view.content) + "\u00a0"; // The nbsp is required in order to preserve trailing newlines - logArea.textContent = fileContent; - file_view = message.file_view; - } - } - function onError(evt) { console.log("error", evt); } - socket.onmessage = function (evt) { - onMessage(evt); - }; + var socket = undefined; - socket.onerror = function (evt) { - onError(evt); - }; + import('/static/external/scripts/ansi_up.js').then(({ AnsiUp }) => { + socket = new WebSocket( + wsStart + location.hostname + wsPort + "/ws/logs/" + analysis_id + ); + + function onMessage(evt) { + var message = JSON.parse(evt.data); + + if (message.file_view) { + var fileContent = Base64.decode(message.file_view.content) // We cannot use atob because of unicode (see https://stackoverflow.com/questions/30106476/using-javascripts-atob-to-decode-base64-doesnt-properly-decode-utf-8-strings) + fileContent = fileContent + "\u00a0"; // The nbsp is required in order to preserve trailing newlines + + var ansi_up = new AnsiUp(); + var coloredFileContent = ansi_up.ansi_to_html(fileContent); + logArea.innerHTML = coloredFileContent; + file_view = message.file_view; + } + } + + socket.onmessage = function (evt) { + onMessage(evt); + }; + + socket.onerror = function (evt) { + onError(evt); + }; + }); function requestUpdate() { var requestView = Object.assign({}, file_view); diff --git a/embark/templates/dashboard/logViewer.html b/embark/templates/dashboard/logViewer.html index 1d4a0dfee..e63fac92f 100644 --- a/embark/templates/dashboard/logViewer.html +++ b/embark/templates/dashboard/logViewer.html @@ -2,6 +2,7 @@ {% load static %} {% block title %}EMBArk Logs{% endblock title %} {% block navigation %}{% include "navigation.html" %}{% endblock navigation %} +{% block style %}{% endblock style %} {% block maincontent %}
    @@ -55,5 +56,6 @@ {% endblock maincontent %} {% block inlinejs %} + {% endblock inlinejs %} \ No newline at end of file diff --git a/installer.sh b/installer.sh index d195deda1..42fa1e679 100755 --- a/installer.sh +++ b/installer.sh @@ -344,6 +344,8 @@ install_embark_default(){ wget -O ./embark/static/external/scripts/bootstrap.js https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/js/bootstrap.bundle.min.js wget -O ./embark/static/external/scripts/datatable.js https://cdn.datatables.net/v/bs5/dt-1.11.2/datatables.min.js wget -O ./embark/static/external/scripts/charts.js https://cdn.jsdelivr.net/npm/chart.js@3.5.1/dist/chart.min.js + wget -O ./embark/static/external/scripts/base64.js https://cdn.jsdelivr.net/npm/js-base64@3.7.5/base64.min.js + wget -O ./embark/static/external/scripts/ansi_up.js https://cdn.jsdelivr.net/npm/ansi_up@6.0.2/ansi_up.min.js wget -O ./embark/static/external/css/confirm.css https://cdnjs.cloudflare.com/ajax/libs/jquery-confirm/3.3.2/jquery-confirm.min.css wget -O ./embark/static/external/css/bootstrap.css https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/css/bootstrap.min.css wget -O ./embark/static/external/css/datatable.css https://cdn.datatables.net/v/bs5/dt-1.11.2/datatables.min.css From 12a0438f05679bd5775e841f3946c2437f712404 Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sun, 1 Oct 2023 17:35:12 +0200 Subject: [PATCH 07/12] Fixed merge issue --- embark/static/scripts/logViewer.js | 4 ---- 1 file changed, 4 deletions(-) diff --git a/embark/static/scripts/logViewer.js b/embark/static/scripts/logViewer.js index c81a48abc..bd6aeb412 100644 --- a/embark/static/scripts/logViewer.js +++ b/embark/static/scripts/logViewer.js @@ -15,10 +15,6 @@ window.addEventListener( wsPort = ":8000"; } - var socket = new WebSocket( - wsStart + location.hostname + wsPort + "/ws/logs/" + analysis_id - ); - window.file_view = { offset: 0, limit: 50, From 5bed8e3927204f284b565f20b474d6bfc913491b Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sun, 1 Oct 2023 17:50:31 +0200 Subject: [PATCH 08/12] Fixed issues with cr lf lines in log files --- embark/embark/logviewer.py | 3 +++ embark/embark/tests/test_linecache.py | 12 ++++++++++++ 2 files changed, 15 insertions(+) diff --git a/embark/embark/logviewer.py b/embark/embark/logviewer.py index 7a66cb187..3ccee4785 100644 --- a/embark/embark/logviewer.py +++ b/embark/embark/logviewer.py @@ -168,6 +168,9 @@ def refresh(self) -> None: if len(line) > 0 and line[-1:] == b'\n': line_ending = line_ending - 1 + if len(line) > 1 and line[-2:] == b'\r\n': + line_ending = line_ending - 1 + self.line_beginnings.append(line_beginning) self.line_endings.append(line_ending) diff --git a/embark/embark/tests/test_linecache.py b/embark/embark/tests/test_linecache.py index 448f7a96b..b40064442 100644 --- a/embark/embark/tests/test_linecache.py +++ b/embark/embark/tests/test_linecache.py @@ -17,6 +17,18 @@ def test_default(self): self.assertEqual(b'10: ggggggggg\n11: hhhhhhhhhhhhh\n', line_cache.read_lines(0, 2), 'The line cache did not return the correct value.') self.assertEqual(b'9: ffffffffff\n10: ggggggggg\n11: hhhhhhhhhhhhh', line_cache.read_lines(1, 3), 'The line cache did not return the correct value.') + def test_cr_lf(self): + line_cache = LineCache('./test/logviewer/line_cache_test_cr_lf.log') + + for _ in range(0, 2): + self.assertEqual(12, line_cache.num_lines(), 'Incorrect number of lines.') + self.assertEqual(len(line_cache.line_endings), len(line_cache.line_beginnings), 'The number of line beginnings and line endings do not match.') + self.assertEqual([0, 7, 10, 22, 30, 40, 49, 52, 55, 69, 83, 102], line_cache.line_beginnings, 'The line beginning cache is not valid.') + line_cache.refresh() + + self.assertEqual(b'10: ggggggggg\n11: hhhhhhhhhhhhh\r\n', line_cache.read_lines(0, 2), 'The line cache did not return the correct value.') + self.assertEqual(b'9: ffffffffff\n10: ggggggggg\n11: hhhhhhhhhhhhh', line_cache.read_lines(1, 3), 'The line cache did not return the correct value.') + def test_no_newline_end(self): line_cache = LineCache('./test/logviewer/line_cache_test_no_newline.log') From 1e5d4193eba71018230048d72d4592e1793e3a8d Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Sun, 1 Oct 2023 18:29:54 +0200 Subject: [PATCH 09/12] Fixed some linting issues --- embark/static/scripts/logViewer.js | 60 +++++++++++------------ embark/templates/dashboard/logViewer.html | 3 +- installer.sh | 32 ++++++------ 3 files changed, 47 insertions(+), 48 deletions(-) diff --git a/embark/static/scripts/logViewer.js b/embark/static/scripts/logViewer.js index bd6aeb412..25206b4b9 100644 --- a/embark/static/scripts/logViewer.js +++ b/embark/static/scripts/logViewer.js @@ -3,6 +3,9 @@ /* global analysis_id */ +import { AnsiUp } from '/static/external/scripts/ansi_up.js'; +import { Base64 } from '/static/external/scripts/base64.js'; + window.addEventListener( "load", function () { @@ -27,38 +30,35 @@ window.addEventListener( console.log("error", evt); } - var socket = undefined; - import('/static/external/scripts/ansi_up.js').then(({ AnsiUp }) => { - socket = new WebSocket( - wsStart + location.hostname + wsPort + "/ws/logs/" + analysis_id - ); + var socket = new WebSocket( + wsStart + location.hostname + wsPort + "/ws/logs/" + analysis_id + ); - function onMessage(evt) { - var message = JSON.parse(evt.data); + function onMessage(evt) { + var message = JSON.parse(evt.data); - if (message.file_view) { - var fileContent = Base64.decode(message.file_view.content) // We cannot use atob because of unicode (see https://stackoverflow.com/questions/30106476/using-javascripts-atob-to-decode-base64-doesnt-properly-decode-utf-8-strings) - fileContent = fileContent + "\u00a0"; // The nbsp is required in order to preserve trailing newlines + if (message.file_view) { + var fileContent = Base64.decode(message.file_view.content); // We cannot use atob because of unicode (see https://stackoverflow.com/questions/30106476/using-javascripts-atob-to-decode-base64-doesnt-properly-decode-utf-8-strings) + fileContent = fileContent + "\u00a0"; // The nbsp is required in order to preserve trailing newlines - var ansi_up = new AnsiUp(); - var coloredFileContent = ansi_up.ansi_to_html(fileContent); - logArea.innerHTML = coloredFileContent; - file_view = message.file_view; - } + var ansi_up = new AnsiUp(); + var coloredFileContent = ansi_up.ansi_to_html(fileContent); + logArea.innerHTML = coloredFileContent; + window.file_view = message.file_view; } + } - socket.onmessage = function (evt) { - onMessage(evt); - }; + socket.onmessage = function (evt) { + onMessage(evt); + }; - socket.onerror = function (evt) { - onError(evt); - }; - }); + socket.onerror = function (evt) { + onError(evt); + }; function requestUpdate() { - var requestView = Object.assign({}, file_view); + var requestView = Object.assign({}, window.file_view); requestView.content = ""; socket.send( JSON.stringify({ action: "change_view", file_view: requestView }) @@ -67,20 +67,20 @@ window.addEventListener( window.LogControls = { move_offset: function (lines) { - file_view.offset += lines; + window.file_view.offset += lines; requestUpdate(); }, set_offset: function (offset) { - file_view.offset = offset; + window.file_view.offset = offset; requestUpdate(); }, increase_view_size: function (lines) { - file_view.limit += lines; - if (file_view.limit < 5) { - file_view.limit = 5; + window.file_view.limit += lines; + if (window.file_view.limit < 5) { + window.file_view.limit = 5; } - if (file_view.num_lines != 0 && file_view.limit > file_view.num_lines) { - file_view.limit = file_view.num_lines; + if (window.file_view.num_lines != 0 && window.file_view.limit > window.file_view.num_lines) { + window.file_view.limit = window.file_view.num_lines; } requestUpdate(); }, diff --git a/embark/templates/dashboard/logViewer.html b/embark/templates/dashboard/logViewer.html index e63fac92f..4ad4fe13e 100644 --- a/embark/templates/dashboard/logViewer.html +++ b/embark/templates/dashboard/logViewer.html @@ -56,6 +56,5 @@ {% endblock maincontent %} {% block inlinejs %} - - + {% endblock inlinejs %} \ No newline at end of file diff --git a/installer.sh b/installer.sh index 42fa1e679..c7ec90d00 100755 --- a/installer.sh +++ b/installer.sh @@ -35,7 +35,7 @@ export GREEN='\033[0;32m' export ORANGE='\033[0;33m' export CYAN='\033[0;36m' export BOLD='\033[1m' -export NC='\033[0m' # no +export NC='\033[0m' # no print_help(){ echo -e "\\n""$CYAN""USAGE""$NC" @@ -88,7 +88,7 @@ write_env(){ local ENV_FILES=() local LAST_PW_HASH="" local CHECK_PW="" - + if [[ -d safe ]]; then mapfile -d '' ENV_FILES < <(find ./safe -iname "*.env" -print0 2> /dev/null) if [[ ${#ENV_FILES[@]} -gt 0 ]] && [[ -f safe/history.env ]]; then @@ -111,11 +111,11 @@ write_env(){ DJANGO_SECRET_KEY=$(python3.10 -c 'from django.core.management.utils import get_random_secret_key; print(get_random_secret_key())') RANDOM_PW=$(openssl rand -base64 12) fi - + echo -e "$ORANGE""$BOLD""Creating a EMBArk configuration file .env""$NC" { echo "DATABASE_NAME=embark" - echo "DATABASE_USER=embark" + echo "DATABASE_USER=embark" echo "DATABASE_PASSWORD=$RANDOM_PW" echo "DATABASE_HOST=172.22.0.5" echo "DATABASE_PORT=3306" @@ -136,7 +136,7 @@ write_env(){ install_emba(){ echo -e "\n$GREEN""$BOLD""Installation of the firmware scanner EMBA on host""$NC" if git submodule status emba | grep --quiet '^-'; then - sudo -u "${SUDO_USER:-${USER}}" git submodule init emba + sudo -u "${SUDO_USER:-${USER}}" git submodule init emba fi sudo -u "${SUDO_USER:-${USER}}" git submodule update --remote sudo -u "${SUDO_USER:-${USER}}" git config --global --add safe.directory "$PWD"/emba @@ -152,13 +152,13 @@ install_emba(){ } create_ca (){ - # FIXME could use some work + # FIXME could use some work echo -e "\n$GREEN""$BOLD""Creating SSL Cert""$NC" if ! [[ -d cert ]]; then sudo -u "${SUDO_USER:-${USER}}" git checkout -- cert fi cd cert || exit 1 - if [[ -f embark.local.csr ]] || [[ -f embark-ws.local.csr ]] || [[ -f embark.local.crt ]] || [[ -f embark-ws.local.crt ]]; then + if [[ -f embark.local.csr ]] || [[ -f embark-ws.local.csr ]] || [[ -f embark.local.crt ]] || [[ -f embark-ws.local.crt ]]; then echo -e "\n$GREEN""$BOLD""Certs already generated, skipping""$NC" else # create CA @@ -284,10 +284,10 @@ install_embark_default(){ if [[ "$WSL" -eq 1 ]]; then echo -e "$RED""$BOLD""EMBArk currently does not support WSL in default mode. (only in Dev-mode)""$NC" fi - + #debs apt-get install -y -q default-libmysqlclient-dev build-essential mysql-client-core-8.0 - + # install pipenv pip3.10 install pipenv @@ -323,7 +323,7 @@ install_embark_default(){ # daemon install_daemon - + #add ssl cert create_ca @@ -333,7 +333,7 @@ install_embark_default(){ #install packages cp ./Pipfile* /var/www/ (cd /var/www && PIPENV_VENV_IN_PROJECT=1 pipenv install) - + # download externals if ! [[ -d ./embark/static/external ]]; then @@ -344,7 +344,7 @@ install_embark_default(){ wget -O ./embark/static/external/scripts/bootstrap.js https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/js/bootstrap.bundle.min.js wget -O ./embark/static/external/scripts/datatable.js https://cdn.datatables.net/v/bs5/dt-1.11.2/datatables.min.js wget -O ./embark/static/external/scripts/charts.js https://cdn.jsdelivr.net/npm/chart.js@3.5.1/dist/chart.min.js - wget -O ./embark/static/external/scripts/base64.js https://cdn.jsdelivr.net/npm/js-base64@3.7.5/base64.min.js + wget -O ./embark/static/external/scripts/base64.js https://cdn.jsdelivr.net/npm/js-base64@3.7.5/+esm wget -O ./embark/static/external/scripts/ansi_up.js https://cdn.jsdelivr.net/npm/ansi_up@6.0.2/ansi_up.min.js wget -O ./embark/static/external/css/confirm.css https://cdnjs.cloudflare.com/ajax/libs/jquery-confirm/3.3.2/jquery-confirm.min.css wget -O ./embark/static/external/css/bootstrap.css https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/css/bootstrap.min.css @@ -383,7 +383,7 @@ install_embark_dev(){ # npm packages npm install -g jshint # npm install -g dockerlinter - + # install pipenv pip3 install pipenv @@ -392,7 +392,7 @@ install_embark_dev(){ echo "${SUDO_USER:-${USER}}"" ALL=(ALL) NOPASSWD: /bin/pkill" | EDITOR='tee -a' visudo echo "root ALL=(ALL) NOPASSWD: ""$PWD""/emba/emba" | EDITOR='tee -a' visudo echo "root ALL=(ALL) NOPASSWD: /bin/pkill" | EDITOR='tee -a' visudo - + #pipenv PIPENV_VENV_IN_PROJECT=1 pipenv install --dev @@ -495,7 +495,7 @@ uninstall (){ # delete user www-embark and reset visudo echo -e "$ORANGE""$BOLD""Delete user""$NC" - + if id -u www-embark &>/dev/null ; then userdel www-embark fi @@ -549,7 +549,7 @@ uninstall (){ systemctl daemon-reload fi sudo -u "${SUDO_USER:-${USER}}" git checkout HEAD -- embark.service - + # reset ownership etc # reset server-certs From 6d249178ca04d89719a857a71f334eac9b5220ac Mon Sep 17 00:00:00 2001 From: Christian Bieg Date: Fri, 6 Oct 2023 08:38:07 +0200 Subject: [PATCH 10/12] Added missing requirement to the dev install script --- installer.sh | 2 ++ 1 file changed, 2 insertions(+) diff --git a/installer.sh b/installer.sh index c7ec90d00..b1f4c9f4c 100755 --- a/installer.sh +++ b/installer.sh @@ -423,6 +423,8 @@ install_embark_dev(){ wget -O ./embark/static/external/scripts/bootstrap.js https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/js/bootstrap.bundle.min.js wget -O ./embark/static/external/scripts/datatable.js https://cdn.datatables.net/v/bs5/dt-1.11.2/datatables.min.js wget -O ./embark/static/external/scripts/charts.js https://cdn.jsdelivr.net/npm/chart.js@3.5.1/dist/chart.min.js + wget -O ./embark/static/external/scripts/base64.js https://cdn.jsdelivr.net/npm/js-base64@3.7.5/+esm + wget -O ./embark/static/external/scripts/ansi_up.js https://cdn.jsdelivr.net/npm/ansi_up@6.0.2/ansi_up.min.js wget -O ./embark/static/external/css/confirm.css https://cdnjs.cloudflare.com/ajax/libs/jquery-confirm/3.3.2/jquery-confirm.min.css wget -O ./embark/static/external/css/bootstrap.css https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/css/bootstrap.min.css wget -O ./embark/static/external/css/datatable.css https://cdn.datatables.net/v/bs5/dt-1.11.2/datatables.min.css From e876bb9a896a7ae1eefa4fb8c45c5138de5cc21c Mon Sep 17 00:00:00 2001 From: Benedikt Kuehne Date: Mon, 9 Oct 2023 13:39:13 +0200 Subject: [PATCH 11/12] add MySQL LDFLAGS --- installer.sh | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/installer.sh b/installer.sh index fc9d7abd8..c9957ff28 100755 --- a/installer.sh +++ b/installer.sh @@ -317,7 +317,7 @@ install_embark_default(){ #install packages cp ./Pipfile* /var/www/ - (cd /var/www && PIPENV_VENV_IN_PROJECT=1 pipenv install) + (cd /var/www && MYSQLCLIENT_LDFLAGS='-L/usr/mysql/lib -lmysqlclient -lssl -lcrypto -lresolv' MYSQLCLIENT_CFLAGS='-I/usr/include/mysql/' PIPENV_VENV_IN_PROJECT=1 pipenv install) # download externals @@ -378,7 +378,7 @@ install_embark_dev(){ #pipenv - PIPENV_VENV_IN_PROJECT=1 pipenv install --dev + MYSQLCLIENT_LDFLAGS='-L/usr/mysql/lib -lmysqlclient -lssl -lcrypto -lresolv' MYSQLCLIENT_CFLAGS='-I/usr/include/mysql/' PIPENV_VENV_IN_PROJECT=1 pipenv install --dev #Server-Dir if ! [[ -d media ]]; then From f3508cf9d90e37eb5f449a0825b4076518432efc Mon Sep 17 00:00:00 2001 From: kali Date: Mon, 6 Nov 2023 13:11:44 +0100 Subject: [PATCH 12/12] shellcheck braces and updated check-project file --- dev-tools/check_project.sh | 188 +++++++++++++++--------------- dev-tools/clean-setup.sh | 56 ++++----- dev-tools/debug-server-start.sh | 48 ++++---- dev-tools/prepare.sh | 32 ++--- dev-tools/test.sh | 26 ++--- embark/entrypoint.sh | 22 ++-- export-DB.sh | 16 +-- helper/helper_embark_general.sh | 76 ++++++------ helper/wickStrictModeFail.sh | 16 +-- import-DB.sh | 18 +-- installer.sh | 200 ++++++++++++++++---------------- run-server.sh | 104 ++++++++--------- supervisor.sh | 4 +- 13 files changed, 403 insertions(+), 403 deletions(-) diff --git a/dev-tools/check_project.sh b/dev-tools/check_project.sh index 8a20a2937..8314fe3b5 100755 --- a/dev-tools/check_project.sh +++ b/dev-tools/check_project.sh @@ -16,10 +16,10 @@ # And check Django if its deployable export DJANGO_SETTINGS_MODULE=embark.settings.dev -export PYTHONPATH="$PYTHONPATH:${PWD}/embark/embark/:${PWD}/embark/" +export PYTHONPATH="${PYTHONPATH}:${PWD}/embark/embark/:${PWD}/embark/" export PIPENV_VENV_IN_PROJECT="True" -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 cd .. || exit 1 RED='\033[0;31m' @@ -28,35 +28,35 @@ ORANGE='\033[0;33m' BOLD='\033[1m' NC='\033[0m' # no color -if [[ $EUID -eq 0 ]]; then - echo -e "\\n$RED""Running this script as root is not supported""$NC\\n" +if [[ ${EUID} -eq 0 ]]; then + echo -e "\\n${RED}""Running this script as root is not supported""${NC}\\n" fi # check that all tools are installed check_tools(){ TOOLS=("jshint" "shellcheck" "pylint" "yamllint") for TOOL in "${TOOLS[@]}";do - if ! command -v "$TOOL" > /dev/null ; then - echo -e "\\n""$RED""$TOOL is not installed correctly""$NC""\\n" + if ! command -v "${TOOL}" > /dev/null ; then + echo -e "\\n""${RED}""${TOOL} is not installed correctly""${NC}""\\n" exit 1 fi done pipenv run djlint --version | grep "version"; RES=$? - if [[ -z "$RES" ]];then - echo -e "\\n""$RED""djlint(pip) is not installed correctly""$NC""\\n" + if [[ -z "${RES}" ]];then + echo -e "\\n""${RED}""djlint(pip) is not installed correctly""${NC}""\\n" fi } # checks django configuration check_django(){ cd ./embark || exit 1 - echo -e "\\n""$ORANGE""$BOLD""EMBArk django-settings check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk django-settings check""${NC}""\\n""${BOLD}""=================================================================""${NC}" if ! pipenv run python ./manage.py check | grep -q "System check identified no issues"; then #TODO add --deploy flag - echo -e "\\n""$RED$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${RED}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) MODULES_TO_CHECK_ARR+=( "./embark/settings.py" ) else - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" fi cd .. || exit 1 } @@ -64,25 +64,25 @@ check_django(){ # checks js-scripts with jshint for errors # config @ .jshintrc jscheck(){ - echo -e "\\n""$ORANGE""$BOLD""EMBArk javascript-files check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk javascript-files check""${NC}""\\n""${BOLD}""=================================================================""${NC}" mapfile -t JS_SCRIPTS < <(find ./embark -iname "*.js") for JS_SCRIPT in "${JS_SCRIPTS[@]}"; do - echo -e "\\n""$GREEN""Run jshint on $JS_SCRIPT:""$NC""\\n" - # mapfile -t JS_RESULT < <(jshint "$JS_SCRIPT") - jshint -c ./.jshintrc "$JS_SCRIPT" + echo -e "\\n""${GREEN}""Run jshint on ${JS_SCRIPT}:""${NC}""\\n" + # mapfile -t JS_RESULT < <(jshint "${JS_SCRIPT}") + jshint -c ./.jshintrc "${JS_SCRIPT}" RES=$? - if [[ $RES -eq 2 ]] ; then - echo -e "\\n""$RED$BOLD==> FIX ERRORS""$NC""\\n" + if [[ ${RES} -eq 2 ]] ; then + echo -e "\\n""${RED}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$JS_SCRIPT" ) - elif [[ $RES -eq 1 ]]; then - echo -e "\\n""$ORANGE$BOLD==> FIX WARNINGS""$NC""\\n" + MODULES_TO_CHECK_ARR+=( "${JS_SCRIPT}" ) + elif [[ ${RES} -eq 1 ]]; then + echo -e "\\n""${ORANGE}${BOLD}==> FIX WARNINGS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$JS_SCRIPT" ) - elif [[ $RES -eq 0 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + MODULES_TO_CHECK_ARR+=( "${JS_SCRIPT}" ) + elif [[ ${RES} -eq 0 ]]; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$RED$BOLD""[jshint]ERRORS in SCRIPT""$NC""\\n" + echo -e "\\n""${RED}${BOLD}""[jshint]ERRORS in SCRIPT""${NC}""\\n" fi done } @@ -90,109 +90,109 @@ jscheck(){ # uses djlint to check for errors in all html-template files inside /embark (Django-root-dir) # no config templatechecker(){ - echo -e "\\n""$ORANGE""$BOLD""EMBArk html-templates check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk html-templates check""${NC}""\\n""${BOLD}""=================================================================""${NC}" mapfile -t HTML_FILE < <(find ./embark -iname "*.html") for HTML_FILE in "${HTML_FILE[@]}"; do - echo -e "\\n""$GREEN""Run djlint on $HTML_FILE:""$NC""\\n" - pipenv run djlint "$HTML_FILE" + echo -e "\\n""${GREEN}""Run djlint on ${HTML_FILE}:""${NC}""\\n" + pipenv run djlint "${HTML_FILE}" RES=$? - if [[ $RES -eq 1 ]]; then - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + if [[ ${RES} -eq 1 ]]; then + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$HTML_FILE" ) - elif [[ $RES -eq 0 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + MODULES_TO_CHECK_ARR+=( "${HTML_FILE}" ) + elif [[ ${RES} -eq 0 ]]; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$RED$BOLD""[html-check(tidy)]ERRORS in SCRIPT""$NC""\\n" + echo -e "\\n""${RED}${BOLD}""[html-check(tidy)]ERRORS in SCRIPT""${NC}""\\n" fi done } shellchecker() { - echo -e "\\n""$ORANGE""$BOLD""EMBArk Shellcheck""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk Shellcheck""${NC}""\\n""${BOLD}""=================================================================""${NC}" if ! command -v shellcheck >/dev/null 2>&1; then - echo -e "\\n""$ORANGE""Shellcheck not found!""$NC""\\n""$ORANGE""Install shellcheck via 'apt-get install shellcheck'!""$NC\\n" + echo -e "\\n""${ORANGE}""Shellcheck not found!""${NC}""\\n""${ORANGE}""Install shellcheck via 'apt-get install shellcheck'!""${NC}\\n" exit 1 fi - echo -e "\\n""$GREEN""Run shellcheck on installer:""$NC""\\n" + echo -e "\\n""${GREEN}""Run shellcheck on installer:""${NC}""\\n" if shellcheck ./installer.sh || [[ $? -ne 1 && $? -ne 2 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) MODULES_TO_CHECK_ARR+=( "installer.sh" ) fi - echo -e "\\n""$GREEN""Run shellcheck on this script:""$NC""\\n" + echo -e "\\n""${GREEN}""Run shellcheck on this script:""${NC}""\\n" if shellcheck ./dev-tools/check_project.sh || [[ $? -ne 1 && $? -ne 2 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) MODULES_TO_CHECK_ARR+=( "dev-tools/check_project.sh" ) fi - echo -e "\\n""$GREEN""Find shell scripts and run shellcheck on them:""$NC""\\n" + echo -e "\\n""${GREEN}""Find shell scripts and run shellcheck on them:""${NC}""\\n" mapfile -t SH_SCRIPTS < <(find embark -iname "*.sh") for SH_SCRIPT in "${SH_SCRIPTS[@]}"; do - echo -e "\\n""$GREEN""Run shellcheck on $SH_SCRIPT:""$NC""\\n" - if shellcheck -x "$SH_SCRIPT" || [[ $? -ne 1 && $? -ne 2 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "\\n""${GREEN}""Run shellcheck on ${SH_SCRIPT}:""${NC}""\\n" + if shellcheck -x -o require-variable-braces "${SH_SCRIPT}" || [[ $? -ne 1 && $? -ne 2 ]]; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$SH_SCRIPT" ) + MODULES_TO_CHECK_ARR+=( "${SH_SCRIPT}" ) fi done } pycodestyle_check(){ - echo -e "\\n""$ORANGE""$BOLD""EMBArk pycodestyle check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk pycodestyle check""${NC}""\\n""${BOLD}""=================================================================""${NC}" echo -e "[*] Searching python files and test with pycodestyle.py" if command -v pipenv run pycodstyle >/dev/null 2>&1; then - echo -e "\\n""$ORANGE""pycodestyle found in PATH!""$NC""\\n" + echo -e "\\n""${ORANGE}""pycodestyle found in PATH!""${NC}""\\n" PYCODESTYLE="pycodestyle" elif [[ -f "/usr/lib/python3/dist-packages/pycodestyle.py" ]]; then - echo -e "\\n""$ORANGE""pycodestyle found!""$NC""\\n" + echo -e "\\n""${ORANGE}""pycodestyle found!""${NC}""\\n" PYCODESTYLE="/usr/lib/python3/dist-packages/pycodestyle.py" else - echo -e "\\n""$ORANGE""pycodestyle not found!""$NC""\\n""$ORANGE""Install pycodestyle via 'apt-get install pycodestyle'!""$NC\\n" + echo -e "\\n""${ORANGE}""pycodestyle not found!""${NC}""\\n""${ORANGE}""Install pycodestyle via 'apt-get install pycodestyle'!""${NC}\\n" exit 1 fi - echo -e "\\n""$GREEN""Find python scripts and run pycodestyle on them:""$NC""\\n" + echo -e "\\n""${GREEN}""Find python scripts and run pycodestyle on them:""${NC}""\\n" mapfile -t PY_SCRIPTS < <(find embark -iname "*.py") for PY_SCRIPT in "${PY_SCRIPTS[@]}"; do - echo -e "\\n""$GREEN""Run pycodestyle on $PY_SCRIPT:""$NC""\\n" - if pipenv run "$PYCODESTYLE" --config=./.pycodestylerc --first "$PY_SCRIPT" 2> >(grep -v "Courtesy Notice\|Loading .env" >&2) || [[ $? -ne 1 && $? -ne 2 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "\\n""${GREEN}""Run pycodestyle on ${PY_SCRIPT}:""${NC}""\\n" + if pipenv run "${PYCODESTYLE}" --config=./.pycodestylerc --first "${PY_SCRIPT}" 2> >(grep -v "Courtesy Notice\|Loading .env" >&2) || [[ $? -ne 1 && $? -ne 2 ]]; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$PY_SCRIPT" ) + MODULES_TO_CHECK_ARR+=( "${PY_SCRIPT}" ) fi done } banditer() { - echo -e "\\n""$ORANGE""$BOLD""EMBArk bandit check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk bandit check""${NC}""\\n""${BOLD}""=================================================================""${NC}" if command -v bandit >/dev/null 2>&1; then - echo -e "\\n""$ORANGE""bandit found in PATH!""$NC""\\n" + echo -e "\\n""${ORANGE}""bandit found in PATH!""${NC}""\\n" else - echo -e "\\n""$ORANGE""bandit not found!""$NC""\\n""$ORANGE""Install bandit via 'apt-get install bandit'!""$NC\\n" + echo -e "\\n""${ORANGE}""bandit not found!""${NC}""\\n""${ORANGE}""Install bandit via 'apt-get install bandit'!""${NC}\\n" exit 1 fi for PY_SCRIPT in "${PY_SCRIPTS[@]}"; do - echo -e "\\n""$GREEN""Run bandit on $PY_SCRIPT:""$NC""\\n" - if bandit -c .banditrc "$PY_SCRIPT" 2> /dev/null | grep -q "No issues identified."; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "\\n""${GREEN}""Run bandit on ${PY_SCRIPT}:""${NC}""\\n" + if bandit -c .banditrc "${PY_SCRIPT}" 2> /dev/null | grep -q "No issues identified."; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - bandit -c .banditrc "$PY_SCRIPT" - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + bandit -c .banditrc "${PY_SCRIPT}" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$PY_SCRIPT" ) + MODULES_TO_CHECK_ARR+=( "${PY_SCRIPT}" ) fi done @@ -200,33 +200,33 @@ banditer() { pylinter(){ cd ./embark || exit 1 - echo -e "\\n""$ORANGE""$BOLD""EMBArk pylint check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk pylint check""${NC}""\\n""${BOLD}""=================================================================""${NC}" mapfile -t PY_SCRIPTS < <(find . -type d -name migrations -prune -false -o -iname "*.py") for PY_SCRIPT in "${PY_SCRIPTS[@]}"; do - echo -e "\\n""$GREEN""Run pylint on $PY_SCRIPT:""$NC""\\n" - mapfile -t PY_RESULT < <(pipenv run pylint --rcfile=../.pylintrc "$PY_SCRIPT" 2> >(grep -v "Courtesy Notice\|Loading .env" >&2) ) + echo -e "\\n""${GREEN}""Run pylint on ${PY_SCRIPT}:""${NC}""\\n" + mapfile -t PY_RESULT < <(pipenv run pylint --rcfile=../.pylintrc "${PY_SCRIPT}" 2> >(grep -v "Courtesy Notice\|Loading .env" >&2) ) local RATING_10=0 if [[ "${#PY_RESULT[@]}" -gt 0 ]]; then if ! printf '%s\n' "${PY_RESULT[@]}" | grep -q -P '^Your code has been rated at 10'; then for LINE in "${PY_RESULT[@]}"; do - echo "$LINE" + echo "${LINE}" done else RATING_10=1 fi - if [[ "$RATING_10" -ne 1 ]]; then - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + if [[ "${RATING_10}" -ne 1 ]]; then + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$PY_SCRIPT" ) + MODULES_TO_CHECK_ARR+=( "${PY_SCRIPT}" ) else - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" fi else - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" fi done - echo -e "\\n""$GREEN""Run pylint on all scripts:""$NC""\\n" + echo -e "\\n""${GREEN}""Run pylint on all scripts:""${NC}""\\n" pipenv run pylint --rcfile=../.pylintrc ./* 2> >(grep -v "Courtesy Notice\|Loading .env" >&2) | grep "Your code has been rated" cd .. || exit 1 } @@ -238,35 +238,35 @@ dockerchecker(){ else ENV=0 fi - echo -e "\\n""$ORANGE""$BOLD""EMBArk docker-files check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk docker-files check""${NC}""\\n""${BOLD}""=================================================================""${NC}" mapfile -t DOCKER_COMPS < <(find . -maxdepth 1 -type d -name migrations -prune -false -o -iname "docker-compose*.yml") for DOCKER_COMP in "${DOCKER_COMPS[@]}"; do - echo -e "\\n""$GREEN""Run docker check on $DOCKER_COMP:""$NC""\\n" - if docker-compose -f "$DOCKER_COMP" config 1>/dev/null || [[ $? -ne 1 ]]; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "\\n""${GREEN}""Run docker check on ${DOCKER_COMP}:""${NC}""\\n" + if docker-compose -f "${DOCKER_COMP}" config 1>/dev/null || [[ $? -ne 1 ]]; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$DOCKER_COMP" ) + MODULES_TO_CHECK_ARR+=( "${DOCKER_COMP}" ) fi done #TODO dockerlinter -f ./Dockerfile - if [[ $ENV -gt 0 ]]; then + if [[ ${ENV} -gt 0 ]]; then rm .env #remove dummy fi } yamlchecker(){ - echo -e "\\n""$ORANGE""$BOLD""EMBArk yaml-files check""$NC""\\n""$BOLD""=================================================================""$NC" + echo -e "\\n""${ORANGE}""${BOLD}""EMBArk yaml-files check""${NC}""\\n""${BOLD}""=================================================================""${NC}" mapfile -t YAML_COMPS < <(find . -maxdepth 1 -type d -name migrations -prune -false -o -iname "*.yml") for YAML_COMP_ in "${YAML_COMPS[@]}"; do - echo -e "\\n""$GREEN""Run docker check on $YAML_COMP_:""$NC""\\n" - if yamllint "$YAML_COMP_" ; then - echo -e "$GREEN""$BOLD""==> SUCCESS""$NC""\\n" + echo -e "\\n""${GREEN}""Run docker check on ${YAML_COMP_}:""${NC}""\\n" + if yamllint "${YAML_COMP_}" ; then + echo -e "${GREEN}""${BOLD}""==> SUCCESS""${NC}""\\n" else - echo -e "\\n""$ORANGE$BOLD==> FIX ERRORS""$NC""\\n" + echo -e "\\n""${ORANGE}${BOLD}==> FIX ERRORS""${NC}""\\n" ((MODULES_TO_CHECK=MODULES_TO_CHECK+1)) - MODULES_TO_CHECK_ARR+=( "$YAML_COMP_" ) + MODULES_TO_CHECK_ARR+=( "${YAML_COMP_}" ) fi done } @@ -286,12 +286,12 @@ check_django yamlchecker if [[ "${#MODULES_TO_CHECK_ARR[@]}" -gt 0 ]]; then - echo -e "\\n\\n""$GREEN$BOLD""SUMMARY:$NC\\n" - echo -e "Modules to check: $MODULES_TO_CHECK\\n" + echo -e "\\n\\n""${GREEN}${BOLD}""SUMMARY:${NC}\\n" + echo -e "Modules to check: ${MODULES_TO_CHECK}\\n" for MODULE in "${MODULES_TO_CHECK_ARR[@]}"; do - echo -e "$RED$BOLD==> FIX MODULE: ""$MODULE""$NC" + echo -e "${RED}${BOLD}==> FIX MODULE: ""${MODULE}""${NC}" done exit 1 fi -echo -e "$GREEN$BOLD===> ALL CHECKS SUCCESSFUL""$NC" -exit 0 \ No newline at end of file +echo -e "${GREEN}${BOLD}===> ALL CHECKS SUCCESSFUL""${NC}" +exit 0 diff --git a/dev-tools/clean-setup.sh b/dev-tools/clean-setup.sh index e179e58c1..7e1806239 100755 --- a/dev-tools/clean-setup.sh +++ b/dev-tools/clean-setup.sh @@ -11,65 +11,65 @@ # Description: Script for cleaning up all docker- container and images -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 cd .. -echo -e "\n$GREEN""$BOLD""Reset docker container & networks""$NC" +echo -e "\n${GREEN}""${BOLD}""Reset docker container & networks""${NC}" docker images docker container ls -a while docker images | grep -qE "\"; do IMAGE_ID=$(docker images | grep -E "\" | awk '{print $3}') - echo -e "$GREEN""$BOLD""Remove failed docker image""$NC" - docker image rm "$IMAGE_ID" -f + echo -e "${GREEN}""${BOLD}""Remove failed docker image""${NC}" + docker image rm "${IMAGE_ID}" -f done if docker images | grep -qE "^mysql[[:space:]]*latest"; then - echo -e "\n$GREEN""$BOLD""Found mysql docker environment - removing it""$NC" + echo -e "\n${GREEN}""${BOLD}""Found mysql docker environment - removing it""${NC}" CONTAINER_ID=$(docker container ls -a | grep -E "embark_db" | awk '{print $1}') - echo -e "$GREEN""$BOLD""Stop mysql docker container""$NC" - docker container stop "$CONTAINER_ID" - echo -e "$GREEN""$BOLD""Remove mysql docker container""$NC" - docker container rm "$CONTAINER_ID" -f - echo -e "$GREEN""$BOLD""Remove mysql docker image""$NC" + echo -e "${GREEN}""${BOLD}""Stop mysql docker container""${NC}" + docker container stop "${CONTAINER_ID}" + echo -e "${GREEN}""${BOLD}""Remove mysql docker container""${NC}" + docker container rm "${CONTAINER_ID}" -f + echo -e "${GREEN}""${BOLD}""Remove mysql docker image""${NC}" docker image rm mysql:latest -f fi if docker images | grep -qE "^redis[[:space:]]*5"; then - echo -e "\n$GREEN""$BOLD""Found redis docker environment - removing it""$NC" + echo -e "\n${GREEN}""${BOLD}""Found redis docker environment - removing it""${NC}" CONTAINER_ID=$(docker container ls -a | grep -E "embark_redis" | awk '{print $1}') - echo -e "$GREEN""$BOLD""Stop redis docker container""$NC" - docker container stop "$CONTAINER_ID" - echo -e "$GREEN""$BOLD""Remove redis docker container""$NC" - docker container rm "$CONTAINER_ID" -f - echo -e "$GREEN""$BOLD""Remove redis docker image""$NC" + echo -e "${GREEN}""${BOLD}""Stop redis docker container""${NC}" + docker container stop "${CONTAINER_ID}" + echo -e "${GREEN}""${BOLD}""Remove redis docker container""${NC}" + docker container rm "${CONTAINER_ID}" -f + echo -e "${GREEN}""${BOLD}""Remove redis docker image""${NC}" docker image rm redis:5 -f fi #networks if docker network ls | grep -E "embark_dev"; then - echo -e "\n$GREEN""$BOLD""Found EMBArk_dev network - removing it""$NC" + echo -e "\n${GREEN}""${BOLD}""Found EMBArk_dev network - removing it""${NC}" NET_ID=$(docker network ls | grep -E "embark_dev" | awk '{print $1}') - echo -e "$GREEN""$BOLD""Remove EMBArk_dev network""$NC" - docker network rm "$NET_ID" + echo -e "${GREEN}""${BOLD}""Remove EMBArk_dev network""${NC}" + docker network rm "${NET_ID}" fi if docker network ls | grep -E "embark_frontend"; then - echo -e "\n$GREEN""$BOLD""Found EMBArk_frontend network - removing it""$NC" + echo -e "\n${GREEN}""${BOLD}""Found EMBArk_frontend network - removing it""${NC}" NET_ID=$(docker network ls | grep -E "embark_frontend" | awk '{print $1}') - echo -e "$GREEN""$BOLD""Remove EMBArk_frontend network""$NC" - docker network rm "$NET_ID" + echo -e "${GREEN}""${BOLD}""Remove EMBArk_frontend network""${NC}" + docker network rm "${NET_ID}" fi if docker network ls | grep -E "embark_backend"; then - echo -e "\n$GREEN""$BOLD""Found EMBArk_backend network - removing it""$NC" + echo -e "\n${GREEN}""${BOLD}""Found EMBArk_backend network - removing it""${NC}" NET_ID=$(docker network ls | grep -E "embark_backend" | awk '{print $1}') - echo -e "$GREEN""$BOLD""Remove EMBArk_backend network""$NC" - docker network rm "$NET_ID" + echo -e "${GREEN}""${BOLD}""Remove EMBArk_backend network""${NC}" + docker network rm "${NET_ID}" fi -echo -e "\n$GREEN""$BOLD""Clearing Migrations""$NC" +echo -e "\n${GREEN}""${BOLD}""Clearing Migrations""${NC}" find . -path "*/migrations/*.py" -not -name "__init__.py" -delete find . -path "*/migrations/*.pyc" -delete @@ -79,8 +79,8 @@ fuser -k 8080/tcp docker container prune -echo -e "$ORANGE""$BOLD""Consider running \$docker system prune""$NC" +echo -e "${ORANGE}""${BOLD}""Consider running \$docker system prune""${NC}" rm -Rf -I ./embark_db find . -path "*/migrations/*.py" -not -name "__init__.py" -delete -pipenv run ./embark/manage.py flush \ No newline at end of file +pipenv run ./embark/manage.py flush diff --git a/dev-tools/debug-server-start.sh b/dev-tools/debug-server-start.sh index 51c468601..b41737309 100755 --- a/dev-tools/debug-server-start.sh +++ b/dev-tools/debug-server-start.sh @@ -28,7 +28,7 @@ export WSL=0 cleaner() { pkill -u root daphne - pkill -u root "$PWD"/emba/emba + pkill -u root "${PWD}"/emba/emba pkill -u root runapscheduler docker container stop embark_db @@ -37,8 +37,8 @@ cleaner() { # docker container prune -f --filter "label=flag" # rm embark_db/* -rf - fuser -k "$PORT"/tcp - chown "${SUDO_USER:-${USER}}" "$PWD" -R + fuser -k "${PORT}"/tcp + chown "${SUDO_USER:-${USER}}" "${PWD}" -R exit 1 } @@ -48,25 +48,25 @@ import_helper() local HELPER_COUNT=0 local HELPER_FILE="" local HELP_DIR='helper' - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } set -a trap cleaner INT -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 -if ! [[ $EUID -eq 0 ]] ; then - echo -e "\\n$RED""Run script with root permissions!""$NC\\n" +if ! [[ ${EUID} -eq 0 ]] ; then + echo -e "\\n${RED}""Run script with root permissions!""${NC}\\n" exit 1 fi @@ -79,31 +79,31 @@ import_helper # WSL/OS version check # WSL support - currently experimental! if grep -q -i wsl /proc/version; then - echo -e "\n${ORANGE}INFO: System running in WSL environment!$NC" - echo -e "\n${ORANGE}INFO: WSL is currently experimental!$NC" + echo -e "\n${ORANGE}INFO: System running in WSL environment!${NC}" + echo -e "\n${ORANGE}INFO: WSL is currently experimental!${NC}" WSL=1 fi -if [[ "$WSL" -eq 1 ]]; then +if [[ "${WSL}" -eq 1 ]]; then check_docker_wsl fi # check emba -echo -e "$BLUE""$BOLD""checking EMBA""$NC" +echo -e "${BLUE}""${BOLD}""checking EMBA""${NC}" if ! (cd ./emba && ./emba -d 1) ; then - echo -e "$RED""EMBA is not configured correctly""$NC" + echo -e "${RED}""EMBA is not configured correctly""${NC}" exit 1 fi check_db -if ! [[ -d "$PWD"/logs ]]; then +if ! [[ -d "${PWD}"/logs ]]; then mkdir logs fi -echo -e "\n[""$BLUE JOB""$NC""] Redis logs are copied to ./logs/redis_dev.log""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] Redis logs are copied to ./logs/redis_dev.log""${NC}" docker container logs embark_redis -f > ./logs/redis_dev.log & -echo -e "\n[""$BLUE JOB""$NC""] DB logs are copied to ./logs/mysql_dev.log""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] DB logs are copied to ./logs/mysql_dev.log""${NC}" docker container logs embark_db -f > ./logs/mysql_dev.log & # shellcheck disable=SC1091 @@ -118,22 +118,22 @@ python3 ./embark/manage.py migrate | tee -a ./logs/migration.log python3 ./embark/manage.py createsuperuser --noinput ## -echo -e "\n[""$BLUE JOB""$NC""] Starting runapscheduler" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting runapscheduler" python3 ./embark/manage.py runapscheduler | tee -a ./logs/scheduler.log & -echo -e "\n[""$BLUE JOB""$NC""] Starting daphne(ASGI) - log to /embark/logs/daphne.log" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting daphne(ASGI) - log to /embark/logs/daphne.log" echo "START DAPHNE" >./logs/daphne.log cd ./embark || exit 1 -pipenv run daphne -v 3 -p 8001 -b "$IP" --root-path="$PWD"/embark embark.asgi:application &>../logs/daphne.log & +pipenv run daphne -v 3 -p 8001 -b "${IP}" --root-path="${PWD}"/embark embark.asgi:application &>../logs/daphne.log & cd .. || exit 1 # start embark # systemctl start embark.service -echo -e "$ORANGE""$BOLD""start EMBArk server (WS/WSS not enabled -a also asgi)""$NC" -python3 ./embark/manage.py runserver "$IP":"$PORT" |& tee -a ./logs/debug-server.log +echo -e "${ORANGE}""${BOLD}""start EMBArk server (WS/WSS not enabled -a also asgi)""${NC}" +python3 ./embark/manage.py runserver "${IP}":"${PORT}" |& tee -a ./logs/debug-server.log wait -echo -e "\n$ORANGE""$BOLD""Done. To clean-up use the clean-setup script""$NC" \ No newline at end of file +echo -e "\n${ORANGE}""${BOLD}""Done. To clean-up use the clean-setup script""${NC}" diff --git a/dev-tools/prepare.sh b/dev-tools/prepare.sh index 136471cff..f094e5143 100755 --- a/dev-tools/prepare.sh +++ b/dev-tools/prepare.sh @@ -27,7 +27,7 @@ export "$(grep PYTHONPATH ./.env)" cleaner() { pkill -u root daphne - pkill -u root "$PWD"/emba/emba + pkill -u root "${PWD}"/emba/emba pkill -u root runapscheduler docker container stop embark_db @@ -36,8 +36,8 @@ cleaner() { # docker container prune -f --filter "label=flag" # rm embark_db/* -rf - fuser -k "$PORT"/tcp - chown "${SUDO_USER:-${USER}}" "$PWD" -R + fuser -k "${PORT}"/tcp + chown "${SUDO_USER:-${USER}}" "${PWD}" -R exit 1 } @@ -47,25 +47,25 @@ import_helper() local HELPER_COUNT=0 local HELPER_FILE="" local HELP_DIR='helper' - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } set -a trap cleaner INT -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 -if ! [[ $EUID -eq 0 ]] ; then - echo -e "\\n$RED""Run script with root permissions!""$NC\\n" +if ! [[ ${EUID} -eq 0 ]] ; then + echo -e "\\n${RED}""Run script with root permissions!""${NC}\\n" exit 1 fi @@ -78,20 +78,20 @@ import_helper # WSL/OS version check # WSL support - currently experimental! if grep -q -i wsl /proc/version; then - echo -e "\n${ORANGE}INFO: System running in WSL environment!$NC" - echo -e "\n${ORANGE}INFO: WSL is currently experimental!$NC" + echo -e "\n${ORANGE}INFO: System running in WSL environment!${NC}" + echo -e "\n${ORANGE}INFO: WSL is currently experimental!${NC}" WSL=1 fi check_db -if ! [[ -d "$PWD"/logs ]]; then +if ! [[ -d "${PWD}"/logs ]]; then mkdir logs fi -echo -e "\n[""$BLUE JOB""$NC""] Redis logs are copied to ./logs/redis_dev.log""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] Redis logs are copied to ./logs/redis_dev.log""${NC}" docker container logs embark_redis -f > ./logs/redis_dev.log & -echo -e "\n[""$BLUE JOB""$NC""] DB logs are copied to ./logs/mysql_dev.log""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] DB logs are copied to ./logs/mysql_dev.log""${NC}" docker container logs embark_db -f > ./logs/mysql_dev.log & # shellcheck disable=SC1091 @@ -103,4 +103,4 @@ python3 ./embark/manage.py makemigrations users uploader reporter dashboard port python3 ./embark/manage.py migrate | tee -a ./logs/migration.log # superuser -python3 ./embark/manage.py createsuperuser --noinput \ No newline at end of file +python3 ./embark/manage.py createsuperuser --noinput diff --git a/dev-tools/test.sh b/dev-tools/test.sh index 208ee4319..9c37b3f29 100755 --- a/dev-tools/test.sh +++ b/dev-tools/test.sh @@ -35,7 +35,7 @@ cleaner() { docker container prune -f --filter "label=flag" - fuser -k "$PORT"/tcp + fuser -k "${PORT}"/tcp exit 1 } @@ -44,25 +44,25 @@ import_helper() local HELPERS=() local HELPER_COUNT=0 local HELPER_FILE="" - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } set -a trap cleaner INT -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 cd .. || exit 1 import_helper -echo -e "\n$GREEN""$BOLD""Configuring Embark""$NC" +echo -e "\n${GREEN}""${BOLD}""Configuring Embark""${NC}" # shellcheck disable=SC1091 source ./.venv/bin/activate || exit 1 @@ -83,19 +83,19 @@ pipenv run ./embark/manage.py migrate | tee -a ./logs/migration.log pipenv run ./embark/manage.py createsuperuser --noinput # add privs -# echo -e "$BLUE""$BOLD""[+] Adding permissions for testing""$NC\\n" -# mysql --host="$(grep DATABASE_HOST ./.env | sed 's/DATABASE\_HOST\=//')" --user=root --password="$MYSQL_ROOT_PASSWORD" -e"GRANT ALL PRIVILEGES ON test_db.* TO 'embark'@'%';" +# echo -e "${BLUE}""${BOLD}""[+] Adding permissions for testing""$NC\\n" +# mysql --host="$(grep DATABASE_HOST ./.env | sed 's/DATABASE\_HOST\=//')" --user=root --password="${MYSQL_ROOT_PASSWORD}" -e"GRANT ALL PRIVILEGES ON test_db.* TO 'embark'@'%';" -echo -e "\n[""$BLUE JOB""$NC""] Redis logs are copied to ./embark/logs/redis.log""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] Redis logs are copied to ./embark/logs/redis.log""${NC}" docker container logs embark_redis -f > ./logs/redis.log & -echo -e "\n[""$BLUE JOB""$NC""] DB logs are copied to ./embark/logs/mysql.log""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] DB logs are copied to ./embark/logs/mysql.log""${NC}" docker container logs embark_db -f > ./logs/mysql.log & ## -echo -e "\n[""$BLUE JOB""$NC""] Testing""$NC" +echo -e "\n[""${BLUE} JOB""${NC}""] Testing""${NC}" pipenv run ./embark/manage.py test pipenv run ./embark/manage.py test embark.test_logreader # pipenv run ./embark/manage.py test users.tests.SeleniumTests.test_register # pipenv run ./embark/manage.py test users.tests.SeleniumTests.test_login pipenv run ./embark/manage.py test porter.tests.TestImport -echo -e "\n$ORANGE""$BOLD""Done. To clean-up use the clean-setup script""$NC" \ No newline at end of file +echo -e "\n${ORANGE}""${BOLD}""Done. To clean-up use the clean-setup script""${NC}" diff --git a/embark/entrypoint.sh b/embark/entrypoint.sh index 1ce740da0..ceb19162a 100755 --- a/embark/entrypoint.sh +++ b/embark/entrypoint.sh @@ -7,10 +7,10 @@ export HTTPS_PORT=443 export BIND_IP='0.0.0.0' export FILE_SIZE=2000000000 -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 # check emba -echo -e "$BLUE""$BOLD""checking EMBA""$NC" +echo -e "${BLUE}""${BOLD}""checking EMBA""${NC}" # TODO pipe # start venv (ignore source in script) @@ -30,36 +30,36 @@ if ! [[ -d /var/www/logs ]]; then fi # db_init -echo -e "\n[""$BLUE JOB""$NC""] Starting migrations - log to embark/logs/migration.log" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting migrations - log to embark/logs/migration.log" python3.10 manage.py makemigrations users uploader dashboard reporter | tee -a /var/www/logs/migration.log python3.10 manage.py migrate | tee -a /var/www/logs/migration.log # collect staticfiles and make accesable for server -echo -e "\n[""$BLUE JOB""$NC""] Collecting static files" +echo -e "\n[""${BLUE} JOB""${NC}""] Collecting static files" python3.10 manage.py collectstatic --no-input chown www-embark /var/www/ -R chmod 760 /var/www/media/ -R # TODO other fileperms -echo -e "\n[""$BLUE JOB""$NC""] Starting runapscheduler" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting runapscheduler" python3.10 manage.py runapscheduler | tee -a /var/www/logs/scheduler.log & sleep 5 -echo -e "\n[""$BLUE JOB""$NC""] Starting Apache" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting Apache" python3.10 manage.py runmodwsgi --user www-embark --group sudo \ ---host "$BIND_IP" --port="$HTTP_PORT" --limit-request-body "$FILE_SIZE" \ +--host "${BIND_IP}" --port="${HTTP_PORT}" --limit-request-body "${FILE_SIZE}" \ --url-alias /static/ /var/www/static/ \ --url-alias /media/ /var/www/media/ \ --allow-localhost --working-directory ./embark/ --server-root /var/www/httpd80/ \ --include-file /var/www/conf/embark.conf \ --server-name embark.local & # --ssl-certificate /var/www/conf/cert/embark.local --ssl-certificate-key-file /var/www/conf/cert/embark.local.key \ -# --https-port "$HTTPS_PORT" & +# --https-port "${HTTPS_PORT}" & # --enable-debugger --https-only \ sleep 5 -echo -e "\n[""$BLUE JOB""$NC""] Starting daphne(ASGI) - log to /embark/logs/daphne.log" -daphne --access-log /var/www/logs/daphne.log -e ssl:8000:privateKey=/var/www/conf/cert/embark-ws.local.key:certKey=/var/www/conf/cert/embark-ws.local.crt -b "$BIND_IP" -p 8001 -s embark-ws.local --root-path=/var/www/embark embark.asgi:application & +echo -e "\n[""${BLUE} JOB""${NC}""] Starting daphne(ASGI) - log to /embark/logs/daphne.log" +daphne --access-log /var/www/logs/daphne.log -e ssl:8000:privateKey=/var/www/conf/cert/embark-ws.local.key:certKey=/var/www/conf/cert/embark-ws.local.crt -b "${BIND_IP}" -p 8001 -s embark-ws.local --root-path=/var/www/embark embark.asgi:application & sleep 5 -wait \ No newline at end of file +wait diff --git a/export-DB.sh b/export-DB.sh index c429c9f35..7f33ea9f1 100755 --- a/export-DB.sh +++ b/export-DB.sh @@ -26,23 +26,23 @@ import_helper() local HELPER_COUNT=0 local HELPER_FILE="" local HELP_DIR='helper' - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 -if ! [[ $EUID -eq 0 ]] ; then - echo -e "\\n$RED""Run script with root permissions!""$NC\\n" +if ! [[ ${EUID} -eq 0 ]] ; then + echo -e "\\n${RED}""Run script with root permissions!""${NC}\\n" exit 1 fi @@ -50,4 +50,4 @@ echo "USER is ${SUDO_USER:-${USER}}" import_helper cp -urvi ./embark_db ./.embark_db_backup -tar -cf full-backup-"$(date +%F)".tar ./.embark_db_backup \ No newline at end of file +tar -cf full-backup-"$(date +%F)".tar ./.embark_db_backup diff --git a/helper/helper_embark_general.sh b/helper/helper_embark_general.sh index 2013d7a81..73a9ca2e9 100644 --- a/helper/helper_embark_general.sh +++ b/helper/helper_embark_general.sh @@ -18,19 +18,19 @@ docker_image_rm(){ local IMAGE_NAME_="${1:-}" local IMAGE_VERSION_="${2:-}" - if [[ $(docker image ls -q "$IMAGE_NAME_"":""$IMAGE_VERSION_" | wc -c ) -ne 0 ]] ; then - if [[ $(docker ps -a -q --filter "ancestor=""$IMAGE_NAME_"":""$IMAGE_VERSION_" | wc -c) -ne 0 ]]; then + if [[ $(docker image ls -q "${IMAGE_NAME_}"":""${IMAGE_VERSION_}" | wc -c ) -ne 0 ]] ; then + if [[ $(docker ps -a -q --filter "ancestor=""${IMAGE_NAME_}"":""${IMAGE_VERSION_}" | wc -c) -ne 0 ]]; then local CONTAINERS_ - mapfile -t CONTAINERS_ < <(docker ps -a -q --filter ancestor="$IMAGE_NAME_"":""$IMAGE_VERSION_" --format="{{.ID}}") + mapfile -t CONTAINERS_ < <(docker ps -a -q --filter ancestor="${IMAGE_NAME_}"":""${IMAGE_VERSION_}" --format="{{.ID}}") for CONTAINER_ID_ in "${CONTAINERS_[@]}" ; do - echo -e "$GREEN""$BOLD""Stopping ""$CONTAINER_ID_"" docker container""$NC" - docker stop "$CONTAINER_ID_" - echo -e "$GREEN""$BOLD""Remove ""$CONTAINER_ID_"" docker container""$NC" - docker container rm "$CONTAINER_ID_" -f + echo -e "${GREEN}""${BOLD}""Stopping ""${CONTAINER_ID_}"" docker container""${NC}" + docker stop "${CONTAINER_ID_}" + echo -e "${GREEN}""${BOLD}""Remove ""${CONTAINER_ID_}"" docker container""${NC}" + docker container rm "${CONTAINER_ID_}" -f done fi - echo -e "$GREEN$BOLD""Removing ""$IMAGE_NAME_"":""$IMAGE_VERSION_" "docker image""$NC\\n" - docker image rm "$IMAGE_NAME_"":""$IMAGE_VERSION_" -f + echo -e "${GREEN}${BOLD}""Removing ""${IMAGE_NAME_}"":""${IMAGE_VERSION_}" "docker image""${NC}\\n" + docker image rm "${IMAGE_NAME_}"":""${IMAGE_VERSION_}" -f fi } @@ -38,11 +38,11 @@ docker_network_rm(){ # removes docker networks by name local NET_NAME="${1:-}" local NET_ID="" - if docker network ls | grep -E "$NET_NAME"; then - echo -e "\n$GREEN""$BOLD""Found ""$NET_NAME"" - removing it""$NC" - NET_ID=$(docker network ls | grep -E "$NET_NAME" | awk '{print $1}') - echo -e "$GREEN""$BOLD""Remove ""$NET_NAME"" network""$NC" - docker network rm "$NET_ID" + if docker network ls | grep -E "${NET_NAME}"; then + echo -e "\n${GREEN}""${BOLD}""Found ""${NET_NAME}"" - removing it""${NC}" + NET_ID=$(docker network ls | grep -E "${NET_NAME}" | awk '{print $1}') + echo -e "${GREEN}""${BOLD}""Remove ""${NET_NAME}"" network""${NC}" + docker network rm "${NET_ID}" fi } @@ -50,20 +50,20 @@ copy_file(){ # check and copy file forcing overwrite local SOURCE_="${1:-}" local DESTINATION_="${2:-}" - if ! [[ -f "$SOURCE_" ]] ; then - echo -e "\\n$RED""Could not find ""$SOURCE_""$NC\\n" + if ! [[ -f "${SOURCE_}" ]] ; then + echo -e "\\n${RED}""Could not find ""${SOURCE_}""${NC}\\n" return 1 - elif ! [[ -d $(dirname "$DESTINATION_") ]] ; then - echo -e "\\n$RED""Could not find ""$DESTINATION_""$NC\\n" + elif ! [[ -d $(dirname "${DESTINATION_}") ]] ; then + echo -e "\\n${RED}""Could not find ""${DESTINATION_}""${NC}\\n" return 1 fi - cp -f "$SOURCE_" "$DESTINATION_" + cp -f "${SOURCE_}" "${DESTINATION_}" } enable_strict_mode() { local STRICT_MODE_="${1:-}" - if [[ "$STRICT_MODE_" -eq 1 ]]; then + if [[ "${STRICT_MODE_}" -eq 1 ]]; then # http://redsymbol.net/articles/unofficial-bash-strict-mode/ # https://github.com/tests-always-included/wick/blob/master/doc/bash-strict-mode.md # shellcheck disable=SC1091 @@ -83,7 +83,7 @@ enable_strict_mode() { check_docker_wsl() { # checks if service docker is running - echo -e "$BLUE""$BOLD""checking docker""$NC\\n" + echo -e "${BLUE}""${BOLD}""checking docker""${NC}\\n" service docker status } @@ -100,8 +100,8 @@ run_mysql_cmd() { PW_ENV=$(grep DATABASE_PASSWORD ./.env | sed 's/DATABASE\_PASSWORD\=//') USER_ENV=$(grep DATABASE_USER ./.env | sed 's/DATABASE\_USER\=//') HOST_ENV=$(grep DATABASE_HOST ./.env | sed 's/DATABASE\_HOST\=//') - echo -e "\n[""$BLUE JOB""$NC""] Running the following command: $SQL_COMMAND ""$NC" - mysql --host="$HOST_ENV" --user="$USER_ENV" --password="$PW_ENV" -e"$SQL_COMMAND" + echo -e "\n[""${BLUE} JOB""${NC}""] Running the following command: ${SQL_COMMAND} ""${NC}" + mysql --host="${HOST_ENV}" --user="${USER_ENV}" --password="${PW_ENV}" -e"${SQL_COMMAND}" } check_db() { @@ -111,32 +111,32 @@ check_db() { PW_ENV=$(grep DATABASE_PASSWORD ./.env | sed 's/DATABASE\_PASSWORD\=//') USER_ENV=$(grep DATABASE_USER ./.env | sed 's/DATABASE\_USER\=//') HOST_ENV=$(grep DATABASE_HOST ./.env | sed 's/DATABASE\_HOST\=//') - echo -e "\\n$ORANGE""$BOLD""checking database""$NC\\n""$BOLD=================================================================$NC" - echo -e "$BLUE""$BOLD""1. checking startup""$NC\\n" + echo -e "\\n${ORANGE}""${BOLD}""checking database""${NC}\\n""${BOLD}=================================================================${NC}" + echo -e "${BLUE}""${BOLD}""1. checking startup""${NC}\\n" if docker-compose -f ./docker-compose.yml up -d ; then - echo -e "$GREEN""$BOLD""Finished setup mysql and redis docker images""$NC" - add_to_env_history "$PW_ENV" "$(docker-compose ps -q embark_db)" + echo -e "${GREEN}""${BOLD}""Finished setup mysql and redis docker images""${NC}" + add_to_env_history "${PW_ENV}" "$(docker-compose ps -q embark_db)" else - echo -e "$ORANGE""$BOLD""Failed setup mysql and redis docker images""$NC" + echo -e "${ORANGE}""${BOLD}""Failed setup mysql and redis docker images""${NC}" exit 1 fi - echo -e "$BLUE""$BOLD""2. checking password""$NC\\n" - if ! mysql --host="$HOST_ENV" --user="$USER_ENV" --password="$PW_ENV" -e"quit"; then - echo -e "$ORANGE""$BOLD""[*] Retesting the mysql connection""$NC" + echo -e "${BLUE}""${BOLD}""2. checking password""${NC}\\n" + if ! mysql --host="${HOST_ENV}" --user="${USER_ENV}" --password="${PW_ENV}" -e"quit"; then + echo -e "${ORANGE}""${BOLD}""[*] Retesting the mysql connection""${NC}" sleep 35s if ! mysql --host="${HOST_ENV}" --user="${USER_ENV}" --password="${PW_ENV}" -e"quit"; then - echo -e "$ORANGE""$BOLD""Failed logging into database with password""$NC" + echo -e "${ORANGE}""${BOLD}""Failed logging into database with password""${NC}" echo -e "---------------------------------------------------------------------------" - echo -e "$CYAN""Old passwords are stored in the \"safe\" folder when uninstalling EMBArk""$NC\\n" - echo -e "$CYAN""You could try recoverying manually by overwriting your\".env\" file""$NC\\n" + echo -e "${CYAN}""Old passwords are stored in the \"safe\" folder when uninstalling EMBArk""${NC}\\n" + echo -e "${CYAN}""You could try recoverying manually by overwriting your\".env\" file""${NC}\\n" if [[ -f safe/history.env ]]; then - echo -e "$CYAN""The mysql-db was first started with the password(sha256sum): $(head -n1 ./safe/history.env | cut -d";" -f1) ""$NC\\n" - echo -e "$CYAN""And the password used was (sha256sum): $(echo "${PW_ENV}" | sha256sum)""$NC\\n" + echo -e "${CYAN}""The mysql-db was first started with the password(sha256sum): $(head -n1 ./safe/history.env | cut -d";" -f1) ""${NC}\\n" + echo -e "${CYAN}""And the password used was (sha256sum): $(echo "${PW_ENV}" | sha256sum)""${NC}\\n" fi exit 1 fi fi - echo -e "$GREEN""$BOLD""[+] Everything checks out""$NC\\n" + echo -e "${GREEN}""${BOLD}""[+] Everything checks out""${NC}\\n" } add_to_env_history(){ @@ -146,6 +146,6 @@ add_to_env_history(){ if ! [[ -d safe ]]; then mkdir safe fi - printf '%s;%s;\n' "$(echo "$PASSWORD_" | sha256sum)" "$CONTAINER_HASH_" >> ./safe/history.env + printf '%s;%s;\n' "$(echo "${PASSWORD_}" | sha256sum)" "${CONTAINER_HASH_}" >> ./safe/history.env } diff --git a/helper/wickStrictModeFail.sh b/helper/wickStrictModeFail.sh index 61d8dd055..93711ade1 100644 --- a/helper/wickStrictModeFail.sh +++ b/helper/wickStrictModeFail.sh @@ -16,9 +16,9 @@ wickStrictModeFail() ( set +x local argsList argsLeft i nextArg - echo -e "Error detected - status code $ORANGE$1$NC" - echo -e "Command: $ORANGE$BASH_COMMAND$NC" - echo -e "Location: $ORANGE${BASH_SOURCE[1]:-unknown}$NC, line $ORANGE${BASH_LINENO[0]:-unknown}$NC" + echo -e "Error detected - status code ${ORANGE}${1}${NC}" + echo -e "Command: ${ORANGE}${BASH_COMMAND}${NC}" + echo -e "Location: ${ORANGE}${BASH_SOURCE[1]:-unknown}${NC}, line ${ORANGE}${BASH_LINENO[0]:-unknown}${NC}" if [[ ${#PIPESTATUS[@]} -gt 1 ]]; then echo "Pipe status: " "${PIPESTATUS[@]}" @@ -27,19 +27,19 @@ wickStrictModeFail() ( i=$# nextArg=$# - if [[ $i -lt ${#BASH_LINENO[@]} ]]; then + if [[ ${i} -lt ${#BASH_LINENO[@]} ]]; then echo "Stack Trace:" else echo "Stack trace is unavailable" fi - while [[ $i -lt ${#BASH_LINENO[@]} ]]; do + while [[ ${i} -lt ${#BASH_LINENO[@]} ]]; do argsList=() - if [[ ${#BASH_ARGC[@]} -gt $i ]] && [[ ${#BASH_ARGV[@]} -ge $(( nextArg + BASH_ARGC[i] )) ]]; then + if [[ ${#BASH_ARGC[@]} -gt ${i} ]] && [[ ${#BASH_ARGV[@]} -ge $(( nextArg + BASH_ARGC[i] )) ]]; then for (( argsLeft = BASH_ARGC[i]; argsLeft; --argsLeft )); do # Note: this reverses the order on purpose - argsList[$argsLeft]=${BASH_ARGV[nextArg]} + argsList[${argsLeft}]=${BASH_ARGV[nextArg]} (( nextArg ++ )) done @@ -56,7 +56,7 @@ wickStrictModeFail() ( argsList="" fi - echo " [$i] ${FUNCNAME[i]:+${FUNCNAME[i]}(): }${BASH_SOURCE[i]}, line ${BASH_LINENO[i - 1]} -> ${FUNCNAME[i]:-${BASH_SOURCE[i]##*/}}$argsList" + echo " [${i}] ${FUNCNAME[i]:+${FUNCNAME[i]}(): }${BASH_SOURCE[i]}, line ${BASH_LINENO[i - 1]} -> ${FUNCNAME[i]:-${BASH_SOURCE[i]##*/}}${argsList}" (( i ++ )) done echo -e "\n${BLUE}${BOLD}Important: Consider filling out a bug report at https://github.com/e-m-b-a/embark/issues${NC}\n" diff --git a/import-DB.sh b/import-DB.sh index 89aea6fae..83a66c2a4 100755 --- a/import-DB.sh +++ b/import-DB.sh @@ -26,23 +26,23 @@ import_helper() local HELPER_COUNT=0 local HELPER_FILE="" local HELP_DIR='helper' - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } ALT_BACKUP_FILE="${1:-}" export BACKUP_FILE="" if [[ -n "${ALT_BACKUP_FILE}" ]]; then if ! [[ -f "${ALT_BACKUP_FILE}" ]]; then - echo -e "\\n$RED""Error with input for BACKUP FILE!""$NC\\n" + echo -e "\\n${RED}""Error with input for BACKUP FILE!""${NC}\\n" exit 1 fi BACKUP_FILE="${ALT_BACKUP_FILE}" @@ -50,10 +50,10 @@ else BACKUP_FILE="$(find . -type f -iname "full-backup-*.tar" | sort -n | tail -n 1)" fi -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 -if ! [[ $EUID -eq 0 ]] ; then - echo -e "\\n$RED""Run script with root permissions!""$NC\\n" +if ! [[ ${EUID} -eq 0 ]] ; then + echo -e "\\n${RED}""Run script with root permissions!""${NC}\\n" exit 1 fi @@ -67,4 +67,4 @@ rsync -a --progress ./.embark_db_backup/ ./embark_db/ rm -rf ./.embark_db_backup/ -echo -e "\\n""==> ""$GREEN""Import successful""$NC" \ No newline at end of file +echo -e "\\n""==> ""${GREEN}""Import successful""${NC}" diff --git a/installer.sh b/installer.sh index 61b6648c7..e86709e33 100755 --- a/installer.sh +++ b/installer.sh @@ -38,33 +38,33 @@ export BOLD='\033[1m' export NC='\033[0m' # no print_help(){ - echo -e "\\n""$CYAN""USAGE""$NC" - echo -e "$CYAN-h$NC Print this help message" - echo -e "$CYAN-d$NC EMBArk default installation" - echo -e "$CYAN-F$NC Installation of EMBArk for developers" - echo -e "$CYAN-e$NC Install EMBA only" - echo -e "$CYAN-s$NC Installation without EMBA (use in combination with d/F)" + echo -e "\\n""${CYAN}""USAGE""${NC}" + echo -e "${CYAN}-h${NC} Print this help message" + echo -e "${CYAN}-d${NC} EMBArk default installation" + echo -e "${CYAN}-F${NC} Installation of EMBArk for developers" + echo -e "${CYAN}-e${NC} Install EMBA only" + echo -e "${CYAN}-s${NC} Installation without EMBA (use in combination with d/F)" echo -e "---------------------------------------------------------------------------" - echo -e "$CYAN-U$NC Uninstall EMBArk" - echo -e "$CYAN-rd$NC Reinstallation of EMBArk with all dependencies" - echo -e "$CYAN-rF$NC Reinstallation of EMBArk with all dependencies in Developer-mode" - echo -e "$RED ! Both options delete all Database-files as well !""$NC" + echo -e "${CYAN}-U${NC} Uninstall EMBArk" + echo -e "${CYAN}-rd${NC} Reinstallation of EMBArk with all dependencies" + echo -e "${CYAN}-rF${NC} Reinstallation of EMBArk with all dependencies in Developer-mode" + echo -e "${RED} ! Both options delete all Database-files as well !""${NC}" } import_helper(){ local HELPERS=() local HELPER_COUNT=0 local HELPER_FILE="" - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } # Source: https://stackoverflow.com/questions/4023830/how-to-compare-two-strings-in-dot-separated-version-format-in-bash @@ -92,8 +92,8 @@ write_env(){ if [[ -d safe ]]; then mapfile -d '' ENV_FILES < <(find ./safe -iname "*.env" -print0 2> /dev/null) if [[ ${#ENV_FILES[@]} -gt 0 ]] && [[ -f safe/history.env ]]; then - echo -e "$ORANGE""$BOLD""Using old env file""$NC" - # check which env file was the last one where $(echo "$PASSWORD_" | sha256sum) matches the first line and entry + echo -e "${ORANGE}""${BOLD}""Using old env file""${NC}" + # check which env file was the last one where $(echo "${PASSWORD_}" | sha256sum) matches the first line and entry LAST_PW_HASH="$(grep -v "$(echo "" | sha256sum)" safe/history.env | tail -n 1 | cut -d";" -f1)" for FILE_ in "${ENV_FILES[@]}"; do CHECK_PW="$(grep "DATABASE_PASSWORD=" "${FILE_}" | sed -e "s/^DATABASE_PASSWORD=//" )" @@ -107,59 +107,59 @@ write_env(){ fi if [[ -z ${DJANGO_SECRET_KEY} ]] || [[ -z ${DJANGO_SECRET_KEY} ]]; then - echo -e "$ORANGE""$BOLD""Did not find safed passwords""$NC" + echo -e "${ORANGE}""${BOLD}""Did not find safed passwords""${NC}" DJANGO_SECRET_KEY=$(python3.10 -c 'from django.core.management.utils import get_random_secret_key; print(get_random_secret_key())') RANDOM_PW=$(openssl rand -base64 12) fi - echo -e "$ORANGE""$BOLD""Creating a EMBArk configuration file .env""$NC" + echo -e "${ORANGE}""${BOLD}""Creating a EMBArk configuration file .env""${NC}" { echo "DATABASE_NAME=embark" echo "DATABASE_USER=embark" - echo "DATABASE_PASSWORD=$RANDOM_PW" + echo "DATABASE_PASSWORD=${RANDOM_PW}" echo "DATABASE_HOST=172.22.0.5" echo "DATABASE_PORT=3306" - echo "MYSQL_PASSWORD=$RANDOM_PW" + echo "MYSQL_PASSWORD=${RANDOM_PW}" echo "MYSQL_USER=embark" echo "MYSQL_DATABASE=embark" echo "REDIS_HOST=172.22.0.8" echo "REDIS_PORT=7777" - echo "SECRET_KEY=$DJANGO_SECRET_KEY" - echo "DJANGO_SUPERUSER_USERNAME=$SUPER_USER" - echo "DJANGO_SUPERUSER_EMAIL=$SUPER_EMAIL" - echo "DJANGO_SUPERUSER_PASSWORD=$SUPER_PW" + echo "SECRET_KEY=${DJANGO_SECRET_KEY}" + echo "DJANGO_SUPERUSER_USERNAME=${SUPER_USER}" + echo "DJANGO_SUPERUSER_EMAIL=${SUPER_EMAIL}" + echo "DJANGO_SUPERUSER_PASSWORD=${SUPER_PW}" echo "PYTHONPATH=${PWD}:${PWD}/embark:/var/www/:/var/www/embark" } > .env chmod 600 .env } install_emba(){ - echo -e "\n$GREEN""$BOLD""Installation of the firmware scanner EMBA on host""$NC" + echo -e "\n${GREEN}""${BOLD}""Installation of the firmware scanner EMBA on host""${NC}" if git submodule status emba | grep --quiet '^-'; then sudo -u "${SUDO_USER:-${USER}}" git submodule init emba fi sudo -u "${SUDO_USER:-${USER}}" git submodule update --remote - sudo -u "${SUDO_USER:-${USER}}" git config --global --add safe.directory "$PWD"/emba + sudo -u "${SUDO_USER:-${USER}}" git config --global --add safe.directory "${PWD}"/emba cd emba ./installer.sh -d || ( echo "Could not install EMBA" && exit 1 ) cd .. if ! (cd emba && ./emba -d 1); then - echo -e "\n$RED""$BOLD""EMBA installation failed""$NC" + echo -e "\n${RED}""${BOLD}""EMBA installation failed""${NC}" exit 1 fi chown -R "${SUDO_USER:-${USER}}" emba - echo -e "\n""--------------------------------------------------------------------""$NC" + echo -e "\n""--------------------------------------------------------------------""${NC}" } create_ca (){ # FIXME could use some work - echo -e "\n$GREEN""$BOLD""Creating SSL Cert""$NC" + echo -e "\n${GREEN}""${BOLD}""Creating SSL Cert""${NC}" if ! [[ -d cert ]]; then sudo -u "${SUDO_USER:-${USER}}" git checkout -- cert fi cd cert || exit 1 if [[ -f embark.local.csr ]] || [[ -f embark-ws.local.csr ]] || [[ -f embark.local.crt ]] || [[ -f embark-ws.local.crt ]]; then - echo -e "\n$GREEN""$BOLD""Certs already generated, skipping""$NC" + echo -e "\n${GREEN}""${BOLD}""Certs already generated, skipping""${NC}" else # create CA openssl genrsa -out rootCA.key 4096 @@ -177,16 +177,16 @@ create_ca (){ } dns_resolve(){ - echo -e "\n$GREEN""$BOLD""Install hostnames for local dns-resolve""$NC" + echo -e "\n${GREEN}""${BOLD}""Install hostnames for local dns-resolve""${NC}" if ! grep -q "embark.local" /etc/hosts ; then printf "0.0.0.0 embark.local\n" >>/etc/hosts else - echo -e "\n$ORANGE""$BOLD""hostname already in use!""$NC" + echo -e "\n${ORANGE}""${BOLD}""hostname already in use!""${NC}" fi } reset_docker(){ - echo -e "\\n$GREEN""$BOLD""Reset EMBArk docker images""$NC\\n" + echo -e "\\n${GREEN}""${BOLD}""Reset EMBArk docker images""${NC}\\n" # EMBArk docker_image_rm "mysql" "latest" @@ -205,7 +205,7 @@ reset_docker(){ install_debs(){ local DOCKER_COMP_VER="" - echo -e "\n$GREEN""$BOLD""Install debian packages for EMBArk installation""$NC" + echo -e "\n${GREEN}""${BOLD}""Install debian packages for EMBArk installation""${NC}" apt-get update -y # Git if ! command -v git > /dev/null ; then @@ -228,8 +228,8 @@ install_debs(){ apt-get install build-essential fi # Docker - if [[ "$WSL" -eq 1 ]]; then - echo -e "\n${ORANGE}WARNING: If you are using WSL2, disable docker integration from the docker-desktop daemon!$NC" + if [[ "${WSL}" -eq 1 ]]; then + echo -e "\n${ORANGE}WARNING: If you are using WSL2, disable docker integration from the docker-desktop daemon!${NC}" read -p "Fix docker stuff, then continue. Press any key to continue ..." -n1 -s -r fi if ! command -v docker > /dev/null ; then @@ -243,9 +243,9 @@ install_debs(){ fi else DOCKER_COMP_VER=$(docker-compose -v | grep version | awk '{print $3}' | tr -d ',') - if [[ $(version "$DOCKER_COMP_VER") -lt $(version "1.28.5") ]]; then - echo -e "\n${ORANGE}WARNING: compatibility of the used docker-compose version is unknown!$NC" - echo -e "\n${ORANGE}Please consider updating your docker-compose installation to version 1.28.5 or later.$NC" + if [[ $(version "${DOCKER_COMP_VER}") -lt $(version "1.28.5") ]]; then + echo -e "\n${ORANGE}WARNING: compatibility of the used docker-compose version is unknown!${NC}" + echo -e "\n${ORANGE}Please consider updating your docker-compose installation to version 1.28.5 or later.${NC}" read -p "If you know what you are doing you can press any key to continue ..." -n1 -s -r fi fi @@ -261,15 +261,15 @@ install_debs(){ } install_daemon(){ - echo -e "\n$GREEN""$BOLD""Install embark daemon""$NC" - sed -i "s|{\$EMBARK_ROOT_DIR}|$PWD|g" embark.service + echo -e "\n${GREEN}""${BOLD}""Install embark daemon""${NC}" + sed -i "s|{\$EMBARK_ROOT_DIR}|${PWD}|g" embark.service if ! [[ -e /etc/systemd/system/embark.service ]] ; then - ln -s "$PWD"/embark.service /etc/systemd/system/embark.service + ln -s "${PWD}"/embark.service /etc/systemd/system/embark.service fi } uninstall_daemon(){ - echo -e "\n$ORANGE""$BOLD""Uninstalling embark daemon""$NC" + echo -e "\n${ORANGE}""${BOLD}""Uninstalling embark daemon""${NC}" if [[ -e /etc/systemd/system/embark.service ]] ; then systemctl stop embark.service systemctl disable embark.service @@ -279,10 +279,10 @@ uninstall_daemon(){ } install_embark_default(){ - echo -e "\n$GREEN""$BOLD""Installation of the firmware scanning environment EMBArk""$NC" + echo -e "\n${GREEN}""${BOLD}""Installation of the firmware scanning environment EMBArk""${NC}" - if [[ "$WSL" -eq 1 ]]; then - echo -e "$RED""$BOLD""EMBArk currently does not support WSL in default mode. (only in Dev-mode)""$NC" + if [[ "${WSL}" -eq 1 ]]; then + echo -e "${RED}""${BOLD}""EMBArk currently does not support WSL in default mode. (only in Dev-mode)""${NC}" fi #debs @@ -337,7 +337,7 @@ install_embark_default(){ # download externals if ! [[ -d ./embark/static/external ]]; then - echo -e "\n$GREEN""$BOLD""Downloading of external files, e.g. jQuery, for the offline usability of EMBArk""$NC" + echo -e "\n${GREEN}""${BOLD}""Downloading of external files, e.g. jQuery, for the offline usability of EMBArk""${NC}" mkdir -p ./embark/static/external/{scripts,css} wget -O ./embark/static/external/scripts/jquery.js https://code.jquery.com/jquery-3.6.0.min.js wget -O ./embark/static/external/scripts/confirm.js https://cdnjs.cloudflare.com/ajax/libs/jquery-confirm/3.3.2/jquery-confirm.min.js @@ -355,7 +355,7 @@ install_embark_default(){ # write env-vars into ./.env write_env - if [[ "$WSL" -eq 1 ]]; then + if [[ "${WSL}" -eq 1 ]]; then check_docker_wsl fi @@ -367,12 +367,12 @@ install_embark_default(){ systemctl start embark.service check_db docker-compose stop - echo -e "$GREEN""$BOLD""Ready to use \$sudo ./run-server.sh ""$NC" - echo -e "$GREEN""$BOLD""Which starts the server on (0.0.0.0) port 80 ""$NC" + echo -e "${GREEN}""${BOLD}""Ready to use \$sudo ./run-server.sh ""${NC}" + echo -e "${GREEN}""${BOLD}""Which starts the server on (0.0.0.0) port 80 ""${NC}" } install_embark_dev(){ - echo -e "\n$GREEN""$BOLD""Building Developent-Enviroment for EMBArk""$NC" + echo -e "\n${GREEN}""${BOLD}""Building Developent-Enviroment for EMBArk""${NC}" # apt packages apt-get install -y npm pycodestyle python3-pylint-django default-libmysqlclient-dev build-essential bandit yamllint mysql-client-core-8.0 # get geckodriver @@ -388,9 +388,9 @@ install_embark_dev(){ pip3 install pipenv #Add user nosudo - echo "${SUDO_USER:-${USER}}"" ALL=(ALL) NOPASSWD: ""$PWD""/emba/emba" | EDITOR='tee -a' visudo + echo "${SUDO_USER:-${USER}}"" ALL=(ALL) NOPASSWD: ""${PWD}""/emba/emba" | EDITOR='tee -a' visudo echo "${SUDO_USER:-${USER}}"" ALL=(ALL) NOPASSWD: /bin/pkill" | EDITOR='tee -a' visudo - echo "root ALL=(ALL) NOPASSWD: ""$PWD""/emba/emba" | EDITOR='tee -a' visudo + echo "root ALL=(ALL) NOPASSWD: ""${PWD}""/emba/emba" | EDITOR='tee -a' visudo echo "root ALL=(ALL) NOPASSWD: /bin/pkill" | EDITOR='tee -a' visudo @@ -416,7 +416,7 @@ install_embark_dev(){ # download externals if ! [[ -d ./embark/static/external ]]; then - echo -e "\n$GREEN""$BOLD""Downloading of external files, e.g. jQuery, for the offline usability of EMBArk""$NC" + echo -e "\n${GREEN}""${BOLD}""Downloading of external files, e.g. jQuery, for the offline usability of EMBArk""${NC}" mkdir -p ./embark/static/external/{scripts,css} wget -O ./embark/static/external/scripts/jquery.js https://code.jquery.com/jquery-3.6.0.min.js wget -O ./embark/static/external/scripts/confirm.js https://cdnjs.cloudflare.com/ajax/libs/jquery-confirm/3.3.2/jquery-confirm.min.js @@ -441,22 +441,22 @@ install_embark_dev(){ check_db docker-compose stop - echo -e "$GREEN""$BOLD""Ready to use \$sudo ./dev-tools/debug-server-start.sh""$NC" - echo -e "$GREEN""$BOLD""Or use otherwise""$NC" + echo -e "${GREEN}""${BOLD}""Ready to use \$sudo ./dev-tools/debug-server-start.sh""${NC}" + echo -e "${GREEN}""${BOLD}""Or use otherwise""${NC}" } uninstall (){ - echo -e "[+]$CYAN""$BOLD""Uninstalling EMBArk""$NC" + echo -e "[+]${CYAN}""${BOLD}""Uninstalling EMBArk""${NC}" # check for changes if [[ $(git status --porcelain --untracked-files=no --ignore-submodules=all) ]]; then # Changes - echo -e "[!!]$RED""$BOLD""Changes detected - please stash or commit them $ORANGE( \$git stash )""$NC" + echo -e "[!!]${RED}""${BOLD}""Changes detected - please stash or commit them ${ORANGE}( \$git stash )""${NC}" git status exit 1 fi # delete directories - echo -e "$ORANGE""$BOLD""Delete Directories""$NC" + echo -e "${ORANGE}""${BOLD}""Delete Directories""${NC}" if [[ -d /var/www ]]; then rm -Rv /var/www fi @@ -481,17 +481,17 @@ uninstall (){ if [[ -d ./logs ]]; then rm -Rvf ./logs fi - if [[ "$REFORCE" -eq 0 ]]; then + if [[ "${REFORCE}" -eq 0 ]]; then # user-files if [[ -d ./emba_logs ]]; then - echo -e "$RED""$BOLD""Do you wish to remove the EMBA-Logs (and backups)""$NC" + echo -e "${RED}""${BOLD}""Do you wish to remove the EMBA-Logs (and backups)""${NC}" rm -RIv ./emba_logs fi if [[ -d ./embark_db ]]; then - echo -e "$RED""$BOLD""Do you wish to remove the database(and backups)""$NC" + echo -e "${RED}""${BOLD}""Do you wish to remove the database(and backups)""${NC}" rm -RIv ./embark_db if [[ -f ./safe/history.env ]]; then - echo -e "$RED""$BOLD""Moved old history file""$NC" + echo -e "${RED}""${BOLD}""Moved old history file""${NC}" mv --force ./safe/history.env ./safe/old_env_history fi fi @@ -499,7 +499,7 @@ uninstall (){ # delete user www-embark and reset visudo - echo -e "$ORANGE""$BOLD""Delete user""$NC" + echo -e "${ORANGE}""${BOLD}""Delete user""${NC}" if id -u www-embark &>/dev/null ; then userdel www-embark @@ -507,28 +507,28 @@ uninstall (){ # remove all emba/embark NOPASSWD entries into sudoer file if grep -qE "NOPASSWD\:.*\/emba\/emba" /etc/sudoers ; then - echo -e "$ORANGE""$BOLD""Deleting EMBA NOPASSWD entries""$NC" + echo -e "${ORANGE}""${BOLD}""Deleting EMBA NOPASSWD entries""${NC}" sed -i '/NOPASSWD\:.*\/emba\/emba/d' /etc/sudoers fi if grep -qE "NOPASSWD\:.*\/bin\/pkill" /etc/sudoers ; then - echo -e "$ORANGE""$BOLD""Deleting pkill NOPASSWD entries""$NC" + echo -e "${ORANGE}""${BOLD}""Deleting pkill NOPASSWD entries""${NC}" sed -i '/NOPASSWD\:.*\/bin\/pkill/d' /etc/sudoers fi # delete .env - echo -e "$ORANGE""$BOLD""Delete env""$NC" + echo -e "${ORANGE}""${BOLD}""Delete env""${NC}" if [[ -f ./.env ]]; then rm -Rvf ./.env fi # delete shared volumes and migrations - echo -e "$ORANGE""$BOLD""Delete migration-files""$NC" + echo -e "${ORANGE}""${BOLD}""Delete migration-files""${NC}" find . -path "*/migrations/*.py" -not -name "__init__.py" -delete find . -path "*/migrations/*.pyc" -delete # delete all docker interfaces and containers + images reset_docker - echo -e "$ORANGE""$BOLD""Consider running " "$CYAN""\$docker system prune""$NC" + echo -e "${ORANGE}""${BOLD}""Consider running " "${CYAN}""\$docker system prune""${NC}" # delete/uninstall submodules # emba @@ -536,11 +536,11 @@ uninstall (){ rm -r ./emba/external/ fi # all submodules - if [[ $REFORCE -eq 1 ]]; then + if [[ ${REFORCE} -eq 1 ]]; then sudo -u "${SUDO_USER:-${USER}}" git submodule status else if [[ $(sudo -u "${SUDO_USER:-${USER}}" git submodule foreach git status --porcelain --untracked-files=no) ]]; then - echo -e "[!!]$RED""$BOLD""Submodule changes detected - please commit them...otherwise they will be lost""$NC" + echo -e "[!!]${RED}""${BOLD}""Submodule changes detected - please commit them...otherwise they will be lost""${NC}" read -p "If you know what you are doing you can press any key to continue ..." -n1 -s -r fi sudo -u "${SUDO_USER:-${USER}}" git submodule foreach git reset --hard @@ -549,7 +549,7 @@ uninstall (){ fi # stop&reset daemon - if [[ "$WSL" -ne 1 ]]; then + if [[ "${WSL}" -ne 1 ]]; then uninstall_daemon systemctl daemon-reload fi @@ -561,19 +561,19 @@ uninstall (){ sudo -u "${SUDO_USER:-${USER}}" git checkout HEAD -- cert # final - if [[ "$REFORCE" -eq 0 ]]; then + if [[ "${REFORCE}" -eq 0 ]]; then sudo -u "${SUDO_USER:-${USER}}" git reset rm -r ./safe fi - echo -e "$ORANGE""$BOLD""Consider ""$CYAN""\$git pull""$ORANGE""$BOLD"" and ""$CYAN""\$git clean""$NC" + echo -e "${ORANGE}""${BOLD}""Consider ""${CYAN}""\$git pull""${ORANGE}""${BOLD}"" and ""${CYAN}""\$git clean""${NC}" } -echo -e "\\n$ORANGE""$BOLD""EMBArk Installer""$NC\\n""$BOLD=================================================================$NC" -echo -e "$ORANGE""$BOLD""WARNING: This script can harm your environment!""$NC\n" +echo -e "\\n${ORANGE}""${BOLD}""EMBArk Installer""${NC}\\n""${BOLD}=================================================================${NC}" +echo -e "${ORANGE}""${BOLD}""WARNING: This script can harm your environment!""${NC}\n" import_helper -if [[ "$STRICT_MODE" -eq 1 ]]; then +if [[ "${STRICT_MODE}" -eq 1 ]]; then # http://redsymbol.net/articles/unofficial-bash-strict-mode/ # https://github.com/tests-always-included/wick/blob/master/doc/bash-strict-mode.md set -e # Exit immediately if a command exits with a non-zero status @@ -586,76 +586,76 @@ if [[ "$STRICT_MODE" -eq 1 ]]; then fi if [ "$#" -ne 1 ]; then - echo -e "$RED""$BOLD""Invalid number of arguments""$NC" + echo -e "${RED}""${BOLD}""Invalid number of arguments""${NC}" print_help exit 1 fi while getopts esFUrdDSh OPT ; do - case $OPT in + case ${OPT} in e) export EMBA_ONLY=1 - echo -e "$GREEN""$BOLD""Install only emba""$NC" + echo -e "${GREEN}""${BOLD}""Install only emba""${NC}" ;; s) export NO_EMBA=1 - echo -e "$GREEN""$BOLD""Install without emba""$NC" + echo -e "${GREEN}""${BOLD}""Install without emba""${NC}" ;; F) export DEV=1 - echo -e "$GREEN""$BOLD""Building Development-Enviroment""$NC" + echo -e "${GREEN}""${BOLD}""Building Development-Enviroment""${NC}" ;; U) export UNINSTALL=1 - echo -e "$GREEN""$BOLD""Uninstall EMBArk""$NC" + echo -e "${GREEN}""${BOLD}""Uninstall EMBArk""${NC}" ;; r) export UNINSTALL=1 export REFORCE=1 - echo -e "$GREEN""$BOLD""Re-Install all dependecies while keeping user-files""$NC" + echo -e "${GREEN}""${BOLD}""Re-Install all dependecies while keeping user-files""${NC}" ;; d) export DEFAULT=1 - echo -e "$GREEN""$BOLD""Default installation of EMBArk""$NC" + echo -e "${GREEN}""${BOLD}""Default installation of EMBArk""${NC}" ;; S) export STRICT_MODE=1 - echo -e "$GREEN""$BOLD""Strict-mode enabled""$NC" + echo -e "${GREEN}""${BOLD}""Strict-mode enabled""${NC}" ;; h) print_help exit 0 ;; *) - echo -e "$RED""$BOLD""Invalid option""$NC" + echo -e "${RED}""${BOLD}""Invalid option""${NC}" print_help exit 1 ;; esac done -enable_strict_mode $STRICT_MODE +enable_strict_mode ${STRICT_MODE} # WSL/OS version check # WSL support - currently experimental! if grep -q -i wsl /proc/version; then - echo -e "\n${ORANGE}INFO: System running in WSL environment!$NC" - echo -e "\n${ORANGE}INFO: WSL is currently experimental!$NC" - echo -e "\n${ORANGE}INFO: Ubuntu 22.04 is required for WSL!$NC" + echo -e "\n${ORANGE}INFO: System running in WSL environment!${NC}" + echo -e "\n${ORANGE}INFO: WSL is currently experimental!${NC}" + echo -e "\n${ORANGE}INFO: Ubuntu 22.04 is required for WSL!${NC}" read -p "If you know what you are doing you can press any key to continue ..." -n1 -s -r WSL=1 fi -if [[ $EUID -ne 0 ]]; then - echo -e "\\n$RED""Run EMBArk installation script with root permissions!""$NC\\n" +if [[ ${EUID} -ne 0 ]]; then + echo -e "\\n${RED}""Run EMBArk installation script with root permissions!""${NC}\\n" print_help exit 1 fi -if [[ $REFORCE -eq 1 ]] && [[ $UNINSTALL -eq 1 ]]; then +if [[ ${REFORCE} -eq 1 ]] && [[ ${UNINSTALL} -eq 1 ]]; then save_old_env uninstall -elif [[ $UNINSTALL -eq 1 ]]; then +elif [[ ${UNINSTALL} -eq 1 ]]; then save_old_env uninstall exit 0 @@ -664,17 +664,17 @@ fi install_debs # mark dir as safe for git -sudo -u "${SUDO_USER:-${USER}}" git config --global --add safe.directory "$PWD" +sudo -u "${SUDO_USER:-${USER}}" git config --global --add safe.directory "${PWD}" -if [[ "$NO_EMBA" -eq 0 ]]; then +if [[ "${NO_EMBA}" -eq 0 ]]; then install_emba fi -if [[ "$EMBA_ONLY" -eq 1 ]]; then +if [[ "${EMBA_ONLY}" -eq 1 ]]; then exit 0 fi -if [[ $DEFAULT -eq 1 ]]; then +if [[ ${DEFAULT} -eq 1 ]]; then install_embark_default -elif [[ $DEV -eq 1 ]]; then +elif [[ ${DEV} -eq 1 ]]; then install_embark_dev fi exit 0 diff --git a/run-server.sh b/run-server.sh index f84526092..3ee7e1748 100755 --- a/run-server.sh +++ b/run-server.sh @@ -36,25 +36,25 @@ import_helper() local HELPERS=() local HELPER_COUNT=0 local HELPER_FILE="" - mapfile -d '' HELPERS < <(find "$HELP_DIR" -iname "helper_embark_*.sh" -print0 2> /dev/null) + mapfile -d '' HELPERS < <(find "${HELP_DIR}" -iname "helper_embark_*.sh" -print0 2> /dev/null) for HELPER_FILE in "${HELPERS[@]}" ; do - if ( file "$HELPER_FILE" | grep -q "shell script" ) && ! [[ "$HELPER_FILE" =~ \ |\' ]] ; then + if ( file "${HELPER_FILE}" | grep -q "shell script" ) && ! [[ "${HELPER_FILE}" =~ \ |\' ]] ; then # https://github.com/koalaman/shellcheck/wiki/SC1090 # shellcheck source=/dev/null - source "$HELPER_FILE" + source "${HELPER_FILE}" (( HELPER_COUNT+=1 )) fi done - echo -e "\\n""==> ""$GREEN""Imported ""$HELPER_COUNT"" necessary files""$NC\\n" + echo -e "\\n""==> ""${GREEN}""Imported ""${HELPER_COUNT}"" necessary files""${NC}\\n" } cleaner() { pkill -u root daphne - pkill -u root "$PWD"/emba/emba + pkill -u root "${PWD}"/emba/emba pkill -u root runapscheduler - fuser -k "$HTTP_PORT"/tcp - fuser -k "$HTTPS_PORT"/tcp + fuser -k "${HTTP_PORT}"/tcp + fuser -k "${HTTPS_PORT}"/tcp fuser -k 8000/tcp fuser -k 8001/tcp @@ -68,73 +68,73 @@ cleaner() { } # main -echo -e "\\n$ORANGE""$BOLD""EMBArk Startup""$NC\\n""$BOLD=================================================================$NC" +echo -e "\\n${ORANGE}""${BOLD}""EMBArk Startup""${NC}\\n""${BOLD}=================================================================${NC}" while getopts "ha:" OPT ; do - case $OPT in + case ${OPT} in h) - echo -e "\\n""$CYAN""USAGE""$NC" - echo -e "$CYAN-h$NC Print this help message" - echo -e "$CYAN-a $NC Add a server Domain-name alias" + echo -e "\\n""${CYAN}""USAGE""${NC}" + echo -e "${CYAN}-h${NC} Print this help message" + echo -e "${CYAN}-a ${NC} Add a server Domain-name alias" echo -e "---------------------------------------------------------------------------" if ip addr show eth0 &>/dev/null ; then IP=$(ip addr show eth0 | grep "inet\b" | awk '{print $2}' | cut -d/ -f1) - echo -e "$GREEN Suggestion:$NC sudo ./run-server.sh -a $IP" - echo -e "$GREEN nslookup helper:$NC" - nslookup "$IP" + echo -e "${GREEN} Suggestion:${NC} sudo ./run-server.sh -a ${IP}" + echo -e "${GREEN} nslookup helper:${NC}" + nslookup "${IP}" fi exit 0 ;; a) - SERVER_ALIAS+=("$OPTARG") + SERVER_ALIAS+=("${OPTARG}") WSGI_FLAGS+=(--server-alias "${OPTARG}") ;; :) - echo -e "$CYAN Usage: [-a ] $NC" + echo -e "${CYAN} Usage: [-a ] ${NC}" exit 1 ;; *) - echo -e "\\n$ORANGE""$BOLD""No Alias set""$NC\\n" + echo -e "\\n${ORANGE}""${BOLD}""No Alias set""${NC}\\n" ;; esac done # Alias if [[ ${#SERVER_ALIAS[@]} -ne 0 ]]; then - echo -e "$GREEN Server-alias:$NC" + echo -e "${GREEN} Server-alias:${NC}" for VAR in "${SERVER_ALIAS[@]}"; do - echo "[*] $VAR" + echo "[*] ${VAR}" done fi -cd "$(dirname "$0")" || exit 1 +cd "$(dirname "${0}")" || exit 1 import_helper -enable_strict_mode "$STRICT_MODE" +enable_strict_mode "${STRICT_MODE}" set -a trap cleaner INT -if ! [[ $EUID -eq 0 ]] ; then - echo -e "\\n$RED""Run Server script with root permissions!""$NC\\n" +if ! [[ ${EUID} -eq 0 ]] ; then + echo -e "\\n${RED}""Run Server script with root permissions!""${NC}\\n" exit 1 fi # check emba -echo -e "$BLUE""$BOLD""checking EMBA""$NC" +echo -e "${BLUE}""${BOLD}""checking EMBA""${NC}" if ! [[ -d ./emba ]]; then - echo -e "$RED""$BOLD""You are using the wrong installation and missing the EMBA subdirectory""$NC" + echo -e "${RED}""${BOLD}""You are using the wrong installation and missing the EMBA subdirectory""${NC}" fi -if ! (cd "$PWD"/emba && ./emba -d 1); then - echo -e "$BLUE""Trying auto-maintain""$NC" +if ! (cd "${PWD}"/emba && ./emba -d 1); then + echo -e "${BLUE}""Trying auto-maintain""${NC}" # automaintain if ! [[ -d ./emba ]]; then - echo -e "$RED""EMBA not installed""$NC" + echo -e "${RED}""EMBA not installed""${NC}" exit 1 fi cd ./emba || exit 1 systemctl restart NetworkManager docker ./emba -d 1 1>/dev/null if [[ $? -eq 1 ]]; then - echo -e "$RED""EMBA is not configured correctly""$NC" + echo -e "${RED}""EMBA is not configured correctly""${NC}" exit 1 fi cd .. || exit 1 @@ -142,7 +142,7 @@ fi # check venv if ! [[ -d /var/www/.venv ]]; then - echo -e "$RED""$BOLD""Pip-enviroment not found!""$NC" + echo -e "${RED}""${BOLD}""Pip-enviroment not found!""${NC}" exit 1 fi if ! nc -zw1 google.com 443 &>/dev/null ; then @@ -164,9 +164,9 @@ if ! [[ -d ./docker_logs ]]; then fi # container-logs (2 jobs) -echo -e "\n[""$BLUE JOB""$NC""] Redis logs are copied to ./docker_logs/redis.log" +echo -e "\n[""${BLUE} JOB""${NC}""] Redis logs are copied to ./docker_logs/redis.log" docker container logs embark_redis -f &> ./docker_logs/redis.log & -echo -e "\n[""$BLUE JOB""$NC""] DB logs are copied to ./embark/logs/mysql.log" +echo -e "\n[""${BLUE} JOB""${NC}""] DB logs are copied to ./embark/logs/mysql.log" docker container logs embark_db -f &> ./docker_logs/mysql.log & # start the supervisor @@ -193,10 +193,10 @@ fi if ! [[ -d /var/www/conf/cert ]]; then mkdir /var/www/conf/cert fi -copy_file "$PWD"/cert/embark.local.crt /var/www/conf/cert -copy_file "$PWD"/cert/embark.local.key /var/www/conf/cert -copy_file "$PWD"/cert/embark-ws.local.key /var/www/conf/cert -copy_file "$PWD"/cert/embark-ws.local.crt /var/www/conf/cert +copy_file "${PWD}"/cert/embark.local.crt /var/www/conf/cert +copy_file "${PWD}"/cert/embark.local.key /var/www/conf/cert +copy_file "${PWD}"/cert/embark-ws.local.key /var/www/conf/cert +copy_file "${PWD}"/cert/embark-ws.local.crt /var/www/conf/cert # cp .env copy_file ./.env /var/www/embark/embark/settings/ @@ -214,23 +214,23 @@ if ! [[ -d /var/www/logs ]]; then fi # db_init -echo -e "\n[""$BLUE JOB""$NC""] Starting migrations - log to embark/logs/migration.log" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting migrations - log to embark/logs/migration.log" pipenv run ./manage.py makemigrations | tee -a /var/www/logs/migration.log pipenv run ./manage.py migrate | tee -a /var/www/logs/migration.log # collect staticfiles and make accesable for server -echo -e "\n[""$BLUE JOB""$NC""] Collecting static files" +echo -e "\n[""${BLUE} JOB""${NC}""] Collecting static files" pipenv run ./manage.py collectstatic --no-input chown www-embark /var/www/ -R chmod 760 /var/www/media/ -R -echo -e "\n[""$BLUE JOB""$NC""] Starting runapscheduler" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting runapscheduler" pipenv run ./manage.py runapscheduler | tee -a /var/www/logs/scheduler.log & sleep 5 -echo -e "\n[""$BLUE JOB""$NC""] Starting Apache" +echo -e "\n[""${BLUE} JOB""${NC}""] Starting Apache" pipenv run ./manage.py runmodwsgi --user www-embark --group sudo \ ---host "$BIND_IP" --port="$HTTP_PORT" --limit-request-body "$FILE_SIZE" \ +--host "${BIND_IP}" --port="${HTTP_PORT}" --limit-request-body "${FILE_SIZE}" \ --url-alias /static/ /var/www/static/ \ --url-alias /media/ /var/www/media/ \ --allow-localhost --working-directory /var/www/embark/ --server-root /var/www/httpd80/ \ @@ -240,22 +240,22 @@ pipenv run ./manage.py runmodwsgi --user www-embark --group sudo \ --log-level debug \ --server-name embark.local "${WSGI_FLAGS[@]}" & # --ssl-certificate /var/www/conf/cert/embark.local --ssl-certificate-key-file /var/www/conf/cert/embark.local.key \ -# --https-port "$HTTPS_PORT" & +# --https-port "${HTTPS_PORT}" & # --https-only --enable-debugger \ sleep 5 -echo -e "\n[""$BLUE JOB""$NC""] Starting daphne(ASGI) - log to /embark/logs/daphne.log" -pipenv run daphne --access-log /var/www/logs/daphne.log -e ssl:8000:privateKey=/var/www/conf/cert/embark-ws.local.key:certKey=/var/www/conf/cert/embark-ws.local.crt -b "$BIND_IP" -p 8001 -s embark-ws.local --root-path=/var/www/embark embark.asgi:application & +echo -e "\n[""${BLUE} JOB""${NC}""] Starting daphne(ASGI) - log to /embark/logs/daphne.log" +pipenv run daphne --access-log /var/www/logs/daphne.log -e ssl:8000:privateKey=/var/www/conf/cert/embark-ws.local.key:certKey=/var/www/conf/cert/embark-ws.local.crt -b "${BIND_IP}" -p 8001 -s embark-ws.local --root-path=/var/www/embark embark.asgi:application & sleep 5 -echo -e "\n""$ORANGE$BOLD""=============================================================""$NC" -echo -e "\n""$ORANGE$BOLD""EMBA logs are under /var/www/emba_logs/ ""$NC" -# echo -e "\n\n""$GREEN$BOLD""the trusted rootCA.key for the ssl encryption is in ./cert""$NC" +echo -e "\n""${ORANGE}${BOLD}""=============================================================""${NC}" +echo -e "\n""${ORANGE}${BOLD}""EMBA logs are under /var/www/emba_logs/ ""${NC}" +# echo -e "\n\n""${GREEN}${BOLD}""the trusted rootCA.key for the ssl encryption is in ./cert""${NC}" if [[ ${#SERVER_ALIAS[@]} -ne 0 ]]; then - echo -e "\n""$ORANGE$BOLD""Server started on http://embark.local with alias:""${SERVER_ALIAS[*]}""$NC" + echo -e "\n""${ORANGE}${BOLD}""Server started on http://embark.local with alias:""${SERVER_ALIAS[*]}""${NC}" else - echo -e "\n""$ORANGE$BOLD""Server started on http://embark.local""$NC" + echo -e "\n""${ORANGE}${BOLD}""Server started on http://embark.local""${NC}" fi -# echo -e "\n""$ORANGE$BOLD""For SSL you may use https://embark.local (Not recommended for local use)""$NC" -wait \ No newline at end of file +# echo -e "\n""${ORANGE}${BOLD}""For SSL you may use https://embark.local (Not recommended for local use)""${NC}" +wait diff --git a/supervisor.sh b/supervisor.sh index 7e1a9f67e..33a5c5db0 100755 --- a/supervisor.sh +++ b/supervisor.sh @@ -16,7 +16,7 @@ LOG_FILE='/var/log/embark-supervisor.log' # Close standard output file descriptors and reroute to log exec 1<&- exec 2<&- -exec 1<>$LOG_FILE +exec 1<>${LOG_FILE} exec 2>&1 while :; do @@ -25,4 +25,4 @@ while :; do echo "$(date +"%D %T")""retstarted docker-networks" fi sleep 1m -done \ No newline at end of file +done