1 /** Various debug printing tools for debug printing in `@safe pure nothrow @nogc` code. 2 * 3 * Copyright: Per Nordlöw 2018-. 4 * License: $(WEB boost.org/LICENSE_1_0.txt, Boost License 1.0). 5 * Authors: $(WEB Per Nordlöw) 6 * 7 * See_Also: https://forum.dlang.org/post/svjjawiezudnugdyriig@forum.dlang.org 8 */ 9 module nxt.dbgio; 10 11 // version = show; 12 13 @safe pure: 14 15 /** Debug print `Names` all being compile-time strings that are mixed in. 16 * 17 * Similar to Rust's `dbg` macro introduced in version 1.32. 18 * 19 * See_Also: https://blog.rust-lang.org/2019/01/17/Rust-1.32.0.html#the-dbg-macro 20 * See_Also: https://forum.dlang.org/post/svjjawiezudnugdyriig@forum.dlang.org 21 */ 22 mixin template dumpObseleted(Names...) 23 { 24 auto dumpObseleted = 25 { 26 import std.stdio : stderr, write; 27 debug write(__FILE__, ":", __LINE__, ": Info: "); 28 foreach (immutable i, name; Names) 29 { 30 debug write(typeof(name).stringof, " ", name, ": ", mixin(name), (i < Names.length-1) ? ", " : "\n"); 31 } 32 return false; 33 }(); 34 } 35 36 /// 37 version(show) 38 @safe pure unittest 39 { 40 const int x = 42; 41 int[2] y = [42, 43]; 42 mixin dumpObseleted!("x", "y"); 43 } 44 45 /** Debug print `args` followed by a newline. 46 * 47 * Similar to Rust's `dbg` macro introduced in version 1.32. 48 * 49 * See_Also: https://blog.rust-lang.org/2019/01/17/Rust-1.32.0.html#the-dbg-macro 50 * See_Also: https://forum.dlang.org/post/svjjawiezudnugdyriig@forum.dlang.org 51 */ 52 void dbg(Args...)(scope Args args, 53 const string file = __FILE_FULL_PATH__, 54 const uint line = __LINE__, 55 const string fun = __FUNCTION__) @trusted pure nothrow @nogc 56 { 57 import std.stdio : stderr, writeln; 58 try 59 debug stderr.writeln(file, "(", line, ",1):", " Debug: \"", args, "\""); 60 catch (Exception) { } 61 } 62 63 /// 64 version(show) 65 @safe pure nothrow @nogc unittest 66 { 67 int x = 42; 68 dbg("x: ", x); 69 static assert(__traits(compiles, { dbg(); })); // ok for dln to discard function qualifiers 70 } 71 72 /** Show the symbol name and variable of $(D Args). 73 * 74 * See_Also: http://forum.dlang.org/thread/yczwqrbkxdiqijtiynrh@forum.dlang.org?page=1 75 * 76 * TODO: use https://forum.dlang.org/post/ypxsqtddxvdxunsoluas@forum.dlang.org 77 * 78 * TODO: is using this https://dlang.org/changelog/2.079.0.html#default_after_variadic preferred? 79 * 80 * TODO: instead use 81 * 82 * void show_(Args...)(Args args, 83 * string file = __FILE__, 84 * uint line = __LINE__, 85 * string fun = __FUNCTION__) 86 * 87 */ 88 template show(Args...) 89 if (Args.length >= 1) 90 { 91 void show(string file = __FILE__, 92 uint line = __LINE__, 93 string fun = __FUNCTION__) 94 { 95 import std.stdio: write, writeln; 96 try 97 { 98 debug 99 { 100 write(file, ":",line, ":" /* , ": in ",fun */, " debug: "); 101 foreach (const i, Arg; Args) 102 { 103 if (i) 104 write(", "); // separator 105 write(Args[i].stringof, ":", Arg); 106 } 107 writeln(); 108 } 109 } 110 catch (Exception) { } 111 } 112 } 113 114 version(show) 115 @safe pure unittest 116 { 117 const x = 11; 118 const y = 12; 119 const z = 13; 120 show!x; 121 show!y; 122 show!z; 123 } 124 125 version(show) unittest 126 { 127 const x = 11, y = 12, z = 13; 128 show!(x, y, z); 129 } 130 131 import std.stdio; 132 133 /** Debug dump arguments `args` to standard error output (`stderr`). 134 * 135 * See_Also: https://forum.dlang.org/post/myxzyfgtcewixwbhvalp@forum.dlang.org 136 */ 137 template dump(args...) 138 { 139 alias dbgImpl!(args).print dump; 140 } 141 142 template dbgImpl(args...) 143 { 144 import std.traits : isBuiltinType, isAggregateType, FieldNameTuple; 145 146 private void print(string file = __FILE__, uint line = __LINE__, string fun = __FUNCTION__) 147 { 148 static foreach (arg; args) 149 { 150 static if (isBuiltinType!(typeof(arg))) 151 { 152 debug stderr.writefln("[%s:%s (%s)] %s = %s", 153 file, line, fun, 154 __traits(identifier, arg), arg); 155 } 156 else static if (isAggregateType!(typeof(arg))) 157 { 158 debug stderr.writefln("[%s:%s (%s)] %s = %s", 159 file, line, fun, 160 __traits(identifier, arg), 161 toDbgString(arg)); 162 } 163 } 164 } 165 166 private string toDbgString(Arg)(Arg o) 167 { 168 string dbgstr = "("; 169 import std.format; 170 static foreach(f; FieldNameTuple!(typeof(o))) 171 { 172 static if (isBuiltinType!(typeof(__traits(getMember, o, f)))) 173 dbgstr ~= format("%s:%s, ", f, __traits(getMember, o, f)); 174 else static if (isAggregateType!(typeof(__traits(getMember, o, f)))) 175 dbgstr ~= format("%s = %s, ", f, toDbgString(__traits(getMember, o, f))); 176 } 177 return dbgstr[0..$-2] ~ ")"; 178 } 179 } 180 181 /// 182 version(show) 183 @safe pure unittest 184 { 185 struct Bar { auto c = 'c';} 186 struct Foo { int s = 2; bool b = false; Bar bar;} 187 class FooBar { int t; Foo f; } 188 189 int i; 190 float f = 3.14; 191 string s = "some string"; 192 Foo foo; 193 Bar bar; 194 195 dump!(i, f, s, foo, 1+3, foo, bar); 196 197 // prints: 198 // [dump.d:54 (dump.main)] i = 0 199 // [dump.d:54 (dump.main)] f = 3.14 200 // [dump.d:54 (dump.main)] s = some string 201 // [dump.d:54 (dump.main)] foo = (s:2, b:false, bar = (c:c)) 202 // [dump.d:54 (dump.main)] _ = 4 203 // [dump.d:54 (dump.main)] foo = (s:2, b:false, bar = (c:c)) 204 // [dump.d:54 (dump.main)] bar = (c:c) 205 // [dump.d:54 (dump.main)] fb = (t:0, f = (s:2, b:false, bar = (c:c))) 206 }