]> git.saurik.com Git - apt.git/blob - apt-pkg/cachefilter.cc
try to avoid removal of crossgraded packages
[apt.git] / apt-pkg / cachefilter.cc
1 // -*- mode: cpp; mode: fold -*-
2 // Description /*{{{*/
3 /** \file cachefilter.h
4 Collection of functor classes */
5 /*}}}*/
6 // Include Files /*{{{*/
7 #include <config.h>
8
9 #include <apt-pkg/cachefile.h>
10 #include <apt-pkg/cachefilter.h>
11 #include <apt-pkg/error.h>
12 #include <apt-pkg/pkgcache.h>
13 #include <apt-pkg/cacheiterators.h>
14 #include <apt-pkg/strutl.h>
15 #include <apt-pkg/macros.h>
16
17 #include <string>
18 #include <string.h>
19 #include <regex.h>
20 #include <fnmatch.h>
21
22 #include <apti18n.h>
23 /*}}}*/
24 namespace APT {
25 namespace CacheFilter {
26 APT_CONST Matcher::~Matcher() {}
27 APT_CONST PackageMatcher::~PackageMatcher() {}
28
29 // Name matches RegEx /*{{{*/
30 PackageNameMatchesRegEx::PackageNameMatchesRegEx(std::string const &Pattern) {
31 pattern = new regex_t;
32 int const Res = regcomp(pattern, Pattern.c_str(), REG_EXTENDED | REG_ICASE | REG_NOSUB);
33 if (Res == 0)
34 return;
35
36 delete pattern;
37 pattern = NULL;
38 char Error[300];
39 regerror(Res, pattern, Error, sizeof(Error));
40 _error->Error(_("Regex compilation error - %s"), Error);
41 }
42 bool PackageNameMatchesRegEx::operator() (pkgCache::PkgIterator const &Pkg) {
43 if (unlikely(pattern == NULL))
44 return false;
45 else
46 return regexec(pattern, Pkg.Name(), 0, 0, 0) == 0;
47 }
48 bool PackageNameMatchesRegEx::operator() (pkgCache::GrpIterator const &Grp) {
49 if (unlikely(pattern == NULL))
50 return false;
51 else
52 return regexec(pattern, Grp.Name(), 0, 0, 0) == 0;
53 }
54 PackageNameMatchesRegEx::~PackageNameMatchesRegEx() {
55 if (pattern == NULL)
56 return;
57 regfree(pattern);
58 delete pattern;
59 }
60 /*}}}*/
61 // Name matches Fnmatch /*{{{*/
62 PackageNameMatchesFnmatch::PackageNameMatchesFnmatch(std::string const &Pattern) :
63 Pattern(Pattern) {}
64 bool PackageNameMatchesFnmatch::operator() (pkgCache::PkgIterator const &Pkg) {
65 return fnmatch(Pattern.c_str(), Pkg.Name(), FNM_CASEFOLD) == 0;
66 }
67 bool PackageNameMatchesFnmatch::operator() (pkgCache::GrpIterator const &Grp) {
68 return fnmatch(Pattern.c_str(), Grp.Name(), FNM_CASEFOLD) == 0;
69 }
70 /*}}}*/
71 // Architecture matches <libc>-<kernel>-<cpu> specification /*{{{*/
72 //----------------------------------------------------------------------
73 /* The complete architecture, consisting of <libc>-<kernel>-<cpu>. */
74 static std::string CompleteArch(std::string const &arch, bool const isPattern) {
75 auto const found = arch.find('-');
76 if (found != std::string::npos)
77 {
78 // ensure that only -any- is replaced and not something like company-
79 std::string complete = std::string("-").append(arch).append("-");
80 size_t pos = 0;
81 char const * const search = "-any-";
82 auto const search_len = strlen(search) - 2;
83 while((pos = complete.find(search, pos)) != std::string::npos) {
84 complete.replace(pos + 1, search_len, "*");
85 pos += 2;
86 }
87 complete = complete.substr(1, complete.size()-2);
88 if (arch.find('-', found+1) != std::string::npos)
89 // <libc>-<kernel>-<cpu> format
90 return complete;
91 // <kernel>-<cpu> format
92 else if (isPattern)
93 return "*-" + complete;
94 else
95 return "gnu-" + complete;
96 }
97 else if (arch == "any")
98 return "*-*-*";
99 else if (isPattern)
100 return "*-linux-" + arch;
101 else
102 return "gnu-linux-" + arch;
103 }
104 PackageArchitectureMatchesSpecification::PackageArchitectureMatchesSpecification(std::string const &pattern, bool const pisPattern) :
105 literal(pattern), complete(CompleteArch(pattern, pisPattern)), isPattern(pisPattern) {
106 }
107 bool PackageArchitectureMatchesSpecification::operator() (char const * const &arch) {
108 if (strcmp(literal.c_str(), arch) == 0 ||
109 strcmp(complete.c_str(), arch) == 0)
110 return true;
111 std::string const pkgarch = CompleteArch(arch, !isPattern);
112 if (isPattern == true)
113 return fnmatch(complete.c_str(), pkgarch.c_str(), 0) == 0;
114 return fnmatch(pkgarch.c_str(), complete.c_str(), 0) == 0;
115 }
116 bool PackageArchitectureMatchesSpecification::operator() (pkgCache::PkgIterator const &Pkg) {
117 return (*this)(Pkg.Arch());
118 }
119 PackageArchitectureMatchesSpecification::~PackageArchitectureMatchesSpecification() {
120 }
121 /*}}}*/
122 // Package is new install /*{{{*/
123 PackageIsNewInstall::PackageIsNewInstall(pkgCacheFile * const Cache) : Cache(Cache) {}
124 APT_PURE bool PackageIsNewInstall::operator() (pkgCache::PkgIterator const &Pkg) {
125 return (*Cache)[Pkg].NewInstall();
126 }
127 PackageIsNewInstall::~PackageIsNewInstall() {}
128 /*}}}*/
129 // Generica like True, False, NOT, AND, OR /*{{{*/
130 APT_CONST bool TrueMatcher::operator() (pkgCache::PkgIterator const &) { return true; }
131 APT_CONST bool TrueMatcher::operator() (pkgCache::GrpIterator const &) { return true; }
132 APT_CONST bool TrueMatcher::operator() (pkgCache::VerIterator const &) { return true; }
133
134 APT_CONST bool FalseMatcher::operator() (pkgCache::PkgIterator const &) { return false; }
135 APT_CONST bool FalseMatcher::operator() (pkgCache::GrpIterator const &) { return false; }
136 APT_CONST bool FalseMatcher::operator() (pkgCache::VerIterator const &) { return false; }
137
138 NOTMatcher::NOTMatcher(Matcher * const matcher) : matcher(matcher) {}
139 bool NOTMatcher::operator() (pkgCache::PkgIterator const &Pkg) { return ! (*matcher)(Pkg); }
140 bool NOTMatcher::operator() (pkgCache::GrpIterator const &Grp) { return ! (*matcher)(Grp); }
141 bool NOTMatcher::operator() (pkgCache::VerIterator const &Ver) { return ! (*matcher)(Ver); }
142 NOTMatcher::~NOTMatcher() { delete matcher; }
143
144 ANDMatcher::ANDMatcher() {}
145 ANDMatcher::ANDMatcher(Matcher * const matcher1) {
146 AND(matcher1);
147 }
148 ANDMatcher::ANDMatcher(Matcher * const matcher1, Matcher * const matcher2) {
149 AND(matcher1).AND(matcher2);
150 }
151 ANDMatcher::ANDMatcher(Matcher * const matcher1, Matcher * const matcher2, Matcher * const matcher3) {
152 AND(matcher1).AND(matcher2).AND(matcher3);
153 }
154 ANDMatcher::ANDMatcher(Matcher * const matcher1, Matcher * const matcher2, Matcher * const matcher3, Matcher * const matcher4) {
155 AND(matcher1).AND(matcher2).AND(matcher3).AND(matcher4);
156 }
157 ANDMatcher::ANDMatcher(Matcher * const matcher1, Matcher * const matcher2, Matcher * const matcher3, Matcher * const matcher4, Matcher * const matcher5) {
158 AND(matcher1).AND(matcher2).AND(matcher3).AND(matcher4).AND(matcher5);
159 }
160 ANDMatcher& ANDMatcher::AND(Matcher * const matcher) { matchers.push_back(matcher); return *this; }
161 bool ANDMatcher::operator() (pkgCache::PkgIterator const &Pkg) {
162 for (std::vector<Matcher *>::const_iterator M = matchers.begin(); M != matchers.end(); ++M)
163 if ((**M)(Pkg) == false)
164 return false;
165 return true;
166 }
167 bool ANDMatcher::operator() (pkgCache::GrpIterator const &Grp) {
168 for (std::vector<Matcher *>::const_iterator M = matchers.begin(); M != matchers.end(); ++M)
169 if ((**M)(Grp) == false)
170 return false;
171 return true;
172 }
173 bool ANDMatcher::operator() (pkgCache::VerIterator const &Ver) {
174 for (std::vector<Matcher *>::const_iterator M = matchers.begin(); M != matchers.end(); ++M)
175 if ((**M)(Ver) == false)
176 return false;
177 return true;
178 }
179 ANDMatcher::~ANDMatcher() {
180 for (std::vector<Matcher *>::iterator M = matchers.begin(); M != matchers.end(); ++M)
181 delete *M;
182 }
183
184 ORMatcher::ORMatcher() {}
185 ORMatcher::ORMatcher(Matcher * const matcher1) {
186 OR(matcher1);
187 }
188 ORMatcher::ORMatcher(Matcher * const matcher1, Matcher * const matcher2) {
189 OR(matcher1).OR(matcher2);
190 }
191 ORMatcher::ORMatcher(Matcher * const matcher1, Matcher * const matcher2, Matcher * const matcher3) {
192 OR(matcher1).OR(matcher2).OR(matcher3);
193 }
194 ORMatcher::ORMatcher(Matcher * const matcher1, Matcher * const matcher2, Matcher * const matcher3, Matcher * const matcher4) {
195 OR(matcher1).OR(matcher2).OR(matcher3).OR(matcher4);
196 }
197 ORMatcher::ORMatcher(Matcher * const matcher1, Matcher * const matcher2, Matcher * const matcher3, Matcher * const matcher4, Matcher * const matcher5) {
198 OR(matcher1).OR(matcher2).OR(matcher3).OR(matcher4).OR(matcher5);
199 }
200 ORMatcher& ORMatcher::OR(Matcher * const matcher) { matchers.push_back(matcher); return *this; }
201 bool ORMatcher::operator() (pkgCache::PkgIterator const &Pkg) {
202 for (std::vector<Matcher *>::const_iterator M = matchers.begin(); M != matchers.end(); ++M)
203 if ((**M)(Pkg) == true)
204 return true;
205 return false;
206 }
207 bool ORMatcher::operator() (pkgCache::GrpIterator const &Grp) {
208 for (std::vector<Matcher *>::const_iterator M = matchers.begin(); M != matchers.end(); ++M)
209 if ((**M)(Grp) == true)
210 return true;
211 return false;
212 }
213 bool ORMatcher::operator() (pkgCache::VerIterator const &Ver) {
214 for (std::vector<Matcher *>::const_iterator M = matchers.begin(); M != matchers.end(); ++M)
215 if ((**M)(Ver) == true)
216 return true;
217 return false;
218 }
219 ORMatcher::~ORMatcher() {
220 for (std::vector<Matcher *>::iterator M = matchers.begin(); M != matchers.end(); ++M)
221 delete *M;
222 }
223 /*}}}*/
224
225 }
226 }