#include "watch.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace phosphor::certs { using ::phosphor::logging::elog; using ::sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; namespace fs = std::filesystem; Watch::Watch(sdeventplus::Event& event, std::string& certFile, Callback cb) : event(event), callback(std::move(cb)) { // get parent directory of certificate file to watch fs::path path = fs::path(certFile).parent_path(); try { if (!fs::exists(path)) { fs::create_directories(path); } } catch (const fs::filesystem_error& e) { lg2::error( "Failed to create directory, ERR:{ERR}, DIRECTORY:{DIRECTORY}", "ERR", e, "DIRECTORY", path); elog(); } watchDir = path; watchFile = fs::path(certFile).filename(); startWatch(); } Watch::~Watch() { stopWatch(); } void Watch::startWatch() { // stop any existing watch stopWatch(); fd = inotify_init1(IN_NONBLOCK); if (-1 == fd) { lg2::error("inotify_init1 failed: {ERR}", "ERR", std::strerror(errno)); elog(); } wd = inotify_add_watch(fd, watchDir.c_str(), IN_CLOSE_WRITE); if (-1 == wd) { close(fd); lg2::error("inotify_add_watch failed, ERR:{ERR}, WATCH:{WATCH}", "ERR", std::strerror(errno), "WATCH", watchDir); elog(); } ioPtr = std::make_unique( event, fd, EPOLLIN, [this](sdeventplus::source::IO&, int fd, uint32_t) { constexpr int size = sizeof(struct inotify_event) + NAME_MAX + 1; std::array buffer{}; ssize_t length = read(fd, buffer.data(), buffer.size()); if (length >= static_cast(sizeof(struct inotify_event))) { struct inotify_event* notifyEvent = reinterpret_cast(&buffer[0]); if (notifyEvent->len) { if (watchFile == notifyEvent->name) { callback(); } } } else { lg2::error("Failed to read inotify event"); } }); } void Watch::stopWatch() { if (-1 != fd) { if (-1 != wd) { inotify_rm_watch(fd, wd); } close(fd); } if (ioPtr) { ioPtr.reset(); } } } // namespace phosphor::certs