4 #include <apt-pkg/cachefile.h>
5 #include <apt-pkg/cacheset.h>
6 #include <apt-pkg/cmndline.h>
7 #include <apt-pkg/pkgrecords.h>
8 #include <apt-pkg/policy.h>
9 #include <apt-pkg/progress.h>
10 #include <apt-pkg/cacheiterators.h>
11 #include <apt-pkg/configuration.h>
12 #include <apt-pkg/depcache.h>
13 #include <apt-pkg/macros.h>
14 #include <apt-pkg/pkgcache.h>
16 #include <apt-private/private-cacheset.h>
17 #include <apt-private/private-output.h>
18 #include <apt-private/private-search.h>
30 bool FullTextSearch(CommandLine
&CmdL
) /*{{{*/
32 pkgCacheFile CacheFile
;
33 pkgCache
*Cache
= CacheFile
.GetPkgCache();
34 pkgDepCache::Policy
*Plcy
= CacheFile
.GetPolicy();
35 pkgRecords
records(CacheFile
);
36 if (unlikely(Cache
== NULL
|| Plcy
== NULL
))
39 // Make sure there is at least one argument
40 unsigned int const NumPatterns
= CmdL
.FileSize() -1;
42 return _error
->Error(_("You must give at least one search pattern"));
44 #define APT_FREE_PATTERNS() for (std::vector<regex_t>::iterator P = Patterns.begin(); \
45 P != Patterns.end(); ++P) { regfree(&(*P)); }
47 // Compile the regex pattern
48 std::vector
<regex_t
> Patterns
;
49 for (unsigned int I
= 0; I
!= NumPatterns
; ++I
)
52 if (regcomp(&pattern
, CmdL
.FileList
[I
+ 1], REG_EXTENDED
| REG_ICASE
| REG_NOSUB
) != 0)
55 return _error
->Error("Regex compilation error");
57 Patterns
.push_back(pattern
);
60 bool const NamesOnly
= _config
->FindB("APT::Cache::NamesOnly", false);
62 std::map
<std::string
, std::string
> output_map
;
63 std::map
<std::string
, std::string
>::const_iterator K
;
65 LocalitySortedVersionSet bag
;
66 OpTextProgress
progress(*_config
);
67 progress
.OverallProgress(0, 100, 50, _("Sorting"));
68 GetLocalitySortedVersionSet(CacheFile
, bag
, progress
);
69 LocalitySortedVersionSet::iterator V
= bag
.begin();
71 progress
.OverallProgress(50, 100, 50, _("Full Text Search"));
72 progress
.SubProgress(bag
.size());
74 for ( ;V
!= bag
.end(); ++V
)
77 progress
.Progress(Done
);
80 pkgCache::DescIterator Desc
= V
.TranslatedDescription();
81 pkgRecords::Parser
&parser
= records
.Lookup(Desc
.FileList());
83 bool all_found
= true;
84 for (std::vector
<regex_t
>::const_iterator pattern
= Patterns
.begin();
85 pattern
!= Patterns
.end(); ++pattern
)
87 if (regexec(&(*pattern
), V
.ParentPkg().Name(), 0, 0, 0) == 0)
89 else if (NamesOnly
== false && regexec(&(*pattern
), parser
.LongDesc().c_str(), 0, 0, 0) == 0)
91 // search patterns are AND, so one failing fails all
95 if (all_found
== true)
97 std::stringstream outs
;
98 ListSingleVersion(CacheFile
, records
, V
, outs
);
99 output_map
.insert(std::make_pair
<std::string
, std::string
>(
100 V
.ParentPkg().Name(), outs
.str()));
106 // FIXME: SORT! and make sorting flexible (alphabetic, by pkg status)
107 // output the sorted map
108 for (K
= output_map
.begin(); K
!= output_map
.end(); ++K
)
109 std::cout
<< (*K
).second
<< std::endl
;