summaryrefslogblamecommitdiffstats
path: root/src/LoggerListeners.cpp
blob: 91f166bda418417ad985c5054b3b5ff57ddc919e (plain) (tree)
1
2
3
4
5
6
7
8
9




                            

                 

                                                                        
                         
                      
                                                            


      
                                        


                                           
                  
 

                                                                             
 


                                                                                           
                                                         




                                              



 



                                                 
                                                       





                                                                                        
 
                             

                                                                                                   
                                                                  



                                                                                                     

 


                                                                                 
                                                                                          


                                                        
                                 
                                                        


                                                                                                     
                                                     
                                 
                                                          


                                                                                                          
                                                        
                                 

                                                                                       

                                              
                                                      

                                                       

                                                                                       
                                 


                                                                   

 
                                                           
                 

                                                                                   
 
                
 


                                            


 
                       


 








                                                                                 
                                 
                                                                          
                                                          
                                              
                                 
                                                     
                                 
                                                          

                                                             
                                 
                                                        
                                 

                                                             
                                              
                                 
                                                      
                                 


                                                                  
                                 

                         

 

                                                           
                                                          


                                          
 

      



 









                                                                                   
                         

                                                       
                         
                                             
                         

                                                        
                         
                                                
                         

                                                           
                         
                                              
                         

                                                         
                         






                                                                            

 












                                                                                   
                                                                         
 

                        
                                                                  

         





                                                                                      
                                                                         

                                                                    
                                                                                                          


                    
                                                                                
                 
                                                    


                                                    
                                                                           


                    
                                                                                

                 
                                                                        


              






                                                                                

                                   
 
       
 
                              
 


                                                                                                            
                                                           










                                                                                                      
 
                                                                                   
         


                                                      
                 





















                                                         
                 

                                                                         
                 
                                       
                 
         
 












                                                                       
         
                                        
         
                                           
 

 

#include "Globals.h"

#include "LoggerListeners.h"

#include <chrono>

#if defined(_WIN32)
	#include <io.h>  // Needed for _isatty(), not available on Linux
	#include <time.h>
#elif defined(__linux)
	#include <unistd.h>  // Needed for isatty() on Linux
#endif


#if defined(_WIN32) || defined (__linux)
	class cColouredConsoleListener
		: public cLogger::cListener
	{
	protected:

		virtual void SetLogColour(cLogger::eLogLevel a_LogLevel) = 0;
		virtual void SetDefaultLogColour() = 0;

		virtual void Log(AString a_Message, cLogger::eLogLevel a_LogLevel) override
		{
			SetLogColour(a_LogLevel);
			fputs(a_Message.c_str(), stdout);
			SetDefaultLogColour();
		}
	};
#endif





#ifdef _WIN32
	class cWindowsConsoleListener
		: public cColouredConsoleListener
	{
		typedef cColouredConsoleListener super;
	public:
		cWindowsConsoleListener(HANDLE a_Console, WORD a_DefaultConsoleAttrib) :
			m_Console(a_Console),
			m_DefaultConsoleAttrib(a_DefaultConsoleAttrib)
		{
		}

		#ifdef _DEBUG
			virtual void Log(AString a_Message, cLogger::eLogLevel a_LogLevel) override
			{
				super::Log(a_Message, a_LogLevel);
				// In a Windows Debug build, output the log to debug console as well:
				OutputDebugStringA(a_Message.c_str());
			}
		#endif


		virtual void SetLogColour(cLogger::eLogLevel a_LogLevel) override
		{
			// by default, gray on black
			WORD Attrib = FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED;
			switch (a_LogLevel)
			{
				case cLogger::llRegular:
				{
					// Gray on black
					Attrib = FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED;
					break;
				}
				case cLogger::llInfo:
				{
					// Yellow on black
					Attrib = FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_INTENSITY;
					break;
				}
				case cLogger::llWarning:
				{
					// Red on black
					Attrib = FOREGROUND_RED | FOREGROUND_INTENSITY;
					break;
				}
				case cLogger::llError:
				{
					// Black on red
					Attrib = BACKGROUND_RED | BACKGROUND_INTENSITY;
					break;
				}
			}
			SetConsoleTextAttribute(m_Console, Attrib);
		}


		virtual void SetDefaultLogColour() override
		{
			SetConsoleTextAttribute(m_Console, m_DefaultConsoleAttrib);
		}

	private:

		HANDLE m_Console;
		WORD m_DefaultConsoleAttrib;
	};



#elif defined (__linux)



	class cLinuxConsoleListener
		: public cColouredConsoleListener
	{
	public:
		virtual void SetLogColour(cLogger::eLogLevel a_LogLevel) override
		{
			switch (a_LogLevel)
			{
				case cLogger::llRegular:
				{
					// Whatever the console default is
					printf("\x1b[0m");
					break;
				}
				case cLogger::llInfo:
				{
					// Yellow on black
					printf("\x1b[33;1m");
					break;
				}
				case cLogger::llWarning:
				{
					// Red on black
					printf("\x1b[31;1m");
					break;
				}
				case cLogger::llError:
				{
					// Yellow on red
					printf("\x1b[1;33;41;1m");
					break;
				}
			}
		}


		virtual void SetDefaultLogColour() override
		{
			// Whatever the console default is
			printf("\x1b[0m");
		}
	};

#endif





class cVanillaCPPConsoleListener
	: public cLogger::cListener
{
public:
	virtual void Log(AString a_Message, cLogger::eLogLevel a_LogLevel) override
	{
		AString LogLevelString;
		switch (a_LogLevel)
		{
			case cLogger::llRegular:
			{
				LogLevelString = "Log";
				break;
			}
			case cLogger::llInfo:
			{
				LogLevelString = "Info";
				break;
			}
			case cLogger::llWarning:
			{
				LogLevelString = "Warning";
				break;
			}
			case cLogger::llError:
			{
				LogLevelString = "Error";
				break;
			}
		}
		printf("%s: %s", LogLevelString.c_str(), a_Message.c_str());
	}
};





// Listener for when stdout is closed, i.e. When running as a daemon.
class cNullConsoleListener
	: public cLogger::cListener
{
	virtual void Log(AString a_Message, cLogger::eLogLevel a_LogLevel) override
	{
	}
};





std::unique_ptr<cLogger::cListener> MakeConsoleListener(bool a_IsService)
{
	if (a_IsService)
	{
		return cpp14::make_unique<cNullConsoleListener>();
	}

	#ifdef _WIN32
		// See whether we are writing to a console the default console attrib:
		bool ShouldColorOutput = (_isatty(_fileno(stdin)) != 0);
		if (ShouldColorOutput)
		{
			CONSOLE_SCREEN_BUFFER_INFO sbi;
			HANDLE Console = GetStdHandle(STD_OUTPUT_HANDLE);
			GetConsoleScreenBufferInfo(Console, &sbi);
			WORD DefaultConsoleAttrib = sbi.wAttributes;
			return cpp14::make_unique<cWindowsConsoleListener>(Console, DefaultConsoleAttrib);
		}
		else
		{
			return cpp14::make_unique<cVanillaCPPConsoleListener>();
		}
	#elif defined (__linux) && !defined(ANDROID)
		// TODO: lookup terminal in terminfo
		if (isatty(fileno(stdout)))
		{
			return cpp14::make_unique<cLinuxConsoleListener>();
		}
		else
		{
			return cpp14::make_unique<cVanillaCPPConsoleListener>();
		}
	#else
		return cpp14::make_unique<cVanillaCPPConsoleListener>();
	#endif
}





////////////////////////////////////////////////////////////////////////////////
// cFileListener:

class cFileListener
	: public cLogger::cListener
{
public:

	cFileListener(void) {}

	bool Open()
	{
		// Assume creation succeeds, as the API does not provide a way to tell if the folder exists.
		cFile::CreateFolder(FILE_IO_PREFIX "logs");
		bool success = m_File.Open(
			FILE_IO_PREFIX + Printf(
				"logs/LOG_%d.txt",
				std::chrono::duration_cast<std::chrono::duration<int, std::ratio<1>>>(
					std::chrono::system_clock::now().time_since_epoch()
				).count()
			),
			cFile::fmAppend
		);
		return success;
	}

	virtual void Log(AString a_Message, cLogger::eLogLevel a_LogLevel) override
	{
		const char * LogLevelPrefix = "Unkn ";
		bool ShouldFlush = false;
		switch (a_LogLevel)
		{
			case cLogger::llRegular:
			{
				LogLevelPrefix = "     ";
				break;
			}
			case cLogger::llInfo:
			{
				LogLevelPrefix = "Info ";
				break;
			}
			case cLogger::llWarning:
			{
				LogLevelPrefix = "Warn ";
				ShouldFlush = true;
				break;
			}
			case cLogger::llError:
			{
				LogLevelPrefix = "Err  ";
				ShouldFlush = true;
				break;
			}
		}
		m_File.Printf("%s%s", LogLevelPrefix, a_Message.c_str());
		if (ShouldFlush)
		{
			m_File.Flush();
		}
	}

private:

	cFile m_File;
};





std::pair<bool, std::unique_ptr<cLogger::cListener>> MakeFileListener()
{
	auto listener = cpp14::make_unique<cFileListener>();
	if (!listener->Open())
	{
		return {false, nullptr};
	}
	return {true, std::move(listener)};
}