/* globals $, LZString */ var smapi = smapi || {}; smapi.logParser = function(sectionUrl) { /********* ** Initialisation *********/ var stage, flags = $("#modflags"), output = $("#output"), filters = 0, memory = "", versionInfo, modInfo, modMap, modErrors, logInfo, templateBody = $("#template-body").text(), templateModentry = $("#template-modentry").text(), templateCss = $("#template-css").text(), templateLogentry = $("#template-logentry").text(), templateLognotice = $("#template-lognotice").text(), regexInfo = /\[[\d\:]+ INFO SMAPI] SMAPI (.*?) with Stardew Valley (.*?) on (.*?)\n/g, regexMods = /\[[^\]]+\] Loaded \d+ mods:(?:\n\[[^\]]+\] .+)+/g, regexLog = /\[([\d\:]+) (TRACE|DEBUG|INFO|WARN|ALERT|ERROR) ? ([^\]]+)\] ?((?:\n|.)*?)(?=(?:\[\d\d:|$))/g, regexMod = /\[(?:.*?)\] *(.*?) (\d+\.?(?:.*?))(?: by (.*?))? \|(?:.*?)$/gm, regexDate = /\[\d{2}:\d{2}:\d{2} TRACE SMAPI\] Log started at (.*?) UTC/g, regexPath = /\[\d{2}:\d{2}:\d{2} DEBUG SMAPI\] Mods go here: (.*?)(?:\n|$)/g; $("#tabs li:not(.notice)").on("click", function(evt) { var t = $(evt.currentTarget); t.toggleClass("active"); $("#output").toggleClass(t.text().toLowerCase()); }); $("#upload-button").on("click", function() { memory = $("#input").val() || ""; $("#input").val(""); $("#popup-upload").fadeIn(); }); $("#popup-upload").on({ 'dragover dragenter': function(e) { e.preventDefault(); e.stopPropagation(); }, 'drop': function(e) { $("#uploader").attr("data-text", "Reading..."); $("#uploader").show(); var dataTransfer = e.originalEvent.dataTransfer; if (dataTransfer && dataTransfer.files.length) { e.preventDefault(); e.stopPropagation(); var file = dataTransfer.files[0]; var reader = new FileReader(); reader.onload = $.proxy(function(file, $input, event) { $input.val(event.target.result); $("#uploader").fadeOut(); $("#submit").click(); }, this, file, $("#input")); reader.readAsText(file); } } }); $("#submit").on("click", function() { $("#popup-upload").fadeOut(); var raw = $("#input").val(); if (raw) { memory = ""; var paste = LZString.compressToUTF16(raw); if (paste.length * 2 > 524288) { $("#output").html('

Unable to save!

This log cannot be saved due to its size.
' + $("#input").val() + "
"); return; } $("#uploader").attr("data-text", "Saving..."); $("#uploader").fadeIn(); $ .ajax({ type: "POST", url: sectionUrl + "/save", data: JSON.stringify(paste), contentType: "application/json" // sent to API }) .fail(function(xhr, textStatus) { $("#uploader").fadeOut(); $("#output").html('

Parsing failed!

Parsing of the log failed, details follow.
 

Stage: Upload

Error: ' + textStatus + ': ' + xhr.responseText + "
" + $("#input").val() + "
"); }) .then(function(data) { $("#uploader").fadeOut(); console.log("Result: ", data); if (!data.success) $("#output").html('

Parsing failed!

Parsing of the log failed, details follow.
 

Stage: Upload

Error: ' + data.error + "
" + $("#input").val() + "
"); else location.href = "?" + data.id; }); } else { alert("Unable to parse log, the input is empty!"); $("#uploader").fadeOut(); } }); $("#cancel").on("click", function() { $("#popup-upload").fadeOut(400, function() { $("#input").val(memory); memory = ""; }); }); $("#closeraw").on("click", function() { $("#popup-raw").fadeOut(400); }); if (location.search) { getData(); } else $("#popup-upload").fadeIn(); /********* ** Helpers *********/ function modClicked(evt) { var id = $(evt.currentTarget).attr("id").split("-")[1], cls = "mod-" + id; if (output.hasClass(cls)) filters--; else filters++; output.toggleClass(cls); if (filters === 0) { output.removeClass("modfilter"); } else { output.addClass("modfilter"); } } function removeFilter() { for (var c = 0; c < modInfo.length; c++) { output.removeClass("mod-" + c); } filters = 0; output.removeClass("modfilter"); } function selectAll() { for (var c = 0; c < modInfo.length; c++) { output.addClass("mod-" + c); } filters = modInfo.length; output.addClass("modfilter"); } function parseData() { stage = "parseData.pre"; var data = $("#input").val(); if (!data) { stage = "parseData.checkNullData"; throw new Error("Field `data` is null"); } var dataInfo = regexInfo.exec(data) || regexInfo.exec(data) || regexInfo.exec(data), dataMods = regexMods.exec(data) || regexMods.exec(data) || regexMods.exec(data), dataDate = regexDate.exec(data) || regexDate.exec(data) || regexDate.exec(data), dataPath = regexPath.exec(data) || regexPath.exec(data) || regexPath.exec(data), match; console.log("dataInfo:", dataInfo); console.log("dataMods:", dataMods); console.log("dataDate:", dataDate); console.log("dataPath:", dataPath); stage = "parseData.doNullCheck"; if (!dataInfo) throw new Error("Field `dataInfo` is null"); if (!dataMods) throw new Error("Field `dataMods` is null"); if (!dataPath) throw new Error("Field `dataPath` is null"); dataMods = dataMods[0]; stage = "parseData.setupDefaults"; modMap = { "SMAPI": 0 }; modErrors = { "SMAPI": 0, "Console.Out": 0 }; logInfo = []; modInfo = [ ["SMAPI", dataInfo[1], "Zoryn, CLxS & Pathoschild"] ]; stage = "parseData.parseInfo"; var date = dataDate ? new Date(dataDate[1] + "Z") : null; versionInfo = [dataInfo[1], dataInfo[2], dataInfo[3], date ? date.getFullYear() + "-" + ("0" + date.getMonth().toString()).substr(-2) + "-" + ("0" + date.getDay().toString()).substr(-2) + " at " + date.getHours() + ":" + date.getMinutes() + ":" + date.getSeconds() + " " + date.toLocaleTimeString("en-us", { timeZoneName: "short" }).split(" ")[2] : "No timestamp found", dataPath[1]]; stage = "parseData.parseMods"; while ((match = regexMod.exec(dataMods))) { modErrors[match[1]] = 0; modMap[match[1]] = modInfo.length; modInfo.push([match[1], match[2], match[3] ? ("by " + match[3]) : "Unknown author"]); } stage = "parseData.parseLog"; while ((match = regexLog.exec(data))) { if (match[2] === "ERROR") modErrors[match[3]]++; logInfo.push([match[1], match[2], match[3], match[4]]); } stage = "parseData.post"; modMap["Console.Out"] = modInfo.length; modInfo.push(["Console.Out", "", ""]); } function renderData() { stage = "renderData.pre"; output.html(prepare(templateBody, versionInfo)); var modslist = $("#modslist"), log = $("#log"), modCache = [], y = 0; for (; y < modInfo.length; y++) { var errors = modErrors[modInfo[y][0]], err, cls = "color-red"; if (errors === 0) { err = "No Errors"; cls = "color-green"; } else if (errors === 1) err = "1 Error"; else err = errors + " Errors"; modCache.push(prepare(templateModentry, [y, modInfo[y][0], modInfo[y][1], modInfo[y][2], cls, err])); } modslist.append(modCache.join("")); for (var z = 0; z < modInfo.length; z++) $("#modlink-" + z).on("click", modClicked); var flagCache = []; for (var c = 0; c < modInfo.length; c++) flagCache.push(prepare(templateCss, [c])); flags.html(flagCache.join("")); var logCache = [], dupeCount = 0, dupeMemory = "|||"; for (var x = 0; x < logInfo.length; x++) { var dm = logInfo[x][1] + "|" + logInfo[x][2] + "|" + logInfo[x][3]; if (dupeMemory !== dm) { if (dupeCount > 0) logCache.push(prepare(templateLognotice, [logInfo[x - 1][1].toLowerCase(), modMap[logInfo[x - 1][2]], dupeCount])); dupeCount = 0; dupeMemory = dm; logCache.push(prepare(templateLogentry, [logInfo[x][1].toLowerCase(), modMap[logInfo[x][2]], logInfo[x][0], logInfo[x][1], logInfo[x][2], logInfo[x][3].split(" ").join("  ").replace(//g, ">").replace(/\n/g, "
")])); } else dupeCount++; } log.append(logCache.join("")); $("#modlink-r").on("click", removeFilter); $("#modlink-a").on("click", selectAll); } function prepare(str, arr) { var regex = /\{(\d)\}/g, match; while ((match = regex.exec(str))) str = str.replace(match[0], arr[match[1]]); return str; } function loadData() { try { stage = "loadData.Pre"; var start = performance.now(); parseData(); renderData(); var end = performance.now(); $(".always").prepend("
Log processed in: " + (Math.round((end - start) * 100) / 100) + ' ms (View raw)

'); $("#viewraw").on("click", function() { $("#dataraw").val($("#input").val()); $("#popup-raw").fadeIn(); }); stage = "loadData.Post"; } catch (err) { $("#output").html('

Parsing failed!

Parsing of the log failed, details follow.
 

Stage: ' + stage + "

" + err + '
'); $("#rawlog").text($("#input").val()); } } function getData() { $("#uploader").attr("data-text", "Loading..."); $("#uploader").fadeIn(); $.get(sectionUrl + "/fetch/" + location.search.substring(1), function(data) { if (data.success) { $("#input").val(LZString.decompressFromUTF16(data.content) || data.content); loadData(); } else { $("#output").html('

Fetching the log failed!

' + data.error + '

'); $("#rawlog").text($("#input").val()); } $("#uploader").fadeOut(); }); } };