diff --git a/ReleaseNotes.md b/ReleaseNotes.md index 9b4c9ed5fe5bf0d97c7f1207a994cbb90dce7ca3..ad6aeda97d857328c79c06ac02d5e18290a80b19 100644 --- a/ReleaseNotes.md +++ b/ReleaseNotes.md @@ -7,6 +7,7 @@ - cta/CTA#213 - Add tool for injecting file into eos - cta/CTA#224 - Improve error message for cta-verify-file whn VID does not exist - cta/CTA#230 - Modify CTA code to enforce VID uppercase +- cta/CTA#239 - Add improvments to the cta-change-storage-class tool ### Bug Fixes - cta/CTA#234 - Replace stoi with toUint64 in standalone cli tool - cta/CTA#238 - Compilation fails when using cta::common::Configuration::getConfEntInt(...) diff --git a/cmdline/CtaAdminCmdParse.hpp b/cmdline/CtaAdminCmdParse.hpp index 7ceba7d4df515b49c3de26c0fdfa87fcd2233307..e7eac64628b9a81b4554f12622ff838077f0a560 100644 --- a/cmdline/CtaAdminCmdParse.hpp +++ b/cmdline/CtaAdminCmdParse.hpp @@ -634,7 +634,7 @@ const std::map<cmd_key_t, cmd_val_t> cmdOptions = { {{ AdminCmd::CMD_STORAGECLASS, AdminCmd::SUBCMD_CH }, { opt_storageclass_alias, opt_copynb_alias.optional(), opt_vo.optional(), opt_comment.optional() }}, {{ AdminCmd::CMD_STORAGECLASS, AdminCmd::SUBCMD_RM }, { opt_storageclass_alias }}, - {{ AdminCmd::CMD_STORAGECLASS, AdminCmd::SUBCMD_LS }, { }}, + {{ AdminCmd::CMD_STORAGECLASS, AdminCmd::SUBCMD_LS }, { opt_storageclass_alias.optional() }}, /*----------------------------------------------------------------------------------------------------*/ {{ AdminCmd::CMD_TAPE, AdminCmd::SUBCMD_ADD }, { opt_vid, opt_mediatype, opt_vendor, opt_logicallibrary, opt_tapepool, opt_full, diff --git a/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.cpp b/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.cpp index a8ff83a083a5baacc50c041e522ba46b0793ed4b..fba8a08df1f103eb5a7abf3e39a301ec11ef51f0 100644 --- a/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.cpp +++ b/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.cpp @@ -34,9 +34,9 @@ #include "cmdline/standalone_cli_tools/common/ConnectionConfiguration.hpp" #include "common/checksum/ChecksumBlob.hpp" #include "common/exception/CommandLineNotParsed.hpp" -#include "common/utils/utils.hpp" +#include "common/exception/UserError.hpp" #include "common/log/StdoutLogger.hpp" - +#include "common/utils/utils.hpp" #include "CtaFrontendApi.hpp" // GLOBAL VARIABLES : used to pass information between main thread and stream handler thread @@ -55,27 +55,29 @@ ChangeStorageClass::ChangeStorageClass( std::istream &inStream, std::ostream &outStream, std::ostream &errStream, - cta::log::StdoutLogger &log, - const int& argc, - char *const *const argv): + cta::log::StdoutLogger &log): CmdLineTool(inStream, outStream, errStream), - m_log(log) { + m_log(log) {} - CmdLineArgs cmdLineArgs(argc, argv, StandaloneCliTool::CTA_CHANGE_STORAGE_CLASS); +//------------------------------------------------------------------------------ +// exceptionThrowingMain +//------------------------------------------------------------------------------ +int ChangeStorageClass::exceptionThrowingMain(const int argc, char *const *const argv) { + CmdLineArgs cmdLineArgs(argc, argv, StandaloneCliTool::CTA_CHANGE_STORAGE_CLASS); if (cmdLineArgs.m_help) { cmdLineArgs.printUsage(std::cout); - exit(0); + throw exception::UserError(""); } if (!cmdLineArgs.m_storageClassName) { cmdLineArgs.printUsage(std::cout); - exit(1); + throw exception::UserError("Missing requried option: storage.class.name"); } if (!cmdLineArgs.m_archiveFileId && !cmdLineArgs.m_archiveFileIds) { cmdLineArgs.printUsage(std::cout); - exit(1); + throw exception::UserError("filename or id must be provided"); } m_storageClassName = cmdLineArgs.m_storageClassName.value(); @@ -85,7 +87,7 @@ ChangeStorageClass::ChangeStorageClass( } if (cmdLineArgs.m_archiveFileIds) { - for (const auto archiveFileId : cmdLineArgs.m_archiveFileIds.value() ) { + for (const auto& archiveFileId : cmdLineArgs.m_archiveFileIds.value() ) { m_archiveFileIds.push_back(archiveFileId); } } @@ -96,20 +98,23 @@ ChangeStorageClass::ChangeStorageClass( m_eosUpdateFrequency = 100; } - auto [serviceProvider, endpointmap] = ConnConfiguration::readAndSetConfiguration(log, getUsername(), cmdLineArgs); + auto [serviceProvider, endpointmap] = ConnConfiguration::readAndSetConfiguration(m_log, getUsername(), cmdLineArgs); m_serviceProviderPtr = std::move(serviceProvider); m_endpointMapPtr = std::move(endpointmap); + + storageClassExists(); + updateStorageClassInEosNamespace(); + updateStorageClassInCatalogue(); + writeSkippedArchiveIdsToFile(); + return 0; } //------------------------------------------------------------------------------ // fileInFlight //------------------------------------------------------------------------------ -bool ChangeStorageClass::fileInFlight(const google::protobuf::RepeatedField<uint32_t> &locations) { - for (auto location : locations) { - // file is not in flight if fsid==65535 - if (location == 65535) { return false; } - } - return true; +bool ChangeStorageClass::fileInFlight(const google::protobuf::RepeatedField<uint32_t> &locations) const { + // file is not in flight if fsid==65535 + return std::all_of(std::begin(locations), std::end(locations), [](const int &location) { return location != 65535; }); } @@ -128,8 +133,8 @@ void ChangeStorageClass::updateStorageClassInEosNamespace() { const auto [diskInstance, diskFileId] = CatalogueFetch::getInstanceAndFid(archiveFileId, m_serviceProviderPtr, m_log); // No files in flight should change storage class - const auto md_response = m_endpointMapPtr->getMD(diskInstance, ::eos::rpc::FILE, cta::utils::toUint64(diskFileId), "", false); - if (fileInFlight(md_response.fmd().locations())) { + if (const auto md_response = m_endpointMapPtr->getMD(diskInstance, ::eos::rpc::FILE, cta::utils::toUint64(diskFileId), "", false); + fileInFlight(md_response.fmd().locations())) { m_archiveIdsNotUpdatedInEos.push_back(archiveFileId); std::list<cta::log::Param> params; params.push_back(cta::log::Param("archiveFileId", archiveFileId)); @@ -154,7 +159,7 @@ void ChangeStorageClass::updateStorageClassInEosNamespace() { //------------------------------------------------------------------------------ // storageClassExists //------------------------------------------------------------------------------ -void ChangeStorageClass::storageClassExists() { +void ChangeStorageClass::storageClassExists() const { cta::xrd::Request request; const auto admincmd = request.mutable_admincmd(); @@ -163,6 +168,13 @@ void ChangeStorageClass::storageClassExists() { admincmd->set_cmd(cta::admin::AdminCmd::CMD_STORAGECLASS); admincmd->set_subcmd(cta::admin::AdminCmd::SUBCMD_LS); + { + const auto key = cta::admin::OptionString::STORAGE_CLASS; + const auto new_opt = admincmd->add_option_str(); + new_opt->set_key(key); + new_opt->set_value(m_storageClassName); + } + cta::xrd::Response response; auto stream_future = m_serviceProviderPtr->SendAsync(request, response); @@ -185,8 +197,7 @@ void ChangeStorageClass::storageClassExists() { // wait until the data stream has been processed before exiting stream_future.wait(); - std::list<std::string>::iterator storageClassFound = std::find(g_storageClasses.begin(), g_storageClasses.end(), m_storageClassName); - if ( g_storageClasses.end() == storageClassFound ){ + if (g_storageClasses.empty()){ throw(exception::UserError("The storage class provided has not been defined.")); } } @@ -194,7 +205,7 @@ void ChangeStorageClass::storageClassExists() { //------------------------------------------------------------------------------ // updateCatalogue //------------------------------------------------------------------------------ -void ChangeStorageClass::updateStorageClassInCatalogue() { +void ChangeStorageClass::updateStorageClassInCatalogue() const { cta::xrd::Request request; const auto admincmd = request.mutable_admincmd(); @@ -249,7 +260,7 @@ void ChangeStorageClass::updateStorageClassInCatalogue() { //------------------------------------------------------------------------------ // writeSkippedArchiveIdsToFile //------------------------------------------------------------------------------ -void ChangeStorageClass::writeSkippedArchiveIdsToFile() { +void ChangeStorageClass::writeSkippedArchiveIdsToFile() const { const std::filesystem::path filePath = "/var/log/skippedArchiveIds.txt"; std::ofstream archiveIdFile(filePath); @@ -258,7 +269,7 @@ void ChangeStorageClass::writeSkippedArchiveIdsToFile() { } if (archiveIdFile.is_open()) { - for (const auto archiveId : m_archiveIdsNotUpdatedInEos) { + for (const auto& archiveId : m_archiveIdsNotUpdatedInEos) { archiveIdFile << archiveId << std::endl; } archiveIdFile.close(); @@ -267,17 +278,5 @@ void ChangeStorageClass::writeSkippedArchiveIdsToFile() { } } -//------------------------------------------------------------------------------ -// exceptionThrowingMain -//------------------------------------------------------------------------------ -int ChangeStorageClass::exceptionThrowingMain(const int argc, char *const *const argv) { - storageClassExists(); - updateStorageClassInEosNamespace(); - updateStorageClassInCatalogue(); - writeSkippedArchiveIdsToFile(); - return 0; -} - - } // namespace admin } // namespace cta diff --git a/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.hpp b/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.hpp index 87a5b6187ad73a831b9d9accf4931be5dcf8e5f6..64246a2b0c7b4beb1d8a3efd2d0b49c50315e751 100644 --- a/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.hpp +++ b/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClass.hpp @@ -50,9 +50,7 @@ public: std::istream &inStream, std::ostream &outStream, std::ostream &errStream, - cta::log::StdoutLogger &log, - const int& argc, - char *const *const argv); + cta::log::StdoutLogger &log); private: @@ -105,22 +103,22 @@ private: * Checks if the storage class provided to the tool is defined, * and throws an exception::UserError if it is not found */ - void storageClassExists(); + void storageClassExists() const; /** * Updates the storage class name in the catalogue */ - void updateStorageClassInCatalogue(); + void updateStorageClassInCatalogue() const; /** * Writes the skipped archive ids to file */ - void writeSkippedArchiveIdsToFile(); + void writeSkippedArchiveIdsToFile() const; /** * Checks if a file is in flight */ - bool fileInFlight(const google::protobuf::RepeatedField<unsigned int> &locations); + bool fileInFlight(const google::protobuf::RepeatedField<unsigned int> &locations) const; /** * An exception throwing version of main(). diff --git a/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClassMain.cpp b/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClassMain.cpp index 5175ab2e58891f811d4aa74395d46903ca208062..c0aedff3d1841ec1d5618eb5896c6e37ca923c72 100644 --- a/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClassMain.cpp +++ b/cmdline/standalone_cli_tools/change_storage_class/ChangeStorageClassMain.cpp @@ -35,7 +35,7 @@ int main(const int argc, char *const *const argv) { cta::log::StdoutLogger log(hostName, "cta-change-storage-class"); - cta::cliTool::ChangeStorageClass cmd(std::cin, std::cout, std::cerr, log, argc, argv); + cta::cliTool::ChangeStorageClass cmd(std::cin, std::cout, std::cerr, log); int ret = cmd.main(argc, argv); // Delete all global objects allocated by libprotobuf google::protobuf::ShutdownProtobufLibrary(); diff --git a/xroot_plugins/XrdCtaStorageClassLs.hpp b/xroot_plugins/XrdCtaStorageClassLs.hpp index ff5cdba7c05b1df85462f52cbeec74358b2309f1..a881f809fa5e7a76bb3a3b85d0c9682ad944b0e7 100644 --- a/xroot_plugins/XrdCtaStorageClassLs.hpp +++ b/xroot_plugins/XrdCtaStorageClassLs.hpp @@ -19,6 +19,7 @@ #include <xroot_plugins/XrdCtaStream.hpp> #include <xroot_plugins/XrdSsiCtaRequestMessage.hpp> +#include <optional> namespace cta { namespace xrd { @@ -35,7 +36,7 @@ public: * @param[in] catalogue CTA Catalogue * @param[in] scheduler CTA Scheduler */ - StorageClassLsStream(const RequestMessage &requestMsg, cta::catalogue::Catalogue &catalogue, cta::Scheduler &scheduler); + StorageClassLsStream(const RequestMessage &requestMsg, cta::catalogue::Catalogue &catalogue, cta::Scheduler &scheduler, const std::optional<std::string> storageClassName); private: /*! @@ -52,17 +53,27 @@ private: std::list<cta::common::dataStructures::StorageClass> m_storageClassList; //!< List of storage classes from the catalogue + std::optional<std::string> m_storageClassName; + static constexpr const char* const LOG_SUFFIX = "StorageClassLsStream"; //!< Identifier for log messages }; -StorageClassLsStream::StorageClassLsStream(const RequestMessage &requestMsg, cta::catalogue::Catalogue &catalogue, cta::Scheduler &scheduler) : +StorageClassLsStream::StorageClassLsStream(const RequestMessage &requestMsg, cta::catalogue::Catalogue &catalogue, cta::Scheduler &scheduler, const std::optional<std::string> storageClassName) : XrdCtaStream(catalogue, scheduler), - m_storageClassList(catalogue.getStorageClasses()) + m_storageClassName(storageClassName) { using namespace cta::admin; XrdSsiPb::Log::Msg(XrdSsiPb::Log::DEBUG, LOG_SUFFIX, "StorageClassLsStream() constructor"); + + if(m_storageClassName) { + m_storageClassList.push_back(m_catalogue.getStorageClass(m_storageClassName.value())); + } else { + for(const auto &storageClass : m_catalogue.getStorageClasses()) { + m_storageClassList.push_back(storageClass); + } + } } int StorageClassLsStream::fillBuffer(XrdSsiPb::OStreamBuffer<Data> *streambuf) { diff --git a/xroot_plugins/XrdSsiCtaRequestMessage.cpp b/xroot_plugins/XrdSsiCtaRequestMessage.cpp index f48b47247b208122876bbd4866aba3a845252fb8..d81226f41ab4f8d7dd488a133ef19c4fcdeb4b99 100644 --- a/xroot_plugins/XrdSsiCtaRequestMessage.cpp +++ b/xroot_plugins/XrdSsiCtaRequestMessage.cpp @@ -1872,8 +1872,10 @@ void RequestMessage::processStorageClass_Ls(cta::xrd::Response &response, XrdSsi { using namespace cta::admin; + std::optional<std::string> scn = getOptional(OptionString::STORAGE_CLASS); + // Create a XrdSsi stream object to return the results - stream = new StorageClassLsStream(*this, m_catalogue, m_scheduler); + stream = new StorageClassLsStream(*this, m_catalogue, m_scheduler, scn); response.set_show_header(HeaderType::STORAGECLASS_LS); response.set_type(cta::xrd::Response::RSP_SUCCESS);