digitalmars.D - __traits(compiles, ...) with syntax errors
- Steven Schveighoffer (18/18) Jan 14 2019 How many times have you written something like:
- Steven Schveighoffer (6/28) Jan 14 2019 Another option is to have __traits(compiles) spit out an error type that...
- Steven Schveighoffer (5/10) Jan 14 2019 Oops, it would have to be success or syntax error, so something like:
- Paul Backus (4/13) Jan 14 2019 Ideally, you would catch this in the unit tests for `bar`. You
- Steven Schveighoffer (12/27) Jan 14 2019 It's not always that simple. Sometimes bar is part of a hook or
- Jonathan M Davis (38/55) Jan 14 2019 Good unit tests catch a lot, but it's hard for them to catch everything,...
- H. S. Teoh (76/96) Jan 14 2019 Yeah, I've run into this problem far too many times. It's extremely
- =?UTF-8?Q?Ali_=c3=87ehreli?= (7/13) Jan 14 2019 I've recently opened the following bug which I believe is due to the
- Steven Schveighoffer (37/90) Jan 14 2019 So to clarify a bit, I'm not expecting miracles here. If your code is
- Olivier FAURE (13/17) Jan 15 2019 Maybe a sub-name system? Like:
- Iain Buclaw (5/6) Jan 14 2019 Compile with -verrors=spec (or -Wspeculative if using gdc).
- Steven Schveighoffer (4/13) Jan 14 2019 But wouldn't that show all errors, not just syntax errors? That might be...
- Steven Schveighoffer (3/16) Jan 14 2019 BTW, I didn't know about this switch, thanks for pointing it out.
- Nicholas Wilson (3/17) Jan 14 2019 Yep, the SNR on that switch is abominable.
- Steven Schveighoffer (4/21) Jan 16 2019 Could there be a way to focus that switch on certain files/lines? That
- Kagamin (10/19) Jan 15 2019 struct A
- Steven Schveighoffer (16/38) Jan 16 2019 Indeed, I realized after posting [1] that it's really stupid typos that
- Atila Neves (4/11) Jan 15 2019 More times than I can count...
- Petar Kirov [ZombineDev] (18/31) Jan 15 2019 Sounds like a job for `static try` too keep things DRY:
- Steven Schveighoffer (15/53) Jan 16 2019 It might be nice to have something like this. Note that the two code
How many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon). I often times, to avoid this, simply just use __traits(hasMember, T, "bar"), and live with the consequences. I'd love to have something like __traits(compiles), but have it be an error if the syntax is incorrect. In other words, the constraint above only fails if there is a missing symbol, or incorrect type conversion, not that there is a stupid syntax error that would NEVER compile. Ironically, it would be something like compilesOrSyntaxError. Any thoughts? -Steve
Jan 14 2019
On 1/14/19 4:47 PM, Steven Schveighoffer wrote:How many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon). I often times, to avoid this, simply just use __traits(hasMember, T, "bar"), and live with the consequences. I'd love to have something like __traits(compiles), but have it be an error if the syntax is incorrect. In other words, the constraint above only fails if there is a missing symbol, or incorrect type conversion, not that there is a stupid syntax error that would NEVER compile. Ironically, it would be something like compilesOrSyntaxError.Another option is to have __traits(compiles) spit out an error type that has truthiness if it's not an error (for backwards compatibility), then you could something like: if(__traits(compiles, ...) != COMPILER_ERROR.SYNTAX) -Steve
Jan 14 2019
On 1/14/19 4:52 PM, Steven Schveighoffer wrote:Another option is to have __traits(compiles) spit out an error type that has truthiness if it's not an error (for backwards compatibility), then you could something like: if(__traits(compiles, ...) != COMPILER_ERROR.SYNTAX)Oops, it would have to be success or syntax error, so something like: if(successOrSyntaxError!(__traits(compiles, ...))) successOrSyntaxError would be a simple template -Steve
Jan 14 2019
On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:How many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).Ideally, you would catch this in the unit tests for `bar`. You are writing unit tests, aren't you? ;)
Jan 14 2019
On 1/14/19 4:57 PM, Paul Backus wrote:On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:It's not always that simple. Sometimes bar is part of a hook or something to a complex system, and you expect it to be called one way, and it's called another way (or the lack of ability to call it changes how the system behaves). Do you unit test all your calls to front(), popFront or empty individually? I generally don't, I just put it in as a range, and see if that works. Then I get an error like "this isn't a range", and it's not super-helpful. Or worse, I get a "I can't call any of these 10 template overloads, each with a 5-term constraint" -SteveHow many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).Ideally, you would catch this in the unit tests for `bar`. You are writing unit tests, aren't you? ;)
Jan 14 2019
On Monday, January 14, 2019 2:57:28 PM MST Paul Backus via Digitalmars-d wrote:On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:Good unit tests catch a lot, but it's hard for them to catch everything, and having to test for stuff that the compiler can trivially tell you can get really annoying. That's not to say that the compiler should find everything for you, but sometimes, having it be able to tell you a bit more than currently it does could save a lot of trouble. In this particular case, it's not uncommon to have alternate implementations depending on compile time conditionals. e.g. stuff like static if(isRandomAccessRange!R) { ... } else { ... } isn't all that uncommon in optimized range code. The function may work just fine whether the range is random-access or not, but the actual code that gets compiled in (and therefore its performance) could vary quite a bit depending on the range's capabilities. Syntax errors are less likely in code like that than code that uses is expressions or __traits(compiles, ...) directly, but the problem that Steven brings up is very real - and in a way that it can actually be rather difficult to test for. In many cases, you're basically forced to put in pragmas or invalid code to see which branches are being compiled. Having a trait similar to __traits(compiles, ...) but which actually gives you an error for syntax errors would catch _some_ of the problems surrounding branching on compile-time conditionals. However, given that it's still not verifying that the correct branch was taken (just that the test is syntactically valid), it still wouldn't solve the entire problem. It would catch a certain class of bugs though. And they're bugs that wouldn't exist anywhere outside of an is expression or __traits(compiles), ...), because everywhere else, such syntax errors would be caught. It's just that with those constructs, those errors basically get shut up and treated simply as false (which is probably the wrong behavior in all cases except perhaps for those that involve checking code that's been mixed in). - Jonathan M DavisHow many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).Ideally, you would catch this in the unit tests for `bar`. You are writing unit tests, aren't you? ;)
Jan 14 2019
On Mon, Jan 14, 2019 at 04:31:06PM -0700, Jonathan M Davis via Digitalmars-d wrote:On Monday, January 14, 2019 2:57:28 PM MST Paul Backus via Digitalmars-d wrote:Yeah, I've run into this problem far too many times. It's extremely annoying because the error is silent, and makes the compiler just move on to a different overload, leaving you scratching your head as to why it's not picking the "correct" overload.On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:How many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).[...] The problem is not just trivial syntax errors like Steven's example. The problem is that sufficiently complex sig constraints will sometimes have unexpected corner cases that works in general, but fails for specific cases. One nasty example that comes to mind is when the sig constraint only works with mutable types (or some other such qualified type) because you used an overly-restrictive test. Initial unittests work because you're not expecting that changing the input to const would cause any trouble. Then some time later, you happen to need to pass a const object, and suddenly the compiler seemingly refuses to acknowledge that this overload of the template function exists, and insists on trying a different overload, or failing with a totally unhelpful error saying no overloads match. If you're dealing with something with as many overloads as std.conv.toImpl, for example, good luck finding what went wrong. You first have to parse every single darned sig constraint on every single overload, just to find the one overload that *should* match, but doesn't. Then you have to parse the sig constraint clause by clause just to determine which clause failed. And *then*, if you're really unlucky and the clause is a complex one (that calls another template, say, like isXyzRange or something equally complex), *then* you have to find the definition of that helper template, then parse through every clause in *that* to find out what went wrong. And if you're *truly* unlucky, the helper template is of the form: isBlahBlahBlah(T) = __traits(compiles, (T t) { ... // arbitrarily complex code here }); then you have to copy-n-paste the stupid lambda and instantiate it with your exact version of T (and that's if the original template function's sig constraint hadn't munged it with things like Unqual or any of the other type-changing templates -- then good luck figuring out what T is for your particular function call), just to get the stupid compiler to tell you the stupid error that made it fail, because by default all errors are gagged inside template instantiations during IFTI, so there's nothing at all that even hints at what went wrong. I've had to do this so many times when working with template-heavy code, that recently I've started to develop a distaste for sig constraints. Unless I'm working with complex overload sets where sig constraints are unavoidable, I'd rather just leave the function without any sig constraints -- then at least when something fails, you've a fighting chance of knowing what went wrong without going through the above dance *every* *single* *time*, because the compiler will now actually tell you what the lousy compile error was. And if you're really cursed, you'd run into a case like Jonathan says, where there's a static if inside the template function that goes to the else clause because it doesn't like your template argument for whatever reason, so you just got switched to a different implementation with not a clue as to how it ended up there. Given enough nested static if's, good luck figuring out just which one of them had the failing condition that sent you down the wrong rabbit hole -- the failure is completely silent; and remember, if the else-branch of the static if fails to compile, the compiler will just gag the error and move on to the next overload at the top level. Good luck finding that one static if that failed. (Oh, and did I mention, almost all of the above requires you to edit the *callee* code in order to find out where the template instantiation went? Good luck if it goes into some common templates used everywhere else, that if you change one line, something else unrelated in your program will stop compiling so you won't even get to the point of failure in the first place. Sometimes the only solution here is to copy-n-paste the entire lousy template function into a different module (possibly along with most/all of its dependencies) and call *that* instead, so that the compiler won't prematurely die in unrelated code.) This is a very real problem, and an extremely annoying one. In a hair-pullingly frustrating manner. And of course, it makes you feel *really* good 2 hours later after all of this jazz to discover that the error came from a simple typo. T -- Life is too short to run proprietary software. -- Bdale GarbeeIdeally, you would catch this in the unit tests for `bar`. You are writing unit tests, aren't you? ;)
Jan 14 2019
On 01/14/2019 03:58 PM, H. S. Teoh wrote:the error is silent, and makes the compiler just move on to a different overload, leaving you scratching your head as to why it's not picking the "correct" overload.I've recently opened the following bug which I believe is due to the same underlying issue: A compilation error is silently skipped. https://issues.dlang.org/show_bug.cgi?id=19541(Oh, and did I mention, almost all of the above requires you to edit the *callee* code in order to find out where the template instantiation went?Yep. Similar to my choice of words in that bug report: "the user [...] is now looking for unnecessary workarounds". Ali
Jan 14 2019
On 1/14/19 6:58 PM, H. S. Teoh wrote:On Mon, Jan 14, 2019 at 04:31:06PM -0700, Jonathan M Davis via Digitalmars-d wrote:So to clarify a bit, I'm not expecting miracles here. If your code is valid, but just isn't callable in the way you expect it to be, I don't see how the compiler can figure out that you meant it another way. But my overall point really is that we have a mechanism to say "Does this compile?", when we really mean "Can I use this thing this way?". I wondered if it wouldn't be worth having a way to assume that all code is syntactically valid, and just throw errors when it's not. And now that I'm thinking about it some more, I'm completely wrong. It is little typos that are syntactically valid that fail to compile -- syntax errors are still flagged (unless you are using mixins). So this doesn't help at all :( But I still face the same problems you bring up.On Monday, January 14, 2019 2:57:28 PM MST Paul Backus via Digitalmars-d wrote:Yeah, I've run into this problem far too many times. It's extremely annoying because the error is silent, and makes the compiler just move on to a different overload, leaving you scratching your head as to why it's not picking the "correct" overload.On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:How many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).[...] The problem is not just trivial syntax errors like Steven's example. The problem is that sufficiently complex sig constraints will sometimes have unexpected corner cases that works in general, but fails for specific cases. One nasty example that comes to mind is when the sig constraint only works with mutable types (or some other such qualified type) because you used an overly-restrictive test. Initial unittests work because you're not expecting that changing the input to const would cause any trouble. Then some time later, you happen to need to pass a const object, and suddenly the compiler seemingly refuses to acknowledge that this overload of the template function exists, and insists on trying a different overload, or failing with a totally unhelpful error saying no overloads match.Ideally, you would catch this in the unit tests for `bar`. You are writing unit tests, aren't you? ;)(Oh, and did I mention, almost all of the above requires you to edit the *callee* code in order to find out where the template instantiation went? Good luck if it goes into some common templates used everywhere else, that if you change one line, something else unrelated in your program will stop compiling so you won't even get to the point of failure in the first place. Sometimes the only solution here is to copy-n-paste the entire lousy template function into a different module (possibly along with most/all of its dependencies) and call *that* instead, so that the compiler won't prematurely die in unrelated code.)Oh yeah, this is a really hard problem. This captures a lot of the frustration that I've had too. I'm wondering instead of having a way to help with this, if we can't specify (for debugging purposes), "this overload should be used". In other words, get rid of the constraints and force the compiler to use a certain call. That would help IMMENSELY. I'm actually thinking of a library mechanism to do this, which is horrible, and awful, but might actually work: struct PickMe { string file; size_t line; } enum isMyOverload(...) more cruft enum ItPickedMe(T, string file == __FILE__, size_t line == __LINE__) = hasUDA!(PickMe) && anySatisfy!(isMyOverload, getUDAs!(T, PickMe)) void foo(T)(T t) if(ItPickedMe!T || __traits(compiles, t.bar())) { t.bar(); } I don't want to do this, but a compiler feature that allowed it would be cool. Having to make copies of templates to be able to debug things is so horrid. -Steve
Jan 14 2019
On Tuesday, 15 January 2019 at 01:59:58 UTC, Steven Schveighoffer wrote:I don't want to do this, but a compiler feature that allowed it would be cool. Having to make copies of templates to be able to debug things is so horrid. -SteveMaybe a sub-name system? Like: T* myFunc.withFoo(T)(T t) if(__traits(compiles, t.foo())); T* myFunc.withBar(T)(T t) if(__traits(compiles, t.bar())); Then if you call `myFunc.withFoo(t)`, you get something like: Error: Cannot call withFoo overload of myFunc Error: because constraint if(__traits(compiles, t.foo())) is not satisfied (it would be especially helpful with the "multiple constraints" DIP) It would probably be pretty hard to implement, and make name resolution even more complicated, though.
Jan 15 2019
On Mon, 14 Jan 2019 at 22:50, Steven Schveighoffer via Digitalmars-d <digitalmars-d puremagic.com> wrote:Any thoughts?Compile with -verrors=spec (or -Wspeculative if using gdc). -- Iain
Jan 14 2019
On 1/14/19 5:25 PM, Iain Buclaw wrote:On Mon, 14 Jan 2019 at 22:50, Steven Schveighoffer via Digitalmars-d <digitalmars-d puremagic.com> wrote:But wouldn't that show all errors, not just syntax errors? That might be a bit difficult to wade through. -SteveAny thoughts?Compile with -verrors=spec (or -Wspeculative if using gdc).
Jan 14 2019
On 1/14/19 5:32 PM, Steven Schveighoffer wrote:On 1/14/19 5:25 PM, Iain Buclaw wrote:BTW, I didn't know about this switch, thanks for pointing it out. -SteveOn Mon, 14 Jan 2019 at 22:50, Steven Schveighoffer via Digitalmars-d <digitalmars-d puremagic.com> wrote:But wouldn't that show all errors, not just syntax errors? That might be a bit difficult to wade through.Any thoughts?Compile with -verrors=spec (or -Wspeculative if using gdc).
Jan 14 2019
On Monday, 14 January 2019 at 22:32:59 UTC, Steven Schveighoffer wrote:On 1/14/19 5:25 PM, Iain Buclaw wrote:Yep, the SNR on that switch is abominable.On Mon, 14 Jan 2019 at 22:50, Steven Schveighoffer via Digitalmars-d <digitalmars-d puremagic.com> wrote:But wouldn't that show all errors, not just syntax errors? That might be a bit difficult to wade through. -SteveAny thoughts?Compile with -verrors=spec (or -Wspeculative if using gdc).
Jan 14 2019
On 1/14/19 5:50 PM, Nicholas Wilson wrote:On Monday, 14 January 2019 at 22:32:59 UTC, Steven Schveighoffer wrote:Could there be a way to focus that switch on certain files/lines? That would be much more useful. -SteveOn 1/14/19 5:25 PM, Iain Buclaw wrote:Yep, the SNR on that switch is abominable.On Mon, 14 Jan 2019 at 22:50, Steven Schveighoffer via Digitalmars-d <digitalmars-d puremagic.com> wrote:But wouldn't that show all errors, not just syntax errors? That might be a bit difficult to wade through.Any thoughts?Compile with -verrors=spec (or -Wspeculative if using gdc).
Jan 16 2019
On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:How many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).struct A { int f()() { return 0 } } Like this? There's no way for it to ever compile.
Jan 15 2019
On 1/15/19 3:26 AM, Kagamin wrote:On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:Indeed, I realized after posting [1] that it's really stupid typos that I make that are still valid syntax, but obviously unintentional. By that I mean, it will never compile. Like: void foo(T)(T t) { int x = 1; t.fun(X); } What I was trying to say is that I wanted a way to separate "obvious" errors from errors that happen because of the template parameters. But I think there simply isn't a way to do it. The best we can do might be to try and force specific instantiations to see why they don't compile. -Steve [1] https://forum.dlang.org/post/q1jequ$r63$1 digitalmars.comHow many times have you written something like: void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } And somehow, somewhere, this isn't called? Then you remove the constraint, and find that there's a syntax error in the bar template function (like a missing semicolon).struct A { int f()() { return 0 } } Like this? There's no way for it to ever compile.
Jan 16 2019
On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:How many times have you written something like:More times than I can count...void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } [...]I'm not sure how best to go about this.
Jan 15 2019
On Tuesday, 15 January 2019 at 11:15:21 UTC, Atila Neves wrote:On Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:Sounds like a job for `static try` too keep things DRY: void fun(T)(T t) if (__traits(compiles, t.foo())) { t.foo(); } void fun(T)(T t) if (__traits(compiles, t.bar())) { t.bar(); } // ----v void fun(T)(T t) { static try { t.bar(); } else static try { t.foo(); } else static assert(0, T.stringof ~ "does not support neither `foo`, nor `bar`"); }How many times have you written something like:More times than I can count...void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } [...]I'm not sure how best to go about this.
Jan 15 2019
On 1/15/19 6:59 AM, Petar Kirov [ZombineDev] wrote:On Tuesday, 15 January 2019 at 11:15:21 UTC, Atila Neves wrote:It might be nice to have something like this. Note that the two code samples aren't equivalent -- something that defines both bar and foo would not work with the first iteration, but would work with the second. The DRYness of the constraint is annoying too, but this misses the point that t.bar and t.foo may be intended to BOTH compile, but only one does because of a "stupid" error. And so the wrong path is taken, and you either get weird other errors, or you get no errors, but the code doesn't work right or is unintentionally lower performance. Understanding WHY a template constraint fails is important, but when you have __traits(compiles), it's a very blunt instrument. There are so many reasons why it might not compile, and it's really hard to figure out the answer, especially if you don't control the templates. Overriding the constraints might be the easiest way to probe the answers out. -SteveOn Monday, 14 January 2019 at 21:47:40 UTC, Steven Schveighoffer wrote:Sounds like a job for `static try` too keep things DRY: void fun(T)(T t) if (__traits(compiles, t.foo())) { t.foo(); } void fun(T)(T t) if (__traits(compiles, t.bar())) { t.bar(); } // ----v void fun(T)(T t) { static try { t.bar(); } else static try { t.foo(); } else static assert(0, T.stringof ~ "does not support neither `foo`, nor `bar`"); }How many times have you written something like:More times than I can count...void foo(T)(T t) if (__traits(compiles, t.bar())) // or is(typeof(t.bar())) { t.bar(); } [...]I'm not sure how best to go about this.
Jan 16 2019