-
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
nix errors phase 2 #3590
nix errors phase 2 #3590
Conversation
src/error-demo/error-demo.cc
Outdated
|
||
// In each program where errors occur, this has to be set. | ||
ErrorInfo::programName = std::optional("error-demo"); | ||
|
||
// 'DemoError' appears as error name. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wonder if it would be possible to turn this demo
file into unit tests instead? Would that be feasible? WDYT @bburdette ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am not opposed. What do you envision as tests? String comparison of predicted vs actual output?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes i think that is what I would envision. I have to admit i didn’t yet find the time to take a closer look so maybe some parts might be hard(er) to test when log functions just print to stderr bur whenever an Exception is thrown with a message that encodes information in a certain format that could be tested.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I actually just spent some time on this and am almost done converting everything from your error-demo.cc
into a unit test. I'll probably have it done later this evening for you to check out.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wow just saw this! I've been working on it too, but sounds like you're further along. I'll look for that!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please have a look at:
gilligan@952e72c
I would suggest to remove error-demo
and replace it with that. Some notes:
- Shortly after i started i realized that this would really be tests for
logging.h
- I am using the google test to capture output and assert that. That seemed like the best approach to me
- You will see notes with
XXX
where i don't quite understand whydescription
is not set. - The asserts do look a bit ugly. It might be nicer to use c++ raw strings so the expected output is actually human readable. Using those in the
json.hh
tests:
ASSERT_EQ(out.str(),
R"#({
"list": [
"element"
]
})#");
I was using those at first but then realized that because of the escape codes for coloring they wouldn't work. I guess ideally we'd still use those and add the escape codes (?). Or we leave that as an improvement for later. I'll leave it up to you.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice! The testing::internal::CaptureStderr() is helpful. My approach was doing a test in bash; this is better I think.
I do wonder about the empty strings on Talkative and etc. On the plus side, its only the logError and logWarning that are used in actual code at this point. Seems like a problem with the stderr capture somehow? "Chatty" still has a "talkative" description; I made a note.
Anyway overall LGTM. Thanks for expediting this!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I’m super excited about this. Very much looking forward to this getting over the finishing line! Would be cool if my comments on testing and documentation could be incorporated.
const std::string nativeSystem = SYSTEM; | ||
|
||
|
||
BaseError & BaseError::addPrefix(const FormatOrString & fs) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you add a short comment documenting this method?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
comment added!
return *this; | ||
} | ||
|
||
const string& BaseError::calcWhat() const |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you add a short comment documenting this method?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
} | ||
|
||
void printCodeLines(const string &prefix, const NixCode &nixCode) | ||
void printCodeLines(std::ostream &out, const string &prefix, const NixCode &nixCode) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you add a short comment documenting this method?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
comment added!
src/libutil/error.hh
Outdated
// format function for hints. same as fmt, except templated values | ||
// are always in yellow. | ||
// ------------------------------------------------- | ||
// ErrorInfo. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Repeating the name of the structure isn’t a very helpful comment 😅
I wonder if you could possibly add a brief description to this header file that would enable someone to grasp the overall design of this “error api”?
The Nix code base mostly doesn’t have a lot of inline comments/documentation but i think it would be fantastic if we could slowly improve that.
Edit: i guess slightly changed parts of the PR description would be a good fit?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok added some comments at the top and removed the "ErrorInfo."
The logging.hh superseeds the demo
@bburdette thanks for incorporating the tests! |
Hope we can get this merged soon! The status quo with the demo merged but not integrated is perfectly understandable as an intermediate step, but not something we should leave around! |
Also, I was just ad-hoc adding some |
@edolstra is there anything that in your view makes this PR not ready to be merged? |
@bburdette Thanks, merged! |
Awesome, thanks for the merge!! |
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, but is only targeted for folks using an unstable Nix version. Also, this improved-error-message feature is still work in progress, so the behavior of `nix-instantiate` is still subject to change. [1] NixOS/nix#3590
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, but is only targeted for folks using an unstable Nix version. Please note that this currently depends on a pending PR[2] which adds support for minimal JSON error-outputs. [1] NixOS/nix#3590 [2] NixOS/nix#3847
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, but is only targeted for folks using an unstable Nix version. Please note that this currently depends on a pending PR[2] which adds support for minimal JSON error-outputs. [1] NixOS/nix#3590 [2] NixOS/nix#3847
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, but is only targeted for folks using an unstable Nix version. Please note that this currently depends on a pending PR[2] which adds support for minimal JSON error-outputs. [1] NixOS/nix#3590 [2] NixOS/nix#3847
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, for both stable Nix (2.3 and older) and unstable Nix (2.4 and newer). [1] NixOS/nix#3590
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, for both stable Nix (2.3 and older) and unstable Nix (2.4 and newer). [1] NixOS/nix#3590
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, for both stable Nix (2.3 and older) and unstable Nix (2.4 and newer). [1] NixOS/nix#3590
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, for both stable Nix (2.3 and older) and unstable Nix (2.4 and newer). [1] NixOS/nix#3590
In the (unreleased) Nix 2.4 the error-messages have been reformatted[1]. This patch aims to retain proper `.nix`-support in `ale`, for both stable Nix (2.3 and older) and unstable Nix (2.4 and newer). [1] NixOS/nix#3590
This PR is for phase 2 of the nix-error-proposal project.
Overall the idea is to enhance the format of error messages, and their content. This phase is just for the formatting part, the content of error messages is largely left alone.
To use the new format there are some new macros in logging.hh:
These require an ErrorInfo struct as an argument instead of a format string. In the default logger, the ErrorInfo is used to print errors to cerr. This late conversion of ErrorInfo to string should help with future language server support, or other similar efforts.
These macros prevent creating unnecessary ErrorInfo structs, similar to the printInfo and printError macros. That is, filtering takes place before the struct is created, not after.
ErrorInfo uses a special fmt - 'hintfmt' - that highlights templated values in yellow. For consistency, I changed the printError/warn/printInfo macros to use the same format style. That also required removing the remaining calls to boost::format.
ErrorInfo usage fits into two big categories: BaseError descendants, and converted printError messages.
Exceptions
BaseError descendants - Error, SysError, and so forth, all use the new format now. The MakeError macro uses the class name as the error name, so SysError will print:
Pos goes into ErrorInfo as a special field, and gets printed at a standardized place in the message. It got removed from the string templates so as not to duplicate information.
printError
For printError messages, there were more decisions to make. The ErrorInfo format is pretty heavy and not always appropriate. I categorized printError calls as follows:
- converted to ErrorInfo format.
- converted to printInfo.
- converted to logWarning()
- left alone.
warn() messages.
these I left alone for now; reasoning that the
printError("warning..."
ones are more important, so those are distiguished by logWarning calls. warn() calls are intact other than converting from boost::format.What's missing
This is already a huge PR, so I'm leaving some things out that belong in phase 2.
show-trace is not included in ErrorInfo. It still works as it has in the past, but its printed to cerr and doesn't make it into the logger except as strings.
lines of code. This is a big part of the new format! But I think there's enough to look at in this PR. Its next on my list.
Moving code around.
BaseError and descendants got moved out of types.hh (and util.cc) and into error.hh.
ErrorInfo now uses Verbosity instead of its own ErrLevel type. Consequently, Verbosity was moved from logging.hh to error.hh.
Theres a new header, fmt.hh, where fmt, FormatOrString, hintfmt, and so forth are defined.
Misc
In eval.cc and eval-inline are some functions like throwEvalError, that used pos in a string template. With pos being passed to ErrorInfo instead, I found it confusing to leave pos as the last argument in the function:
It seems like cpu affinity was meant to be printed in affinity.cc, but wasn't. I added an ostream operator<< for affinity, so I could print it: