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 }