1 /* 2 * Copyright (c) 2017-2019 sel-project 3 * 4 * Permission is hereby granted, free of charge, to any person obtaining a copy 5 * of this software and associated documentation files (the "Software"), to deal 6 * in the Software without restriction, including without limitation the rights 7 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 8 * copies of the Software, and to permit persons to whom the Software is 9 * furnished to do so, subject to the following conditions: 10 * 11 * The above copyright notice and this permission notice shall be included in all 12 * copies or substantial portions of the Software. 13 * 14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 17 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 18 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 19 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 20 * SOFTWARE. 21 * 22 */ 23 /** 24 * Copyright: Copyright (c) 2017-2019 sel-project 25 * License: MIT 26 * Authors: Kripth 27 * Source: $(HTTP github.com/sel-project/selery/source/selery/crash.d, selery/crash.d) 28 */ 29 module selery.crash; 30 31 import std.algorithm : min, max; 32 import std.ascii : newline; 33 import std.conv : to; 34 import std.datetime : Clock; 35 import std.file : write, read, exists, mkdir; 36 import std..string : split, replace; 37 38 import sel.format : Format; 39 import sel.terminal : writeln; 40 41 import selery.about : Software; 42 import selery.lang : LanguageManager; 43 44 public string logCrash(string type, inout LanguageManager lang, Throwable e) { 45 46 string filename = "crash/" ~ type ~ "_" ~ Clock.currTime().toSimpleString().split(".")[0].replace(" ", "_").replace(":", ".") ~ ".txt"; 47 48 writeln(Format.red ~ lang.translate("warning.crash", [typeid(e).to!string.split(".")[$-1], e.msg, e.file, e.line.to!string])); 49 50 string file = "Critical " ~ (cast(Error)e ? "error" : "exception") ~ " on " ~ Software.display ~ newline ~ newline; 51 file ~= "Message: " ~ e.msg ~ newline; 52 file ~= "Type: " ~ typeid(e).to!string.split(".")[$-1] ~ newline; 53 file ~= "File: " ~ e.file ~ newline; 54 file ~= "Line: " ~ e.line.to!string ~ newline ~ newline; 55 file ~= e.info.to!string.replace("\n", newline) ~ newline; 56 if(exists(e.file)) { 57 file ~= newline; 58 string[] errfile = (cast(string)read(e.file)).split(newline); 59 foreach(uint i ; to!uint(max(0, e.line-32))..to!uint(min(errfile.length, e.line+32))) { 60 file ~= "[" ~ (i + 1).to!string ~ "] " ~ errfile[i] ~ newline; 61 } 62 } 63 if(!exists("crash")) mkdir("crash"); 64 write(filename, file); 65 66 writeln(Format.red ~ lang.translate("warning.savedCrash", [filename])); 67 68 return filename; 69 70 }