From 8d4bc7d77a62558458f7d7e635af0e8e0f0ec65b Mon Sep 17 00:00:00 2001 From: Victor Poughon <victor.poughon@cnes.fr> Date: Tue, 16 May 2017 19:03:46 +0200 Subject: [PATCH] WIP: Better error messages (RFC-91) --- .../IO/ImageIO/include/otbImageFileReader.h | 15 +- .../IO/ImageIO/include/otbImageFileReader.txx | 124 ++++------- .../IO/ImageIO/include/otbImageFileWriter.txx | 15 +- Modules/IO/ImageIO/src/otbImageIOFactory.cxx | 33 ++- .../ApplicationEngine/include/otbLogger.h | 50 +++++ .../include/otbWrapperApplication.h | 30 ++- .../include/otbWrapperInputImageParameter.txx | 9 +- .../include/otbWrapperMacros.h | 13 +- .../ApplicationEngine/src/CMakeLists.txt | 1 + .../ApplicationEngine/src/otbLogger.cxx | 61 ++++++ .../src/otbWrapperApplication.cxx | 11 +- .../otbWrapperInputFilenameListParameter.cxx | 43 +--- .../src/otbWrapperInputImageListParameter.cxx | 57 ++--- .../src/otbWrapperInputImageParameter.cxx | 25 +-- .../src/otbWrapperCommandLineLauncher.cxx | 194 +++++++++--------- 15 files changed, 337 insertions(+), 344 deletions(-) create mode 100644 Modules/Wrappers/ApplicationEngine/include/otbLogger.h create mode 100644 Modules/Wrappers/ApplicationEngine/src/otbLogger.cxx diff --git a/Modules/IO/ImageIO/include/otbImageFileReader.h b/Modules/IO/ImageIO/include/otbImageFileReader.h index daf6c58e80..c6553512d8 100644 --- a/Modules/IO/ImageIO/include/otbImageFileReader.h +++ b/Modules/IO/ImageIO/include/otbImageFileReader.h @@ -54,19 +54,14 @@ public: /** Constructor. */ ImageFileReaderException(const char *file, unsigned int line, - const char* message = "Error in IO", - const char* loc = "Unknown") : - ExceptionObject(file, line, message, loc) + const std::string& desc = "", + const std::string& filename = "") : + ExceptionObject(file, line, desc), + m_Filename(filename) { } - /** Constructor. */ - ImageFileReaderException(const std::string &file, unsigned int line, - const char* message = "Error in IO", - const char* loc = "Unknown") : - ExceptionObject(file, line, message, loc) - { - } + std::string m_Filename; }; /** \class ImageFileReader diff --git a/Modules/IO/ImageIO/include/otbImageFileReader.txx b/Modules/IO/ImageIO/include/otbImageFileReader.txx index 8f529967ad..900d1e9562 100644 --- a/Modules/IO/ImageIO/include/otbImageFileReader.txx +++ b/Modules/IO/ImageIO/include/otbImageFileReader.txx @@ -65,7 +65,6 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> m_UserSpecifiedImageIO(false), m_FileName(""), m_UseStreaming(true), - m_ExceptionMessage(""), m_ActualIORegion(), m_FilenameHelper(FNameHelperType::New()), m_AdditionalNumber(0), @@ -242,8 +241,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> } else { - throw otb::ImageFileReaderException(__FILE__, __LINE__, - "Invalid output object type"); + throw otb::ImageFileReaderException(__FILE__, __LINE__, "Invalid output object type"); } } } @@ -259,11 +257,10 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> itkDebugMacro(<< "Reading file for GenerateOutputInformation()" << this->m_FileName); // Check to see if we can read the file given the name or prefix - // if (this->m_FileName == "") - { - throw otb::ImageFileReaderException(__FILE__, __LINE__, "FileName must be specified"); - } + { + throw otb::ImageFileReaderException(__FILE__, __LINE__, "Filename must be specified."); + } // Find real image file name // !!!! Update FileName @@ -271,56 +268,25 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> bool found = GetGdalReadImageFileName(this->m_FileName, lFileName); if (found == false) { - otbMsgDebugMacro(<< "Filename was NOT unknown. May be recognized by a Image factory ! "); + otbMsgDebugMacro(<< "Filename was NOT unknown. May be recognized by a Image factory!"); } // Update FileName this->m_FileName = lFileName; // Test if the file exists and if it can be opened. // An exception will be thrown otherwise. - // We catch the exception because some ImageIO's may not actually - // open a file. Still reports file error if no ImageIO is loaded. - - try - { - m_ExceptionMessage = ""; - this->TestFileExistenceAndReadability(); - } - catch (itk::ExceptionObject & err) - { - m_ExceptionMessage = err.GetDescription(); - } - + this->TestFileExistenceAndReadability(); + if (this->m_UserSpecifiedImageIO == false) //try creating via factory - { + { this->m_ImageIO = ImageIOFactory::CreateImageIO(this->m_FileName.c_str(), otb::ImageIOFactory::ReadMode); - } - - if (this->m_ImageIO.IsNull()) - { - //this->Print(std::cerr); - otb::ImageFileReaderException e(__FILE__, __LINE__); - std::ostringstream msg; - msg << " Could not create IO object for file " - << this->m_FileName.c_str() << std::endl; - msg << " Tried to create one of the following:" << std::endl; - std::list<itk::LightObject::Pointer> allobjects = - itk::ObjectFactoryBase::CreateAllInstance("otbImageIOBase"); - for (std::list<itk::LightObject::Pointer>::iterator i = allobjects.begin(); - i != allobjects.end(); ++i) - { - otb::ImageIOBase* io = dynamic_cast<otb::ImageIOBase*>(i->GetPointer()); - // IO should never be null, but we would better check for it - if(io) - msg << " " << io->GetNameOfClass() << std::endl; - } - msg << " You probably failed to set a file suffix, or" << std::endl; - msg << " set the suffix to an unsupported type." << std::endl; - e.SetDescription(msg.str().c_str()); - throw e; - return; - } + } + // Throw error if the image wasn't loaded + if (this->m_ImageIO.IsNull()) + { + throw otb::ImageFileReaderException(__FILE__, __LINE__, "Cannot read image (probably unsupported or incorrect filename extension).", this->m_FileName); + } // Get the ImageIO MetaData Dictionary itk::MetaDataDictionary& dict = this->m_ImageIO->GetMetaDataDictionary(); @@ -331,7 +297,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> bool lVectorImage = false; if (strcmp(output->GetNameOfClass(), "VectorImage") == 0) lVectorImage= true; - + this->m_ImageIO->SetOutputImagePixelType(PixelIsComplex(dummy),lVectorImage); // Pass the dataset number (used for hdf files for example) @@ -435,9 +401,9 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> if(!m_KeywordListUpToDate && !m_FilenameHelper->GetSkipGeom()) { - + std::string lFileNameOssimKeywordlist = GetDerivedDatasetSourceFileName(m_FileName); - + // Update otb Keywordlist ImageKeywordlist otb_kwl; if (!m_FilenameHelper->ExtGEOMFileNameIsSet()) @@ -450,7 +416,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> otb_kwl = ReadGeometryFromGEOMFile(m_FilenameHelper->GetExtGEOMFileName()); otbMsgDevMacro(<< "Loading external kwl"); } - + // Don't add an empty ossim keyword list if(!otb_kwl.Empty()) { @@ -511,7 +477,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> } - + // If Skip ProjectionRef is activated, remove ProjRef from dict if (m_FilenameHelper->GetSkipCarto()) { @@ -576,9 +542,9 @@ std::string ImageFileReader<TOutputImage, ConvertPixelTraits> ::GetDerivedDatasetSourceFileName(const std::string & filename) const { - + const size_t dsds_pos = filename.find(DerivedSubdatasetPrefix); - + if(dsds_pos != std::string::npos) { // Derived subdataset from gdal @@ -605,40 +571,26 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> } std::string fileToCheck = GetDerivedDatasetSourceFileName(m_FileName); - + // Test if the file exists. if (!itksys::SystemTools::FileExists(fileToCheck.c_str())) - { - otb::ImageFileReaderException e(__FILE__, __LINE__); - std::ostringstream msg; - msg << "The file doesn't exist. " - << std::endl << "Filename = " << fileToCheck - << std::endl; - e.SetDescription(msg.str().c_str()); - throw e; - return; - } + { + throw otb::ImageFileReaderException (__FILE__, __LINE__, "The file does not exist.", fileToCheck); + } // Test if the file can be open for reading access. //Only if m_FileName specify a filename (not a dirname) if (itksys::SystemTools::FileExists(fileToCheck.c_str(), true)) - { + { std::ifstream readTester; readTester.open(fileToCheck.c_str()); if (readTester.fail()) - { + { readTester.close(); - std::ostringstream msg; - msg << "The file couldn't be opened for reading. " - << std::endl << "Filename: " << fileToCheck - << std::endl; - otb::ImageFileReaderException e(__FILE__, __LINE__, msg.str().c_str(), ITK_LOCATION); - throw e; - return; - - } - readTester.close(); + throw otb::ImageFileReaderException(__FILE__, __LINE__, "The file cannot be opened for reading.", fileToCheck); } + readTester.close(); + } } template <class TOutputImage, class ConvertPixelTraits> @@ -695,7 +647,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> } fic_trouve = true; } - + otbMsgDevMacro(<< "lFileNameGdal : " << GdalFileName.c_str()); otbMsgDevMacro(<< "fic_trouve : " << fic_trouve); return (fic_trouve); @@ -716,12 +668,12 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> { const std::string skip_geom_key = "skipgeom"; const std::string geom_key = "geom"; - + if (in) { // First, see if the simple filename has changed typename FNameHelperType::Pointer helper = FNameHelperType::New(); - + helper->SetExtendedFileName(in); std::string simpleFileName = helper->GetSimpleFileName(); @@ -735,7 +687,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> if(oldMap.size() != newMap.size() || !std::equal(oldMap.begin(),oldMap.end(),newMap.begin())) { this->Modified(); - + // Now check if keywordlist needs to be generated again // Condition is: one of the old or new map has the skip_geom // key and the other does not @@ -759,7 +711,7 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> m_KeywordListUpToDate = false; this->Modified(); } - + m_FilenameHelper = helper; } } @@ -781,15 +733,15 @@ ImageFileReader<TOutputImage, ConvertPixelTraits> return this->m_ImageIO->GetOverviewsCount(); } - - + + template <class TOutputImage, class ConvertPixelTraits> std::vector<std::string> ImageFileReader<TOutputImage, ConvertPixelTraits> ::GetOverviewsInfo() { this->UpdateOutputInformation(); - + return this->m_ImageIO->GetOverviewsInfo(); } diff --git a/Modules/IO/ImageIO/include/otbImageFileWriter.txx b/Modules/IO/ImageIO/include/otbImageFileWriter.txx index 3224f5a386..9531c69244 100644 --- a/Modules/IO/ImageIO/include/otbImageFileWriter.txx +++ b/Modules/IO/ImageIO/include/otbImageFileWriter.txx @@ -448,20 +448,7 @@ ImageFileWriter<TInputImage> { itk::ImageFileWriterException e(__FILE__, __LINE__); std::ostringstream msg; - msg << " Could not create IO object for file " - << m_FileName.c_str() << std::endl; - msg << " Tried to create one of the following:" << std::endl; - std::list<itk::LightObject::Pointer> allobjects = - itk::ObjectFactoryBase::CreateAllInstance("otbImageIOBase"); - for (std::list<itk::LightObject::Pointer>::iterator i = allobjects.begin(); - i != allobjects.end(); ++i) - { - otb::ImageIOBase* io = dynamic_cast<otb::ImageIOBase*>(i->GetPointer()); - if(io) - msg << " " << io->GetNameOfClass() << std::endl; - } - msg << " You probably failed to set a file suffix, or" << std::endl; - msg << " set the suffix to an unsupported type." << std::endl; + msg << "Cannot write file " << m_FileName.c_str() << " (probably unsupported or incorrect filename extension)."; e.SetDescription(msg.str().c_str()); e.SetLocation(ITK_LOCATION); throw e; diff --git a/Modules/IO/ImageIO/src/otbImageIOFactory.cxx b/Modules/IO/ImageIO/src/otbImageIOFactory.cxx index 671a62d39c..1f084b6afd 100644 --- a/Modules/IO/ImageIO/src/otbImageIOFactory.cxx +++ b/Modules/IO/ImageIO/src/otbImageIOFactory.cxx @@ -48,38 +48,35 @@ ImageIOFactory::CreateImageIO(const char* path, FileModeType mode) itk::ObjectFactoryBase::CreateAllInstance("otbImageIOBase"); for(std::list<itk::LightObject::Pointer>::iterator i = allobjects.begin(); i != allobjects.end(); ++i) - { + { otb::ImageIOBase* io = dynamic_cast<otb::ImageIOBase*>(i->GetPointer()); if(io) - { + { possibleImageIO.push_back(io); - } - else - { - std::cerr << "Error ImageIO factory did not return an ImageIOBase: " - << (*i)->GetNameOfClass() - << std::endl; - } } - for(std::list<otb::ImageIOBase::Pointer>::iterator k = possibleImageIO.begin(); - k != possibleImageIO.end(); ++k) + else { + itkGenericExceptionMacro(<< "ImageIO factory did not return an ImageIOBase but a " << (*i)->GetNameOfClass()); + } + } + + for(std::list<otb::ImageIOBase::Pointer>::iterator k = possibleImageIO.begin(); k != possibleImageIO.end(); ++k) + { if( mode == ReadMode ) - { + { if((*k)->CanReadFile(path)) - { + { return *k; - } } + } else if( mode == WriteMode ) - { + { if((*k)->CanWriteFile(path)) - { + { return *k; - } - } } + } return ITK_NULLPTR; } diff --git a/Modules/Wrappers/ApplicationEngine/include/otbLogger.h b/Modules/Wrappers/ApplicationEngine/include/otbLogger.h new file mode 100644 index 0000000000..c7a3b65b10 --- /dev/null +++ b/Modules/Wrappers/ApplicationEngine/include/otbLogger.h @@ -0,0 +1,50 @@ +/* + * Copyright (C) 2005-2017 Centre National d'Etudes Spatiales (CNES) + * + * This file is part of Orfeo Toolbox + * + * https://www.orfeo-toolbox.org/ + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef otbLogger_h +#define otbLogger_h + +#include "itkLoggerBase.h" +#include "itkLogger.h" + +namespace otb { + +class Logger : public itk::Logger +{ +public: + typedef Logger Self; + typedef itk::Logger Superclass; + typedef itk::SmartPointer< Self > Pointer; + typedef itk::SmartPointer< const Self > ConstPointer; + + itkTypeMacro(Logger, Object); + itkNewMacro(Self); + + // Overwrite this to provide custom formatting of log entries + std::string BuildFormattedEntry(itk::Logger::PriorityLevelType, std::string const&); + +protected: + Logger(); + virtual ~Logger(); +}; // class Logger + +} // namespace otb + +#endif // otbLogger_h diff --git a/Modules/Wrappers/ApplicationEngine/include/otbWrapperApplication.h b/Modules/Wrappers/ApplicationEngine/include/otbWrapperApplication.h index 741f5c4e80..97ade53dc2 100644 --- a/Modules/Wrappers/ApplicationEngine/include/otbWrapperApplication.h +++ b/Modules/Wrappers/ApplicationEngine/include/otbWrapperApplication.h @@ -26,7 +26,7 @@ #include "otbWrapperTags.h" #include "otbWrapperParameterGroup.h" -#include "itkLogger.h" +#include "otbLogger.h" #include "itkTimeProbe.h" #include "otbWrapperMacros.h" #include "otbWrapperInputImageParameter.h" @@ -40,6 +40,30 @@ namespace otb { + +class ApplicationException : public itk::ExceptionObject +{ +public: + /** Run-time information. */ + itkTypeMacro( ApplicationException, ExceptionObject ); + + /** Constructor. */ + ApplicationException(const char *file, unsigned int line, + const char* message = "Application error.", + const char* loc = "Unknown") : + ExceptionObject(file, line, message, loc) + { + } + + /** Constructor. */ + ApplicationException(const std::string &file, unsigned int line, + const char* message = "Application error.", + const char* loc = "Unknown") : + ExceptionObject(file, line, message, loc) + { + } +}; + namespace Wrapper { @@ -70,6 +94,7 @@ public: m_Name = name; GetDocExample()->SetApplicationName(name); this->Modified(); + m_Logger->SetName(name); } itkGetStringMacro(Name); @@ -916,7 +941,6 @@ protected: } } - private: /* Implement this method to add parameters */ virtual void DoInit() = 0; @@ -938,7 +962,7 @@ private: std::string m_Name; std::string m_Description; ParameterGroup::Pointer m_ParameterList; - itk::Logger::Pointer m_Logger; + otb::Logger::Pointer m_Logger; itk::ProcessObject::Pointer m_ProgressSource; std::string m_ProgressSourceDescription; diff --git a/Modules/Wrappers/ApplicationEngine/include/otbWrapperInputImageParameter.txx b/Modules/Wrappers/ApplicationEngine/include/otbWrapperInputImageParameter.txx index f8aa185ced..64a620f706 100644 --- a/Modules/Wrappers/ApplicationEngine/include/otbWrapperInputImageParameter.txx +++ b/Modules/Wrappers/ApplicationEngine/include/otbWrapperInputImageParameter.txx @@ -56,14 +56,7 @@ InputImageParameter::GetImage() typename ReaderType::Pointer reader = ReaderType::New(); reader->SetFileName(m_FileName); - try - { - reader->UpdateOutputInformation(); - } - catch (itk::ExceptionObject &) - { - this->ClearValue(); - } + reader->UpdateOutputInformation(); m_Image = reader->GetOutput(); m_Reader = reader; diff --git a/Modules/Wrappers/ApplicationEngine/include/otbWrapperMacros.h b/Modules/Wrappers/ApplicationEngine/include/otbWrapperMacros.h index 49da20449c..39e4c57d93 100644 --- a/Modules/Wrappers/ApplicationEngine/include/otbWrapperMacros.h +++ b/Modules/Wrappers/ApplicationEngine/include/otbWrapperMacros.h @@ -24,10 +24,15 @@ #define otbAppLogFATAL(x) \ { \ - std::ostringstream msg; \ - msg << "" x << std::endl; \ - this->GetLogger()->Write( itk::LoggerBase::FATAL, msg.str() ); \ - itkGenericExceptionMacro(<< "Fatal error caught. Aborting..."); \ + std::ostringstream msg; \ + msg << "" x << std::endl; \ + this->GetLogger()->Fatal(msg.str()); \ + { \ + std::ostringstream message; \ + message << "otb::ApplicationException " x; \ + ::otb::ApplicationException e_(__FILE__, __LINE__, message.str().c_str(), ITK_LOCATION); \ + throw e_; /* Explicit naming to work around Intel compiler bug. */ \ + } \ } #define otbAppLogCRITICAL(x) \ diff --git a/Modules/Wrappers/ApplicationEngine/src/CMakeLists.txt b/Modules/Wrappers/ApplicationEngine/src/CMakeLists.txt index 6bfea3af16..3251a05d8e 100644 --- a/Modules/Wrappers/ApplicationEngine/src/CMakeLists.txt +++ b/Modules/Wrappers/ApplicationEngine/src/CMakeLists.txt @@ -48,6 +48,7 @@ set(OTBApplicationEngine_SRC otbWrapperApplicationRegistry.cxx otbWrapperApplicationFactoryBase.cxx otbWrapperCompositeApplication.cxx + otbLogger.cxx ) add_library(OTBApplicationEngine ${OTBApplicationEngine_SRC}) diff --git a/Modules/Wrappers/ApplicationEngine/src/otbLogger.cxx b/Modules/Wrappers/ApplicationEngine/src/otbLogger.cxx new file mode 100644 index 0000000000..da3e0e6a6e --- /dev/null +++ b/Modules/Wrappers/ApplicationEngine/src/otbLogger.cxx @@ -0,0 +1,61 @@ +/* + * Copyright (C) 2005-2017 Centre National d'Etudes Spatiales (CNES) + * + * This file is part of Orfeo Toolbox + * + * https://www.orfeo-toolbox.org/ + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "otbLogger.h" +#include "itksys/SystemTools.hxx" + +namespace otb +{ + +Logger::Logger() : + itk::Logger::Logger() +{ +} + +Logger::~Logger() +{ +} + +std::string Logger::BuildFormattedEntry(itk::Logger::PriorityLevelType level, std::string const & content) +{ + static std::string m_LevelString[] = { "(MUSTFLUSH)", "(FATAL)", "(CRITICAL)", + "(WARNING)", "(INFO)", "(DEBUG)", "(NOTSET)" }; + std::ostringstream s; + + switch ( this->m_TimeStampFormat ) + { + case REALVALUE: + { + s.precision(30); + s << m_Clock->GetTimeInSeconds(); + break; + } + case HUMANREADABLE: + { + s << itksys::SystemTools::GetCurrentDateTime( this->m_HumanReadableFormat.c_str() ); + break; + } + } + s << " " << m_LevelString[level] << ": " << content; + + return s.str(); +} + +} // namespace otb diff --git a/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplication.cxx b/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplication.cxx index 9ad9107432..a8a6d3ff59 100644 --- a/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplication.cxx +++ b/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplication.cxx @@ -53,7 +53,7 @@ namespace Wrapper Application::Application() : m_Name(""), m_Description(""), - m_Logger(itk::Logger::New()), + m_Logger(otb::Logger::New()), m_ProgressSourceDescription(""), m_DocName(""), m_DocLongDescription(""), @@ -68,7 +68,12 @@ Application::Application() { // Don't call Init from the constructor, since it calls a virtual method ! m_Logger->SetName("Application.logger"); +#if OTB_DEBUG m_Logger->SetPriorityLevel(itk::LoggerBase::DEBUG); +#else + m_Logger->SetPriorityLevel(itk::LoggerBase::INFO); +#endif + m_Logger->SetLevelForFlushing(itk::LoggerBase::CRITICAL); } @@ -199,9 +204,7 @@ void Application::SetParameterString(std::string parameter, std::string value, b else if (dynamic_cast<InputImageParameter*>(param)) { InputImageParameter* paramDown = dynamic_cast<InputImageParameter*>(param); - if ( !paramDown->SetFromFileName(value) ) - otbAppLogCRITICAL( <<"Invalid image filename " << value <<"."); - + paramDown->SetFromFileName(value); } else if (dynamic_cast<ComplexInputImageParameter*>(param)) { diff --git a/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputFilenameListParameter.cxx b/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputFilenameListParameter.cxx index eb91c4e568..d04f55061c 100644 --- a/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputFilenameListParameter.cxx +++ b/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputFilenameListParameter.cxx @@ -43,38 +43,21 @@ InputFilenameListParameter::SetListFromFileName(const std::vector<std::string> & // First clear previous file chosen this->ClearValue(); - bool isOk = true; for(unsigned int i=0; i<filenames.size(); i++) { std::string filename = filenames[i]; - // TODO : when the logger will be available, redirect the exception - // in the logger (like what is done in MsgReporter) // File existence checked by the reader if (!filename.empty()) { StringParameter::Pointer strParameter = StringParameter::New(); strParameter->SetValue(filename); - try - { - strParameter->HasValue(); - } - catch(itk::ExceptionObject & /*err*/) - { - this->ClearValue(); - isOk = false; - break; - } + strParameter->HasValue(); // everything went fine, store the object reference m_FilenameList->PushBack(strParameter); } } - if( !isOk ) - { - return false; - } - SetActive(true); this->Modified(); return true; @@ -92,22 +75,12 @@ InputFilenameListParameter::AddNullElement() bool InputFilenameListParameter::AddFromFileName(const std::string & filename) { - // TODO : when the logger will be available, redirect the exception - // in the logger (like what is done in MsgReporter) // File existence checked by the reader if (!filename.empty()) { StringParameter::Pointer strParameter = StringParameter::New(); strParameter->SetValue(filename); - try - { - strParameter->HasValue(); - } - catch(itk::ExceptionObject & /*err*/) - { - this->ClearValue(); - return false; - } + strParameter->HasValue(); // everything went fine, store the object references m_FilenameList->PushBack(strParameter); @@ -127,22 +100,12 @@ InputFilenameListParameter::SetNthFileName( const unsigned int id, const std::st itkExceptionMacro(<< "No file "<<id<<". Only "<<m_FilenameList->Size()<<" filenames available."); } - // TODO : when the logger will be available, redirect the exception - // in the logger (like what is done in MsgReporter) // File existence checked by the reader if (!filename.empty()) { StringParameter::Pointer strParameter = StringParameter::New(); strParameter->SetValue(filename); - try - { - strParameter->HasValue(); - } - catch(itk::ExceptionObject &) - { - this->ClearValue(); - return false; - } + strParameter->HasValue(); m_FilenameList->SetNthElement(id, strParameter); diff --git a/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageListParameter.cxx b/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageListParameter.cxx index 01be700ea7..1e0dc32da9 100644 --- a/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageListParameter.cxx +++ b/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageListParameter.cxx @@ -202,19 +202,10 @@ void InputImageListParameter::SetImageList(FloatVectorImageListType* imList) { // Check input availability - // TODO : when the logger will be available, redirect the exception - // in the logger (like what is done in MsgReporter) - try - { - for(unsigned int i=0; i<imList->Size(); i++) - { - imList->GetNthElement( i )->UpdateOutputInformation(); - } - } - catch(itk::ExceptionObject &) - { - return; - } + for(unsigned int i = 0; i < imList->Size(); i++) + { + imList->GetNthElement(i)->UpdateOutputInformation(); + } // Clear previous values this->ClearValue(); @@ -223,13 +214,13 @@ InputImageListParameter::SetImageList(FloatVectorImageListType* imList) { // Try to build a new ParameterInputImage InputImageParameter::Pointer tmpInputImageParameter = InputImageParameter::New(); - + tmpInputImageParameter->SetImage(imList->GetNthElement(i)); - + m_InputImageParameterVector.push_back(tmpInputImageParameter); m_ImageList->PushBack(tmpInputImageParameter->GetFloatVectorImage()); } - + SetActive(true); this->Modified(); } @@ -237,21 +228,16 @@ InputImageListParameter::SetImageList(FloatVectorImageListType* imList) void InputImageListParameter::SetNthImage(unsigned int i, ImageBaseType * img) { if(m_ImageList->Size()<i) - { + { itkExceptionMacro(<< "No image "<<i<<". Only "<<m_ImageList->Size()<<" images available."); - } - try - { - img->UpdateOutputInformation(); - } - catch(itk::ExceptionObject &) - { - return; - } - + } + + // Check input availability + img->UpdateOutputInformation(); + // Try to build a new ParameterInputImage InputImageParameter::Pointer tmpInputImageParameter = InputImageParameter::New(); - + tmpInputImageParameter->SetImage(img); m_InputImageParameterVector[i] = tmpInputImageParameter; @@ -263,21 +249,12 @@ void InputImageListParameter::AddImage(ImageBaseType* image) { // Check input availability - // TODO : when the logger will be available, redirect the exception - // in the logger (like what is done in MsgReporter) - try - { - image->UpdateOutputInformation(); - } - catch(itk::ExceptionObject &) - { - return; - } + image->UpdateOutputInformation(); InputImageParameter::Pointer tmpInputImageParameter = InputImageParameter::New(); - + tmpInputImageParameter->SetImage(image); - + m_InputImageParameterVector.push_back(tmpInputImageParameter); m_ImageList->PushBack(tmpInputImageParameter->GetFloatVectorImage()); diff --git a/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageParameter.cxx b/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageParameter.cxx index dcd9d21528..08c0e1c362 100644 --- a/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageParameter.cxx +++ b/Modules/Wrappers/ApplicationEngine/src/otbWrapperInputImageParameter.cxx @@ -53,27 +53,10 @@ InputImageParameter::SetFromFileName(const std::string& filename) // - Done in the reader // - allow appending additional information to the filename // myfile.tif:2 for example, or myfile.tif:nocarto - if (!filename.empty()) - { - FloatVectorReaderType::Pointer reader = FloatVectorReaderType::New(); - - try - { - reader->SetFileName(filename); - reader->UpdateOutputInformation(); - } - catch(itk::ExceptionObject & /*err*/) - { - return false; - } - - // the specified filename is valid => store the value - m_FileName = filename; - m_UseFilename = true; - SetActive(true); - return true; - } - return false; + m_FileName = filename; + m_UseFilename = true; + SetActive(true); + return true; } diff --git a/Modules/Wrappers/CommandLine/src/otbWrapperCommandLineLauncher.cxx b/Modules/Wrappers/CommandLine/src/otbWrapperCommandLineLauncher.cxx index d1cc388ebd..fa1ec13636 100644 --- a/Modules/Wrappers/CommandLine/src/otbWrapperCommandLineLauncher.cxx +++ b/Modules/Wrappers/CommandLine/src/otbWrapperCommandLineLauncher.cxx @@ -49,6 +49,8 @@ #include <string> #include <iostream> +using std::string; + namespace otb { namespace Wrapper @@ -96,18 +98,19 @@ bool CommandLineLauncher::Load() { if (m_VExpression.empty()) { - itkExceptionMacro("No expression specified..."); + std::cerr << "ERROR: No expression specified." << std::endl; + return false; } if (this->CheckParametersPrefix() == false) { - std::cerr << "ERROR: Parameter keys have to set using \"-\", not \"--\"" << std::endl; + std::cerr << "ERROR: Parameter keys have to set using \"-\", not \"--\"." << std::endl; return false; } if (this->CheckUnicity() == false) { - std::cerr << "ERROR: At least one key is not unique in the expression..." << std::endl; + std::cerr << "ERROR: At least one key is not unique in the expression." << std::endl; return false; } @@ -116,7 +119,7 @@ bool CommandLineLauncher::Load() if (m_Parser->GetPathsAsString(m_VExpression).size() != 0) { std::cerr << "ERROR: At least one specified path within \"" << m_Parser->GetPathsAsString(m_VExpression) - << "\" is invalid or doesn't exist..." << std::endl; + << "\" is invalid or doesn't exist." << std::endl; return false; } } @@ -145,31 +148,54 @@ bool CommandLineLauncher::Execute() bool CommandLineLauncher::ExecuteAndWriteOutput() { try - { + { if (this->BeforeExecute() == false) - { + { return false; - } + } if( m_Application->ExecuteAndWriteOutput() == 0 ) - { + { this->DisplayOutputParameters(); - } + } else + { return false; } + } + catch(otb::ApplicationException& err) + { + // These are thrown with otbAppLogFATAL, a macro which logs a user + // friendly error message before throwing. So log exception details only + // in debug. + m_Application->GetLogger()->Debug("Caught otb::ApplicationException during application execution:\n"); + m_Application->GetLogger()->Debug(string(err.what()) + "\n"); + return false; + } + catch(otb::ImageFileReaderException& err) + { + m_Application->GetLogger()->Debug("Caught otb::ImageFileReaderException during application execution:\n"); + m_Application->GetLogger()->Debug(string(err.what()) + "\n"); + m_Application->GetLogger()->Fatal(string("Cannot open file ") + err.m_Filename + string(". ") + err.GetDescription() + string("\n")); + return false; + } + catch(itk::ExceptionObject& err) + { + m_Application->GetLogger()->Debug("Caught itk::ExceptionObject during application execution:\n"); + m_Application->GetLogger()->Debug(string(err.what()) + "\n"); + m_Application->GetLogger()->Fatal(string(err.GetDescription()) + "\n"); + return false; + } catch(std::exception& err) - { - std::ostringstream message; - message << "The following error occurred during application execution : " << err.what() << std::endl; - m_Application->GetLogger()->Write( itk::LoggerBase::FATAL, message.str() ); + { + m_Application->GetLogger()->Fatal(std::string("Caught std::exception during application execution: ") + err.what() + "\n"); return false; - } + } catch(...) - { - m_Application->GetLogger()->Write( itk::LoggerBase::FATAL, "An unknown exception has been raised during application execution" ); + { + m_Application->GetLogger()->Fatal("Caught unknown exception during application execution.\n"); return false; - } + } return true; } @@ -177,10 +203,10 @@ bool CommandLineLauncher::ExecuteAndWriteOutput() bool CommandLineLauncher::BeforeExecute() { if (m_Application.IsNull()) - { - std::cerr << "ERROR: No loaded application..." << std::endl; + { + std::cerr << "ERROR: No loaded application." << std::endl; return false; - } + } // Check if there's keys in the expression if the application takes // at least 1 mandatory parameter @@ -189,7 +215,7 @@ bool CommandLineLauncher::BeforeExecute() if( appKeyList.size()!=0 && keyList.size()==0 ) { - std::cerr << "ERROR: Waiting for at least one parameter..." << std::endl; + std::cerr << "ERROR: Waiting for at least one parameter." << std::endl; this->DisplayHelp(); return false; } @@ -218,64 +244,21 @@ bool CommandLineLauncher::BeforeExecute() std::string unknownKey; if (this->CheckKeyValidity(unknownKey) == false) { - std::cerr << "ERROR: option -"<<unknownKey<<" does not exist in the application." << std::endl; - this->DisplayHelp(); + std::cerr << "ERROR: Parameter -" << unknownKey <<" does not exist in the application." << std::endl; return false; } - try - { - if (this->LoadParameters() != OKPARAM) - { - std::cerr << "ERROR: Troubles loading parameter, please check your line argument..." << std::endl; - // Force to reload the application, the LoadParameters can change wrong values - this->LoadApplication(); - m_Application->Init(); - this->DisplayHelp(); - return false; - } - } - catch (itk::ExceptionObject& err) - { - std::cerr << "ERROR: Troubles in parameter setting, please check your line argument..." << std::endl; - std::cerr << err.GetDescription() << std::endl; - // Force to reload the application, the LoadParameters can change wrong values - this->LoadApplication(); - m_Application->Init(); - this->DisplayHelp(); + ParamResultType result = this->LoadParameters(); + if (result == MISSINGMANDATORYPARAMETER) + { + this->DisplayHelp(); return false; - } - - // Check for the progress report - if (m_Parser->IsAttributExists("-progress", m_VExpression) == true) - { - std::vector<std::string> val; - val = m_Parser->GetAttribut("-progress", m_VExpression); - if (val.size() != 1) - { - std::cerr << "ERROR: Invalid progress argument, must be unique value..." << std::endl; - return false; - } - if (val[0] == "1" || val[0] == "true") - { - m_ReportProgress = true; - } - else - if (val[0] == "0" || val[0] == "false") - { - m_ReportProgress = false; - } - else - { - std::cerr << "ERROR: Invalid progress argument, must be 0, 1, false or true..." << std::endl; - // Force to reload the application, the LoadParameters can change wrong values - this->LoadApplication(); - this->DisplayHelp(); - - return false; - } - } + } + else if (result != OKPARAM) + { + return false; + } return true; } @@ -307,7 +290,7 @@ void CommandLineLauncher::LoadApplication() //if (m_Parser->GetModuleName(moduleName, m_Expression) != CommandLineParser::OK) if (m_Parser->GetModuleName(moduleName, m_VExpression) != CommandLineParser::OK) { - std::cerr << "ERROR: LoadApplication, no module found..." << std::endl; + std::cerr << "ERROR: LoadApplication, no module found." << std::endl; return; } @@ -338,6 +321,7 @@ void CommandLineLauncher::LoadApplication() { // Attach log output to the Application logger m_Application->GetLogger()->SetTimeStampFormat(itk::LoggerBase::HUMANREADABLE); + m_Application->GetLogger()->SetHumanReadableFormat("%Y-%m-%d %H:%M:%S"); m_Application->GetLogger()->AddLogOutput(m_LogOutput); // Add an observer to the AddedProcess event @@ -367,6 +351,25 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() m_Application->UpdateParameters(); } + // Check for the progress report parameter + if (m_Parser->IsAttributExists("-progress", m_VExpression) == true) + { + std::vector<std::string> val = m_Parser->GetAttribut("-progress", m_VExpression); + if (val.size() == 1 && (val[0] == "1" || val[0] == "true")) + { + m_ReportProgress = true; + } + else if (val.size() == 1 && (val[0] == "0" || val[0] == "false")) + { + m_ReportProgress = false; + } + else + { + std::cerr << "ERROR: Invalid value for parameter -progress. It must be 0, 1, false or true." << std::endl; + return WRONGPARAMETERVALUE; + } + } + const std::vector<std::string> appKeyList = m_Application->GetParametersKeys(true); // Loop over each parameter key declared in the application // FIRST PASS : set parameter values @@ -393,8 +396,7 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() // Check if there is a value associated to the attribute if ( values.empty() ) { - std::cerr << "ERROR: No value associated to the parameter : \"" << paramKey - << "\", invalid number of values " << values.size() << std::endl; + std::cerr << "ERROR: No value associated to parameter -" << paramKey << "." << std::endl; return INVALIDNUMBEROFVALUE; } @@ -449,16 +451,16 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() else if (values[1] == "double") outPixType = ImagePixelType_double; else - { + { + std::cerr << "ERROR: Invalid output type for parameter -" << paramKey << ": " << values[1] << "." << std::endl; return WRONGPARAMETERVALUE; - } + } dynamic_cast<OutputImageParameter *> (param.GetPointer())->SetPixelType(outPixType); } else - if (values.size() != 1 && values.size() != 2) + if (values.size() > 2) { - std::cerr << "ERROR: Invalid number of value for: \"" << paramKey - << "\", invalid number of values " << values.size() << std::endl; + std::cerr << "ERROR: Too many values for parameter -" << paramKey << " (expected 2 or less, got " << values.size() << ")." << std::endl; return INVALIDNUMBEROFVALUE; } } @@ -474,9 +476,10 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() else if (values[1] == "cdouble") outPixType = ComplexImagePixelType_double; else - { + { + std::cerr << "ERROR: Invalid output type for parameter -" << paramKey << ": " << values[1] << "." << std::endl; return WRONGPARAMETERVALUE; - } + } dynamic_cast<ComplexOutputImageParameter *> (param.GetPointer())->SetComplexPixelType(outPixType); } else @@ -551,7 +554,7 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() } else { - std::cerr << "ERROR: Wrong parameter value: " << paramKey << std::endl; + std::cerr << "ERROR: Wrong value for parameter -" << paramKey << "." << std::endl; return WRONGPARAMETERVALUE; } } @@ -614,27 +617,26 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() } if( mustBeSet ) - { + { if (!paramExists) - { + { // If key doesn't exist and parameter hasn't default value set... if (!hasValue) - { - std::cerr << "ERROR: Missing mandatory parameter: " << paramKey << " " << m_Application->HasValue(paramKey) - << std::endl; + { + std::cerr << "ERROR: Missing mandatory parameter -" << paramKey << "." << std::endl; return MISSINGMANDATORYPARAMETER; - } } + } else - { + { values = m_Parser->GetAttribut(std::string("-").append(paramKey), m_VExpression); if (values.size() == 0 && !m_Application->HasValue(paramKey)) - { - std::cerr << "ERROR: Missing mandatory parameter: " << paramKey << std::endl; + { + std::cerr << "ERROR: Missing mandatory parameter -" << paramKey << "." << std::endl; return MISSINGPARAMETERVALUE; - } } } + } // Check if non mandatory parameter have values else { @@ -643,7 +645,7 @@ CommandLineLauncher::ParamResultType CommandLineLauncher::LoadParameters() values = m_Parser->GetAttribut(std::string("-").append(paramKey), m_VExpression); if (values.size() == 0) { - std::cerr << "ERROR: Missing non-mandatory parameter: " << paramKey << std::endl; + std::cerr << "ERROR: Missing non-mandatory parameter -" << paramKey << "." << std::endl; return MISSINGPARAMETERVALUE; } } -- GitLab