123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334 |
- // Includes /*{{{*/
- #include <config.h>
- #include <apt-pkg/cachefile.h>
- #include <apt-pkg/cacheset.h>
- #include <apt-pkg/cmndline.h>
- #include <apt-pkg/pkgrecords.h>
- #include <apt-pkg/policy.h>
- #include <apt-pkg/progress.h>
- #include <apt-pkg/cacheiterators.h>
- #include <apt-pkg/configuration.h>
- #include <apt-pkg/depcache.h>
- #include <apt-pkg/macros.h>
- #include <apt-pkg/pkgcache.h>
- #include <apt-private/private-cacheset.h>
- #include <apt-private/private-output.h>
- #include <apt-private/private-search.h>
- #include <apt-private/private-show.h>
- #include <string.h>
- #include <iostream>
- #include <sstream>
- #include <map>
- #include <string>
- #include <utility>
- #include <apti18n.h>
- /*}}}*/
- static bool FullTextSearch(CommandLine &CmdL) /*{{{*/
- {
- pkgCacheFile CacheFile;
- pkgCache *Cache = CacheFile.GetPkgCache();
- pkgDepCache::Policy *Plcy = CacheFile.GetPolicy();
- if (unlikely(Cache == NULL || Plcy == NULL))
- return false;
- // Make sure there is at least one argument
- unsigned int const NumPatterns = CmdL.FileSize() -1;
- if (NumPatterns < 1)
- return _error->Error(_("You must give at least one search pattern"));
- #define APT_FREE_PATTERNS() for (std::vector<regex_t>::iterator P = Patterns.begin(); \
- P != Patterns.end(); ++P) { regfree(&(*P)); }
- // Compile the regex pattern
- std::vector<regex_t> Patterns;
- for (unsigned int I = 0; I != NumPatterns; ++I)
- {
- regex_t pattern;
- if (regcomp(&pattern, CmdL.FileList[I + 1], REG_EXTENDED | REG_ICASE | REG_NOSUB) != 0)
- {
- APT_FREE_PATTERNS();
- return _error->Error("Regex compilation error");
- }
- Patterns.push_back(pattern);
- }
- std::map<std::string, std::string> output_map;
- LocalitySortedVersionSet bag;
- OpTextProgress progress(*_config);
- progress.OverallProgress(0, 100, 50, _("Sorting"));
- GetLocalitySortedVersionSet(CacheFile, &bag, &progress);
- LocalitySortedVersionSet::iterator V = bag.begin();
- progress.OverallProgress(50, 100, 50, _("Full Text Search"));
- progress.SubProgress(bag.size());
- pkgRecords records(CacheFile);
- std::string format = "${color:highlight}${Package}${color:neutral}/${Origin} ${Version} ${Architecture}${ }${apt:Status}\n";
- if (_config->FindB("APT::Cache::ShowFull",false) == false)
- format += " ${Description}\n";
- else
- format += " ${LongDescription}\n";
- bool const NamesOnly = _config->FindB("APT::Cache::NamesOnly", false);
- int Done = 0;
- std::vector<bool> PkgsDone(Cache->Head().PackageCount, false);
- for ( ;V != bag.end(); ++V)
- {
- if (Done%500 == 0)
- progress.Progress(Done);
- ++Done;
- // we want to list each package only once
- pkgCache::PkgIterator const P = V.ParentPkg();
- if (PkgsDone[P->ID] == true)
- continue;
- char const * const PkgName = P.Name();
- pkgCache::DescIterator Desc = V.TranslatedDescription();
- std::string LongDesc = "";
- if (Desc.end() == false)
- {
- pkgRecords::Parser &parser = records.Lookup(Desc.FileList());
- LongDesc = parser.LongDesc();
- }
- bool all_found = true;
- for (std::vector<regex_t>::const_iterator pattern = Patterns.begin();
- pattern != Patterns.end(); ++pattern)
- {
- if (regexec(&(*pattern), PkgName, 0, 0, 0) == 0)
- continue;
- else if (NamesOnly == false && regexec(&(*pattern), LongDesc.c_str(), 0, 0, 0) == 0)
- continue;
- // search patterns are AND, so one failing fails all
- all_found = false;
- break;
- }
- if (all_found == true)
- {
- PkgsDone[P->ID] = true;
- std::stringstream outs;
- ListSingleVersion(CacheFile, records, V, outs, format);
- output_map.insert(std::make_pair<std::string, std::string>(
- PkgName, outs.str()));
- }
- }
- APT_FREE_PATTERNS();
- progress.Done();
- // FIXME: SORT! and make sorting flexible (alphabetic, by pkg status)
- // output the sorted map
- std::map<std::string, std::string>::const_iterator K;
- for (K = output_map.begin(); K != output_map.end(); ++K)
- std::cout << (*K).second << std::endl;
- return true;
- }
- /*}}}*/
- // LocalitySort - Sort a version list by package file locality /*{{{*/
- static int LocalityCompare(const void * const a, const void * const b)
- {
- pkgCache::VerFile const * const A = *(pkgCache::VerFile const * const * const)a;
- pkgCache::VerFile const * const B = *(pkgCache::VerFile const * const * const)b;
- if (A == 0 && B == 0)
- return 0;
- if (A == 0)
- return 1;
- if (B == 0)
- return -1;
- if (A->File == B->File)
- return A->Offset - B->Offset;
- return A->File - B->File;
- }
- void LocalitySort(pkgCache::VerFile ** const begin, unsigned long long const Count,size_t const Size)
- {
- qsort(begin,Count,Size,LocalityCompare);
- }
- static void LocalitySort(pkgCache::DescFile ** const begin, unsigned long long const Count,size_t const Size)
- {
- qsort(begin,Count,Size,LocalityCompare);
- }
- /*}}}*/
- // Search - Perform a search /*{{{*/
- // ---------------------------------------------------------------------
- /* This searches the package names and package descriptions for a pattern */
- struct ExDescFile
- {
- pkgCache::DescFile *Df;
- pkgCache::VerIterator V;
- map_id_t ID;
- };
- static bool Search(CommandLine &CmdL)
- {
- bool const ShowFull = _config->FindB("APT::Cache::ShowFull",false);
- unsigned int const NumPatterns = CmdL.FileSize() -1;
-
- pkgCacheFile CacheFile;
- pkgCache *Cache = CacheFile.GetPkgCache();
- pkgDepCache::Policy *Plcy = CacheFile.GetPolicy();
- if (unlikely(Cache == NULL || Plcy == NULL))
- return false;
- // Make sure there is at least one argument
- if (NumPatterns < 1)
- return _error->Error(_("You must give at least one search pattern"));
-
- // Compile the regex pattern
- regex_t *Patterns = new regex_t[NumPatterns];
- memset(Patterns,0,sizeof(*Patterns)*NumPatterns);
- for (unsigned I = 0; I != NumPatterns; I++)
- {
- if (regcomp(&Patterns[I],CmdL.FileList[I+1],REG_EXTENDED | REG_ICASE |
- REG_NOSUB) != 0)
- {
- for (; I != 0; I--)
- regfree(&Patterns[I]);
- return _error->Error("Regex compilation error");
- }
- }
-
- if (_error->PendingError() == true)
- {
- for (unsigned I = 0; I != NumPatterns; I++)
- regfree(&Patterns[I]);
- return false;
- }
-
- size_t const descCount = Cache->HeaderP->GroupCount + 1;
- ExDescFile *DFList = new ExDescFile[descCount];
- memset(DFList,0,sizeof(*DFList) * descCount);
- bool *PatternMatch = new bool[descCount * NumPatterns];
- memset(PatternMatch,false,sizeof(*PatternMatch) * descCount * NumPatterns);
- // Map versions that we want to write out onto the VerList array.
- bool const NamesOnly = _config->FindB("APT::Cache::NamesOnly",false);
- for (pkgCache::GrpIterator G = Cache->GrpBegin(); G.end() == false; ++G)
- {
- size_t const PatternOffset = G->ID * NumPatterns;
- size_t unmatched = 0, matched = 0;
- for (unsigned I = 0; I < NumPatterns; ++I)
- {
- if (PatternMatch[PatternOffset + I] == true)
- ++matched;
- else if (regexec(&Patterns[I],G.Name(),0,0,0) == 0)
- PatternMatch[PatternOffset + I] = true;
- else
- ++unmatched;
- }
- // already dealt with this package?
- if (matched == NumPatterns)
- continue;
- // Doing names only, drop any that don't match..
- if (NamesOnly == true && unmatched == NumPatterns)
- continue;
- // Find the proper version to use
- pkgCache::PkgIterator P = G.FindPreferredPkg();
- if (P.end() == true)
- continue;
- pkgCache::VerIterator V = Plcy->GetCandidateVer(P);
- if (V.end() == false)
- {
- pkgCache::DescIterator const D = V.TranslatedDescription();
- //FIXME: packages without a description can't be found
- if (D.end() == true)
- continue;
- DFList[G->ID].Df = D.FileList();
- DFList[G->ID].V = V;
- DFList[G->ID].ID = G->ID;
- }
- if (unmatched == NumPatterns)
- continue;
- // Include all the packages that provide matching names too
- for (pkgCache::PrvIterator Prv = P.ProvidesList() ; Prv.end() == false; ++Prv)
- {
- pkgCache::VerIterator V = Plcy->GetCandidateVer(Prv.OwnerPkg());
- if (V.end() == true)
- continue;
- unsigned long id = Prv.OwnerPkg().Group()->ID;
- pkgCache::DescIterator const D = V.TranslatedDescription();
- //FIXME: packages without a description can't be found
- if (D.end() == true)
- continue;
- DFList[id].Df = D.FileList();
- DFList[id].V = V;
- DFList[id].ID = id;
- size_t const PrvPatternOffset = id * NumPatterns;
- for (unsigned I = 0; I < NumPatterns; ++I)
- PatternMatch[PrvPatternOffset + I] |= PatternMatch[PatternOffset + I];
- }
- }
- LocalitySort(&DFList->Df, Cache->HeaderP->GroupCount, sizeof(*DFList));
- // Create the text record parser
- pkgRecords Recs(*Cache);
- // Iterate over all the version records and check them
- for (ExDescFile *J = DFList; J->Df != 0; ++J)
- {
- pkgRecords::Parser &P = Recs.Lookup(pkgCache::DescFileIterator(*Cache,J->Df));
- size_t const PatternOffset = J->ID * NumPatterns;
- if (NamesOnly == false)
- {
- std::string const LongDesc = P.LongDesc();
- for (unsigned I = 0; I < NumPatterns; ++I)
- {
- if (PatternMatch[PatternOffset + I] == true)
- continue;
- else if (regexec(&Patterns[I],LongDesc.c_str(),0,0,0) == 0)
- PatternMatch[PatternOffset + I] = true;
- }
- }
- bool matchedAll = true;
- for (unsigned I = 0; I < NumPatterns; ++I)
- if (PatternMatch[PatternOffset + I] == false)
- {
- matchedAll = false;
- break;
- }
- if (matchedAll == true)
- {
- if (ShowFull == true)
- DisplayRecordV1(CacheFile, J->V, std::cout);
- else
- printf("%s - %s\n",P.Name().c_str(),P.ShortDesc().c_str());
- }
- }
-
- delete [] DFList;
- delete [] PatternMatch;
- for (unsigned I = 0; I != NumPatterns; I++)
- regfree(&Patterns[I]);
- delete [] Patterns;
- if (ferror(stdout))
- return _error->Error("Write to stdout failed");
- return true;
- }
- /*}}}*/
- bool DoSearch(CommandLine &CmdL) /*{{{*/
- {
- int const ShowVersion = _config->FindI("APT::Cache::Search::Version", 1);
- if (ShowVersion <= 1)
- return Search(CmdL);
- return FullTextSearch(CmdL);
- }
|