mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
application.hpp
Go to the documentation of this file.
1/** \file application.hpp
2 * \author Jared R. Males
3 * \brief A class for managing application configuration and execution
4 *
5 */
6
7//***********************************************************************//
8// Copyright 2015, 2016, 2017, 2018 Jared R. Males (jaredmales@gmail.com)
9//
10// This file is part of mxlib.
11//
12// mxlib is free software: you can redistribute it and/or modify
13// it under the terms of the GNU General Public License as published by
14// the Free Software Foundation, either version 3 of the License, or
15// (at your option) any later version.
16//
17// mxlib is distributed in the hope that it will be useful,
18// but WITHOUT ANY WARRANTY; without even the implied warranty of
19// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20// GNU General Public License for more details.
21//
22// You should have received a copy of the GNU General Public License
23// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
24//***********************************************************************//
25
26#ifndef app_application_hpp
27#define app_application_hpp
28
29#include "appConfigurator.hpp"
30#include "../ioutils/textTable.hpp"
31
32namespace mx
33{
34namespace app
35{
36
37/// A class for managing application configuration and execution
38/** Derived classes should implement at a minimum
39 *
40 * \code
41 virtual void setupConfig();
42 virtual void loadConfig();
43 virtual int execute();
44 \endcode
45 *
46 * These are executed in the order shown by the call to \ref main().
47 *
48 * This class uses a cascaded configuration system. The application configuration is built up from the following
49 sources, in increasing order of precedence:
50 * - A global configuration file
51 * - A user configuration file (which may be specified relative to the users home directory)
52 * - A local configuration file (in the pwd)
53 * - A configuration file specified on the command line
54 * - The command line
55 *
56 * At each step in the above order, values read from that configuration source override any previous settings. So the
57 command line has the highest precedence.
58 *
59 * The configuration is set up and accessed using an object of type \ref appConfigurator named `config`.
60 * For specification of the configuration file syntax and command line arguments see \ref appConfigurator.
61 *
62 * The configuration should be set up in the \ref setupConfig function. This is normally done using
63 * the appConfigurator::add method, as in:
64 * \code
65 void derived_class::setupConfig()
66 {
67 config.add("name", "s", "long", argType::Required, "section", "keyword", false, "int", "help message");
68 }
69 \endcode
70 * The configuration is then accessed using the `config` member's operator as in
71 \code
72 void derived_class::loadConfig()
73 {
74 int val = 0;
75 config(val, "name");
76 }
77 \endcode
78 * In these examples, a config target with name "name" is created. If the user sets a value of 2
79 * via the command line with `-s 2`, `--long=2` or in a configuration file with
80 \verbatim
81 [section]
82 keyword=2
83 \endverbatim
84 then val will be set to 2. Otherwise, `val` will remain 0.
85 *
86 * Standard target `-h`/`--help` with name "help" to trigger printing a help message,
87 * and `-c`/`--config` with name "config" to pass a config file via the command line. This
88 * behavior can be changed by overriding functions.
89 *
90 * \note After loadConfig() but before execute(), the containers in \ref config are de-allocated , so they can not be
91 used inside execute.
92 *
93 * A standard help message is produced when requested by the `-h`/`--help` option. This behavior can be changed
94 * by overriding the \ref help method.
95 *
96 * \ingroup mxApp
97 */
99{
100
101 protected:
102 std::string invokedName; ///< The name used to invoke this application.
103
104 std::string m_configPathGlobal; ///< The path to the gobal configuration file. Set in constructor to use.
105 std::string m_configPathGlobal_env; ///< Environment variable to check for the global config path. Set in
106 ///< constructor to use.
107 std::string m_configPathUser; ///< The path to the user's configuration file. If the first character is not '/' or
108 ///< '~', this is added to HOME. Set in constructor to use.
109 std::string
110 m_configPathUser_env; ///< Environment variable to check fo the user config path. Set in constructor to use.
111 std::string m_configPathLocal; ///< The path to a local configuration file. Set in constructor to use.
112 std::string
113 m_configPathLocal_env; ///< Environment variable to check for the local config path. Set in constructor to use.
114
116 true }; ///< Flag controlling whether lack of a global configuration file should be reported.
118 true }; ///< Flag controlling whether lack of a user configuration file should be reported.
120 true }; ///< Flag controlling whether lack of a local configuration file should be reported.
121
122 std::string m_configPathCL; ///< The path to a configuration file specified on the command line.
123 std::string m_configPathCLBase; ///< A base path to add to the CL path. Set in constructor to use.
124 std::string
125 m_configPathCLBase_env; ///< Environment variable to check for the CL base path. Set in constructor to use.
126
127 appConfigurator config; ///< The structure used for parsing and storing the configuration.
128
129 bool m_preserveConfig{ false }; ///< Flag controlling whether the configuration is cleared before execution. Set in
130 ///< derived constructor.
131
132 bool doHelp{ false }; /**< Flag to control whether the help message is printed or not.
133 If true, it is printed and the app exits */
134
135 bool m_configOnly {false}; /**< Flag to set if application should exit after loading the configuration. This
136 could be used to have a config validation mode.*/
137
138 int m_helpWidth{ 120 }; ///< The total text width available for the help message.
139 int m_helpSOColWidth{ 2 }; ///< The width of the short option (-o) column in the help message.
140 int m_helpLOColWidth{ 25 }; ///< The width of the long option (--option) column in the help message.
141 int m_helpCFColWidth{ 25 }; ///< The width of the config file option column in the help message.
142 int m_helpTypeColWidth{ 15 }; ///< The width of the argument type column in the help message.
143
144 std::string m_nonOptionHelp; ///< String to print to describe non-option arguments.
145
146 int m_argc; ///< Store argc for later use. E.g. in reReadConfig().
147 char **m_argv; ///< Store argv for later use. E.g. in reReadConfig().
148
149 public:
150 // application();
151
152 virtual ~application();
153
154 /// The application main function.
155 /** Call this from the true main function, passing the command line arguments to be processed.
156 * This calls \ref setup(), then checks if the m_doHelp flag was set. If so, it calls \ref help() and returns.
157 * If m_doHelp is not set, it then clears the config structure, and then calls \ref execute().
158 *
159 * The configuration is cleared before the call to execute, unless m_preserveConfig = true.
160 *
161 * \returns 1 if help is executed.
162 * \returns -1 on error.
163 * \returns the value of \ref execute() otherwise.
164 */
165 int main( int argc, ///< [in] standard command line result specifying number of arguments in argv
166 char **argv ///< [in] standard command line result containing the arguments.
167 );
168
169 /// Set the global configuration path
170 void setConfigPathGlobal( const std::string &s /**< [in] the new path */ );
171
172 /// Set the user configuration path
173 /** If the provided path does not star with a '/' or '~', then it will appended to
174 * the users HOME directory obtained from the environment.
175 */
176 void setConfigPathUser( const std::string &s /**< [in] the new path to the user config file*/ );
177
178 /// Set the local configuration path
179 void setConfigPathLocal( const std::string &s /**< [in] the new path */ );
180
181 protected:
182 /** \name Required Overrides
183 * These methods should be implemented in derived classes to use an mx::application in its default behavior.
184 * @{
185 */
186
187 /// In derived classes this is where the config targets are added to \ref config.
188 virtual void setupConfig();
189
190 /// Override this function to extract the configured values and set up the application.
191 virtual void loadConfig();
192
193 /// This function is where the derived class should do its work.
194 /** The application will exit with the return value of execute.
195 */
196 virtual int execute();
197
198 ///@}
199
200 /** \name Optional Overrides
201 * These methods do not need to be implemented in derived classes unless it is desired to change behavior.
202 * @{
203 */
204
205 /// Sets up the application by executing the configuration steps
206 /** This is the key method which defines the mx::application configuration system.
207 * This will not normally need to be implemented by derived clasess --
208 * only do so if you intend to change the configuration process!
209 */
210 virtual void setup( int argc, ///< [in] standard command line result specifying number of arguments in argv
211 char **argv ///< [in] standard command line result containing the arguments.
212 );
213
214 /// Re-read the config stack.
215 /** This would be used if some config targets can only be constructed after
216 * a first pass. Note that all previously read values will be appended as if
217 * entered twice, so you must be careful to only access new targets
218 * after calling this.
219 *
220 * \returns 0 on success.
221 * \returns -1 on error.
222 */
223 int reReadConfig();
224
225 /// Set the default search paths for config files
226 /** In general you should not need to redefine this.
227 *
228 * The comand line arguments are not used by the default version, but are parameters in
229 * case derived classes need access when overriding.
230 *
231 */
232 virtual void setDefaults( int argc, ///< [in] standard command line result specifying number of arguments in argv
233 char **argv ///< [in] standard command line result containing the arguments.
234 );
235
236 /// Set up the command-line config option in a standard way.
237 /** This adds "-c --config" as command line options.
238 * You can override this function to change this behavior, or simply clear config
239 * at the beginning of \ref setupConfig(). If you do this, you should also override \ref loadStandardConfig().
240 */
241 virtual void setupStandardConfig();
242
243 /// Set up the help an options in a standard way.
244 /** This adds "-h and --help" as command line options.
245 * You can override this function to change this behavior, or simply clear config
246 * at the beginning of \ref setupConfig(). If you do this, you should also override \ref loadStandardHelp().
247 */
248 virtual void setupStandardHelp();
249
250 /// Loads the values of "config".
251 /** Override this function if you do not want to use this, or have different behavior.
252 * See also \ref setupStandardConfig().
253 */
254 virtual void loadStandardConfig();
255
256 /// Loads the value of "help" into m_doHelp.
257 /** Override this function if you do not want to use this, or have different behavior.
258 * See also \ref setupStandardConfig().
259 */
260 virtual void loadStandardHelp();
261
262 /// Format a configTarget for the help message.
263 virtual void optionHelp( configTarget &tgt, ///< [in] The Target
264 ioutils::textTable &tt ///< [out] the textTable being populated
265 );
266
267 /// Setup a basic configuration. Can be used in an intermediate derived class to allow its children to use
268 /// setupConfig.
269 /** This is called just before setupConfig().
270 */
271 virtual void setupBasicConfig();
272
273 /// Load a basic configuration. Can be used in an intermediate derived class to allow its children to use
274 /// loadConfig.
275 /** This is called just before loadConfig().
276 */
277 virtual void loadBasicConfig();
278
279 /// Check the config. This is called at the end of setup, before the configuration is cleared.
280 /** It is up to you to decide how to handle the outcome. If a bad config results in printing help,
281 * you can set the m_doHelp flag.
282 */
283 virtual void checkConfig();
284
285 /// Print a formatted help message, based on the config target inputs.
286 virtual void help();
287
288 ///@}
289};
290
291} // namespace app
292} // namespace mx
293
294#endif // app_application_hpp
295
296// Deprecations:
297// Produce errors if older code is compiled which attempts to define-config the config:
298
299#ifdef MX_APP_DEFAULT_configPathGlobal
300#error MX_APP_DEFAULT_configPathGlobal no longer works. Set m_configPathGlobal in constructor.
301#endif
302
303#ifdef MX_APP_DEFAULT_configPathGlobal_env
304#error MX_APP_DEFAULT_configPathGlobal_env no longer works. Set m_configPathGlobal_env in constructor.
305#endif
306
307#ifdef MX_APP_DEFAULT_configPathUser
308#error MX_APP_DEFAULT_configPathUser no longer works. Set m_configPathUser in constructor.
309#endif
310
311#ifdef MX_APP_DEFAULT_configPathUser_env
312#error MX_APP_DEFAULT_configPathUser_env no longer works. Set m_configPathUser_env in constructor.
313#endif
314
315#ifdef MX_APP_DEFAULT_configPathLocal
316#error MX_APP_DEFAULT_configPathLocal no longer works. Set m_configPathLocal in constructor.
317#endif
318
319#ifdef MX_APP_DEFAULT_configPathLocal_env
320#error MX_APP_DEFAULT_configPathLocal_env no longer works. Set m_configPathLocal_env in constructor.
321#endif
322
323#ifdef MX_APP_DEFAULT_configPathCLBase_env
324#error MX_APP_DEFAULT_configPathCLBase_env no longer works. Set m_configPathCLBase_env in constructor.
325#endif
An application configuration manager.
A class for managing application configuration and execution.
virtual void setup(int argc, char **argv)
Sets up the application by executing the configuration steps.
std::string invokedName
The name used to invoke this application.
std::string m_configPathCLBase
A base path to add to the CL path. Set in constructor to use.
virtual void setupStandardConfig()
Set up the command-line config option in a standard way.
std::string m_configPathUser
bool m_requireConfigPathUser
Flag controlling whether lack of a user configuration file should be reported.
std::string m_configPathUser_env
Environment variable to check fo the user config path. Set in constructor to use.
virtual void checkConfig()
Check the config. This is called at the end of setup, before the configuration is cleared.
virtual void setupStandardHelp()
Set up the help an options in a standard way.
virtual void loadBasicConfig()
int reReadConfig()
Re-read the config stack.
virtual void loadStandardConfig()
Loads the values of "config".
int main(int argc, char **argv)
The application main function.
char ** m_argv
Store argv for later use. E.g. in reReadConfig().
virtual void loadConfig()
Override this function to extract the configured values and set up the application.
int m_argc
Store argc for later use. E.g. in reReadConfig().
void setConfigPathUser(const std::string &s)
Set the user configuration path.
std::string m_configPathLocal_env
Environment variable to check for the local config path. Set in constructor to use.
int m_helpLOColWidth
The width of the long option (–option) column in the help message.
int m_helpCFColWidth
The width of the config file option column in the help message.
int m_helpTypeColWidth
The width of the argument type column in the help message.
appConfigurator config
The structure used for parsing and storing the configuration.
bool m_requireConfigPathLocal
Flag controlling whether lack of a local configuration file should be reported.
int m_helpWidth
The total text width available for the help message.
virtual void optionHelp(configTarget &tgt, ioutils::textTable &tt)
Format a configTarget for the help message.
virtual void setDefaults(int argc, char **argv)
Set the default search paths for config files.
virtual void help()
Print a formatted help message, based on the config target inputs.
virtual void setupBasicConfig()
virtual int execute()
This function is where the derived class should do its work.
std::string m_configPathCL
The path to a configuration file specified on the command line.
std::string m_configPathGlobal
The path to the gobal configuration file. Set in constructor to use.
virtual void loadStandardHelp()
Loads the value of "help" into m_doHelp.
std::string m_configPathGlobal_env
void setConfigPathGlobal(const std::string &s)
Set the global configuration path.
virtual void setupConfig()
In derived classes this is where the config targets are added to config.
int m_helpSOColWidth
The width of the short option (-o) column in the help message.
std::string m_configPathLocal
The path to a local configuration file. Set in constructor to use.
std::string m_configPathCLBase_env
Environment variable to check for the CL base path. Set in constructor to use.
std::string m_nonOptionHelp
String to print to describe non-option arguments.
void setConfigPathLocal(const std::string &s)
Set the local configuration path.
bool m_requireConfigPathGlobal
Flag controlling whether lack of a global configuration file should be reported.
The mxlib c++ namespace.
Definition mxError.hpp:106
Class to manage a set of configurable values, and read their values from config/ini files and the com...
A configuration target.
An ascii table formatter.
Definition textTable.hpp:61