diff --git a/catalogue/3.0/mysql_catalogue_schema.sql b/catalogue/3.0/mysql_catalogue_schema.sql index dcac6016cb89b5b59f44a434c6c230f3a2765ecf..b236e948620088c3826fb3682bc35dde8e4cd45d 100644 --- a/catalogue/3.0/mysql_catalogue_schema.sql +++ b/catalogue/3.0/mysql_catalogue_schema.sql @@ -33,7 +33,9 @@ CREATE TABLE CTA_CATALOGUE( SCHEMA_VERSION_MINOR BIGINT UNSIGNED CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL, NEXT_SCHEMA_VERSION_MAJOR BIGINT UNSIGNED, NEXT_SCHEMA_VERSION_MINOR BIGINT UNSIGNED, - STATUS VARCHAR(100) + STATUS VARCHAR(100), + IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL, + CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1')) ); CREATE TABLE ADMIN_USER( ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL, diff --git a/catalogue/3.0/oracle_catalogue_schema.sql b/catalogue/3.0/oracle_catalogue_schema.sql index 82459ed1ec59e755490592b74af750f647c9631d..11dd0659ca578e315a91725c90d01e2086104e93 100644 --- a/catalogue/3.0/oracle_catalogue_schema.sql +++ b/catalogue/3.0/oracle_catalogue_schema.sql @@ -62,7 +62,9 @@ CREATE TABLE CTA_CATALOGUE( SCHEMA_VERSION_MINOR NUMERIC(20, 0) CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL, NEXT_SCHEMA_VERSION_MAJOR NUMERIC(20, 0), NEXT_SCHEMA_VERSION_MINOR NUMERIC(20, 0), - STATUS VARCHAR2(100) + STATUS VARCHAR2(100), + IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL, + CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1')) ); CREATE TABLE ADMIN_USER( ADMIN_USER_NAME VARCHAR2(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL, diff --git a/catalogue/3.0/postgres_catalogue_schema.sql b/catalogue/3.0/postgres_catalogue_schema.sql index 00f49e130ce6c8bd8f45f30fbde26839f4f5089d..e0c29901f61782721d5c8fdc765bebdf54259e96 100644 --- a/catalogue/3.0/postgres_catalogue_schema.sql +++ b/catalogue/3.0/postgres_catalogue_schema.sql @@ -45,7 +45,9 @@ CREATE TABLE CTA_CATALOGUE( SCHEMA_VERSION_MINOR NUMERIC(20, 0) CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL, NEXT_SCHEMA_VERSION_MAJOR NUMERIC(20, 0), NEXT_SCHEMA_VERSION_MINOR NUMERIC(20, 0), - STATUS VARCHAR(100) + STATUS VARCHAR(100), + IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL, + CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1')) ); CREATE TABLE ADMIN_USER( ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL, diff --git a/catalogue/3.0/sqlite_catalogue_schema.sql b/catalogue/3.0/sqlite_catalogue_schema.sql index 26da00179c9de142adc9af3017e1e4613680040e..f6d8fc0c85861e3f18f1ab4eed49b7d93fe0c28e 100644 --- a/catalogue/3.0/sqlite_catalogue_schema.sql +++ b/catalogue/3.0/sqlite_catalogue_schema.sql @@ -21,7 +21,9 @@ CREATE TABLE CTA_CATALOGUE( SCHEMA_VERSION_MINOR INTEGER CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL, NEXT_SCHEMA_VERSION_MAJOR INTEGER, NEXT_SCHEMA_VERSION_MINOR INTEGER, - STATUS VARCHAR(100) + STATUS VARCHAR(100), + IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL, + CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1')) ); CREATE TABLE ADMIN_USER( ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL, diff --git a/catalogue/AllCatalogueSchema.hpp b/catalogue/AllCatalogueSchema.hpp index e4563f3f8f0990513dc19d41240506975b372c9a..3a5b25001cedee23b6e2586e07672654add45603 100644 --- a/catalogue/AllCatalogueSchema.hpp +++ b/catalogue/AllCatalogueSchema.hpp @@ -3866,7 +3866,9 @@ namespace catalogue{ " SCHEMA_VERSION_MINOR NUMERIC(20, 0) CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL," " NEXT_SCHEMA_VERSION_MAJOR NUMERIC(20, 0)," " NEXT_SCHEMA_VERSION_MINOR NUMERIC(20, 0)," - " STATUS VARCHAR2(100)" + " STATUS VARCHAR2(100)," + " IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL," + " CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1'))" ");" "CREATE TABLE ADMIN_USER(" " ADMIN_USER_NAME VARCHAR2(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL," @@ -4266,7 +4268,9 @@ namespace catalogue{ " SCHEMA_VERSION_MINOR BIGINT UNSIGNED CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL," " NEXT_SCHEMA_VERSION_MAJOR BIGINT UNSIGNED," " NEXT_SCHEMA_VERSION_MINOR BIGINT UNSIGNED," - " STATUS VARCHAR(100)" + " STATUS VARCHAR(100)," + " IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL," + " CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1'))" ");" "CREATE TABLE ADMIN_USER(" " ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL," @@ -4653,7 +4657,9 @@ namespace catalogue{ " SCHEMA_VERSION_MINOR INTEGER CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL," " NEXT_SCHEMA_VERSION_MAJOR INTEGER," " NEXT_SCHEMA_VERSION_MINOR INTEGER," - " STATUS VARCHAR(100)" + " STATUS VARCHAR(100)," + " IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL," + " CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1'))" ");" "CREATE TABLE ADMIN_USER(" " ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL," @@ -5062,7 +5068,9 @@ namespace catalogue{ " SCHEMA_VERSION_MINOR NUMERIC(20, 0) CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL," " NEXT_SCHEMA_VERSION_MAJOR NUMERIC(20, 0)," " NEXT_SCHEMA_VERSION_MINOR NUMERIC(20, 0)," - " STATUS VARCHAR(100)" + " STATUS VARCHAR(100)," + " IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL," + " CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1'))" ");" "CREATE TABLE ADMIN_USER(" " ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL," diff --git a/catalogue/CMakeLists.txt b/catalogue/CMakeLists.txt index b67b276fe9c47a208964fdf7c620c567864b8d56..fea9123a2716d25b4676b63ea0806da7212c8367 100644 --- a/catalogue/CMakeLists.txt +++ b/catalogue/CMakeLists.txt @@ -104,7 +104,8 @@ install (TARGETS ctacatalogue DESTINATION usr/${CMAKE_INSTALL_LIBDIR}) target_link_libraries (ctacatalogue ctacommon - ctardbms) + ctardbms +) include(${PROJECT_SOURCE_DIR}/cmake/CTAVersions.cmake) @@ -258,6 +259,27 @@ add_executable(cta-catalogue-schema-create PostgresCatalogueSchema.cpp MysqlCatalogueSchema.cpp) +set (SCHEMA_CHECKER_LIB_SRC_FILES + SQLiteSchemaInserter.cpp + SchemaSqlStatementsReader.cpp + SchemaComparer.cpp + SQLiteSchemaComparer.cpp + DbToSQLiteStatementTransformer.cpp + SchemaCheckerResult.cpp + SchemaChecker.cpp + DatabaseMetadataGetter.cpp + CatalogueSchema.cpp + OracleCatalogueSchema.cpp + SqliteCatalogueSchema.cpp + PostgresCatalogueSchema.cpp + MysqlCatalogueSchema.cpp +) + +add_library (ctaschemachecker SHARED + ${SCHEMA_CHECKER_LIB_SRC_FILES}) + +install (TARGETS ctaschemachecker DESTINATION usr/${CMAKE_INSTALL_LIBDIR}) + target_link_libraries (cta-catalogue-schema-create ctacatalogue) set_property(TARGET cta-catalogue-schema-create APPEND PROPERTY INSTALL_RPATH ${PROTOBUF3_RPATH}) set_property(TARGET cta-catalogue-schema-create APPEND PROPERTY INSTALL_RPATH ${ORACLE-INSTANTCLIENT_RPATH}) @@ -270,7 +292,7 @@ add_executable(cta-catalogue-schema-drop DropSchemaCmdLineArgs.cpp DropSchemaCmdMain.cpp) -target_link_libraries (cta-catalogue-schema-drop ctacatalogue) +target_link_libraries (cta-catalogue-schema-drop ctacatalogue ctaschemachecker) set_property(TARGET cta-catalogue-schema-drop APPEND PROPERTY INSTALL_RPATH ${PROTOBUF3_RPATH}) set_property(TARGET cta-catalogue-schema-drop APPEND PROPERTY INSTALL_RPATH ${ORACLE-INSTANTCLIENT_RPATH}) @@ -301,27 +323,6 @@ set_property(TARGET cta-catalogue-admin-user-create APPEND PROPERTY INSTALL_RPAT install(TARGETS cta-catalogue-admin-user-create DESTINATION /usr/bin) install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/cta-catalogue-admin-user-create.1cta DESTINATION /usr/share/man/man1) -set (SCHEMA_CHECKER_LIB_SRC_FILES - SQLiteSchemaInserter.cpp - SchemaSqlStatementsReader.cpp - SchemaComparer.cpp - SQLiteSchemaComparer.cpp - DbToSQLiteStatementTransformer.cpp - SchemaComparerResult.cpp - SchemaChecker.cpp - DatabaseMetadataGetter.cpp - CatalogueSchema.cpp - OracleCatalogueSchema.cpp - SqliteCatalogueSchema.cpp - PostgresCatalogueSchema.cpp - MysqlCatalogueSchema.cpp -) - -add_library (ctaschemachecker SHARED - ${SCHEMA_CHECKER_LIB_SRC_FILES}) - -install (TARGETS ctaschemachecker DESTINATION usr/${CMAKE_INSTALL_LIBDIR}) - add_executable(cta-catalogue-schema-verify VerifySchemaCmd.cpp VerifySchemaCmdLineArgs.cpp diff --git a/catalogue/DropSchemaCmd.cpp b/catalogue/DropSchemaCmd.cpp index 63a65b967e045e6432ac2f7adfbf882d87e4b051..187c096f5bf8f46b11eb74ae3d4eef574b1568c3 100644 --- a/catalogue/DropSchemaCmd.cpp +++ b/catalogue/DropSchemaCmd.cpp @@ -20,6 +20,7 @@ #include "catalogue/DropSchemaCmdLineArgs.hpp" #include "common/exception/Exception.hpp" #include "rdbms/ConnPool.hpp" +#include "catalogue/SchemaChecker.hpp" #include <algorithm> @@ -65,6 +66,12 @@ int DropSchemaCmd::exceptionThrowingMain(const int argc, char *const *const argv return 0; } + if(isProductionProtectionCheckable(conn,dbLogin.dbType)){ + if(isProductionSet(conn)){ + throw cta::exception::Exception("Cannot drop a production database. If you still wish to proceed then please modify the database manually to remove its production status before trying again."); + } + } + if(userConfirmsDropOfSchema(dbLogin)) { m_out << "DROPPING the schema of the CTA calalogue database" << std::endl; dropCatalogueSchema(dbLogin.dbType, conn); @@ -351,6 +358,28 @@ void DropSchemaCmd::dropDatabaseSequences(rdbms::Conn &conn, const std::list<std } } +bool DropSchemaCmd::isProductionSet(cta::rdbms::Conn & conn){ + const char * const sql = "SELECT CTA_CATALOGUE.IS_PRODUCTION AS IS_PRODUCTION FROM CTA_CATALOGUE"; + auto stmt = conn.createStmt(sql); + auto rset = stmt.executeQuery(); + if(rset.next()){ + return rset.columnBool("IS_PRODUCTION"); + } + //We should never arrive here + throw cta::exception::Exception("Cannot check the IS_PRODUCTION bit because the CTA_CATALOGUE table is empty or does not exist."); +} + + +bool DropSchemaCmd::isProductionProtectionCheckable(rdbms::Conn& conn, const cta::rdbms::Login::DbType dbType) { + cta::catalogue::SchemaChecker::Builder builder("catalogue",dbType,conn); + auto checker = builder.build(); + SchemaCheckerResult res = checker->checkTableContainsColumns("CTA_CATALOGUE",{"IS_PRODUCTION"}); + if(res.getStatus() == SchemaCheckerResult::Status::FAILED){ + return false; + } + return true; +} + //------------------------------------------------------------------------------ // printUsage //------------------------------------------------------------------------------ diff --git a/catalogue/DropSchemaCmd.hpp b/catalogue/DropSchemaCmd.hpp index 54da56020581ff5c69cb7537b190d73cafbe5343..e6806fe6297c804de7920b55de3d9d6b09fb9ce0 100644 --- a/catalogue/DropSchemaCmd.hpp +++ b/catalogue/DropSchemaCmd.hpp @@ -128,6 +128,21 @@ private: * @param seqeuncesToDrop The names of the database sequences to be dropped. */ void dropDatabaseSequences(rdbms::Conn &conn, const std::list<std::string> &sequencesToDrop); + + /** + * Checks if the IS_PRODUCTION bit is set on the CTA_CATALOGUE table + * @return true if the IS_PRODUCTION bit is set, false otherwise + */ + bool isProductionSet(rdbms::Conn & conn); + + /** + * Checks if we can check the IS_PRODUCTION bit. This allows the backward-compatibility of + * this tool if we use it against a schema that does not have the IS_PRODUCTION bit. + * @param conn the connection to the Catalogue database + * @param dbType the type of the Catalogue database + * @return true if the production bit is set, false otherwise + */ + bool isProductionProtectionCheckable(rdbms::Conn & conn, const cta::rdbms::Login::DbType dbType); }; // class DropSchemaCmd diff --git a/catalogue/RdbmsCatalogue.cpp b/catalogue/RdbmsCatalogue.cpp index 8172b35feb3edb0514c11c5ec32f5825bee0936e..530453216d5fa43b2b35bce87de743b82f539ae8 100644 --- a/catalogue/RdbmsCatalogue.cpp +++ b/catalogue/RdbmsCatalogue.cpp @@ -7971,16 +7971,16 @@ SchemaVersion RdbmsCatalogue::getSchemaVersion() const { if(rset.next()) { SchemaVersion::Builder schemaVersionBuilder; - schemaVersionBuilder.schemaVersionMajor(rset.columnUint64("SCHEMA_VERSION_MAJOR")) - .schemaVersionMinor(rset.columnUint64("SCHEMA_VERSION_MINOR")) - .status(rset.columnString("STATUS")); - auto schemaVersionMajorNext = rset.columnOptionalUint64("NEXT_SCHEMA_VERSION_MAJOR"); - auto schemaVersionMinorNext = rset.columnOptionalUint64("NEXT_SCHEMA_VERSION_MINOR"); - if(schemaVersionMajorNext && schemaVersionMinorNext){ - schemaVersionBuilder.nextSchemaVersionMajor(schemaVersionMajorNext.value()) - .nextSchemaVersionMinor(schemaVersionMinorNext.value()); - } - return schemaVersionBuilder.build(); + schemaVersionBuilder.schemaVersionMajor(rset.columnUint64("SCHEMA_VERSION_MAJOR")) + .schemaVersionMinor(rset.columnUint64("SCHEMA_VERSION_MINOR")) + .status(rset.columnString("STATUS")); + auto schemaVersionMajorNext = rset.columnOptionalUint64("NEXT_SCHEMA_VERSION_MAJOR"); + auto schemaVersionMinorNext = rset.columnOptionalUint64("NEXT_SCHEMA_VERSION_MINOR"); + if(schemaVersionMajorNext && schemaVersionMinorNext){ + schemaVersionBuilder.nextSchemaVersionMajor(schemaVersionMajorNext.value()) + .nextSchemaVersionMinor(schemaVersionMinorNext.value()); + } + return schemaVersionBuilder.build(); } else { throw exception::Exception("CTA_CATALOGUE does not contain any row"); } diff --git a/catalogue/SQLiteSchemaComparer.cpp b/catalogue/SQLiteSchemaComparer.cpp index 50dd9fb37e2b2e65da1979869b59fb4ef1bb2c1c..7526ade6e7ae9d0e3fafc86960bc3dbab2d9940d 100644 --- a/catalogue/SQLiteSchemaComparer.cpp +++ b/catalogue/SQLiteSchemaComparer.cpp @@ -43,23 +43,23 @@ SQLiteSchemaComparer::~SQLiteSchemaComparer() { m_sqliteConnPool.reset(); } -SchemaComparerResult SQLiteSchemaComparer::compareAll(){ +SchemaCheckerResult SQLiteSchemaComparer::compareAll(){ insertSchemaInSQLite(); - SchemaComparerResult res; + SchemaCheckerResult res; res += compareTables(); res += compareIndexes(); return res; } -SchemaComparerResult SQLiteSchemaComparer::compareTables(){ +SchemaCheckerResult SQLiteSchemaComparer::compareTables(){ insertSchemaInSQLite(); std::list<std::string> catalogueTables = m_databaseMetadataGetter.getTableNames(); std::list<std::string> schemaTables = m_schemaMetadataGetter->getTableNames(); - SchemaComparerResult res = compareTables(catalogueTables,schemaTables); + SchemaCheckerResult res = compareTables(catalogueTables,schemaTables); return res; } -SchemaComparerResult SQLiteSchemaComparer::compareTablesLocatedInSchema(){ +SchemaCheckerResult SQLiteSchemaComparer::compareTablesLocatedInSchema(){ insertSchemaInSQLite(); std::list<std::string> databaseTables = m_databaseMetadataGetter.getTableNames(); std::list<std::string> schemaTables = m_schemaMetadataGetter->getTableNames(); @@ -68,7 +68,7 @@ SchemaComparerResult SQLiteSchemaComparer::compareTablesLocatedInSchema(){ return schemaTable == catalogueTable; }) == schemaTables.end(); }); - SchemaComparerResult res = compareTables(databaseTables,schemaTables); + SchemaCheckerResult res = compareTables(databaseTables,schemaTables); return res; } @@ -84,30 +84,30 @@ void SQLiteSchemaComparer::insertSchemaInSQLite() { m_isSchemaInserted = true; } -SchemaComparerResult SQLiteSchemaComparer::compareIndexes(){ +SchemaCheckerResult SQLiteSchemaComparer::compareIndexes(){ insertSchemaInSQLite(); std::list<std::string> catalogueIndexes = m_databaseMetadataGetter.getIndexNames(); std::list<std::string> schemaIndexes = m_schemaMetadataGetter->getIndexNames(); return compareItems("INDEX", catalogueIndexes, schemaIndexes); } -SchemaComparerResult SQLiteSchemaComparer::compareItems(const std::string &itemType, const std::list<std::string>& itemsFromDatabase, const std::list<std::string>& itemsFromSQLite){ - SchemaComparerResult result; +SchemaCheckerResult SQLiteSchemaComparer::compareItems(const std::string &itemType, const std::list<std::string>& itemsFromDatabase, const std::list<std::string>& itemsFromSQLite){ + SchemaCheckerResult result; for(auto &databaseItem: itemsFromDatabase){ if(std::find(itemsFromSQLite.begin(),itemsFromSQLite.end(),databaseItem) == itemsFromSQLite.end()){ - result.addDiff(itemType+" "+databaseItem+" is missing in the schema but defined in the "+m_databaseToCheckName+" database."); + result.addError(itemType+" "+databaseItem+" is missing in the schema but defined in the "+m_databaseToCheckName+" database."); } } for(auto &sqliteItem: itemsFromSQLite){ if(std::find(itemsFromDatabase.begin(),itemsFromDatabase.end(),sqliteItem) == itemsFromDatabase.end()){ - result.addDiff(itemType+" "+sqliteItem+" is missing in the "+m_databaseToCheckName+" database but is defined in the schema."); + result.addError(itemType+" "+sqliteItem+" is missing in the "+m_databaseToCheckName+" database but is defined in the schema."); } } return result; } -SchemaComparerResult SQLiteSchemaComparer::compareTables(const std::list<std::string>& databaseTables, const std::list<std::string>& schemaTables){ - SchemaComparerResult result; +SchemaCheckerResult SQLiteSchemaComparer::compareTables(const std::list<std::string>& databaseTables, const std::list<std::string>& schemaTables){ + SchemaCheckerResult result; std::map<std::string, std::map<std::string, std::string>> databaseTableColumns; std::map<std::string, std::map<std::string, std::string>> schemaTableColumns; std::map<std::string,std::list<std::string>> databaseTableConstraints; @@ -132,8 +132,8 @@ SchemaComparerResult SQLiteSchemaComparer::compareTables(const std::list<std::st return result; } -SchemaComparerResult SQLiteSchemaComparer::compareTableColumns(const TableColumns & schema1TableColumns, const std::string &schema1Type,const TableColumns & schema2TableColumns, const std::string &schema2Type){ - SchemaComparerResult result; +SchemaCheckerResult SQLiteSchemaComparer::compareTableColumns(const TableColumns & schema1TableColumns, const std::string &schema1Type,const TableColumns & schema2TableColumns, const std::string &schema2Type){ + SchemaCheckerResult result; for(auto &kvFirstSchemaTableColumns: schema1TableColumns){ //For each firstSchema table, get the corresponding secondSchema table //If it does not exist, add a difference and go ahead @@ -149,14 +149,14 @@ SchemaComparerResult SQLiteSchemaComparer::compareTableColumns(const TableColumn try { std::string schemaColumnType = mapSchema2ColumnType.at(schema1ColumnName); if( schema1ColumnType != schemaColumnType){ - result.addDiff("TABLE "+schema1TableName+" from "+schema1Type+" has a column named "+schema1ColumnName+" that has a type "+schema1ColumnType+" that does not match the column type from the "+schema2Type+" ("+schemaColumnType+")"); + result.addError("TABLE "+schema1TableName+" from "+schema1Type+" has a column named "+schema1ColumnName+" that has a type "+schema1ColumnType+" that does not match the column type from the "+schema2Type+" ("+schemaColumnType+")"); } } catch (const std::out_of_range &) { - result.addDiff("TABLE "+schema1TableName+" from "+schema1Type+" has a column named " + schema1ColumnName + " that is missing in the "+schema2Type+"."); + result.addError("TABLE "+schema1TableName+" from "+schema1Type+" has a column named " + schema1ColumnName + " that is missing in the "+schema2Type+"."); } } } catch (const std::out_of_range &) { - result.addDiff("TABLE "+schema1TableName+" is missing in the "+schema2Type+" but is defined in the "+schema1Type+"."); + result.addError("TABLE "+schema1TableName+" is missing in the "+schema2Type+" but is defined in the "+schema1Type+"."); } } return result; diff --git a/catalogue/SQLiteSchemaComparer.hpp b/catalogue/SQLiteSchemaComparer.hpp index 2f146938b75df50947d2422f3df9d972844be3e3..b501c0a6d59dc0daa212cee169a0a7ae381ccc8c 100644 --- a/catalogue/SQLiteSchemaComparer.hpp +++ b/catalogue/SQLiteSchemaComparer.hpp @@ -38,19 +38,19 @@ public: * Compare the catalogue schema against the InMemory SQLite catalogue schema * @return a SchemaComparerResult object that will contain the differences if there are some */ - SchemaComparerResult compareAll() override; - SchemaComparerResult compareIndexes() override; - SchemaComparerResult compareTables() override; - SchemaComparerResult compareTablesLocatedInSchema() override; + SchemaCheckerResult compareAll() override; + SchemaCheckerResult compareIndexes() override; + SchemaCheckerResult compareTables() override; + SchemaCheckerResult compareTablesLocatedInSchema() override; virtual ~SQLiteSchemaComparer(); private: void insertSchemaInSQLite(); - SchemaComparerResult compareItems(const std::string &itemType, const std::list<std::string>& itemsFromDatabase, const std::list<std::string>& itemsFromSQLite); - SchemaComparerResult compareTables(const std::list<std::string> &databaseTables, const std::list<std::string> &schemaTables); + SchemaCheckerResult compareItems(const std::string &itemType, const std::list<std::string>& itemsFromDatabase, const std::list<std::string>& itemsFromSQLite); + SchemaCheckerResult compareTables(const std::list<std::string> &databaseTables, const std::list<std::string> &schemaTables); typedef std::map<std::string, std::map<std::string, std::string>> TableColumns; - SchemaComparerResult compareTableColumns(const TableColumns & schema1TableColumns, const std::string &schema1Type,const TableColumns & schema2TableColumns, const std::string &schema2Type); + SchemaCheckerResult compareTableColumns(const TableColumns & schema1TableColumns, const std::string &schema1Type,const TableColumns & schema2TableColumns, const std::string &schema2Type); rdbms::Conn m_sqliteConn; std::unique_ptr<rdbms::ConnPool> m_sqliteConnPool; std::unique_ptr<SchemaMetadataGetter> m_schemaMetadataGetter; diff --git a/catalogue/SchemaChecker.cpp b/catalogue/SchemaChecker.cpp index d671da69d99e807e32f09005b5d877c12abfea98..0721964913f88ead54691e89569c57069a035d72 100644 --- a/catalogue/SchemaChecker.cpp +++ b/catalogue/SchemaChecker.cpp @@ -36,71 +36,70 @@ void SchemaChecker::checkSchemaComparerNotNull(const std::string& methodName){ } } -SchemaChecker::Status SchemaChecker::compareSchema(){ - checkSchemaComparerNotNull(__PRETTY_FUNCTION__); - SchemaComparerResult totalResult; - std::cout << "Schema version : " << m_databaseMetadataGetter->getCatalogueVersion().getSchemaVersion<std::string>() << std::endl; - std::cout << "Checking indexes..." << std::endl; - cta::catalogue::SchemaComparerResult resIndex = m_schemaComparer->compareIndexes(); +SchemaCheckerResult SchemaChecker::displayingCompareSchema(std::ostream & stdOut, std::ostream & stdErr){ + checkSchemaComparerNotNull(__PRETTY_FUNCTION__); + SchemaCheckerResult totalResult; + stdOut << "Schema version : " << m_databaseMetadataGetter->getCatalogueVersion().getSchemaVersion<std::string>() << std::endl; + stdOut << "Checking indexes..." << std::endl; + cta::catalogue::SchemaCheckerResult resIndex = m_schemaComparer->compareIndexes(); totalResult += resIndex; - resIndex.printDiffs(); - std::cout <<" "<<resIndex.statusToString(resIndex.getStatus())<<std::endl; + resIndex.displayErrors(stdErr); + stdOut <<" "<<resIndex.statusToString(resIndex.getStatus())<<std::endl; if(m_dbType == rdbms::Login::DbType::DBTYPE_MYSQL){ - std::cout << "Checking tables and columns..." << std::endl; + stdOut << "Checking tables and columns..." << std::endl; } else { - std::cout << "Checking tables, columns and constraints..." << std::endl; + stdOut << "Checking tables, columns and constraints..." << std::endl; } - cta::catalogue::SchemaComparerResult resTables = m_schemaComparer->compareTables(); + cta::catalogue::SchemaCheckerResult resTables = m_schemaComparer->compareTables(); totalResult += resTables; - resTables.printDiffs(); - std::cout <<" "<<resTables.statusToString(resTables.getStatus())<<std::endl; - std::cout << "Status of the checking : " << cta::catalogue::SchemaComparerResult::statusToString(totalResult.getStatus()) << std::endl; - if(totalResult.getStatus() == SchemaComparerResult::Status::FAILED){ - return SchemaChecker::FAILURE; - } - return SchemaChecker::OK; + resTables.displayErrors(stdErr); + stdOut <<" "<<resTables.statusToString(resTables.getStatus())<<std::endl; + stdOut << "Status of the checking : " << cta::catalogue::SchemaCheckerResult::statusToString(totalResult.getStatus()) << std::endl; + return totalResult; } -void SchemaChecker::checkNoParallelTables(){ +SchemaCheckerResult SchemaChecker::checkNoParallelTables(){ + SchemaCheckerResult res; std::list<std::string> parallelTables = m_databaseMetadataGetter->getParallelTableNames(); for(auto& table:parallelTables) { - std::cout << "WARNING : TABLE " << table << " is set as PARALLEL" << std::endl; + std::string warning = "TABLE " + table + " is set as PARALLEL"; + res.addWarning(warning); } + return res; } -void SchemaChecker::checkSchemaNotUpgrading(){ +SchemaCheckerResult SchemaChecker::checkSchemaNotUpgrading(){ + SchemaCheckerResult res; SchemaVersion catalogueVersion = m_databaseMetadataGetter->getCatalogueVersion(); if(catalogueVersion.getStatus<SchemaVersion::Status>() == SchemaVersion::Status::UPGRADING){ - std::cout << "WARNING : The status of the schema is " << catalogueVersion.getStatus<std::string>() << ", the future version is " << catalogueVersion.getSchemaVersionNext<std::string>() << std::endl; + std::string warning = "The status of the schema is " + catalogueVersion.getStatus<std::string>() + ", the future version is " + catalogueVersion.getSchemaVersionNext<std::string>(); + res.addWarning(warning); } + return res; } -SchemaChecker::Status SchemaChecker::compareTablesLocatedInSchema(){ +SchemaCheckerResult SchemaChecker::compareTablesLocatedInSchema(){ checkSchemaComparerNotNull(__PRETTY_FUNCTION__); - SchemaComparerResult res = m_schemaComparer->compareTablesLocatedInSchema(); - if(res.getStatus() == SchemaComparerResult::Status::FAILED){ - res.printDiffs(); - return SchemaChecker::Status::FAILURE; - } - return SchemaChecker::Status::OK; + return m_schemaComparer->compareTablesLocatedInSchema(); } -SchemaChecker::Status SchemaChecker::checkTableContainsColumns(const std::string& tableName, const std::list<std::string> columnNames){ +SchemaCheckerResult SchemaChecker::checkTableContainsColumns(const std::string& tableName, const std::list<std::string> columnNames){ std::map<std::string, std::string> mapColumnsTypes = m_databaseMetadataGetter->getColumns(tableName); - SchemaChecker::Status status = SchemaChecker::Status::OK; + SchemaCheckerResult res; if(mapColumnsTypes.empty()){ - std::cerr << "TABLE " << tableName << " does not exist." << std::endl; - return SchemaChecker::Status::FAILURE; + std::string error = "TABLE " + tableName +" does not exist."; + res.addError(error); + return res; } for(auto &columnName: columnNames){ try{ mapColumnsTypes.at(columnName); } catch(std::out_of_range &){ - std::cerr << "TABLE " << tableName << " does not contain the column " << columnName << "."<< std::endl; - status = SchemaChecker::Status::FAILURE; + std::string error = "TABLE " + tableName + " does not contain the column " + columnName + "."; + res.addError(error); } } - return status; + return res; } ///////////////////////////////////////// @@ -139,6 +138,4 @@ std::unique_ptr<SchemaChecker> SchemaChecker::Builder::build() { return std::move(schemaChecker); } - - }} \ No newline at end of file diff --git a/catalogue/SchemaChecker.hpp b/catalogue/SchemaChecker.hpp index b499112f61076dcd1440447a949cbce465eaf304..381a27ce022d2101dfdfa4b0e794af5691e1cf56 100644 --- a/catalogue/SchemaChecker.hpp +++ b/catalogue/SchemaChecker.hpp @@ -21,6 +21,7 @@ #include "rdbms/Conn.hpp" #include "SchemaComparer.hpp" #include "CatalogueSchema.hpp" +#include "SchemaCheckerResult.hpp" namespace cta{ namespace catalogue{ @@ -39,43 +40,38 @@ namespace catalogue{ */ class SchemaChecker { public: - /** - * The status of the checking of the database - */ - enum Status { - OK, - FAILURE - }; /** * Destructor */ virtual ~SchemaChecker(); /** - * Compare the schema by using a SchemaComparer + * Compare the schema by using a SchemaComparer and output what it is doing and errors * @throws Exception if no SchemaComparer has been set. - * @return a Status : OK or FAILURE + * @return a SchemaCheckerResult */ - Status compareSchema(); + SchemaCheckerResult displayingCompareSchema(std::ostream & stdout, std::ostream & stderr); + /** * Checks if the catalogue database contains PARALLEL tables * It will display a warning with the table name if this is the case + * @return a SchemaCheckerResult */ - void checkNoParallelTables(); + SchemaCheckerResult checkNoParallelTables(); /** * Checks if the catalogue database schema is upgrading or not */ - void checkSchemaNotUpgrading(); + SchemaCheckerResult checkSchemaNotUpgrading(); /** * Compare the schema tables whose names are located in the tableList parameter * @param tableList the table names we would like to compare * @return a Status OK or FAILURE */ - Status compareTablesLocatedInSchema(); + SchemaCheckerResult compareTablesLocatedInSchema(); - Status checkTableContainsColumns(const std::string &tableName, const std::list<std::string> columnNames); + SchemaCheckerResult checkTableContainsColumns(const std::string &tableName, const std::list<std::string> columnNames); class Builder { public: diff --git a/catalogue/SchemaCheckerResult.cpp b/catalogue/SchemaCheckerResult.cpp new file mode 100644 index 0000000000000000000000000000000000000000..e254736c36c2045c473b2150cc56da1aa5914bfb --- /dev/null +++ b/catalogue/SchemaCheckerResult.cpp @@ -0,0 +1,94 @@ +/** + * The CERN Tape Archive (CTA) project + * Copyright © 2018 CERN + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <iostream> + +#include "SchemaCheckerResult.hpp" + +namespace cta { +namespace catalogue { + +SchemaCheckerResult::SchemaCheckerResult():m_status(Status::SUCCESS) { +} + +SchemaCheckerResult::SchemaCheckerResult(const SchemaCheckerResult& orig) { + m_errors = orig.m_errors; + m_warnings = orig.m_warnings; + m_status = orig.m_status; +} + +SchemaCheckerResult SchemaCheckerResult::operator=(const SchemaCheckerResult &other){ + if(this != &other){ + m_errors = other.m_errors; + m_warnings = other.m_warnings; + m_status = other.m_status; + } + return *this; +} + +SchemaCheckerResult SchemaCheckerResult::operator +=(const SchemaCheckerResult& other){ + m_errors.insert(m_errors.end(),other.m_errors.begin(),other.m_errors.end()); + m_warnings.insert(m_warnings.end(),other.m_warnings.begin(), other.m_warnings.end()); + if(m_status == Status::SUCCESS){ + // The status should not change if it is failed + m_status = other.m_status; + } + return *this; +} + +SchemaCheckerResult::~SchemaCheckerResult() { +} + +void SchemaCheckerResult::addError(const std::string& error){ + m_errors.emplace_back(error); + m_status = Status::FAILED; +} + +void SchemaCheckerResult::addWarning(const std::string& warning){ + m_warnings.emplace_back(warning); +} + +SchemaCheckerResult::Status SchemaCheckerResult::getStatus() const { + return m_status; +} + +void SchemaCheckerResult::displayErrors(std::ostream & os) const { + for(auto &error: m_errors){ + os << " ERROR: " << error << std::endl; + } +} + +void SchemaCheckerResult::displayWarnings(std::ostream& os) const{ + for(auto &warning: m_warnings){ + os << " WARNING: " << warning << std::endl; + } +} + + +std::string SchemaCheckerResult::statusToString(const Status& status){ + switch(status){ + case Status::SUCCESS: + return "SUCCESS"; + case Status::FAILED: + return "FAILED"; + default: + return " UnknownStatus"; + } +} + +}} \ No newline at end of file diff --git a/catalogue/SchemaComparerResult.hpp b/catalogue/SchemaCheckerResult.hpp similarity index 67% rename from catalogue/SchemaComparerResult.hpp rename to catalogue/SchemaCheckerResult.hpp index 685acad1e45ea7fd0bfa7f6904c54adb7d0785be..dc7cabc03846626350a9e1ce0b935257c933323e 100644 --- a/catalogue/SchemaComparerResult.hpp +++ b/catalogue/SchemaCheckerResult.hpp @@ -27,9 +27,10 @@ namespace catalogue { * This class holds the results of the Schema comparison against the catalogue database schema * It is simply composed of: * - a list of differences between the catalogue schema and the schema we are comparing it against + * - a list of warnings * - a Status (SUCCESS or FAILED) */ -class SchemaComparerResult { +class SchemaCheckerResult { public: /** * The comparison is either SUCCESS or FAILED @@ -40,9 +41,9 @@ public: }; static std::string statusToString(const Status& status); - SchemaComparerResult(); - SchemaComparerResult(const SchemaComparerResult& orig); - SchemaComparerResult operator=(const SchemaComparerResult &other); + SchemaCheckerResult(); + SchemaCheckerResult(const SchemaCheckerResult& orig); + SchemaCheckerResult operator=(const SchemaCheckerResult &other); /** * We can combine the SchemaComparerResult in order to add other Results to it * @param other the SchemaComparerResult object to add @@ -51,24 +52,39 @@ public: * Note: The status will never change if it is failed (this or other) * It will simply append the list of differences of other to this SchemaComparerResult */ - SchemaComparerResult operator+=(const SchemaComparerResult &other); + SchemaCheckerResult operator+=(const SchemaCheckerResult &other); + + /** + * Prints the errors the ostream + * @param os the ostream to print the errors + */ + void displayErrors(std::ostream & os) const; + /** - * Prints the differences in this result + * Prints the warnings the ostream + * @param os the ostream to print the warnings */ - void printDiffs() const; + void displayWarnings(std::ostream & os) const; /** * Returns the status of the SchemaComparerResult * @return the status of the SchemaComparerResult */ Status getStatus() const; /** - * Add a difference to this Result - * @param diff the difference to add + * Add an error to this result + * @param error the error to add */ - void addDiff(const std::string &diff); - virtual ~SchemaComparerResult(); + void addError(const std::string &error); + /** + * Add a warning to this result + * @param warning the warning to add + */ + void addWarning(const std::string & warning); + + virtual ~SchemaCheckerResult(); private: - std::list<std::string> m_diffs; + std::list<std::string> m_errors; + std::list<std::string> m_warnings; Status m_status; }; diff --git a/catalogue/SchemaComparer.hpp b/catalogue/SchemaComparer.hpp index 1a281d8e3ffbefd8129f0521abbc01d6ca5bcdc8..2db94cde3b3c6c82af2513b71f48d93923e1d08a 100644 --- a/catalogue/SchemaComparer.hpp +++ b/catalogue/SchemaComparer.hpp @@ -25,7 +25,7 @@ #pragma once #include "rdbms/ConnPool.hpp" -#include "SchemaComparerResult.hpp" +#include "SchemaCheckerResult.hpp" #include "DatabaseMetadataGetter.hpp" #include "SchemaSqlStatementsReader.hpp" @@ -51,24 +51,24 @@ public: * Compare the schema to compare against the database * @return a SchemaComparerResult object that will contain the differences if there are some */ - virtual SchemaComparerResult compareAll() = 0; + virtual SchemaCheckerResult compareAll() = 0; /** * Compare the tables of the schema against the catalogue database * @return a SchemaComparerResult that will contain the differences if there are some */ - virtual SchemaComparerResult compareTables() = 0; + virtual SchemaCheckerResult compareTables() = 0; /** * Compare the indexes of the schema against the catalogue database * @return a SchemaComparerResult that will contain the differences if there are some */ - virtual SchemaComparerResult compareIndexes() = 0; + virtual SchemaCheckerResult compareIndexes() = 0; /** * Compare only the tables that are located in the schema * This is useful when want to compare ONLY tables that are defined in a schema * @return a SchemaComparerResult that will contain the differences if there are some */ - virtual SchemaComparerResult compareTablesLocatedInSchema() = 0; + virtual SchemaCheckerResult compareTablesLocatedInSchema() = 0; /** * Sets the way the schema sql statements will be read to do the schemas comparison * @param schemaSqlStatementsReader the reader used to get the schema sql statements in order to do schema comparison diff --git a/catalogue/SchemaComparerResult.cpp b/catalogue/SchemaComparerResult.cpp deleted file mode 100644 index d90165ccb55b17dfe20812e6bf484b770f8f826b..0000000000000000000000000000000000000000 --- a/catalogue/SchemaComparerResult.cpp +++ /dev/null @@ -1,80 +0,0 @@ -/** - * The CERN Tape Archive (CTA) project - * Copyright © 2018 CERN - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see <http://www.gnu.org/licenses/>. - */ - -#include <iostream> - -#include "SchemaComparerResult.hpp" - -namespace cta { -namespace catalogue { - -SchemaComparerResult::SchemaComparerResult():m_status(Status::SUCCESS) { -} - -SchemaComparerResult::SchemaComparerResult(const SchemaComparerResult& orig) { - m_diffs = orig.m_diffs; - m_status = orig.m_status; -} - -SchemaComparerResult SchemaComparerResult::operator=(const SchemaComparerResult &other){ - if(this != &other){ - m_diffs = other.m_diffs; - m_status = other.m_status; - } - return *this; -} - -SchemaComparerResult SchemaComparerResult::operator +=(const SchemaComparerResult& other){ - m_diffs.insert(m_diffs.end(),other.m_diffs.begin(),other.m_diffs.end()); - if(m_status == Status::SUCCESS){ - // The status should not change if it is failed - m_status = other.m_status; - } - return *this; -} - -SchemaComparerResult::~SchemaComparerResult() { -} - -void SchemaComparerResult::addDiff(const std::string& diff){ - m_diffs.emplace_back(diff); - m_status = Status::FAILED; -} - -SchemaComparerResult::Status SchemaComparerResult::getStatus() const { - return m_status; -} - -void SchemaComparerResult::printDiffs() const { - for(auto &diff: m_diffs){ - std::cerr << " ERROR: " << diff << std::endl; - } -} - -std::string SchemaComparerResult::statusToString(const Status& status){ - switch(status){ - case Status::SUCCESS: - return "SUCCESS"; - case Status::FAILED: - return "FAILED"; - default: - return " UnknownStatus"; - } -} - -}} \ No newline at end of file diff --git a/catalogue/SchemaVersion.cpp b/catalogue/SchemaVersion.cpp index a4eca8333e0a2a72fb3d069874b8faa7ce9d236d..929d20f521e50fad6eecef7e0d63fddacfedb115 100644 --- a/catalogue/SchemaVersion.cpp +++ b/catalogue/SchemaVersion.cpp @@ -24,10 +24,6 @@ namespace cta { namespace catalogue { SchemaVersion::SchemaVersion(){} -SchemaVersion::SchemaVersion(const uint64_t schemaVersionMajor, const uint64_t schemaVersionMinor, const cta::optional<uint64_t> schemaVersionMajorNext, const cta::optional<uint64_t> schemaVersionMinorNext, const Status status):m_schemaVersionMajor(schemaVersionMajor), m_schemaVersionMinor(schemaVersionMinor), m_nextSchemaVersionMajor(schemaVersionMajorNext), m_nextSchemaVersionMinor(schemaVersionMinorNext), m_status(status) { - -} - SchemaVersion::SchemaVersion(const SchemaVersion& orig) { if(this != &orig){ m_schemaVersionMajor = orig.m_schemaVersionMajor; diff --git a/catalogue/SchemaVersion.hpp b/catalogue/SchemaVersion.hpp index 6536a3f1109a2b15811c203e2b67e88f05345a10..7998dd9163a737b59afbb8838459c82403c0e48d 100644 --- a/catalogue/SchemaVersion.hpp +++ b/catalogue/SchemaVersion.hpp @@ -54,7 +54,6 @@ private: cta::optional<uint64_t> m_nextSchemaVersionMinor; SchemaVersion::Status m_status; SchemaVersion(); - SchemaVersion(const uint64_t schemaVersionMajor, const uint64_t schemaVersionMinor, const cta::optional<uint64_t> nextSchemaVersionMajor, const cta::optional<uint64_t> nextSchemaVersionMinor, const Status status); }; class SchemaVersion::Builder{ diff --git a/catalogue/VerifySchemaCmd.cpp b/catalogue/VerifySchemaCmd.cpp index cc8f70d3507ef158ada0fd45fd24b88145db9c61..5bea49f513099ab061f392162b451fe534f4e418 100644 --- a/catalogue/VerifySchemaCmd.cpp +++ b/catalogue/VerifySchemaCmd.cpp @@ -80,10 +80,11 @@ int VerifySchemaCmd::exceptionThrowingMain(const int argc, char *const *const ar .useSQLiteSchemaComparer() .build()); - SchemaChecker::Status comparisonStatus = schemaChecker->compareSchema(); - schemaChecker->checkNoParallelTables(); - schemaChecker->checkSchemaNotUpgrading(); - if(comparisonStatus == SchemaChecker::Status::FAILURE){ + SchemaCheckerResult result = schemaChecker->displayingCompareSchema(std::cout,std::cerr); + result += schemaChecker->checkNoParallelTables(); + result += schemaChecker->checkSchemaNotUpgrading(); + result.displayWarnings(std::cout); + if(result.getStatus() == SchemaCheckerResult::Status::FAILED){ return 1; } return 0; diff --git a/catalogue/common_catalogue_schema.sql b/catalogue/common_catalogue_schema.sql index c69e545a346850f5dd5de994519b9daaa71f378f..f740467c7494c4f7d4761b30a54eaae16cf122e6 100644 --- a/catalogue/common_catalogue_schema.sql +++ b/catalogue/common_catalogue_schema.sql @@ -3,7 +3,9 @@ CREATE TABLE CTA_CATALOGUE( SCHEMA_VERSION_MINOR UINT64TYPE CONSTRAINT CTA_CATALOGUE_SVM2_NN NOT NULL, NEXT_SCHEMA_VERSION_MAJOR UINT64TYPE, NEXT_SCHEMA_VERSION_MINOR UINT64TYPE, - STATUS VARCHAR(100) + STATUS VARCHAR(100), + IS_PRODUCTION CHAR(1) DEFAULT '0' CONSTRAINT CTA_CATALOGUE_IP_NN NOT NULL, + CONSTRAINT CTA_CATALOGUE_IP_BOOL_CK CHECK(IS_PRODUCTION IN ('0','1')) ); CREATE TABLE ADMIN_USER( ADMIN_USER_NAME VARCHAR(100) CONSTRAINT ADMIN_USER_AUN_NN NOT NULL, diff --git a/statistics/StatisticsSaveCmd.cpp b/statistics/StatisticsSaveCmd.cpp index 7a7e0c25a965b687d424c625ab5cd1560cb9e182..a0937f4831dfed991735b36576b38563bbd6a77c 100644 --- a/statistics/StatisticsSaveCmd.cpp +++ b/statistics/StatisticsSaveCmd.cpp @@ -229,7 +229,7 @@ void StatisticsSaveCmd::checkCatalogueSchema(cta::rdbms::Conn& catalogueConn, ct SchemaChecker::Builder catalogueCheckerBuilder("catalogue",dbType,catalogueConn); std::unique_ptr<cta::catalogue::SchemaChecker> catalogueChecker = catalogueCheckerBuilder.build(); - SchemaChecker::Status tapeTableStatus = + SchemaCheckerResult result = catalogueChecker->checkTableContainsColumns("TAPE",{ "NB_MASTER_FILES", "MASTER_DATA_IN_BYTES", @@ -240,10 +240,10 @@ void StatisticsSaveCmd::checkCatalogueSchema(cta::rdbms::Conn& catalogueConn, ct "DIRTY" } ); - SchemaChecker::Status voTableStatus = - catalogueChecker->checkTableContainsColumns("TAPE_POOL",{"VIRTUAL_ORGANIZATION_ID"}); + result += catalogueChecker->checkTableContainsColumns("TAPE_POOL",{"VIRTUAL_ORGANIZATION_ID"}); - if(tapeTableStatus == SchemaChecker::Status::FAILURE || voTableStatus == SchemaChecker::Status::FAILURE ){ + if(result.getStatus() == SchemaCheckerResult::FAILED){ + result.displayErrors(std::cerr); throw cta::exception::Exception("Catalogue schema checking failed."); } } @@ -258,8 +258,9 @@ void StatisticsSaveCmd::checkStatisticsSchema(cta::rdbms::Conn& statisticsDataba statisticsCheckerBuilder.useCppSchemaStatementsReader(*statisticsSchema) .useSQLiteSchemaComparer() .build(); - SchemaChecker::Status compareTablesStatus = statisticsChecker->compareTablesLocatedInSchema(); - if(compareTablesStatus == SchemaChecker::Status::FAILURE){ + SchemaCheckerResult compareTablesStatus = statisticsChecker->compareTablesLocatedInSchema(); + if(compareTablesStatus.getStatus() == SchemaCheckerResult::FAILED){ + compareTablesStatus.displayErrors(std::cerr); throw cta::exception::Exception("Statistics schema checking failed."); } } diff --git a/statistics/StatisticsUpdateCmd.cpp b/statistics/StatisticsUpdateCmd.cpp index e06702a449cb9d2fa03ee697560afe9afa8e78ce..dd5bd9e090ae9c60645f725789a8221b52ea2de6 100644 --- a/statistics/StatisticsUpdateCmd.cpp +++ b/statistics/StatisticsUpdateCmd.cpp @@ -66,7 +66,7 @@ int StatisticsUpdateCmd::exceptionThrowingMain(const int argc, char *const *cons std::unique_ptr<cta::catalogue::SchemaChecker> catalogueChecker; catalogueChecker = catalogueCheckerBuilder.build(); - SchemaChecker::Status tapeTableStatus = catalogueChecker->checkTableContainsColumns("TAPE",{"VID", + SchemaCheckerResult result = catalogueChecker->checkTableContainsColumns("TAPE",{"VID", "DIRTY", "NB_MASTER_FILES", "MASTER_DATA_IN_BYTES", @@ -75,10 +75,11 @@ int StatisticsUpdateCmd::exceptionThrowingMain(const int argc, char *const *cons "NB_COPY_NB_GT_1", "COPY_NB_GT_1_IN_BYTES" }); - SchemaChecker::Status tapeFileTableStatus = catalogueChecker->checkTableContainsColumns("TAPE_FILE",{"VID","ARCHIVE_FILE_ID","FSEQ","COPY_NB","SUPERSEDED_BY_VID","SUPERSEDED_BY_FSEQ"}); - SchemaChecker::Status archiveFileTableStatus = catalogueChecker->checkTableContainsColumns("ARCHIVE_FILE",{"ARCHIVE_FILE_ID","SIZE_IN_BYTES"}); + result += catalogueChecker->checkTableContainsColumns("TAPE_FILE",{"VID","ARCHIVE_FILE_ID","FSEQ","COPY_NB","SUPERSEDED_BY_VID","SUPERSEDED_BY_FSEQ"}); + result += catalogueChecker->checkTableContainsColumns("ARCHIVE_FILE",{"ARCHIVE_FILE_ID","SIZE_IN_BYTES"}); - if(tapeTableStatus == SchemaChecker::Status::FAILURE || tapeFileTableStatus == SchemaChecker::Status::FAILURE || archiveFileTableStatus == SchemaChecker::Status::FAILURE){ + if(result.getStatus() == SchemaCheckerResult::FAILED){ + result.displayErrors(std::cerr); return EXIT_FAILURE; }