rcpp/rcpp.cc

191 lines
3.7 KiB
C++
Raw Normal View History

2023-10-13 11:23:34 +00:00
///
/// \file rcpp.cc
/// \author kyle
/// \created 2023-10-12
/// \brief rcpp is a recursive copy by pattern program.
///
/// \section COPYRIGHT
///
/// Copyright 2023 K. Isom <kyle@imap.cc>
///
/// Permission to use, copy, modify, and/or distribute this software for
/// any purpose with or without fee is hereby granted, provided that
/// the above copyright notice and this permission notice appear in all
/// copies.
///
/// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
/// WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
/// WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
/// AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
/// DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA
/// OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
/// TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
/// PERFORMANCE OF THIS SOFTWARE.
///
#include <filesystem>
#include <fstream>
#include <iostream>
#include <regex>
2023-10-13 11:24:32 +00:00
#include "Flag.h"
using namespace klib;
2023-10-13 11:23:34 +00:00
#define ALIAS_FS namespace fs = std::filesystem
typedef std::filesystem::path Path;
#ifdef NO_TRACE
static std::ofstream null("/dev/null");
#define TRACE null
#else
#define TRACE std::cout
#endif
int
preflightDirectory(Path dst)
{
ALIAS_FS;
if (!fs::exists(dst)) {
fs::create_directory(dst);
} else if (!fs::is_directory(dst)) {
return -1;
}
return 0;
}
int
preflightChecks(Path dst, Path src)
{
ALIAS_FS;
if (preflightDirectory(dst) != 0) {
return -1;
}
if (!fs::exists(src)) {
return -1;
}
if (!fs::is_regular_file(src)) {
return -1;
}
return 0;
}
int
copyToPath(Path dst, Path src)
{
ALIAS_FS;
TRACE << "copyToPath " << dst << " " << src << "\n";
if (preflightChecks(dst, src)) {
return -1;
}
Path dstFile = dst / src.filename();
TRACE << "destination file: " << dstFile << "\n";
fs::copy(src, dst, fs::copy_options::update_existing);
return 0;
}
int
maybeCopyPattern(Path dst, Path entry, std::regex &pattern)
{
ALIAS_FS;
TRACE << "maybeCopyPattern " << entry << "\n";
if (!fs::is_regular_file(entry)) {
TRACE << "\tnot a file\n";
return 0;
}
if (!std::regex_search(entry.string(), pattern)) {
TRACE << "\tpattern doesn't match\n";
return 0;
}
return (copyToPath(dst, entry));
}
int
recursiveCopyPattern(Path &dst, Path &src, std::regex &pattern)
{
ALIAS_FS;
if (!fs::exists(src)) {
return -1;
}
if (!fs::is_directory(src)) {
return -1;
}
for (auto entry : fs::recursive_directory_iterator(src)) {
TRACE << "found " << entry << "\n";
if (maybeCopyPattern(dst, entry, pattern)) {
return -1;
}
}
return 0;
}
int
main(int argc, char *argv[])
{
ALIAS_FS;
auto regexOptions = std::regex_constants::optimize;
2023-10-13 11:24:32 +00:00
Flags optArgs("rcpp", "recursively copy files by pattern");
optArgs.Register("-e", FlagType::Boolean, "pattern is a file extension");
optArgs.Register("-i", FlagType::Boolean, "case-insensitive maches");
if (optArgs.Parse(argc, argv)) {
optArgs.Usage(std::cerr, 1);
}
if (optArgs.NumArgs() < 3) {
optArgs.Usage(std::cerr, 1);
2023-10-13 11:23:34 +00:00
}
2023-10-13 11:24:32 +00:00
auto pattern = optArgs.Arg(0);
auto destDir = fs::path(optArgs.Arg(optArgs.NumArgs() - 1));
2023-10-13 11:23:34 +00:00
bool value;
2023-10-13 11:24:32 +00:00
if (optArgs.GetBool("-e", value) && value) {
pattern = "\\." + pattern + "$";
}
if (optArgs.GetBool("-i", value) && value) {
regexOptions |= std::regex_constants::icase;
}
destDir = fs::absolute(destDir);
2023-10-13 11:23:34 +00:00
TRACE << "argc: " << argc << "\n";
auto rePattern = std::regex(pattern);
for (int i = 2; i < argc-1; i++) {
TRACE << "[" << i << "] consider " << argv[i] << "\n";
auto src = fs::path(argv[i]);
if (recursiveCopyPattern(destDir, src, rePattern) != 0) {
TRACE << "failed\n";
}
}
return 0;
}