1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-10 00:43:59 +02:00
vcmi/docs/developers/Logging_API.md

164 lines
6.3 KiB
Markdown
Raw Normal View History

2024-07-16 20:29:20 +02:00
# Logging API
## Features
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
- A logger belongs to a "domain", this enables us to change log level settings more selectively
2023-08-12 23:17:38 +02:00
- The log format can be customized
2023-09-01 14:25:21 +02:00
- The color of a log entry can be customized based on logger domain and logger level
2023-08-12 23:17:38 +02:00
- Logger settings can be changed in the settings.json file
- No std::endl at the end of a log entry required
- Thread-safe
- Macros for tracing the application flow
- Provides stream-like and function-like logging
2024-07-16 20:29:20 +02:00
## Class diagram
2023-08-12 23:17:38 +02:00
2023-09-07 11:57:03 +02:00
![Logging_Class_Diagram](https://github.com/IvanSavenko/vcmi/assets/1576820/31c9b14e-a055-4b07-87fe-00da43430a2b)
2023-08-12 23:17:38 +02:00
Some notes:
2023-09-01 14:25:21 +02:00
- There are two methods `configure` and `configureDefault` of the class `CBasicLogConfigurator` to initialize and setup the logging system. The latter one setups default logging and isn't dependent on VCMI's filesystem, whereas the first one setups logging based on the user's settings which can be configured in the settings.json.
- The methods `isDebugEnabled` and `isTraceEnabled` return true if a log record of level debug respectively trace will be logged. This can be useful if composing the log message is a expensive task and performance is important.
2023-08-12 23:17:38 +02:00
2024-07-16 20:29:20 +02:00
## Usage
2023-08-12 23:17:38 +02:00
2024-07-16 20:29:20 +02:00
### Setup settings.json
2023-08-12 23:17:38 +02:00
``` javascript
{
"logging" : {
"console" : {
"threshold" : "debug",
"colorMapping" : [
{
"domain" : "network",
"level" : "trace",
"color" : "magenta"
}
]
},
"loggers" : [
{
"domain" : "global",
"level" : "debug"
},
{
"domain" : "ai",
"level" : "trace"
}
]
}
}
```
2023-09-01 14:25:21 +02:00
The above code is an example on how to configure logging. It sets the log level to debug globally and the log level of the domain ai to trace. In addition, it tells the console to log debug messages as well with the threshold attribute. Finally, it configures the console so that it logs network trace messages in magenta.
2023-08-12 23:17:38 +02:00
2024-07-16 20:29:20 +02:00
### Configuration
2023-08-12 23:17:38 +02:00
The following code shows how the logging system can be configured:
2023-09-01 14:25:21 +02:00
```cpp
2023-08-12 23:17:38 +02:00
console = new CConsoleHandler;
CBasicLogConfigurator logConfig(VCMIDirs::get().localPath() + "/VCMI_Server_log.txt", console);
logConfig.configureDefault(); // Initialize default logging due to that the filesystem and settings are not available
preinitDLL(console); // Init filesystem
settings.init(); // Init settings
logConfig.configure(); // Now setup "real" logging system, overwrites default settings
2023-09-01 14:25:21 +02:00
```
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
If `configureDefault` or `configure` won't be called, then logs aren't written either to the console or to the file. The default logging setups a system like this:
2023-08-12 23:17:38 +02:00
**Console**
Format: %m
Threshold: info
coloredOutputEnabled: true
2023-09-01 14:25:21 +02:00
colorMapping: trace -\> gray, debug -\> white, info -\> green, warn -\> yellow, error -\> red
2023-08-12 23:17:38 +02:00
**File**
Format: %d %l %n \[%t\] - %m
**Loggers**
global -\> info
2024-07-16 20:29:20 +02:00
### How to get a logger
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
There exist only one logger object per domain. A logger object cannot be copied. You can get access to a logger object by using the globally defined ones like `logGlobal` or `logAi`, etc... or by getting one manually:
```cpp
Logger * logger = CLogger::getLogger(CLoggerDomain("rmg"));
```
2023-08-12 23:17:38 +02:00
2024-07-16 20:29:20 +02:00
### Logging
2023-08-12 23:17:38 +02:00
Logging can be done via two ways, stream-like or function-like.
2023-09-01 14:25:21 +02:00
```cpp
2023-08-12 23:17:38 +02:00
logGlobal->warnStream() << "Call to loadBitmap with void fname!";
logGlobal->warn("Call to loadBitmap with void fname!");
2023-09-01 14:25:21 +02:00
```
Don't include a '\n' or std::endl at the end of your log message, a new line will be appended automatically.
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
The following list shows several log levels from the highest one to the lowest one:
2023-10-17 22:06:08 +02:00
- error -\> for errors, e.g. if resource is not available, if a initialization fault has occurred, if a exception has been thrown (can result in program termination)
2023-09-01 14:25:21 +02:00
- warn -\> for warnings, e.g. if sth. is wrong, but the program can continue execution "normally"
- info -\> informational messages, e.g. Filesystem initialized, Map loaded, Server started, etc...
- debug -\> for debugging, e.g. hero moved to (12,3,0), direction 3', 'following artifacts influence X: .. or pattern detected at pos (10,15,0), p-nr. 30, flip 1, repl. 'D'
- trace -\> for logging the control flow, the execution progress or fine-grained events, e.g. hero movement completed, entering CMapEditManager::updateTerrainViews: posx '10', posy '5', width '10', height '10', mapLevel '0',...
2023-08-12 23:17:38 +02:00
The following colors are available for console output:
- default
- green
- red
- magenta
- yellow
- white
- gray
- teal
2024-07-16 20:29:20 +02:00
### How to trace execution
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
The program execution can be traced by using the macros TRACE_BEGIN, TRACE_END and their \_PARAMS counterparts. This can be important if you want to analyze the operations/internal workings of the AI or the communication of the client-server. In addition, it can help you to find bugs on a foreign VCMI installation with a custom mod configuration.
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
```cpp
2023-08-12 23:17:38 +02:00
int calculateMovementPointsForPath(int3 start, int3 end, CHero * hero) // This is just an example, the function is fictive
{
TRACE_BEGIN_PARAMS(logGlobal, "start '%s', end '%s', hero '%s'", start.toString() % end.toString() % hero.getName()); // toString is fictive as well and returns a string representation of the int3 pos, ....
int movPoints;
// Do some stuff
// ...
TRACE_END_PARAMS(logGlobal, "movPoints '%i'", movPoints);
return movPoints;
}
2023-09-01 14:25:21 +02:00
```
2023-08-12 23:17:38 +02:00
2024-07-16 20:29:20 +02:00
## Concepts
2023-08-12 23:17:38 +02:00
2024-07-16 20:29:20 +02:00
### Domain
2023-08-12 23:17:38 +02:00
2023-09-01 14:25:21 +02:00
A domain is a specific part of the software. In VCMI there exist several domains:
2023-08-12 23:17:38 +02:00
- network
- ai
- bonus
- network
2023-09-01 14:25:21 +02:00
In addition to these domains, there exist always a super domain called "global". Sub-domains can be created with "ai.battle" or "ai.adventure" for example. The dot between the "ai" and "battle" is important and notes the parent-child relationship of those two domains. A few examples how the log level will be inherited:
2023-08-12 23:17:38 +02:00
global, level=info
network, level=not set, effective level=info
global, level=warn
network, level=trace, effective level=trace
global, level=debug
ai, level=not set, effective level=debug
ai.battle, level=trace, effective level=trace
The same technique is applied to the console colors. If you want to have another debug color for the domain ai, you can explicitly set a color for that domain and level.