1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133
|
// Copyright Contributors to the DNF5 project.
// Copyright Contributors to the libdnf project.
// SPDX-License-Identifier: LGPL-2.1-or-later
//
// This file is part of libdnf: https://github.com/rpm-software-management/libdnf/
//
// Libdnf is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 2.1 of the License, or
// (at your option) any later version.
//
// Libdnf 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 Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with libdnf. If not, see <https://www.gnu.org/licenses/>.
#include "libdnf5/common/sack/match_string.hpp"
#include "common/sack/query_cmp_private.hpp"
#include "utils/string.hpp"
#include "libdnf5/common/exception.hpp"
#include <fnmatch.h>
#include <regex>
#include <stdexcept>
// TODO: Implement proper FNM_EXTMATCH support to fix MUSL errors.
#if !defined(FNM_EXTMATCH)
#define FNM_EXTMATCH (0)
#endif
namespace libdnf5::sack {
bool match_string(const std::string & value, QueryCmp cmp, const std::string & pattern) {
bool result = false;
switch (cmp - QueryCmp::NOT) {
case QueryCmp::EXACT:
result = value == pattern;
break;
case QueryCmp::IEXACT:
result = libdnf5::utils::string::tolower(value) == libdnf5::utils::string::tolower(pattern);
break;
case QueryCmp::GLOB:
result = fnmatch(pattern.c_str(), value.c_str(), FNM_EXTMATCH) == 0;
break;
case QueryCmp::IGLOB:
result = fnmatch(pattern.c_str(), value.c_str(), FNM_CASEFOLD | FNM_EXTMATCH) == 0;
break;
case QueryCmp::REGEX:
result = std::regex_match(value, std::regex(pattern));
break;
case QueryCmp::IREGEX:
result = std::regex_match(value, std::regex(pattern, std::regex::icase));
break;
case QueryCmp::CONTAINS:
result = value.find(pattern) != std::string::npos;
break;
case QueryCmp::ICONTAINS:
result = libdnf5::utils::string::tolower(value).find(libdnf5::utils::string::tolower(pattern)) !=
std::string::npos;
break;
case QueryCmp::STARTSWITH:
result = libdnf5::utils::string::starts_with(value, pattern);
break;
case QueryCmp::ISTARTSWITH:
result = libdnf5::utils::string::starts_with(
libdnf5::utils::string::tolower(value), libdnf5::utils::string::tolower(pattern));
break;
case QueryCmp::ENDSWITH:
result = libdnf5::utils::string::ends_with(value, pattern);
break;
case QueryCmp::IENDSWITH:
result = libdnf5::utils::string::ends_with(
libdnf5::utils::string::tolower(value), libdnf5::utils::string::tolower(pattern));
break;
default:
libdnf_assert(cmp - QueryCmp::NOT - QueryCmp::ICASE, "NOT and ICASE modifiers cannot be used standalone");
libdnf_throw_assert_unsupported_query_cmp_type(cmp);
}
return static_cast<bool>(cmp & QueryCmp::NOT) ? !result : result;
}
// cmp is positive: return true if the value matches at least one of patterns
// cmp is negative: return true if value doesn't match any of patterns
bool match_string(const std::string & value, QueryCmp cmp, const std::vector<std::string> & patterns) {
bool result = (cmp & libdnf5::sack::QueryCmp::NOT) == libdnf5::sack::QueryCmp::NOT;
for (auto & pattern : patterns) {
if (match_string(value, cmp, pattern) != result) {
return !result;
}
}
return result;
}
// cmp is positive: return true if at least one of the values matches the pattern
// cmp is negative: return true if neither of the values matches the pattern
bool match_string(const std::vector<std::string> & values, QueryCmp cmp, const std::string & pattern) {
bool result = (cmp & libdnf5::sack::QueryCmp::NOT) == libdnf5::sack::QueryCmp::NOT;
for (auto & value : values) {
if (match_string(value, cmp, pattern) != result) {
return !result;
}
}
return result;
}
// cmp is positive: return true if at least one of the values matches at least one of the patterns
// cmp is negative: return true if none of the values matches none of the patterns
bool match_string(const std::vector<std::string> & values, QueryCmp cmp, const std::vector<std::string> & patterns) {
bool result = (cmp & libdnf5::sack::QueryCmp::NOT) == libdnf5::sack::QueryCmp::NOT;
for (auto & value : values) {
for (auto & pattern : patterns) {
if (match_string(value, cmp, pattern) != result) {
return !result;
}
}
}
return result;
}
} // namespace libdnf5::sack
|