mirror of
				https://github.com/FairRootGroup/FairLogger.git
				synced 2025-10-20 20:11:43 +00:00 
			
		
		
		
	Compare commits
	
		
			70 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | cd9e392ce0 | ||
|  | 27527ad87b | ||
|  | f3d68fb4ba | ||
|  | 0ae9a697b4 | ||
|  | fdbc047591 | ||
|  | ea4067c474 | ||
|  | 067dcc0e26 | ||
|  | 0ab390c465 | ||
|  | 56b90a7445 | ||
|  | 7fc05f3808 | ||
|  | d93fb0870b | ||
|  | 953eac19c8 | ||
|  | 9a6644c931 | ||
|  | 27040ea292 | ||
|  | 6545d0efa2 | ||
|  | 159a21c24b | ||
|  | 2494a9e85a | ||
|  | d0b109015d | ||
|  | ba7da2f79a | ||
|  | 1c7e7f566a | ||
|  | b2dfdd1768 | ||
|  | 13ebedca3d | ||
|  | 1c43450582 | ||
|  | f9af3a7269 | ||
|  | 064e034d25 | ||
|  | 3f55a16850 | ||
|  | 340b005997 | ||
|  | f012a870e7 | ||
|  | bcfe438862 | ||
|  | 4b3e6d3837 | ||
|  | b30cacab12 | ||
|  | c8d59d11fb | ||
|  | a737a1de9c | ||
|  | 1cb941021c | ||
|  | de1014dabb | ||
|  | ce64f628b0 | ||
|  | 35ebc10204 | ||
|  | 1253bbbac8 | ||
|  | e9bd3f2b62 | ||
|  | 1f3b2a2c82 | ||
|  | 2b37d0147e | ||
|  | f0cbe0bd47 | ||
|  | 5f7197d987 | ||
|  | b1cb6f8a99 | ||
|  | 56780689fc | ||
|  | 8446c6db0c | ||
|  | a0ff4eba50 | ||
|  | cfe0f9dc53 | ||
|  | cdf887f5da | ||
|  | 6555aa1dc0 | ||
|  | d1f73fe9f0 | ||
|  | 86ab87de7b | ||
|  | b9edcd623d | ||
|  | 0670b5dba6 | ||
|  | b65084f1ce | ||
|  | da9c36702c | ||
|  | 6cc60e962b | ||
|  | de955b78da | ||
|  | e6a279f58a | ||
|  | 8073d2982c | ||
|  | 2a7d4dfd9a | ||
|  | bb5e67a5e7 | ||
|  | edbc8e6270 | ||
|  | afb468406a | ||
|  | a5f3e95238 | ||
|  | 3d36ffeb40 | ||
|  | d9ac93552e | ||
|  | dcc27744cd | ||
|  | 4b883688c9 | ||
|  | 5eb2612636 | 
							
								
								
									
										101
									
								
								CMakeLists.txt
									
									
									
									
									
								
							
							
						
						
									
										101
									
								
								CMakeLists.txt
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| ################################################################################ | ################################################################################ | ||||||
| # Copyright (C) 2018-2019 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | # Copyright (C) 2018-2024 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | ||||||
| #                                                                              # | #                                                                              # | ||||||
| #              This software is distributed under the terms of the             # | #              This software is distributed under the terms of the             # | ||||||
| #              GNU Lesser General Public Licence (LGPL) version 3,             # | #              GNU Lesser General Public Licence (LGPL) version 3,             # | ||||||
| @@ -7,15 +7,16 @@ | |||||||
| ################################################################################ | ################################################################################ | ||||||
|  |  | ||||||
| cmake_minimum_required(VERSION 3.9.4 FATAL_ERROR) | cmake_minimum_required(VERSION 3.9.4 FATAL_ERROR) | ||||||
| cmake_policy(VERSION 3.9...3.14) | cmake_policy(VERSION 3.9...3.30) | ||||||
|  |  | ||||||
| # Project ###################################################################### | # Project ###################################################################### | ||||||
| set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake) | set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake) | ||||||
| include(FairLoggerLib) | include(FairLoggerLib) | ||||||
|  | include(FairFindPackage2) | ||||||
|  |  | ||||||
| get_git_version() | get_git_version() | ||||||
|  |  | ||||||
| project(FairLogger VERSION ${PROJECT_VERSION} LANGUAGES C CXX) | project(FairLogger VERSION ${PROJECT_VERSION} LANGUAGES CXX) | ||||||
| message(STATUS "${BWhite}${PROJECT_NAME}${CR} ${PROJECT_GIT_VERSION} from ${PROJECT_DATE}") | message(STATUS "${BWhite}${PROJECT_NAME}${CR} ${PROJECT_GIT_VERSION} from ${PROJECT_DATE}") | ||||||
|  |  | ||||||
| set_fairlogger_defaults() | set_fairlogger_defaults() | ||||||
| @@ -23,10 +24,10 @@ set_fairlogger_defaults() | |||||||
| include(CTest) | include(CTest) | ||||||
|  |  | ||||||
| option(USE_BOOST_PRETTY_FUNCTION "Use Boost BOOST_PRETTY_FUNCTION macro" OFF) | option(USE_BOOST_PRETTY_FUNCTION "Use Boost BOOST_PRETTY_FUNCTION macro" OFF) | ||||||
|  | option(USE_EXTERNAL_FMT "Use external fmt library instead of the bundled one" OFF) | ||||||
| ################################################################################ | ################################################################################ | ||||||
|  |  | ||||||
|  | # Dependencies ################################################################# | ||||||
| # Dependency ################################################################### |  | ||||||
| if(USE_BOOST_PRETTY_FUNCTION) | if(USE_BOOST_PRETTY_FUNCTION) | ||||||
|   if(NOT DEFINED Boost_NO_BOOST_CMAKE AND CMAKE_VERSION VERSION_LESS 3.15) |   if(NOT DEFINED Boost_NO_BOOST_CMAKE AND CMAKE_VERSION VERSION_LESS 3.15) | ||||||
|     # Since Boost 1.70 a CMake package is shipped by default. Unfortunately, it has a number |     # Since Boost 1.70 a CMake package is shipped by default. Unfortunately, it has a number | ||||||
| @@ -36,8 +37,11 @@ if(USE_BOOST_PRETTY_FUNCTION) | |||||||
|   endif() |   endif() | ||||||
|   find_package2(PUBLIC Boost REQUIRED) |   find_package2(PUBLIC Boost REQUIRED) | ||||||
| endif() | endif() | ||||||
| ################################################################################ |  | ||||||
|  |  | ||||||
|  | if(USE_EXTERNAL_FMT) | ||||||
|  |   find_package2(PUBLIC fmt REQUIRED VERSION 5.3.0) | ||||||
|  | endif() | ||||||
|  | ################################################################################ | ||||||
|  |  | ||||||
| # Targets ###################################################################### | # Targets ###################################################################### | ||||||
| # Configure Version.h | # Configure Version.h | ||||||
| @@ -50,16 +54,34 @@ add_library(FairLogger | |||||||
|   logger/Logger.cxx |   logger/Logger.cxx | ||||||
|   logger/Logger.h |   logger/Logger.h | ||||||
| ) | ) | ||||||
|  | target_compile_features(FairLogger PUBLIC cxx_std_17) | ||||||
|  |  | ||||||
| if(USE_BOOST_PRETTY_FUNCTION) | if(USE_BOOST_PRETTY_FUNCTION) | ||||||
|   target_link_libraries(FairLogger PUBLIC Boost::boost) |   target_link_libraries(FairLogger PUBLIC Boost::boost) | ||||||
|   target_compile_definitions(FairLogger PUBLIC FAIRLOGGER_USE_BOOST_PRETTY_FUNCTION) |   target_compile_definitions(FairLogger PUBLIC FAIRLOGGER_USE_BOOST_PRETTY_FUNCTION) | ||||||
| endif() | endif() | ||||||
|  |  | ||||||
|  | if(USE_EXTERNAL_FMT) | ||||||
|  |   target_link_libraries(FairLogger PUBLIC fmt::fmt) | ||||||
|  | else() | ||||||
|  |   add_library(fmt INTERFACE) | ||||||
|  |   target_include_directories(fmt INTERFACE | ||||||
|  |     $<BUILD_INTERFACE:${CMAKE_SOURCE_DIR}/logger/bundled> | ||||||
|  |     $<INSTALL_INTERFACE:${PROJECT_INSTALL_BUNDLEDINCDIR}> | ||||||
|  |   ) | ||||||
|  |   target_compile_features(fmt INTERFACE cxx_std_11) | ||||||
|  |   target_compile_definitions(fmt INTERFACE FMT_HEADER_ONLY) | ||||||
|  |   target_link_libraries(FairLogger PUBLIC fmt) | ||||||
|  | endif() | ||||||
|  |  | ||||||
|  | if(DEFINED FAIR_MIN_SEVERITY) | ||||||
|  |   target_compile_definitions(FairLogger PUBLIC "FAIR_MIN_SEVERITY=${FAIR_MIN_SEVERITY}") | ||||||
|  | endif() | ||||||
|  |  | ||||||
| target_include_directories(FairLogger | target_include_directories(FairLogger | ||||||
|     PUBLIC |   PUBLIC | ||||||
|     $<BUILD_INTERFACE:${CMAKE_SOURCE_DIR}/logger> |   $<BUILD_INTERFACE:${CMAKE_SOURCE_DIR}/logger> | ||||||
|     $<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}> |   $<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}> | ||||||
| ) | ) | ||||||
| set_target_properties(FairLogger PROPERTIES | set_target_properties(FairLogger PROPERTIES | ||||||
|   VERSION ${PROJECT_GIT_VERSION} |   VERSION ${PROJECT_GIT_VERSION} | ||||||
| @@ -67,19 +89,35 @@ set_target_properties(FairLogger PROPERTIES | |||||||
| ) | ) | ||||||
|  |  | ||||||
| if(BUILD_TESTING) | if(BUILD_TESTING) | ||||||
|   add_executable(loggerTest test/loggerTest.cxx) |   add_executable(cycleTest test/cycle.cxx) | ||||||
|   target_link_libraries(loggerTest FairLogger pthread) |   target_link_libraries(cycleTest FairLogger) | ||||||
|  |   add_executable(loggerTest test/logger.cxx) | ||||||
|  |   target_link_libraries(loggerTest FairLogger) | ||||||
|  |   add_executable(macrosTest test/macros.cxx) | ||||||
|  |   target_link_libraries(macrosTest FairLogger) | ||||||
|  |   add_executable(nologTest test/nolog.cxx) | ||||||
|  |   target_link_libraries(nologTest FairLogger) | ||||||
|  |   add_executable(severityTest test/severity.cxx) | ||||||
|  |   target_link_libraries(severityTest FairLogger) | ||||||
|  |   add_executable(sinksTest test/sinks.cxx) | ||||||
|  |   target_link_libraries(sinksTest FairLogger) | ||||||
|  |   add_executable(threadsTest test/threads.cxx) | ||||||
|  |   target_link_libraries(threadsTest FairLogger pthread) | ||||||
|  |   add_executable(verbosityTest test/verbosity.cxx) | ||||||
|  |   target_link_libraries(verbosityTest FairLogger) | ||||||
| endif() | endif() | ||||||
| ################################################################################ | ################################################################################ | ||||||
|  |  | ||||||
|  |  | ||||||
| # Installation ################################################################# | # Installation ################################################################# | ||||||
| if(BUILD_TESTING) | if(BUILD_TESTING) | ||||||
|   set(test_targets ${targets} loggerTest) |   set(test_targets ${targets} loggerTest) | ||||||
| endif() | endif() | ||||||
|  | if(NOT USE_EXTERNAL_FMT) | ||||||
|  |   set(fmt_target fmt) | ||||||
|  | endif() | ||||||
| install(TARGETS | install(TARGETS | ||||||
|   FairLogger |   FairLogger | ||||||
|   ${test_targets} |   ${fmt_target} | ||||||
|  |  | ||||||
|   EXPORT ${PROJECT_EXPORT_SET} |   EXPORT ${PROJECT_EXPORT_SET} | ||||||
|   LIBRARY DESTINATION ${PROJECT_INSTALL_LIBDIR} |   LIBRARY DESTINATION ${PROJECT_INSTALL_LIBDIR} | ||||||
| @@ -94,18 +132,36 @@ install(FILES | |||||||
|   DESTINATION ${PROJECT_INSTALL_INCDIR} |   DESTINATION ${PROJECT_INSTALL_INCDIR} | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | if(NOT USE_EXTERNAL_FMT) | ||||||
|  |   install(DIRECTORY | ||||||
|  |     logger/bundled | ||||||
|  |     DESTINATION ${PROJECT_INSTALL_INCDIR} | ||||||
|  |   ) | ||||||
|  | endif() | ||||||
|  |  | ||||||
| install_cmake_package() | install_cmake_package() | ||||||
| ################################################################################ | ################################################################################ | ||||||
|  |  | ||||||
|  |  | ||||||
| # Testing ###################################################################### | # Testing ###################################################################### | ||||||
| if(BUILD_TESTING) | if(BUILD_TESTING) | ||||||
|   add_test(NAME loggerTest |   add_test(NAME cycle COMMAND $<TARGET_FILE:cycleTest>) | ||||||
|     COMMAND $<TARGET_FILE:loggerTest> |   add_test(NAME logger COMMAND $<TARGET_FILE:loggerTest>) | ||||||
|   ) |   add_test(NAME macros COMMAND $<TARGET_FILE:macrosTest>) | ||||||
|  |   add_test(NAME nolog COMMAND $<TARGET_FILE:nologTest>) | ||||||
|  |   add_test(NAME severity COMMAND $<TARGET_FILE:severityTest>) | ||||||
|  |   add_test(NAME sinks COMMAND $<TARGET_FILE:sinksTest>) | ||||||
|  |   add_test(NAME threads COMMAND $<TARGET_FILE:threadsTest>) | ||||||
|  |   add_test(NAME verbosity COMMAND $<TARGET_FILE:verbosityTest>) | ||||||
| endif() | endif() | ||||||
| ################################################################################ | ################################################################################ | ||||||
|  |  | ||||||
|  | # Utils ######################################################################## | ||||||
|  | find_program(BASH bash HINTS "/bin") | ||||||
|  | find_program(SINGULARITY singularity) | ||||||
|  | if(BASH AND SINGULARITY) | ||||||
|  |   add_subdirectory(test/ci) | ||||||
|  | endif() | ||||||
|  | ################################################################################ | ||||||
|  |  | ||||||
| # Summary ###################################################################### | # Summary ###################################################################### | ||||||
| message(STATUS "  ") | message(STATUS "  ") | ||||||
| @@ -166,6 +222,9 @@ if(PROJECT_PACKAGE_DEPENDENCIES) | |||||||
|         get_target_property(boost_include Boost::boost INTERFACE_INCLUDE_DIRECTORIES) |         get_target_property(boost_include Boost::boost INTERFACE_INCLUDE_DIRECTORIES) | ||||||
|       endif() |       endif() | ||||||
|       get_filename_component(prefix ${boost_include}/.. ABSOLUTE) |       get_filename_component(prefix ${boost_include}/.. ABSOLUTE) | ||||||
|  |     elseif(${dep} STREQUAL fmt) | ||||||
|  |       get_target_property(fmt_include fmt::fmt INTERFACE_INCLUDE_DIRECTORIES) | ||||||
|  |       get_filename_component(prefix ${fmt_include}/.. ABSOLUTE) | ||||||
|     else() |     else() | ||||||
|       get_filename_component(prefix ${${dep}_INCLUDE_DIR}/.. ABSOLUTE) |       get_filename_component(prefix ${${dep}_INCLUDE_DIR}/.. ABSOLUTE) | ||||||
|     endif() |     endif() | ||||||
| @@ -185,4 +244,12 @@ else() | |||||||
| endif() | endif() | ||||||
| message(STATUS "  ${BWhite}tests${CR}       ${testing_summary}") | message(STATUS "  ${BWhite}tests${CR}       ${testing_summary}") | ||||||
| message(STATUS "  ") | message(STATUS "  ") | ||||||
|  | if(DEFINED FAIR_MIN_SEVERITY) | ||||||
|  |   message(STATUS "  ${Cyan}FAIR_MIN_SEVERITY${CR}  ${BGreen}${FAIR_MIN_SEVERITY}${CR} (change with ${BMagenta}-DFAIR_MIN_SEVERITY=...${CR})") | ||||||
|  | else() | ||||||
|  |   message(STATUS "  ${Cyan}FAIR_MIN_SEVERITY${CR}  not defined${CR}, enabling all severities (change with ${BMagenta}-DFAIR_MIN_SEVERITY=...${CR})") | ||||||
|  | endif() | ||||||
|  | message(STATUS "  ") | ||||||
|  | message(STATUS "  ${Cyan}INSTALL PREFIX${CR}     ${BGreen}${CMAKE_INSTALL_PREFIX}${CR} (change with ${BMagenta}-DCMAKE_INSTALL_PREFIX=...${CR})") | ||||||
|  | message(STATUS "  ") | ||||||
| ################################################################################ | ################################################################################ | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								COPYRIGHT
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								COPYRIGHT
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/ | ||||||
|  | Upstream-Name: FairLogger | ||||||
|  | Upstream-Contact: Mohammad Al-Turany <m.al-turany@gsi.de> | ||||||
|  | Source: https://github.com/FairRootGroup/FairLogger | ||||||
|  |  | ||||||
|  | Files: * | ||||||
|  | Copyright: 2017-2019, GSI Helmholtzzentrum fuer Schwerionenforschung GmbH | ||||||
|  | Comment: The copyright of individual contributors is documented in the | ||||||
|  |   Git history. | ||||||
|  | License: LGPL-3.0-only | ||||||
|  |  | ||||||
|  | Files: logger/bundled/fmt/* | ||||||
|  | Copyright (c) 2012 - present, Victor Zverovich | ||||||
|  | License: FMT | ||||||
|  |  | ||||||
|  | License: LGPL-3.0-only | ||||||
|  |   [see LICENSE file] | ||||||
|  |  | ||||||
|  | License: FMT | ||||||
|  |   All rights reserved. | ||||||
|  |  | ||||||
|  |   Redistribution and use in source and binary forms, with or without | ||||||
|  |   modification, are permitted provided that the following conditions are met: | ||||||
|  |  | ||||||
|  |   1. Redistributions of source code must retain the above copyright notice, this | ||||||
|  |     list of conditions and the following disclaimer. | ||||||
|  |   2. Redistributions in binary form must reproduce the above copyright notice, | ||||||
|  |     this list of conditions and the following disclaimer in the documentation | ||||||
|  |     and/or other materials provided with the distribution. | ||||||
|  |  | ||||||
|  |   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND | ||||||
|  |   ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||||||
|  |   WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||||
|  |   DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR | ||||||
|  |   ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||||||
|  |   (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||||||
|  |   LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||||||
|  |   ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||||||
|  |   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||||||
|  |   SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||||
							
								
								
									
										133
									
								
								Dart.sh
									
									
									
									
									
								
							
							
						
						
									
										133
									
								
								Dart.sh
									
									
									
									
									
								
							| @@ -1,133 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
|  |  | ||||||
| function print_example(){ |  | ||||||
| echo "##################################################################" |  | ||||||
| echo "# To set the required parameters as source and the build         #" |  | ||||||
| echo "# directory for ctest, the linux flavour and the SIMPATH         #" |  | ||||||
| echo "# put the export commands below to a separate file which is read #" |  | ||||||
| echo "# during execution and which is defined on the command line.     #" |  | ||||||
| echo "# Set all parameters according to your needs.                    #" |  | ||||||
| echo "# LINUX_FLAVOUR should be set to the distribution you are using  #" |  | ||||||
| echo "# eg Debian, SuSe etc.                                           #" |  | ||||||
| echo "# An additional varibale NCPU can overwrite the default number   #" |  | ||||||
| echo "# of parallel processes used to compile the project.             #" |  | ||||||
| echo "# This can be usefull if one can use a distributed build system  #" |  | ||||||
| echo "# like icecream.                                                 #" |  | ||||||
| echo "# For example                                                    #" |  | ||||||
| echo "#!/bin/bash                                                      #" |  | ||||||
| echo "export LINUX_FLAVOUR=<your linux flavour>                        #" |  | ||||||
| echo "export FAIRSOFT_VERSION=<version of FairSoft>                    #" |  | ||||||
| echo "export SIMPATH=<path to your FairSoft version>                   #" |  | ||||||
| echo "export GIT_BRANCH=< master or dev>                               #" |  | ||||||
| echo "export BUILDDIR=<dir where the build files go>                   #" |  | ||||||
| echo "export SOURCEDIR=<location of the FairRoot sources>              #" |  | ||||||
| echo "#export NCPU=100                                                 #" |  | ||||||
| echo "##################################################################" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| if [ "$#" -lt "2" ]; then |  | ||||||
|   echo "" |  | ||||||
|   echo "-- Error -- Please start script with two parameters" |  | ||||||
|   echo "-- Error -- The first parameter is the ctest model." |  | ||||||
|   echo "-- Error -- Possible arguments are Nightly, Experimental, " |  | ||||||
|   echo "-- Error -- Continuous or Profile." |  | ||||||
|   echo "-- Error -- The second parameter is the file containg the" |  | ||||||
|   echo "-- Error -- Information about the setup at the client" |  | ||||||
|   echo "-- Error -- installation (see example below)." |  | ||||||
|   echo "" |  | ||||||
|   print_example |  | ||||||
|   exit 1 |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # test if a valid ctest model is defined |  | ||||||
| if [ "$1" == "Experimental" -o "$1" == "Nightly" -o "$1" == "Continuous" -o "$1" == "Profile" -o "$1" == "alfa_ci" ]; then |  | ||||||
|   echo "" |  | ||||||
| else |  | ||||||
|   echo "-- Error -- This ctest model is not supported." |  | ||||||
|   echo "-- Error -- Possible arguments are Nightly, Experimental, Continuous or Profile." |  | ||||||
|   exit 1 |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # test if the input file exists and execute it |  | ||||||
| if [ -e "$2" ];then |  | ||||||
|   source $2 |  | ||||||
| else |  | ||||||
|   echo "-- Error -- Input file does not exist." |  | ||||||
|   echo "-- Error -- Please choose existing input file." |  | ||||||
|   exit 1 |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # set the ctest model to command line parameter |  | ||||||
| if [ "$1" == "alfa_ci" ]; then |  | ||||||
|   export ctest_model=Experimental |  | ||||||
| else |  | ||||||
|   export ctest_model=$1 |  | ||||||
| fi |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # test for architecture |  | ||||||
| arch=$(uname -s | tr '[A-Z]' '[a-z]') |  | ||||||
| chip=$(uname -m | tr '[A-Z]' '[a-z]') |  | ||||||
|  |  | ||||||
| # extract information about the system and the machine and set |  | ||||||
| # environment variables used by ctest |  | ||||||
| SYSTEM=$arch-$chip |  | ||||||
| if test -z $CXX ; then |  | ||||||
|   if [ "$arch" == "darwin" ]; then |  | ||||||
|     COMPILER=$(clang --version | head -n 1 | cut -d' ' -f1,2,4 | tr -d ' ') |  | ||||||
|   else |  | ||||||
|     COMPILER=gcc$(gcc -dumpversion) |  | ||||||
|   fi |  | ||||||
| else |  | ||||||
|   COMPILER=$CXX$($CXX -dumpversion) |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if [ "$1" == "alfa_ci" ]; then |  | ||||||
|   export LABEL1=alfa_ci-$COMPILER-FairLogger_$GIT_BRANCH |  | ||||||
|   export LABEL=$(echo $LABEL1 | sed -e 's#/#_#g') |  | ||||||
| else |  | ||||||
|   export LABEL1=${LINUX_FLAVOUR}-$chip-$COMPILER-FairLogger_$GIT_BRANCH |  | ||||||
|   export LABEL=$(echo $LABEL1 | sed -e 's#/#_#g') |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # get the number of processors |  | ||||||
| # and information about the host |  | ||||||
| if [ "$arch" = "linux" ]; |  | ||||||
| then |  | ||||||
|   if [ "$NCPU" != "" ]; |  | ||||||
|   then |  | ||||||
|     export number_of_processors=$NCPU |  | ||||||
|   else |  | ||||||
|     export number_of_processors=$(cat /proc/cpuinfo | grep processor | wc -l) |  | ||||||
|   fi |  | ||||||
|   if [ -z "$SITE" ]; then |  | ||||||
|     export SITE=$(hostname -f) |  | ||||||
|     if [ -z "$SITE" ]; then |  | ||||||
|       export SITE=$(uname -n) |  | ||||||
|     fi |  | ||||||
|   fi |  | ||||||
| elif [ "$arch" = "darwin" ]; |  | ||||||
| then |  | ||||||
|   if [ "$NCPU" != "" ]; |  | ||||||
|   then |  | ||||||
|     export number_of_processors=$NCPU |  | ||||||
|   else |  | ||||||
|     export number_of_processors=$(sysctl -n hw.ncpu) |  | ||||||
|   fi |  | ||||||
|   if [ -z "$SITE" ]; then |  | ||||||
|     export SITE=$(hostname -s) |  | ||||||
|   fi |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| echo "************************" |  | ||||||
| date |  | ||||||
| echo "LABEL: " $LABEL |  | ||||||
| echo "SITE: " $SITE |  | ||||||
| echo "Model: " ${ctest_model} |  | ||||||
| echo "Nr. of processes: " $number_of_processors |  | ||||||
| echo "************************" |  | ||||||
|  |  | ||||||
| cd $SOURCEDIR |  | ||||||
|  |  | ||||||
| ctest -S FairLoggerTest.cmake -V --VV |  | ||||||
| @@ -1,69 +1,62 @@ | |||||||
| ################################################################################ | ################################################################################ | ||||||
| #    Copyright (C) 2014 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH    # | #    Copyright (C) 2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH    # | ||||||
| #                                                                              # | #                                                                              # | ||||||
| #              This software is distributed under the terms of the             # | #              This software is distributed under the terms of the             # | ||||||
| #              GNU Lesser General Public Licence (LGPL) version 3,             # | #              GNU Lesser General Public Licence (LGPL) version 3,             # | ||||||
| #                  copied verbatim in the file "LICENSE"                       # | #                  copied verbatim in the file "LICENSE"                       # | ||||||
| ################################################################################ | ################################################################################ | ||||||
| Set(CTEST_SOURCE_DIRECTORY $ENV{SOURCEDIR}) |  | ||||||
| Set(CTEST_BINARY_DIRECTORY $ENV{BUILDDIR}) |  | ||||||
| Set(CTEST_SITE $ENV{SITE}) |  | ||||||
| Set(CTEST_BUILD_NAME $ENV{LABEL}) |  | ||||||
| Set(CTEST_CMAKE_GENERATOR "Unix Makefiles") |  | ||||||
| Set(CTEST_PROJECT_NAME "FairLogger") |  | ||||||
|  |  | ||||||
| Find_Program(CTEST_GIT_COMMAND NAMES git) | cmake_host_system_information(RESULT fqdn QUERY FQDN) | ||||||
| Set(CTEST_UPDATE_COMMAND "${CTEST_GIT_COMMAND}") |  | ||||||
|  |  | ||||||
| Set(BUILD_COMMAND "make") | set(CTEST_SOURCE_DIRECTORY .) | ||||||
| Set(CTEST_BUILD_COMMAND "${BUILD_COMMAND} -j$ENV{number_of_processors}") | set(CTEST_BINARY_DIRECTORY build) | ||||||
|  | set(CTEST_CMAKE_GENERATOR "Ninja") | ||||||
|  | set(CTEST_USE_LAUNCHERS ON) | ||||||
|  | set(CTEST_CONFIGURATION_TYPE "RelWithDebInfo") | ||||||
|  |  | ||||||
| String(TOUPPER $ENV{ctest_model} _Model) | if(NOT NCPUS) | ||||||
| Set(configure_options "-DCMAKE_BUILD_TYPE=${_Model}") |   if(ENV{SLURM_CPUS_PER_TASK}) | ||||||
|  |     set(NCPUS $ENV{SLURM_CPUS_PER_TASK}) | ||||||
|  |   else() | ||||||
|  |     include(ProcessorCount) | ||||||
|  |     ProcessorCount(NCPUS) | ||||||
|  |     if(NCPUS EQUAL 0) | ||||||
|  |       set(NCPUS 1) | ||||||
|  |     endif() | ||||||
|  |   endif() | ||||||
|  | endif() | ||||||
|  |  | ||||||
| Set(CTEST_USE_LAUNCHERS 1) | if ("$ENV{CTEST_SITE}" STREQUAL "") | ||||||
| Set(configure_options "${configure_options};-DCTEST_USE_LAUNCHERS=${CTEST_USE_LAUNCHERS}") |   set(CTEST_SITE "${fqdn}") | ||||||
|  | else() | ||||||
|  |   set(CTEST_SITE $ENV{CTEST_SITE}) | ||||||
|  | endif() | ||||||
|  |  | ||||||
| Set(configure_options "${configure_options};-DDISABLE_COLOR=ON") | if ("$ENV{LABEL}" STREQUAL "") | ||||||
|  |   set(CTEST_BUILD_NAME "build") | ||||||
|  | else() | ||||||
|  |   set(CTEST_BUILD_NAME $ENV{LABEL}) | ||||||
|  | endif() | ||||||
|  |  | ||||||
| Set(EXTRA_FLAGS $ENV{EXTRA_FLAGS}) | ctest_start(Continuous) | ||||||
| If(EXTRA_FLAGS) |  | ||||||
|   Set(configure_options "${configure_options};${EXTRA_FLAGS}")  |  | ||||||
| EndIf() |  | ||||||
|  |  | ||||||
| If($ENV{ctest_model} MATCHES Nightly OR $ENV{ctest_model} MATCHES Profile) | list(APPEND options | ||||||
|  |   "-DDISABLE_COLOR=ON" | ||||||
|  |   "-DUSE_EXTERNAL_FMT=ON" | ||||||
|  |   "-DUSE_BOOST_PRETTY_FUNCTION=ON" | ||||||
|  |   ) | ||||||
|  | list(JOIN options ";" optionsstr) | ||||||
|  | ctest_configure(OPTIONS "${optionsstr}") | ||||||
|  |  | ||||||
|   Find_Program(GCOV_COMMAND gcov) | ctest_build(FLAGS "-j${NCPUS}") | ||||||
|   If(GCOV_COMMAND) |  | ||||||
|     Message("Found GCOV: ${GCOV_COMMAND}") |  | ||||||
|     Set(CTEST_COVERAGE_COMMAND ${GCOV_COMMAND}) |  | ||||||
|   EndIf(GCOV_COMMAND) |  | ||||||
|  |  | ||||||
|   Set(ENV{ctest_model} Nightly) | ctest_test(BUILD "${CTEST_BINARY_DIRECTORY}" | ||||||
|  |            PARALLEL_LEVEL 1 | ||||||
|  |            SCHEDULE_RANDOM ON | ||||||
|  |            RETURN_VALUE _ctest_test_ret_val) | ||||||
|  |  | ||||||
|   CTEST_EMPTY_BINARY_DIRECTORY(${CTEST_BINARY_DIRECTORY}) | ctest_submit() | ||||||
|  |  | ||||||
| EndIf() | if(_ctest_test_ret_val) | ||||||
|  |  | ||||||
| Ctest_Start($ENV{ctest_model}) |  | ||||||
|  |  | ||||||
| Ctest_Configure(BUILD "${CTEST_BINARY_DIRECTORY}" |  | ||||||
|                 OPTIONS "${configure_options}" |  | ||||||
|                ) |  | ||||||
|  |  | ||||||
| Ctest_Build(BUILD "${CTEST_BINARY_DIRECTORY}") |  | ||||||
|  |  | ||||||
| Ctest_Test(BUILD "${CTEST_BINARY_DIRECTORY}"  |  | ||||||
|            PARALLEL_LEVEL $ENV{number_of_processors} |  | ||||||
|            RETURN_VALUE _ctest_test_ret_val |  | ||||||
|           ) |  | ||||||
|  |  | ||||||
| If(GCOV_COMMAND) |  | ||||||
|   Ctest_Coverage(BUILD "${CTEST_BINARY_DIRECTORY}") |  | ||||||
| EndIf() |  | ||||||
|  |  | ||||||
| Ctest_Submit() |  | ||||||
|   |  | ||||||
| if (_ctest_test_ret_val) |  | ||||||
|   Message(FATAL_ERROR "Some tests failed.") |   Message(FATAL_ERROR "Some tests failed.") | ||||||
| endif() | endif() | ||||||
|   | |||||||
							
								
								
									
										85
									
								
								Jenkinsfile
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										85
									
								
								Jenkinsfile
									
									
									
									
										vendored
									
									
								
							| @@ -1,51 +1,44 @@ | |||||||
| #!groovy | #!groovy | ||||||
|  |  | ||||||
| def specToLabel(Map spec) { | def jobMatrix(String prefix, String type, List specs) { | ||||||
|   return "${spec.os}-${spec.arch}-${spec.compiler}-FairSoft_${spec.fairsoft}" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def jobMatrix(String prefix, List specs, Closure callback) { |  | ||||||
|   def nodes = [:] |   def nodes = [:] | ||||||
|   for (spec in specs) { |   for (spec in specs) { | ||||||
|     def label = specToLabel(spec) |     job = "${spec.os}-${spec.ver}-${spec.arch}-${spec.compiler}" | ||||||
|     def fairsoft = spec.fairsoft |     def label = "${type}/${job}" | ||||||
|  |     def selector = "${spec.os}-${spec.ver}-${spec.arch}" | ||||||
|     def os = spec.os |     def os = spec.os | ||||||
|     def compiler = spec.compiler |     def ver = spec.ver | ||||||
|     nodes["${prefix}/${label}"] = { |     def check = spec.check | ||||||
|       node(label) { |  | ||||||
|  |     nodes[label] = { | ||||||
|  |       node(selector) { | ||||||
|         githubNotify(context: "${prefix}/${label}", description: 'Building ...', status: 'PENDING') |         githubNotify(context: "${prefix}/${label}", description: 'Building ...', status: 'PENDING') | ||||||
|         try { |         try { | ||||||
|           deleteDir() |           deleteDir() | ||||||
|           checkout scm |           checkout scm | ||||||
|  |  | ||||||
|           sh """\ |           def jobscript = 'job.sh' | ||||||
|             echo "export SIMPATH=\${SIMPATH_PREFIX}${fairsoft}" >> Dart.cfg |           def ctestcmd = "ctest -S FairLoggerTest.cmake -V --output-on-failure" | ||||||
|             echo "export FAIRSOFT_VERSION=${fairsoft}" >> Dart.cfg |           sh "echo \"set -e\" >> ${jobscript}" | ||||||
|           """ |           sh "echo \"export LABEL=\\\"\${JOB_BASE_NAME} ${label}\\\"\" >> ${jobscript}" | ||||||
|           if (os =~ /Debian/ && compiler =~ /gcc8/) { |           if (selector =~ /^macos/) { | ||||||
|             sh '''\ |             sh "echo \"${ctestcmd}\" >> ${jobscript}" | ||||||
|               echo "source /etc/profile.d/modules.sh" >> Dart.cfg |             sh "cat ${jobscript}" | ||||||
|               echo "module use /cvmfs/it.gsi.de/modulefiles" >> Dart.cfg |             sh "bash ${jobscript}" | ||||||
|               echo "module load compiler/gcc/8" >> Dart.cfg |  | ||||||
|             ''' |  | ||||||
|           } |  | ||||||
|           if (os =~ /MacOS/) { |  | ||||||
|             sh "echo \"export EXTRA_FLAGS='-DCMAKE_CXX_COMPILER=clang++'\" >> Dart.cfg" |  | ||||||
|           } else { |           } else { | ||||||
|             sh "echo \"export EXTRA_FLAGS='-DCMAKE_CXX_COMPILER=g++'\" >> Dart.cfg" |             def containercmd = "singularity exec -B/shared ${env.SINGULARITY_CONTAINER_ROOT}/fairlogger/${os}.${ver}.sif bash -l -c \\\"${ctestcmd}\\\"" | ||||||
|  |             sh """\ | ||||||
|  |               echo \"echo \\\"*** Job started at .......: \\\$(date -R)\\\"\" >> ${jobscript} | ||||||
|  |               echo \"echo \\\"*** Job ID ...............: \\\${SLURM_JOB_ID}\\\"\" >> ${jobscript} | ||||||
|  |               echo \"echo \\\"*** Compute node .........: \\\$(hostname -f)\\\"\" >> ${jobscript} | ||||||
|  |               echo \"unset http_proxy\" >> ${jobscript} | ||||||
|  |               echo \"unset HTTP_PROXY\" >> ${jobscript} | ||||||
|  |               echo \"${containercmd}\" >> ${jobscript} | ||||||
|  |             """ | ||||||
|  |             sh "cat ${jobscript}" | ||||||
|  |             sh "test/ci/slurm-submit.sh \"FairLogger \${JOB_BASE_NAME} ${label}\" ${jobscript}" | ||||||
|           } |           } | ||||||
|  |  | ||||||
|           sh '''\ |  | ||||||
|             echo "export BUILDDIR=$PWD/build" >> Dart.cfg |  | ||||||
|             echo "export SOURCEDIR=$PWD" >> Dart.cfg |  | ||||||
|             echo "export PATH=\\\$SIMPATH/bin:\\\$PATH" >> Dart.cfg |  | ||||||
|             echo "export GIT_BRANCH=$JOB_BASE_NAME" >> Dart.cfg |  | ||||||
|             echo "echo \\\$PATH" >> Dart.cfg |  | ||||||
|           ''' |  | ||||||
|           sh 'cat Dart.cfg' |  | ||||||
|  |  | ||||||
|           callback.call(spec, label) |  | ||||||
|  |  | ||||||
|           deleteDir() |           deleteDir() | ||||||
|           githubNotify(context: "${prefix}/${label}", description: 'Success', status: 'SUCCESS') |           githubNotify(context: "${prefix}/${label}", description: 'Success', status: 'SUCCESS') | ||||||
|         } catch (e) { |         } catch (e) { | ||||||
| @@ -65,13 +58,21 @@ pipeline{ | |||||||
|     stage("Run CI Matrix") { |     stage("Run CI Matrix") { | ||||||
|       steps{ |       steps{ | ||||||
|         script { |         script { | ||||||
|           parallel(jobMatrix('alfa-ci/build', [ |           def builds = jobMatrix('alfa-ci', 'build', [ | ||||||
|             [os: 'Debian8',    arch: 'x86_64', compiler: 'gcc8.1.0',        fairsoft: 'fairmq_dev'], |             [os: 'fedora', ver: '32',    arch: 'x86_64', compiler: 'gcc-10'], | ||||||
|             [os: 'MacOS10.13', arch: 'x86_64', compiler: 'AppleLLVM10.0.0', fairsoft: 'fairmq_dev'], |             [os: 'fedora', ver: '33',    arch: 'x86_64', compiler: 'gcc-10'], | ||||||
|             [os: 'MacOS10.14', arch: 'x86_64', compiler: 'AppleLLVM10.0.0', fairsoft: 'fairmq_dev'], |             [os: 'fedora', ver: '34',    arch: 'x86_64', compiler: 'gcc-11'], | ||||||
|           ]) { spec, label -> |             [os: 'fedora', ver: '35',    arch: 'x86_64', compiler: 'gcc-11'], | ||||||
|             sh './Dart.sh alfa_ci Dart.cfg' |             [os: 'fedora', ver: '36',    arch: 'x86_64', compiler: 'gcc-12'], | ||||||
|           }) |             [os: 'fedora', ver: '37',    arch: 'x86_64', compiler: 'gcc-12'], | ||||||
|  |             [os: 'fedora', ver: '38',    arch: 'x86_64', compiler: 'gcc-13'], | ||||||
|  |             [os: 'fedora', ver: '39',    arch: 'x86_64', compiler: 'gcc-13'], | ||||||
|  |             [os: 'macos',  ver: '13',    arch: 'x86_64', compiler: 'apple-clang-15'], | ||||||
|  |             [os: 'macos',  ver: '14',    arch: 'x86_64', compiler: 'apple-clang-15'], | ||||||
|  |             [os: 'macos',  ver: '14',    arch: 'arm64',  compiler: 'apple-clang-15'], | ||||||
|  |           ]) | ||||||
|  |  | ||||||
|  |           parallel(builds) | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,77 +0,0 @@ | |||||||
| #!groovy |  | ||||||
|  |  | ||||||
| def specToLabel(Map spec) { |  | ||||||
|   return "${spec.os}-${spec.arch}-${spec.compiler}-FairSoft_${spec.fairsoft}" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def buildMatrix(List specs, Closure callback) { |  | ||||||
|   def nodes = [:] |  | ||||||
|   for (spec in specs) { |  | ||||||
|     def label = specToLabel(spec) |  | ||||||
|     def fairsoft = spec.fairsoft |  | ||||||
|     def os = spec.os |  | ||||||
|     def compiler = spec.compiler |  | ||||||
|     nodes[label] = { |  | ||||||
|       node(label) { |  | ||||||
|         try { |  | ||||||
|           deleteDir() |  | ||||||
|           checkout scm |  | ||||||
|  |  | ||||||
|           sh """\ |  | ||||||
|             echo "export SIMPATH=\${SIMPATH_PREFIX}${fairsoft}" >> Dart.cfg |  | ||||||
|             echo "export FAIRSOFT_VERSION=${fairsoft}" >> Dart.cfg |  | ||||||
|           """ |  | ||||||
|           if (os =~ /Debian/ && compiler =~ /gcc8/) { |  | ||||||
|             sh '''\ |  | ||||||
|               echo "source /etc/profile.d/modules.sh" >> Dart.cfg |  | ||||||
|               echo "module use /cvmfs/it.gsi.de/modulefiles" >> Dart.cfg |  | ||||||
|               echo "module load compiler/gcc/8" >> Dart.cfg |  | ||||||
|             ''' |  | ||||||
|           } |  | ||||||
|           if (os =~ /MacOS/) { |  | ||||||
|             sh "echo \"export EXTRA_FLAGS='-DCMAKE_CXX_COMPILER=clang++'\" >> Dart.cfg" |  | ||||||
|           } else { |  | ||||||
|             sh "echo \"export EXTRA_FLAGS='-DCMAKE_CXX_COMPILER=g++'\" >> Dart.cfg" |  | ||||||
|           } |  | ||||||
|           sh '''\ |  | ||||||
|             echo "export BUILDDIR=$PWD/build" >> Dart.cfg |  | ||||||
|             echo "export SOURCEDIR=$PWD" >> Dart.cfg |  | ||||||
|             echo "export PATH=\\\$SIMPATH/bin:\\\$PATH" >> Dart.cfg |  | ||||||
|             echo "export GIT_BRANCH=dev" >> Dart.cfg |  | ||||||
|             echo "echo \\\$PATH" >> Dart.cfg |  | ||||||
|           ''' |  | ||||||
|           sh 'cat Dart.cfg' |  | ||||||
|  |  | ||||||
|           callback.call(spec, label) |  | ||||||
|  |  | ||||||
|           deleteDir() |  | ||||||
|         } catch (e) { |  | ||||||
|           deleteDir() |  | ||||||
|           throw e |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   return nodes |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pipeline{ |  | ||||||
|   agent none |  | ||||||
|   triggers { cron('H 2 * * *') } |  | ||||||
|   stages { |  | ||||||
|     stage("Run Nightly Build/Test Matrix") { |  | ||||||
|       steps{ |  | ||||||
|         script { |  | ||||||
|           parallel(buildMatrix([ |  | ||||||
|             [os: 'Debian8',    arch: 'x86_64', compiler: 'gcc8.1.0',        fairsoft: 'fairmq_dev'], |  | ||||||
|             [os: 'MacOS10.13', arch: 'x86_64', compiler: 'AppleLLVM10.0.0', fairsoft: 'fairmq_dev'], |  | ||||||
|             [os: 'MacOS10.14', arch: 'x86_64', compiler: 'AppleLLVM10.0.0', fairsoft: 'fairmq_dev'], |  | ||||||
|           ]) { spec, label -> |  | ||||||
|             sh './Dart.sh Nightly Dart.cfg' |  | ||||||
|             sh './Dart.sh Profile Dart.cfg' |  | ||||||
|           }) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
							
								
								
									
										98
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										98
									
								
								README.md
									
									
									
									
									
								
							| @@ -18,6 +18,8 @@ cmake -DCMAKE_INSTALL_PREFIX=./FairLogger_install ../FairLogger | |||||||
| cmake --build . --target install | cmake --build . --target install | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | FairLogger bundles a version of the [fmt](https://github.com/fmtlib/fmt) library. You can override this with your own fmt installation via the `-DUSE_EXTERNAL_FMT=ON` and `-DFMT_ROOT=/fmt/location` CMake switches. | ||||||
|  |  | ||||||
| ## Usage | ## Usage | ||||||
|  |  | ||||||
| In your `CMakeLists.txt`: | In your `CMakeLists.txt`: | ||||||
| @@ -35,6 +37,15 @@ find_package(FairLogger) | |||||||
|  |  | ||||||
| `find_package(FairLogger)` will define an imported target `FairLogger::FairLogger`. | `find_package(FairLogger)` will define an imported target `FairLogger::FairLogger`. | ||||||
|  |  | ||||||
|  | If FairLogger is built with `-DUSE_BOOST_PRETTY_FUNCTION=ON` and/or `-DUSE_EXTERNAL_FMT=ON`, your project needs to find the external dependencies, too, e.g. | ||||||
|  |  | ||||||
|  | ```cmake | ||||||
|  | find_package(FairLogger) | ||||||
|  | foreach(dep IN LISTS FairLogger_PACKAGE_DEPENDENCIES) | ||||||
|  |   find_package(${dep} ${FairLogger_${dep}_VERSION}) | ||||||
|  | endforeach() | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## CMake options | ## CMake options | ||||||
|  |  | ||||||
| On command line: | On command line: | ||||||
| @@ -42,6 +53,7 @@ On command line: | |||||||
|   * `-DDISABLE_COLOR=ON` disables coloured console output. |   * `-DDISABLE_COLOR=ON` disables coloured console output. | ||||||
|   * `-DBUILD_TESTING=OFF` disables building of unit tests. |   * `-DBUILD_TESTING=OFF` disables building of unit tests. | ||||||
|   * `-DUSE_BOOST_PRETTY_FUNCTION=ON` enables usage of `BOOST_PRETTY_FUNCTION` macro. |   * `-DUSE_BOOST_PRETTY_FUNCTION=ON` enables usage of `BOOST_PRETTY_FUNCTION` macro. | ||||||
|  |   * `-DUSE_EXTERNAL_FMT=ON` uses external fmt instead of the bundled one. | ||||||
|  |  | ||||||
| ## Documentation | ## Documentation | ||||||
|  |  | ||||||
| @@ -49,7 +61,18 @@ On command line: | |||||||
|  |  | ||||||
| All log calls go through the provided LOG(severity) macro. Output through this macro is thread-safe. Logging is done to cout, file output and/or custom sinks. | All log calls go through the provided LOG(severity) macro. Output through this macro is thread-safe. Logging is done to cout, file output and/or custom sinks. | ||||||
|  |  | ||||||
| ## 2. Severity | ## 2. Additional macros | ||||||
|  |  | ||||||
|  | A number of additional logging macros are provided: | ||||||
|  |  | ||||||
|  | - `LOGV(severity, verbosity)` Log the line with the provided verbosity, e.g. `LOG(info, veryhigh) << "abcd";` | ||||||
|  | - `LOGF(severity, ...)` The arguments are given to `fmt::printf`, which formats the string using a [printf syntax](https://fmt.dev/dev/api.html#printf-formatting) and the result is logged, e.g. `LOGF(info, "Hello %s!", "world");` | ||||||
|  | - `LOGP(severity, ...)` The arguments are given to `fmt::format`, which formats the string using a [Python-like syntax](https://fmt.dev/dev/syntax.html) and the result is logged, e.g. `LOGP(info, "Hello {}!", "world");` | ||||||
|  | - `LOGN(severity)` Logs an empty line, e.g. `LOGN(info);` | ||||||
|  | - `LOG_IF(severity, condition)` Logs the line if the provided condition if true | ||||||
|  | - `LOGD(severity, file, line, f)` Logs the line with the provided file, line and function parameters (only if the active verbosity allows it). | ||||||
|  |  | ||||||
|  | ## 3. Severity | ||||||
|  |  | ||||||
| The log severity is controlled via: | The log severity is controlled via: | ||||||
| ```C++ | ```C++ | ||||||
| @@ -64,36 +87,55 @@ where severity level is one of the following: | |||||||
|  |  | ||||||
| ```C++ | ```C++ | ||||||
| "nolog", | "nolog", | ||||||
| "fatal", |  | ||||||
| "error", |  | ||||||
| "warn", |  | ||||||
| "state", |  | ||||||
| "info", |  | ||||||
| "debug", |  | ||||||
| "debug1", |  | ||||||
| "debug2", |  | ||||||
| "debug3", |  | ||||||
| "debug4", |  | ||||||
| "trace", | "trace", | ||||||
|  | "debug4", | ||||||
|  | "debug3", | ||||||
|  | "debug2", | ||||||
|  | "debug1", | ||||||
|  | "debug", | ||||||
|  | "detail", | ||||||
|  | "info", | ||||||
|  | "state", | ||||||
|  | "warn", | ||||||
|  | "important", | ||||||
|  | "alarm", | ||||||
|  | "error", | ||||||
|  | "critical", | ||||||
|  | "fatal", | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Logger will log the chosen severity and all above it (except "nolog", which deactivates logging for that sink completely). Fatal severity is always logged. | Logger will log the chosen severity and all above it (except "nolog", which deactivates logging for that sink completely). Fatal severity is always logged. | ||||||
|  |  | ||||||
| ## 3. Verbosity | ## 3.1 Compile-time severity switch | ||||||
|  |  | ||||||
|  | The minimum severity level can be configured at compile time via definition of `FAIR_MIN_SEVERITY`: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | cmake -DFAIR_MIN_SEVERITY=warn .. | ||||||
|  | ``` | ||||||
|  | The above would only log severities equal to or above `warn`. | ||||||
|  |  | ||||||
|  | When `FAIR_MIN_SEVERITY` is not provided all severities are enabled. | ||||||
|  |  | ||||||
|  | ## 4. Verbosity | ||||||
|  |  | ||||||
| The log verbosity is controlled via: | The log verbosity is controlled via: | ||||||
| ```C++ | ```C++ | ||||||
| fair::Logger::SetVerbosity("<verbosity level>"); | fair::Logger::SetVerbosity("<verbosity level>"); | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| it is same for all sinks, and is one of the following values: `verylow`, `low`, `medium`, `high`, `veryhigh`, which translates to following output: | it is same for all sinks, and is one of the following values: `verylow`, `low`, `medium`, `high`, `veryhigh`, `user1`, `user2`, `user3`, `user4`, which translates to following output: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| verylow:  message | verylow:  message | ||||||
| low:      [severity] message | low:      [severity] message | ||||||
| medium:   [HH:MM:SS][severity] message | medium:   [HH:MM:SS][severity] message | ||||||
| high:     [process name][HH:MM:SS:µS][severity] message | high:     [process_name][HH:MM:SS][severity] message | ||||||
| veryhigh: [process name][HH:MM:SS:µS][severity][file:line:function] message | veryhigh: [process_name][HH:MM:SS:µS][severity][file:line:function] message | ||||||
|  | user1:    [severity] message | ||||||
|  | user2:    [severity] message | ||||||
|  | user3:    [severity] message | ||||||
|  | user4:    [severity] message | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| When running a FairMQ device, the log severity can be simply provided via `--verbosity <level>` cmd option. | When running a FairMQ device, the log severity can be simply provided via `--verbosity <level>` cmd option. | ||||||
| @@ -108,12 +150,12 @@ The `fair::Logger::VerbositySpec` object can e.g. be created like this: | |||||||
| ```C++ | ```C++ | ||||||
| auto spec = fair::VerbositySpec::Make(VerbositySpec::Info::timestamp_s, | auto spec = fair::VerbositySpec::Make(VerbositySpec::Info::timestamp_s, | ||||||
|                                       VerbositySpec::Info::process_name); |                                       VerbositySpec::Info::process_name); | ||||||
| // results in [HH:MM:SS][process name] message | // results in [HH:MM:SS][process_name] message | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| | **Argument** | **Result** | | | **Argument** | **Result** | | ||||||
| | --- | --- | | | --- | --- | | ||||||
| | `fair::VerbositySpec::Info::process_name`       | `[process name]`       | | | `fair::VerbositySpec::Info::process_name`       | `[process_name]`       | | ||||||
| | `fair::VerbositySpec::Info::timestamp_s`        | `[HH:MM:SS]`           | | | `fair::VerbositySpec::Info::timestamp_s`        | `[HH:MM:SS]`           | | ||||||
| | `fair::VerbositySpec::Info::timestamp_us`       | `[HH:MM:SS:µS]`        | | | `fair::VerbositySpec::Info::timestamp_us`       | `[HH:MM:SS:µS]`        | | ||||||
| | `fair::VerbositySpec::Info::severity`           | `[severity]`           | | | `fair::VerbositySpec::Info::severity`           | `[severity]`           | | ||||||
| @@ -122,7 +164,7 @@ auto spec = fair::VerbositySpec::Make(VerbositySpec::Info::timestamp_s, | |||||||
| | `fair::VerbositySpec::Info::file_line_function` | `[file:line:function]` | | | `fair::VerbositySpec::Info::file_line_function` | `[file:line:function]` | | ||||||
|  |  | ||||||
|  |  | ||||||
| ### 3.1 `BOOST_PRETTY_FUNCTION` support | ### 4.1 `BOOST_PRETTY_FUNCTION` support | ||||||
|  |  | ||||||
| By default, the `veryhigh` verbosity prints the function name from which the `LOG` macro was invoked. If you desire a more verbose function signature including the full namespace, return value and function arguments, you can enable support for `BOOST_PRETTY_FUNCTION` | By default, the `veryhigh` verbosity prints the function name from which the `LOG` macro was invoked. If you desire a more verbose function signature including the full namespace, return value and function arguments, you can enable support for `BOOST_PRETTY_FUNCTION` | ||||||
|  |  | ||||||
| @@ -136,7 +178,7 @@ By default, the `veryhigh` verbosity prints the function name from which the `LO | |||||||
|  |  | ||||||
| In the latter case, the user needs to take care of adding the boost include path to the compiler search path manually (e.g. `-I/path/to/boost/include`). | In the latter case, the user needs to take care of adding the boost include path to the compiler search path manually (e.g. `-I/path/to/boost/include`). | ||||||
|  |  | ||||||
| ## 4. Color | ## 5. Color | ||||||
|  |  | ||||||
| Colored output on console can be activated with: | Colored output on console can be activated with: | ||||||
| ```C++ | ```C++ | ||||||
| @@ -145,7 +187,7 @@ Logger::SetConsoleColor(true); | |||||||
|  |  | ||||||
| When running a FairMQ device, the log color (console) can be simply provided via `--color <true/false>` cmd option (default is true). | When running a FairMQ device, the log color (console) can be simply provided via `--color <true/false>` cmd option (default is true). | ||||||
|  |  | ||||||
| ## 5. File output | ## 6. File output | ||||||
|  |  | ||||||
| Output to file can be enabled via: | Output to file can be enabled via: | ||||||
| ```C++ | ```C++ | ||||||
| @@ -155,7 +197,7 @@ which will add output to "test_log" filename (if third parameter is `true` it wi | |||||||
|  |  | ||||||
| When running a FairMQ device, the log file can be simply provided via `--log-to-file <filename_prefix>` cmd option (this will also turn off console output). | When running a FairMQ device, the log file can be simply provided via `--log-to-file <filename_prefix>` cmd option (this will also turn off console output). | ||||||
|  |  | ||||||
| ## 5.5 Custom sinks | ## 7. Custom sinks | ||||||
|  |  | ||||||
| Custom sinks can be added via `Logger::AddCustomSink("sink name", "<severity>", callback)` method. | Custom sinks can be added via `Logger::AddCustomSink("sink name", "<severity>", callback)` method. | ||||||
|  |  | ||||||
| @@ -180,8 +222,20 @@ Here is an example adding a custom sink for all severities ("trace" and above). | |||||||
|  |  | ||||||
| If only output from custom sinks is desirable, console/file sinks must be deactivated by setting their severity to `"nolog"`. | If only output from custom sinks is desirable, console/file sinks must be deactivated by setting their severity to `"nolog"`. | ||||||
|  |  | ||||||
|  | ## Naming conflicts? | ||||||
|  |  | ||||||
|  | By default, `<fairlogger/Logger.h>` defines unprefixed macros: `LOG`, `LOGV`, `LOGF`, `LOGP`, `LOGN`, `LOGD`, `LOG_IF`. | ||||||
|  |  | ||||||
|  | Define an option `FAIR_NO_LOG*` to prevent the above unprefixed macros to be defined, e.g. | ||||||
|  |  | ||||||
|  | ```C++ | ||||||
|  | #define FAIR_NO_LOG | ||||||
|  | #define FAIR_NO_LOGF | ||||||
|  | #include <fairlogger/Logger.h> | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
| GNU Lesser General Public Licence (LGPL) version 3, see [LICENSE](LICENSE). | GNU Lesser General Public Licence (LGPL) version 3, see [LICENSE](LICENSE). | ||||||
|  |  | ||||||
| Copyright (C) 2017-2018 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH | Copyright (C) 2017-2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH | ||||||
|   | |||||||
							
								
								
									
										102
									
								
								cmake/FairFindPackage2.cmake
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								cmake/FairFindPackage2.cmake
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | ################################################################################ | ||||||
|  | # Copyright (C) 2018-2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | ||||||
|  | #                                                                              # | ||||||
|  | #              This software is distributed under the terms of the             # | ||||||
|  | #              GNU Lesser General Public Licence (LGPL) version 3,             # | ||||||
|  | #                  copied verbatim in the file "LICENSE"                       # | ||||||
|  | ################################################################################ | ||||||
|  |  | ||||||
|  | # | ||||||
|  | # find_package2(PRIVATE|PUBLIC|INTERFACE <pkgname> | ||||||
|  | #               [VERSION <version>] | ||||||
|  | #               [COMPONENTS <list of components>] | ||||||
|  | #               [ADD_REQUIREMENTS_OF <list of dep_pgkname>] | ||||||
|  | #               [any other option the native find_package supports]...) | ||||||
|  | # | ||||||
|  | # Wrapper around CMake's native find_package command to add some features and bookkeeping. | ||||||
|  | # | ||||||
|  | # The qualifier (PRIVATE|PUBLIC|INTERFACE) to the package to populate | ||||||
|  | # the variables PROJECT_[INTERFACE]_<pkgname>_([VERSION]|[COMPONENTS]|PACKAGE_DEPENDENCIES) | ||||||
|  | # accordingly. This bookkeeping information is used to print our dependency found summary | ||||||
|  | # table and to generate a part of our CMake package. | ||||||
|  | # | ||||||
|  | # When a dependending package is listed with ADD_REQUIREMENTS_OF the variables | ||||||
|  | # <dep_pkgname>_<pkgname>_VERSION|COMPONENTS are looked up to and added to the native | ||||||
|  | # VERSION (selected highest version) and COMPONENTS (deduplicated) args. | ||||||
|  | # | ||||||
|  | # COMPONENTS and VERSION args are then just passed to the native find_package. | ||||||
|  | # | ||||||
|  | macro(find_package2 qualifier pkgname) | ||||||
|  |   cmake_parse_arguments(ARGS "" "VERSION" "COMPONENTS;ADD_REQUIREMENTS_OF" ${ARGN}) | ||||||
|  |  | ||||||
|  |   string(TOUPPER ${pkgname} pkgname_upper) | ||||||
|  |   set(__old_cpp__ ${CMAKE_PREFIX_PATH}) | ||||||
|  |   set(CMAKE_PREFIX_PATH ${${pkgname_upper}_ROOT} $ENV{${pkgname_upper}_ROOT} ${CMAKE_PREFIX_PATH}) | ||||||
|  |  | ||||||
|  |   # build lists of required versions and components | ||||||
|  |   unset(__required_versions__) | ||||||
|  |   unset(__components__) | ||||||
|  |   if(ARGS_VERSION) | ||||||
|  |     list(APPEND __required_versions__ ${ARGS_VERSION}) | ||||||
|  |   endif() | ||||||
|  |   if(ARGS_COMPONENTS) | ||||||
|  |     list(APPEND __components__ ${ARGS_COMPONENTS}) | ||||||
|  |   endif() | ||||||
|  |   if(ARGS_ADD_REQUIREMENTS_OF) | ||||||
|  |     foreach(dep_pkgname IN LISTS ARGS_ADD_REQUIREMENTS_OF) | ||||||
|  |       if(${dep_pkgname}_${pkgname}_VERSION) | ||||||
|  |         list(APPEND __required_versions__ ${${dep_pkgname}_${pkgname}_VERSION}) | ||||||
|  |       endif() | ||||||
|  |       if(${dep_pkgname}_${pkgname}_COMPONENTS) | ||||||
|  |         list(APPEND __components__ ${${dep_pkgname}_${pkgname}_COMPONENTS}) | ||||||
|  |       endif() | ||||||
|  |     endforeach() | ||||||
|  |   endif() | ||||||
|  |  | ||||||
|  |   # select highest required version | ||||||
|  |   unset(__version__) | ||||||
|  |   if(__required_versions__) | ||||||
|  |     list(GET __required_versions__ 0 __version__) | ||||||
|  |     foreach(v IN LISTS __required_versions__) | ||||||
|  |       if(${v} VERSION_GREATER ${__version__}) | ||||||
|  |         set(__version__ ${v}) | ||||||
|  |       endif() | ||||||
|  |     endforeach() | ||||||
|  |   endif() | ||||||
|  |   # deduplicate required component list | ||||||
|  |   if(__components__) | ||||||
|  |     list(REMOVE_DUPLICATES ARGS_COMPONENTS) | ||||||
|  |   endif() | ||||||
|  |  | ||||||
|  |   # call native find_package | ||||||
|  |   if(__components__) | ||||||
|  |     find_package(${pkgname} ${__version__} QUIET COMPONENTS ${__components__} ${ARGS_UNPARSED_ARGUMENTS}) | ||||||
|  |   else() | ||||||
|  |     find_package(${pkgname} ${__version__} QUIET ${ARGS_UNPARSED_ARGUMENTS}) | ||||||
|  |   endif() | ||||||
|  |  | ||||||
|  |   if(${pkgname}_FOUND) | ||||||
|  |     if(${qualifier} STREQUAL PRIVATE) | ||||||
|  |       set(PROJECT_${pkgname}_VERSION ${__version__}) | ||||||
|  |       set(PROJECT_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) | ||||||
|  |       set(PROJECT_PACKAGE_DEPENDENCIES ${PROJECT_PACKAGE_DEPENDENCIES} ${pkgname}) | ||||||
|  |     elseif(${qualifier} STREQUAL PUBLIC) | ||||||
|  |       set(PROJECT_${pkgname}_VERSION ${__version__}) | ||||||
|  |       set(PROJECT_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) | ||||||
|  |       set(PROJECT_PACKAGE_DEPENDENCIES ${PROJECT_PACKAGE_DEPENDENCIES} ${pkgname}) | ||||||
|  |       set(PROJECT_INTERFACE_${pkgname}_VERSION ${__version__}) | ||||||
|  |       set(PROJECT_INTERFACE_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) | ||||||
|  |       set(PROJECT_INTERFACE_PACKAGE_DEPENDENCIES ${PROJECT_INTERFACE_PACKAGE_DEPENDENCIES} ${pkgname}) | ||||||
|  |     elseif(${qualifier} STREQUAL INTERFACE) | ||||||
|  |       set(PROJECT_INTERFACE_${pkgname}_VERSION ${__version__}) | ||||||
|  |       set(PROJECT_INTERFACE_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) | ||||||
|  |       set(PROJECT_INTERFACE_PACKAGE_DEPENDENCIES ${PROJECT_INTERFACE_PACKAGE_DEPENDENCIES} ${pkgname}) | ||||||
|  |     endif() | ||||||
|  |   endif() | ||||||
|  |  | ||||||
|  |   unset(__version__) | ||||||
|  |   unset(__components__) | ||||||
|  |   unset(__required_versions__) | ||||||
|  |   set(CMAKE_PREFIX_PATH ${__old_cpp__}) | ||||||
|  |   unset(__old_cpp__) | ||||||
|  | endmacro() | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| ################################################################################ | ################################################################################ | ||||||
| # Copyright (C) 2018-2019 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | # Copyright (C) 2018-2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | ||||||
| #                                                                              # | #                                                                              # | ||||||
| #              This software is distributed under the terms of the             # | #              This software is distributed under the terms of the             # | ||||||
| #              GNU Lesser General Public Licence (LGPL) version 3,             # | #              GNU Lesser General Public Licence (LGPL) version 3,             # | ||||||
| @@ -15,7 +15,8 @@ set(@PROJECT_NAME@_GIT_DATE @PROJECT_GIT_DATE@) | |||||||
|  |  | ||||||
| set_and_check(@PROJECT_NAME@_PREFIX @PACKAGE_CMAKE_INSTALL_PREFIX@) | set_and_check(@PROJECT_NAME@_PREFIX @PACKAGE_CMAKE_INSTALL_PREFIX@) | ||||||
| set(@PROJECT_NAME@_BINDIR @PACKAGE_CMAKE_INSTALL_PREFIX@/@PROJECT_INSTALL_BINDIR@) | set(@PROJECT_NAME@_BINDIR @PACKAGE_CMAKE_INSTALL_PREFIX@/@PROJECT_INSTALL_BINDIR@) | ||||||
| set_and_check(@PROJECT_NAME@_INCDIR @PACKAGE_CMAKE_INSTALL_PREFIX@/@CMAKE_INSTALL_INCLUDEDIR@) | set(@PROJECT_NAME@_INCDIR @PACKAGE_INSTALL_INCDIRS@) | ||||||
|  | set(@PROJECT_NAME@_INCDIRS @PACKAGE_INSTALL_INCDIRS@) | ||||||
| set_and_check(@PROJECT_NAME@_LIBDIR @PACKAGE_CMAKE_INSTALL_PREFIX@/@PROJECT_INSTALL_LIBDIR@) | set_and_check(@PROJECT_NAME@_LIBDIR @PACKAGE_CMAKE_INSTALL_PREFIX@/@PROJECT_INSTALL_LIBDIR@) | ||||||
|  |  | ||||||
| set(@PROJECT_NAME@_CXX_STANDARD_REQUIRED @CMAKE_CXX_STANDARD_REQUIRED@) | set(@PROJECT_NAME@_CXX_STANDARD_REQUIRED @CMAKE_CXX_STANDARD_REQUIRED@) | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| ################################################################################ | ################################################################################ | ||||||
| # Copyright (C) 2018-2019 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | # Copyright (C) 2018-2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  # | ||||||
| #                                                                              # | #                                                                              # | ||||||
| #              This software is distributed under the terms of the             # | #              This software is distributed under the terms of the             # | ||||||
| #              GNU Lesser General Public Licence (LGPL) version 3,             # | #              GNU Lesser General Public Licence (LGPL) version 3,             # | ||||||
| @@ -94,24 +94,22 @@ macro(set_fairlogger_defaults) | |||||||
|   endif() |   endif() | ||||||
|  |  | ||||||
|   # Handle C++ standard level |   # Handle C++ standard level | ||||||
|   set(PROJECT_MIN_CXX_STANDARD 11) |   set(PROJECT_MIN_CXX_STANDARD 17) | ||||||
|   set(CMAKE_CXX_STANDARD_REQUIRED ON) |   if(CMAKE_CXX_STANDARD LESS PROJECT_MIN_CXX_STANDARD) | ||||||
|   if(NOT CMAKE_CXX_STANDARD) |  | ||||||
|     set(CMAKE_CXX_STANDARD ${PROJECT_MIN_CXX_STANDARD}) |  | ||||||
|   elseif(${CMAKE_CXX_STANDARD} LESS ${PROJECT_MIN_CXX_STANDARD}) |  | ||||||
|     message(FATAL_ERROR "A minimum CMAKE_CXX_STANDARD of ${PROJECT_MIN_CXX_STANDARD} is required.") |     message(FATAL_ERROR "A minimum CMAKE_CXX_STANDARD of ${PROJECT_MIN_CXX_STANDARD} is required.") | ||||||
|   endif() |   endif() | ||||||
|   set(CMAKE_CXX_EXTENSIONS OFF) |  | ||||||
|  |  | ||||||
|   # Generate compile_commands.json file (https://clang.llvm.org/docs/JSONCompilationDatabase.html) |   # Generate compile_commands.json file (https://clang.llvm.org/docs/JSONCompilationDatabase.html) | ||||||
|   set(CMAKE_EXPORT_COMPILE_COMMANDS ON) |   if(NOT DEFINED CMAKE_EXPORT_COMPILE_COMMANDS) | ||||||
|  |     set(CMAKE_EXPORT_COMPILE_COMMANDS ON) | ||||||
|  |   endif() | ||||||
|  |  | ||||||
|   if(NOT BUILD_SHARED_LIBS) |   if(NOT DEFINED BUILD_SHARED_LIBS) | ||||||
|     set(BUILD_SHARED_LIBS ON CACHE BOOL "Whether to build shared libraries or static archives") |     set(BUILD_SHARED_LIBS ON CACHE BOOL "Whether to build shared libraries or static archives") | ||||||
|   endif() |   endif() | ||||||
|  |  | ||||||
|   # Set -fPIC as default for all library types |   # Set -fPIC as default for all library types | ||||||
|   if(NOT CMAKE_POSITION_INDEPENDENT_CODE) |   if(NOT DEFINED CMAKE_POSITION_INDEPENDENT_CODE) | ||||||
|     set(CMAKE_POSITION_INDEPENDENT_CODE ON) |     set(CMAKE_POSITION_INDEPENDENT_CODE ON) | ||||||
|   endif() |   endif() | ||||||
|  |  | ||||||
| @@ -122,6 +120,7 @@ macro(set_fairlogger_defaults) | |||||||
|   set(PROJECT_INSTALL_BINDIR ${CMAKE_INSTALL_BINDIR}) |   set(PROJECT_INSTALL_BINDIR ${CMAKE_INSTALL_BINDIR}) | ||||||
|   set(PROJECT_INSTALL_LIBDIR ${CMAKE_INSTALL_LIBDIR}) |   set(PROJECT_INSTALL_LIBDIR ${CMAKE_INSTALL_LIBDIR}) | ||||||
|   set(PROJECT_INSTALL_INCDIR ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME_LOWER}) |   set(PROJECT_INSTALL_INCDIR ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME_LOWER}) | ||||||
|  |   set(PROJECT_INSTALL_BUNDLEDINCDIR ${PROJECT_INSTALL_INCDIR}/bundled) | ||||||
|   set(PROJECT_INSTALL_DATADIR ${CMAKE_INSTALL_DATADIR}/${PROJECT_NAME_LOWER}) |   set(PROJECT_INSTALL_DATADIR ${CMAKE_INSTALL_DATADIR}/${PROJECT_NAME_LOWER}) | ||||||
|  |  | ||||||
|   # https://cmake.org/Wiki/CMake_RPATH_handling |   # https://cmake.org/Wiki/CMake_RPATH_handling | ||||||
| @@ -129,13 +128,13 @@ macro(set_fairlogger_defaults) | |||||||
|   list(FIND CMAKE_PLATFORM_IMPLICIT_LINK_DIRECTORIES "${CMAKE_INSTALL_PREFIX}/${PROJECT_INSTALL_LIBDIR}" isSystemDir) |   list(FIND CMAKE_PLATFORM_IMPLICIT_LINK_DIRECTORIES "${CMAKE_INSTALL_PREFIX}/${PROJECT_INSTALL_LIBDIR}" isSystemDir) | ||||||
|   if("${isSystemDir}" STREQUAL "-1") |   if("${isSystemDir}" STREQUAL "-1") | ||||||
|     if(CMAKE_SYSTEM_NAME STREQUAL "Linux") |     if(CMAKE_SYSTEM_NAME STREQUAL "Linux") | ||||||
|       set(CMAKE_EXE_LINKER_FLAGS ${CMAKE_EXE_LINKER_FLAGS} "-Wl,--enable-new-dtags") |       string(APPEND CMAKE_EXE_LINKER_FLAGS " -Wl,--enable-new-dtags") | ||||||
|       set(CMAKE_SHARED_LINKER_FLAGS ${CMAKE_SHARED_LINKER_FLAGS} "-Wl,--enable-new-dtags") |       string(APPEND CMAKE_SHARED_LINKER_FLAGS " -Wl,--enable-new-dtags") | ||||||
|       set(CMAKE_INSTALL_RPATH "$ORIGIN/../${PROJECT_INSTALL_LIBDIR}") |       list(APPEND CMAKE_INSTALL_RPATH "$ORIGIN/../${PROJECT_INSTALL_LIBDIR}") | ||||||
|     elseif(CMAKE_SYSTEM_NAME STREQUAL "Darwin") |     elseif(CMAKE_SYSTEM_NAME STREQUAL "Darwin") | ||||||
|       set(CMAKE_INSTALL_RPATH "@loader_path/../${PROJECT_INSTALL_LIBDIR}") |       list(APPEND CMAKE_INSTALL_RPATH "@loader_path/../${PROJECT_INSTALL_LIBDIR}") | ||||||
|     else() |     else() | ||||||
|       set(CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/${PROJECT_INSTALL_LIBDIR}") |       list(APPEND CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/${PROJECT_INSTALL_LIBDIR}") | ||||||
|     endif() |     endif() | ||||||
|   endif() |   endif() | ||||||
|  |  | ||||||
| @@ -227,6 +226,13 @@ macro(install_cmake_package) | |||||||
|     VERSION ${PROJECT_VERSION} |     VERSION ${PROJECT_VERSION} | ||||||
|     COMPATIBILITY AnyNewerVersion |     COMPATIBILITY AnyNewerVersion | ||||||
|   ) |   ) | ||||||
|  |   unset(PACKAGE_INSTALL_INCDIRS) | ||||||
|  |   list(APPEND PACKAGE_INSTALL_INCDIRS | ||||||
|  |     \$\{PACKAGE_PREFIX_DIR\}/${CMAKE_INSTALL_INCLUDEDIR}) | ||||||
|  |   if(NOT USE_EXTERNAL_FMT) | ||||||
|  |     list(APPEND PACKAGE_INSTALL_INCDIRS | ||||||
|  |       \$\{PACKAGE_PREFIX_DIR\}/${PROJECT_INSTALL_BUNDLEDINCDIR}) | ||||||
|  |   endif() | ||||||
|   generate_package_dependencies() # fills ${PACKAGE_DEPENDENCIES} |   generate_package_dependencies() # fills ${PACKAGE_DEPENDENCIES} | ||||||
|   string(TOUPPER ${CMAKE_BUILD_TYPE} PROJECT_BUILD_TYPE_UPPER) |   string(TOUPPER ${CMAKE_BUILD_TYPE} PROJECT_BUILD_TYPE_UPPER) | ||||||
|   set(PROJECT_CXX_FLAGS ${CMAKE_CXX_FLAGS} ${CMAKE_CXX_FLAGS_${PROJECT_BUILD_TYPE_UPPER}}) |   set(PROJECT_CXX_FLAGS ${CMAKE_CXX_FLAGS} ${CMAKE_CXX_FLAGS_${PROJECT_BUILD_TYPE_UPPER}}) | ||||||
| @@ -242,98 +248,3 @@ macro(install_cmake_package) | |||||||
|     DESTINATION ${PACKAGE_INSTALL_DESTINATION} |     DESTINATION ${PACKAGE_INSTALL_DESTINATION} | ||||||
|   ) |   ) | ||||||
| endmacro() | endmacro() | ||||||
|  |  | ||||||
| # |  | ||||||
| # find_package2(PRIVATE|PUBLIC|INTERFACE <pkgname> |  | ||||||
| #               [VERSION <version>] |  | ||||||
| #               [COMPONENTS <list of components>] |  | ||||||
| #               [ADD_REQUIREMENTS_OF <list of dep_pgkname>] |  | ||||||
| #               [any other option the native find_package supports]...) |  | ||||||
| # |  | ||||||
| # Wrapper around CMake's native find_package command to add some features and bookkeeping. |  | ||||||
| # |  | ||||||
| # The qualifier (PRIVATE|PUBLIC|INTERFACE) to the package to populate |  | ||||||
| # the variables PROJECT_[INTERFACE]_<pkgname>_([VERSION]|[COMPONENTS]|PACKAGE_DEPENDENCIES) |  | ||||||
| # accordingly. This bookkeeping information is used to print our dependency found summary |  | ||||||
| # table and to generate a part of our CMake package. |  | ||||||
| # |  | ||||||
| # When a dependending package is listed with ADD_REQUIREMENTS_OF the variables |  | ||||||
| # <dep_pkgname>_<pkgname>_VERSION|COMPONENTS are looked up to and added to the native |  | ||||||
| # VERSION (selected highest version) and COMPONENTS (deduplicated) args. |  | ||||||
| # |  | ||||||
| # COMPONENTS and VERSION args are then just passed to the native find_package. |  | ||||||
| # |  | ||||||
| macro(find_package2 qualifier pkgname) |  | ||||||
|   cmake_parse_arguments(ARGS "" "VERSION" "COMPONENTS;ADD_REQUIREMENTS_OF" ${ARGN}) |  | ||||||
|  |  | ||||||
|   string(TOUPPER ${pkgname} pkgname_upper) |  | ||||||
|   set(__old_cpp__ ${CMAKE_PREFIX_PATH}) |  | ||||||
|   set(CMAKE_PREFIX_PATH ${${pkgname_upper}_ROOT} $ENV{${pkgname_upper}_ROOT} ${CMAKE_PREFIX_PATH}) |  | ||||||
|  |  | ||||||
|   # build lists of required versions and components |  | ||||||
|   unset(__required_versions__) |  | ||||||
|   unset(__components__) |  | ||||||
|   if(ARGS_VERSION) |  | ||||||
|     list(APPEND __required_versions__ ${ARGS_VERSION}) |  | ||||||
|   endif() |  | ||||||
|   if(ARGS_COMPONENTS) |  | ||||||
|     list(APPEND __components__ ${ARGS_COMPONENTS}) |  | ||||||
|   endif() |  | ||||||
|   if(ARGS_ADD_REQUIREMENTS_OF) |  | ||||||
|     foreach(dep_pkgname IN LISTS ARGS_ADD_REQUIREMENTS_OF) |  | ||||||
|       if(${dep_pkgname}_${pkgname}_VERSION) |  | ||||||
|         list(APPEND __required_versions__ ${${dep_pkgname}_${pkgname}_VERSION}) |  | ||||||
|       endif() |  | ||||||
|       if(${dep_pkgname}_${pkgname}_COMPONENTS) |  | ||||||
|         list(APPEND __components__ ${${dep_pkgname}_${pkgname}_COMPONENTS}) |  | ||||||
|       endif() |  | ||||||
|     endforeach() |  | ||||||
|   endif() |  | ||||||
|  |  | ||||||
|   # select highest required version |  | ||||||
|   unset(__version__) |  | ||||||
|   if(__required_versions__) |  | ||||||
|     list(GET __required_versions__ 0 __version__) |  | ||||||
|     foreach(v IN LISTS __required_versions__) |  | ||||||
|       if(${v} VERSION_GREATER ${__version__}) |  | ||||||
|         set(__version__ ${v}) |  | ||||||
|       endif() |  | ||||||
|     endforeach() |  | ||||||
|   endif() |  | ||||||
|   # deduplicate required component list |  | ||||||
|   if(__components__) |  | ||||||
|     list(REMOVE_DUPLICATES ARGS_COMPONENTS) |  | ||||||
|   endif() |  | ||||||
|  |  | ||||||
|   # call native find_package |  | ||||||
|   if(__components__) |  | ||||||
|     find_package(${pkgname} ${__version__} QUIET COMPONENTS ${__components__} ${ARGS_UNPARSED_ARGUMENTS}) |  | ||||||
|   else() |  | ||||||
|     find_package(${pkgname} ${__version__} QUIET ${ARGS_UNPARSED_ARGUMENTS}) |  | ||||||
|   endif() |  | ||||||
|  |  | ||||||
|   if(${pkgname}_FOUND) |  | ||||||
|     if(${qualifier} STREQUAL PRIVATE) |  | ||||||
|       set(PROJECT_${pkgname}_VERSION ${__version__}) |  | ||||||
|       set(PROJECT_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) |  | ||||||
|       set(PROJECT_PACKAGE_DEPENDENCIES ${PROJECT_PACKAGE_DEPENDENCIES} ${pkgname}) |  | ||||||
|     elseif(${qualifier} STREQUAL PUBLIC) |  | ||||||
|       set(PROJECT_${pkgname}_VERSION ${__version__}) |  | ||||||
|       set(PROJECT_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) |  | ||||||
|       set(PROJECT_PACKAGE_DEPENDENCIES ${PROJECT_PACKAGE_DEPENDENCIES} ${pkgname}) |  | ||||||
|       set(PROJECT_INTERFACE_${pkgname}_VERSION ${__version__}) |  | ||||||
|       set(PROJECT_INTERFACE_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) |  | ||||||
|       set(PROJECT_INTERFACE_PACKAGE_DEPENDENCIES ${PROJECT_INTERFACE_PACKAGE_DEPENDENCIES} ${pkgname}) |  | ||||||
|     elseif(${qualifier} STREQUAL INTERFACE) |  | ||||||
|       set(PROJECT_INTERFACE_${pkgname}_VERSION ${__version__}) |  | ||||||
|       set(PROJECT_INTERFACE_${pkgname}_COMPONENTS ${ARGS_COMPONENTS}) |  | ||||||
|       set(PROJECT_INTERFACE_PACKAGE_DEPENDENCIES ${PROJECT_INTERFACE_PACKAGE_DEPENDENCIES} ${pkgname}) |  | ||||||
|     endif() |  | ||||||
|   endif() |  | ||||||
|  |  | ||||||
|   unset(__version__) |  | ||||||
|   unset(__components__) |  | ||||||
|   unset(__required_versions__) |  | ||||||
|   set(CMAKE_PREFIX_PATH ${__old_cpp__}) |  | ||||||
|   unset(__old_cpp__) |  | ||||||
| endmacro() |  | ||||||
|   | |||||||
| @@ -1,87 +1,36 @@ | |||||||
| /******************************************************************************** | /******************************************************************************** | ||||||
|  * Copyright (C) 2014-2019 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * |  * Copyright (C) 2014-2025 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  *                                                                              * |  *                                                                              * | ||||||
|  *              This software is distributed under the terms of the             * |  *              This software is distributed under the terms of the             * | ||||||
|  *              GNU Lesser General Public Licence (LGPL) version 3,             * |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  *                  copied verbatim in the file "LICENSE"                       * |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  ********************************************************************************/ |  ********************************************************************************/ | ||||||
| #include "Logger.h" | #include "Logger.h" | ||||||
|  | #include <string_view> | ||||||
|  |  | ||||||
|  | #if FMT_VERSION < 60000 | ||||||
|  | #include <fmt/time.h> | ||||||
|  | #else | ||||||
|  | #include <fmt/chrono.h> | ||||||
|  | #endif | ||||||
|  |  | ||||||
| #include <iostream> |  | ||||||
| #include <ostream> |  | ||||||
| #include <array> |  | ||||||
| #include <chrono> |  | ||||||
| #include <ctime> // strftime |  | ||||||
| #include <iomanip> // setw, setfill |  | ||||||
| #include <cstdio> // printf | #include <cstdio> // printf | ||||||
|  | #include <iostream> | ||||||
|  | #include <iterator> // std::back_inserter | ||||||
|  |  | ||||||
| using namespace std; | using namespace std; | ||||||
|  |  | ||||||
| namespace fair | namespace fair | ||||||
| { | { | ||||||
|  |  | ||||||
| class ColoredSeverityWriter | using VSpec = VerbositySpec; | ||||||
| { |  | ||||||
|   public: |  | ||||||
|     ColoredSeverityWriter(Severity severity) |  | ||||||
|         : fSeverity(severity) |  | ||||||
|     {} |  | ||||||
|  |  | ||||||
|     friend ostream& operator<<(ostream& os, const ColoredSeverityWriter& w) |  | ||||||
|     { |  | ||||||
|         switch (w.fSeverity) { |  | ||||||
|             case Severity::nolog: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgDefault) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::fatal: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::bgRed) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::error: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgRed) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::warn: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgYellow) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::state: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgMagenta) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::info: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgGreen) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::debug: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgBlue) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::debug1: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgBlue) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::debug2: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgBlue) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::debug3: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgBlue) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::debug4: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgBlue) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             case Severity::trace: |  | ||||||
|                 return os << "\033[01;" << static_cast<int>(Logger::Color::fgCyan) << "m" << Logger::SeverityName(w.fSeverity) << "\033[0m"; |  | ||||||
|                 break; |  | ||||||
|             default: |  | ||||||
|                 return os << "UNKNOWN"; |  | ||||||
|                 break; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|   private: |  | ||||||
|     Severity fSeverity; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| bool Logger::fColored = false; | bool Logger::fColored = false; | ||||||
| fstream Logger::fFileStream; | fstream Logger::fFileStream; | ||||||
| Verbosity Logger::fVerbosity = Verbosity::low; | Verbosity Logger::fVerbosity = Verbosity::low; | ||||||
| Severity Logger::fConsoleSeverity = Severity::info; | Severity Logger::fConsoleSeverity = Severity::FAIR_MIN_SEVERITY > Severity::info ? Severity::FAIR_MIN_SEVERITY : Severity::info; | ||||||
|  | Severity Logger::fMinSeverity = Severity::FAIR_MIN_SEVERITY > Severity::info ? Severity::FAIR_MIN_SEVERITY : Severity::info; | ||||||
| Severity Logger::fFileSeverity = Severity::nolog; | Severity Logger::fFileSeverity = Severity::nolog; | ||||||
| Severity Logger::fMinSeverity = Severity::info; |  | ||||||
| function<void()> Logger::fFatalCallback; | function<void()> Logger::fFatalCallback; | ||||||
| unordered_map<string, pair<Severity, function<void(const string& content, const LogMetaData& metadata)>>> Logger::fCustomSinks; | unordered_map<string, pair<Severity, function<void(const string& content, const LogMetaData& metadata)>>> Logger::fCustomSinks; | ||||||
| mutex Logger::fMtx; | mutex Logger::fMtx; | ||||||
| @@ -96,129 +45,276 @@ const string Logger::fProcessName = program_invocation_short_name; | |||||||
| const string Logger::fProcessName = "?"; | const string Logger::fProcessName = "?"; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| const unordered_map<string, Verbosity> Logger::fVerbosityMap = | const unordered_map<string_view, Verbosity> Logger::fVerbosityMap = | ||||||
| { | { | ||||||
|     { "veryhigh",  Verbosity::veryhigh  }, |     { {"veryhigh"}, Verbosity::veryhigh }, | ||||||
|     { "high",      Verbosity::high      }, |     { {"high"},     Verbosity::high     }, | ||||||
|     { "medium",    Verbosity::medium    }, |     { {"medium"},   Verbosity::medium   }, | ||||||
|     { "low",       Verbosity::low       }, |     { {"low"},      Verbosity::low      }, | ||||||
|     { "verylow",   Verbosity::verylow   }, |     { {"verylow"},  Verbosity::verylow  }, | ||||||
|     { "VERYHIGH",  Verbosity::veryhigh  }, |     { {"VERYHIGH"}, Verbosity::veryhigh }, | ||||||
|     { "HIGH",      Verbosity::high      }, |     { {"HIGH"},     Verbosity::high     }, | ||||||
|     { "MEDIUM",    Verbosity::medium    }, |     { {"MEDIUM"},   Verbosity::medium   }, | ||||||
|     { "LOW",       Verbosity::low       }, |     { {"LOW"},      Verbosity::low      }, | ||||||
|     { "VERYLOW",   Verbosity::verylow   }, |     { {"VERYLOW"},  Verbosity::verylow  }, | ||||||
|     { "user1",     Verbosity::user1     }, |     { {"user1"},    Verbosity::user1    }, | ||||||
|     { "user2",     Verbosity::user2     }, |     { {"user2"},    Verbosity::user2    }, | ||||||
|     { "user3",     Verbosity::user3     }, |     { {"user3"},    Verbosity::user3    }, | ||||||
|     { "user4",     Verbosity::user4     } |     { {"user4"},    Verbosity::user4    } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const unordered_map<string, Severity> Logger::fSeverityMap = | const unordered_map<string_view, Severity> Logger::fSeverityMap = | ||||||
| { | { | ||||||
|     { "nolog",   Severity::nolog   }, |     { {"nolog"},     Severity::nolog     }, | ||||||
|     { "NOLOG",   Severity::nolog   }, |     { {"NOLOG"},     Severity::nolog     }, | ||||||
|     { "error",   Severity::error   }, |     { {"fatal"},     Severity::fatal     }, | ||||||
|     { "ERROR",   Severity::error   }, |     { {"FATAL"},     Severity::fatal     }, | ||||||
|     { "warn",    Severity::warn    }, |     { {"critical"},  Severity::critical  }, | ||||||
|     { "WARN",    Severity::warn    }, |     { {"error"},     Severity::error     }, | ||||||
|     { "warning", Severity::warn    }, |     { {"ERROR"},     Severity::error     }, | ||||||
|     { "WARNING", Severity::warn    }, |     { {"alarm"},     Severity::alarm     }, | ||||||
|     { "state",   Severity::state   }, |     { {"important"}, Severity::important }, | ||||||
|     { "STATE",   Severity::state   }, |     { {"warn"},      Severity::warn      }, | ||||||
|     { "info",    Severity::info    }, |     { {"WARN"},      Severity::warn      }, | ||||||
|     { "INFO",    Severity::info    }, |     { {"warning"},   Severity::warn      }, | ||||||
|     { "debug",   Severity::debug   }, |     { {"WARNING"},   Severity::warn      }, | ||||||
|     { "DEBUG",   Severity::debug   }, |     { {"state"},     Severity::state     }, | ||||||
|     { "debug1",  Severity::debug1  }, |     { {"STATE"},     Severity::state     }, | ||||||
|     { "DEBUG1",  Severity::debug1  }, |     { {"info"},      Severity::info      }, | ||||||
|     { "debug2",  Severity::debug2  }, |     { {"INFO"},      Severity::info      }, | ||||||
|     { "DEBUG2",  Severity::debug2  }, |     { {"detail"},    Severity::detail    }, | ||||||
|     { "debug3",  Severity::debug3  }, |     { {"debug"},     Severity::debug     }, | ||||||
|     { "DEBUG3",  Severity::debug3  }, |     { {"DEBUG"},     Severity::debug     }, | ||||||
|     { "debug4",  Severity::debug4  }, |     { {"debug1"},    Severity::debug1    }, | ||||||
|     { "DEBUG4",  Severity::debug4  }, |     { {"DEBUG1"},    Severity::debug1    }, | ||||||
|     { "trace",   Severity::trace   }, |     { {"debug2"},    Severity::debug2    }, | ||||||
|     { "TRACE",   Severity::trace   } |     { {"DEBUG2"},    Severity::debug2    }, | ||||||
|  |     { {"debug3"},    Severity::debug3    }, | ||||||
|  |     { {"DEBUG3"},    Severity::debug3    }, | ||||||
|  |     { {"debug4"},    Severity::debug4    }, | ||||||
|  |     { {"DEBUG4"},    Severity::debug4    }, | ||||||
|  |     { {"trace"},     Severity::trace     }, | ||||||
|  |     { {"TRACE"},     Severity::trace     } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const array<string, 12> Logger::fSeverityNames = | const array<string_view, 16> Logger::fSeverityNames = | ||||||
| { | { | ||||||
|     { |     { | ||||||
|         "NOLOG", |         "NOLOG", | ||||||
|         "FATAL", |         "TRACE", | ||||||
|         "ERROR", |  | ||||||
|         "WARN", |  | ||||||
|         "STATE", |  | ||||||
|         "INFO", |  | ||||||
|         "DEBUG", |  | ||||||
|         "DEBUG1", |  | ||||||
|         "DEBUG2", |  | ||||||
|         "DEBUG3", |  | ||||||
|         "DEBUG4", |         "DEBUG4", | ||||||
|         "TRACE" |         "DEBUG3", | ||||||
|  |         "DEBUG2", | ||||||
|  |         "DEBUG1", | ||||||
|  |         "DEBUG", | ||||||
|  |         "DETAIL", | ||||||
|  |         "INFO", | ||||||
|  |         "STATE", | ||||||
|  |         "WARN", | ||||||
|  |         "IMPORTANT", | ||||||
|  |         "ALARM", | ||||||
|  |         "ERROR", | ||||||
|  |         "CRITICAL", | ||||||
|  |         "FATAL" | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const array<string, 5> Logger::fVerbosityNames = | const array<string_view, 9> Logger::fVerbosityNames = | ||||||
| { | { | ||||||
|     { |     { | ||||||
|         "verylow", |         "verylow", | ||||||
|         "low", |         "low", | ||||||
|         "medium", |         "medium", | ||||||
|         "high", |         "high", | ||||||
|         "veryhigh" |         "veryhigh", | ||||||
|  |         "user1", | ||||||
|  |         "user2", | ||||||
|  |         "user3", | ||||||
|  |         "user4" | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| std::map<Verbosity, VerbositySpec> Logger::fVerbosities = | map<Verbosity, VSpec> Logger::fVerbosities = | ||||||
| { | { | ||||||
|     {   Verbosity::verylow,  VerbositySpec::Make()                                        }, |     { Verbosity::verylow,  VSpec::Make()                                                                                                             }, | ||||||
|     {   Verbosity::low,      VerbositySpec::Make(VerbositySpec::Info::severity)           }, |     { Verbosity::low,      VSpec::Make(VSpec::Info::severity)                                                                                        }, | ||||||
|     {   Verbosity::medium,   VerbositySpec::Make(VerbositySpec::Info::timestamp_s, |     { Verbosity::medium,   VSpec::Make(VSpec::Info::timestamp_s, VSpec::Info::severity)                                                              }, | ||||||
|                                                  VerbositySpec::Info::severity)           }, |     { Verbosity::high,     VSpec::Make(VSpec::Info::process_name, VSpec::Info::timestamp_s, VSpec::Info::severity)                                   }, | ||||||
|     {   Verbosity::high,     VerbositySpec::Make(VerbositySpec::Info::process_name, |     { Verbosity::veryhigh, VSpec::Make(VSpec::Info::process_name, VSpec::Info::timestamp_us, VSpec::Info::severity, VSpec::Info::file_line_function) }, | ||||||
|                                                  VerbositySpec::Info::timestamp_s, |     { Verbosity::user1,    VSpec::Make(VSpec::Info::severity)                                                                                        }, | ||||||
|                                                  VerbositySpec::Info::severity)           }, |     { Verbosity::user2,    VSpec::Make(VSpec::Info::severity)                                                                                        }, | ||||||
|     {   Verbosity::veryhigh, VerbositySpec::Make(VerbositySpec::Info::process_name, |     { Verbosity::user3,    VSpec::Make(VSpec::Info::severity)                                                                                        }, | ||||||
|                                                  VerbositySpec::Info::timestamp_s, |     { Verbosity::user4,    VSpec::Make(VSpec::Info::severity)                                                                                        } | ||||||
|                                                  VerbositySpec::Info::severity, |  | ||||||
|                                                  VerbositySpec::Info::file_line_function) }, |  | ||||||
|     {   Verbosity::user1,    VerbositySpec::Make(VerbositySpec::Info::severity)           }, |  | ||||||
|     {   Verbosity::user2,    VerbositySpec::Make(VerbositySpec::Info::severity)           }, |  | ||||||
|     {   Verbosity::user3,    VerbositySpec::Make(VerbositySpec::Info::severity)           }, |  | ||||||
|     {   Verbosity::user4,    VerbositySpec::Make(VerbositySpec::Info::severity)           } |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| string Logger::SeverityName(Severity severity) | Logger::Logger(Severity severity, Verbosity verbosity, std::string_view file, std::string_view line, std::string_view func) | ||||||
| { |     : fTimeCalculated(false) | ||||||
|     return fSeverityNames.at(static_cast<size_t>(severity)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| string Logger::VerbosityName(Verbosity verbosity) |  | ||||||
| { |  | ||||||
|     return fVerbosityNames.at(static_cast<size_t>(verbosity)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Logger::Logger(Severity severity, const string& file, const string& line, const string& func) |  | ||||||
| { | { | ||||||
|     if (!fIsDestructed) { |     if (!fIsDestructed) { | ||||||
|         chrono::time_point<chrono::system_clock> now = chrono::system_clock::now(); |  | ||||||
|         size_t pos = file.rfind("/"); |         size_t pos = file.rfind("/"); | ||||||
|  |         // fInfos.timestamp is filled conditionally | ||||||
|  |         // fInfos.us is filled conditionally | ||||||
|  |         fInfos.process_name = fProcessName; | ||||||
|  |         fInfos.file = file.substr(pos + 1); | ||||||
|  |         fInfos.line = line; | ||||||
|  |         fInfos.func = func; | ||||||
|  |         fInfos.severity_name = fSeverityNames.at(static_cast<size_t>(severity)); | ||||||
|  |         fInfos.severity = severity; | ||||||
|  |  | ||||||
|         fMetaData.timestamp = chrono::system_clock::to_time_t(now); |         auto spec = fVerbosities[verbosity]; | ||||||
|         fMetaData.us = chrono::duration_cast<chrono::microseconds>(now.time_since_epoch()) % 1000000; |  | ||||||
|         fMetaData.process_name = fProcessName; |         if ((!fColored && LoggingToConsole()) || LoggingToFile()) { | ||||||
|         fMetaData.file = file.substr(pos + 1); |             for (const auto info : spec.fInfos) { | ||||||
|         fMetaData.line = line; |                 switch (info) { | ||||||
|         fMetaData.func = func; |                     case VSpec::Info::process_name: | ||||||
|         fMetaData.severity_name = fSeverityNames.at(static_cast<size_t>(severity)); |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{}]", fInfos.process_name); | ||||||
|         fMetaData.severity = severity; |                         break; | ||||||
|  |                     case VSpec::Info::timestamp_us: | ||||||
|  |                         FillTimeInfos(); | ||||||
|  |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{:%H:%M:%S}.{:06}]", fmt::localtime(fInfos.timestamp), fInfos.us.count()); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::timestamp_s: | ||||||
|  |                         FillTimeInfos(); | ||||||
|  |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{:%H:%M:%S}]", fmt::localtime(fInfos.timestamp)); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::severity: | ||||||
|  |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{}]", fInfos.severity_name); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::file_line_function: | ||||||
|  |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{}:{}:{}]", fInfos.file, fInfos.line, fInfos.func); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::file_line: | ||||||
|  |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{}:{}]", fInfos.file, fInfos.line); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::file: | ||||||
|  |                         fmt::format_to(std::back_inserter(fBWPrefix), "[{}]", fInfos.file); | ||||||
|  |                         break; | ||||||
|  |                     default: | ||||||
|  |                         break; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (spec.fSize > 0) { | ||||||
|  |                 fmt::format_to(std::back_inserter(fBWPrefix), " "); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (fColored && LoggingToConsole()) { | ||||||
|  |             for (const auto info : spec.fInfos) { | ||||||
|  |                 switch (info) { | ||||||
|  |                     case VSpec::Info::process_name: | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}]", ColorOut(Color::fgBlue, fInfos.process_name)); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::timestamp_us: | ||||||
|  |                         FillTimeInfos(); | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}{:%H:%M:%S}.{:06}{}]", startColor(Color::fgCyan), fmt::localtime(fInfos.timestamp), fInfos.us.count(), endColor()); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::timestamp_s: | ||||||
|  |                         FillTimeInfos(); | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}{:%H:%M:%S}{}]", startColor(Color::fgCyan), fmt::localtime(fInfos.timestamp), endColor()); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::severity: | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}]", GetColoredSeverityString(fInfos.severity)); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::file_line_function: | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}:{}:{}]", ColorOut(Color::fgBlue, fInfos.file), ColorOut(Color::fgYellow, fInfos.line), ColorOut(Color::fgBlue, fInfos.func)); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::file_line: | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}:{}]", ColorOut(Color::fgBlue, fInfos.file), ColorOut(Color::fgYellow, fInfos.line)); | ||||||
|  |                         break; | ||||||
|  |                     case VSpec::Info::file: | ||||||
|  |                         fmt::format_to(std::back_inserter(fColorPrefix), "[{}]", ColorOut(Color::fgBlue, fInfos.file)); | ||||||
|  |                         break; | ||||||
|  |                     default: | ||||||
|  |                         break; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (spec.fSize > 0) { | ||||||
|  |                 fmt::format_to(std::back_inserter(fColorPrefix), " "); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!fCustomSinks.empty()) { | ||||||
|  |             FillTimeInfos(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Logger::~Logger() noexcept(false) | ||||||
|  | { | ||||||
|  |     if (fIsDestructed) { | ||||||
|  |         printf("post-static destruction output: %s\n", fContent.str().c_str()); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     for (auto& it : fCustomSinks) { | ||||||
|  |         if (LoggingCustom(it.second.first)) { | ||||||
|  |             lock_guard<mutex> lock(fMtx); | ||||||
|  |             it.second.second(fContent.str(), fInfos); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // "\n" + flush instead of endl makes output thread safe. | ||||||
|  |  | ||||||
|  |     if (LoggingToConsole()) { | ||||||
|  |         if (fColored) { | ||||||
|  |             fmt::print("{}{}\n", to_string(fColorPrefix), fContent.str()); | ||||||
|  |         } else { | ||||||
|  |             fmt::print("{}{}\n", to_string(fBWPrefix), fContent.str()); | ||||||
|  |         } | ||||||
|  |         cout << flush; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (LoggingToFile()) { | ||||||
|  |         lock_guard<mutex> lock(fMtx); | ||||||
|  |         if (fFileStream.is_open()) { | ||||||
|  |             fFileStream << fmt::format("{}{}\n", to_string(fBWPrefix), fContent.str()) << flush; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (fInfos.severity == Severity::fatal) { | ||||||
|  |         if (fFatalCallback) { | ||||||
|  |             fFatalCallback(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Logger::LogEmptyLine() | ||||||
|  | { | ||||||
|  |     // do nothing, line break is added by the destructor | ||||||
|  |     // this call just to prevent any output to be added to the logger object | ||||||
|  | } | ||||||
|  |  | ||||||
|  | string Logger::GetColoredSeverityString(Severity severity) | ||||||
|  | { | ||||||
|  |     switch (severity) { | ||||||
|  |         case Severity::nolog:     return "\033[01;39mNOLOG\033[0m";     break; | ||||||
|  |         case Severity::fatal:     return "\033[01;31mFATAL\033[0m";     break; | ||||||
|  |         case Severity::critical:  return "\033[01;31mCRITICAL\033[0m";  break; | ||||||
|  |         case Severity::error:     return "\033[01;31mERROR\033[0m";     break; | ||||||
|  |         case Severity::alarm:     return "\033[01;33mALARM\033[0m";     break; | ||||||
|  |         case Severity::important: return "\033[01;32mIMPORTANT\033[0m"; break; | ||||||
|  |         case Severity::warn:      return "\033[01;33mWARN\033[0m";      break; | ||||||
|  |         case Severity::state:     return "\033[01;35mSTATE\033[0m";     break; | ||||||
|  |         case Severity::info:      return "\033[01;32mINFO\033[0m";      break; | ||||||
|  |         case Severity::detail:    return "\033[01;32mDETAIL\033[0m";    break; | ||||||
|  |         case Severity::debug:     return "\033[01;34mDEBUG\033[0m";     break; | ||||||
|  |         case Severity::debug1:    return "\033[01;34mDEBUG1\033[0m";    break; | ||||||
|  |         case Severity::debug2:    return "\033[01;34mDEBUG2\033[0m";    break; | ||||||
|  |         case Severity::debug3:    return "\033[01;34mDEBUG3\033[0m";    break; | ||||||
|  |         case Severity::debug4:    return "\033[01;34mDEBUG4\033[0m";    break; | ||||||
|  |         case Severity::trace:     return "\033[01;36mTRACE\033[0m";     break; | ||||||
|  |         default:                  return "UNKNOWN";                     break; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| void Logger::SetConsoleSeverity(const Severity severity) | void Logger::SetConsoleSeverity(const Severity severity) | ||||||
| { | { | ||||||
|  |     if (severity < Severity::FAIR_MIN_SEVERITY && severity != Severity::nolog) { | ||||||
|  |         cout << "Requested severity is higher than the enabled compile-time FAIR_MIN_SEVERITY (" << Severity::FAIR_MIN_SEVERITY << "), ignoring" << endl; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|     fConsoleSeverity = severity; |     fConsoleSeverity = severity; | ||||||
|     UpdateMinSeverity(); |     UpdateMinSeverity(); | ||||||
| } | } | ||||||
| @@ -240,6 +336,10 @@ Severity Logger::GetConsoleSeverity() | |||||||
|  |  | ||||||
| void Logger::SetFileSeverity(const Severity severity) | void Logger::SetFileSeverity(const Severity severity) | ||||||
| { | { | ||||||
|  |     if (severity < Severity::FAIR_MIN_SEVERITY && severity != Severity::nolog) { | ||||||
|  |         cout << "Requested severity is higher than the enabled compile-time FAIR_MIN_SEVERITY (" << Severity::FAIR_MIN_SEVERITY << "), ignoring" << endl; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|     fFileSeverity = severity; |     fFileSeverity = severity; | ||||||
|     UpdateMinSeverity(); |     UpdateMinSeverity(); | ||||||
| } | } | ||||||
| @@ -256,8 +356,17 @@ void Logger::SetFileSeverity(const string& severityStr) | |||||||
|  |  | ||||||
| void Logger::SetCustomSeverity(const string& key, const Severity severity) | void Logger::SetCustomSeverity(const string& key, const Severity severity) | ||||||
| { | { | ||||||
|     fCustomSinks.at(key).first = severity; // TODO: range checks |     try { | ||||||
|     UpdateMinSeverity(); |         if (severity < Severity::FAIR_MIN_SEVERITY && severity != Severity::nolog) { | ||||||
|  |             cout << "Requested severity is higher than the enabled compile-time FAIR_MIN_SEVERITY (" << Severity::FAIR_MIN_SEVERITY << "), ignoring" << endl; | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         fCustomSinks.at(key).first = severity; | ||||||
|  |         UpdateMinSeverity(); | ||||||
|  |     } catch (const out_of_range& oor) { | ||||||
|  |         LOG(error) << "No custom sink with id '" << key << "' found"; | ||||||
|  |         throw; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| void Logger::SetCustomSeverity(const string& key, const string& severityStr) | void Logger::SetCustomSeverity(const string& key, const string& severityStr) | ||||||
| @@ -270,11 +379,21 @@ void Logger::SetCustomSeverity(const string& key, const string& severityStr) | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| auto Logger::CycleConsoleSeverityUp() -> void | Severity Logger::GetCustomSeverity(const std::string& key) | ||||||
|  | { | ||||||
|  |     try { | ||||||
|  |         return fCustomSinks.at(key).first; | ||||||
|  |     } catch (const out_of_range& oor) { | ||||||
|  |         LOG(error) << "No custom sink with id '" << key << "' found"; | ||||||
|  |         throw; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Logger::CycleConsoleSeverityUp() | ||||||
| { | { | ||||||
|     int current = static_cast<int>(fConsoleSeverity); |     int current = static_cast<int>(fConsoleSeverity); | ||||||
|     if (current == static_cast<int>(fSeverityNames.size()) - 1) { |     if (current == static_cast<int>(fSeverityNames.size()) - 1) { | ||||||
|         SetConsoleSeverity(static_cast<Severity>(0)); |         SetConsoleSeverity(Severity::FAIR_MIN_SEVERITY); | ||||||
|     } else { |     } else { | ||||||
|         SetConsoleSeverity(static_cast<Severity>(current + 1)); |         SetConsoleSeverity(static_cast<Severity>(current + 1)); | ||||||
|     } |     } | ||||||
| @@ -282,17 +401,17 @@ auto Logger::CycleConsoleSeverityUp() -> void | |||||||
|     stringstream ss; |     stringstream ss; | ||||||
|  |  | ||||||
|     for (int i = 0; i < static_cast<int>(fSeverityNames.size()); ++i) { |     for (int i = 0; i < static_cast<int>(fSeverityNames.size()); ++i) { | ||||||
|         ss << (i == newCurrent ? ">" : " ") << fSeverityNames.at(i) << (i == newCurrent ? "<" : " "); |         ss << (i == newCurrent ? "<" : " ") << fSeverityNames.at(i) << (i == newCurrent ? ">" : " "); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ss << "\n\n"; |     ss << "\n\n"; | ||||||
|     cout << ss.str() << flush; |     cout << ss.str() << flush; | ||||||
| } | } | ||||||
|  |  | ||||||
| auto Logger::CycleConsoleSeverityDown() -> void | void Logger::CycleConsoleSeverityDown() | ||||||
| { | { | ||||||
|     int current = static_cast<int>(fConsoleSeverity); |     int current = static_cast<int>(fConsoleSeverity); | ||||||
|     if (current == 0) { |     if (current == static_cast<int>(Severity::FAIR_MIN_SEVERITY)) { | ||||||
|         SetConsoleSeverity(static_cast<Severity>(fSeverityNames.size() - 1)); |         SetConsoleSeverity(static_cast<Severity>(fSeverityNames.size() - 1)); | ||||||
|     } else { |     } else { | ||||||
|         SetConsoleSeverity(static_cast<Severity>(current - 1)); |         SetConsoleSeverity(static_cast<Severity>(current - 1)); | ||||||
| @@ -301,14 +420,14 @@ auto Logger::CycleConsoleSeverityDown() -> void | |||||||
|     stringstream ss; |     stringstream ss; | ||||||
|  |  | ||||||
|     for (int i = 0; i < static_cast<int>(fSeverityNames.size()); ++i) { |     for (int i = 0; i < static_cast<int>(fSeverityNames.size()); ++i) { | ||||||
|         ss << (i == newCurrent ? ">" : " ") << fSeverityNames.at(i) << (i == newCurrent ? "<" : " "); |         ss << (i == newCurrent ? "<" : " ") << fSeverityNames.at(i) << (i == newCurrent ? ">" : " "); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ss << "\n\n"; |     ss << "\n\n"; | ||||||
|     cout << ss.str() << flush; |     cout << ss.str() << flush; | ||||||
| } | } | ||||||
|  |  | ||||||
| auto Logger::CycleVerbosityUp() -> void | void Logger::CycleVerbosityUp() | ||||||
| { | { | ||||||
|     int current = static_cast<int>(fVerbosity); |     int current = static_cast<int>(fVerbosity); | ||||||
|     if (current == static_cast<int>(fVerbosityNames.size() - 1)) { |     if (current == static_cast<int>(fVerbosityNames.size() - 1)) { | ||||||
| @@ -320,14 +439,14 @@ auto Logger::CycleVerbosityUp() -> void | |||||||
|     stringstream ss; |     stringstream ss; | ||||||
|  |  | ||||||
|     for (int i = 0; i < static_cast<int>(fVerbosityNames.size()); ++i) { |     for (int i = 0; i < static_cast<int>(fVerbosityNames.size()); ++i) { | ||||||
|         ss << (i == newCurrent ? ">" : " ") << fVerbosityNames.at(i) << (i == newCurrent ? "<" : " "); |         ss << (i == newCurrent ? "<" : " ") << fVerbosityNames.at(i) << (i == newCurrent ? ">" : " "); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ss << "\n\n"; |     ss << "\n\n"; | ||||||
|     cout << ss.str() << flush; |     cout << ss.str() << flush; | ||||||
| } | } | ||||||
|  |  | ||||||
| auto Logger::CycleVerbosityDown() -> void | void Logger::CycleVerbosityDown() | ||||||
| { | { | ||||||
|     int current = static_cast<int>(fVerbosity); |     int current = static_cast<int>(fVerbosity); | ||||||
|     if (current == 0) { |     if (current == 0) { | ||||||
| @@ -339,7 +458,7 @@ auto Logger::CycleVerbosityDown() -> void | |||||||
|     stringstream ss; |     stringstream ss; | ||||||
|  |  | ||||||
|     for (int i = 0; i < static_cast<int>(fVerbosityNames.size()); ++i) { |     for (int i = 0; i < static_cast<int>(fVerbosityNames.size()); ++i) { | ||||||
|         ss << (i == newCurrent ? ">" : " ") << fVerbosityNames.at(i) << (i == newCurrent ? "<" : " "); |         ss << (i == newCurrent ? "<" : " ") << fVerbosityNames.at(i) << (i == newCurrent ? ">" : " "); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ss << "\n\n"; |     ss << "\n\n"; | ||||||
| @@ -348,28 +467,22 @@ auto Logger::CycleVerbosityDown() -> void | |||||||
|  |  | ||||||
| void Logger::UpdateMinSeverity() | void Logger::UpdateMinSeverity() | ||||||
| { | { | ||||||
|     fMinSeverity = (fConsoleSeverity <= fFileSeverity) ? fFileSeverity : fConsoleSeverity; |     if (fFileSeverity == Severity::nolog) { | ||||||
|  |         fMinSeverity = fConsoleSeverity; | ||||||
|  |     } else { | ||||||
|  |         fMinSeverity = std::max(fConsoleSeverity, fFileSeverity); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     for (auto& it : fCustomSinks) { |     for (auto& it : fCustomSinks) { | ||||||
|         if (fMinSeverity <= it.second.first) { |         if (fMinSeverity == Severity::nolog) { | ||||||
|             fMinSeverity = it.second.first; |             fMinSeverity = std::max(fMinSeverity, it.second.first); | ||||||
|  |         } else if (it.second.first != Severity::nolog) { | ||||||
|  |             fMinSeverity = std::min(fMinSeverity, it.second.first); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| bool Logger::Logging(Severity severity) | bool Logger::Logging(const string& severityStr) | ||||||
| { |  | ||||||
|     if (Severity::fatal == severity) { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|     if (severity <= fMinSeverity && severity > Severity::nolog) { |  | ||||||
|         return true; |  | ||||||
|     } else { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool Logger::Logging(const std::string& severityStr) |  | ||||||
| { | { | ||||||
|     if (fSeverityMap.count(severityStr)) { |     if (fSeverityMap.count(severityStr)) { | ||||||
|         return Logging(fSeverityMap.at(severityStr)); |         return Logging(fSeverityMap.at(severityStr)); | ||||||
| @@ -404,7 +517,7 @@ void Logger::DefineVerbosity(const Verbosity verbosity, const VerbositySpec spec | |||||||
|     fVerbosities[verbosity] = spec; |     fVerbosities[verbosity] = spec; | ||||||
| } | } | ||||||
|  |  | ||||||
| void Logger::DefineVerbosity(const std::string& verbosityStr, const VerbositySpec spec) | void Logger::DefineVerbosity(const string& verbosityStr, const VerbositySpec spec) | ||||||
| { | { | ||||||
|     if (fVerbosityMap.count(verbosityStr)) { |     if (fVerbosityMap.count(verbosityStr)) { | ||||||
|         DefineVerbosity(fVerbosityMap.at(verbosityStr), spec); |         DefineVerbosity(fVerbosityMap.at(verbosityStr), spec); | ||||||
| @@ -418,7 +531,7 @@ void Logger::SetConsoleColor(const bool colored) | |||||||
|     fColored = colored; |     fColored = colored; | ||||||
| } | } | ||||||
|  |  | ||||||
| void Logger::InitFileSink(const Severity severity, const string& filename, bool customizeName) | string Logger::InitFileSink(const Severity severity, const string& filename, bool customizeName) | ||||||
| { | { | ||||||
|     lock_guard<mutex> lock(fMtx); |     lock_guard<mutex> lock(fMtx); | ||||||
|     if (fFileStream.is_open()) { |     if (fFileStream.is_open()) { | ||||||
| @@ -433,8 +546,7 @@ void Logger::InitFileSink(const Severity severity, const string& filename, bool | |||||||
|         stringstream ss; |         stringstream ss; | ||||||
|         ss << "_"; |         ss << "_"; | ||||||
|         char tsstr[32]; |         char tsstr[32]; | ||||||
|         if (strftime(tsstr, sizeof(tsstr), "%Y-%m-%d_%H_%M_%S", localtime(&now))) |         if (strftime(tsstr, sizeof(tsstr), "%Y-%m-%d_%H_%M_%S", localtime(&now))) { | ||||||
|         { |  | ||||||
|             ss << tsstr; |             ss << tsstr; | ||||||
|         } |         } | ||||||
|         ss << ".log"; |         ss << ".log"; | ||||||
| @@ -444,21 +556,27 @@ void Logger::InitFileSink(const Severity severity, const string& filename, bool | |||||||
|     fFileStream.open(fullName, fstream::out | fstream::app); |     fFileStream.open(fullName, fstream::out | fstream::app); | ||||||
|  |  | ||||||
|     if (fFileStream.is_open()) { |     if (fFileStream.is_open()) { | ||||||
|         fFileSeverity = severity; |         if (severity < Severity::FAIR_MIN_SEVERITY && severity != Severity::nolog) { | ||||||
|  |             cout << "Requested file sink severity is higher than the enabled compile-time FAIR_MIN_SEVERITY (" << Severity::FAIR_MIN_SEVERITY << "), setting to " << Severity::FAIR_MIN_SEVERITY << endl; | ||||||
|  |             fFileSeverity = Severity::FAIR_MIN_SEVERITY; | ||||||
|  |         } else { | ||||||
|  |             fFileSeverity = severity; | ||||||
|  |         } | ||||||
|         UpdateMinSeverity(); |         UpdateMinSeverity(); | ||||||
|     } else { |     } else { | ||||||
|         cout << "Error opening file: " << fullName; |         cout << "Error opening file: " << fullName; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     return fullName; | ||||||
| } | } | ||||||
|  |  | ||||||
| void Logger::InitFileSink(const string& severityStr, const string& filename, bool customizeName) | string Logger::InitFileSink(const string& severityStr, const string& filename, bool customizeName) | ||||||
| { | { | ||||||
|     if (fSeverityMap.count(severityStr)) { |     if (fSeverityMap.count(severityStr)) { | ||||||
|         InitFileSink(fSeverityMap.at(severityStr), filename, customizeName); |         return InitFileSink(fSeverityMap.at(severityStr), filename, customizeName); | ||||||
|     } else { |     } else { | ||||||
|         LOG(error) << "Unknown severity setting: '" << severityStr << "', setting to default 'info'."; |         LOG(error) << "Unknown severity setting: '" << severityStr << "', setting to default 'info'."; | ||||||
|         InitFileSink(Severity::info, filename); |         return InitFileSink(Severity::info, filename); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -467,28 +585,30 @@ void Logger::RemoveFileSink() | |||||||
|     lock_guard<mutex> lock(fMtx); |     lock_guard<mutex> lock(fMtx); | ||||||
|     if (fFileStream.is_open()) { |     if (fFileStream.is_open()) { | ||||||
|         fFileStream.close(); |         fFileStream.close(); | ||||||
|  |         fFileSeverity = Severity::nolog; | ||||||
|  |         UpdateMinSeverity(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| bool Logger::LoggingToConsole() const | bool Logger::LoggingToConsole() const | ||||||
| { | { | ||||||
|     return (fMetaData.severity <= fConsoleSeverity && |     return (fInfos.severity >= fConsoleSeverity && | ||||||
|            fMetaData.severity > Severity::nolog) || |             fConsoleSeverity > Severity::nolog) || | ||||||
|            fMetaData.severity == Severity::fatal; |             fInfos.severity == Severity::fatal; | ||||||
| } | } | ||||||
|  |  | ||||||
| bool Logger::LoggingToFile() const | bool Logger::LoggingToFile() const | ||||||
| { | { | ||||||
|     return (fMetaData.severity <= fFileSeverity && |     return (fInfos.severity >= fFileSeverity && | ||||||
|            fMetaData.severity > Severity::nolog) || |             fFileSeverity   >  Severity::nolog) || | ||||||
|            fMetaData.severity == Severity::fatal; |             fInfos.severity == Severity::fatal; | ||||||
| } | } | ||||||
|  |  | ||||||
| bool Logger::LoggingCustom(const Severity severity) const | bool Logger::LoggingCustom(const Severity severity) const | ||||||
| { | { | ||||||
|     return (fMetaData.severity <= severity && |     return (fInfos.severity >= severity && | ||||||
|            fMetaData.severity > Severity::nolog) || |             severity        > Severity::nolog) || | ||||||
|            fMetaData.severity == Severity::fatal; |             fInfos.severity == Severity::fatal; | ||||||
| } | } | ||||||
|  |  | ||||||
| void Logger::OnFatal(function<void()> func) | void Logger::OnFatal(function<void()> func) | ||||||
| @@ -500,10 +620,16 @@ void Logger::AddCustomSink(const string& key, Severity severity, function<void(c | |||||||
| { | { | ||||||
|     lock_guard<mutex> lock(fMtx); |     lock_guard<mutex> lock(fMtx); | ||||||
|     if (fCustomSinks.count(key) == 0) { |     if (fCustomSinks.count(key) == 0) { | ||||||
|         fCustomSinks.insert(make_pair(key, make_pair(severity, func))); |         if (severity < Severity::FAIR_MIN_SEVERITY && severity != Severity::nolog) { | ||||||
|  |             cout << "Requested custom sink severity is higher than the enabled compile-time FAIR_MIN_SEVERITY (" << Severity::FAIR_MIN_SEVERITY << "), setting to " << Severity::FAIR_MIN_SEVERITY << endl; | ||||||
|  |             fCustomSinks.insert(make_pair(key, make_pair(Severity::FAIR_MIN_SEVERITY, func))); | ||||||
|  |         } else { | ||||||
|  |             fCustomSinks.insert(make_pair(key, make_pair(severity, func))); | ||||||
|  |         } | ||||||
|         UpdateMinSeverity(); |         UpdateMinSeverity(); | ||||||
|     } else { |     } else { | ||||||
|         cout << "Logger::AddCustomSink: sink '" << key << "' already exists, will not add again. Remove first with Logger::RemoveCustomSink(const string& key)" << endl; |         cout << "Logger::AddCustomSink: sink '" << key << "' already exists, will not add again. Remove first with Logger::RemoveCustomSink(const string& key)" << endl; | ||||||
|  |         throw runtime_error("Adding a sink with a key that already exists. Remove first."); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -524,116 +650,10 @@ void Logger::RemoveCustomSink(const string& key) | |||||||
|         UpdateMinSeverity(); |         UpdateMinSeverity(); | ||||||
|     } else { |     } else { | ||||||
|         cout << "Logger::RemoveCustomSink: sink '" << key << "' doesn't exists, will not remove." << endl; |         cout << "Logger::RemoveCustomSink: sink '" << key << "' doesn't exists, will not remove." << endl; | ||||||
|  |         throw runtime_error("Trying to remove a sink with a key that does not exist."); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| Logger& Logger::Log() |  | ||||||
| { |  | ||||||
|     if (fIsDestructed) { |  | ||||||
|         return *this; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     char tsstr[32]; |  | ||||||
|     { |  | ||||||
|         lock_guard<mutex> lock(fMtx); // localtime is not threadsafe, guard it |  | ||||||
|         if (!strftime(tsstr, sizeof(tsstr), "%H:%M:%S", localtime(&(fMetaData.timestamp)))) { |  | ||||||
|             tsstr[0] = 'u'; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     auto spec = fVerbosities[fVerbosity]; |  | ||||||
|  |  | ||||||
|     if ((!fColored && LoggingToConsole()) || LoggingToFile()) { |  | ||||||
|         bool appendSpace = false; |  | ||||||
|         for (const auto info : spec.fOrder) { |  | ||||||
|             switch (info) { |  | ||||||
|                 case VerbositySpec::Info::process_name: |  | ||||||
|                     fBWOut << "[" << fMetaData.process_name << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::timestamp_us: |  | ||||||
|                     fBWOut << "[" << tsstr << "." << setw(6) << setfill('0') << fMetaData.us.count() << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::timestamp_s: |  | ||||||
|                     fBWOut << "[" << tsstr << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::severity: |  | ||||||
|                     fBWOut << "[" << fMetaData.severity_name << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::file_line_function: |  | ||||||
|                     fBWOut << "[" << fMetaData.file << ":" << fMetaData.line << ":" << fMetaData.func << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::file_line: |  | ||||||
|                     fBWOut << "[" << fMetaData.file << ":" << fMetaData.line << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::file: |  | ||||||
|                     fBWOut << "[" << fMetaData.file << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 default: |  | ||||||
|                     break; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (appendSpace) { |  | ||||||
|             fBWOut << " "; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (fColored && LoggingToConsole()) { |  | ||||||
|         bool appendSpace = false; |  | ||||||
|         for (const auto info : spec.fOrder) { |  | ||||||
|             switch (info) { |  | ||||||
|                 case VerbositySpec::Info::process_name: |  | ||||||
|                     fColorOut << "[" << ColorOut(Color::fgBlue, fMetaData.process_name) << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::timestamp_us: |  | ||||||
|                     fColorOut << "[" << startColor(Color::fgCyan) << tsstr << "." |  | ||||||
|                                     << setw(6) << setfill('0') << fMetaData.us.count() << endColor() << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::timestamp_s: |  | ||||||
|                     fColorOut << "[" << startColor(Color::fgCyan) << tsstr << endColor() << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::severity: |  | ||||||
|                     fColorOut << "[" << ColoredSeverityWriter(fMetaData.severity) << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::file_line_function: |  | ||||||
|                     fColorOut << "[" << ColorOut(Color::fgBlue, fMetaData.file) << ":" |  | ||||||
|                                     << ColorOut(Color::fgYellow, fMetaData.line) << ":" |  | ||||||
|                                     << ColorOut(Color::fgBlue, fMetaData.func) << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::file_line: |  | ||||||
|                     fColorOut << "[" << ColorOut(Color::fgBlue, fMetaData.file) << ":" |  | ||||||
|                                     << ColorOut(Color::fgYellow, fMetaData.line) << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 case VerbositySpec::Info::file: |  | ||||||
|                     fColorOut << "[" << ColorOut(Color::fgBlue, fMetaData.file) << "]"; |  | ||||||
|                     appendSpace = true; |  | ||||||
|                     break; |  | ||||||
|                 default: |  | ||||||
|                     break; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (appendSpace) { |  | ||||||
|             fColorOut << " "; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return *this; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Logger& Logger::operator<<(ios_base& (*manip) (ios_base&)) | Logger& Logger::operator<<(ios_base& (*manip) (ios_base&)) | ||||||
| { | { | ||||||
|     fContent << manip; |     fContent << manip; | ||||||
| @@ -646,44 +666,13 @@ Logger& Logger::operator<<(ostream& (*manip) (ostream&)) | |||||||
|     return *this; |     return *this; | ||||||
| } | } | ||||||
|  |  | ||||||
| Logger::~Logger() noexcept(false) | void Logger::FillTimeInfos() | ||||||
| { | { | ||||||
|     if (fIsDestructed) { |     if (!fTimeCalculated) { | ||||||
|         printf("post-static destruction output: %s\n", fContent.str().c_str()); |         chrono::time_point<chrono::system_clock> now = chrono::system_clock::now(); | ||||||
|         return; |         fInfos.timestamp = chrono::system_clock::to_time_t(now); | ||||||
|     } |         fInfos.us = chrono::duration_cast<chrono::microseconds>(now.time_since_epoch()) % 1000000; | ||||||
|  |         fTimeCalculated = true; | ||||||
|     for (auto& it : fCustomSinks) { |  | ||||||
|         if (LoggingCustom(it.second.first)) { |  | ||||||
|             lock_guard<mutex> lock(fMtx); |  | ||||||
|             it.second.second(fContent.str(), fMetaData); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fContent << "\n"; // "\n" + flush instead of endl makes output thread safe. |  | ||||||
|  |  | ||||||
|     fBWOut << fContent.str(); |  | ||||||
|  |  | ||||||
|     if (LoggingToConsole()) { |  | ||||||
|         if (fColored) { |  | ||||||
|             fColorOut << fContent.str(); |  | ||||||
|             cout << fColorOut.str() << flush; |  | ||||||
|         } else { |  | ||||||
|             cout << fBWOut.str() << flush; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (LoggingToFile()) { |  | ||||||
|         lock_guard<mutex> lock(fMtx); |  | ||||||
|         if (fFileStream.is_open()) { |  | ||||||
|             fFileStream << fBWOut.str() << flush; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (fMetaData.severity == Severity::fatal) { |  | ||||||
|         if (fFatalCallback) { |  | ||||||
|             fFatalCallback(); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										341
									
								
								logger/Logger.h
									
									
									
									
									
								
							
							
						
						
									
										341
									
								
								logger/Logger.h
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| /******************************************************************************** | /******************************************************************************** | ||||||
|  *    Copyright (C) 2014 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH    * |  * Copyright (C) 2014-2025 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  *                                                                              * |  *                                                                              * | ||||||
|  *              This software is distributed under the terms of the             * |  *              This software is distributed under the terms of the             * | ||||||
|  *              GNU Lesser General Public Licence (LGPL) version 3,             * |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
| @@ -13,83 +13,104 @@ | |||||||
| #warning "The symbol 'DEBUG' is used in FairRoot Logger. undefining..." | #warning "The symbol 'DEBUG' is used in FairRoot Logger. undefining..." | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #include <sstream> | #ifndef FAIR_MIN_SEVERITY | ||||||
| #include <fstream> | #define FAIR_MIN_SEVERITY nolog | ||||||
| #include <string> | #endif | ||||||
| #include <unordered_map> |  | ||||||
| #include <functional> |  | ||||||
| #include <map> |  | ||||||
| #include <chrono> |  | ||||||
| #include <mutex> |  | ||||||
| #include <utility> // pair |  | ||||||
| #include <time.h> // time_t |  | ||||||
| #include <array> |  | ||||||
| #include <type_traits> |  | ||||||
| #include <cassert> |  | ||||||
| #include <algorithm> |  | ||||||
|  |  | ||||||
| #ifdef FAIRLOGGER_USE_BOOST_PRETTY_FUNCTION | #ifdef FAIRLOGGER_USE_BOOST_PRETTY_FUNCTION | ||||||
| #include <boost/current_function.hpp> | #include <boost/current_function.hpp> | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | #pragma GCC diagnostic push | ||||||
|  | #pragma GCC diagnostic ignored "-Wshadow" | ||||||
|  |  | ||||||
|  | #include <fmt/core.h> | ||||||
|  | #include <fmt/printf.h> | ||||||
|  | #include <fmt/ostream.h> | ||||||
|  |  | ||||||
|  | #pragma GCC diagnostic pop | ||||||
|  |  | ||||||
|  | #include <algorithm> | ||||||
|  | #include <array> | ||||||
|  | #include <cassert> | ||||||
|  | #include <chrono> | ||||||
|  | #include <fstream> | ||||||
|  | #include <functional> | ||||||
|  | #include <map> | ||||||
|  | #include <mutex> | ||||||
|  | #include <ostream> | ||||||
|  | #include <sstream> | ||||||
|  | #include <stdexcept> | ||||||
|  | #include <string> | ||||||
|  | #include <time.h> // time_t | ||||||
|  | #include <type_traits> // is_same | ||||||
|  | #include <unordered_map> | ||||||
|  | #include <string_view> | ||||||
|  | #include <utility> // pair | ||||||
|  |  | ||||||
| namespace fair | namespace fair | ||||||
| { | { | ||||||
|  |  | ||||||
| enum class Severity : int | enum class Severity : int | ||||||
| { | { | ||||||
|     nolog, |     nolog = 0, | ||||||
|     fatal, |     trace = 1, | ||||||
|     error, |     debug4 = 2, | ||||||
|     warn, |     debug3 = 3, | ||||||
|     state, |     debug2 = 4, | ||||||
|     info, |     debug1 = 5, | ||||||
|     debug, |     debug = 6, | ||||||
|     debug1, |     detail = 7, | ||||||
|     debug2, |     info = 8, | ||||||
|     debug3, |     state = 9, | ||||||
|     debug4, |     warn = 10, | ||||||
|     trace, |     important = 11, | ||||||
|     // backwards-compatibility: |     alarm = 12, | ||||||
|     NOLOG = nolog, |     error = 13, | ||||||
|     FATAL = fatal, |     critical = 14, | ||||||
|     ERROR = error, |     fatal = 15, | ||||||
|     WARN = warn, |     // aliases | ||||||
|     warning = warn, |     warning = warn, | ||||||
|     WARNING = warn, |     // backwards-compatibility | ||||||
|     STATE = state, |     NOLOG   __attribute__((deprecated("Use LOG(nolog) instead (lowercase severity name)."))) = nolog, | ||||||
|     INFO = info, |     FATAL   __attribute__((deprecated("Use LOG(fatal) instead (lowercase severity name)."))) = fatal, | ||||||
|     DEBUG = debug, |     ERROR   __attribute__((deprecated("Use LOG(error) instead (lowercase severity name)."))) = error, | ||||||
|     DEBUG1 = debug1, |     WARN    __attribute__((deprecated("Use LOG(warn) instead (lowercase severity name)."))) = warn, | ||||||
|     DEBUG2 = debug2, |     WARNING __attribute__((deprecated("Use LOG(warning) instead (lowercase severity name)."))) = warn, | ||||||
|     DEBUG3 = debug3, |     STATE   __attribute__((deprecated("Use LOG(state) instead (lowercase severity name)."))) = state, | ||||||
|     DEBUG4 = debug4, |     INFO    __attribute__((deprecated("Use LOG(info) instead (lowercase severity name)."))) = info, | ||||||
|     TRACE = trace |     DEBUG   __attribute__((deprecated("Use LOG(debug) instead (lowercase severity name)."))) = debug, | ||||||
|  |     DEBUG1  __attribute__((deprecated("Use LOG(debug1) instead (lowercase severity name)."))) = debug1, | ||||||
|  |     DEBUG2  __attribute__((deprecated("Use LOG(debug2) instead (lowercase severity name)."))) = debug2, | ||||||
|  |     DEBUG3  __attribute__((deprecated("Use LOG(debug3) instead (lowercase severity name)."))) = debug3, | ||||||
|  |     DEBUG4  __attribute__((deprecated("Use LOG(debug4) instead (lowercase severity name)."))) = debug4, | ||||||
|  |     TRACE   __attribute__((deprecated("Use LOG(trace) instead (lowercase severity name)."))) = trace | ||||||
| }; | }; | ||||||
|  |  | ||||||
| // verbosity levels: | // verbosity levels: | ||||||
| // verylow:  message | // verylow:  message | ||||||
| // low:      [severity] message | // low:      [severity] message | ||||||
| // medium:   [HH:MM:SS][severity] message | // medium:   [HH:MM:SS][severity] message | ||||||
| // high:     [process name][HH:MM:SS:µS][severity] message | // high:     [process_name][HH:MM:SS][severity] message | ||||||
| // veryhigh: [process name][HH:MM:SS:µS][severity][file:line:function] message | // veryhigh: [process_name][HH:MM:SS:µS][severity][file:line:function] message | ||||||
| enum class Verbosity : int | enum class Verbosity : int | ||||||
| { | { | ||||||
|     verylow, |     verylow = 0, | ||||||
|     low, |     low, | ||||||
|     medium, |     medium, | ||||||
|     high, |     high, | ||||||
|     veryhigh, |     veryhigh, | ||||||
|     // backwards-compatibility: |  | ||||||
|     VERYLOW = verylow, |  | ||||||
|     LOW = low, |  | ||||||
|     MEDIUM = medium, |  | ||||||
|     HIGH = high, |  | ||||||
|     VERYHIGH = veryhigh, |  | ||||||
|     // extra slots for user-defined verbosities: |     // extra slots for user-defined verbosities: | ||||||
|     user1, |     user1, | ||||||
|     user2, |     user2, | ||||||
|     user3, |     user3, | ||||||
|     user4, |     user4, | ||||||
|  |     // backwards-compatibility: | ||||||
|  |     VERYLOW = verylow, | ||||||
|  |     LOW = low, | ||||||
|  |     MEDIUM = medium, | ||||||
|  |     HIGH = high, | ||||||
|  |     VERYHIGH = veryhigh | ||||||
| }; | }; | ||||||
|  |  | ||||||
| struct VerbositySpec | struct VerbositySpec | ||||||
| @@ -97,7 +118,7 @@ struct VerbositySpec | |||||||
|     enum class Info : int |     enum class Info : int | ||||||
|     { |     { | ||||||
|         __empty__ = 0,         // used to initialize order array |         __empty__ = 0,         // used to initialize order array | ||||||
|         process_name,          // [process name] |         process_name,          // [process_name] | ||||||
|         timestamp_s,           // [HH:MM:SS] |         timestamp_s,           // [HH:MM:SS] | ||||||
|         timestamp_us,          // [HH:MM:SS:µS] |         timestamp_us,          // [HH:MM:SS:µS] | ||||||
|         severity,              // [severity] |         severity,              // [severity] | ||||||
| @@ -107,59 +128,46 @@ struct VerbositySpec | |||||||
|         __max__                // needs to be last in enum |         __max__                // needs to be last in enum | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     std::array<Info, static_cast<int>(Info::__max__)> fOrder; |     std::array<Info, static_cast<int>(Info::__max__)> fInfos; | ||||||
|  |     int fSize; | ||||||
|  |  | ||||||
|     VerbositySpec() : fOrder({Info::__empty__}) {} |     VerbositySpec() : fInfos({Info::__empty__}), fSize(0) {} | ||||||
|  |  | ||||||
|     template<typename ... Ts> |     template<typename ... Ts> | ||||||
|     static VerbositySpec Make(Ts ... options) |     static VerbositySpec Make(Ts ... options) | ||||||
|     { |     { | ||||||
|       static_assert(sizeof...(Ts) < static_cast<int>(Info::__max__), |         static_assert(sizeof...(Ts) < static_cast<int>(Info::__max__), "Maximum number of VerbositySpec::Info parameters exceeded."); | ||||||
|                     "Maximum number of VerbositySpec::Info parameters exceeded."); |         return Make(VerbositySpec(), 0, options...); | ||||||
|  |  | ||||||
|       return Make(VerbositySpec(), 0, options...); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   private: |   private: | ||||||
|     template<typename T, typename ... Ts> |     template<typename T, typename ... Ts> | ||||||
|     static VerbositySpec Make(VerbositySpec spec, int i, T option, Ts ... options) |     static VerbositySpec Make(VerbositySpec spec, int i, T option, Ts ... options) | ||||||
|     { |     { | ||||||
|         static_assert(std::is_same<T, Info>::value, |         static_assert(std::is_same<T, Info>::value, "Only arguments of type VerbositySpec::Info are allowed."); | ||||||
|                       "Only arguments of type VerbositySpec::Info are allowed."); |  | ||||||
|  |  | ||||||
|         assert(option > Info::__empty__); |         assert(option > Info::__empty__); | ||||||
|         assert(option < Info::__max__); |         assert(option < Info::__max__); | ||||||
|  |  | ||||||
|         if (std::find(spec.fOrder.begin(), spec.fOrder.end(), option) == spec.fOrder.end()) { |         if (std::find(spec.fInfos.begin(), spec.fInfos.end(), option) == spec.fInfos.end()) { | ||||||
|             spec.fOrder[i] = option; |             spec.fInfos[i] = option; | ||||||
|             ++i; |             ++i; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return Make(spec, i, options ...); |         return Make(spec, i, options ...); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     static VerbositySpec Make(VerbositySpec spec, int) |     static VerbositySpec Make(VerbositySpec spec, int i) { spec.fSize = i; return spec; } | ||||||
|     { |  | ||||||
|         return spec; |  | ||||||
|     } |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| // non-std exception to avoid undesirable catches - fatal should exit in a way we want. | // non-std exception to avoid undesirable catches - fatal should exit in a way we want. | ||||||
| class FatalException | class FatalException | ||||||
| { | { | ||||||
|   public: |   public: | ||||||
|     FatalException() |     FatalException() : fWhat() {} | ||||||
|         : fWhat() |     FatalException(std::string what) : fWhat(what) {} | ||||||
|     {} |  | ||||||
|  |  | ||||||
|     FatalException(std::string what) |     std::string What() { return fWhat; } | ||||||
|         : fWhat(what) |  | ||||||
|     {} |  | ||||||
|  |  | ||||||
|     std::string What() |  | ||||||
|     { |  | ||||||
|         return fWhat; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|   private: |   private: | ||||||
|     std::string fWhat; |     std::string fWhat; | ||||||
| @@ -169,18 +177,26 @@ struct LogMetaData | |||||||
| { | { | ||||||
|     std::time_t timestamp; |     std::time_t timestamp; | ||||||
|     std::chrono::microseconds us; |     std::chrono::microseconds us; | ||||||
|     std::string process_name; |     std::string_view process_name; | ||||||
|     std::string file; |     std::string_view file; | ||||||
|     std::string line; |     std::string_view line; | ||||||
|     std::string func; |     std::string_view func; | ||||||
|     std::string severity_name; |     std::string_view severity_name; | ||||||
|     fair::Severity severity; |     fair::Severity severity; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| class Logger | class Logger | ||||||
| { | { | ||||||
|   public: |   public: | ||||||
|     Logger(Severity severity, const std::string& file, const std::string& line, const std::string& func); |     Logger(Severity severity, Verbosity verbosity, std::string_view file, std::string_view line, std::string_view func); | ||||||
|  |     Logger(Severity severity, std::string_view file, std::string_view line, std::string_view func) | ||||||
|  |         : Logger(severity, fVerbosity, file, line, func) | ||||||
|  |     {} | ||||||
|  |     virtual ~Logger() noexcept(false); | ||||||
|  |  | ||||||
|  |     Logger& Log() { return *this; } | ||||||
|  |  | ||||||
|  |     void LogEmptyLine(); | ||||||
|  |  | ||||||
|     enum class Color : int |     enum class Color : int | ||||||
|     { |     { | ||||||
| @@ -228,35 +244,10 @@ class Logger | |||||||
|         bgWhite        = 107 |         bgWhite        = 107 | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     static std::string startColor(Color color) |     static std::string startColor(Color color) { return fmt::format("\033[01;{}m", static_cast<int>(color)); } | ||||||
|     { |     static std::string endColor() { return "\033[0m"; } | ||||||
|         std::ostringstream os; |     static std::string ColorOut(Color c, std::string_view s) { return fmt::format("\033[01;{}m{}\033[0m", static_cast<int>(c), s); } | ||||||
|         os << "\033[01;" << static_cast<int>(color) << "m"; |     static std::string GetColoredSeverityString(Severity severity); | ||||||
|         return os.str(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static std::string endColor() |  | ||||||
|     { |  | ||||||
|         return "\033[0m"; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     class ColorOut |  | ||||||
|     { |  | ||||||
|       public: |  | ||||||
|         ColorOut(Color color, const std::string& str) |  | ||||||
|             : fColor(color) |  | ||||||
|             , fStr(str) |  | ||||||
|         {} |  | ||||||
|  |  | ||||||
|         friend std::ostream& operator<<(std::ostream& os, const ColorOut& w) |  | ||||||
|         { |  | ||||||
|             return os << "\033[01;" << static_cast<int>(w.fColor) << "m" << w.fStr << "\033[0m"; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|       private: |  | ||||||
|         Color fColor; |  | ||||||
|         const std::string& fStr; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     static void SetConsoleSeverity(const Severity severity); |     static void SetConsoleSeverity(const Severity severity); | ||||||
|     static void SetConsoleSeverity(const std::string& severityStr); |     static void SetConsoleSeverity(const std::string& severityStr); | ||||||
| @@ -264,16 +255,23 @@ class Logger | |||||||
|  |  | ||||||
|     static void SetFileSeverity(const Severity severity); |     static void SetFileSeverity(const Severity severity); | ||||||
|     static void SetFileSeverity(const std::string& severityStr); |     static void SetFileSeverity(const std::string& severityStr); | ||||||
|  |     static Severity GetFileSeverity() { return fFileSeverity; } | ||||||
|  |  | ||||||
|     static void SetCustomSeverity(const std::string& key, const Severity severity); |     static void SetCustomSeverity(const std::string& key, const Severity severity); | ||||||
|     static void SetCustomSeverity(const std::string& key, const std::string& severityStr); |     static void SetCustomSeverity(const std::string& key, const std::string& severityStr); | ||||||
|  |     static Severity GetCustomSeverity(const std::string& key); | ||||||
|  |  | ||||||
|     static void CycleConsoleSeverityUp(); |     static void CycleConsoleSeverityUp(); | ||||||
|     static void CycleConsoleSeverityDown(); |     static void CycleConsoleSeverityDown(); | ||||||
|     static void CycleVerbosityUp(); |     static void CycleVerbosityUp(); | ||||||
|     static void CycleVerbosityDown(); |     static void CycleVerbosityDown(); | ||||||
|  |  | ||||||
|     static bool Logging(const Severity severity); |     static bool Logging(const Severity severity) | ||||||
|  |     { | ||||||
|  |         return (severity >= fMinSeverity && | ||||||
|  |                 fMinSeverity > Severity::nolog) || | ||||||
|  |                 severity == Severity::fatal; | ||||||
|  |     } | ||||||
|     static bool Logging(const std::string& severityStr); |     static bool Logging(const std::string& severityStr); | ||||||
|  |  | ||||||
|     static void SetVerbosity(const Verbosity verbosity); |     static void SetVerbosity(const Verbosity verbosity); | ||||||
| @@ -284,13 +282,13 @@ class Logger | |||||||
|  |  | ||||||
|     static void SetConsoleColor(const bool colored = true); |     static void SetConsoleColor(const bool colored = true); | ||||||
|  |  | ||||||
|     static void InitFileSink(const Severity severity, const std::string& filename, bool customizeName = true); |     static std::string InitFileSink(const Severity severity, const std::string& filename, bool customizeName = true); | ||||||
|     static void InitFileSink(const std::string& severityStr, const std::string& filename, bool customizeName = true); |     static std::string InitFileSink(const std::string& severityStr, const std::string& filename, bool customizeName = true); | ||||||
|  |  | ||||||
|     static void RemoveFileSink(); |     static void RemoveFileSink(); | ||||||
|  |  | ||||||
|     static std::string SeverityName(Severity); |     static std::string_view SeverityName(Severity s) { return fSeverityNames.at(static_cast<size_t>(s)); } | ||||||
|     static std::string VerbosityName(Verbosity); |     static std::string_view VerbosityName(Verbosity v) { return fVerbosityNames.at(static_cast<size_t>(v)); } | ||||||
|  |  | ||||||
|     static void OnFatal(std::function<void()> func); |     static void OnFatal(std::function<void()> func); | ||||||
|  |  | ||||||
| @@ -298,8 +296,6 @@ class Logger | |||||||
|     static void AddCustomSink(const std::string& key, const std::string& severityStr, std::function<void(const std::string& content, const LogMetaData& metadata)> sink); |     static void AddCustomSink(const std::string& key, const std::string& severityStr, std::function<void(const std::string& content, const LogMetaData& metadata)> sink); | ||||||
|     static void RemoveCustomSink(const std::string& key); |     static void RemoveCustomSink(const std::string& key); | ||||||
|  |  | ||||||
|     Logger& Log(); |  | ||||||
|  |  | ||||||
|     template<typename T> |     template<typename T> | ||||||
|     Logger& operator<<(const T& t) |     Logger& operator<<(const T& t) | ||||||
|     { |     { | ||||||
| @@ -328,23 +324,26 @@ class Logger | |||||||
|     Logger& operator<<(std::ios_base& (*manip) (std::ios_base&)); |     Logger& operator<<(std::ios_base& (*manip) (std::ios_base&)); | ||||||
|     Logger& operator<<(std::ostream& (*manip) (std::ostream&)); |     Logger& operator<<(std::ostream& (*manip) (std::ostream&)); | ||||||
|  |  | ||||||
|     static const std::unordered_map<std::string, Verbosity> fVerbosityMap; |     static const std::unordered_map<std::string_view, Verbosity> fVerbosityMap; | ||||||
|     static const std::unordered_map<std::string, Severity> fSeverityMap; |     static const std::unordered_map<std::string_view, Severity> fSeverityMap; | ||||||
|     static const std::array<std::string, 12> fSeverityNames; |     static const std::array<std::string_view, 16> fSeverityNames; | ||||||
|     static const std::array<std::string, 5> fVerbosityNames; |     static const std::array<std::string_view, 9> fVerbosityNames; | ||||||
|  |  | ||||||
|     virtual ~Logger() noexcept(false); |  | ||||||
|  |  | ||||||
|     // protection for use after static destruction took place |     // protection for use after static destruction took place | ||||||
|     static bool fIsDestructed; |     static bool fIsDestructed; | ||||||
|     static struct DestructionHelper { ~DestructionHelper() { Logger::fIsDestructed = true; }} fDestructionHelper; |     static struct DestructionHelper { ~DestructionHelper() { Logger::fIsDestructed = true; }} fDestructionHelper; | ||||||
|  |  | ||||||
|  |     static bool constexpr SuppressSeverity(Severity sev) | ||||||
|  |     { | ||||||
|  |         return sev < Severity::FAIR_MIN_SEVERITY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|   private: |   private: | ||||||
|     LogMetaData fMetaData; |     LogMetaData fInfos; | ||||||
|  |  | ||||||
|     std::ostringstream fContent; |     std::ostringstream fContent; | ||||||
|     std::ostringstream fColorOut; |     fmt::memory_buffer fColorPrefix; | ||||||
|     std::ostringstream fBWOut; |     fmt::memory_buffer fBWPrefix; | ||||||
|     static const std::string fProcessName; |     static const std::string fProcessName; | ||||||
|     static bool fColored; |     static bool fColored; | ||||||
|     static std::fstream fFileStream; |     static std::fstream fFileStream; | ||||||
| @@ -365,31 +364,93 @@ class Logger | |||||||
|  |  | ||||||
|     static void UpdateMinSeverity(); |     static void UpdateMinSeverity(); | ||||||
|  |  | ||||||
|  |     void FillTimeInfos(); | ||||||
|  |     bool fTimeCalculated; | ||||||
|  |  | ||||||
|     static std::map<Verbosity, VerbositySpec> fVerbosities; |     static std::map<Verbosity, VerbositySpec> fVerbosities; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | inline std::ostream& operator<<(std::ostream& os, const Severity& s) { return os << Logger::SeverityName(s); } | ||||||
|  | inline std::ostream& operator<<(std::ostream& os, const Verbosity& v) { return os << Logger::VerbosityName(v); } | ||||||
|  |  | ||||||
| } // namespace fair | } // namespace fair | ||||||
|  |  | ||||||
| #define IMP_CONVERTTOSTRING(s) # s | #define IMP_CONVERTTOSTRING(s) # s | ||||||
| #define CONVERTTOSTRING(s) IMP_CONVERTTOSTRING(s) | #define CONVERTTOSTRING(s) IMP_CONVERTTOSTRING(s) | ||||||
|  |  | ||||||
| #ifdef FAIRLOGGER_USE_BOOST_PRETTY_FUNCTION | #ifdef FAIRLOGGER_USE_BOOST_PRETTY_FUNCTION | ||||||
| #define LOG(severity) \ | #define MSG_ORIGIN __FILE__, CONVERTTOSTRING(__LINE__), static_cast<const char*>(BOOST_CURRENT_FUNCTION) | ||||||
|     for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(fair::Severity::severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ |  | ||||||
|         fair::Logger(fair::Severity::severity, __FILE__, CONVERTTOSTRING(__LINE__), BOOST_CURRENT_FUNCTION).Log() |  | ||||||
| #else | #else | ||||||
| #define LOG(severity) \ | #define MSG_ORIGIN __FILE__, CONVERTTOSTRING(__LINE__), static_cast<const char*>(__FUNCTION__) | ||||||
|     for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(fair::Severity::severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ |  | ||||||
|         fair::Logger(fair::Severity::severity, __FILE__, CONVERTTOSTRING(__LINE__), __FUNCTION__).Log() |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| // with custom file, line, function | // allow user of this header file to prevent definition of the LOG macro, by defining FAIR_NO_LOG before including this header | ||||||
| #define LOGD(severity, file, line, function) \ | #ifndef FAIR_NO_LOG | ||||||
|     for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ | #undef LOG | ||||||
|         fair::Logger(severity, file, line, function).Log() | #define LOG FAIR_LOG | ||||||
|  | #endif | ||||||
|  | // allow user of this header file to prevent definition of the LOGV macro, by defining FAIR_NO_LOGV before including this header | ||||||
|  | #ifndef FAIR_NO_LOGV | ||||||
|  | #undef LOGV | ||||||
|  | #define LOGV FAIR_LOGV | ||||||
|  | #endif | ||||||
|  | // allow user of this header file to prevent definition of the LOGF macro, by defining FAIR_NO_LOGF before including this header | ||||||
|  | #ifndef FAIR_NO_LOGF | ||||||
|  | #undef LOGF | ||||||
|  | #define LOGF FAIR_LOGF | ||||||
|  | #endif | ||||||
|  | // allow user of this header file to prevent definition of the LOGP macro, by defining FAIR_NO_LOGP before including this header | ||||||
|  | #ifndef FAIR_NO_LOGP | ||||||
|  | #undef LOGP | ||||||
|  | #define LOGP FAIR_LOGP | ||||||
|  | #endif | ||||||
|  | // allow user of this header file to prevent definition of the LOGN macro, by defining FAIR_NO_LOGN before including this header | ||||||
|  | #ifndef FAIR_NO_LOGN | ||||||
|  | #undef LOGN | ||||||
|  | #define LOGN FAIR_LOGN | ||||||
|  | #endif | ||||||
|  | // allow user of this header file to prevent definition of the LOGD macro, by defining FAIR_NO_LOGD before including this header | ||||||
|  | #ifndef FAIR_NO_LOGD | ||||||
|  | #undef LOGD | ||||||
|  | #define LOGD FAIR_LOGD | ||||||
|  | #endif | ||||||
|  | // allow user of this header file to prevent definition of the LOG_IF macro, by defining FAIR_NO_LOG_IF before including this header | ||||||
|  | #ifndef FAIR_NO_LOG_IF | ||||||
|  | #undef LOG_IF | ||||||
|  | #define LOG_IF FAIR_LOG_IF | ||||||
|  | #endif | ||||||
|  |  | ||||||
| #define LOG_IF(severity, condition) \ | // Log line if the provided severity is below or equals the configured one | ||||||
|     for (bool fairLOggerunLikelyvariable2 = false; condition && !fairLOggerunLikelyvariable2; fairLOggerunLikelyvariable2 = true) \ | #define FAIR_LOG(severity) \ | ||||||
|         LOG(severity) |     for (bool fairLOggerunLikelyvariable3 = false; !fair::Logger::SuppressSeverity(fair::Severity::severity) && !fairLOggerunLikelyvariable3; fairLOggerunLikelyvariable3 = true) \ | ||||||
|  |         for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(fair::Severity::severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ | ||||||
|  |             fair::Logger(fair::Severity::severity, MSG_ORIGIN) | ||||||
|  |  | ||||||
|  | // Log line with the given verbosity if the provided severity is below or equals the configured one | ||||||
|  | #define FAIR_LOGV(severity, verbosity) \ | ||||||
|  |     for (bool fairLOggerunLikelyvariable3 = false; !fair::Logger::SuppressSeverity(fair::Severity::severity) && !fairLOggerunLikelyvariable3; fairLOggerunLikelyvariable3 = true) \ | ||||||
|  |         for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(fair::Severity::severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ | ||||||
|  |             fair::Logger(fair::Severity::severity, fair::Verbosity::verbosity, MSG_ORIGIN) | ||||||
|  |  | ||||||
|  | // Log with fmt- or printf-like formatting | ||||||
|  | #define FAIR_LOGP(severity, ...) LOG(severity) << fmt::format(__VA_ARGS__) | ||||||
|  | #define FAIR_LOGF(severity, ...) LOG(severity) << fmt::sprintf(__VA_ARGS__) | ||||||
|  |  | ||||||
|  | // Log an empty line | ||||||
|  | #define FAIR_LOGN(severity) \ | ||||||
|  |     for (bool fairLOggerunLikelyvariable3 = false; !fair::Logger::SuppressSeverity(fair::Severity::severity) && !fairLOggerunLikelyvariable3; fairLOggerunLikelyvariable3 = true) \ | ||||||
|  |         for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(fair::Severity::severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ | ||||||
|  |             fair::Logger(fair::Severity::severity, fair::Verbosity::verylow, MSG_ORIGIN).LogEmptyLine() | ||||||
|  |  | ||||||
|  | // Log with custom file, line, function | ||||||
|  | #define FAIR_LOGD(severity, file, line, f) \ | ||||||
|  |     for (bool fairLOggerunLikelyvariable3 = false; !fair::Logger::SuppressSeverity(severity) && !fairLOggerunLikelyvariable3; fairLOggerunLikelyvariable3 = true) \ | ||||||
|  |         for (bool fairLOggerunLikelyvariable = false; fair::Logger::Logging(severity) && !fairLOggerunLikelyvariable; fairLOggerunLikelyvariable = true) \ | ||||||
|  |             fair::Logger(severity, file, line, f) | ||||||
|  |  | ||||||
|  | #define FAIR_LOG_IF(severity, condition) \ | ||||||
|  |     for (bool fairLOggerunLikelyvariable4 = false; !fair::Logger::SuppressSeverity(fair::Severity::severity) && !fairLOggerunLikelyvariable4; fairLOggerunLikelyvariable4 = true) \ | ||||||
|  |         for (bool fairLOggerunLikelyvariable2 = false; condition && !fairLOggerunLikelyvariable2; fairLOggerunLikelyvariable2 = true) \ | ||||||
|  |             LOG(severity) | ||||||
|  |  | ||||||
| #endif // FAIR_LOGGER_H | #endif // FAIR_LOGGER_H | ||||||
|   | |||||||
							
								
								
									
										452
									
								
								logger/bundled/fmt/chrono.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										452
									
								
								logger/bundled/fmt/chrono.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,452 @@ | |||||||
|  | // Formatting library for C++ - chrono support | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - present, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_CHRONO_H_ | ||||||
|  | #define FMT_CHRONO_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  | #include "locale.h" | ||||||
|  |  | ||||||
|  | #include <chrono> | ||||||
|  | #include <ctime> | ||||||
|  | #include <locale> | ||||||
|  | #include <sstream> | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | namespace internal{ | ||||||
|  |  | ||||||
|  | enum class numeric_system { | ||||||
|  |   standard, | ||||||
|  |   // Alternative numeric system, e.g. 十二 instead of 12 in ja_JP locale. | ||||||
|  |   alternative | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Parses a put_time-like format string and invokes handler actions. | ||||||
|  | template <typename Char, typename Handler> | ||||||
|  | FMT_CONSTEXPR const Char *parse_chrono_format( | ||||||
|  |     const Char *begin, const Char *end, Handler &&handler) { | ||||||
|  |   auto ptr = begin; | ||||||
|  |   while (ptr != end) { | ||||||
|  |     auto c = *ptr; | ||||||
|  |     if (c == '}') break; | ||||||
|  |     if (c != '%') { | ||||||
|  |       ++ptr; | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     if (begin != ptr) | ||||||
|  |       handler.on_text(begin, ptr); | ||||||
|  |     ++ptr; // consume '%' | ||||||
|  |     if (ptr == end) | ||||||
|  |       throw format_error("invalid format"); | ||||||
|  |     c = *ptr++; | ||||||
|  |     switch (c) { | ||||||
|  |     case '%': | ||||||
|  |       handler.on_text(ptr - 1, ptr); | ||||||
|  |       break; | ||||||
|  |     case 'n': { | ||||||
|  |       const char newline[] = "\n"; | ||||||
|  |       handler.on_text(newline, newline + 1); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case 't': { | ||||||
|  |       const char tab[] = "\t"; | ||||||
|  |       handler.on_text(tab, tab + 1); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     // Day of the week: | ||||||
|  |     case 'a': | ||||||
|  |       handler.on_abbr_weekday(); | ||||||
|  |       break; | ||||||
|  |     case 'A': | ||||||
|  |       handler.on_full_weekday(); | ||||||
|  |       break; | ||||||
|  |     case 'w': | ||||||
|  |       handler.on_dec0_weekday(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'u': | ||||||
|  |       handler.on_dec1_weekday(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     // Month: | ||||||
|  |     case 'b': | ||||||
|  |       handler.on_abbr_month(); | ||||||
|  |       break; | ||||||
|  |     case 'B': | ||||||
|  |       handler.on_full_month(); | ||||||
|  |       break; | ||||||
|  |     // Hour, minute, second: | ||||||
|  |     case 'H': | ||||||
|  |       handler.on_24_hour(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'I': | ||||||
|  |       handler.on_12_hour(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'M': | ||||||
|  |       handler.on_minute(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'S': | ||||||
|  |       handler.on_second(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     // Other: | ||||||
|  |     case 'c': | ||||||
|  |       handler.on_datetime(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'x': | ||||||
|  |       handler.on_loc_date(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'X': | ||||||
|  |       handler.on_loc_time(numeric_system::standard); | ||||||
|  |       break; | ||||||
|  |     case 'D': | ||||||
|  |       handler.on_us_date(); | ||||||
|  |       break; | ||||||
|  |     case 'F': | ||||||
|  |       handler.on_iso_date(); | ||||||
|  |       break; | ||||||
|  |     case 'r': | ||||||
|  |       handler.on_12_hour_time(); | ||||||
|  |       break; | ||||||
|  |     case 'R': | ||||||
|  |       handler.on_24_hour_time(); | ||||||
|  |       break; | ||||||
|  |     case 'T': | ||||||
|  |       handler.on_iso_time(); | ||||||
|  |       break; | ||||||
|  |     case 'p': | ||||||
|  |       handler.on_am_pm(); | ||||||
|  |       break; | ||||||
|  |     case 'z': | ||||||
|  |       handler.on_utc_offset(); | ||||||
|  |       break; | ||||||
|  |     case 'Z': | ||||||
|  |       handler.on_tz_name(); | ||||||
|  |       break; | ||||||
|  |     // Alternative representation: | ||||||
|  |     case 'E': { | ||||||
|  |       if (ptr == end) | ||||||
|  |         throw format_error("invalid format"); | ||||||
|  |       c = *ptr++; | ||||||
|  |       switch (c) { | ||||||
|  |       case 'c': | ||||||
|  |         handler.on_datetime(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'x': | ||||||
|  |         handler.on_loc_date(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'X': | ||||||
|  |         handler.on_loc_time(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         throw format_error("invalid format"); | ||||||
|  |       } | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case 'O': | ||||||
|  |       if (ptr == end) | ||||||
|  |         throw format_error("invalid format"); | ||||||
|  |       c = *ptr++; | ||||||
|  |       switch (c) { | ||||||
|  |       case 'w': | ||||||
|  |         handler.on_dec0_weekday(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'u': | ||||||
|  |         handler.on_dec1_weekday(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'H': | ||||||
|  |         handler.on_24_hour(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'I': | ||||||
|  |         handler.on_12_hour(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'M': | ||||||
|  |         handler.on_minute(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       case 'S': | ||||||
|  |         handler.on_second(numeric_system::alternative); | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         throw format_error("invalid format"); | ||||||
|  |       } | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       throw format_error("invalid format"); | ||||||
|  |     } | ||||||
|  |     begin = ptr; | ||||||
|  |   } | ||||||
|  |   if (begin != ptr) | ||||||
|  |     handler.on_text(begin, ptr); | ||||||
|  |   return ptr; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct chrono_format_checker { | ||||||
|  |   void report_no_date() { throw format_error("no date"); } | ||||||
|  |  | ||||||
|  |   template <typename Char> | ||||||
|  |   void on_text(const Char *, const Char *) {} | ||||||
|  |   void on_abbr_weekday() { report_no_date(); } | ||||||
|  |   void on_full_weekday() { report_no_date(); } | ||||||
|  |   void on_dec0_weekday(numeric_system) { report_no_date(); } | ||||||
|  |   void on_dec1_weekday(numeric_system) { report_no_date(); } | ||||||
|  |   void on_abbr_month() { report_no_date(); } | ||||||
|  |   void on_full_month() { report_no_date(); } | ||||||
|  |   void on_24_hour(numeric_system) {} | ||||||
|  |   void on_12_hour(numeric_system) {} | ||||||
|  |   void on_minute(numeric_system) {} | ||||||
|  |   void on_second(numeric_system) {} | ||||||
|  |   void on_datetime(numeric_system) { report_no_date(); } | ||||||
|  |   void on_loc_date(numeric_system) { report_no_date(); } | ||||||
|  |   void on_loc_time(numeric_system) { report_no_date(); } | ||||||
|  |   void on_us_date() { report_no_date(); } | ||||||
|  |   void on_iso_date() { report_no_date(); } | ||||||
|  |   void on_12_hour_time() {} | ||||||
|  |   void on_24_hour_time() {} | ||||||
|  |   void on_iso_time() {} | ||||||
|  |   void on_am_pm() {} | ||||||
|  |   void on_utc_offset() { report_no_date(); } | ||||||
|  |   void on_tz_name() { report_no_date(); } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Int> | ||||||
|  | inline int to_int(Int value) { | ||||||
|  |   FMT_ASSERT(value >= (std::numeric_limits<int>::min)() && | ||||||
|  |              value <= (std::numeric_limits<int>::max)(), "invalid value"); | ||||||
|  |   return static_cast<int>(value); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename FormatContext, typename OutputIt> | ||||||
|  | struct chrono_formatter { | ||||||
|  |   FormatContext &context; | ||||||
|  |   OutputIt out; | ||||||
|  |   std::chrono::seconds s; | ||||||
|  |   std::chrono::milliseconds ms; | ||||||
|  |  | ||||||
|  |   typedef typename FormatContext::char_type char_type; | ||||||
|  |  | ||||||
|  |   explicit chrono_formatter(FormatContext &ctx, OutputIt o) | ||||||
|  |     : context(ctx), out(o) {} | ||||||
|  |  | ||||||
|  |   int hour() const { return to_int((s.count() / 3600) % 24); } | ||||||
|  |  | ||||||
|  |   int hour12() const { | ||||||
|  |     auto hour = to_int((s.count() / 3600) % 12); | ||||||
|  |     return hour > 0 ? hour : 12; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   int minute() const { return to_int((s.count() / 60) % 60); } | ||||||
|  |   int second() const { return to_int(s.count() % 60); } | ||||||
|  |  | ||||||
|  |   std::tm time() const { | ||||||
|  |     auto time = std::tm(); | ||||||
|  |     time.tm_hour = hour(); | ||||||
|  |     time.tm_min = minute(); | ||||||
|  |     time.tm_sec = second(); | ||||||
|  |     return time; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void write(int value, int width) { | ||||||
|  |     typedef typename int_traits<int>::main_type main_type; | ||||||
|  |     main_type n = to_unsigned(value); | ||||||
|  |     int num_digits = internal::count_digits(n); | ||||||
|  |     if (width > num_digits) | ||||||
|  |       out = std::fill_n(out, width - num_digits, '0'); | ||||||
|  |     out = format_decimal<char_type>(out, n, num_digits); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void format_localized(const tm &time, const char *format) { | ||||||
|  |     auto locale = context.locale().template get<std::locale>(); | ||||||
|  |     auto &facet = std::use_facet<std::time_put<char_type>>(locale); | ||||||
|  |     std::basic_ostringstream<char_type> os; | ||||||
|  |     os.imbue(locale); | ||||||
|  |     facet.put(os, os, ' ', &time, format, format + std::strlen(format)); | ||||||
|  |     auto str = os.str(); | ||||||
|  |     std::copy(str.begin(), str.end(), out); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_text(const char_type *begin, const char_type *end) { | ||||||
|  |     std::copy(begin, end, out); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // These are not implemented because durations don't have date information. | ||||||
|  |   void on_abbr_weekday() {} | ||||||
|  |   void on_full_weekday() {} | ||||||
|  |   void on_dec0_weekday(numeric_system) {} | ||||||
|  |   void on_dec1_weekday(numeric_system) {} | ||||||
|  |   void on_abbr_month() {} | ||||||
|  |   void on_full_month() {} | ||||||
|  |   void on_datetime(numeric_system) {} | ||||||
|  |   void on_loc_date(numeric_system) {} | ||||||
|  |   void on_loc_time(numeric_system) {} | ||||||
|  |   void on_us_date() {} | ||||||
|  |   void on_iso_date() {} | ||||||
|  |   void on_utc_offset() {} | ||||||
|  |   void on_tz_name() {} | ||||||
|  |  | ||||||
|  |   void on_24_hour(numeric_system ns) { | ||||||
|  |     if (ns == numeric_system::standard) | ||||||
|  |       return write(hour(), 2); | ||||||
|  |     auto time = tm(); | ||||||
|  |     time.tm_hour = hour(); | ||||||
|  |     format_localized(time, "%OH"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_12_hour(numeric_system ns) { | ||||||
|  |     if (ns == numeric_system::standard) | ||||||
|  |       return write(hour12(), 2); | ||||||
|  |     auto time = tm(); | ||||||
|  |     time.tm_hour = hour(); | ||||||
|  |     format_localized(time, "%OI"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_minute(numeric_system ns) { | ||||||
|  |     if (ns == numeric_system::standard) | ||||||
|  |       return write(minute(), 2); | ||||||
|  |     auto time = tm(); | ||||||
|  |     time.tm_min = minute(); | ||||||
|  |     format_localized(time, "%OM"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_second(numeric_system ns) { | ||||||
|  |     if (ns == numeric_system::standard) { | ||||||
|  |       write(second(), 2); | ||||||
|  |       if (ms != std::chrono::milliseconds(0)) { | ||||||
|  |         *out++ = '.'; | ||||||
|  |         write(to_int(ms.count()), 3); | ||||||
|  |       } | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     auto time = tm(); | ||||||
|  |     time.tm_sec = second(); | ||||||
|  |     format_localized(time, "%OS"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_12_hour_time() { format_localized(time(), "%r"); } | ||||||
|  |  | ||||||
|  |   void on_24_hour_time() { | ||||||
|  |     write(hour(), 2); | ||||||
|  |     *out++ = ':'; | ||||||
|  |     write(minute(), 2); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_iso_time() { | ||||||
|  |     on_24_hour_time(); | ||||||
|  |     *out++ = ':'; | ||||||
|  |     write(second(), 2); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void on_am_pm() { format_localized(time(), "%p"); } | ||||||
|  | }; | ||||||
|  | }  // namespace internal | ||||||
|  |  | ||||||
|  | template <typename Period> FMT_CONSTEXPR const char *get_units() { | ||||||
|  |   return FMT_NULL; | ||||||
|  | } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::atto>() { return "as"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::femto>() { return "fs"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::pico>() { return "ps"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::nano>() { return "ns"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::micro>() { return "µs"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::milli>() { return "ms"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::centi>() { return "cs"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::deci>() { return "ds"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::ratio<1>>() { return "s"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::deca>() { return "das"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::hecto>() { return "hs"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::kilo>() { return "ks"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::mega>() { return "Ms"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::giga>() { return "Gs"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::tera>() { return "Ts"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::peta>() { return "Ps"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::exa>() { return "Es"; } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::ratio<60>>() { | ||||||
|  |   return "m"; | ||||||
|  | } | ||||||
|  | template <> FMT_CONSTEXPR const char *get_units<std::ratio<3600>>() { | ||||||
|  |   return "h"; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Rep, typename Period, typename Char> | ||||||
|  | struct formatter<std::chrono::duration<Rep, Period>, Char> { | ||||||
|  |  private: | ||||||
|  |   align_spec spec; | ||||||
|  |   internal::arg_ref<Char> width_ref; | ||||||
|  |   mutable basic_string_view<Char> format_str; | ||||||
|  |   typedef std::chrono::duration<Rep, Period> duration; | ||||||
|  |  | ||||||
|  |   struct spec_handler { | ||||||
|  |     formatter &f; | ||||||
|  |     basic_parse_context<Char> &context; | ||||||
|  |  | ||||||
|  |     typedef internal::arg_ref<Char> arg_ref_type; | ||||||
|  |  | ||||||
|  |     template <typename Id> | ||||||
|  |     FMT_CONSTEXPR arg_ref_type make_arg_ref(Id arg_id) { | ||||||
|  |       context.check_arg_id(arg_id); | ||||||
|  |       return arg_ref_type(arg_id); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     FMT_CONSTEXPR arg_ref_type make_arg_ref(internal::auto_id) { | ||||||
|  |       return arg_ref_type(context.next_arg_id()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void on_error(const char *msg) { throw format_error(msg); } | ||||||
|  |     void on_fill(Char fill) { f.spec.fill_ = fill; } | ||||||
|  |     void on_align(alignment align) { f.spec.align_ = align; } | ||||||
|  |     void on_width(unsigned width) { f.spec.width_ = width; } | ||||||
|  |  | ||||||
|  |     template <typename Id> | ||||||
|  |     void on_dynamic_width(Id arg_id) { | ||||||
|  |       f.width_ref = make_arg_ref(arg_id); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   formatter() : spec() {} | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR auto parse(basic_parse_context<Char> &ctx) | ||||||
|  |       -> decltype(ctx.begin()) { | ||||||
|  |     auto begin = ctx.begin(), end = ctx.end(); | ||||||
|  |     if (begin == end) return begin; | ||||||
|  |     spec_handler handler{*this, ctx}; | ||||||
|  |     begin = internal::parse_align(begin, end, handler); | ||||||
|  |     if (begin == end) return begin; | ||||||
|  |     begin = internal::parse_width(begin, end, handler); | ||||||
|  |     end = parse_chrono_format(begin, end, internal::chrono_format_checker()); | ||||||
|  |     format_str = basic_string_view<Char>(&*begin, internal::to_unsigned(end - begin)); | ||||||
|  |     return end; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename FormatContext> | ||||||
|  |   auto format(const duration &d, FormatContext &ctx) | ||||||
|  |       -> decltype(ctx.out()) { | ||||||
|  |     auto begin = format_str.begin(), end = format_str.end(); | ||||||
|  |     memory_buffer buf; | ||||||
|  |     typedef output_range<decltype(ctx.out()), Char> range; | ||||||
|  |     basic_writer<range> w(range(ctx.out())); | ||||||
|  |     if (begin == end || *begin == '}') { | ||||||
|  |       if (const char *unit = get_units<Period>()) | ||||||
|  |         format_to(buf, "{}{}", d.count(), unit); | ||||||
|  |       else if (Period::den == 1) | ||||||
|  |         format_to(buf, "{}[{}]s", d.count(), Period::num); | ||||||
|  |       else | ||||||
|  |         format_to(buf, "{}[{}/{}]s", d.count(), Period::num, Period::den); | ||||||
|  |       internal::handle_dynamic_spec<internal::width_checker>( | ||||||
|  |         spec.width_, width_ref, ctx); | ||||||
|  |     } else { | ||||||
|  |       auto out = std::back_inserter(buf); | ||||||
|  |       internal::chrono_formatter<FormatContext, decltype(out)> f(ctx, out); | ||||||
|  |       f.s = std::chrono::duration_cast<std::chrono::seconds>(d); | ||||||
|  |       f.ms = std::chrono::duration_cast<std::chrono::milliseconds>(d - f.s); | ||||||
|  |       parse_chrono_format(begin, end, f); | ||||||
|  |     } | ||||||
|  |     w.write(buf.data(), buf.size(), spec); | ||||||
|  |     return w.out(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_CHRONO_H_ | ||||||
							
								
								
									
										577
									
								
								logger/bundled/fmt/color.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										577
									
								
								logger/bundled/fmt/color.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,577 @@ | |||||||
|  | // Formatting library for C++ - color support | ||||||
|  | // | ||||||
|  | // Copyright (c) 2018 - present, Victor Zverovich and fmt contributors | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_COLOR_H_ | ||||||
|  | #define FMT_COLOR_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | #ifdef FMT_DEPRECATED_COLORS | ||||||
|  |  | ||||||
|  | // color and (v)print_colored are deprecated. | ||||||
|  | enum color { black, red, green, yellow, blue, magenta, cyan, white }; | ||||||
|  | FMT_API void vprint_colored(color c, string_view format, format_args args); | ||||||
|  | FMT_API void vprint_colored(color c, wstring_view format, wformat_args args); | ||||||
|  | template <typename... Args> | ||||||
|  | inline void print_colored(color c, string_view format_str, | ||||||
|  |                           const Args & ... args) { | ||||||
|  |   vprint_colored(c, format_str, make_format_args(args...)); | ||||||
|  | } | ||||||
|  | template <typename... Args> | ||||||
|  | inline void print_colored(color c, wstring_view format_str, | ||||||
|  |                           const Args & ... args) { | ||||||
|  |   vprint_colored(c, format_str, make_format_args<wformat_context>(args...)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline void vprint_colored(color c, string_view format, format_args args) { | ||||||
|  |   char escape[] = "\x1b[30m"; | ||||||
|  |   escape[3] = static_cast<char>('0' + c); | ||||||
|  |   std::fputs(escape, stdout); | ||||||
|  |   vprint(format, args); | ||||||
|  |   std::fputs(internal::data::RESET_COLOR, stdout); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline void vprint_colored(color c, wstring_view format, wformat_args args) { | ||||||
|  |   wchar_t escape[] = L"\x1b[30m"; | ||||||
|  |   escape[3] = static_cast<wchar_t>('0' + c); | ||||||
|  |   std::fputws(escape, stdout); | ||||||
|  |   vprint(format, args); | ||||||
|  |   std::fputws(internal::data::WRESET_COLOR, stdout); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #else | ||||||
|  |  | ||||||
|  | enum class color : uint32_t { | ||||||
|  |   alice_blue              = 0xF0F8FF, // rgb(240,248,255) | ||||||
|  |   antique_white           = 0xFAEBD7, // rgb(250,235,215) | ||||||
|  |   aqua                    = 0x00FFFF, // rgb(0,255,255) | ||||||
|  |   aquamarine              = 0x7FFFD4, // rgb(127,255,212) | ||||||
|  |   azure                   = 0xF0FFFF, // rgb(240,255,255) | ||||||
|  |   beige                   = 0xF5F5DC, // rgb(245,245,220) | ||||||
|  |   bisque                  = 0xFFE4C4, // rgb(255,228,196) | ||||||
|  |   black                   = 0x000000, // rgb(0,0,0) | ||||||
|  |   blanched_almond         = 0xFFEBCD, // rgb(255,235,205) | ||||||
|  |   blue                    = 0x0000FF, // rgb(0,0,255) | ||||||
|  |   blue_violet             = 0x8A2BE2, // rgb(138,43,226) | ||||||
|  |   brown                   = 0xA52A2A, // rgb(165,42,42) | ||||||
|  |   burly_wood              = 0xDEB887, // rgb(222,184,135) | ||||||
|  |   cadet_blue              = 0x5F9EA0, // rgb(95,158,160) | ||||||
|  |   chartreuse              = 0x7FFF00, // rgb(127,255,0) | ||||||
|  |   chocolate               = 0xD2691E, // rgb(210,105,30) | ||||||
|  |   coral                   = 0xFF7F50, // rgb(255,127,80) | ||||||
|  |   cornflower_blue         = 0x6495ED, // rgb(100,149,237) | ||||||
|  |   cornsilk                = 0xFFF8DC, // rgb(255,248,220) | ||||||
|  |   crimson                 = 0xDC143C, // rgb(220,20,60) | ||||||
|  |   cyan                    = 0x00FFFF, // rgb(0,255,255) | ||||||
|  |   dark_blue               = 0x00008B, // rgb(0,0,139) | ||||||
|  |   dark_cyan               = 0x008B8B, // rgb(0,139,139) | ||||||
|  |   dark_golden_rod         = 0xB8860B, // rgb(184,134,11) | ||||||
|  |   dark_gray               = 0xA9A9A9, // rgb(169,169,169) | ||||||
|  |   dark_green              = 0x006400, // rgb(0,100,0) | ||||||
|  |   dark_khaki              = 0xBDB76B, // rgb(189,183,107) | ||||||
|  |   dark_magenta            = 0x8B008B, // rgb(139,0,139) | ||||||
|  |   dark_olive_green        = 0x556B2F, // rgb(85,107,47) | ||||||
|  |   dark_orange             = 0xFF8C00, // rgb(255,140,0) | ||||||
|  |   dark_orchid             = 0x9932CC, // rgb(153,50,204) | ||||||
|  |   dark_red                = 0x8B0000, // rgb(139,0,0) | ||||||
|  |   dark_salmon             = 0xE9967A, // rgb(233,150,122) | ||||||
|  |   dark_sea_green          = 0x8FBC8F, // rgb(143,188,143) | ||||||
|  |   dark_slate_blue         = 0x483D8B, // rgb(72,61,139) | ||||||
|  |   dark_slate_gray         = 0x2F4F4F, // rgb(47,79,79) | ||||||
|  |   dark_turquoise          = 0x00CED1, // rgb(0,206,209) | ||||||
|  |   dark_violet             = 0x9400D3, // rgb(148,0,211) | ||||||
|  |   deep_pink               = 0xFF1493, // rgb(255,20,147) | ||||||
|  |   deep_sky_blue           = 0x00BFFF, // rgb(0,191,255) | ||||||
|  |   dim_gray                = 0x696969, // rgb(105,105,105) | ||||||
|  |   dodger_blue             = 0x1E90FF, // rgb(30,144,255) | ||||||
|  |   fire_brick              = 0xB22222, // rgb(178,34,34) | ||||||
|  |   floral_white            = 0xFFFAF0, // rgb(255,250,240) | ||||||
|  |   forest_green            = 0x228B22, // rgb(34,139,34) | ||||||
|  |   fuchsia                 = 0xFF00FF, // rgb(255,0,255) | ||||||
|  |   gainsboro               = 0xDCDCDC, // rgb(220,220,220) | ||||||
|  |   ghost_white             = 0xF8F8FF, // rgb(248,248,255) | ||||||
|  |   gold                    = 0xFFD700, // rgb(255,215,0) | ||||||
|  |   golden_rod              = 0xDAA520, // rgb(218,165,32) | ||||||
|  |   gray                    = 0x808080, // rgb(128,128,128) | ||||||
|  |   green                   = 0x008000, // rgb(0,128,0) | ||||||
|  |   green_yellow            = 0xADFF2F, // rgb(173,255,47) | ||||||
|  |   honey_dew               = 0xF0FFF0, // rgb(240,255,240) | ||||||
|  |   hot_pink                = 0xFF69B4, // rgb(255,105,180) | ||||||
|  |   indian_red              = 0xCD5C5C, // rgb(205,92,92) | ||||||
|  |   indigo                  = 0x4B0082, // rgb(75,0,130) | ||||||
|  |   ivory                   = 0xFFFFF0, // rgb(255,255,240) | ||||||
|  |   khaki                   = 0xF0E68C, // rgb(240,230,140) | ||||||
|  |   lavender                = 0xE6E6FA, // rgb(230,230,250) | ||||||
|  |   lavender_blush          = 0xFFF0F5, // rgb(255,240,245) | ||||||
|  |   lawn_green              = 0x7CFC00, // rgb(124,252,0) | ||||||
|  |   lemon_chiffon           = 0xFFFACD, // rgb(255,250,205) | ||||||
|  |   light_blue              = 0xADD8E6, // rgb(173,216,230) | ||||||
|  |   light_coral             = 0xF08080, // rgb(240,128,128) | ||||||
|  |   light_cyan              = 0xE0FFFF, // rgb(224,255,255) | ||||||
|  |   light_golden_rod_yellow = 0xFAFAD2, // rgb(250,250,210) | ||||||
|  |   light_gray              = 0xD3D3D3, // rgb(211,211,211) | ||||||
|  |   light_green             = 0x90EE90, // rgb(144,238,144) | ||||||
|  |   light_pink              = 0xFFB6C1, // rgb(255,182,193) | ||||||
|  |   light_salmon            = 0xFFA07A, // rgb(255,160,122) | ||||||
|  |   light_sea_green         = 0x20B2AA, // rgb(32,178,170) | ||||||
|  |   light_sky_blue          = 0x87CEFA, // rgb(135,206,250) | ||||||
|  |   light_slate_gray        = 0x778899, // rgb(119,136,153) | ||||||
|  |   light_steel_blue        = 0xB0C4DE, // rgb(176,196,222) | ||||||
|  |   light_yellow            = 0xFFFFE0, // rgb(255,255,224) | ||||||
|  |   lime                    = 0x00FF00, // rgb(0,255,0) | ||||||
|  |   lime_green              = 0x32CD32, // rgb(50,205,50) | ||||||
|  |   linen                   = 0xFAF0E6, // rgb(250,240,230) | ||||||
|  |   magenta                 = 0xFF00FF, // rgb(255,0,255) | ||||||
|  |   maroon                  = 0x800000, // rgb(128,0,0) | ||||||
|  |   medium_aquamarine       = 0x66CDAA, // rgb(102,205,170) | ||||||
|  |   medium_blue             = 0x0000CD, // rgb(0,0,205) | ||||||
|  |   medium_orchid           = 0xBA55D3, // rgb(186,85,211) | ||||||
|  |   medium_purple           = 0x9370DB, // rgb(147,112,219) | ||||||
|  |   medium_sea_green        = 0x3CB371, // rgb(60,179,113) | ||||||
|  |   medium_slate_blue       = 0x7B68EE, // rgb(123,104,238) | ||||||
|  |   medium_spring_green     = 0x00FA9A, // rgb(0,250,154) | ||||||
|  |   medium_turquoise        = 0x48D1CC, // rgb(72,209,204) | ||||||
|  |   medium_violet_red       = 0xC71585, // rgb(199,21,133) | ||||||
|  |   midnight_blue           = 0x191970, // rgb(25,25,112) | ||||||
|  |   mint_cream              = 0xF5FFFA, // rgb(245,255,250) | ||||||
|  |   misty_rose              = 0xFFE4E1, // rgb(255,228,225) | ||||||
|  |   moccasin                = 0xFFE4B5, // rgb(255,228,181) | ||||||
|  |   navajo_white            = 0xFFDEAD, // rgb(255,222,173) | ||||||
|  |   navy                    = 0x000080, // rgb(0,0,128) | ||||||
|  |   old_lace                = 0xFDF5E6, // rgb(253,245,230) | ||||||
|  |   olive                   = 0x808000, // rgb(128,128,0) | ||||||
|  |   olive_drab              = 0x6B8E23, // rgb(107,142,35) | ||||||
|  |   orange                  = 0xFFA500, // rgb(255,165,0) | ||||||
|  |   orange_red              = 0xFF4500, // rgb(255,69,0) | ||||||
|  |   orchid                  = 0xDA70D6, // rgb(218,112,214) | ||||||
|  |   pale_golden_rod         = 0xEEE8AA, // rgb(238,232,170) | ||||||
|  |   pale_green              = 0x98FB98, // rgb(152,251,152) | ||||||
|  |   pale_turquoise          = 0xAFEEEE, // rgb(175,238,238) | ||||||
|  |   pale_violet_red         = 0xDB7093, // rgb(219,112,147) | ||||||
|  |   papaya_whip             = 0xFFEFD5, // rgb(255,239,213) | ||||||
|  |   peach_puff              = 0xFFDAB9, // rgb(255,218,185) | ||||||
|  |   peru                    = 0xCD853F, // rgb(205,133,63) | ||||||
|  |   pink                    = 0xFFC0CB, // rgb(255,192,203) | ||||||
|  |   plum                    = 0xDDA0DD, // rgb(221,160,221) | ||||||
|  |   powder_blue             = 0xB0E0E6, // rgb(176,224,230) | ||||||
|  |   purple                  = 0x800080, // rgb(128,0,128) | ||||||
|  |   rebecca_purple          = 0x663399, // rgb(102,51,153) | ||||||
|  |   red                     = 0xFF0000, // rgb(255,0,0) | ||||||
|  |   rosy_brown              = 0xBC8F8F, // rgb(188,143,143) | ||||||
|  |   royal_blue              = 0x4169E1, // rgb(65,105,225) | ||||||
|  |   saddle_brown            = 0x8B4513, // rgb(139,69,19) | ||||||
|  |   salmon                  = 0xFA8072, // rgb(250,128,114) | ||||||
|  |   sandy_brown             = 0xF4A460, // rgb(244,164,96) | ||||||
|  |   sea_green               = 0x2E8B57, // rgb(46,139,87) | ||||||
|  |   sea_shell               = 0xFFF5EE, // rgb(255,245,238) | ||||||
|  |   sienna                  = 0xA0522D, // rgb(160,82,45) | ||||||
|  |   silver                  = 0xC0C0C0, // rgb(192,192,192) | ||||||
|  |   sky_blue                = 0x87CEEB, // rgb(135,206,235) | ||||||
|  |   slate_blue              = 0x6A5ACD, // rgb(106,90,205) | ||||||
|  |   slate_gray              = 0x708090, // rgb(112,128,144) | ||||||
|  |   snow                    = 0xFFFAFA, // rgb(255,250,250) | ||||||
|  |   spring_green            = 0x00FF7F, // rgb(0,255,127) | ||||||
|  |   steel_blue              = 0x4682B4, // rgb(70,130,180) | ||||||
|  |   tan                     = 0xD2B48C, // rgb(210,180,140) | ||||||
|  |   teal                    = 0x008080, // rgb(0,128,128) | ||||||
|  |   thistle                 = 0xD8BFD8, // rgb(216,191,216) | ||||||
|  |   tomato                  = 0xFF6347, // rgb(255,99,71) | ||||||
|  |   turquoise               = 0x40E0D0, // rgb(64,224,208) | ||||||
|  |   violet                  = 0xEE82EE, // rgb(238,130,238) | ||||||
|  |   wheat                   = 0xF5DEB3, // rgb(245,222,179) | ||||||
|  |   white                   = 0xFFFFFF, // rgb(255,255,255) | ||||||
|  |   white_smoke             = 0xF5F5F5, // rgb(245,245,245) | ||||||
|  |   yellow                  = 0xFFFF00, // rgb(255,255,0) | ||||||
|  |   yellow_green            = 0x9ACD32  // rgb(154,205,50) | ||||||
|  | };  // enum class color | ||||||
|  |  | ||||||
|  | enum class terminal_color : uint8_t { | ||||||
|  |   black = 30, | ||||||
|  |   red, | ||||||
|  |   green, | ||||||
|  |   yellow, | ||||||
|  |   blue, | ||||||
|  |   magenta, | ||||||
|  |   cyan, | ||||||
|  |   white, | ||||||
|  |   bright_black = 90, | ||||||
|  |   bright_red, | ||||||
|  |   bright_green, | ||||||
|  |   bright_yellow, | ||||||
|  |   bright_blue, | ||||||
|  |   bright_magenta, | ||||||
|  |   bright_cyan, | ||||||
|  |   bright_white | ||||||
|  | };  // enum class terminal_color | ||||||
|  |  | ||||||
|  | enum class emphasis : uint8_t { | ||||||
|  |   bold = 1, | ||||||
|  |   italic = 1 << 1, | ||||||
|  |   underline = 1 << 2, | ||||||
|  |   strikethrough = 1 << 3 | ||||||
|  | };  // enum class emphasis | ||||||
|  |  | ||||||
|  | // rgb is a struct for red, green and blue colors. | ||||||
|  | // We use rgb as name because some editors will show it as color direct in the | ||||||
|  | // editor. | ||||||
|  | struct rgb { | ||||||
|  |   FMT_CONSTEXPR_DECL rgb() : r(0), g(0), b(0) {} | ||||||
|  |   FMT_CONSTEXPR_DECL rgb(uint8_t r_, uint8_t g_, uint8_t b_) | ||||||
|  |     : r(r_), g(g_), b(b_) {} | ||||||
|  |   FMT_CONSTEXPR_DECL rgb(uint32_t hex) | ||||||
|  |     : r((hex >> 16) & 0xFF), g((hex >> 8) & 0xFF), b((hex) & 0xFF) {} | ||||||
|  |   FMT_CONSTEXPR_DECL rgb(color hex) | ||||||
|  |     : r((uint32_t(hex) >> 16) & 0xFF), g((uint32_t(hex) >> 8) & 0xFF), | ||||||
|  |       b(uint32_t(hex) & 0xFF) {} | ||||||
|  |   uint8_t r; | ||||||
|  |   uint8_t g; | ||||||
|  |   uint8_t b; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | namespace internal { | ||||||
|  |  | ||||||
|  | // color is a struct of either a rgb color or a terminal color. | ||||||
|  | struct color_type { | ||||||
|  |   FMT_CONSTEXPR color_type() FMT_NOEXCEPT | ||||||
|  |     : is_rgb(), value{} {} | ||||||
|  |   FMT_CONSTEXPR color_type(color rgb_color) FMT_NOEXCEPT | ||||||
|  |     : is_rgb(true), value{} { | ||||||
|  |     value.rgb_color = static_cast<uint32_t>(rgb_color); | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR color_type(rgb rgb_color) FMT_NOEXCEPT | ||||||
|  |     : is_rgb(true), value{} { | ||||||
|  |     value.rgb_color = (static_cast<uint32_t>(rgb_color.r) << 16) | ||||||
|  |        | (static_cast<uint32_t>(rgb_color.g) << 8) | rgb_color.b; | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR color_type(terminal_color term_color) FMT_NOEXCEPT | ||||||
|  |     : is_rgb(), value{} { | ||||||
|  |     value.term_color = static_cast<uint8_t>(term_color); | ||||||
|  |   } | ||||||
|  |   bool is_rgb; | ||||||
|  |   union color_union { | ||||||
|  |     uint8_t term_color; | ||||||
|  |     uint32_t rgb_color; | ||||||
|  |   } value; | ||||||
|  | }; | ||||||
|  | } // namespace internal | ||||||
|  |  | ||||||
|  | // Experimental text formatting support. | ||||||
|  | class text_style { | ||||||
|  |  public: | ||||||
|  |   FMT_CONSTEXPR text_style(emphasis em = emphasis()) FMT_NOEXCEPT | ||||||
|  |       : set_foreground_color(), set_background_color(), ems(em) {} | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR text_style &operator|=(const text_style &rhs) { | ||||||
|  |     if (!set_foreground_color) { | ||||||
|  |       set_foreground_color = rhs.set_foreground_color; | ||||||
|  |       foreground_color = rhs.foreground_color; | ||||||
|  |     } else if (rhs.set_foreground_color) { | ||||||
|  |       if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb) | ||||||
|  |         throw format_error("can't OR a terminal color"); | ||||||
|  |       foreground_color.value.rgb_color |= rhs.foreground_color.value.rgb_color; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (!set_background_color) { | ||||||
|  |       set_background_color = rhs.set_background_color; | ||||||
|  |       background_color = rhs.background_color; | ||||||
|  |     } else if (rhs.set_background_color) { | ||||||
|  |       if (!background_color.is_rgb || !rhs.background_color.is_rgb) | ||||||
|  |         throw format_error("can't OR a terminal color"); | ||||||
|  |       background_color.value.rgb_color |= rhs.background_color.value.rgb_color; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ems = static_cast<emphasis>(static_cast<uint8_t>(ems) | | ||||||
|  |                                 static_cast<uint8_t>(rhs.ems)); | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   friend FMT_CONSTEXPR | ||||||
|  |   text_style operator|(text_style lhs, const text_style &rhs) { | ||||||
|  |     return lhs |= rhs; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR text_style &operator&=(const text_style &rhs) { | ||||||
|  |     if (!set_foreground_color) { | ||||||
|  |       set_foreground_color = rhs.set_foreground_color; | ||||||
|  |       foreground_color = rhs.foreground_color; | ||||||
|  |     } else if (rhs.set_foreground_color) { | ||||||
|  |       if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb) | ||||||
|  |         throw format_error("can't AND a terminal color"); | ||||||
|  |       foreground_color.value.rgb_color &= rhs.foreground_color.value.rgb_color; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (!set_background_color) { | ||||||
|  |       set_background_color = rhs.set_background_color; | ||||||
|  |       background_color = rhs.background_color; | ||||||
|  |     } else if (rhs.set_background_color) { | ||||||
|  |       if (!background_color.is_rgb || !rhs.background_color.is_rgb) | ||||||
|  |         throw format_error("can't AND a terminal color"); | ||||||
|  |       background_color.value.rgb_color &= rhs.background_color.value.rgb_color; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ems = static_cast<emphasis>(static_cast<uint8_t>(ems) & | ||||||
|  |                                 static_cast<uint8_t>(rhs.ems)); | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   friend FMT_CONSTEXPR | ||||||
|  |   text_style operator&(text_style lhs, const text_style &rhs) { | ||||||
|  |     return lhs &= rhs; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR bool has_foreground() const FMT_NOEXCEPT { | ||||||
|  |     return set_foreground_color; | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR bool has_background() const FMT_NOEXCEPT { | ||||||
|  |     return set_background_color; | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR bool has_emphasis() const FMT_NOEXCEPT { | ||||||
|  |     return static_cast<uint8_t>(ems) != 0; | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR internal::color_type get_foreground() const FMT_NOEXCEPT { | ||||||
|  |     assert(has_foreground() && "no foreground specified for this style"); | ||||||
|  |     return foreground_color; | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR internal::color_type get_background() const FMT_NOEXCEPT { | ||||||
|  |     assert(has_background() && "no background specified for this style"); | ||||||
|  |     return background_color; | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR emphasis get_emphasis() const FMT_NOEXCEPT { | ||||||
|  |     assert(has_emphasis() && "no emphasis specified for this style"); | ||||||
|  |     return ems; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |  FMT_CONSTEXPR text_style(bool is_foreground, | ||||||
|  |                           internal::color_type text_color) FMT_NOEXCEPT | ||||||
|  |      : set_foreground_color(), | ||||||
|  |        set_background_color(), | ||||||
|  |        ems() { | ||||||
|  |    if (is_foreground) { | ||||||
|  |      foreground_color = text_color; | ||||||
|  |      set_foreground_color = true; | ||||||
|  |    } else { | ||||||
|  |      background_color = text_color; | ||||||
|  |      set_background_color = true; | ||||||
|  |    } | ||||||
|  |  } | ||||||
|  |  | ||||||
|  |   friend FMT_CONSTEXPR_DECL text_style fg(internal::color_type foreground) | ||||||
|  |       FMT_NOEXCEPT; | ||||||
|  |   friend FMT_CONSTEXPR_DECL text_style bg(internal::color_type background) | ||||||
|  |       FMT_NOEXCEPT; | ||||||
|  |  | ||||||
|  |   internal::color_type foreground_color; | ||||||
|  |   internal::color_type background_color; | ||||||
|  |   bool set_foreground_color; | ||||||
|  |   bool set_background_color; | ||||||
|  |   emphasis ems; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | FMT_CONSTEXPR text_style fg(internal::color_type foreground) FMT_NOEXCEPT { | ||||||
|  |   return text_style(/*is_foreground=*/true, foreground); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_CONSTEXPR text_style bg(internal::color_type background) FMT_NOEXCEPT { | ||||||
|  |   return text_style(/*is_foreground=*/false, background); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_CONSTEXPR text_style operator|(emphasis lhs, emphasis rhs) FMT_NOEXCEPT { | ||||||
|  |   return text_style(lhs) | rhs; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | namespace internal { | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | struct ansi_color_escape { | ||||||
|  |   FMT_CONSTEXPR ansi_color_escape(internal::color_type text_color, | ||||||
|  |                                   const char * esc) FMT_NOEXCEPT { | ||||||
|  |     // If we have a terminal color, we need to output another escape code | ||||||
|  |     // sequence. | ||||||
|  |     if (!text_color.is_rgb) { | ||||||
|  |       bool is_background = esc == internal::data::BACKGROUND_COLOR; | ||||||
|  |       uint32_t value = text_color.value.term_color; | ||||||
|  |       // Background ASCII codes are the same as the foreground ones but with | ||||||
|  |       // 10 more. | ||||||
|  |       if (is_background) | ||||||
|  |         value += 10u; | ||||||
|  |  | ||||||
|  |       std::size_t index = 0; | ||||||
|  |       buffer[index++] = static_cast<Char>('\x1b'); | ||||||
|  |       buffer[index++] = static_cast<Char>('['); | ||||||
|  |  | ||||||
|  |       if (value >= 100u) { | ||||||
|  |         buffer[index++] = static_cast<Char>('1'); | ||||||
|  |         value %= 100u; | ||||||
|  |       } | ||||||
|  |       buffer[index++] = static_cast<Char>('0' + value / 10u); | ||||||
|  |       buffer[index++] = static_cast<Char>('0' + value % 10u); | ||||||
|  |  | ||||||
|  |       buffer[index++] = static_cast<Char>('m'); | ||||||
|  |       buffer[index++] = static_cast<Char>('\0'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     for (int i = 0; i < 7; i++) { | ||||||
|  |       buffer[i] = static_cast<Char>(esc[i]); | ||||||
|  |     } | ||||||
|  |     rgb color(text_color.value.rgb_color); | ||||||
|  |     to_esc(color.r, buffer +  7, ';'); | ||||||
|  |     to_esc(color.g, buffer + 11, ';'); | ||||||
|  |     to_esc(color.b, buffer + 15, 'm'); | ||||||
|  |     buffer[19] = static_cast<Char>(0); | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR ansi_color_escape(emphasis em) FMT_NOEXCEPT { | ||||||
|  |     uint8_t em_codes[4] = {}; | ||||||
|  |     uint8_t em_bits = static_cast<uint8_t>(em); | ||||||
|  |     if (em_bits & static_cast<uint8_t>(emphasis::bold)) | ||||||
|  |       em_codes[0] = 1; | ||||||
|  |     if (em_bits & static_cast<uint8_t>(emphasis::italic)) | ||||||
|  |       em_codes[1] = 3; | ||||||
|  |     if (em_bits & static_cast<uint8_t>(emphasis::underline)) | ||||||
|  |       em_codes[2] = 4; | ||||||
|  |     if (em_bits & static_cast<uint8_t>(emphasis::strikethrough)) | ||||||
|  |       em_codes[3] = 9; | ||||||
|  |  | ||||||
|  |     std::size_t index = 0; | ||||||
|  |     for (int i = 0; i < 4; ++i) { | ||||||
|  |       if (!em_codes[i]) | ||||||
|  |         continue; | ||||||
|  |       buffer[index++] = static_cast<Char>('\x1b'); | ||||||
|  |       buffer[index++] = static_cast<Char>('['); | ||||||
|  |       buffer[index++] = static_cast<Char>('0' + em_codes[i]); | ||||||
|  |       buffer[index++] = static_cast<Char>('m'); | ||||||
|  |     } | ||||||
|  |     buffer[index++] = static_cast<Char>(0); | ||||||
|  |   } | ||||||
|  |   FMT_CONSTEXPR operator const Char *() const FMT_NOEXCEPT { return buffer; } | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |   Char buffer[7u + 3u * 4u + 1u]; | ||||||
|  |  | ||||||
|  |   static FMT_CONSTEXPR void to_esc(uint8_t c, Char *out, | ||||||
|  |                                    char delimiter) FMT_NOEXCEPT { | ||||||
|  |     out[0] = static_cast<Char>('0' + c / 100); | ||||||
|  |     out[1] = static_cast<Char>('0' + c / 10 % 10); | ||||||
|  |     out[2] = static_cast<Char>('0' + c % 10); | ||||||
|  |     out[3] = static_cast<Char>(delimiter); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | FMT_CONSTEXPR ansi_color_escape<Char> | ||||||
|  | make_foreground_color(internal::color_type foreground) FMT_NOEXCEPT { | ||||||
|  |   return ansi_color_escape<Char>(foreground, internal::data::FOREGROUND_COLOR); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | FMT_CONSTEXPR ansi_color_escape<Char> | ||||||
|  | make_background_color(internal::color_type background) FMT_NOEXCEPT { | ||||||
|  |   return ansi_color_escape<Char>(background, internal::data::BACKGROUND_COLOR); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | FMT_CONSTEXPR ansi_color_escape<Char> | ||||||
|  | make_emphasis(emphasis em) FMT_NOEXCEPT { | ||||||
|  |   return ansi_color_escape<Char>(em); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | inline void fputs(const Char *chars, FILE *stream) FMT_NOEXCEPT { | ||||||
|  |   std::fputs(chars, stream); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <> | ||||||
|  | inline void fputs<wchar_t>(const wchar_t *chars, FILE *stream) FMT_NOEXCEPT { | ||||||
|  |   std::fputws(chars, stream); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | inline void reset_color(FILE *stream) FMT_NOEXCEPT { | ||||||
|  |   fputs(internal::data::RESET_COLOR, stream); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <> | ||||||
|  | inline void reset_color<wchar_t>(FILE *stream) FMT_NOEXCEPT { | ||||||
|  |   fputs(internal::data::WRESET_COLOR, stream); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // The following specialiazation disables using std::FILE as a character type, | ||||||
|  | // which is needed because or else | ||||||
|  | //   fmt::print(stderr, fmt::emphasis::bold, ""); | ||||||
|  | // would take stderr (a std::FILE *) as the format string. | ||||||
|  | template <> | ||||||
|  | struct is_string<std::FILE *> : std::false_type {}; | ||||||
|  | template <> | ||||||
|  | struct is_string<const std::FILE *> : std::false_type {}; | ||||||
|  | } // namespace internal | ||||||
|  |  | ||||||
|  | template < | ||||||
|  |   typename S, typename Char = typename internal::char_t<S>::type> | ||||||
|  | void vprint(std::FILE *f, const text_style &ts, const S &format, | ||||||
|  |             basic_format_args<typename buffer_context<Char>::type> args) { | ||||||
|  |   bool has_style = false; | ||||||
|  |   if (ts.has_emphasis()) { | ||||||
|  |     has_style = true; | ||||||
|  |     internal::fputs<Char>( | ||||||
|  |           internal::make_emphasis<Char>(ts.get_emphasis()), f); | ||||||
|  |   } | ||||||
|  |   if (ts.has_foreground()) { | ||||||
|  |     has_style = true; | ||||||
|  |     internal::fputs<Char>( | ||||||
|  |           internal::make_foreground_color<Char>(ts.get_foreground()), f); | ||||||
|  |   } | ||||||
|  |   if (ts.has_background()) { | ||||||
|  |     has_style = true; | ||||||
|  |     internal::fputs<Char>( | ||||||
|  |         internal::make_background_color<Char>(ts.get_background()), f); | ||||||
|  |   } | ||||||
|  |   vprint(f, format, args); | ||||||
|  |   if (has_style) { | ||||||
|  |     internal::reset_color<Char>(f); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   Formats a string and prints it to the specified file stream using ANSI | ||||||
|  |   escape sequences to specify text formatting. | ||||||
|  |   Example: | ||||||
|  |     fmt::print(fmt::emphasis::bold | fg(fmt::color::red), | ||||||
|  |                "Elapsed time: {0:.2f} seconds", 1.23); | ||||||
|  |  */ | ||||||
|  | template <typename String, typename... Args> | ||||||
|  | typename std::enable_if<internal::is_string<String>::value>::type print( | ||||||
|  |     std::FILE *f, const text_style &ts, const String &format_str, | ||||||
|  |     const Args &... args) { | ||||||
|  |   internal::check_format_string<Args...>(format_str); | ||||||
|  |   typedef typename internal::char_t<String>::type char_t; | ||||||
|  |   typedef typename buffer_context<char_t>::type context_t; | ||||||
|  |   format_arg_store<context_t, Args...> as{args...}; | ||||||
|  |   vprint(f, ts, format_str, basic_format_args<context_t>(as)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   Formats a string and prints it to stdout using ANSI escape sequences to | ||||||
|  |   specify text formatting. | ||||||
|  |   Example: | ||||||
|  |     fmt::print(fmt::emphasis::bold | fg(fmt::color::red), | ||||||
|  |                "Elapsed time: {0:.2f} seconds", 1.23); | ||||||
|  |  */ | ||||||
|  | template <typename String, typename... Args> | ||||||
|  | typename std::enable_if<internal::is_string<String>::value>::type print( | ||||||
|  |     const text_style &ts, const String &format_str, | ||||||
|  |     const Args &... args) { | ||||||
|  |   return print(stdout, ts, format_str, args...); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_COLOR_H_ | ||||||
							
								
								
									
										1502
									
								
								logger/bundled/fmt/core.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1502
									
								
								logger/bundled/fmt/core.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										972
									
								
								logger/bundled/fmt/format-inl.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										972
									
								
								logger/bundled/fmt/format-inl.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,972 @@ | |||||||
|  | // Formatting library for C++ | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - 2016, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_FORMAT_INL_H_ | ||||||
|  | #define FMT_FORMAT_INL_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  |  | ||||||
|  | #include <string.h> | ||||||
|  |  | ||||||
|  | #include <cctype> | ||||||
|  | #include <cerrno> | ||||||
|  | #include <climits> | ||||||
|  | #include <cmath> | ||||||
|  | #include <cstdarg> | ||||||
|  | #include <cstddef>  // for std::ptrdiff_t | ||||||
|  | #include <cstring>  // for std::memmove | ||||||
|  | #if !defined(FMT_STATIC_THOUSANDS_SEPARATOR) | ||||||
|  | # include <locale> | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #if FMT_USE_WINDOWS_H | ||||||
|  | # if !defined(FMT_HEADER_ONLY) && !defined(WIN32_LEAN_AND_MEAN) | ||||||
|  | #  define WIN32_LEAN_AND_MEAN | ||||||
|  | # endif | ||||||
|  | # if defined(NOMINMAX) || defined(FMT_WIN_MINMAX) | ||||||
|  | #  include <windows.h> | ||||||
|  | # else | ||||||
|  | #  define NOMINMAX | ||||||
|  | #  include <windows.h> | ||||||
|  | #  undef NOMINMAX | ||||||
|  | # endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #if FMT_EXCEPTIONS | ||||||
|  | # define FMT_TRY try | ||||||
|  | # define FMT_CATCH(x) catch (x) | ||||||
|  | #else | ||||||
|  | # define FMT_TRY if (true) | ||||||
|  | # define FMT_CATCH(x) if (false) | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef _MSC_VER | ||||||
|  | # pragma warning(push) | ||||||
|  | # pragma warning(disable: 4127)  // conditional expression is constant | ||||||
|  | # pragma warning(disable: 4702)  // unreachable code | ||||||
|  | // Disable deprecation warning for strerror. The latter is not called but | ||||||
|  | // MSVC fails to detect it. | ||||||
|  | # pragma warning(disable: 4996) | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | // Dummy implementations of strerror_r and strerror_s called if corresponding | ||||||
|  | // system functions are not available. | ||||||
|  | inline fmt::internal::null<> strerror_r(int, char *, ...) { | ||||||
|  |   return fmt::internal::null<>(); | ||||||
|  | } | ||||||
|  | inline fmt::internal::null<> strerror_s(char *, std::size_t, ...) { | ||||||
|  |   return fmt::internal::null<>(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | namespace { | ||||||
|  |  | ||||||
|  | #ifndef _MSC_VER | ||||||
|  | # define FMT_SNPRINTF snprintf | ||||||
|  | #else  // _MSC_VER | ||||||
|  | inline int fmt_snprintf(char *buffer, size_t size, const char *format, ...) { | ||||||
|  |   va_list args; | ||||||
|  |   va_start(args, format); | ||||||
|  |   int result = vsnprintf_s(buffer, size, _TRUNCATE, format, args); | ||||||
|  |   va_end(args); | ||||||
|  |   return result; | ||||||
|  | } | ||||||
|  | # define FMT_SNPRINTF fmt_snprintf | ||||||
|  | #endif  // _MSC_VER | ||||||
|  |  | ||||||
|  | #if defined(_WIN32) && defined(__MINGW32__) && !defined(__NO_ISOCEXT) | ||||||
|  | # define FMT_SWPRINTF snwprintf | ||||||
|  | #else | ||||||
|  | # define FMT_SWPRINTF swprintf | ||||||
|  | #endif // defined(_WIN32) && defined(__MINGW32__) && !defined(__NO_ISOCEXT) | ||||||
|  |  | ||||||
|  | typedef void (*FormatFunc)(internal::buffer &, int, string_view); | ||||||
|  |  | ||||||
|  | // Portable thread-safe version of strerror. | ||||||
|  | // Sets buffer to point to a string describing the error code. | ||||||
|  | // This can be either a pointer to a string stored in buffer, | ||||||
|  | // or a pointer to some static immutable string. | ||||||
|  | // Returns one of the following values: | ||||||
|  | //   0      - success | ||||||
|  | //   ERANGE - buffer is not large enough to store the error message | ||||||
|  | //   other  - failure | ||||||
|  | // Buffer should be at least of size 1. | ||||||
|  | int safe_strerror( | ||||||
|  |     int error_code, char *&buffer, std::size_t buffer_size) FMT_NOEXCEPT { | ||||||
|  |   FMT_ASSERT(buffer != FMT_NULL && buffer_size != 0, "invalid buffer"); | ||||||
|  |  | ||||||
|  |   class dispatcher { | ||||||
|  |    private: | ||||||
|  |     int error_code_; | ||||||
|  |     char *&buffer_; | ||||||
|  |     std::size_t buffer_size_; | ||||||
|  |  | ||||||
|  |     // A noop assignment operator to avoid bogus warnings. | ||||||
|  |     void operator=(const dispatcher &) {} | ||||||
|  |  | ||||||
|  |     // Handle the result of XSI-compliant version of strerror_r. | ||||||
|  |     int handle(int result) { | ||||||
|  |       // glibc versions before 2.13 return result in errno. | ||||||
|  |       return result == -1 ? errno : result; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Handle the result of GNU-specific version of strerror_r. | ||||||
|  |     int handle(char *message) { | ||||||
|  |       // If the buffer is full then the message is probably truncated. | ||||||
|  |       if (message == buffer_ && strlen(buffer_) == buffer_size_ - 1) | ||||||
|  |         return ERANGE; | ||||||
|  |       buffer_ = message; | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Handle the case when strerror_r is not available. | ||||||
|  |     int handle(internal::null<>) { | ||||||
|  |       return fallback(strerror_s(buffer_, buffer_size_, error_code_)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Fallback to strerror_s when strerror_r is not available. | ||||||
|  |     int fallback(int result) { | ||||||
|  |       // If the buffer is full then the message is probably truncated. | ||||||
|  |       return result == 0 && strlen(buffer_) == buffer_size_ - 1 ? | ||||||
|  |             ERANGE : result; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | #if !FMT_MSC_VER | ||||||
|  |     // Fallback to strerror if strerror_r and strerror_s are not available. | ||||||
|  |     int fallback(internal::null<>) { | ||||||
|  |       errno = 0; | ||||||
|  |       buffer_ = strerror(error_code_); | ||||||
|  |       return errno; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |    public: | ||||||
|  |     dispatcher(int err_code, char *&buf, std::size_t buf_size) | ||||||
|  |       : error_code_(err_code), buffer_(buf), buffer_size_(buf_size) {} | ||||||
|  |  | ||||||
|  |     int run() { | ||||||
|  |       return handle(strerror_r(error_code_, buffer_, buffer_size_)); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |   return dispatcher(error_code, buffer, buffer_size).run(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void format_error_code(internal::buffer &out, int error_code, | ||||||
|  |                        string_view message) FMT_NOEXCEPT { | ||||||
|  |   // Report error code making sure that the output fits into | ||||||
|  |   // inline_buffer_size to avoid dynamic memory allocation and potential | ||||||
|  |   // bad_alloc. | ||||||
|  |   out.resize(0); | ||||||
|  |   static const char SEP[] = ": "; | ||||||
|  |   static const char ERROR_STR[] = "error "; | ||||||
|  |   // Subtract 2 to account for terminating null characters in SEP and ERROR_STR. | ||||||
|  |   std::size_t error_code_size = sizeof(SEP) + sizeof(ERROR_STR) - 2; | ||||||
|  |   typedef internal::int_traits<int>::main_type main_type; | ||||||
|  |   main_type abs_value = static_cast<main_type>(error_code); | ||||||
|  |   if (internal::is_negative(error_code)) { | ||||||
|  |     abs_value = 0 - abs_value; | ||||||
|  |     ++error_code_size; | ||||||
|  |   } | ||||||
|  |   error_code_size += internal::to_unsigned(internal::count_digits(abs_value)); | ||||||
|  |   writer w(out); | ||||||
|  |   if (message.size() <= inline_buffer_size - error_code_size) { | ||||||
|  |     w.write(message); | ||||||
|  |     w.write(SEP); | ||||||
|  |   } | ||||||
|  |   w.write(ERROR_STR); | ||||||
|  |   w.write(error_code); | ||||||
|  |   assert(out.size() <= inline_buffer_size); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void report_error(FormatFunc func, int error_code, | ||||||
|  |                   string_view message) FMT_NOEXCEPT { | ||||||
|  |   memory_buffer full_message; | ||||||
|  |   func(full_message, error_code, message); | ||||||
|  |   // Use Writer::data instead of Writer::c_str to avoid potential memory | ||||||
|  |   // allocation. | ||||||
|  |   std::fwrite(full_message.data(), full_message.size(), 1, stderr); | ||||||
|  |   std::fputc('\n', stderr); | ||||||
|  | } | ||||||
|  | }  // namespace | ||||||
|  |  | ||||||
|  | FMT_FUNC size_t internal::count_code_points(basic_string_view<char8_t> s) { | ||||||
|  |   const char8_t *data = s.data(); | ||||||
|  |   size_t num_code_points = 0; | ||||||
|  |   for (size_t i = 0, size = s.size(); i != size; ++i) { | ||||||
|  |     if ((data[i] & 0xc0) != 0x80) | ||||||
|  |       ++num_code_points; | ||||||
|  |   } | ||||||
|  |   return num_code_points; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #if !defined(FMT_STATIC_THOUSANDS_SEPARATOR) | ||||||
|  | namespace internal { | ||||||
|  |  | ||||||
|  | template <typename Locale> | ||||||
|  | locale_ref::locale_ref(const Locale &loc) : locale_(&loc) { | ||||||
|  |   static_assert(std::is_same<Locale, std::locale>::value, ""); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Locale> | ||||||
|  | Locale locale_ref::get() const { | ||||||
|  |   static_assert(std::is_same<Locale, std::locale>::value, ""); | ||||||
|  |   return locale_ ? *static_cast<const std::locale*>(locale_) : std::locale(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | FMT_FUNC Char thousands_sep_impl(locale_ref loc) { | ||||||
|  |   return std::use_facet<std::numpunct<Char> >( | ||||||
|  |     loc.get<std::locale>()).thousands_sep(); | ||||||
|  | } | ||||||
|  | } | ||||||
|  | #else | ||||||
|  | template <typename Char> | ||||||
|  | FMT_FUNC Char internal::thousands_sep_impl(locale_ref) { | ||||||
|  |   return FMT_STATIC_THOUSANDS_SEPARATOR; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | FMT_FUNC void system_error::init( | ||||||
|  |     int err_code, string_view format_str, format_args args) { | ||||||
|  |   error_code_ = err_code; | ||||||
|  |   memory_buffer buffer; | ||||||
|  |   format_system_error(buffer, err_code, vformat(format_str, args)); | ||||||
|  |   std::runtime_error &base = *this; | ||||||
|  |   base = std::runtime_error(to_string(buffer)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | namespace internal { | ||||||
|  | template <typename T> | ||||||
|  | int char_traits<char>::format_float( | ||||||
|  |     char *buf, std::size_t size, const char *format, int precision, T value) { | ||||||
|  |   return precision < 0 ? | ||||||
|  |       FMT_SNPRINTF(buf, size, format, value) : | ||||||
|  |       FMT_SNPRINTF(buf, size, format, precision, value); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | int char_traits<wchar_t>::format_float( | ||||||
|  |     wchar_t *buf, std::size_t size, const wchar_t *format, int precision, | ||||||
|  |     T value) { | ||||||
|  |   return precision < 0 ? | ||||||
|  |       FMT_SWPRINTF(buf, size, format, value) : | ||||||
|  |       FMT_SWPRINTF(buf, size, format, precision, value); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | const char basic_data<T>::DIGITS[] = | ||||||
|  |     "0001020304050607080910111213141516171819" | ||||||
|  |     "2021222324252627282930313233343536373839" | ||||||
|  |     "4041424344454647484950515253545556575859" | ||||||
|  |     "6061626364656667686970717273747576777879" | ||||||
|  |     "8081828384858687888990919293949596979899"; | ||||||
|  |  | ||||||
|  | #define FMT_POWERS_OF_10(factor) \ | ||||||
|  |   factor * 10, \ | ||||||
|  |   factor * 100, \ | ||||||
|  |   factor * 1000, \ | ||||||
|  |   factor * 10000, \ | ||||||
|  |   factor * 100000, \ | ||||||
|  |   factor * 1000000, \ | ||||||
|  |   factor * 10000000, \ | ||||||
|  |   factor * 100000000, \ | ||||||
|  |   factor * 1000000000 | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | const uint32_t basic_data<T>::POWERS_OF_10_32[] = { | ||||||
|  |   1, FMT_POWERS_OF_10(1) | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | const uint32_t basic_data<T>::ZERO_OR_POWERS_OF_10_32[] = { | ||||||
|  |   0, FMT_POWERS_OF_10(1) | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | const uint64_t basic_data<T>::ZERO_OR_POWERS_OF_10_64[] = { | ||||||
|  |   0, | ||||||
|  |   FMT_POWERS_OF_10(1), | ||||||
|  |   FMT_POWERS_OF_10(1000000000ull), | ||||||
|  |   10000000000000000000ull | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Normalized 64-bit significands of pow(10, k), for k = -348, -340, ..., 340. | ||||||
|  | // These are generated by support/compute-powers.py. | ||||||
|  | template <typename T> | ||||||
|  | const uint64_t basic_data<T>::POW10_SIGNIFICANDS[] = { | ||||||
|  |   0xfa8fd5a0081c0288, 0xbaaee17fa23ebf76, 0x8b16fb203055ac76, | ||||||
|  |   0xcf42894a5dce35ea, 0x9a6bb0aa55653b2d, 0xe61acf033d1a45df, | ||||||
|  |   0xab70fe17c79ac6ca, 0xff77b1fcbebcdc4f, 0xbe5691ef416bd60c, | ||||||
|  |   0x8dd01fad907ffc3c, 0xd3515c2831559a83, 0x9d71ac8fada6c9b5, | ||||||
|  |   0xea9c227723ee8bcb, 0xaecc49914078536d, 0x823c12795db6ce57, | ||||||
|  |   0xc21094364dfb5637, 0x9096ea6f3848984f, 0xd77485cb25823ac7, | ||||||
|  |   0xa086cfcd97bf97f4, 0xef340a98172aace5, 0xb23867fb2a35b28e, | ||||||
|  |   0x84c8d4dfd2c63f3b, 0xc5dd44271ad3cdba, 0x936b9fcebb25c996, | ||||||
|  |   0xdbac6c247d62a584, 0xa3ab66580d5fdaf6, 0xf3e2f893dec3f126, | ||||||
|  |   0xb5b5ada8aaff80b8, 0x87625f056c7c4a8b, 0xc9bcff6034c13053, | ||||||
|  |   0x964e858c91ba2655, 0xdff9772470297ebd, 0xa6dfbd9fb8e5b88f, | ||||||
|  |   0xf8a95fcf88747d94, 0xb94470938fa89bcf, 0x8a08f0f8bf0f156b, | ||||||
|  |   0xcdb02555653131b6, 0x993fe2c6d07b7fac, 0xe45c10c42a2b3b06, | ||||||
|  |   0xaa242499697392d3, 0xfd87b5f28300ca0e, 0xbce5086492111aeb, | ||||||
|  |   0x8cbccc096f5088cc, 0xd1b71758e219652c, 0x9c40000000000000, | ||||||
|  |   0xe8d4a51000000000, 0xad78ebc5ac620000, 0x813f3978f8940984, | ||||||
|  |   0xc097ce7bc90715b3, 0x8f7e32ce7bea5c70, 0xd5d238a4abe98068, | ||||||
|  |   0x9f4f2726179a2245, 0xed63a231d4c4fb27, 0xb0de65388cc8ada8, | ||||||
|  |   0x83c7088e1aab65db, 0xc45d1df942711d9a, 0x924d692ca61be758, | ||||||
|  |   0xda01ee641a708dea, 0xa26da3999aef774a, 0xf209787bb47d6b85, | ||||||
|  |   0xb454e4a179dd1877, 0x865b86925b9bc5c2, 0xc83553c5c8965d3d, | ||||||
|  |   0x952ab45cfa97a0b3, 0xde469fbd99a05fe3, 0xa59bc234db398c25, | ||||||
|  |   0xf6c69a72a3989f5c, 0xb7dcbf5354e9bece, 0x88fcf317f22241e2, | ||||||
|  |   0xcc20ce9bd35c78a5, 0x98165af37b2153df, 0xe2a0b5dc971f303a, | ||||||
|  |   0xa8d9d1535ce3b396, 0xfb9b7cd9a4a7443c, 0xbb764c4ca7a44410, | ||||||
|  |   0x8bab8eefb6409c1a, 0xd01fef10a657842c, 0x9b10a4e5e9913129, | ||||||
|  |   0xe7109bfba19c0c9d, 0xac2820d9623bf429, 0x80444b5e7aa7cf85, | ||||||
|  |   0xbf21e44003acdd2d, 0x8e679c2f5e44ff8f, 0xd433179d9c8cb841, | ||||||
|  |   0x9e19db92b4e31ba9, 0xeb96bf6ebadf77d9, 0xaf87023b9bf0ee6b, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Binary exponents of pow(10, k), for k = -348, -340, ..., 340, corresponding | ||||||
|  | // to significands above. | ||||||
|  | template <typename T> | ||||||
|  | const int16_t basic_data<T>::POW10_EXPONENTS[] = { | ||||||
|  |   -1220, -1193, -1166, -1140, -1113, -1087, -1060, -1034, -1007,  -980,  -954, | ||||||
|  |    -927,  -901,  -874,  -847,  -821,  -794,  -768,  -741,  -715,  -688,  -661, | ||||||
|  |    -635,  -608,  -582,  -555,  -529,  -502,  -475,  -449,  -422,  -396,  -369, | ||||||
|  |    -343,  -316,  -289,  -263,  -236,  -210,  -183,  -157,  -130,  -103,   -77, | ||||||
|  |     -50,   -24,     3,    30,    56,    83,   109,   136,   162,   189,   216, | ||||||
|  |     242,   269,   295,   322,   348,   375,   402,   428,   455,   481,   508, | ||||||
|  |     534,   561,   588,   614,   641,   667,   694,   720,   747,   774,   800, | ||||||
|  |     827,   853,   880,   907,   933,   960,   986,  1013,  1039,  1066 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> const char basic_data<T>::FOREGROUND_COLOR[] = "\x1b[38;2;"; | ||||||
|  | template <typename T> const char basic_data<T>::BACKGROUND_COLOR[] = "\x1b[48;2;"; | ||||||
|  | template <typename T> const char basic_data<T>::RESET_COLOR[] = "\x1b[0m"; | ||||||
|  | template <typename T> const wchar_t basic_data<T>::WRESET_COLOR[] = L"\x1b[0m"; | ||||||
|  |  | ||||||
|  | // A handmade floating-point number f * pow(2, e). | ||||||
|  | class fp { | ||||||
|  |  private: | ||||||
|  |   typedef uint64_t significand_type; | ||||||
|  |  | ||||||
|  |   // All sizes are in bits. | ||||||
|  |   static FMT_CONSTEXPR_DECL const int char_size = | ||||||
|  |     std::numeric_limits<unsigned char>::digits; | ||||||
|  |   // Subtract 1 to account for an implicit most significant bit in the | ||||||
|  |   // normalized form. | ||||||
|  |   static FMT_CONSTEXPR_DECL const int double_significand_size = | ||||||
|  |     std::numeric_limits<double>::digits - 1; | ||||||
|  |   static FMT_CONSTEXPR_DECL const uint64_t implicit_bit = | ||||||
|  |     1ull << double_significand_size; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   significand_type f; | ||||||
|  |   int e; | ||||||
|  |  | ||||||
|  |   static FMT_CONSTEXPR_DECL const int significand_size = | ||||||
|  |     sizeof(significand_type) * char_size; | ||||||
|  |  | ||||||
|  |   fp(): f(0), e(0) {} | ||||||
|  |   fp(uint64_t f_val, int e_val): f(f_val), e(e_val) {} | ||||||
|  |  | ||||||
|  |   // Constructs fp from an IEEE754 double. It is a template to prevent compile | ||||||
|  |   // errors on platforms where double is not IEEE754. | ||||||
|  |   template <typename Double> | ||||||
|  |   explicit fp(Double d) { | ||||||
|  |     // Assume double is in the format [sign][exponent][significand]. | ||||||
|  |     typedef std::numeric_limits<Double> limits; | ||||||
|  |     const int double_size = static_cast<int>(sizeof(Double) * char_size); | ||||||
|  |     const int exponent_size = | ||||||
|  |       double_size - double_significand_size - 1;  // -1 for sign | ||||||
|  |     const uint64_t significand_mask = implicit_bit - 1; | ||||||
|  |     const uint64_t exponent_mask = (~0ull >> 1) & ~significand_mask; | ||||||
|  |     const int exponent_bias = (1 << exponent_size) - limits::max_exponent - 1; | ||||||
|  |     auto u = bit_cast<uint64_t>(d); | ||||||
|  |     auto biased_e = (u & exponent_mask) >> double_significand_size; | ||||||
|  |     f = u & significand_mask; | ||||||
|  |     if (biased_e != 0) | ||||||
|  |       f += implicit_bit; | ||||||
|  |     else | ||||||
|  |       biased_e = 1;  // Subnormals use biased exponent 1 (min exponent). | ||||||
|  |     e = static_cast<int>(biased_e - exponent_bias - double_significand_size); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Normalizes the value converted from double and multiplied by (1 << SHIFT). | ||||||
|  |   template <int SHIFT = 0> | ||||||
|  |   void normalize() { | ||||||
|  |     // Handle subnormals. | ||||||
|  |     auto shifted_implicit_bit = implicit_bit << SHIFT; | ||||||
|  |     while ((f & shifted_implicit_bit) == 0) { | ||||||
|  |       f <<= 1; | ||||||
|  |       --e; | ||||||
|  |     } | ||||||
|  |     // Subtract 1 to account for hidden bit. | ||||||
|  |     auto offset = significand_size - double_significand_size - SHIFT - 1; | ||||||
|  |     f <<= offset; | ||||||
|  |     e -= offset; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Compute lower and upper boundaries (m^- and m^+ in the Grisu paper), where | ||||||
|  |   // a boundary is a value half way between the number and its predecessor | ||||||
|  |   // (lower) or successor (upper). The upper boundary is normalized and lower | ||||||
|  |   // has the same exponent but may be not normalized. | ||||||
|  |   void compute_boundaries(fp &lower, fp &upper) const { | ||||||
|  |     lower = f == implicit_bit ? | ||||||
|  |           fp((f << 2) - 1, e - 2) : fp((f << 1) - 1, e - 1); | ||||||
|  |     upper = fp((f << 1) + 1, e - 1); | ||||||
|  |     upper.normalize<1>();  // 1 is to account for the exponent shift above. | ||||||
|  |     lower.f <<= lower.e - upper.e; | ||||||
|  |     lower.e = upper.e; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Returns an fp number representing x - y. Result may not be normalized. | ||||||
|  | inline fp operator-(fp x, fp y) { | ||||||
|  |   FMT_ASSERT(x.f >= y.f && x.e == y.e, "invalid operands"); | ||||||
|  |   return fp(x.f - y.f, x.e); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Computes an fp number r with r.f = x.f * y.f / pow(2, 64) rounded to nearest | ||||||
|  | // with half-up tie breaking, r.e = x.e + y.e + 64. Result may not be normalized. | ||||||
|  | FMT_API fp operator*(fp x, fp y); | ||||||
|  |  | ||||||
|  | // Returns cached power (of 10) c_k = c_k.f * pow(2, c_k.e) such that its | ||||||
|  | // (binary) exponent satisfies min_exponent <= c_k.e <= min_exponent + 3. | ||||||
|  | FMT_API fp get_cached_power(int min_exponent, int &pow10_exponent); | ||||||
|  |  | ||||||
|  | FMT_FUNC fp operator*(fp x, fp y) { | ||||||
|  |   // Multiply 32-bit parts of significands. | ||||||
|  |   uint64_t mask = (1ULL << 32) - 1; | ||||||
|  |   uint64_t a = x.f >> 32, b = x.f & mask; | ||||||
|  |   uint64_t c = y.f >> 32, d = y.f & mask; | ||||||
|  |   uint64_t ac = a * c, bc = b * c, ad = a * d, bd = b * d; | ||||||
|  |   // Compute mid 64-bit of result and round. | ||||||
|  |   uint64_t mid = (bd >> 32) + (ad & mask) + (bc & mask) + (1U << 31); | ||||||
|  |   return fp(ac + (ad >> 32) + (bc >> 32) + (mid >> 32), x.e + y.e + 64); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC fp get_cached_power(int min_exponent, int &pow10_exponent) { | ||||||
|  |   const double one_over_log2_10 = 0.30102999566398114;  // 1 / log2(10) | ||||||
|  |   int index = static_cast<int>(std::ceil( | ||||||
|  |         (min_exponent + fp::significand_size - 1) * one_over_log2_10)); | ||||||
|  |   // Decimal exponent of the first (smallest) cached power of 10. | ||||||
|  |   const int first_dec_exp = -348; | ||||||
|  |   // Difference between 2 consecutive decimal exponents in cached powers of 10. | ||||||
|  |   const int dec_exp_step = 8; | ||||||
|  |   index = (index - first_dec_exp - 1) / dec_exp_step + 1; | ||||||
|  |   pow10_exponent = first_dec_exp + index * dec_exp_step; | ||||||
|  |   return fp(data::POW10_SIGNIFICANDS[index], data::POW10_EXPONENTS[index]); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC bool grisu2_round( | ||||||
|  |     char *buf, int &size, int max_digits, uint64_t delta, | ||||||
|  |     uint64_t remainder, uint64_t exp, uint64_t diff, int &exp10) { | ||||||
|  |   while (remainder < diff && delta - remainder >= exp && | ||||||
|  |         (remainder + exp < diff || diff - remainder > remainder + exp - diff)) { | ||||||
|  |     --buf[size - 1]; | ||||||
|  |     remainder += exp; | ||||||
|  |   } | ||||||
|  |   if (size > max_digits) { | ||||||
|  |     --size; | ||||||
|  |     ++exp10; | ||||||
|  |     if (buf[size] >= '5') | ||||||
|  |       return false; | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Generates output using Grisu2 digit-gen algorithm. | ||||||
|  | FMT_FUNC bool grisu2_gen_digits( | ||||||
|  |     char *buf, int &size, uint32_t hi, uint64_t lo, int &exp, | ||||||
|  |     uint64_t delta, const fp &one, const fp &diff, int max_digits) { | ||||||
|  |   // Generate digits for the most significant part (hi). | ||||||
|  |   while (exp > 0) { | ||||||
|  |     uint32_t digit = 0; | ||||||
|  |     // This optimization by miloyip reduces the number of integer divisions by | ||||||
|  |     // one per iteration. | ||||||
|  |     switch (exp) { | ||||||
|  |     case 10: digit = hi / 1000000000; hi %= 1000000000; break; | ||||||
|  |     case  9: digit = hi /  100000000; hi %=  100000000; break; | ||||||
|  |     case  8: digit = hi /   10000000; hi %=   10000000; break; | ||||||
|  |     case  7: digit = hi /    1000000; hi %=    1000000; break; | ||||||
|  |     case  6: digit = hi /     100000; hi %=     100000; break; | ||||||
|  |     case  5: digit = hi /      10000; hi %=      10000; break; | ||||||
|  |     case  4: digit = hi /       1000; hi %=       1000; break; | ||||||
|  |     case  3: digit = hi /        100; hi %=        100; break; | ||||||
|  |     case  2: digit = hi /         10; hi %=         10; break; | ||||||
|  |     case  1: digit = hi;              hi =           0; break; | ||||||
|  |     default: | ||||||
|  |       FMT_ASSERT(false, "invalid number of digits"); | ||||||
|  |     } | ||||||
|  |     if (digit != 0 || size != 0) | ||||||
|  |       buf[size++] = static_cast<char>('0' + digit); | ||||||
|  |     --exp; | ||||||
|  |     uint64_t remainder = (static_cast<uint64_t>(hi) << -one.e) + lo; | ||||||
|  |     if (remainder <= delta || size > max_digits) { | ||||||
|  |       return grisu2_round( | ||||||
|  |             buf, size, max_digits, delta, remainder, | ||||||
|  |             static_cast<uint64_t>(data::POWERS_OF_10_32[exp]) << -one.e, | ||||||
|  |             diff.f, exp); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   // Generate digits for the least significant part (lo). | ||||||
|  |   for (;;) { | ||||||
|  |     lo *= 10; | ||||||
|  |     delta *= 10; | ||||||
|  |     char digit = static_cast<char>(lo >> -one.e); | ||||||
|  |     if (digit != 0 || size != 0) | ||||||
|  |       buf[size++] = static_cast<char>('0' + digit); | ||||||
|  |     lo &= one.f - 1; | ||||||
|  |     --exp; | ||||||
|  |     if (lo < delta || size > max_digits) { | ||||||
|  |       return grisu2_round(buf, size, max_digits, delta, lo, one.f, | ||||||
|  |                           diff.f * data::POWERS_OF_10_32[-exp], exp); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #if FMT_CLANG_VERSION | ||||||
|  | # define FMT_FALLTHROUGH [[clang::fallthrough]]; | ||||||
|  | #elif FMT_GCC_VERSION >= 700 | ||||||
|  | # define FMT_FALLTHROUGH [[gnu::fallthrough]]; | ||||||
|  | #else | ||||||
|  | # define FMT_FALLTHROUGH | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | struct gen_digits_params { | ||||||
|  |   int num_digits; | ||||||
|  |   bool fixed; | ||||||
|  |   bool upper; | ||||||
|  |   bool trailing_zeros; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct prettify_handler { | ||||||
|  |   char *data; | ||||||
|  |   ptrdiff_t size; | ||||||
|  |   buffer &buf; | ||||||
|  |  | ||||||
|  |   explicit prettify_handler(buffer &b, ptrdiff_t n) | ||||||
|  |     : data(b.data()), size(n), buf(b) {} | ||||||
|  |   ~prettify_handler() { | ||||||
|  |     assert(buf.size() >= to_unsigned(size)); | ||||||
|  |     buf.resize(to_unsigned(size)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename F> | ||||||
|  |   void insert(ptrdiff_t pos, ptrdiff_t n, F f) { | ||||||
|  |     std::memmove(data + pos + n, data + pos, to_unsigned(size - pos)); | ||||||
|  |     f(data + pos); | ||||||
|  |     size += n; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void insert(ptrdiff_t pos, char c) { | ||||||
|  |     std::memmove(data + pos + 1, data + pos, to_unsigned(size - pos)); | ||||||
|  |     data[pos] = c; | ||||||
|  |     ++size; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void append(ptrdiff_t n, char c) { | ||||||
|  |     std::uninitialized_fill_n(data + size, n, c); | ||||||
|  |     size += n; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void append(char c) { data[size++] = c; } | ||||||
|  |  | ||||||
|  |   void remove_trailing(char c) { | ||||||
|  |     while (data[size - 1] == c) --size; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Writes the exponent exp in the form "[+-]d{2,3}" to buffer. | ||||||
|  | template <typename Handler> | ||||||
|  | FMT_FUNC void write_exponent(int exp, Handler &&h) { | ||||||
|  |   FMT_ASSERT(-1000 < exp && exp < 1000, "exponent out of range"); | ||||||
|  |   if (exp < 0) { | ||||||
|  |     h.append('-'); | ||||||
|  |     exp = -exp; | ||||||
|  |   } else { | ||||||
|  |     h.append('+'); | ||||||
|  |   } | ||||||
|  |   if (exp >= 100) { | ||||||
|  |     h.append(static_cast<char>('0' + exp / 100)); | ||||||
|  |     exp %= 100; | ||||||
|  |     const char *d = data::DIGITS + exp * 2; | ||||||
|  |     h.append(d[0]); | ||||||
|  |     h.append(d[1]); | ||||||
|  |   } else { | ||||||
|  |     const char *d = data::DIGITS + exp * 2; | ||||||
|  |     h.append(d[0]); | ||||||
|  |     h.append(d[1]); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct fill { | ||||||
|  |   size_t n; | ||||||
|  |   void operator()(char *buf) const { | ||||||
|  |     buf[0] = '0'; | ||||||
|  |     buf[1] = '.'; | ||||||
|  |     std::uninitialized_fill_n(buf + 2, n, '0'); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // The number is given as v = f * pow(10, exp), where f has size digits. | ||||||
|  | template <typename Handler> | ||||||
|  | FMT_FUNC void grisu2_prettify(const gen_digits_params ¶ms, | ||||||
|  |                               int size, int exp, Handler &&handler) { | ||||||
|  |   if (!params.fixed) { | ||||||
|  |     // Insert a decimal point after the first digit and add an exponent. | ||||||
|  |     handler.insert(1, '.'); | ||||||
|  |     exp += size - 1; | ||||||
|  |     if (size < params.num_digits) | ||||||
|  |       handler.append(params.num_digits - size, '0'); | ||||||
|  |     handler.append(params.upper ? 'E' : 'e'); | ||||||
|  |     write_exponent(exp, handler); | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   // pow(10, full_exp - 1) <= v <= pow(10, full_exp). | ||||||
|  |   int full_exp = size + exp; | ||||||
|  |   const int exp_threshold = 21; | ||||||
|  |   if (size <= full_exp && full_exp <= exp_threshold) { | ||||||
|  |     // 1234e7 -> 12340000000[.0+] | ||||||
|  |     handler.append(full_exp - size, '0'); | ||||||
|  |     int num_zeros = params.num_digits - full_exp; | ||||||
|  |     if (num_zeros > 0 && params.trailing_zeros) { | ||||||
|  |       handler.append('.'); | ||||||
|  |       handler.append(num_zeros, '0'); | ||||||
|  |     } | ||||||
|  |   } else if (full_exp > 0) { | ||||||
|  |     // 1234e-2 -> 12.34[0+] | ||||||
|  |     handler.insert(full_exp, '.'); | ||||||
|  |     if (!params.trailing_zeros) { | ||||||
|  |       // Remove trailing zeros. | ||||||
|  |       handler.remove_trailing('0'); | ||||||
|  |     } else if (params.num_digits > size) { | ||||||
|  |       // Add trailing zeros. | ||||||
|  |       ptrdiff_t num_zeros = params.num_digits - size; | ||||||
|  |       handler.append(num_zeros, '0'); | ||||||
|  |     } | ||||||
|  |   } else { | ||||||
|  |     // 1234e-6 -> 0.001234 | ||||||
|  |     handler.insert(0, 2 - full_exp, fill{to_unsigned(-full_exp)}); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct char_counter { | ||||||
|  |   ptrdiff_t size; | ||||||
|  |  | ||||||
|  |   template <typename F> | ||||||
|  |   void insert(ptrdiff_t, ptrdiff_t n, F) { size += n; } | ||||||
|  |   void insert(ptrdiff_t, char) { ++size; } | ||||||
|  |   void append(ptrdiff_t n, char) { size += n; } | ||||||
|  |   void append(char) { ++size; } | ||||||
|  |   void remove_trailing(char) {} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Converts format specifiers into parameters for digit generation and computes | ||||||
|  | // output buffer size for a number in the range [pow(10, exp - 1), pow(10, exp) | ||||||
|  | // or 0 if exp == 1. | ||||||
|  | FMT_FUNC gen_digits_params process_specs(const core_format_specs &specs, | ||||||
|  |                                          int exp, buffer &buf) { | ||||||
|  |   auto params = gen_digits_params(); | ||||||
|  |   int num_digits = specs.precision >= 0 ? specs.precision : 6; | ||||||
|  |   switch (specs.type) { | ||||||
|  |   case 'G': | ||||||
|  |     params.upper = true; | ||||||
|  |     FMT_FALLTHROUGH | ||||||
|  |   case '\0': case 'g': | ||||||
|  |     params.trailing_zeros = (specs.flags & HASH_FLAG) != 0; | ||||||
|  |     if (-4 <= exp && exp < num_digits + 1) { | ||||||
|  |       params.fixed = true; | ||||||
|  |       if (!specs.type && params.trailing_zeros && exp >= 0) | ||||||
|  |         num_digits = exp + 1; | ||||||
|  |     } | ||||||
|  |     break; | ||||||
|  |   case 'F': | ||||||
|  |     params.upper = true; | ||||||
|  |     FMT_FALLTHROUGH | ||||||
|  |   case 'f': { | ||||||
|  |     params.fixed = true; | ||||||
|  |     params.trailing_zeros = true; | ||||||
|  |     int adjusted_min_digits = num_digits + exp; | ||||||
|  |     if (adjusted_min_digits > 0) | ||||||
|  |       num_digits = adjusted_min_digits; | ||||||
|  |     break; | ||||||
|  |   } | ||||||
|  |   case 'E': | ||||||
|  |     params.upper = true; | ||||||
|  |     FMT_FALLTHROUGH | ||||||
|  |   case 'e': | ||||||
|  |     ++num_digits; | ||||||
|  |     break; | ||||||
|  |   } | ||||||
|  |   params.num_digits = num_digits; | ||||||
|  |   char_counter counter{num_digits}; | ||||||
|  |   grisu2_prettify(params, params.num_digits, exp - num_digits, counter); | ||||||
|  |   buf.resize(to_unsigned(counter.size)); | ||||||
|  |   return params; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Double> | ||||||
|  | FMT_FUNC typename std::enable_if<sizeof(Double) == sizeof(uint64_t), bool>::type | ||||||
|  |     grisu2_format(Double value, buffer &buf, core_format_specs specs) { | ||||||
|  |   FMT_ASSERT(value >= 0, "value is negative"); | ||||||
|  |   if (value == 0) { | ||||||
|  |     gen_digits_params params = process_specs(specs, 1, buf); | ||||||
|  |     const size_t size = 1; | ||||||
|  |     buf[0] = '0'; | ||||||
|  |     grisu2_prettify(params, size, 0, prettify_handler(buf, size)); | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   fp fp_value(value); | ||||||
|  |   fp lower, upper;  // w^- and w^+ in the Grisu paper. | ||||||
|  |   fp_value.compute_boundaries(lower, upper); | ||||||
|  |  | ||||||
|  |   // Find a cached power of 10 close to 1 / upper and use it to scale upper. | ||||||
|  |   const int min_exp = -60;  // alpha in Grisu. | ||||||
|  |   int cached_exp = 0;  // K in Grisu. | ||||||
|  |   auto cached_pow = get_cached_power(  // \tilde{c}_{-k} in Grisu. | ||||||
|  |       min_exp - (upper.e + fp::significand_size), cached_exp); | ||||||
|  |   cached_exp = -cached_exp; | ||||||
|  |   upper = upper * cached_pow;  // \tilde{M}^+ in Grisu. | ||||||
|  |   --upper.f;  // \tilde{M}^+ - 1 ulp -> M^+_{\downarrow}. | ||||||
|  |   fp one(1ull << -upper.e, upper.e); | ||||||
|  |   // hi (p1 in Grisu) contains the most significant digits of scaled_upper. | ||||||
|  |   // hi = floor(upper / one). | ||||||
|  |   uint32_t hi = static_cast<uint32_t>(upper.f >> -one.e); | ||||||
|  |   int exp = count_digits(hi);  // kappa in Grisu. | ||||||
|  |   gen_digits_params params = process_specs(specs, cached_exp + exp, buf); | ||||||
|  |   fp_value.normalize(); | ||||||
|  |   fp scaled_value = fp_value * cached_pow; | ||||||
|  |   lower = lower * cached_pow;  // \tilde{M}^- in Grisu. | ||||||
|  |   ++lower.f;  // \tilde{M}^- + 1 ulp -> M^-_{\uparrow}. | ||||||
|  |   uint64_t delta = upper.f - lower.f; | ||||||
|  |   fp diff = upper - scaled_value; // wp_w in Grisu. | ||||||
|  |   // lo (p2 in Grisu) contains the least significants digits of scaled_upper. | ||||||
|  |   // lo = supper % one. | ||||||
|  |   uint64_t lo = upper.f & (one.f - 1); | ||||||
|  |   int size = 0; | ||||||
|  |   if (!grisu2_gen_digits(buf.data(), size, hi, lo, exp, delta, one, diff, | ||||||
|  |                          params.num_digits)) { | ||||||
|  |     buf.clear(); | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |   grisu2_prettify(params, size, cached_exp + exp, prettify_handler(buf, size)); | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Double> | ||||||
|  | void sprintf_format(Double value, internal::buffer &buf, | ||||||
|  |                     core_format_specs spec) { | ||||||
|  |   // Buffer capacity must be non-zero, otherwise MSVC's vsnprintf_s will fail. | ||||||
|  |   FMT_ASSERT(buf.capacity() != 0, "empty buffer"); | ||||||
|  |  | ||||||
|  |   // Build format string. | ||||||
|  |   enum { MAX_FORMAT_SIZE = 10}; // longest format: %#-*.*Lg | ||||||
|  |   char format[MAX_FORMAT_SIZE]; | ||||||
|  |   char *format_ptr = format; | ||||||
|  |   *format_ptr++ = '%'; | ||||||
|  |   if (spec.has(HASH_FLAG)) | ||||||
|  |     *format_ptr++ = '#'; | ||||||
|  |   if (spec.precision >= 0) { | ||||||
|  |     *format_ptr++ = '.'; | ||||||
|  |     *format_ptr++ = '*'; | ||||||
|  |   } | ||||||
|  |   if (std::is_same<Double, long double>::value) | ||||||
|  |     *format_ptr++ = 'L'; | ||||||
|  |   *format_ptr++ = spec.type; | ||||||
|  |   *format_ptr = '\0'; | ||||||
|  |  | ||||||
|  |   // Format using snprintf. | ||||||
|  |   char *start = FMT_NULL; | ||||||
|  |   for (;;) { | ||||||
|  |     std::size_t buffer_size = buf.capacity(); | ||||||
|  |     start = &buf[0]; | ||||||
|  |     int result = internal::char_traits<char>::format_float( | ||||||
|  |         start, buffer_size, format, spec.precision, value); | ||||||
|  |     if (result >= 0) { | ||||||
|  |       unsigned n = internal::to_unsigned(result); | ||||||
|  |       if (n < buf.capacity()) { | ||||||
|  |         buf.resize(n); | ||||||
|  |         break;  // The buffer is large enough - continue with formatting. | ||||||
|  |       } | ||||||
|  |       buf.reserve(n + 1); | ||||||
|  |     } else { | ||||||
|  |       // If result is negative we ask to increase the capacity by at least 1, | ||||||
|  |       // but as std::vector, the buffer grows exponentially. | ||||||
|  |       buf.reserve(buf.capacity() + 1); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | }  // namespace internal | ||||||
|  |  | ||||||
|  | #if FMT_USE_WINDOWS_H | ||||||
|  |  | ||||||
|  | FMT_FUNC internal::utf8_to_utf16::utf8_to_utf16(string_view s) { | ||||||
|  |   static const char ERROR_MSG[] = "cannot convert string from UTF-8 to UTF-16"; | ||||||
|  |   if (s.size() > INT_MAX) | ||||||
|  |     FMT_THROW(windows_error(ERROR_INVALID_PARAMETER, ERROR_MSG)); | ||||||
|  |   int s_size = static_cast<int>(s.size()); | ||||||
|  |   if (s_size == 0) { | ||||||
|  |     // MultiByteToWideChar does not support zero length, handle separately. | ||||||
|  |     buffer_.resize(1); | ||||||
|  |     buffer_[0] = 0; | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   int length = MultiByteToWideChar( | ||||||
|  |       CP_UTF8, MB_ERR_INVALID_CHARS, s.data(), s_size, FMT_NULL, 0); | ||||||
|  |   if (length == 0) | ||||||
|  |     FMT_THROW(windows_error(GetLastError(), ERROR_MSG)); | ||||||
|  |   buffer_.resize(length + 1); | ||||||
|  |   length = MultiByteToWideChar( | ||||||
|  |     CP_UTF8, MB_ERR_INVALID_CHARS, s.data(), s_size, &buffer_[0], length); | ||||||
|  |   if (length == 0) | ||||||
|  |     FMT_THROW(windows_error(GetLastError(), ERROR_MSG)); | ||||||
|  |   buffer_[length] = 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC internal::utf16_to_utf8::utf16_to_utf8(wstring_view s) { | ||||||
|  |   if (int error_code = convert(s)) { | ||||||
|  |     FMT_THROW(windows_error(error_code, | ||||||
|  |         "cannot convert string from UTF-16 to UTF-8")); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC int internal::utf16_to_utf8::convert(wstring_view s) { | ||||||
|  |   if (s.size() > INT_MAX) | ||||||
|  |     return ERROR_INVALID_PARAMETER; | ||||||
|  |   int s_size = static_cast<int>(s.size()); | ||||||
|  |   if (s_size == 0) { | ||||||
|  |     // WideCharToMultiByte does not support zero length, handle separately. | ||||||
|  |     buffer_.resize(1); | ||||||
|  |     buffer_[0] = 0; | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   int length = WideCharToMultiByte( | ||||||
|  |         CP_UTF8, 0, s.data(), s_size, FMT_NULL, 0, FMT_NULL, FMT_NULL); | ||||||
|  |   if (length == 0) | ||||||
|  |     return GetLastError(); | ||||||
|  |   buffer_.resize(length + 1); | ||||||
|  |   length = WideCharToMultiByte( | ||||||
|  |     CP_UTF8, 0, s.data(), s_size, &buffer_[0], length, FMT_NULL, FMT_NULL); | ||||||
|  |   if (length == 0) | ||||||
|  |     return GetLastError(); | ||||||
|  |   buffer_[length] = 0; | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void windows_error::init( | ||||||
|  |     int err_code, string_view format_str, format_args args) { | ||||||
|  |   error_code_ = err_code; | ||||||
|  |   memory_buffer buffer; | ||||||
|  |   internal::format_windows_error(buffer, err_code, vformat(format_str, args)); | ||||||
|  |   std::runtime_error &base = *this; | ||||||
|  |   base = std::runtime_error(to_string(buffer)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void internal::format_windows_error( | ||||||
|  |     internal::buffer &out, int error_code, string_view message) FMT_NOEXCEPT { | ||||||
|  |   FMT_TRY { | ||||||
|  |     wmemory_buffer buf; | ||||||
|  |     buf.resize(inline_buffer_size); | ||||||
|  |     for (;;) { | ||||||
|  |       wchar_t *system_message = &buf[0]; | ||||||
|  |       int result = FormatMessageW( | ||||||
|  |           FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, | ||||||
|  |           FMT_NULL, error_code, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), | ||||||
|  |           system_message, static_cast<uint32_t>(buf.size()), FMT_NULL); | ||||||
|  |       if (result != 0) { | ||||||
|  |         utf16_to_utf8 utf8_message; | ||||||
|  |         if (utf8_message.convert(system_message) == ERROR_SUCCESS) { | ||||||
|  |           writer w(out); | ||||||
|  |           w.write(message); | ||||||
|  |           w.write(": "); | ||||||
|  |           w.write(utf8_message); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) | ||||||
|  |         break;  // Can't get error message, report error code instead. | ||||||
|  |       buf.resize(buf.size() * 2); | ||||||
|  |     } | ||||||
|  |   } FMT_CATCH(...) {} | ||||||
|  |   format_error_code(out, error_code, message); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif  // FMT_USE_WINDOWS_H | ||||||
|  |  | ||||||
|  | FMT_FUNC void format_system_error( | ||||||
|  |     internal::buffer &out, int error_code, string_view message) FMT_NOEXCEPT { | ||||||
|  |   FMT_TRY { | ||||||
|  |     memory_buffer buf; | ||||||
|  |     buf.resize(inline_buffer_size); | ||||||
|  |     for (;;) { | ||||||
|  |       char *system_message = &buf[0]; | ||||||
|  |       int result = safe_strerror(error_code, system_message, buf.size()); | ||||||
|  |       if (result == 0) { | ||||||
|  |         writer w(out); | ||||||
|  |         w.write(message); | ||||||
|  |         w.write(": "); | ||||||
|  |         w.write(system_message); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       if (result != ERANGE) | ||||||
|  |         break;  // Can't get error message, report error code instead. | ||||||
|  |       buf.resize(buf.size() * 2); | ||||||
|  |     } | ||||||
|  |   } FMT_CATCH(...) {} | ||||||
|  |   format_error_code(out, error_code, message); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void internal::error_handler::on_error(const char *message) { | ||||||
|  |   FMT_THROW(format_error(message)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void report_system_error( | ||||||
|  |     int error_code, fmt::string_view message) FMT_NOEXCEPT { | ||||||
|  |   report_error(format_system_error, error_code, message); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #if FMT_USE_WINDOWS_H | ||||||
|  | FMT_FUNC void report_windows_error( | ||||||
|  |     int error_code, fmt::string_view message) FMT_NOEXCEPT { | ||||||
|  |   report_error(internal::format_windows_error, error_code, message); | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | FMT_FUNC void vprint(std::FILE *f, string_view format_str, format_args args) { | ||||||
|  |   memory_buffer buffer; | ||||||
|  |   internal::vformat_to(buffer, format_str, | ||||||
|  |                        basic_format_args<buffer_context<char>::type>(args)); | ||||||
|  |   std::fwrite(buffer.data(), 1, buffer.size(), f); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void vprint(std::FILE *f, wstring_view format_str, wformat_args args) { | ||||||
|  |   wmemory_buffer buffer; | ||||||
|  |   internal::vformat_to(buffer, format_str, args); | ||||||
|  |   std::fwrite(buffer.data(), sizeof(wchar_t), buffer.size(), f); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void vprint(string_view format_str, format_args args) { | ||||||
|  |   vprint(stdout, format_str, args); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_FUNC void vprint(wstring_view format_str, wformat_args args) { | ||||||
|  |   vprint(stdout, format_str, args); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #ifdef _MSC_VER | ||||||
|  | # pragma warning(pop) | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #endif  // FMT_FORMAT_INL_H_ | ||||||
							
								
								
									
										3555
									
								
								logger/bundled/fmt/format.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3555
									
								
								logger/bundled/fmt/format.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										77
									
								
								logger/bundled/fmt/locale.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								logger/bundled/fmt/locale.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | |||||||
|  | // Formatting library for C++ - std::locale support | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - present, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_LOCALE_H_ | ||||||
|  | #define FMT_LOCALE_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  | #include <locale> | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | namespace internal { | ||||||
|  | template <typename Char> | ||||||
|  | typename buffer_context<Char>::type::iterator vformat_to( | ||||||
|  |     const std::locale &loc, basic_buffer<Char> &buf, | ||||||
|  |     basic_string_view<Char> format_str, | ||||||
|  |     basic_format_args<typename buffer_context<Char>::type> args) { | ||||||
|  |   typedef back_insert_range<basic_buffer<Char> > range; | ||||||
|  |   return vformat_to<arg_formatter<range>>( | ||||||
|  |     buf, to_string_view(format_str), args, internal::locale_ref(loc)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | std::basic_string<Char> vformat( | ||||||
|  |     const std::locale &loc, basic_string_view<Char> format_str, | ||||||
|  |     basic_format_args<typename buffer_context<Char>::type> args) { | ||||||
|  |   basic_memory_buffer<Char> buffer; | ||||||
|  |   internal::vformat_to(loc, buffer, format_str, args); | ||||||
|  |   return fmt::to_string(buffer); | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename S, typename Char = FMT_CHAR(S)> | ||||||
|  | inline std::basic_string<Char> vformat( | ||||||
|  |     const std::locale &loc, const S &format_str, | ||||||
|  |     basic_format_args<typename buffer_context<Char>::type> args) { | ||||||
|  |   return internal::vformat(loc, to_string_view(format_str), args); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename S, typename... Args> | ||||||
|  | inline std::basic_string<FMT_CHAR(S)> format( | ||||||
|  |     const std::locale &loc, const S &format_str, const Args &... args) { | ||||||
|  |   return internal::vformat( | ||||||
|  |     loc, to_string_view(format_str), | ||||||
|  |     *internal::checked_args<S, Args...>(format_str, args...)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename String, typename OutputIt, typename... Args> | ||||||
|  | inline typename std::enable_if<internal::is_output_iterator<OutputIt>::value, | ||||||
|  |                                OutputIt>::type | ||||||
|  |     vformat_to(OutputIt out, const std::locale &loc, const String &format_str, | ||||||
|  |                typename format_args_t<OutputIt, FMT_CHAR(String)>::type args) { | ||||||
|  |   typedef output_range<OutputIt, FMT_CHAR(String)> range; | ||||||
|  |   return vformat_to<arg_formatter<range>>( | ||||||
|  |     range(out), to_string_view(format_str), args, internal::locale_ref(loc)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename OutputIt, typename S, typename... Args> | ||||||
|  | inline typename std::enable_if< | ||||||
|  |     internal::is_string<S>::value && | ||||||
|  |     internal::is_output_iterator<OutputIt>::value, OutputIt>::type | ||||||
|  |     format_to(OutputIt out, const std::locale &loc, const S &format_str, | ||||||
|  |               const Args &... args) { | ||||||
|  |   internal::check_format_string<Args...>(format_str); | ||||||
|  |   typedef typename format_context_t<OutputIt, FMT_CHAR(S)>::type context; | ||||||
|  |   format_arg_store<context, Args...> as{args...}; | ||||||
|  |   return vformat_to(out, loc, to_string_view(format_str), | ||||||
|  |                     basic_format_args<context>(as)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_LOCALE_H_ | ||||||
							
								
								
									
										153
									
								
								logger/bundled/fmt/ostream.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										153
									
								
								logger/bundled/fmt/ostream.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,153 @@ | |||||||
|  | // Formatting library for C++ - std::ostream support | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - present, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_OSTREAM_H_ | ||||||
|  | #define FMT_OSTREAM_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  | #include <ostream> | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  | namespace internal { | ||||||
|  |  | ||||||
|  | template <class Char> | ||||||
|  | class formatbuf : public std::basic_streambuf<Char> { | ||||||
|  |  private: | ||||||
|  |   typedef typename std::basic_streambuf<Char>::int_type int_type; | ||||||
|  |   typedef typename std::basic_streambuf<Char>::traits_type traits_type; | ||||||
|  |  | ||||||
|  |   basic_buffer<Char> &buffer_; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   formatbuf(basic_buffer<Char> &buffer) : buffer_(buffer) {} | ||||||
|  |  | ||||||
|  |  protected: | ||||||
|  |   // The put-area is actually always empty. This makes the implementation | ||||||
|  |   // simpler and has the advantage that the streambuf and the buffer are always | ||||||
|  |   // in sync and sputc never writes into uninitialized memory. The obvious | ||||||
|  |   // disadvantage is that each call to sputc always results in a (virtual) call | ||||||
|  |   // to overflow. There is no disadvantage here for sputn since this always | ||||||
|  |   // results in a call to xsputn. | ||||||
|  |  | ||||||
|  |   int_type overflow(int_type ch = traits_type::eof()) FMT_OVERRIDE { | ||||||
|  |     if (!traits_type::eq_int_type(ch, traits_type::eof())) | ||||||
|  |       buffer_.push_back(static_cast<Char>(ch)); | ||||||
|  |     return ch; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   std::streamsize xsputn(const Char *s, std::streamsize count) FMT_OVERRIDE { | ||||||
|  |     buffer_.append(s, s + count); | ||||||
|  |     return count; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | struct test_stream : std::basic_ostream<Char> { | ||||||
|  |  private: | ||||||
|  |   struct null; | ||||||
|  |   // Hide all operator<< from std::basic_ostream<Char>. | ||||||
|  |   void operator<<(null); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Checks if T has a user-defined operator<< (e.g. not a member of std::ostream). | ||||||
|  | template <typename T, typename Char> | ||||||
|  | class is_streamable { | ||||||
|  |  private: | ||||||
|  |   template <typename U> | ||||||
|  |   static decltype( | ||||||
|  |     internal::declval<test_stream<Char>&>() | ||||||
|  |       << internal::declval<U>(), std::true_type()) test(int); | ||||||
|  |  | ||||||
|  |   template <typename> | ||||||
|  |   static std::false_type test(...); | ||||||
|  |  | ||||||
|  |   typedef decltype(test<T>(0)) result; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   static const bool value = result::value; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Write the content of buf to os. | ||||||
|  | template <typename Char> | ||||||
|  | void write(std::basic_ostream<Char> &os, basic_buffer<Char> &buf) { | ||||||
|  |   const Char *data = buf.data(); | ||||||
|  |   typedef std::make_unsigned<std::streamsize>::type UnsignedStreamSize; | ||||||
|  |   UnsignedStreamSize size = buf.size(); | ||||||
|  |   UnsignedStreamSize max_size = | ||||||
|  |       internal::to_unsigned((std::numeric_limits<std::streamsize>::max)()); | ||||||
|  |   do { | ||||||
|  |     UnsignedStreamSize n = size <= max_size ? size : max_size; | ||||||
|  |     os.write(data, static_cast<std::streamsize>(n)); | ||||||
|  |     data += n; | ||||||
|  |     size -= n; | ||||||
|  |   } while (size != 0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char, typename T> | ||||||
|  | void format_value(basic_buffer<Char> &buffer, const T &value) { | ||||||
|  |   internal::formatbuf<Char> format_buf(buffer); | ||||||
|  |   std::basic_ostream<Char> output(&format_buf); | ||||||
|  |   output.exceptions(std::ios_base::failbit | std::ios_base::badbit); | ||||||
|  |   output << value; | ||||||
|  |   buffer.resize(buffer.size()); | ||||||
|  | } | ||||||
|  | }  // namespace internal | ||||||
|  |  | ||||||
|  | // Disable conversion to int if T has an overloaded operator<< which is a free | ||||||
|  | // function (not a member of std::ostream). | ||||||
|  | template <typename T, typename Char> | ||||||
|  | struct convert_to_int<T, Char, void> { | ||||||
|  |   static const bool value = | ||||||
|  |     convert_to_int<T, Char, int>::value && | ||||||
|  |     !internal::is_streamable<T, Char>::value; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Formats an object of type T that has an overloaded ostream operator<<. | ||||||
|  | template <typename T, typename Char> | ||||||
|  | struct formatter<T, Char, | ||||||
|  |     typename std::enable_if< | ||||||
|  |       internal::is_streamable<T, Char>::value && | ||||||
|  |       !internal::format_type< | ||||||
|  |         typename buffer_context<Char>::type, T>::value>::type> | ||||||
|  |     : formatter<basic_string_view<Char>, Char> { | ||||||
|  |  | ||||||
|  |   template <typename Context> | ||||||
|  |   auto format(const T &value, Context &ctx) -> decltype(ctx.out()) { | ||||||
|  |     basic_memory_buffer<Char> buffer; | ||||||
|  |     internal::format_value(buffer, value); | ||||||
|  |     basic_string_view<Char> str(buffer.data(), buffer.size()); | ||||||
|  |     return formatter<basic_string_view<Char>, Char>::format(str, ctx); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | inline void vprint(std::basic_ostream<Char> &os, | ||||||
|  |                    basic_string_view<Char> format_str, | ||||||
|  |                    basic_format_args<typename buffer_context<Char>::type> args) { | ||||||
|  |   basic_memory_buffer<Char> buffer; | ||||||
|  |   internal::vformat_to(buffer, format_str, args); | ||||||
|  |   internal::write(os, buffer); | ||||||
|  | } | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Prints formatted data to the stream *os*. | ||||||
|  |  | ||||||
|  |   **Example**:: | ||||||
|  |  | ||||||
|  |     fmt::print(cerr, "Don't {}!", "panic"); | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template <typename S, typename... Args> | ||||||
|  | inline typename std::enable_if<internal::is_string<S>::value>::type | ||||||
|  | print(std::basic_ostream<FMT_CHAR(S)> &os, const S &format_str, | ||||||
|  |       const Args & ... args) { | ||||||
|  |   internal::checked_args<S, Args...> ca(format_str, args...); | ||||||
|  |   vprint(os, to_string_view(format_str), *ca); | ||||||
|  | } | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_OSTREAM_H_ | ||||||
							
								
								
									
										324
									
								
								logger/bundled/fmt/posix.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										324
									
								
								logger/bundled/fmt/posix.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,324 @@ | |||||||
|  | // A C++ interface to POSIX functions. | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - 2016, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_POSIX_H_ | ||||||
|  | #define FMT_POSIX_H_ | ||||||
|  |  | ||||||
|  | #if defined(__MINGW32__) || defined(__CYGWIN__) | ||||||
|  | // Workaround MinGW bug https://sourceforge.net/p/mingw/bugs/2024/. | ||||||
|  | # undef __STRICT_ANSI__ | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #include <errno.h> | ||||||
|  | #include <fcntl.h>   // for O_RDONLY | ||||||
|  | #include <locale.h>  // for locale_t | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h>  // for strtod_l | ||||||
|  |  | ||||||
|  | #include <cstddef> | ||||||
|  |  | ||||||
|  | #if defined __APPLE__ || defined(__FreeBSD__) | ||||||
|  | # include <xlocale.h>  // for LC_NUMERIC_MASK on OS X | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  |  | ||||||
|  | #ifndef FMT_POSIX | ||||||
|  | # if defined(_WIN32) && !defined(__MINGW32__) | ||||||
|  | // Fix warnings about deprecated symbols. | ||||||
|  | #  define FMT_POSIX(call) _##call | ||||||
|  | # else | ||||||
|  | #  define FMT_POSIX(call) call | ||||||
|  | # endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | // Calls to system functions are wrapped in FMT_SYSTEM for testability. | ||||||
|  | #ifdef FMT_SYSTEM | ||||||
|  | # define FMT_POSIX_CALL(call) FMT_SYSTEM(call) | ||||||
|  | #else | ||||||
|  | # define FMT_SYSTEM(call) call | ||||||
|  | # ifdef _WIN32 | ||||||
|  | // Fix warnings about deprecated symbols. | ||||||
|  | #  define FMT_POSIX_CALL(call) ::_##call | ||||||
|  | # else | ||||||
|  | #  define FMT_POSIX_CALL(call) ::call | ||||||
|  | # endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | // Retries the expression while it evaluates to error_result and errno | ||||||
|  | // equals to EINTR. | ||||||
|  | #ifndef _WIN32 | ||||||
|  | # define FMT_RETRY_VAL(result, expression, error_result) \ | ||||||
|  |   do { \ | ||||||
|  |     result = (expression); \ | ||||||
|  |   } while (result == error_result && errno == EINTR) | ||||||
|  | #else | ||||||
|  | # define FMT_RETRY_VAL(result, expression, error_result) result = (expression) | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #define FMT_RETRY(result, expression) FMT_RETRY_VAL(result, expression, -1) | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   A reference to a null-terminated string. It can be constructed from a C | ||||||
|  |   string or ``std::string``. | ||||||
|  |  | ||||||
|  |   You can use one of the following typedefs for common character types: | ||||||
|  |  | ||||||
|  |   +---------------+-----------------------------+ | ||||||
|  |   | Type          | Definition                  | | ||||||
|  |   +===============+=============================+ | ||||||
|  |   | cstring_view  | basic_cstring_view<char>    | | ||||||
|  |   +---------------+-----------------------------+ | ||||||
|  |   | wcstring_view | basic_cstring_view<wchar_t> | | ||||||
|  |   +---------------+-----------------------------+ | ||||||
|  |  | ||||||
|  |   This class is most useful as a parameter type to allow passing | ||||||
|  |   different types of strings to a function, for example:: | ||||||
|  |  | ||||||
|  |     template <typename... Args> | ||||||
|  |     std::string format(cstring_view format_str, const Args & ... args); | ||||||
|  |  | ||||||
|  |     format("{}", 42); | ||||||
|  |     format(std::string("{}"), 42); | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template <typename Char> | ||||||
|  | class basic_cstring_view { | ||||||
|  |  private: | ||||||
|  |   const Char *data_; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   /** Constructs a string reference object from a C string. */ | ||||||
|  |   basic_cstring_view(const Char *s) : data_(s) {} | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |     \rst | ||||||
|  |     Constructs a string reference from an ``std::string`` object. | ||||||
|  |     \endrst | ||||||
|  |    */ | ||||||
|  |   basic_cstring_view(const std::basic_string<Char> &s) : data_(s.c_str()) {} | ||||||
|  |  | ||||||
|  |   /** Returns the pointer to a C string. */ | ||||||
|  |   const Char *c_str() const { return data_; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | typedef basic_cstring_view<char> cstring_view; | ||||||
|  | typedef basic_cstring_view<wchar_t> wcstring_view; | ||||||
|  |  | ||||||
|  | // An error code. | ||||||
|  | class error_code { | ||||||
|  |  private: | ||||||
|  |   int value_; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   explicit error_code(int value = 0) FMT_NOEXCEPT : value_(value) {} | ||||||
|  |  | ||||||
|  |   int get() const FMT_NOEXCEPT { return value_; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // A buffered file. | ||||||
|  | class buffered_file { | ||||||
|  |  private: | ||||||
|  |   FILE *file_; | ||||||
|  |  | ||||||
|  |   friend class file; | ||||||
|  |  | ||||||
|  |   explicit buffered_file(FILE *f) : file_(f) {} | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   // Constructs a buffered_file object which doesn't represent any file. | ||||||
|  |   buffered_file() FMT_NOEXCEPT : file_(FMT_NULL) {} | ||||||
|  |  | ||||||
|  |   // Destroys the object closing the file it represents if any. | ||||||
|  |   FMT_API ~buffered_file() FMT_NOEXCEPT; | ||||||
|  |  | ||||||
|  |  private: | ||||||
|  |   buffered_file(const buffered_file &) = delete; | ||||||
|  |   void operator=(const buffered_file &) = delete; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   buffered_file(buffered_file &&other) FMT_NOEXCEPT : file_(other.file_) { | ||||||
|  |     other.file_ = FMT_NULL; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   buffered_file& operator=(buffered_file &&other) { | ||||||
|  |     close(); | ||||||
|  |     file_ = other.file_; | ||||||
|  |     other.file_ = FMT_NULL; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Opens a file. | ||||||
|  |   FMT_API buffered_file(cstring_view filename, cstring_view mode); | ||||||
|  |  | ||||||
|  |   // Closes the file. | ||||||
|  |   FMT_API void close(); | ||||||
|  |  | ||||||
|  |   // Returns the pointer to a FILE object representing this file. | ||||||
|  |   FILE *get() const FMT_NOEXCEPT { return file_; } | ||||||
|  |  | ||||||
|  |   // We place parentheses around fileno to workaround a bug in some versions | ||||||
|  |   // of MinGW that define fileno as a macro. | ||||||
|  |   FMT_API int (fileno)() const; | ||||||
|  |  | ||||||
|  |   void vprint(string_view format_str, format_args args) { | ||||||
|  |     fmt::vprint(file_, format_str, args); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename... Args> | ||||||
|  |   inline void print(string_view format_str, const Args & ... args) { | ||||||
|  |     vprint(format_str, make_format_args(args...)); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // A file. Closed file is represented by a file object with descriptor -1. | ||||||
|  | // Methods that are not declared with FMT_NOEXCEPT may throw | ||||||
|  | // fmt::system_error in case of failure. Note that some errors such as | ||||||
|  | // closing the file multiple times will cause a crash on Windows rather | ||||||
|  | // than an exception. You can get standard behavior by overriding the | ||||||
|  | // invalid parameter handler with _set_invalid_parameter_handler. | ||||||
|  | class file { | ||||||
|  |  private: | ||||||
|  |   int fd_;  // File descriptor. | ||||||
|  |  | ||||||
|  |   // Constructs a file object with a given descriptor. | ||||||
|  |   explicit file(int fd) : fd_(fd) {} | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   // Possible values for the oflag argument to the constructor. | ||||||
|  |   enum { | ||||||
|  |     RDONLY = FMT_POSIX(O_RDONLY), // Open for reading only. | ||||||
|  |     WRONLY = FMT_POSIX(O_WRONLY), // Open for writing only. | ||||||
|  |     RDWR   = FMT_POSIX(O_RDWR)    // Open for reading and writing. | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // Constructs a file object which doesn't represent any file. | ||||||
|  |   file() FMT_NOEXCEPT : fd_(-1) {} | ||||||
|  |  | ||||||
|  |   // Opens a file and constructs a file object representing this file. | ||||||
|  |   FMT_API file(cstring_view path, int oflag); | ||||||
|  |  | ||||||
|  |  private: | ||||||
|  |   file(const file &) = delete; | ||||||
|  |   void operator=(const file &) = delete; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   file(file &&other) FMT_NOEXCEPT : fd_(other.fd_) { | ||||||
|  |     other.fd_ = -1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   file& operator=(file &&other) { | ||||||
|  |     close(); | ||||||
|  |     fd_ = other.fd_; | ||||||
|  |     other.fd_ = -1; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Destroys the object closing the file it represents if any. | ||||||
|  |   FMT_API ~file() FMT_NOEXCEPT; | ||||||
|  |  | ||||||
|  |   // Returns the file descriptor. | ||||||
|  |   int descriptor() const FMT_NOEXCEPT { return fd_; } | ||||||
|  |  | ||||||
|  |   // Closes the file. | ||||||
|  |   FMT_API void close(); | ||||||
|  |  | ||||||
|  |   // Returns the file size. The size has signed type for consistency with | ||||||
|  |   // stat::st_size. | ||||||
|  |   FMT_API long long size() const; | ||||||
|  |  | ||||||
|  |   // Attempts to read count bytes from the file into the specified buffer. | ||||||
|  |   FMT_API std::size_t read(void *buffer, std::size_t count); | ||||||
|  |  | ||||||
|  |   // Attempts to write count bytes from the specified buffer to the file. | ||||||
|  |   FMT_API std::size_t write(const void *buffer, std::size_t count); | ||||||
|  |  | ||||||
|  |   // Duplicates a file descriptor with the dup function and returns | ||||||
|  |   // the duplicate as a file object. | ||||||
|  |   FMT_API static file dup(int fd); | ||||||
|  |  | ||||||
|  |   // Makes fd be the copy of this file descriptor, closing fd first if | ||||||
|  |   // necessary. | ||||||
|  |   FMT_API void dup2(int fd); | ||||||
|  |  | ||||||
|  |   // Makes fd be the copy of this file descriptor, closing fd first if | ||||||
|  |   // necessary. | ||||||
|  |   FMT_API void dup2(int fd, error_code &ec) FMT_NOEXCEPT; | ||||||
|  |  | ||||||
|  |   // Creates a pipe setting up read_end and write_end file objects for reading | ||||||
|  |   // and writing respectively. | ||||||
|  |   FMT_API static void pipe(file &read_end, file &write_end); | ||||||
|  |  | ||||||
|  |   // Creates a buffered_file object associated with this file and detaches | ||||||
|  |   // this file object from the file. | ||||||
|  |   FMT_API buffered_file fdopen(const char *mode); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Returns the memory page size. | ||||||
|  | long getpagesize(); | ||||||
|  |  | ||||||
|  | #if (defined(LC_NUMERIC_MASK) || defined(_MSC_VER)) && \ | ||||||
|  |     !defined(__ANDROID__) && !defined(__CYGWIN__) && !defined(__OpenBSD__) && \ | ||||||
|  |     !defined(__NEWLIB_H__) | ||||||
|  | # define FMT_LOCALE | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef FMT_LOCALE | ||||||
|  | // A "C" numeric locale. | ||||||
|  | class Locale { | ||||||
|  |  private: | ||||||
|  | # ifdef _MSC_VER | ||||||
|  |   typedef _locale_t locale_t; | ||||||
|  |  | ||||||
|  |   enum { LC_NUMERIC_MASK = LC_NUMERIC }; | ||||||
|  |  | ||||||
|  |   static locale_t newlocale(int category_mask, const char *locale, locale_t) { | ||||||
|  |     return _create_locale(category_mask, locale); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static void freelocale(locale_t locale) { | ||||||
|  |     _free_locale(locale); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static double strtod_l(const char *nptr, char **endptr, _locale_t locale) { | ||||||
|  |     return _strtod_l(nptr, endptr, locale); | ||||||
|  |   } | ||||||
|  | # endif | ||||||
|  |  | ||||||
|  |   locale_t locale_; | ||||||
|  |  | ||||||
|  |   Locale(const Locale &) = delete; | ||||||
|  |   void operator=(const Locale &) = delete; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   typedef locale_t Type; | ||||||
|  |  | ||||||
|  |   Locale() : locale_(newlocale(LC_NUMERIC_MASK, "C", FMT_NULL)) { | ||||||
|  |     if (!locale_) | ||||||
|  |       FMT_THROW(system_error(errno, "cannot create locale")); | ||||||
|  |   } | ||||||
|  |   ~Locale() { freelocale(locale_); } | ||||||
|  |  | ||||||
|  |   Type get() const { return locale_; } | ||||||
|  |  | ||||||
|  |   // Converts string to floating-point number and advances str past the end | ||||||
|  |   // of the parsed input. | ||||||
|  |   double strtod(const char *&str) const { | ||||||
|  |     char *end = FMT_NULL; | ||||||
|  |     double result = strtod_l(str, &end, locale_); | ||||||
|  |     str = end; | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | #endif  // FMT_LOCALE | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_POSIX_H_ | ||||||
							
								
								
									
										855
									
								
								logger/bundled/fmt/printf.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										855
									
								
								logger/bundled/fmt/printf.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,855 @@ | |||||||
|  | // Formatting library for C++ | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - 2016, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_PRINTF_H_ | ||||||
|  | #define FMT_PRINTF_H_ | ||||||
|  |  | ||||||
|  | #include <algorithm>  // std::fill_n | ||||||
|  | #include <limits>     // std::numeric_limits | ||||||
|  |  | ||||||
|  | #include "ostream.h" | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  | namespace internal { | ||||||
|  |  | ||||||
|  | // An iterator that produces a null terminator on *end. This simplifies parsing | ||||||
|  | // and allows comparing the performance of processing a null-terminated string | ||||||
|  | // vs string_view. | ||||||
|  | template <typename Char> | ||||||
|  | class null_terminating_iterator { | ||||||
|  |  public: | ||||||
|  |   typedef std::ptrdiff_t difference_type; | ||||||
|  |   typedef Char value_type; | ||||||
|  |   typedef const Char* pointer; | ||||||
|  |   typedef const Char& reference; | ||||||
|  |   typedef std::random_access_iterator_tag iterator_category; | ||||||
|  |  | ||||||
|  |   null_terminating_iterator() : ptr_(0), end_(0) {} | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator(const Char *ptr, const Char *end) | ||||||
|  |     : ptr_(ptr), end_(end) {} | ||||||
|  |  | ||||||
|  |   template <typename Range> | ||||||
|  |   FMT_CONSTEXPR explicit null_terminating_iterator(const Range &r) | ||||||
|  |     : ptr_(r.begin()), end_(r.end()) {} | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator &operator=(const Char *ptr) { | ||||||
|  |     assert(ptr <= end_); | ||||||
|  |     ptr_ = ptr; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR Char operator*() const { | ||||||
|  |     return ptr_ != end_ ? *ptr_ : Char(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator operator++() { | ||||||
|  |     ++ptr_; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator operator++(int) { | ||||||
|  |     null_terminating_iterator result(*this); | ||||||
|  |     ++ptr_; | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator operator--() { | ||||||
|  |     --ptr_; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator operator+(difference_type n) { | ||||||
|  |     return null_terminating_iterator(ptr_ + n, end_); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator operator-(difference_type n) { | ||||||
|  |     return null_terminating_iterator(ptr_ - n, end_); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR null_terminating_iterator operator+=(difference_type n) { | ||||||
|  |     ptr_ += n; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR difference_type operator-( | ||||||
|  |       null_terminating_iterator other) const { | ||||||
|  |     return ptr_ - other.ptr_; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FMT_CONSTEXPR bool operator!=(null_terminating_iterator other) const { | ||||||
|  |     return ptr_ != other.ptr_; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   bool operator>=(null_terminating_iterator other) const { | ||||||
|  |     return ptr_ >= other.ptr_; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // This should be a friend specialization pointer_from<Char> but the latter | ||||||
|  |   // doesn't compile by gcc 5.1 due to a compiler bug. | ||||||
|  |   template <typename CharT> | ||||||
|  |   friend FMT_CONSTEXPR_DECL const CharT *pointer_from( | ||||||
|  |       null_terminating_iterator<CharT> it); | ||||||
|  |  | ||||||
|  |  private: | ||||||
|  |   const Char *ptr_; | ||||||
|  |   const Char *end_; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | FMT_CONSTEXPR const T *pointer_from(const T *p) { return p; } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | FMT_CONSTEXPR const Char *pointer_from(null_terminating_iterator<Char> it) { | ||||||
|  |   return it.ptr_; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // DEPRECATED: Parses the input as an unsigned integer. This function assumes | ||||||
|  | // that the first character is a digit and presence of a non-digit character at | ||||||
|  | // the end. | ||||||
|  | // it: an iterator pointing to the beginning of the input range. | ||||||
|  | template <typename Iterator, typename ErrorHandler> | ||||||
|  | FMT_CONSTEXPR unsigned parse_nonnegative_int(Iterator &it, ErrorHandler &&eh) { | ||||||
|  |   assert('0' <= *it && *it <= '9'); | ||||||
|  |   if (*it == '0') { | ||||||
|  |     ++it; | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   unsigned value = 0; | ||||||
|  |   // Convert to unsigned to prevent a warning. | ||||||
|  |   unsigned max_int = (std::numeric_limits<int>::max)(); | ||||||
|  |   unsigned big = max_int / 10; | ||||||
|  |   do { | ||||||
|  |     // Check for overflow. | ||||||
|  |     if (value > big) { | ||||||
|  |       value = max_int + 1; | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     value = value * 10 + unsigned(*it - '0'); | ||||||
|  |     // Workaround for MSVC "setup_exception stack overflow" error: | ||||||
|  |     auto next = it; | ||||||
|  |     ++next; | ||||||
|  |     it = next; | ||||||
|  |   } while ('0' <= *it && *it <= '9'); | ||||||
|  |   if (value > max_int) | ||||||
|  |     eh.on_error("number is too big"); | ||||||
|  |   return value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Checks if a value fits in int - used to avoid warnings about comparing | ||||||
|  | // signed and unsigned integers. | ||||||
|  | template <bool IsSigned> | ||||||
|  | struct int_checker { | ||||||
|  |   template <typename T> | ||||||
|  |   static bool fits_in_int(T value) { | ||||||
|  |     unsigned max = std::numeric_limits<int>::max(); | ||||||
|  |     return value <= max; | ||||||
|  |   } | ||||||
|  |   static bool fits_in_int(bool) { return true; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <> | ||||||
|  | struct int_checker<true> { | ||||||
|  |   template <typename T> | ||||||
|  |   static bool fits_in_int(T value) { | ||||||
|  |     return value >= std::numeric_limits<int>::min() && | ||||||
|  |            value <= std::numeric_limits<int>::max(); | ||||||
|  |   } | ||||||
|  |   static bool fits_in_int(int) { return true; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | class printf_precision_handler: public function<int> { | ||||||
|  |  public: | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<std::is_integral<T>::value, int>::type | ||||||
|  |       operator()(T value) { | ||||||
|  |     if (!int_checker<std::numeric_limits<T>::is_signed>::fits_in_int(value)) | ||||||
|  |       FMT_THROW(format_error("number is too big")); | ||||||
|  |     return static_cast<int>(value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<!std::is_integral<T>::value, int>::type operator()(T) { | ||||||
|  |     FMT_THROW(format_error("precision is not integer")); | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // An argument visitor that returns true iff arg is a zero integer. | ||||||
|  | class is_zero_int: public function<bool> { | ||||||
|  |  public: | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<std::is_integral<T>::value, bool>::type | ||||||
|  |       operator()(T value) { return value == 0; } | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<!std::is_integral<T>::value, bool>::type | ||||||
|  |       operator()(T) { return false; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | struct make_unsigned_or_bool : std::make_unsigned<T> {}; | ||||||
|  |  | ||||||
|  | template <> | ||||||
|  | struct make_unsigned_or_bool<bool> { | ||||||
|  |   typedef bool type; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T, typename Context> | ||||||
|  | class arg_converter: public function<void> { | ||||||
|  |  private: | ||||||
|  |   typedef typename Context::char_type Char; | ||||||
|  |  | ||||||
|  |   basic_format_arg<Context> &arg_; | ||||||
|  |   typename Context::char_type type_; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   arg_converter(basic_format_arg<Context> &arg, Char type) | ||||||
|  |     : arg_(arg), type_(type) {} | ||||||
|  |  | ||||||
|  |   void operator()(bool value) { | ||||||
|  |     if (type_ != 's') | ||||||
|  |       operator()<bool>(value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename U> | ||||||
|  |   typename std::enable_if<std::is_integral<U>::value>::type | ||||||
|  |       operator()(U value) { | ||||||
|  |     bool is_signed = type_ == 'd' || type_ == 'i'; | ||||||
|  |     typedef typename std::conditional< | ||||||
|  |         std::is_same<T, void>::value, U, T>::type TargetType; | ||||||
|  |     if (const_check(sizeof(TargetType) <= sizeof(int))) { | ||||||
|  |       // Extra casts are used to silence warnings. | ||||||
|  |       if (is_signed) { | ||||||
|  |         arg_ = internal::make_arg<Context>( | ||||||
|  |           static_cast<int>(static_cast<TargetType>(value))); | ||||||
|  |       } else { | ||||||
|  |         typedef typename make_unsigned_or_bool<TargetType>::type Unsigned; | ||||||
|  |         arg_ = internal::make_arg<Context>( | ||||||
|  |           static_cast<unsigned>(static_cast<Unsigned>(value))); | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       if (is_signed) { | ||||||
|  |         // glibc's printf doesn't sign extend arguments of smaller types: | ||||||
|  |         //   std::printf("%lld", -42);  // prints "4294967254" | ||||||
|  |         // but we don't have to do the same because it's a UB. | ||||||
|  |         arg_ = internal::make_arg<Context>(static_cast<long long>(value)); | ||||||
|  |       } else { | ||||||
|  |         arg_ = internal::make_arg<Context>( | ||||||
|  |           static_cast<typename make_unsigned_or_bool<U>::type>(value)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename U> | ||||||
|  |   typename std::enable_if<!std::is_integral<U>::value>::type operator()(U) { | ||||||
|  |     // No coversion needed for non-integral types. | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Converts an integer argument to T for printf, if T is an integral type. | ||||||
|  | // If T is void, the argument is converted to corresponding signed or unsigned | ||||||
|  | // type depending on the type specifier: 'd' and 'i' - signed, other - | ||||||
|  | // unsigned). | ||||||
|  | template <typename T, typename Context, typename Char> | ||||||
|  | void convert_arg(basic_format_arg<Context> &arg, Char type) { | ||||||
|  |   visit_format_arg(arg_converter<T, Context>(arg, type), arg); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Converts an integer argument to char for printf. | ||||||
|  | template <typename Context> | ||||||
|  | class char_converter: public function<void> { | ||||||
|  |  private: | ||||||
|  |   basic_format_arg<Context> &arg_; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   explicit char_converter(basic_format_arg<Context> &arg) : arg_(arg) {} | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<std::is_integral<T>::value>::type | ||||||
|  |       operator()(T value) { | ||||||
|  |     typedef typename Context::char_type Char; | ||||||
|  |     arg_ = internal::make_arg<Context>(static_cast<Char>(value)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<!std::is_integral<T>::value>::type operator()(T) { | ||||||
|  |     // No coversion needed for non-integral types. | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Checks if an argument is a valid printf width specifier and sets | ||||||
|  | // left alignment if it is negative. | ||||||
|  | template <typename Char> | ||||||
|  | class printf_width_handler: public function<unsigned> { | ||||||
|  |  private: | ||||||
|  |   typedef basic_format_specs<Char> format_specs; | ||||||
|  |  | ||||||
|  |   format_specs &spec_; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   explicit printf_width_handler(format_specs &spec) : spec_(spec) {} | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<std::is_integral<T>::value, unsigned>::type | ||||||
|  |       operator()(T value) { | ||||||
|  |     typedef typename internal::int_traits<T>::main_type UnsignedType; | ||||||
|  |     UnsignedType width = static_cast<UnsignedType>(value); | ||||||
|  |     if (internal::is_negative(value)) { | ||||||
|  |       spec_.align_ = ALIGN_LEFT; | ||||||
|  |       width = 0 - width; | ||||||
|  |     } | ||||||
|  |     unsigned int_max = std::numeric_limits<int>::max(); | ||||||
|  |     if (width > int_max) | ||||||
|  |       FMT_THROW(format_error("number is too big")); | ||||||
|  |     return static_cast<unsigned>(width); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<!std::is_integral<T>::value, unsigned>::type | ||||||
|  |       operator()(T) { | ||||||
|  |     FMT_THROW(format_error("width is not integer")); | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char, typename Context> | ||||||
|  | void printf(basic_buffer<Char> &buf, basic_string_view<Char> format, | ||||||
|  |             basic_format_args<Context> args) { | ||||||
|  |   Context(std::back_inserter(buf), format, args).format(); | ||||||
|  | } | ||||||
|  | }  // namespace internal | ||||||
|  |  | ||||||
|  | using internal::printf;  // For printing into memory_buffer. | ||||||
|  |  | ||||||
|  | template <typename Range> | ||||||
|  | class printf_arg_formatter; | ||||||
|  |  | ||||||
|  | template < | ||||||
|  |     typename OutputIt, typename Char, | ||||||
|  |     typename ArgFormatter = | ||||||
|  |       printf_arg_formatter<back_insert_range<internal::basic_buffer<Char>>>> | ||||||
|  | class basic_printf_context; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   The ``printf`` argument formatter. | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template <typename Range> | ||||||
|  | class printf_arg_formatter: | ||||||
|  |   public internal::function< | ||||||
|  |     typename internal::arg_formatter_base<Range>::iterator>, | ||||||
|  |   public internal::arg_formatter_base<Range> { | ||||||
|  |  private: | ||||||
|  |   typedef typename Range::value_type char_type; | ||||||
|  |   typedef decltype(internal::declval<Range>().begin()) iterator; | ||||||
|  |   typedef internal::arg_formatter_base<Range> base; | ||||||
|  |   typedef basic_printf_context<iterator, char_type> context_type; | ||||||
|  |  | ||||||
|  |   context_type &context_; | ||||||
|  |  | ||||||
|  |   void write_null_pointer(char) { | ||||||
|  |     this->spec()->type = 0; | ||||||
|  |     this->write("(nil)"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void write_null_pointer(wchar_t) { | ||||||
|  |     this->spec()->type = 0; | ||||||
|  |     this->write(L"(nil)"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   typedef typename base::format_specs format_specs; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |     \rst | ||||||
|  |     Constructs an argument formatter object. | ||||||
|  |     *buffer* is a reference to the output buffer and *spec* contains format | ||||||
|  |     specifier information for standard argument types. | ||||||
|  |     \endrst | ||||||
|  |    */ | ||||||
|  |   printf_arg_formatter(internal::basic_buffer<char_type> &buffer, | ||||||
|  |                        format_specs &spec, context_type &ctx) | ||||||
|  |     : base(back_insert_range<internal::basic_buffer<char_type>>(buffer), &spec, | ||||||
|  |            ctx.locale()), | ||||||
|  |       context_(ctx) {} | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<std::is_integral<T>::value, iterator>::type | ||||||
|  |       operator()(T value) { | ||||||
|  |     // MSVC2013 fails to compile separate overloads for bool and char_type so | ||||||
|  |     // use std::is_same instead. | ||||||
|  |     if (std::is_same<T, bool>::value) { | ||||||
|  |       format_specs &fmt_spec = *this->spec(); | ||||||
|  |       if (fmt_spec.type != 's') | ||||||
|  |         return base::operator()(value ? 1 : 0); | ||||||
|  |       fmt_spec.type = 0; | ||||||
|  |       this->write(value != 0); | ||||||
|  |     } else if (std::is_same<T, char_type>::value) { | ||||||
|  |       format_specs &fmt_spec = *this->spec(); | ||||||
|  |       if (fmt_spec.type && fmt_spec.type != 'c') | ||||||
|  |         return (*this)(static_cast<int>(value)); | ||||||
|  |       fmt_spec.flags = 0; | ||||||
|  |       fmt_spec.align_ = ALIGN_RIGHT; | ||||||
|  |       return base::operator()(value); | ||||||
|  |     } else { | ||||||
|  |       return base::operator()(value); | ||||||
|  |     } | ||||||
|  |     return this->out(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   typename std::enable_if<std::is_floating_point<T>::value, iterator>::type | ||||||
|  |       operator()(T value) { | ||||||
|  |     return base::operator()(value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** Formats a null-terminated C string. */ | ||||||
|  |   iterator operator()(const char *value) { | ||||||
|  |     if (value) | ||||||
|  |       base::operator()(value); | ||||||
|  |     else if (this->spec()->type == 'p') | ||||||
|  |       write_null_pointer(char_type()); | ||||||
|  |     else | ||||||
|  |       this->write("(null)"); | ||||||
|  |     return this->out(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** Formats a null-terminated wide C string. */ | ||||||
|  |   iterator operator()(const wchar_t *value) { | ||||||
|  |     if (value) | ||||||
|  |       base::operator()(value); | ||||||
|  |     else if (this->spec()->type == 'p') | ||||||
|  |       write_null_pointer(char_type()); | ||||||
|  |     else | ||||||
|  |       this->write(L"(null)"); | ||||||
|  |     return this->out(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   iterator operator()(basic_string_view<char_type> value) { | ||||||
|  |     return base::operator()(value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   iterator operator()(monostate value) { | ||||||
|  |     return base::operator()(value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** Formats a pointer. */ | ||||||
|  |   iterator operator()(const void *value) { | ||||||
|  |     if (value) | ||||||
|  |       return base::operator()(value); | ||||||
|  |     this->spec()->type = 0; | ||||||
|  |     write_null_pointer(char_type()); | ||||||
|  |     return this->out(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** Formats an argument of a custom (user-defined) type. */ | ||||||
|  |   iterator operator()(typename basic_format_arg<context_type>::handle handle) { | ||||||
|  |     handle.format(context_); | ||||||
|  |     return this->out(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | struct printf_formatter { | ||||||
|  |   template <typename ParseContext> | ||||||
|  |   auto parse(ParseContext &ctx) -> decltype(ctx.begin()) { return ctx.begin(); } | ||||||
|  |  | ||||||
|  |   template <typename FormatContext> | ||||||
|  |   auto format(const T &value, FormatContext &ctx) -> decltype(ctx.out()) { | ||||||
|  |     internal::format_value(internal::get_container(ctx.out()), value); | ||||||
|  |     return ctx.out(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** This template formats data and writes the output to a writer. */ | ||||||
|  | template <typename OutputIt, typename Char, typename ArgFormatter> | ||||||
|  | class basic_printf_context : | ||||||
|  |   // Inherit publicly as a workaround for the icc bug | ||||||
|  |   // https://software.intel.com/en-us/forums/intel-c-compiler/topic/783476. | ||||||
|  |   public internal::context_base< | ||||||
|  |     OutputIt, basic_printf_context<OutputIt, Char, ArgFormatter>, Char> { | ||||||
|  |  public: | ||||||
|  |   /** The character type for the output. */ | ||||||
|  |   typedef Char char_type; | ||||||
|  |  | ||||||
|  |   template <typename T> | ||||||
|  |   struct formatter_type { typedef printf_formatter<T> type; }; | ||||||
|  |  | ||||||
|  |  private: | ||||||
|  |   typedef internal::context_base<OutputIt, basic_printf_context, Char> base; | ||||||
|  |   typedef typename base::format_arg format_arg; | ||||||
|  |   typedef basic_format_specs<char_type> format_specs; | ||||||
|  |   typedef internal::null_terminating_iterator<char_type> iterator; | ||||||
|  |  | ||||||
|  |   void parse_flags(format_specs &spec, iterator &it); | ||||||
|  |  | ||||||
|  |   // Returns the argument with specified index or, if arg_index is equal | ||||||
|  |   // to the maximum unsigned value, the next argument. | ||||||
|  |   format_arg get_arg( | ||||||
|  |       iterator it, | ||||||
|  |       unsigned arg_index = (std::numeric_limits<unsigned>::max)()); | ||||||
|  |  | ||||||
|  |   // Parses argument index, flags and width and returns the argument index. | ||||||
|  |   unsigned parse_header(iterator &it, format_specs &spec); | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   /** | ||||||
|  |    \rst | ||||||
|  |    Constructs a ``printf_context`` object. References to the arguments and | ||||||
|  |    the writer are stored in the context object so make sure they have | ||||||
|  |    appropriate lifetimes. | ||||||
|  |    \endrst | ||||||
|  |    */ | ||||||
|  |   basic_printf_context(OutputIt out, basic_string_view<char_type> format_str, | ||||||
|  |                        basic_format_args<basic_printf_context> args) | ||||||
|  |     : base(out, format_str, args) {} | ||||||
|  |  | ||||||
|  |   using base::parse_context; | ||||||
|  |   using base::out; | ||||||
|  |   using base::advance_to; | ||||||
|  |  | ||||||
|  |   /** Formats stored arguments and writes the output to the range. */ | ||||||
|  |   void format(); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename OutputIt, typename Char, typename AF> | ||||||
|  | void basic_printf_context<OutputIt, Char, AF>::parse_flags( | ||||||
|  |     format_specs &spec, iterator &it) { | ||||||
|  |   for (;;) { | ||||||
|  |     switch (*it++) { | ||||||
|  |       case '-': | ||||||
|  |         spec.align_ = ALIGN_LEFT; | ||||||
|  |         break; | ||||||
|  |       case '+': | ||||||
|  |         spec.flags |= SIGN_FLAG | PLUS_FLAG; | ||||||
|  |         break; | ||||||
|  |       case '0': | ||||||
|  |         spec.fill_ = '0'; | ||||||
|  |         break; | ||||||
|  |       case ' ': | ||||||
|  |         spec.flags |= SIGN_FLAG; | ||||||
|  |         break; | ||||||
|  |       case '#': | ||||||
|  |         spec.flags |= HASH_FLAG; | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         --it; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename OutputIt, typename Char, typename AF> | ||||||
|  | typename basic_printf_context<OutputIt, Char, AF>::format_arg | ||||||
|  |   basic_printf_context<OutputIt, Char, AF>::get_arg( | ||||||
|  |     iterator it, unsigned arg_index) { | ||||||
|  |   (void)it; | ||||||
|  |   if (arg_index == std::numeric_limits<unsigned>::max()) | ||||||
|  |     return this->do_get_arg(this->parse_context().next_arg_id()); | ||||||
|  |   return base::get_arg(arg_index - 1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename OutputIt, typename Char, typename AF> | ||||||
|  | unsigned basic_printf_context<OutputIt, Char, AF>::parse_header( | ||||||
|  |   iterator &it, format_specs &spec) { | ||||||
|  |   unsigned arg_index = std::numeric_limits<unsigned>::max(); | ||||||
|  |   char_type c = *it; | ||||||
|  |   if (c >= '0' && c <= '9') { | ||||||
|  |     // Parse an argument index (if followed by '$') or a width possibly | ||||||
|  |     // preceded with '0' flag(s). | ||||||
|  |     internal::error_handler eh; | ||||||
|  |     unsigned value = parse_nonnegative_int(it, eh); | ||||||
|  |     if (*it == '$') {  // value is an argument index | ||||||
|  |       ++it; | ||||||
|  |       arg_index = value; | ||||||
|  |     } else { | ||||||
|  |       if (c == '0') | ||||||
|  |         spec.fill_ = '0'; | ||||||
|  |       if (value != 0) { | ||||||
|  |         // Nonzero value means that we parsed width and don't need to | ||||||
|  |         // parse it or flags again, so return now. | ||||||
|  |         spec.width_ = value; | ||||||
|  |         return arg_index; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   parse_flags(spec, it); | ||||||
|  |   // Parse width. | ||||||
|  |   if (*it >= '0' && *it <= '9') { | ||||||
|  |     internal::error_handler eh; | ||||||
|  |     spec.width_ = parse_nonnegative_int(it, eh); | ||||||
|  |   } else if (*it == '*') { | ||||||
|  |     ++it; | ||||||
|  |     spec.width_ = visit_format_arg( | ||||||
|  |           internal::printf_width_handler<char_type>(spec), get_arg(it)); | ||||||
|  |   } | ||||||
|  |   return arg_index; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename OutputIt, typename Char, typename AF> | ||||||
|  | void basic_printf_context<OutputIt, Char, AF>::format() { | ||||||
|  |   auto &buffer = internal::get_container(this->out()); | ||||||
|  |   auto start = iterator(this->parse_context()); | ||||||
|  |   auto it = start; | ||||||
|  |   using internal::pointer_from; | ||||||
|  |   while (*it) { | ||||||
|  |     char_type c = *it++; | ||||||
|  |     if (c != '%') continue; | ||||||
|  |     if (*it == c) { | ||||||
|  |       buffer.append(pointer_from(start), pointer_from(it)); | ||||||
|  |       start = ++it; | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     buffer.append(pointer_from(start), pointer_from(it) - 1); | ||||||
|  |  | ||||||
|  |     format_specs spec; | ||||||
|  |     spec.align_ = ALIGN_RIGHT; | ||||||
|  |  | ||||||
|  |     // Parse argument index, flags and width. | ||||||
|  |     unsigned arg_index = parse_header(it, spec); | ||||||
|  |  | ||||||
|  |     // Parse precision. | ||||||
|  |     if (*it == '.') { | ||||||
|  |       ++it; | ||||||
|  |       if ('0' <= *it && *it <= '9') { | ||||||
|  |         internal::error_handler eh; | ||||||
|  |         spec.precision = static_cast<int>(parse_nonnegative_int(it, eh)); | ||||||
|  |       } else if (*it == '*') { | ||||||
|  |         ++it; | ||||||
|  |         spec.precision = | ||||||
|  |             visit_format_arg(internal::printf_precision_handler(), get_arg(it)); | ||||||
|  |       } else { | ||||||
|  |         spec.precision = 0; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     format_arg arg = get_arg(it, arg_index); | ||||||
|  |     if (spec.has(HASH_FLAG) && visit_format_arg(internal::is_zero_int(), arg)) | ||||||
|  |       spec.flags = static_cast<uint_least8_t>(spec.flags & (~internal::to_unsigned<int>(HASH_FLAG))); | ||||||
|  |     if (spec.fill_ == '0') { | ||||||
|  |       if (arg.is_arithmetic()) | ||||||
|  |         spec.align_ = ALIGN_NUMERIC; | ||||||
|  |       else | ||||||
|  |         spec.fill_ = ' ';  // Ignore '0' flag for non-numeric types. | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Parse length and convert the argument to the required type. | ||||||
|  |     using internal::convert_arg; | ||||||
|  |     switch (*it++) { | ||||||
|  |     case 'h': | ||||||
|  |       if (*it == 'h') | ||||||
|  |         convert_arg<signed char>(arg, *++it); | ||||||
|  |       else | ||||||
|  |         convert_arg<short>(arg, *it); | ||||||
|  |       break; | ||||||
|  |     case 'l': | ||||||
|  |       if (*it == 'l') | ||||||
|  |         convert_arg<long long>(arg, *++it); | ||||||
|  |       else | ||||||
|  |         convert_arg<long>(arg, *it); | ||||||
|  |       break; | ||||||
|  |     case 'j': | ||||||
|  |       convert_arg<intmax_t>(arg, *it); | ||||||
|  |       break; | ||||||
|  |     case 'z': | ||||||
|  |       convert_arg<std::size_t>(arg, *it); | ||||||
|  |       break; | ||||||
|  |     case 't': | ||||||
|  |       convert_arg<std::ptrdiff_t>(arg, *it); | ||||||
|  |       break; | ||||||
|  |     case 'L': | ||||||
|  |       // printf produces garbage when 'L' is omitted for long double, no | ||||||
|  |       // need to do the same. | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       --it; | ||||||
|  |       convert_arg<void>(arg, *it); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Parse type. | ||||||
|  |     if (!*it) | ||||||
|  |       FMT_THROW(format_error("invalid format string")); | ||||||
|  |     spec.type = static_cast<char>(*it++); | ||||||
|  |     if (arg.is_integral()) { | ||||||
|  |       // Normalize type. | ||||||
|  |       switch (spec.type) { | ||||||
|  |       case 'i': case 'u': | ||||||
|  |         spec.type = 'd'; | ||||||
|  |         break; | ||||||
|  |       case 'c': | ||||||
|  |         // TODO: handle wchar_t better? | ||||||
|  |         visit_format_arg( | ||||||
|  |               internal::char_converter<basic_printf_context>(arg), arg); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     start = it; | ||||||
|  |  | ||||||
|  |     // Format argument. | ||||||
|  |     visit_format_arg(AF(buffer, spec, *this), arg); | ||||||
|  |   } | ||||||
|  |   buffer.append(pointer_from(start), pointer_from(it)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Buffer> | ||||||
|  | struct basic_printf_context_t { | ||||||
|  |   typedef basic_printf_context< | ||||||
|  |     std::back_insert_iterator<Buffer>, typename Buffer::value_type> type; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | typedef basic_printf_context_t<internal::buffer>::type printf_context; | ||||||
|  | typedef basic_printf_context_t<internal::wbuffer>::type wprintf_context; | ||||||
|  |  | ||||||
|  | typedef basic_format_args<printf_context> printf_args; | ||||||
|  | typedef basic_format_args<wprintf_context> wprintf_args; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Constructs an `~fmt::format_arg_store` object that contains references to | ||||||
|  |   arguments and can be implicitly converted to `~fmt::printf_args`.  | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template<typename... Args> | ||||||
|  | inline format_arg_store<printf_context, Args...> | ||||||
|  |   make_printf_args(const Args &... args) { return {args...}; } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Constructs an `~fmt::format_arg_store` object that contains references to | ||||||
|  |   arguments and can be implicitly converted to `~fmt::wprintf_args`.  | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template<typename... Args> | ||||||
|  | inline format_arg_store<wprintf_context, Args...> | ||||||
|  |   make_wprintf_args(const Args &... args) { return {args...}; } | ||||||
|  |  | ||||||
|  | template <typename S, typename Char = FMT_CHAR(S)> | ||||||
|  | inline std::basic_string<Char> | ||||||
|  | vsprintf(const S &format, | ||||||
|  |          basic_format_args<typename basic_printf_context_t< | ||||||
|  |            internal::basic_buffer<Char>>::type> args) { | ||||||
|  |   basic_memory_buffer<Char> buffer; | ||||||
|  |   printf(buffer, to_string_view(format), args); | ||||||
|  |   return to_string(buffer); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Formats arguments and returns the result as a string. | ||||||
|  |  | ||||||
|  |   **Example**:: | ||||||
|  |  | ||||||
|  |     std::string message = fmt::sprintf("The answer is %d", 42); | ||||||
|  |   \endrst | ||||||
|  | */ | ||||||
|  | template <typename S, typename... Args> | ||||||
|  | inline FMT_ENABLE_IF_T( | ||||||
|  |     internal::is_string<S>::value, std::basic_string<FMT_CHAR(S)>) | ||||||
|  |     sprintf(const S &format, const Args & ... args) { | ||||||
|  |   internal::check_format_string<Args...>(format); | ||||||
|  |   typedef internal::basic_buffer<FMT_CHAR(S)> buffer; | ||||||
|  |   typedef typename basic_printf_context_t<buffer>::type context; | ||||||
|  |   format_arg_store<context, Args...> as{ args... }; | ||||||
|  |   return vsprintf(to_string_view(format), | ||||||
|  |                   basic_format_args<context>(as)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename S, typename Char = FMT_CHAR(S)> | ||||||
|  | inline int vfprintf(std::FILE *f, const S &format, | ||||||
|  |                     basic_format_args<typename basic_printf_context_t< | ||||||
|  |                       internal::basic_buffer<Char>>::type> args) { | ||||||
|  |   basic_memory_buffer<Char> buffer; | ||||||
|  |   printf(buffer, to_string_view(format), args); | ||||||
|  |   std::size_t size = buffer.size(); | ||||||
|  |   return std::fwrite( | ||||||
|  |     buffer.data(), sizeof(Char), size, f) < size ? -1 : static_cast<int>(size); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Prints formatted data to the file *f*. | ||||||
|  |  | ||||||
|  |   **Example**:: | ||||||
|  |  | ||||||
|  |     fmt::fprintf(stderr, "Don't %s!", "panic"); | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template <typename S, typename... Args> | ||||||
|  | inline FMT_ENABLE_IF_T(internal::is_string<S>::value, int) | ||||||
|  |     fprintf(std::FILE *f, const S &format, const Args & ... args) { | ||||||
|  |   internal::check_format_string<Args...>(format); | ||||||
|  |   typedef internal::basic_buffer<FMT_CHAR(S)> buffer; | ||||||
|  |   typedef typename basic_printf_context_t<buffer>::type context; | ||||||
|  |   format_arg_store<context, Args...> as{ args... }; | ||||||
|  |   return vfprintf(f, to_string_view(format), | ||||||
|  |                   basic_format_args<context>(as)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename S, typename Char = FMT_CHAR(S)> | ||||||
|  | inline int vprintf(const S &format, | ||||||
|  |                    basic_format_args<typename basic_printf_context_t< | ||||||
|  |                     internal::basic_buffer<Char>>::type> args) { | ||||||
|  |   return vfprintf(stdout, to_string_view(format), args); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Prints formatted data to ``stdout``. | ||||||
|  |  | ||||||
|  |   **Example**:: | ||||||
|  |  | ||||||
|  |     fmt::printf("Elapsed time: %.2f seconds", 1.23); | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template <typename S, typename... Args> | ||||||
|  | inline FMT_ENABLE_IF_T(internal::is_string<S>::value, int) | ||||||
|  |     printf(const S &format_str, const Args & ... args) { | ||||||
|  |   internal::check_format_string<Args...>(format_str); | ||||||
|  |   typedef internal::basic_buffer<FMT_CHAR(S)> buffer; | ||||||
|  |   typedef typename basic_printf_context_t<buffer>::type context; | ||||||
|  |   format_arg_store<context, Args...> as{ args... }; | ||||||
|  |   return vprintf(to_string_view(format_str), | ||||||
|  |                  basic_format_args<context>(as)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename S, typename Char = FMT_CHAR(S)> | ||||||
|  | inline int vfprintf(std::basic_ostream<Char> &os, | ||||||
|  |                     const S &format, | ||||||
|  |                     basic_format_args<typename basic_printf_context_t< | ||||||
|  |                       internal::basic_buffer<Char>>::type> args) { | ||||||
|  |   basic_memory_buffer<Char> buffer; | ||||||
|  |   printf(buffer, to_string_view(format), args); | ||||||
|  |   internal::write(os, buffer); | ||||||
|  |   return static_cast<int>(buffer.size()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |   \rst | ||||||
|  |   Prints formatted data to the stream *os*. | ||||||
|  |  | ||||||
|  |   **Example**:: | ||||||
|  |  | ||||||
|  |     fmt::fprintf(cerr, "Don't %s!", "panic"); | ||||||
|  |   \endrst | ||||||
|  |  */ | ||||||
|  | template <typename S, typename... Args> | ||||||
|  | inline FMT_ENABLE_IF_T(internal::is_string<S>::value, int) | ||||||
|  |     fprintf(std::basic_ostream<FMT_CHAR(S)> &os, | ||||||
|  |             const S &format_str, const Args & ... args) { | ||||||
|  |   internal::check_format_string<Args...>(format_str); | ||||||
|  |   typedef internal::basic_buffer<FMT_CHAR(S)> buffer; | ||||||
|  |   typedef typename basic_printf_context_t<buffer>::type context; | ||||||
|  |   format_arg_store<context, Args...> as{ args... }; | ||||||
|  |   return vfprintf(os, to_string_view(format_str), | ||||||
|  |                   basic_format_args<context>(as)); | ||||||
|  | } | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_PRINTF_H_ | ||||||
							
								
								
									
										308
									
								
								logger/bundled/fmt/ranges.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										308
									
								
								logger/bundled/fmt/ranges.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,308 @@ | |||||||
|  | // Formatting library for C++ - the core API | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - present, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  | // | ||||||
|  | // Copyright (c) 2018 - present, Remotion (Igor Schulz) | ||||||
|  | // All Rights Reserved | ||||||
|  | // {fmt} support for ranges, containers and types tuple interface. | ||||||
|  |  | ||||||
|  | #ifndef FMT_RANGES_H_ | ||||||
|  | #define FMT_RANGES_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  | #include <type_traits> | ||||||
|  |  | ||||||
|  | // output only up to N items from the range. | ||||||
|  | #ifndef FMT_RANGE_OUTPUT_LENGTH_LIMIT | ||||||
|  | # define FMT_RANGE_OUTPUT_LENGTH_LIMIT 256 | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | struct formatting_base { | ||||||
|  |   template <typename ParseContext> | ||||||
|  |   FMT_CONSTEXPR auto parse(ParseContext &ctx) -> decltype(ctx.begin()) { | ||||||
|  |     return ctx.begin(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char, typename Enable = void> | ||||||
|  | struct formatting_range : formatting_base<Char> { | ||||||
|  |   static FMT_CONSTEXPR_DECL const std::size_t range_length_limit = | ||||||
|  |       FMT_RANGE_OUTPUT_LENGTH_LIMIT; // output only up to N items from the range. | ||||||
|  |   Char prefix; | ||||||
|  |   Char delimiter; | ||||||
|  |   Char postfix; | ||||||
|  |   formatting_range() : prefix('{'), delimiter(','), postfix('}') {} | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool add_delimiter_spaces = true; | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool add_prepostfix_space = false; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char, typename Enable = void> | ||||||
|  | struct formatting_tuple : formatting_base<Char> { | ||||||
|  |   Char prefix; | ||||||
|  |   Char delimiter; | ||||||
|  |   Char postfix; | ||||||
|  |   formatting_tuple() : prefix('('), delimiter(','), postfix(')') {} | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool add_delimiter_spaces = true; | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool add_prepostfix_space = false; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | namespace internal { | ||||||
|  |  | ||||||
|  | template <typename RangeT, typename OutputIterator> | ||||||
|  | void copy(const RangeT &range, OutputIterator out) { | ||||||
|  |   for (auto it = range.begin(), end = range.end(); it != end; ++it) | ||||||
|  |     *out++ = *it; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename OutputIterator> | ||||||
|  | void copy(const char *str, OutputIterator out) { | ||||||
|  |   const char *p_curr = str; | ||||||
|  |   while (*p_curr) { | ||||||
|  |     *out++ = *p_curr++; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename OutputIterator> | ||||||
|  | void copy(char ch, OutputIterator out) { | ||||||
|  |   *out++ = ch; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Return true value if T has std::string interface, like std::string_view. | ||||||
|  | template <typename T> | ||||||
|  | class is_like_std_string { | ||||||
|  |   template <typename U> | ||||||
|  |   static auto check(U *p) -> | ||||||
|  |     decltype(p->find('a'), p->length(), p->data(), int()); | ||||||
|  |   template <typename> | ||||||
|  |   static void check(...); | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool value = | ||||||
|  |     !std::is_void<decltype(check<T>(FMT_NULL))>::value; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | struct is_like_std_string<fmt::basic_string_view<Char>> : std::true_type {}; | ||||||
|  |  | ||||||
|  | template <typename... Ts> | ||||||
|  | struct conditional_helper {}; | ||||||
|  |  | ||||||
|  | template <typename T, typename _ = void> | ||||||
|  | struct is_range_ : std::false_type {}; | ||||||
|  |  | ||||||
|  | #if !FMT_MSC_VER || FMT_MSC_VER > 1800 | ||||||
|  | template <typename T> | ||||||
|  | struct is_range_<T, typename std::conditional< | ||||||
|  |                     false, | ||||||
|  |                     conditional_helper<decltype(internal::declval<T>().begin()), | ||||||
|  |                                        decltype(internal::declval<T>().end())>, | ||||||
|  |                     void>::type> : std::true_type {}; | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /// tuple_size and tuple_element check. | ||||||
|  | template <typename T> | ||||||
|  | class is_tuple_like_ { | ||||||
|  |   template <typename U> | ||||||
|  |   static auto check(U *p) -> | ||||||
|  |     decltype(std::tuple_size<U>::value, | ||||||
|  |       internal::declval<typename std::tuple_element<0, U>::type>(), int()); | ||||||
|  |   template <typename> | ||||||
|  |   static void check(...); | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool value = | ||||||
|  |     !std::is_void<decltype(check<T>(FMT_NULL))>::value; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Check for integer_sequence | ||||||
|  | #if defined(__cpp_lib_integer_sequence) || FMT_MSC_VER >= 1900 | ||||||
|  | template <typename T, T... N> | ||||||
|  | using integer_sequence = std::integer_sequence<T, N...>; | ||||||
|  | template <std::size_t... N> | ||||||
|  | using index_sequence = std::index_sequence<N...>; | ||||||
|  | template <std::size_t N> | ||||||
|  | using make_index_sequence = std::make_index_sequence<N>; | ||||||
|  | #else | ||||||
|  | template <typename T, T... N> | ||||||
|  | struct integer_sequence { | ||||||
|  |   typedef T value_type; | ||||||
|  |  | ||||||
|  |   static FMT_CONSTEXPR std::size_t size() { | ||||||
|  |     return sizeof...(N); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <std::size_t... N> | ||||||
|  | using index_sequence = integer_sequence<std::size_t, N...>; | ||||||
|  |  | ||||||
|  | template <typename T, std::size_t N, T... Ns> | ||||||
|  | struct make_integer_sequence : make_integer_sequence<T, N - 1, N - 1, Ns...> {}; | ||||||
|  | template <typename T, T... Ns> | ||||||
|  | struct make_integer_sequence<T, 0, Ns...> : integer_sequence<T, Ns...> {}; | ||||||
|  |  | ||||||
|  | template <std::size_t N> | ||||||
|  | using make_index_sequence = make_integer_sequence<std::size_t, N>; | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | template <class Tuple, class F, size_t... Is> | ||||||
|  | void for_each(index_sequence<Is...>, Tuple &&tup, F &&f) FMT_NOEXCEPT { | ||||||
|  |   using std::get; | ||||||
|  |   // using free function get<I>(T) now. | ||||||
|  |   const int _[] = {0, ((void)f(get<Is>(tup)), 0)...}; | ||||||
|  |   (void)_;  // blocks warnings | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <class T> | ||||||
|  | FMT_CONSTEXPR make_index_sequence<std::tuple_size<T>::value>  | ||||||
|  | get_indexes(T const &) { return {}; } | ||||||
|  |  | ||||||
|  | template <class Tuple, class F> | ||||||
|  | void for_each(Tuple &&tup, F &&f) { | ||||||
|  |   const auto indexes = get_indexes(tup); | ||||||
|  |   for_each(indexes, std::forward<Tuple>(tup), std::forward<F>(f)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<typename Arg> | ||||||
|  | FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const Arg&,  | ||||||
|  |   typename std::enable_if< | ||||||
|  |     !is_like_std_string<typename std::decay<Arg>::type>::value>::type* = nullptr) { | ||||||
|  |   return add_space ? " {}" : "{}"; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<typename Arg> | ||||||
|  | FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const Arg&,  | ||||||
|  |   typename std::enable_if< | ||||||
|  |     is_like_std_string<typename std::decay<Arg>::type>::value>::type* = nullptr) { | ||||||
|  |   return add_space ? " \"{}\"" : "\"{}\""; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const char*) { | ||||||
|  |   return add_space ? " \"{}\"" : "\"{}\""; | ||||||
|  | } | ||||||
|  | FMT_CONSTEXPR const wchar_t* format_str_quoted(bool add_space, const wchar_t*) { | ||||||
|  |     return add_space ? L" \"{}\"" : L"\"{}\""; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const char) { | ||||||
|  |     return add_space ? " '{}'" : "'{}'"; | ||||||
|  | } | ||||||
|  | FMT_CONSTEXPR const wchar_t* format_str_quoted(bool add_space, const wchar_t) { | ||||||
|  |     return add_space ? L" '{}'" : L"'{}'"; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | }  // namespace internal | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | struct is_tuple_like { | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool value = | ||||||
|  |     internal::is_tuple_like_<T>::value && !internal::is_range_<T>::value; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename TupleT, typename Char> | ||||||
|  | struct formatter<TupleT, Char,  | ||||||
|  |     typename std::enable_if<fmt::is_tuple_like<TupleT>::value>::type> { | ||||||
|  | private: | ||||||
|  |   // C++11 generic lambda for format() | ||||||
|  |   template <typename FormatContext> | ||||||
|  |   struct format_each { | ||||||
|  |     template <typename T> | ||||||
|  |     void operator()(const T& v) { | ||||||
|  |       if (i > 0) { | ||||||
|  |         if (formatting.add_prepostfix_space) { | ||||||
|  |           *out++ = ' '; | ||||||
|  |         } | ||||||
|  |         internal::copy(formatting.delimiter, out); | ||||||
|  |       } | ||||||
|  |       format_to(out, | ||||||
|  |                 internal::format_str_quoted( | ||||||
|  |                     (formatting.add_delimiter_spaces && i > 0), v), | ||||||
|  |                 v); | ||||||
|  |       ++i; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     formatting_tuple<Char>& formatting; | ||||||
|  |     std::size_t& i; | ||||||
|  |     typename std::add_lvalue_reference<decltype(std::declval<FormatContext>().out())>::type out; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |   formatting_tuple<Char> formatting; | ||||||
|  |  | ||||||
|  |   template <typename ParseContext> | ||||||
|  |   FMT_CONSTEXPR auto parse(ParseContext &ctx) -> decltype(ctx.begin()) { | ||||||
|  |     return formatting.parse(ctx); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename FormatContext = format_context> | ||||||
|  |   auto format(const TupleT &values, FormatContext &ctx) -> decltype(ctx.out()) { | ||||||
|  |     auto out = ctx.out(); | ||||||
|  |     std::size_t i = 0; | ||||||
|  |     internal::copy(formatting.prefix, out); | ||||||
|  |  | ||||||
|  |     internal::for_each(values, format_each<FormatContext>{formatting, i, out}); | ||||||
|  |     if (formatting.add_prepostfix_space) { | ||||||
|  |       *out++ = ' '; | ||||||
|  |     } | ||||||
|  |     internal::copy(formatting.postfix, out); | ||||||
|  |  | ||||||
|  |     return ctx.out(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> | ||||||
|  | struct is_range { | ||||||
|  |   static FMT_CONSTEXPR_DECL const bool value = | ||||||
|  |     internal::is_range_<T>::value && !internal::is_like_std_string<T>::value; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename RangeT, typename Char> | ||||||
|  | struct formatter<RangeT, Char, | ||||||
|  |     typename std::enable_if<fmt::is_range<RangeT>::value>::type> { | ||||||
|  |  | ||||||
|  |   formatting_range<Char> formatting; | ||||||
|  |  | ||||||
|  |   template <typename ParseContext> | ||||||
|  |   FMT_CONSTEXPR auto parse(ParseContext &ctx) -> decltype(ctx.begin()) { | ||||||
|  |     return formatting.parse(ctx); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename FormatContext> | ||||||
|  |   typename FormatContext::iterator format( | ||||||
|  |       const RangeT &values, FormatContext &ctx) { | ||||||
|  |     auto out = ctx.out(); | ||||||
|  |     internal::copy(formatting.prefix, out); | ||||||
|  |     std::size_t i = 0; | ||||||
|  |     for (auto it = values.begin(), end = values.end(); it != end; ++it) { | ||||||
|  |       if (i > 0) { | ||||||
|  |         if (formatting.add_prepostfix_space) { | ||||||
|  |           *out++ = ' '; | ||||||
|  |         } | ||||||
|  |         internal::copy(formatting.delimiter, out); | ||||||
|  |       } | ||||||
|  |       format_to(out, | ||||||
|  |                 internal::format_str_quoted( | ||||||
|  |                     (formatting.add_delimiter_spaces && i > 0), *it), | ||||||
|  |                 *it); | ||||||
|  |       if (++i > formatting.range_length_limit) { | ||||||
|  |         format_to(out, " ... <other elements>"); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (formatting.add_prepostfix_space) { | ||||||
|  |       *out++ = ' '; | ||||||
|  |     } | ||||||
|  |     internal::copy(formatting.postfix, out); | ||||||
|  |     return ctx.out(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif // FMT_RANGES_H_ | ||||||
|  |  | ||||||
							
								
								
									
										160
									
								
								logger/bundled/fmt/time.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										160
									
								
								logger/bundled/fmt/time.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,160 @@ | |||||||
|  | // Formatting library for C++ - time formatting | ||||||
|  | // | ||||||
|  | // Copyright (c) 2012 - present, Victor Zverovich | ||||||
|  | // All rights reserved. | ||||||
|  | // | ||||||
|  | // For the license information refer to format.h. | ||||||
|  |  | ||||||
|  | #ifndef FMT_TIME_H_ | ||||||
|  | #define FMT_TIME_H_ | ||||||
|  |  | ||||||
|  | #include "format.h" | ||||||
|  | #include <ctime> | ||||||
|  | #include <locale> | ||||||
|  |  | ||||||
|  | FMT_BEGIN_NAMESPACE | ||||||
|  |  | ||||||
|  | // Prevents expansion of a preceding token as a function-style macro. | ||||||
|  | // Usage: f FMT_NOMACRO() | ||||||
|  | #define FMT_NOMACRO | ||||||
|  |  | ||||||
|  | namespace internal{ | ||||||
|  | inline null<> localtime_r FMT_NOMACRO(...) { return null<>(); } | ||||||
|  | inline null<> localtime_s(...) { return null<>(); } | ||||||
|  | inline null<> gmtime_r(...) { return null<>(); } | ||||||
|  | inline null<> gmtime_s(...) { return null<>(); } | ||||||
|  | }  // namespace internal | ||||||
|  |  | ||||||
|  | // Thread-safe replacement for std::localtime | ||||||
|  | inline std::tm localtime(std::time_t time) { | ||||||
|  |   struct dispatcher { | ||||||
|  |     std::time_t time_; | ||||||
|  |     std::tm tm_; | ||||||
|  |  | ||||||
|  |     dispatcher(std::time_t t): time_(t) {} | ||||||
|  |  | ||||||
|  |     bool run() { | ||||||
|  |       using namespace fmt::internal; | ||||||
|  |       return handle(localtime_r(&time_, &tm_)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     bool handle(std::tm *tm) { return tm != FMT_NULL; } | ||||||
|  |  | ||||||
|  |     bool handle(internal::null<>) { | ||||||
|  |       using namespace fmt::internal; | ||||||
|  |       return fallback(localtime_s(&tm_, &time_)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     bool fallback(int res) { return res == 0; } | ||||||
|  |  | ||||||
|  | #if !FMT_MSC_VER | ||||||
|  |     bool fallback(internal::null<>) { | ||||||
|  |       using namespace fmt::internal; | ||||||
|  |       std::tm *tm = std::localtime(&time_); | ||||||
|  |       if (tm) tm_ = *tm; | ||||||
|  |       return tm != FMT_NULL; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |   }; | ||||||
|  |   dispatcher lt(time); | ||||||
|  |   // Too big time values may be unsupported. | ||||||
|  |   if (!lt.run()) | ||||||
|  |     FMT_THROW(format_error("time_t value out of range")); | ||||||
|  |   return lt.tm_; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Thread-safe replacement for std::gmtime | ||||||
|  | inline std::tm gmtime(std::time_t time) { | ||||||
|  |   struct dispatcher { | ||||||
|  |     std::time_t time_; | ||||||
|  |     std::tm tm_; | ||||||
|  |  | ||||||
|  |     dispatcher(std::time_t t): time_(t) {} | ||||||
|  |  | ||||||
|  |     bool run() { | ||||||
|  |       using namespace fmt::internal; | ||||||
|  |       return handle(gmtime_r(&time_, &tm_)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     bool handle(std::tm *tm) { return tm != FMT_NULL; } | ||||||
|  |  | ||||||
|  |     bool handle(internal::null<>) { | ||||||
|  |       using namespace fmt::internal; | ||||||
|  |       return fallback(gmtime_s(&tm_, &time_)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     bool fallback(int res) { return res == 0; } | ||||||
|  |  | ||||||
|  | #if !FMT_MSC_VER | ||||||
|  |     bool fallback(internal::null<>) { | ||||||
|  |       std::tm *tm = std::gmtime(&time_); | ||||||
|  |       if (tm) tm_ = *tm; | ||||||
|  |       return tm != FMT_NULL; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |   }; | ||||||
|  |   dispatcher gt(time); | ||||||
|  |   // Too big time values may be unsupported. | ||||||
|  |   if (!gt.run()) | ||||||
|  |     FMT_THROW(format_error("time_t value out of range")); | ||||||
|  |   return gt.tm_; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | namespace internal { | ||||||
|  | inline std::size_t strftime(char *str, std::size_t count, const char *format, | ||||||
|  |                             const std::tm *time) { | ||||||
|  |   return std::strftime(str, count, format, time); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline std::size_t strftime(wchar_t *str, std::size_t count, | ||||||
|  |                             const wchar_t *format, const std::tm *time) { | ||||||
|  |   return std::wcsftime(str, count, format, time); | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename Char> | ||||||
|  | struct formatter<std::tm, Char> { | ||||||
|  |   template <typename ParseContext> | ||||||
|  |   auto parse(ParseContext &ctx) -> decltype(ctx.begin()) { | ||||||
|  |     auto it = ctx.begin(); | ||||||
|  |     if (it != ctx.end() && *it == ':') | ||||||
|  |       ++it; | ||||||
|  |     auto end = it; | ||||||
|  |     while (end != ctx.end() && *end != '}') | ||||||
|  |       ++end; | ||||||
|  |     tm_format.reserve(internal::to_unsigned(end - it + 1)); | ||||||
|  |     tm_format.append(it, end); | ||||||
|  |     tm_format.push_back('\0'); | ||||||
|  |     return end; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   template <typename FormatContext> | ||||||
|  |   auto format(const std::tm &tm, FormatContext &ctx) -> decltype(ctx.out()) { | ||||||
|  |     basic_memory_buffer<Char> buf; | ||||||
|  |     std::size_t start = buf.size(); | ||||||
|  |     for (;;) { | ||||||
|  |       std::size_t size = buf.capacity() - start; | ||||||
|  |       std::size_t count = | ||||||
|  |         internal::strftime(&buf[start], size, &tm_format[0], &tm); | ||||||
|  |       if (count != 0) { | ||||||
|  |         buf.resize(start + count); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       if (size >= tm_format.size() * 256) { | ||||||
|  |         // If the buffer is 256 times larger than the format string, assume | ||||||
|  |         // that `strftime` gives an empty result. There doesn't seem to be a | ||||||
|  |         // better way to distinguish the two cases: | ||||||
|  |         // https://github.com/fmtlib/fmt/issues/367 | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       const std::size_t MIN_GROWTH = 10; | ||||||
|  |       buf.reserve(buf.capacity() + (size > MIN_GROWTH ? size : MIN_GROWTH)); | ||||||
|  |     } | ||||||
|  |     return std::copy(buf.begin(), buf.end(), ctx.out()); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   basic_memory_buffer<Char> tm_format; | ||||||
|  | }; | ||||||
|  | FMT_END_NAMESPACE | ||||||
|  |  | ||||||
|  | #endif  // FMT_TIME_H_ | ||||||
							
								
								
									
										100
									
								
								test/Common.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								test/Common.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #ifndef FAIR_LOGGER_TEST_COMMON_H | ||||||
|  | #define FAIR_LOGGER_TEST_COMMON_H | ||||||
|  |  | ||||||
|  | #include <fstream> | ||||||
|  | #include <functional> | ||||||
|  | #include <iostream> | ||||||
|  | #include <regex> | ||||||
|  | #include <stdexcept> | ||||||
|  | #include <sstream> | ||||||
|  | #include <string> | ||||||
|  |  | ||||||
|  | #include <stdio.h> // fflush | ||||||
|  | #include <unistd.h> // dup, dup2, close | ||||||
|  |  | ||||||
|  | namespace fair | ||||||
|  | { | ||||||
|  | namespace logger | ||||||
|  | { | ||||||
|  | namespace test | ||||||
|  | { | ||||||
|  |  | ||||||
|  | template<typename ... T> | ||||||
|  | auto ToStr(T&&... t) -> std::string | ||||||
|  | { | ||||||
|  |     std::stringstream ss; | ||||||
|  |     (void)std::initializer_list<int>{(ss << t, 0)...}; | ||||||
|  |     return ss.str(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<int S> | ||||||
|  | class StreamCapturer | ||||||
|  | { | ||||||
|  |   public: | ||||||
|  |     explicit StreamCapturer() | ||||||
|  |         : mFd(S) | ||||||
|  |         , mOriginalFd(dup(S)) // create a copy of the given file descriptor | ||||||
|  |     { | ||||||
|  |         char name[] = "/tmp/fairlogger_test_capture.XXXXXX"; | ||||||
|  |  | ||||||
|  |         const int capturedFd = mkstemp(name); // create a unique temporary file | ||||||
|  |         if (capturedFd == -1) { | ||||||
|  |             std::cout << "Could not create tmp file " << name << " for test; does the test have access to the /tmp directory?" << std::endl; | ||||||
|  |             throw std::runtime_error("Could not create tmp file for test; does the test have access to the /tmp directory?"); | ||||||
|  |         } | ||||||
|  |         mTmpFile = name; | ||||||
|  |  | ||||||
|  |         fflush(nullptr); // flushes all open output streams | ||||||
|  |         dup2(capturedFd, mFd); | ||||||
|  |         close(capturedFd); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     std::string GetCapture() | ||||||
|  |     { | ||||||
|  |         fflush(nullptr); // flushes all open output streams | ||||||
|  |         std::ifstream t(mTmpFile); | ||||||
|  |         std::stringstream buffer; | ||||||
|  |         buffer << t.rdbuf(); | ||||||
|  |         return buffer.str(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ~StreamCapturer() | ||||||
|  |     { | ||||||
|  |         dup2(mOriginalFd, mFd); | ||||||
|  |         close(mOriginalFd); | ||||||
|  |         remove(mTmpFile.c_str()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |   private: | ||||||
|  |     const int mFd; | ||||||
|  |     int mOriginalFd; | ||||||
|  |     std::string mTmpFile; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | void CheckOutput(std::string const& expected, std::function<void()> f) | ||||||
|  | { | ||||||
|  |     std::string output; | ||||||
|  |     { | ||||||
|  |         StreamCapturer<1> c; | ||||||
|  |         f(); | ||||||
|  |         output = c.GetCapture(); | ||||||
|  |     } | ||||||
|  |     const std::regex e(expected); | ||||||
|  |     if (!std::regex_match(output, e)) { | ||||||
|  |         throw std::runtime_error(std::string("MISMATCH:\n##### expected (regex):\n" + expected + "\n##### found:\n" + output)); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } // namespace test | ||||||
|  | } // namespace logger | ||||||
|  | } // namespace fair | ||||||
|  |  | ||||||
|  | #endif // FAIR_LOGGER_TEST_COMMON_H | ||||||
							
								
								
									
										41
									
								
								test/ci/CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								test/ci/CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | |||||||
|  | ################################################################################ | ||||||
|  | #   Copyright (C) 2021 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH     # | ||||||
|  | #                                                                              # | ||||||
|  | #              This software is distributed under the terms of the             # | ||||||
|  | #              GNU Lesser General Public Licence (LGPL) version 3,             # | ||||||
|  | #                  copied verbatim in the file "LICENSE"                       # | ||||||
|  | ################################################################################ | ||||||
|  |  | ||||||
|  | function(container) | ||||||
|  |   cmake_parse_arguments(ARGS "" "OS;VERSION" "" ${ARGN}) | ||||||
|  |  | ||||||
|  |   set(container "${ARGS_OS}.${ARGS_VERSION}") | ||||||
|  |   set(def "${container}.def") | ||||||
|  |   set(log "${container}.log") | ||||||
|  |   set(target "${container}.sif") | ||||||
|  |  | ||||||
|  |   configure_file("${CMAKE_CURRENT_SOURCE_DIR}/linux.def.in" ${def} @ONLY) | ||||||
|  |  | ||||||
|  |   add_custom_command(OUTPUT ${target} | ||||||
|  |     COMMAND ${CMAKE_COMMAND} -E env ${BASH} -c | ||||||
|  |       "${SINGULARITY} build -f -F ${target} ${def} > ${log} 2>&1" | ||||||
|  |     VERBATIM | ||||||
|  |     WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR} | ||||||
|  |     COMMENT "Building fairlogger test container ${target}, logging to ${CMAKE_CURRENT_BINARY_DIR}/${log}" | ||||||
|  |     DEPENDS ${PACKAGE_SETUP_SCRIPT} ${bootstrap_cmake_script} | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   list(APPEND containers ${target}) | ||||||
|  |   set(containers ${containers} PARENT_SCOPE) | ||||||
|  | endfunction() | ||||||
|  |  | ||||||
|  | container(OS fedora VERSION 32) | ||||||
|  | container(OS fedora VERSION 33) | ||||||
|  | container(OS fedora VERSION 34) | ||||||
|  | container(OS fedora VERSION 35) | ||||||
|  | container(OS fedora VERSION 36) | ||||||
|  | container(OS fedora VERSION 37) | ||||||
|  | container(OS fedora VERSION 38) | ||||||
|  | container(OS fedora VERSION 39) | ||||||
|  |  | ||||||
|  | add_custom_target(all-containers DEPENDS ${containers}) | ||||||
							
								
								
									
										8
									
								
								test/ci/linux.def.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								test/ci/linux.def.in
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | Bootstrap: docker | ||||||
|  | From: @ARGS_OS@:@ARGS_VERSION@ | ||||||
|  |  | ||||||
|  | %files | ||||||
|  |     @CMAKE_CURRENT_SOURCE_DIR@/setup-@ARGS_OS@.sh /setup.sh | ||||||
|  |  | ||||||
|  | %post | ||||||
|  |     bash /setup.sh @ARGS_VERSION@ | ||||||
							
								
								
									
										7
									
								
								test/ci/setup-fedora.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										7
									
								
								test/ci/setup-fedora.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | #! /bin/bash | ||||||
|  |  | ||||||
|  | dnf -y update | ||||||
|  | dnf -y install https://alfa-ci.gsi.de/packages/rpm/fedora-$1-x86_64/fairsoft-release-dev.rpm | ||||||
|  | dnf -y install boost-devel ninja-build 'dnf-command(builddep)' libasan liblsan libtsan libubsan clang-tools-extra | ||||||
|  | dnf -y builddep fairlogger | ||||||
|  | dnf -y clean all | ||||||
							
								
								
									
										35
									
								
								test/ci/slurm-submit.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										35
									
								
								test/ci/slurm-submit.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | #! /bin/bash | ||||||
|  |  | ||||||
|  | label="$1" | ||||||
|  | jobsh="$2" | ||||||
|  |  | ||||||
|  | ALFACI_SLURM_CPUS=4 | ||||||
|  | if [ -z "$ALFACI_SLURM_EXTRA_OPTS" ] | ||||||
|  | then | ||||||
|  | 	ALFACI_SLURM_EXTRA_OPTS="--hint=compute_bound" | ||||||
|  | fi | ||||||
|  | ALFACI_SLURM_TIMEOUT=10 | ||||||
|  | if [ -z "$ALFACI_SLURM_QUEUE" ] | ||||||
|  | then | ||||||
|  | 	ALFACI_SLURM_QUEUE=main | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | echo "*** Slurm request options :" | ||||||
|  | echo "***   Working directory ..: $PWD" | ||||||
|  | echo "***   Queue ..............: $ALFACI_SLURM_QUEUE" | ||||||
|  | echo "***   CPUs ...............: $ALFACI_SLURM_CPUS" | ||||||
|  | echo "***   Wall Time ..........: $ALFACI_SLURM_TIMEOUT min" | ||||||
|  | echo "***   Job Name ...........: ${label}" | ||||||
|  | echo "***   Extra Options ......: ${ALFACI_SLURM_EXTRA_OPTS}" | ||||||
|  | echo "*** Submitting job at ....: $(date -R)" | ||||||
|  | ( | ||||||
|  | 	set -x | ||||||
|  | 	srun -p $ALFACI_SLURM_QUEUE -c $ALFACI_SLURM_CPUS -n 1 \ | ||||||
|  | 		-t $ALFACI_SLURM_TIMEOUT \ | ||||||
|  | 		--job-name="${label}" \ | ||||||
|  | 		${ALFACI_SLURM_EXTRA_OPTS} \ | ||||||
|  | 		bash "${jobsh}" | ||||||
|  | ) | ||||||
|  | retval=$? | ||||||
|  | echo "*** Exit Code ............: $retval" | ||||||
|  | exit "$retval" | ||||||
							
								
								
									
										81
									
								
								test/cycle.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								test/cycle.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2015-2025 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "Common.h" | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  | using namespace fair::logger::test; | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     try { | ||||||
|  |         Logger::SetConsoleColor(false); | ||||||
|  |         Logger::SetVerbosity(Verbosity::user4); | ||||||
|  |  | ||||||
|  |         cout << "initial verbosity >" << Logger::GetVerbosity() << "<" << endl << endl; | ||||||
|  |  | ||||||
|  |         array<Verbosity, 10> verbositiesUp{{ Verbosity::verylow, Verbosity::low, Verbosity::medium, Verbosity::high, Verbosity::veryhigh, Verbosity::user1, Verbosity::user2, Verbosity::user3, Verbosity::user4, Verbosity::verylow }}; | ||||||
|  |         for (unsigned int i = 0; i < verbositiesUp.size(); ++i) { | ||||||
|  |             Logger::CycleVerbosityUp(); | ||||||
|  |             if (Logger::GetVerbosity() != verbositiesUp.at(i)) { throw runtime_error(ToStr("Expected verbosity to be ", verbositiesUp.at(i), ", but it is ", Logger::GetVerbosity())); } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         array<Verbosity, 10> verbositiesDown{{ Verbosity::user4, Verbosity::user3, Verbosity::user2, Verbosity::user1, Verbosity::veryhigh, Verbosity::high, Verbosity::medium, Verbosity::low, Verbosity::verylow, Verbosity::user4 }}; | ||||||
|  |         for (unsigned int i = 0; i < verbositiesDown.size(); ++i) { | ||||||
|  |             Logger::CycleVerbosityDown(); | ||||||
|  |             if (Logger::GetVerbosity() != verbositiesDown.at(i)) { throw runtime_error(ToStr("Expected verbosity to be ", verbositiesDown.at(i), ", but it is ", Logger::GetVerbosity())); } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Logger::SetConsoleSeverity(Severity::fatal); | ||||||
|  |         cout << "initial severity >" << Logger::GetConsoleSeverity() << "<" << endl << endl; | ||||||
|  |  | ||||||
|  |         array<Severity, 16> severitiesUp{{ Severity::nolog, Severity::trace, Severity::debug4, Severity::debug3, Severity::debug2, Severity::debug1, Severity::debug, Severity::detail, Severity::info, Severity::state, Severity::warn, Severity::important, Severity::alarm, Severity::error, Severity::critical, Severity::fatal }}; | ||||||
|  | #ifdef FAIR_MIN_SEVERITY | ||||||
|  |         for (unsigned int i = static_cast<int>(Severity::FAIR_MIN_SEVERITY); i < severitiesUp.size(); ++i) { | ||||||
|  | #else | ||||||
|  |         for (unsigned int i = 0; i < severitiesUp.size(); ++i) { | ||||||
|  | #endif | ||||||
|  |             Logger::CycleConsoleSeverityUp(); | ||||||
|  |             if (Logger::GetConsoleSeverity() != severitiesUp.at(i)) { throw runtime_error(ToStr("Expected severity to be ", severitiesUp.at(i), ", but it is ", Logger::GetConsoleSeverity())); } | ||||||
|  |         } | ||||||
|  |         Logger::CycleConsoleSeverityUp(); | ||||||
|  | #ifdef FAIR_MIN_SEVERITY | ||||||
|  |         if (Logger::GetConsoleSeverity() != Severity::FAIR_MIN_SEVERITY) { throw runtime_error(ToStr("Expected severity to be ", Severity::nolog, ", but it is ", Logger::GetConsoleSeverity())); } | ||||||
|  | #else | ||||||
|  |         if (Logger::GetConsoleSeverity() != Severity::nolog) { throw runtime_error(ToStr("Expected severity to be ", Severity::nolog, ", but it is ", Logger::GetConsoleSeverity())); } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |         Logger::SetConsoleSeverity(Severity::fatal); | ||||||
|  |         cout << "initial severity >" << Logger::GetConsoleSeverity() << "<" << endl << endl; | ||||||
|  |  | ||||||
|  |         array<Severity, 16> severitiesDown{{ Severity::critical, Severity::error, Severity::alarm, Severity::important, Severity::warn, Severity::state, Severity::info, Severity::detail, Severity::debug, Severity::debug1, Severity::debug2, Severity::debug3, Severity::debug4, Severity::trace, Severity::nolog, Severity::fatal }}; | ||||||
|  | #ifdef FAIR_MIN_SEVERITY | ||||||
|  |         for (unsigned int i = 0; i < severitiesDown.size() - static_cast<int>(Severity::FAIR_MIN_SEVERITY) - 1; ++i) { | ||||||
|  | #else | ||||||
|  |         for (unsigned int i = 0; i < severitiesDown.size(); ++i) { | ||||||
|  | #endif | ||||||
|  |             Logger::CycleConsoleSeverityDown(); | ||||||
|  |             if (Logger::GetConsoleSeverity() != severitiesDown.at(i)) { throw runtime_error(ToStr("Expected severity to be ", severitiesDown.at(i), ", but it is ", Logger::GetConsoleSeverity())); } | ||||||
|  |         } | ||||||
|  |         Logger::CycleConsoleSeverityDown(); | ||||||
|  | #ifdef FAIR_MIN_SEVERITY | ||||||
|  |         if (Logger::GetConsoleSeverity() != Severity::fatal) { throw runtime_error(ToStr("Expected severity to be ", Severity::fatal, ", but it is ", Logger::GetConsoleSeverity())); } | ||||||
|  | #else | ||||||
|  |         if (Logger::GetConsoleSeverity() != Severity::error) { throw runtime_error(ToStr("Expected severity to be ", Severity::error, ", but it is ", Logger::GetConsoleSeverity())); } | ||||||
|  | #endif | ||||||
|  |     } catch (runtime_error& rte) { | ||||||
|  |         cout << rte.what() << endl; | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
							
								
								
									
										62
									
								
								test/logger.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								test/logger.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  |  | ||||||
|  | void printEverySeverity() | ||||||
|  | { | ||||||
|  |     static int i = 1; | ||||||
|  |  | ||||||
|  |     LOG(nolog)     << "nolog message, counter: "     << i++; | ||||||
|  |     LOG(trace)     << "trace message, counter: "     << i++; | ||||||
|  |     LOG(debug4)    << "debug4 message, counter: "    << i++; | ||||||
|  |     LOG(debug3)    << "debug3 message, counter: "    << i++; | ||||||
|  |     LOG(debug2)    << "debug2 message, counter: "    << i++; | ||||||
|  |     LOG(debug1)    << "debug1 message, counter: "    << i++; | ||||||
|  |     LOG(debug)     << "debug message, counter: "     << i++; | ||||||
|  |     LOG(detail)    << "detail message, counter: "    << i++; | ||||||
|  |     LOG(info)      << "info message, counter: "      << i++; | ||||||
|  |     LOG(state)     << "state message, counter: "     << i++; | ||||||
|  |     LOG(warn)      << "warning message, counter: "   << i++; | ||||||
|  |     LOG(important) << "important message, counter: " << i++; | ||||||
|  |     LOG(alarm)     << "alarm message, counter: "     << i++; | ||||||
|  |     LOG(error)     << "error message, counter: "     << i++; | ||||||
|  |     LOG(critical)  << "critical message, counter: "  << i++; | ||||||
|  |     LOG(fatal)     << "fatal message, counter: "     << i++; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void printAllVerbositiesWithSeverity(Severity sev) | ||||||
|  | { | ||||||
|  |     Logger::SetConsoleSeverity(sev); | ||||||
|  |  | ||||||
|  |     for (uint32_t i = 0; i < Logger::fVerbosityNames.size(); ++i) { | ||||||
|  |         cout << "##### testing severity '" << sev << "' with verbosity '" << Logger::fVerbosityNames.at(i) << "'" << endl; | ||||||
|  |         Logger::SetVerbosity(static_cast<Verbosity>(i)); | ||||||
|  |         printEverySeverity(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     Logger::SetConsoleColor(true); | ||||||
|  |     Logger::SetVerbosity(Verbosity::veryhigh); | ||||||
|  |     cout << "##### GetConsoleSeverity = " << Logger::SeverityName(Logger::GetConsoleSeverity()) << endl; | ||||||
|  |  | ||||||
|  |     cout << "##### testing severities..." << endl; | ||||||
|  |  | ||||||
|  |     for (uint32_t i = 0; i < Logger::fSeverityNames.size(); ++i) { | ||||||
|  |         printAllVerbositiesWithSeverity(static_cast<Severity>(i)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
| @@ -1,200 +0,0 @@ | |||||||
| /******************************************************************************** |  | ||||||
|  * Copyright (C) 2014-2019 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * |  | ||||||
|  *                                                                              * |  | ||||||
|  *              This software is distributed under the terms of the             * |  | ||||||
|  *              GNU Lesser General Public Licence (LGPL) version 3,             * |  | ||||||
|  *                  copied verbatim in the file "LICENSE"                       * |  | ||||||
|  ********************************************************************************/ |  | ||||||
|  |  | ||||||
| // WARNING : pragma commands to hide boost warning |  | ||||||
| // TODO : remove these pragma commands when boost will fix this issue in future release |  | ||||||
|  |  | ||||||
| #include <Logger.h> |  | ||||||
|  |  | ||||||
| #include <iostream> |  | ||||||
| #include <vector> |  | ||||||
| #include <thread> |  | ||||||
|  |  | ||||||
| using namespace std; |  | ||||||
| using namespace fair; |  | ||||||
|  |  | ||||||
| namespace test |  | ||||||
| { |  | ||||||
|  |  | ||||||
| void printEverySeverity() |  | ||||||
| { |  | ||||||
|     static int i = 1; |  | ||||||
|  |  | ||||||
|     LOG(nolog) << "nolog message "   << i++; |  | ||||||
|     LOG(error) << "error message "   << i++; |  | ||||||
|     LOG(warn)  << "warning message " << i++; |  | ||||||
|     LOG(state) << "state message "   << i++; |  | ||||||
|     LOG(info)  << "info message "    << i++; |  | ||||||
|     LOG(debug) << "debug message "   << i++; |  | ||||||
|     LOG(debug1) << "debug1 message "   << i++; |  | ||||||
|     LOG(debug2) << "debug2 message "   << i++; |  | ||||||
|     LOG(debug3) << "debug3 message "   << i++; |  | ||||||
|     LOG(debug4) << "debug4 message "   << i++; |  | ||||||
|     LOG(trace) << "trace message "   << i++; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void printAllVerbositiesWithSeverity(Severity sev) |  | ||||||
| { |  | ||||||
|     Logger::SetConsoleSeverity(sev); |  | ||||||
|  |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'verylow' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::verylow); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'low' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::low); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'medium' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::medium); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'high' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::high); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'veryhigh' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::veryhigh); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'user1' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::user1); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'user2' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::user2); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'user3' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::user3); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << endl << "cout: >>> testing severity '" << Logger::SeverityName(sev) << "' with 'user4' verbosity..." << endl; |  | ||||||
|     Logger::SetVerbosity(Verbosity::user4); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void silentlyPrintAllVerbositiesWithSeverity(Severity sev) |  | ||||||
| { |  | ||||||
|     Logger::SetConsoleSeverity(sev); |  | ||||||
|  |  | ||||||
|     Logger::SetVerbosity(Verbosity::verylow); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::low); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::medium); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::high); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::veryhigh); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::user1); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::user2); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::user3); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     Logger::SetVerbosity(Verbosity::user4); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| int main() |  | ||||||
| { |  | ||||||
|     Logger::SetConsoleColor(true); |  | ||||||
|  |  | ||||||
|     auto spec = VerbositySpec::Make(VerbositySpec::Info::file_line_function, |  | ||||||
|                                     VerbositySpec::Info::process_name,VerbositySpec::Info::process_name); |  | ||||||
|     cout << "Defining custom verbosity \"user2\"" << endl; |  | ||||||
|     Logger::DefineVerbosity(Verbosity::user2, spec); |  | ||||||
|  |  | ||||||
|     cout << "cout: testing severities..." << endl; |  | ||||||
|  |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::trace); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::debug4); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::debug3); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::debug2); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::debug1); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::debug); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::info); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::state); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::warn); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::error); |  | ||||||
|     printAllVerbositiesWithSeverity(Severity::nolog); |  | ||||||
|  |  | ||||||
|     cout << endl; |  | ||||||
|     cout << "cout: resetting severity to 'info' and verbosity to 'medium'" << endl; |  | ||||||
|     Logger::SetConsoleSeverity(Severity::info); |  | ||||||
|     Logger::SetVerbosity(Verbosity::medium); |  | ||||||
|  |  | ||||||
|     cout << "cout: is logging trace: " << fair::Logger::Logging(Severity::trace) << endl; |  | ||||||
|     cout << "cout: is logging debug4: " << fair::Logger::Logging(Severity::debug) << endl; |  | ||||||
|     cout << "cout: is logging debug3: " << fair::Logger::Logging(Severity::debug) << endl; |  | ||||||
|     cout << "cout: is logging debug2: " << fair::Logger::Logging(Severity::debug) << endl; |  | ||||||
|     cout << "cout: is logging debug1: " << fair::Logger::Logging(Severity::debug) << endl; |  | ||||||
|     cout << "cout: is logging debug: " << fair::Logger::Logging(Severity::debug) << endl; |  | ||||||
|     cout << "cout: is logging info: " << fair::Logger::Logging(Severity::info) << endl; |  | ||||||
|     cout << "cout: is logging state: " << fair::Logger::Logging(Severity::state) << endl; |  | ||||||
|     cout << "cout: is logging warn: " << fair::Logger::Logging(Severity::warn) << endl; |  | ||||||
|     cout << "cout: is logging error: " << fair::Logger::Logging(Severity::error) << endl; |  | ||||||
|     cout << "cout: is logging fatal: " << fair::Logger::Logging(Severity::fatal) << endl; |  | ||||||
|     cout << "cout: is logging nolog: " << fair::Logger::Logging(Severity::nolog) << endl; |  | ||||||
|  |  | ||||||
|     for (int i = 0; i < 1000000; ++i) |  | ||||||
|     { |  | ||||||
|         silentlyPrintAllVerbositiesWithSeverity(Severity::nolog); |  | ||||||
|     } |  | ||||||
|     cout << endl; |  | ||||||
|     cout << "cout: resetting severity to 'trace' and verbosity to 'veryhigh'" << endl; |  | ||||||
|     Logger::SetConsoleSeverity(Severity::trace); |  | ||||||
|     Logger::SetVerbosity(Verbosity::veryhigh); |  | ||||||
|  |  | ||||||
|     cout << endl; |  | ||||||
|     cout << "cout: testing conditional logging..." << endl; |  | ||||||
|     int x = 0; |  | ||||||
|     LOG(info) << "x = " << x << " (initial)"; |  | ||||||
|     LOG_IF(info, (x == 0)) << "incrementing x to " << ++x; |  | ||||||
|     LOG(info) << "x = " << x << " (after increment)"; |  | ||||||
|     LOG_IF(info, (x == 0)) << "this should not be printed and x not incremented: " << ++x; |  | ||||||
|     LOG(info) << "x = " << x << " (after conditional increment)"; |  | ||||||
|  |  | ||||||
|     cout << endl; |  | ||||||
|     cout << "cout: resetting severity to 'nolog'" << endl; |  | ||||||
|     Logger::SetConsoleSeverity(Severity::nolog); |  | ||||||
|  |  | ||||||
|     cout << "cout: ----------------------------" << endl; |  | ||||||
|     cout << "cout: open log file with severity 'error'" << endl; |  | ||||||
|     Logger::InitFileSink(Severity::error, "test_log", true); |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|     cout << "cout: closing log file" << endl; |  | ||||||
|     Logger::RemoveFileSink(); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     cout << "cout: resetting severity to 'nolog'" << endl; |  | ||||||
|     Logger::SetConsoleSeverity(Severity::nolog); |  | ||||||
|     cout << "cout: ----------------------------" << endl; |  | ||||||
|     cout << "cout: adding custom sink with error severity" << endl << endl; |  | ||||||
|  |  | ||||||
|     Logger::AddCustomSink("CustomSink", "error", [](const string& content, const LogMetaData& metadata) |  | ||||||
|     { |  | ||||||
|         cout << "CustomSink: content: " << content << endl; |  | ||||||
|  |  | ||||||
|         cout << "CustomSink: available metadata: " << endl; |  | ||||||
|         cout << "CustomSink: \tstd::time_t timestamp: " << metadata.timestamp << endl; |  | ||||||
|         cout << "CustomSink: \tstd::chrono::microseconds us: " << metadata.us.count() << endl; |  | ||||||
|         cout << "CustomSink: \tstd::string process_name: " << metadata.process_name << endl; |  | ||||||
|         cout << "CustomSink: \tstd::string file: " << metadata.file << endl; |  | ||||||
|         cout << "CustomSink: \tstd::string line: " << metadata.line << endl; |  | ||||||
|         cout << "CustomSink: \tstd::string func: " << metadata.func << endl; |  | ||||||
|         cout << "CustomSink: \tstd::string severity_name: " << metadata.severity_name << endl; |  | ||||||
|         cout << "CustomSink: \tfair::Severity severity: " << static_cast<int>(metadata.severity) << endl; |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     test::printEverySeverity(); |  | ||||||
|  |  | ||||||
|     cout << endl << "cout: removing custom sink with info severity" << endl; |  | ||||||
|  |  | ||||||
|     Logger::AddCustomSink("CustomSink", Severity::error, [](const string& /*content*/, const LogMetaData& /*metadata*/){}); |  | ||||||
|     Logger::RemoveCustomSink("CustomSink"); |  | ||||||
|     Logger::RemoveCustomSink("bla"); |  | ||||||
|  |  | ||||||
|     return 0; |  | ||||||
| } |  | ||||||
							
								
								
									
										60
									
								
								test/macros.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								test/macros.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "Common.h" | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  | using namespace fair::logger::test; | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     try { | ||||||
|  |         Logger::SetConsoleColor(false); | ||||||
|  |         Logger::SetConsoleSeverity(Severity::fatal); | ||||||
|  |         Logger::SetVerbosity(Verbosity::verylow); | ||||||
|  |  | ||||||
|  |         int x = 0; | ||||||
|  |  | ||||||
|  |         CheckOutput("^incrementing x to 1\n$", [&]() { LOG_IF(fatal, true) << "incrementing x to " << ++x; }); | ||||||
|  |         if (x != 1) { | ||||||
|  |             throw runtime_error(ToStr("expected x to be 1, but it is: ", x)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         CheckOutput("^$", [&]() { LOG_IF(fatal, false) << "incrementing x to " << ++x; }); | ||||||
|  |         if (x != 1) { | ||||||
|  |             throw runtime_error(ToStr("expected x to be 1, but it is: ", x)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         CheckOutput("^Hello world :-\\)!\n$", []() { LOGP(fatal, "Hello {} {}!", "world", ":-)"); }); | ||||||
|  |         CheckOutput("^Hello world :-\\)!\n$", []() { LOGF(fatal, "Hello %s %s!", "world", ":-)"); }); | ||||||
|  |  | ||||||
|  |         CheckOutput(ToStr(R"(^\[FATAL\])", " content\n$"), []() { LOGV(fatal, low) << "content"; }); | ||||||
|  |  | ||||||
|  |         CheckOutput("^\n\n\n\n$", []() { | ||||||
|  |             LOGN(fatal); | ||||||
|  |             LOGN(fatal); | ||||||
|  |             LOGN(fatal); | ||||||
|  |             LOGN(fatal); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::veryhigh); | ||||||
|  |  | ||||||
|  |         CheckOutput(ToStr(R"(^\[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[a:4:b\])", " c\n$"), []() { | ||||||
|  |             LOGD(Severity::fatal, "a", "4", "b") << "c"; | ||||||
|  |         }); | ||||||
|  |     } catch (runtime_error& rte) { | ||||||
|  |         cout << rte.what() << endl; | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
							
								
								
									
										64
									
								
								test/nolog.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								test/nolog.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  |  | ||||||
|  | void printEverySeverity() | ||||||
|  | { | ||||||
|  |     LOG(nolog)     << "nolog message "; | ||||||
|  |     LOG(trace)     << "trace message "; | ||||||
|  |     LOG(debug4)    << "debug4 message "; | ||||||
|  |     LOG(debug3)    << "debug3 message "; | ||||||
|  |     LOG(debug2)    << "debug2 message "; | ||||||
|  |     LOG(debug1)    << "debug1 message "; | ||||||
|  |     LOG(debug)     << "debug message "; | ||||||
|  |     LOG(detail)    << "detail message "; | ||||||
|  |     LOG(info)      << "info message "; | ||||||
|  |     LOG(state)     << "state message "; | ||||||
|  |     LOG(warn)      << "warning message "; | ||||||
|  |     LOG(important) << "important message "; | ||||||
|  |     LOG(alarm)     << "alarm message "; | ||||||
|  |     LOG(error)     << "error message "; | ||||||
|  |     LOG(critical)  << "critical message "; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void silentlyPrintAllVerbositiesWithSeverity(Severity sev) | ||||||
|  | { | ||||||
|  |     Logger::SetConsoleSeverity(sev); | ||||||
|  |  | ||||||
|  |     Logger::SetVerbosity(Verbosity::verylow); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::low); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::medium); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::high); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::veryhigh); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::user1); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::user2); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::user3); | ||||||
|  |     printEverySeverity(); | ||||||
|  |     Logger::SetVerbosity(Verbosity::user4); | ||||||
|  |     printEverySeverity(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     for (int i = 0; i < 1000000; ++i) { | ||||||
|  |         silentlyPrintAllVerbositiesWithSeverity(Severity::nolog); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
							
								
								
									
										102
									
								
								test/severity.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								test/severity.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "Common.h" | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <cstdint> | ||||||
|  | #include <iostream> | ||||||
|  | #include <stdexcept> | ||||||
|  | #include <string> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  | using namespace fair::logger::test; | ||||||
|  |  | ||||||
|  | uint32_t printEverySeverity(uint32_t i) | ||||||
|  | { | ||||||
|  |     LOG(nolog)     << "nolog message, counter: "     << i++; | ||||||
|  |     LOG(trace)     << "trace message, counter: "     << i++; | ||||||
|  |     LOG(debug4)    << "debug4 message, counter: "    << i++; | ||||||
|  |     LOG(debug3)    << "debug3 message, counter: "    << i++; | ||||||
|  |     LOG(debug2)    << "debug2 message, counter: "    << i++; | ||||||
|  |     LOG(debug1)    << "debug1 message, counter: "    << i++; | ||||||
|  |     LOG(debug)     << "debug message, counter: "     << i++; | ||||||
|  |     LOG(detail)    << "detail message, counter: "    << i++; | ||||||
|  |     LOG(info)      << "info message, counter: "      << i++; | ||||||
|  |     LOG(state)     << "state message, counter: "     << i++; | ||||||
|  |     LOG(warn)      << "warning message, counter: "   << i++; | ||||||
|  |     LOG(important) << "important message, counter: " << i++; | ||||||
|  |     LOG(alarm)     << "alarm message, counter: "     << i++; | ||||||
|  |     LOG(error)     << "error message, counter: "     << i++; | ||||||
|  |     LOG(critical)  << "critical message, counter: "  << i++; | ||||||
|  |     LOG(fatal)     << "fatal message, counter: "     << i++; | ||||||
|  |  | ||||||
|  |     return i; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void CheckSeverity(Severity severity) | ||||||
|  | { | ||||||
|  |     Logger::SetConsoleSeverity(severity); | ||||||
|  |     auto sev = Logger::GetConsoleSeverity(); | ||||||
|  |  | ||||||
|  |     cout << "##### testing severity '" << Logger::SeverityName(sev) << "' (" << static_cast<int>(sev) << "), Logging(): " << std::boolalpha << Logger::Logging(sev) << endl; | ||||||
|  |  | ||||||
|  |     for (uint32_t i = 0; i < Logger::fSeverityNames.size(); ++i) { | ||||||
|  |         if (sev == Severity::nolog) { | ||||||
|  |             if (i == static_cast<int>(fair::Severity::fatal)) { | ||||||
|  |                 if (!Logger::Logging(static_cast<Severity>(i))) { | ||||||
|  |                     throw runtime_error(ToStr("expecting to be logging ", Logger::fSeverityNames.at(i), " during ", sev, ", but it is not.")); | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 if (Logger::Logging(static_cast<Severity>(i))) { | ||||||
|  |                     throw runtime_error(ToStr("expecting to NOT be logging ", Logger::fSeverityNames.at(i), " during ", sev, ", but it is.")); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             if (i >= static_cast<unsigned int>(sev)) { | ||||||
|  |                 if (!Logger::Logging(static_cast<Severity>(i))) { | ||||||
|  |                     throw runtime_error(ToStr("expecting to be logging ", Logger::fSeverityNames.at(i), " during ", sev, ", but it is not.")); | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 if (Logger::Logging(static_cast<Severity>(i))) { | ||||||
|  |                     throw runtime_error(ToStr("expecting to NOT be logging ", Logger::fSeverityNames.at(i), " during ", sev, ", but it is.")); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     uint32_t i = 0; | ||||||
|  |     i = printEverySeverity(i); | ||||||
|  |     if (sev == Severity::nolog) { | ||||||
|  |         if (i != 1) { | ||||||
|  |             throw runtime_error(ToStr("expected: i==1, found: i==", i)); | ||||||
|  |         } | ||||||
|  |     } else { | ||||||
|  |         if (i != Logger::fSeverityNames.size() - static_cast<int>(sev)) { | ||||||
|  |             throw runtime_error(ToStr("expected: i==", Logger::fSeverityNames.size() - static_cast<int>(sev) - 1, ", found: i==", i)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     try { | ||||||
|  |         Logger::SetConsoleColor(true); | ||||||
|  |  | ||||||
|  |         cout << "##### testing " << Logger::fSeverityNames.size() << " severities..." << endl; | ||||||
|  |         for (uint32_t i = 0; i < Logger::fSeverityNames.size(); ++i) { | ||||||
|  |             CheckSeverity(static_cast<Severity>(i)); | ||||||
|  |         } | ||||||
|  |     } catch (runtime_error& rte) { | ||||||
|  |         cout << rte.what() << endl; | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
							
								
								
									
										151
									
								
								test/sinks.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								test/sinks.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "Common.h" | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <fstream> | ||||||
|  | #include <iostream> | ||||||
|  | #include <random> | ||||||
|  | #include <sstream> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  | using namespace fair::logger::test; | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  | #ifdef FAIR_MIN_SEVERITY | ||||||
|  |     if (static_cast<int>(Severity::FAIR_MIN_SEVERITY) > static_cast<int>(Severity::warn)) { | ||||||
|  |         cout << "test requires at least FAIR_MIN_SEVERITY == warn to run, skipping" << endl; | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |         Logger::SetConsoleColor(false); | ||||||
|  |         Logger::SetConsoleSeverity(Severity::nolog); | ||||||
|  |         Logger::SetVerbosity(Verbosity::low); | ||||||
|  |  | ||||||
|  |         if (Logger::Logging(Severity::warn)) { cout << "Logger expected to NOT log warn, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (Logger::Logging(Severity::error)) { cout << "Logger expected to NOT log error, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::fatal)) { cout << "Logger expected to log fatal, but it reports not to" << endl; return 1; } | ||||||
|  |  | ||||||
|  |         cout << "##### adding file sink with warn severity" << endl; | ||||||
|  |         random_device rd; | ||||||
|  |         mt19937 gen(rd()); | ||||||
|  |         uniform_int_distribution<> distrib(1, 65536); | ||||||
|  |         string name = Logger::InitFileSink(Severity::warn, string("test_log_" + to_string(distrib(gen))), true); | ||||||
|  |  | ||||||
|  |         if (Logger::GetFileSeverity() != Severity::warn) { | ||||||
|  |             throw runtime_error(ToStr("File sink severity (", Logger::GetFileSeverity(), ") does not match the expected one (", Severity::warn, ")")); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         CheckOutput("^\\[FATAL\\] fatal\n$", [](){ | ||||||
|  |             LOG(state) << "state"; | ||||||
|  |             LOG(warn) << "warning"; | ||||||
|  |             LOG(error) << "error"; | ||||||
|  |             LOG(fatal) << "fatal"; | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         if (Logger::Logging(Severity::state)) { cout << "Logger expected to NOT log warn, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::warn)) { cout << "Logger expected to log warn, but it reports not to" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::error)) { cout << "Logger expected to log error, but it reports not to" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::fatal)) { cout << "Logger expected to log fatal, but it reports not to" << endl; return 1; } | ||||||
|  |  | ||||||
|  |         ifstream t(name); | ||||||
|  |         stringstream buffer; | ||||||
|  |         buffer << t.rdbuf(); | ||||||
|  |         string fileContent = buffer.str(); | ||||||
|  |  | ||||||
|  |         if (fileContent != "[WARN] warning\n[ERROR] error\n[FATAL] fatal\n") { | ||||||
|  |             throw runtime_error(ToStr("unexpected file sink output. expected:\n[WARN] warning\n[ERROR] error\n[FATAL] fatal\nfound:\n", fileContent)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         cout << "##### removing file sink with warn severity" << endl; | ||||||
|  |         Logger::RemoveFileSink(); | ||||||
|  |  | ||||||
|  |         if (Logger::Logging(Severity::warn)) { cout << "Logger expected to NOT log warn, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (Logger::Logging(Severity::error)) { cout << "Logger expected to NOT log error, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::fatal)) { cout << "Logger expected to log fatal, but it reports not to" << endl; return 1; } | ||||||
|  |  | ||||||
|  |         cout << "##### adding custom sink with warn severity" << endl; | ||||||
|  |  | ||||||
|  |         Logger::AddCustomSink("CustomSink", "warn", [](const string& content, const LogMetaData& metadata) | ||||||
|  |         { | ||||||
|  |             cout << "CustomSink " << content << endl; | ||||||
|  |  | ||||||
|  |             if (metadata.severity != Severity::warn && metadata.severity != Severity::error && metadata.severity != Severity::fatal) { | ||||||
|  |                 throw runtime_error(ToStr("unexpected severity message arrived at custom sink that accepts only warn,error,fatal: ", metadata.severity)); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (metadata.severity_name != "WARN" && metadata.severity_name != "ERROR" && metadata.severity_name != "FATAL") { | ||||||
|  |                 throw runtime_error(ToStr("unexpected severity name arrived at custom sink that accepts only warn,error,fatal: ", metadata.severity_name)); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         if (Logger::GetCustomSeverity("CustomSink") != Severity::warn) { | ||||||
|  |             throw runtime_error(ToStr("File sink severity (", Logger::GetCustomSeverity("CustomSink"), ") does not match the expected one (", Severity::warn, ")")); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         bool oorThrown = false; | ||||||
|  |         try { | ||||||
|  |             Logger::GetCustomSeverity("NonExistentSink"); | ||||||
|  |         } catch (const out_of_range& oor) { | ||||||
|  |             oorThrown = true; | ||||||
|  |         } | ||||||
|  |         if (!oorThrown) { | ||||||
|  |             throw runtime_error("Did not detect a severity request from a non-existent sink"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         CheckOutput("^CustomSink warning\nCustomSink error\nCustomSink fatal\n\\[FATAL\\] fatal\n$", [](){ | ||||||
|  |             LOG(state) << "state"; | ||||||
|  |             LOG(warn) << "warning"; | ||||||
|  |             LOG(error) << "error"; | ||||||
|  |             LOG(fatal) << "fatal"; | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         if (Logger::Logging(Severity::state)) { cout << "Logger expected to NOT log warn, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::warn)) { cout << "Logger expected to log warn, but it reports not to" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::error)) { cout << "Logger expected to log error, but it reports not to" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::fatal)) { cout << "Logger expected to log fatal, but it reports not to" << endl; return 1; } | ||||||
|  |  | ||||||
|  |         cout << "##### removing custom sink with error severity" << endl; | ||||||
|  |  | ||||||
|  |         bool caught = false; | ||||||
|  |         try { | ||||||
|  |             Logger::AddCustomSink("CustomSink", Severity::error, [](const string& /*content*/, const LogMetaData& /*metadata*/){}); | ||||||
|  |         } catch (runtime_error& rte) { | ||||||
|  |             caught = true; | ||||||
|  |         } | ||||||
|  |         if (!caught) { | ||||||
|  |             throw runtime_error("expected to throw a runtime_error upon adding sink with same key, but none was thrown"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Logger::RemoveCustomSink("CustomSink"); | ||||||
|  |  | ||||||
|  |         if (Logger::Logging(Severity::warn)) { cout << "Logger expected to NOT log warn, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (Logger::Logging(Severity::error)) { cout << "Logger expected to NOT log error, but it reports to do so" << endl; return 1; } | ||||||
|  |         if (!Logger::Logging(Severity::fatal)) { cout << "Logger expected to log fatal, but it reports not to" << endl; return 1; } | ||||||
|  |  | ||||||
|  |         caught = false; | ||||||
|  |         try { | ||||||
|  |             Logger::RemoveCustomSink("bla"); | ||||||
|  |         } catch (runtime_error& rte) { | ||||||
|  |             caught = true; | ||||||
|  |         } | ||||||
|  |         if (!caught) { | ||||||
|  |             throw runtime_error("expected to throw a runtime_error upon removing non-existent sink, but none was thrown"); | ||||||
|  |         } | ||||||
|  |     } catch (runtime_error& rte) { | ||||||
|  |         cout << rte.what() << endl; | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
							
								
								
									
										67
									
								
								test/threads.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								test/threads.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "Common.h" | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  | #include <thread> | ||||||
|  | #include <vector> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  | using namespace fair::logger::test; | ||||||
|  |  | ||||||
|  | void f() | ||||||
|  | { | ||||||
|  |     LOG(fatal) << "a" << "b" << "c" << "d" << "e" << "f" << "g" << "h" << "i" << "j" << "k" << "l" << "m" << "n" << "o" << "p" << "q" << "r" << "s" << "t" << "u" << "v" << "w" << "x" << "y" << "z"; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     try { | ||||||
|  |         Logger::SetConsoleColor(false); | ||||||
|  |         Logger::SetConsoleSeverity(Severity::fatal); | ||||||
|  |         Logger::SetVerbosity(Verbosity::veryhigh); | ||||||
|  |  | ||||||
|  |         CheckOutput( | ||||||
|  | R"(^\[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[.*:\d+:.*\] abcdefghijklmnopqrstuvwxyz | ||||||
|  | \[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[.*:\d+:.*\] abcdefghijklmnopqrstuvwxyz | ||||||
|  | \[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[.*:\d+:.*\] abcdefghijklmnopqrstuvwxyz | ||||||
|  | \[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[.*:\d+:.*\] abcdefghijklmnopqrstuvwxyz | ||||||
|  | \[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[.*:\d+:.*\] abcdefghijklmnopqrstuvwxyz | ||||||
|  | $)", []() { | ||||||
|  |             thread t1(f); | ||||||
|  |             thread t2(f); | ||||||
|  |             thread t3(f); | ||||||
|  |             thread t4(f); | ||||||
|  |             thread t5(f); | ||||||
|  |             t1.join(); | ||||||
|  |             t2.join(); | ||||||
|  |             t3.join(); | ||||||
|  |             t4.join(); | ||||||
|  |             t5.join(); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         thread t1(f); | ||||||
|  |         thread t2(f); | ||||||
|  |         thread t3(f); | ||||||
|  |         thread t4(f); | ||||||
|  |         thread t5(f); | ||||||
|  |         t1.join(); | ||||||
|  |         t2.join(); | ||||||
|  |         t3.join(); | ||||||
|  |         t4.join(); | ||||||
|  |         t5.join(); | ||||||
|  |     } catch (runtime_error& rte) { | ||||||
|  |         cout << rte.what() << endl; | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
							
								
								
									
										102
									
								
								test/verbosity.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								test/verbosity.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | /******************************************************************************** | ||||||
|  |  * Copyright (C) 2014-2020 GSI Helmholtzzentrum fuer Schwerionenforschung GmbH  * | ||||||
|  |  *                                                                              * | ||||||
|  |  *              This software is distributed under the terms of the             * | ||||||
|  |  *              GNU Lesser General Public Licence (LGPL) version 3,             * | ||||||
|  |  *                  copied verbatim in the file "LICENSE"                       * | ||||||
|  |  ********************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "Common.h" | ||||||
|  | #include <Logger.h> | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  | #include <string> | ||||||
|  |  | ||||||
|  | using namespace std; | ||||||
|  | using namespace fair; | ||||||
|  | using namespace fair::logger::test; | ||||||
|  |  | ||||||
|  | int main() | ||||||
|  | { | ||||||
|  |     try { | ||||||
|  |         Logger::SetConsoleColor(false); | ||||||
|  |         Logger::SetConsoleSeverity(Severity::fatal); | ||||||
|  |  | ||||||
|  |         auto spec1 = VerbositySpec::Make(VerbositySpec::Info::file_line_function, VerbositySpec::Info::process_name); | ||||||
|  |         auto spec2 = VerbositySpec::Make(VerbositySpec::Info::process_name, VerbositySpec::Info::file_line_function); | ||||||
|  |  | ||||||
|  |         Logger::DefineVerbosity(Verbosity::user1, spec1); | ||||||
|  |         Logger::SetVerbosity(Verbosity::user1); // spec1 on user1 | ||||||
|  |         CheckOutput(ToStr(R"(^\[.*:\d{2}:.*\]\[.*\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::DefineVerbosity(Verbosity::user1, spec2); | ||||||
|  |         Logger::SetVerbosity(Verbosity::user1); // spec2 on user1 | ||||||
|  |         CheckOutput(ToStr(R"(^\[.*\]\[.*:\d{2}:.*\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::DefineVerbosity(Verbosity::user2, spec1); | ||||||
|  |         Logger::SetVerbosity(Verbosity::user2); // spec1 on user2 | ||||||
|  |         CheckOutput(ToStr(R"(^\[.*:\d{2}:.*\]\[.*\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::verylow); // content | ||||||
|  |         CheckOutput("^content\n$", []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::low); // [severity] content | ||||||
|  |         CheckOutput(ToStr(R"(^\[FATAL\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::medium); // [HH:MM:SS][severity] content | ||||||
|  |         CheckOutput(ToStr(R"(^\[\d{2}:\d{2}:\d{2}\]\[FATAL\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::high); // [process_name][HH:MM:SS][severity] content | ||||||
|  |         CheckOutput(ToStr(R"(^\[.*\]\[\d{2}:\d{2}:\d{2}\]\[FATAL\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::veryhigh); // [process_name][HH:MM:SS:µS][severity][file:line:function] content | ||||||
|  |         CheckOutput(ToStr(R"(^\[.*\]\[\d{2}:\d{2}:\d{2}\.\d{6}\]\[FATAL\]\[.*:\d+:.*\])", " content\n$"), []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetConsoleColor(true); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::verylow); // content | ||||||
|  |         CheckOutput( | ||||||
|  |             "^" | ||||||
|  |             "content\n" | ||||||
|  |             "$", []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::low); // [severity] content | ||||||
|  |         CheckOutput( | ||||||
|  |             "^" | ||||||
|  |             "\\[\033\\[01;31mFATAL\033\\[0m\\]" | ||||||
|  |             " content\n" | ||||||
|  |             "$", []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::medium); // [HH:MM:SS][severity] content | ||||||
|  |         CheckOutput( | ||||||
|  |             "^" | ||||||
|  |             "\\[\033\\[01;36m\\d{2}:\\d{2}:\\d{2}\033\\[0m\\]" | ||||||
|  |             "\\[\033\\[01;31mFATAL\033\\[0m\\]" | ||||||
|  |             " content\n" | ||||||
|  |             "$", []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::high); // [process_name][HH:MM:SS][severity] content | ||||||
|  |         CheckOutput( | ||||||
|  |             "^" | ||||||
|  |             "\\[\033\\[01;34m.*\033\\[0m\\]" | ||||||
|  |             "\\[\033\\[01;36m\\d{2}:\\d{2}:\\d{2}\033\\[0m\\]" | ||||||
|  |             "\\[\033\\[01;31mFATAL\033\\[0m\\]" | ||||||
|  |             " content\n" | ||||||
|  |             "$", []() { LOG(fatal) << "content"; }); | ||||||
|  |  | ||||||
|  |         Logger::SetVerbosity(Verbosity::veryhigh); // [process_name][HH:MM:SS:µS][severity][file:line:function] content | ||||||
|  |         CheckOutput( | ||||||
|  |             "^" | ||||||
|  |             "\\[\033\\[01;34m.*\033\\[0m\\]" | ||||||
|  |             "\\[\033\\[01;36m\\d{2}:\\d{2}:\\d{2}\\.\\d{6}\033\\[0m\\]" | ||||||
|  |             "\\[\033\\[01;31mFATAL\033\\[0m\\]" | ||||||
|  |             "\\[\033\\[01;34m.*\033\\[0m:\033\\[01;33m\\d+\033\\[0m:\033\\[01;34m.*\033\\[0m\\]" | ||||||
|  |             " content\n" | ||||||
|  |             "$", []() { LOG(fatal) << "content"; }); | ||||||
|  |     } catch (runtime_error& rte) { | ||||||
|  |         cout << rte.what() << endl; | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user