|
|
|
|
|
|
|
|
return ScanNameFormatter.str();
|
|
|
return ScanNameFormatter.str();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const int NoScanYet = -1;
|
|
|
|
|
|
LogicFault FaultIfSettingScanResultConfigBeforeScan("Job::setConfigurationFromScanResult() Fault(NoScanYet == ScanResultCode)");
|
|
|
|
|
|
|
|
|
void Job::setConfigurationFromScanResult(ScopeScanner& myScanner) {
|
|
|
void Job::setConfigurationFromScanResult(ScopeScanner& myScanner) {
|
|
|
|
|
|
|
|
|
|
|
|
FaultIfSettingScanResultConfigBeforeScan(NoScanYet == ScanResultCode);
|
|
|
ScanResultConfiguration = Scanners.ConfigurationForResultCode(ScanResultCode);
|
|
|
ScanResultConfiguration = Scanners.ConfigurationForResultCode(ScanResultCode);
|
|
|
|
|
|
|
|
|
//// Testing -- forcing scan results
|
|
|
//// Testing -- forcing scan results
|
|
|
|
|
|
|
|
|
ScanResultConfiguration.Action = ResultConfiguration::Hold;
|
|
|
ScanResultConfiguration.Action = ResultConfiguration::Hold;
|
|
|
ScanResultConfiguration.HoldPath = "C:\\M\\Projects\\MessageSniffer\\SNF4CGP_Work\\TestEnvironment\\hold\\";
|
|
|
ScanResultConfiguration.HoldPath = "C:\\M\\Projects\\MessageSniffer\\SNF4CGP_Work\\TestEnvironment\\hold\\";
|
|
|
ScanResultConfiguration.RejectionReason = "I don't like the look of it -- so there!";
|
|
|
ScanResultConfiguration.RejectionReason = "I don't like the look of it -- so there!";
|
|
|
|
|
|
ScanResultConfiguration.LogComment = "This is my happy little log comment.";
|
|
|
ScanResultConfiguration.EmitXMLLog = true;
|
|
|
ScanResultConfiguration.EmitXMLLog = true;
|
|
|
ScanResultConfiguration.EmitClassicLog = true;
|
|
|
|
|
|
|
|
|
ScanResultConfiguration.EmitClassicLog = false;
|
|
|
ScanResultConfiguration.InjectHeaders = true;
|
|
|
ScanResultConfiguration.InjectHeaders = true;
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Job::emitLogComment() {
|
|
|
|
|
|
ostringstream O;
|
|
|
|
|
|
O << "* SNF4CGP[" << CurrentCommand.Number << "] Comment: " << ScanResultConfiguration.LogComment << endl;
|
|
|
|
|
|
OutputBuffer.append(O.str());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job::emitXMLLogData() {
|
|
|
void Job::emitXMLLogData() {
|
|
|
ostringstream O;
|
|
|
ostringstream O;
|
|
|
O << "* SNF4CGP[" << CurrentCommand.Number << "] XML Scan log data: " << endl;
|
|
|
O << "* SNF4CGP[" << CurrentCommand.Number << "] XML Scan log data: " << endl;
|
|
|
|
|
|
|
|
|
LogicFault FaultUhandledScanResultAction("Job::doAction() switch(ScanResultConfiguration.Action) fell to default!");
|
|
|
LogicFault FaultUhandledScanResultAction("Job::doAction() switch(ScanResultConfiguration.Action) fell to default!");
|
|
|
|
|
|
|
|
|
void Job::doAction() {
|
|
|
void Job::doAction() {
|
|
|
|
|
|
if(0 < ScanResultConfiguration.LogComment.length()) emitLogComment();
|
|
|
if(ScanResultConfiguration.EmitXMLLog) emitXMLLogData();
|
|
|
if(ScanResultConfiguration.EmitXMLLog) emitXMLLogData();
|
|
|
if(ScanResultConfiguration.EmitClassicLog) emitClassicLogData();
|
|
|
if(ScanResultConfiguration.EmitClassicLog) emitClassicLogData();
|
|
|
switch(ScanResultConfiguration.Action) {
|
|
|
switch(ScanResultConfiguration.Action) {
|
|
|
|
|
|
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
RuntimeCheck CheckLineEndsIdentified("Job::findHeaderInsertPoint() Check(0 < LocalLineEnd.length())");
|
|
|
RuntimeCheck CheckLineEndsIdentified("Job::findHeaderInsertPoint() Check(0 < LocalLineEnd.length())");
|
|
|
|
|
|
RuntimeCheck CheckInsertCursorPlausable("Job::findHeaderInsertPoint() Check(isMinimumSafeDistanceFromEOF(InsertCursor))");
|
|
|
|
|
|
|
|
|
|
|
|
bool Job::isMinimumSafeDistanceFromEOF(const size_t Position) {
|
|
|
|
|
|
string MinimalEmptyMessageBody = LocalLineEnd + "." + LocalLineEnd;
|
|
|
|
|
|
size_t MinimalDistanceToEOF = MinimalEmptyMessageBody.length();
|
|
|
|
|
|
size_t SafetyCalculation = Position + MinimalDistanceToEOF;
|
|
|
|
|
|
size_t AvailableMessageLength = Reader().gcount();
|
|
|
|
|
|
return (SafetyCalculation <= AvailableMessageLength);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Job::moveInsertCursorForwardToBlockTerminus(size_t& InsertCursor, string& BlockTerminus) {
|
|
|
|
|
|
size_t SafetyLimit = (Reader().gcount() - BlockTerminus.length());
|
|
|
|
|
|
for(;InsertCursor < SafetyLimit; InsertCursor++) { // Scan forward safely until we find it.
|
|
|
|
|
|
if( // At each position compare the BlockTerminus
|
|
|
|
|
|
0 == BlockTerminus.compare( // with an equal length string in the buffer.
|
|
|
|
|
|
0, BlockTerminus.length(),
|
|
|
|
|
|
reinterpret_cast<char*>(&ReadBuffer[InsertCursor]), BlockTerminus.length())
|
|
|
|
|
|
) return;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
size_t Job::findHeaderInsertPoint() { // How to find the insert point:
|
|
|
size_t Job::findHeaderInsertPoint() { // How to find the insert point:
|
|
|
identifyLocalLineEnd();
|
|
|
identifyLocalLineEnd();
|
|
|
|
|
|
|
|
|
string BlockTerminus = LocalLineEnd + LocalLineEnd;
|
|
|
string BlockTerminus = LocalLineEnd + LocalLineEnd;
|
|
|
size_t InsertCursor = 0;
|
|
|
size_t InsertCursor = 0;
|
|
|
|
|
|
|
|
|
//// Search for the end of the CGP control block.
|
|
|
|
|
|
|
|
|
moveInsertCursorForwardToBlockTerminus(InsertCursor, BlockTerminus); // Find the end of the CGP control block.
|
|
|
|
|
|
InsertCursor += BlockTerminus.length(); // Move past this block terminus.
|
|
|
|
|
|
moveInsertCursorForwardToBlockTerminus(InsertCursor, BlockTerminus); // Find the end of the message headers.
|
|
|
|
|
|
|
|
|
for(;InsertCursor < (Reader().gcount() - BlockTerminus.length()); InsertCursor++) {
|
|
|
|
|
|
if(0 == BlockTerminus.compare(reinterpret_cast<char*>(&ReadBuffer[InsertCursor]))) break;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
InsertCursor += LocalLineEnd.length(); // Split the headers BlockTerminus.
|
|
|
|
|
|
|
|
|
//// Search for the end of the message headers.
|
|
|
|
|
|
|
|
|
CheckInsertCursorPlausable(isMinimumSafeDistanceFromEOF(InsertCursor));
|
|
|
|
|
|
|
|
|
InsertCursor += BlockTerminus.length();
|
|
|
|
|
|
for(;InsertCursor < (Reader().gcount() - BlockTerminus.length()); InsertCursor++) {
|
|
|
|
|
|
if(0 == BlockTerminus.compare(reinterpret_cast<char*>(&ReadBuffer[InsertCursor]))) break;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
return InsertCursor;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
RuntimeCheck CheckFirstSegmentWriterOk("Job::writeUpToInjectionPoint() Check(Writer().good())");
|
|
|
|
|
|
|
|
|
|
|
|
void Job::writeUpToInjectionPoint(size_t InjectionPoint) {
|
|
|
|
|
|
Writer().write(reinterpret_cast<char*>(&ReadBuffer[0]), InjectionPoint);
|
|
|
|
|
|
CheckFirstSegmentWriterOk(Writer().good());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
//// The correct insert point is between the two LocalLineEnd in the BlockTerminus
|
|
|
|
|
|
|
|
|
RuntimeCheck CheckHeaderWriterOk("Job::writeHeaders() Check(Writer().good())");
|
|
|
|
|
|
|
|
|
|
|
|
void Job::writeHeaders() { // Write (inject) headers while converting
|
|
|
|
|
|
for(size_t i = 0; i < HeadersToInject.length(); i++) { // line ends to the format observed in the
|
|
|
|
|
|
switch(HeadersToInject.at(i)) { // message file.
|
|
|
|
|
|
case '\r' : { break; } // Convert by ignoring <CR> and
|
|
|
|
|
|
case '\n' : { Writer() << LocalLineEnd; break; } // triggering the LocalLineEnd on <LF>
|
|
|
|
|
|
default : { Writer().put(HeadersToInject.at(i)); break; } // All other bytes as seen in the message.
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
CheckHeaderWriterOk(Writer().good());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
InsertCursor += LocalLineEnd.length();
|
|
|
|
|
|
|
|
|
RuntimeCheck CheckEndFirstBufferWriterOk("Job::writeRestOfFirstBuffer() Check(Writer().good())");
|
|
|
|
|
|
|
|
|
return InsertCursor;
|
|
|
|
|
|
|
|
|
void Job::writeRestOfFirstBuffer(size_t InjectionPoint) {
|
|
|
|
|
|
int LengthOfRestOfBuffer = Reader().gcount() - (InjectionPoint + 1);
|
|
|
|
|
|
Writer().write(reinterpret_cast<char*>(&ReadBuffer[InjectionPoint]), LengthOfRestOfBuffer);
|
|
|
|
|
|
CheckEndFirstBufferWriterOk(Writer().good());
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job::writeUpToInjectionPoint(size_t InjectionPoint) {
|
|
|
|
|
|
|
|
|
void Job::copyFirstBufferAndInjectHeadersInMessageMove() {
|
|
|
|
|
|
size_t HeaderInjectionPoint = findHeaderInsertPoint();
|
|
|
|
|
|
writeUpToInjectionPoint(HeaderInjectionPoint);
|
|
|
|
|
|
writeHeaders();
|
|
|
|
|
|
writeRestOfFirstBuffer(HeaderInjectionPoint);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job::writeHeaders() {
|
|
|
|
|
|
|
|
|
RuntimeCheck CheckFirstBufferWriterOk("Job::copyFirstBufferInMessageMove() Check(Writer().good())");
|
|
|
|
|
|
|
|
|
|
|
|
void Job::copyFirstBufferInMessageMove() {
|
|
|
|
|
|
Writer().write(reinterpret_cast<char*>(&ReadBuffer[0]), Reader().gcount());
|
|
|
|
|
|
CheckFirstBufferWriterOk(Writer().good());
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job:: writeTheRestOfTheFirstBuffer() {
|
|
|
|
|
|
|
|
|
RuntimeFault FaultLongCopyReaderBad("Job::readLongCopyBlock() Fault(Reader().bad())");
|
|
|
|
|
|
|
|
|
|
|
|
void Job::readLongCopyBlock() {
|
|
|
|
|
|
Reader().read(reinterpret_cast<char*>(&ReadBuffer[0]), ReadBufferSize);
|
|
|
|
|
|
FaultLongCopyReaderBad(Reader().bad());
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job::injectHeadersInMessageMove() {
|
|
|
|
|
|
//// writeUpToTheInjectionPoint()
|
|
|
|
|
|
//// writeTheHeaders()
|
|
|
|
|
|
//// writeTheRestOfTheFirstBuffer()
|
|
|
|
|
|
|
|
|
RuntimeFault FaultLongCopyWriterBad("Job::writeLongCopyBlock() Fault(Writer().bad())");
|
|
|
|
|
|
|
|
|
|
|
|
void Job::writeLongCopyBlock() {
|
|
|
|
|
|
if(Reader().gcount()) {
|
|
|
|
|
|
Writer().write(reinterpret_cast<char*>(&ReadBuffer[0]), Reader().gcount());
|
|
|
|
|
|
FaultLongCopyWriterBad(Writer().bad());
|
|
|
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job::copyRestOfLongMessage() {
|
|
|
void Job::copyRestOfLongMessage() {
|
|
|
//// while(Reader.good() && !Reader.eof()) {
|
|
|
|
|
|
//// readBlockOfData()
|
|
|
|
|
|
//// writeBlockOfData()
|
|
|
|
|
|
//// }
|
|
|
|
|
|
|
|
|
while(Reader().good()) {
|
|
|
|
|
|
readLongCopyBlock();
|
|
|
|
|
|
writeLongCopyBlock();
|
|
|
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void Job::moveMessageToHoldPath(string& Destination) {
|
|
|
void Job::moveMessageToHoldPath(string& Destination) {
|
|
|
openWriter(Destination);
|
|
|
openWriter(Destination);
|
|
|
if(ScanResultConfiguration.InjectHeaders) injectHeadersInMessageMove();
|
|
|
|
|
|
|
|
|
if(ScanResultConfiguration.InjectHeaders) {
|
|
|
|
|
|
copyFirstBufferAndInjectHeadersInMessageMove();
|
|
|
|
|
|
} else {
|
|
|
|
|
|
copyFirstBufferInMessageMove();
|
|
|
|
|
|
}
|
|
|
copyRestOfLongMessage();
|
|
|
copyRestOfLongMessage();
|
|
|
closeWriter();
|
|
|
closeWriter();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
emitDISCARD();
|
|
|
emitDISCARD();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
const int NoScanYet = -1;
|
|
|
|
|
|
|
|
|
|
|
|
Job::Job(ScannerPool& S, OutputProcessor& O) : // Construct with important links.
|
|
|
Job::Job(ScannerPool& S, OutputProcessor& O) : // Construct with important links.
|
|
|
Scanners(S),
|
|
|
Scanners(S),
|
|
|
Output(O),
|
|
|
Output(O),
|