Skip to content
/ CECS Public
forked from terablade2001/CECS

CECS (C/C++ Error Control System) is a pure C libary, with C++ support, for textual informative error control. Errors, Warnings or Info that is recording during runtime can be traced back to the file and line where the error occured.

License

Notifications You must be signed in to change notification settings

D-os/CECS

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

19 Commits
 
 
 
 
 
 
 
 

Repository files navigation

CECS

What is CECS ?

CECS (C Error Control System) is a C/C++ libary for textual informative error control. Errors, Warnings, Info (or other types) that are recording during runtime can be traced back to the file and line where the error occurred. Error Logging include information like:

  • Error Type: You can define your own error types usign a number. Default types are: Error, Warning, Info, Debug.
  • Module-Name (Each Function/File/Library can be a Module)
  • Error ID number (To describe an error in a typical way)
  • File-Name where the error occured.
  • Line-Number where the error occured.
  • Analytical Textual Information for further understanding of the error.

Why should I use CECS ?

If you need a bit more advanced system of Error Control / Logging, you can try CECS which is self-included as source code in each compilation. All Errors are being stored with information tags in internal memory and can be printed-out in any format when they are requested. Useful to track-down errors, on complicated projects.

C++ Target design

CECS was start as a C implementation, and over the C implementation a C++ class and relative MACROs have been implemented for C++ support. The C++ design which is proposed to be used, is aiming on four targets:

  • Full Error tracking The user should be able to understand an error and track back to specific code of functions that called before that error occurs. This error tracking is providing a full understanding of the system behavior that led to the error.
  • Reduce required type-writing An error should be checked almost like an "if" statement with a "then" report.
  • Support for basic exception handling For each error checking condition, the user should be able to decide easily the stop actions; return something or throw an exception?
  • Common error reporting among different libraries. In a program different libraries are called that may use functions from other libraries. If they've build using CECS, for each captured error there will be generated an error tracking report referring properly each executed library's function that contributed to the error generation.

Design rules

For maximizing the CECS abilities there are some rules to be followed by the user:

  • For each function user should do all possible error checks using _ERR/I/N/T(expression, report) macros.
  • After each function call user should check possible error either via _ERR/I/N/T(expression, report) macros either via the _CHECKR/I/N/T_ macros. The _ERR/I/N/T(expression, report) macros are preferred as they provide the generation of custom report of the error at that point, which means more semantic information.
  • If there is a main() function in the program there should be a try-catch statement inside the main() and all errors in the main() function to be checked via the _ERRT(expression, report) macro or the _CHECKRT_ macro.
    • If there is not any main() function then the first level functions should return an integer or a pointer. When error should be returned the integer should be < 0 and the pointer to be NULL. In this case the _ERRI(expression, report) and the _ERRN(expression, report) macros should be used in those functions.

Easy C++ CECS Setup

In order to use the C++ CECS copy the 4 files of the folder C++/src/ as is (with the includefolder) in your project. For example:

mkdir <myproject>/CECS
cp -rf C++/src/ <myproject>/CECS/

Then include the two .cpp files (CECS.cpp and CECS-c.cpp) in your project and also the <myproject>/CECs/include/CECS.hpp file in every C++ file that you are going to use CECS to.

For each C++ file create a static CECS instance as below, where XXX is a specific short-name that can be used to describe a group of files (i.e. a module or a library or even a single C++ file).

static CECS ECS("XXX")
// or if need to provide specific unique CECS Name:
static CES ECS("XXX","ECS-Name")

Enable the definition #define __ECSOBJ__ ECS in your project (i.e. -D__ECSOBJ__=ECS in compilation flags)

C++ Quick Macros

1. Expression Errors with reports

Given an expression that validates to True and a custom error report:

  • _ERR
    Logs an Error and returns (void)
  • _ERRI
    Logs an Error and return _CECS_DEFAULT_ERRID=-9999
  • _ERRN
    Logs an Error and return NULL
  • _ERRT
    Logs an Error and throw exception
  • _ERRL
    Just logs an error and continue
  • _ERRO
    Logs an error and apply custom user-defined code.

2. Automated Error Checking from records.

If an error has been reported by the use of the previous macros, then the following macros automatically search for errors and if found they do the following:

  • _CHECKR_
    Logs a new error with standard report and returns (void)
  • _CHECKRI_
    Logs a new error with standard report and return _CECS_DEFAULT_ERRID=-9999
  • _CHECKRN_
    Logs a new error with standard report and return NULL
  • _CHECKRT_
    Logs a new error with standard report and throw exception
  • _CHECKRL_
    Logs a new error with standard report and apply custom user-defined code.
  • _CHECKRO_
    Logs a new error with standard report and return

C++ Example

#include "CECS/include/CECS.hpp"

#define __ECSOBJ__ ECS
static CECS ECS("Main-Module","Main-CECS");
using namespace std;



int TableAt = 2;



int TableIsAt(int _tableat) {
	_ERRI((_tableat < 0) || (_tableat > 2),"TableIsAt():: Works only for input 0, 1 or 2. Got input = (%i)",_tableat)
	_ERRL(_tableat==0,"TableIsAt():: Zero value for table!")
	_ERRO(_tableat==2,  { cout<<"User Code Executed!"<<endl; return 8; }, "TableIsAt():: User code execution detected!")
	return 7;
}



int PersonIsInTable(const char* name) {
	string namestr(name);
	_ERRI(namestr.compare("Vasileios"),"Unknown name on tables!")
	int TableNo = TableIsAt(TableAt);
	_ERRI(TableNo < 0,"Wrong position of Table given the person [%s]",name)
	_WARN(TableNo==8, "Warning: TableId == 8!")
	_CHECKRI_
	return TableNo;
}



const char* BallIsInHandsOf(const string& nickname) {
	_ERRN(nickname.compare("terablade2001"),"Unknown nickname (%s)",nickname.c_str())
	static string name("Vasileios");
	
	int TableNo = PersonIsInTable(name.c_str());
	_ERRN(TableNo < 0,"Return of PersonIsInTable(%s) was negative",nickname.c_str())
	return name.c_str();
}



void FindTheBall() {
	const string nickname("terablade2001");
	const char* name = BallIsInHandsOf(nickname);
	_ERRT(name == NULL,"Return of BallIsInHandsOf() is NULL!")
	cout << "The ball is at person ["<< name << "]" << endl;
}



int main(int argc, char **argv)
{
	try {
		FindTheBall();
		_CHECKRT_
		cout << "Test Completed Succesfully!" << endl;
	} catch(std::exception &e) {
		 std::cout<< std::endl<<"(*) Exception occured: "<< std::endl << "  --> " << e.what() << std::endl;
	}
}

The above code produce the following results:

  • If TableAt == 1
  The ball is at person [Vasileios]
  Test Completed Succesfully!
  • If TableAt == 0
(*) Exception occured:
  --> ------- Main-CECS:: 4 Record(s) of ALL Types recorded! -------
[ERROR  ]> #Main-Module: [main.cpp], 15 |> TableIsAt():: Zero value for table!
[ERROR  ]> #Main-Module: [main.cpp], 28 |> CECS_CHECKERROR captured: Function return executed.
[ERROR  ]> #Main-Module: [main.cpp], 39 |> Return of PersonIsInTable(terablade2001) was negative
[ERROR  ]> #Main-Module: [main.cpp], 48 |> Return of BallIsInHandsOf() is NULL!
-------------------------------------------------------
  • If TableAt == 2
User Code Executed!

(*) Exception occured:
  --> ------- Main-CECS:: 5 Record(s) of ALL Types recorded! -------
[ERROR  ]> #Main-Module: [main.cpp], 16 |> TableIsAt():: User code execution detected!
[WARNING]> #Main-Module: [main.cpp], 27 |> Warning: TableId == 8!
[ERROR  ]> #Main-Module: [main.cpp], 28 |> CECS_CHECKERROR captured: Function return executed.
[ERROR  ]> #Main-Module: [main.cpp], 39 |> Return of PersonIsInTable(terablade2001) was negative
[ERROR  ]> #Main-Module: [main.cpp], 48 |> Return of BallIsInHandsOf() is NULL!
-------------------------------------------------------
  • For any other value of TableAt:
(*) Exception occured:
  --> ------- Main-CECS:: 4 Record(s) of ALL Types recorded! -------
[ERROR  ]> #Main-Module: [main.cpp], 14 |> TableIsAt():: Works only for input 0, 1 or 2. Got input = (-1)
[ERROR  ]> #Main-Module: [main.cpp], 26 |> Wrong position of Table given the person [Vasileios]
[ERROR  ]> #Main-Module: [main.cpp], 39 |> Return of PersonIsInTable(terablade2001) was negative
[ERROR  ]> #Main-Module: [main.cpp], 48 |> Return of BallIsInHandsOf() is NULL!
-------------------------------------------------------

C Example

An example taken from the test code "t00_test.c":

#include "CECS.h"

// Creating Error and Warning records in a function which represents Module B.
int test05_ModB(sCECS* cecs) {
	CECS_Initialize("MainCECS", cecs);
	const char CECS_ModName[] = "Module B";
	CECS_MERR(1,"Test error in test05_ModB.");
	CECS_MWARN(1,"Test warning in test05_ModB.");
	return 0;
}

// Creating Error and Warning records in another function which represents Module A.
// Both test05_ModA and test05_ModB are sharing the same CECS object, thus they write their records in the same Error Control System.
int test05_ModA(void) {
	sCECS* CECS = CECS_Initialize("MainCECS", NULL);
	const char CECS_ModName[] = "Module A";
	CECS_MERR(1,"Test Error in test05_ModA.");
	test05_ModB(CECS);
	CECS_MWARN(1,"Test Warning in test05_ModA.");

	printf("%s\n", CECS_str(_CECS_ERRTYPE_ALL));
	
	CECS_Shutdown(NULL);
	return 0;
}

// The results after Calling they main() function will be: 
// ------- CECS:: 4 Errors of ALL Types recorded! -------
// [ET:0]-> #Module A: -9999 | [t00_test.c], 122:: Test Error in test05_ModA.
// [ET:0]-> #Module B: -9999 | [t00_test.c], 134:: Test error in test05_ModB.
// [ET:1]-> #Module B: -9998 | [t00_test.c], 135:: Test warning in test05_ModB.
// [ET:1]-> #Module A: -9998 | [t00_test.c], 124:: Test Warning in test05_ModA.
// -------------------------------------------------------
int main(int argc, char **argv) {
	return test05_ModA();
}

About

CECS (C/C++ Error Control System) is a pure C libary, with C++ support, for textual informative error control. Errors, Warnings or Info that is recording during runtime can be traced back to the file and line where the error occured.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • C 58.8%
  • C++ 39.5%
  • CMake 1.6%
  • Batchfile 0.1%