var canvas = document.getElementById("myCanvas"); var ctx = canvas.getContext("2d"); var ballRadius = 7; var x = canvas.width/2; var y = canvas.height-30; var dx = 2; var dy = -2; var spaceshipHeight = 60; var spaceshipWidth = 60; var spaceshipX = (canvas.width-spaceshipWidth)/2; var spaceshipY = (canvas.height-spaceshipHeight)/2; var clu_endpoint = "endpoint_placeholder"; var clu_insicure = "insecure_endpoint_placeholder"; var demo_mode = "platform_engineering_demo_mode_placeholder" var k8s_url = ""; var chaos_report_post_data = ""; // when zoomIn is 12 var maxAliensPerRow = 20; var startYforHelp = 700; if (clu_insicure == "true") { k8s_url = "http://" + clu_endpoint; } else { k8s_url = "https://" + clu_endpoint; } var namespaces = []; var namespaces_index = 0; var namespace = namespaces[namespaces_index]; var endpoint = ""; var modal_opened = false; var autoPilot = false; var autoPilotDirection = 0; var spaceshipxOld = 0; var randomFactor = 10; // pods list from kubernetes var pods = []; var game_mode_switch = false; var programming_mode_switch = false; var now = ""; var game_buttons = document.getElementById("game-buttons"); var game_screen = document.getElementById("game-screen"); var chaos_program_screen = document.getElementById("chaos-program-screen"); var programming_mode_buttons = document.getElementById("programming-mode-buttons"); var log_tail_switch = false; var log_tail_div = document.getElementById("logTailDiv"); var log_tail_screen = document.getElementById("logTailScreen"); var random_code = (Math.random() + 1).toString(36).substring(7); // nodes list from kubernetes var nodes = []; // Hash of aliens related to pods or nodes var aliens = []; var aliensWidth = 40; var aliensHeight = 40; // Button vars var rightPressed = false; var leftPressed = false; var upPressed = false; var downPressed = false; // The is true the rocket can move var shot = false; // Keep track of rocket launch var rocketLaunched = false; // Rocket position var rocketX = -400; var rocketY = -400; var rocketSpeed = 7; var collisionDetected = false; // Aliens Vars. Keep track of Y positions where there is an alien. var aliensY = []; var aliensIncrementY = 50; var shuffle = true; var help = false; var chaos_nodes = false; var chaos_pods = true; var log_tail_alert = '"); enableLogTail(); setLogRegex(); $('#logTailRegex').val('{"since": "60", "pod":".*", "namespace":"' + namespace + '", "labels":".*", "annotations":".*", "containers":".*"}'); if (!log_tail_switch) { setLogConsole(); } };; $('#currentKubeLinterResult').text('KubeLinter launched. Set this regex and start log tail: {"since": "60", "pod":".*", "namespace":"' + namespace + '", "labels":".*", "annotations":".*", "containers":".*"}'); oReq.open("GET", k8s_url + "/kube/kube-linter?logid=" + random_code +"&namespace=" + namespace); oReq.send(); } function getNamespaces() { var oReq = new XMLHttpRequest(); oReq.onload = function () { namespaces = this.responseText; namespaces = namespaces.split(","); namespace = namespaces[namespaces_index]; console.log("[CURRENT-NAMESPACE] " + namespace); $('#currentGameNamespace').text(namespace); };; oReq.open("GET", k8s_url + "/kube/namespaces"); oReq.send(); } function getEndpoint() { var oReq = new XMLHttpRequest(); oReq.onload = function () { endpoint = this.responseText; };; oReq.open("GET", k8s_url + "/kube/endpoint"); oReq.send(); } function getCurrentChaosContainer() { var oReq = new XMLHttpRequest(); oReq.onload = function () { job_parsed = JSON.stringify(JSON.parse(this.responseText), null, 4); $('#currentChaosContainerYaml').text(job_parsed); $('#currentChaosContainerJsonTextArea').val(job_parsed); };; oReq.open("GET", k8s_url + "/kube/chaos/containers?action=container_definition"); oReq.send(); } function enableLogTail() { var oReq = new XMLHttpRequest(); oReq.open("POST", k8s_url + "/kube/chaos/containers?action=enable_log_tail&id=" + random_code, true); oReq.onreadystatechange = function () { if (this.readyState === XMLHttpRequest.DONE && this.status === 200) { $('#alert_placeholder3').replaceWith(log_tail_alert + 'Logs tail started '); } };; oReq.setRequestHeader("Content-Type", "application/json"); oReq.send("{}"); setLogRegex(); } function disableLogTail() { var oReq = new XMLHttpRequest(); oReq.open("POST", k8s_url + "/kube/chaos/containers?action=disable_log_tail&id=" + random_code, true); oReq.onreadystatechange = function () { if (this.readyState === XMLHttpRequest.DONE && this.status === 200) { $('#alert_placeholder3').replaceWith(log_tail_alert + 'Logs tail stopped '); } };; oReq.setRequestHeader("Content-Type", "application/json"); // TODO: send payload for auth... oReq.send("foobar"); } function setLogRegex() { log_tail_div.style.display = "block"; $('#alert_placeholder3').replaceWith(log_tail_alert + 'Setting regex for filtering log source (by pod name)'); var oReq = new XMLHttpRequest(); oReq.open("POST", k8s_url + "/kube/chaos/containers?action=set_log_regex&id=" + random_code, true); oReq.onreadystatechange = function () { if (this.readyState === XMLHttpRequest.DONE && this.status === 200) { $('#alert_placeholder3').replaceWith(log_tail_alert + 'New regex has been configured'); } };; oReq.setRequestHeader("Content-Type", "application/json"); oReq.send($('#logTailRegex').val()); } function setChaosContainer() { if (!IsJsonString($('#currentChaosContainerJsonTextArea').val())) { $('#alert_placeholder2').text('JSON syntax not valid.'); } else { var oReq = new XMLHttpRequest(); oReq.open("POST", k8s_url + "/kube/chaos/containers?action=set", true); oReq.onreadystatechange = function () { if (this.readyState === XMLHttpRequest.DONE && this.status === 200) { $('#alert_placeholder2').text('New container definition has been saved.'); } };; oReq.setRequestHeader("Content-Type", "application/json"); oReq.send($('#currentChaosContainerJsonTextArea').val()); } } function startChaosNode(node_name) { var oReq = new XMLHttpRequest(); oReq.onload = function () { $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Launched chaos job against ' + node_name + ''); };; $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Start chaos job against ' + node_name + ''); oReq.open("GET", k8s_url + "/kube/chaos/nodes?nodename=" + node_name + "&namespace=" + namespace); oReq.send(); } function deletePods(pod_name) { var oReq = new XMLHttpRequest(); oReq.onload = function () { $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Kill ' + pod_name + ''); };; oReq.open("GET", k8s_url + "/kube/pods?action=delete&pod_name=" + pod_name + "&namespace=" + namespace); oReq.send(); } function getPods() { if (chaos_pods) { var oReq = new XMLHttpRequest(); oReq.onload = function () { new_pods = JSON.parse(this.responseText)["items"]; // Pod might just be killed in game, but not terminated in k8s yet. for (i=0; i alien.name == new_pods[i].name && alien.status == "killed")) { new_pods[i].status = "killed"; } } if (nodes && nodes.length > 0) { pods = new_pods.concat(nodes); } else { pods = new_pods; } };; oReq.open("GET", k8s_url + "/kube/pods?action=list&namespace=" + namespace); oReq.send(); } else { if (nodes && nodes.length > 0) { pods = nodes; } else { pods = []; } } } function getNodes() { if (chaos_nodes) { var oReq = new XMLHttpRequest(); oReq.onload = function () { json_parsed = JSON.parse(this.responseText); nodes = json_parsed["items"]; };; oReq.open("GET", k8s_url + "/kube/nodes"); oReq.send(); } else { nodes = [] } } window.setInterval(function getKubeItems() { if (game_mode_switch) { getNodes(); getPods(); } }, 500) function keyDownHandler(e) { if (!modal_opened && game_mode_switch) { e.preventDefault(); if (e.key == "Right" || e.key == "ArrowRight") { rightPressed = true; } else if (e.key == "Left" || e.key == "ArrowLeft") { leftPressed = true; } if (e.key == "Up" || e.key == "ArrowUp") { upPressed = true; } else if (e.key == "Down" || e.key == "ArrowDown") { downPressed = true; } else if (e.keyCode == 83) { if (shuffle) { shuffle = false; $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Disable shuffle'); } else { shuffle = true $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Enable shuffle'); } } else if (e.keyCode == 32) { shot = true } else if (e.keyCode == 78) { switchNamespace(); } else if (e.keyCode == 72) { if (help) { help = false; } else { help = true } } else if (e.keyCode == 67) { if (is_demo_mode()) { demo_mode_alert(); return; } if (chaos_nodes) { chaos_nodes = false; $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Hide nodes'); } else { chaos_nodes = true $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Show nodes'); } } else if (e.keyCode == 80) { if (chaos_pods) { chaos_pods = false; $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Hide pods'); } else { chaos_pods = true $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Show pods'); } } } } function keyUpHandler(e) { if (e.key == "Right" || e.key == "ArrowRight") { rightPressed = false; } else if (e.key == "Left" || e.key == "ArrowLeft") { leftPressed = false; } else if (e.key == "Up" || e.key == "ArrowUp") { upPressed = false; } else if (e.key == "Down" || e.key == "ArrowDown") { downPressed = false; } } function drawAlien(alienX, alienY, name, status) { var image = new Image(); // Image constructor if (nodes.some((node) => node.name == name)) { image.src = './images/k8s_node.png'; ctx.drawImage(image, alienX, alienY, 30, 40); } else { image.src = `./images/sprite_invader_${status}.png`; ctx.font = '8px pixel'; ctx.drawImage(image, alienX, alienY, 40, 40); if (showPodName) { ctx.fillText(name.substring(0, 19) + '..', alienX, alienY + 40); } } ctx.closePath(); } function checkRocketAlienCollision() { if (contains(aliensY, rocketY)) { var i; for (i=aliens.length - 1; i >= 0; i--) { if (aliens[i]["active"] && (rocketY - aliens[i]["y"] < 5)) { var rangeX = [] rangeX.push(aliens[i]["x"]); for (k=aliens[i]["x"]; k node.name == aliens[i]["name"])) { aliens[i]["active"] = false; startChaosNode(aliens[i]["name"]); } else { deletePods(aliens[i]["name"]); } return true; } } } } return false; } function shuffleAliens() { pods = pods.sort(() => Math.random() - 0.5) } function drawRocket() { var image = new Image(); // Image constructor image.src = './images/kuberocket.png'; ctx.drawImage(image, rocketX, rocketY, 20, 20); ctx.closePath(); if (checkRocketAlienCollision()) { rocketY = -100; rocketX = -100; collisionDetected = false; return } if (shot && rocketLaunched) { if (rocketY < 0) { shot = false; rocketLaunched = false; } else { rocketY = rocketY -= rocketSpeed; } } else { rocketX = spaceshipX + (spaceshipWidth / 3); rocketY = spaceshipY; rocketLaunched = true } } function drawSpaceship() { var image = new Image(); // Image constructor image.src = './images/spaceship.png'; ctx.drawImage(image, spaceshipX, spaceshipY, 60, 60); ctx.closePath(); } window.setInterval(function draw() { if (namespacesJumpFlag){ randNamespaceJump(1, 10, 8); } }, 1000) window.setInterval(function draw() { ctx.clearRect(0, 0, canvas.width, canvas.height); for (i=0; i canvas.width-ballRadius || x + dx < ballRadius) { dx = -dx; } if (y + dy > canvas.height-ballRadius || y + dy < ballRadius) { dy = -dy; } if (autoPilot){ spaceshipY = 340; if (getRandomInt(100) < randomFactor) { shot = true; } if (autoPilotDirection == 0) { autoPilotDirection = getRandomInt(canvas.width-spaceshipWidth); spaceshipxOld = spaceshipX; } else if ((spaceshipX == autoPilotDirection)) { autoPilotDirection = getRandomInt(canvas.width-spaceshipWidth); spaceshipxOld = spaceshipX; } else if ((autoPilotDirection < spaceshipxOld) && (spaceshipX < autoPilotDirection)) { autoPilotDirection = getRandomInt(canvas.width-spaceshipWidth); spaceshipxOld = spaceshipX; } else if ((autoPilotDirection > spaceshipxOld) && (spaceshipX > autoPilotDirection)) { autoPilotDirection = getRandomInt(canvas.width-spaceshipWidth); spaceshipxOld = spaceshipX; } else { if (autoPilotDirection > spaceshipX) { spaceshipX += 5; } else { spaceshipX -= 5; } } } if (rightPressed) { spaceshipX += 3; if (spaceshipX + spaceshipWidth > canvas.width) { spaceshipX = canvas.width - spaceshipWidth; } } else if (leftPressed) { spaceshipX -= 3; if (spaceshipX < 0) { spaceshipX = 0; } } if (upPressed) { spaceshipY -= 3; if (spaceshipY < 0) { spaceshipY = 0; } } else if (downPressed) { spaceshipY += 3; if (spaceshipY + spaceshipHeight > canvas.height) { spaceshipY = canvas.height - spaceshipHeight; } } ctx.fillStyle = 'white'; ctx.font = '16px pixel'; ctx.fillText('Cluster: ' + endpoint, 10, startYforHelp); ctx.fillText('Current Namespace: ' + namespace, 10, startYforHelp + 20); ctx.fillText('Alien Shuffle: ' + shuffle, 10, startYforHelp + 40); ctx.fillText('Auto Namespaces Switch: ' + namespacesJumpStatus, 10, startYforHelp + 60); ctx.fillText('press \'h\' for help!', 10, startYforHelp + 80); if (help) { ctx.fillText('h => Activate or deactivate help', 10, 280); ctx.fillText('s => Activate or deactivate shuffle for aliens', 10, 300); ctx.fillText('n => Change namespace', 10, 320); ctx.fillText('p => Activate or deactivate chaos engineering against pods', 10, 340); ctx.fillText('c => Activate or deactivate chaos engineering against nodes', 10, 360); } }, 10) function buttonShuffleHelper() { if (shuffle) { shuffle = false; $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Shuffle Disable'); $("#buttonShuffle").text("Enable Shuffle"); } else { shuffle = true $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Shuffle Enabled'); $("#buttonShuffle").text("Disable Shuffle"); } } function namespacesJumpControl() { if (namespacesJumpFlag) { namespacesJumpFlag = false; $("#namespacesJumpButton").text("Enable Auto NS Switch"); $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Disabled automatic switch of namespace'); namespacesJumpStatus = 'Disabled' } else { namespacesJumpFlag = true; $("#namespacesJumpButton").text("Disable Auto NS Switch"); $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Enabled automatic switch of namespace '); namespacesJumpStatus = 'Enabled' } } function showPodNameControl() { if (showPodName) { showPodName = false; $("#buttonOnlyPodName").text("Show Pods Name"); } else { showPodName = true $("#buttonOnlyPodName").text("Hide Pods Name"); } } function podExists(podName) { for (i=0; i jumpRandomFactor) { $('#alert_placeholder').replaceWith(alert_div + 'Latest action: Switch Namespace'); switchNamespace(); } } window.setInterval(function setAliens() { if (shuffle) { pods = pods.sort(() => Math.random() - 0.5) } aliens = []; if (pods.length > 0) { for (k=10; k>0; k--) { if (!contains(aliensY, k)) { aliensY.push(k); } } var x = 10; var y = 10; var yInc = false; for (i=0; i=y; k--) { if (!contains(aliensY, k)) { aliensY.push(k); } } } else { x += 60; } } } } }, 1000) window.setInterval(function backgroundTasks() { if (!codename_configured) { chaosProgram = $('#chaosProgramTextArea').val(); chaosProgramWithCodename = chaosProgram.replace(codename_regex, "chaos-codename: " + codename); $('#chaosProgramTextArea').val(chaosProgramWithCodename); codename_configured = true; } if (game_mode_switch || programming_mode_switch || log_tail_switch) { getMetrics(); getChaosJobsPodsPhase(); updateMainMetricsChart(); } if (log_tail_switch) { getChaosJobsLogs(); getTotalLogsPos(); } if (programming_mode_switch) { drawChaosProgramFlow(); } if (chaos_report_switch) { updateElapsedTimeArray(chaosReportprojectName); updateChaosReportStartTime(chaosReportprojectName); drawCanvasHTTPStatusCodeStats(); chaosReportKeepAlive(chaosReportprojectName); } }, 2000) document.addEventListener("keydown", keyDownHandler, false); document.addEventListener("keyup", keyUpHandler, false); getEndpoint(); getNamespaces(); getSavedPresets(); document.getElementById("gameContainer").style.visibility = "hidden"; document.getElementById("metricsPresetsRow").style.visibility = "hidden"; document.getElementById("gameContainer").style.opacity = 0; document.getElementById("metricsPresetsRow").style.opacity = 0; document.getElementById("gameContainer").style.visibility = "visible"; document.getElementById("metricsPresetsRow").style.visibility = "visible"; document.getElementById("gameContainer").style.opacity = 1; document.getElementById("metricsPresetsRow").style.opacity = 1;