Welcome to Software Development on Codidact!
Will you help us build our independent community of developers helping developers? We're small and trying to grow. We welcome questions about all aspects of software development, from design to code to QA and more. Got questions? Got answers? Got code you'd like someone to review? Please join us.
Unit Testing #define Directives in Visual Studio
I have some methods in my codebase that utilize the #if DEBUG
preprocessor directive to provide specialized behavior that differs between the debugging environment and the production environment. I would like to ensure that I have unit tests that validate both of these paths.
Is there a way to do this in Visual Studio without needing to re-run the tests from a different environment? Something like an Attribute that can be attached to the test method? MSTest or xUnit solutions would be preferred.
I tried simply having a RowData/InlineData and using the method parameter to #define inside of an if block, but that is explicitly disallowed:
if (debugTestState)
{
#define DEBUG
}
else
{
#undef DEBUG
}
... results in:
CS1032: Cannot define/undefine preprocessor symbols after first token in file
1 answer
To clarify why what you show can't work, it's because you are mixing build-time and run-time constructs.
All the #xxx commands are to the C pre-processor. They are used to modify the source code that the compiler ultimately sees and turns into machine language (or some type of intermediate instructions in the case of managed code like C#). These things only exist as the source code is being compiled at build time.
The IF statement is a run-time construct. It results in code the compiler puts in the executable for the program to make a decision when that program is eventually run. By the time the machine is running the program and executing the code resulting from the IF statement, the rest of the code has already been compiled and the machine code fixed. It is too late to go back and modify what the compiler is given.
Some languages are interpreted, where such things are possible. Some are partway between, like C# which gets compiled into low level opcodes that are executed by an interpreter at run time. This blurs the build-time versus run-time distinction, but you still can't do what you ask in C#.
0 comment threads