/** * @file setenv.h * @author Nat Goodspeed * @date 2020-04-01 * @brief Provide a way for a particular test program to alter the * environment before entry to main(). * * $LicenseInfo:firstyear=2020&license=viewerlgpl$ * Copyright (c) 2020, Linden Research, Inc. * $/LicenseInfo$ */ #if ! defined(LL_SETENV_H) #define LL_SETENV_H #include <stdlib.h> // setenv() /** * Our test.cpp main program responds to environment variables LOGTEST and * LOGFAIL. But if you set (e.g.) LOGTEST=DEBUG before a viewer build, @em * every test program in the build emits debug log output. This can be so * voluminous as to slow down the build. * * With an integration test program, you can specifically build (e.g.) the * INTEGRATION_TEST_llstring target, and set any environment variables you * want for that. But with a unit test program, since executing the program is * a side effect rather than an explicit target, specifically building (e.g.) * PROJECT_lllogin_TEST_lllogin only builds the executable without running it. * * To set an environment variable for a particular test program, declare a * static instance of SetEnv in its .cpp file. SetEnv's constructor takes * pairs of strings, e.g. * * @code * static SetEnv sLOGGING("LOGTEST", "INFO"); * @endcode * * Declaring a static instance of SetEnv is important because that ensures * that the environment variables are set before main() is entered, since it * is main() that examines LOGTEST and LOGFAIL. */ struct SetEnv { // degenerate constructor, terminate recursion SetEnv() {} /** * SetEnv() accepts an arbitrary number of pairs of strings: variable * name, value, variable name, value ... Entering the constructor sets * those variables in the process environment using Posix setenv(), * overriding any previous value. If static SetEnv declarations in * different translation units specify overlapping sets of variable names, * it is indeterminate which instance will "win." */ template <typename VAR, typename VAL, typename... ARGS> SetEnv(VAR&& var, VAL&& val, ARGS&&... rest): // constructor forwarding handles the tail of the list SetEnv(std::forward<ARGS>(rest)...) { // set just the first (variable, value) pair // 1 means override previous value if any setenv(std::forward<VAR>(var), std::forward<VAL>(val), 1); } }; #endif /* ! defined(LL_SETENV_H) */