X-Git-Url: https://git.saurik.com/apt.git/blobdiff_plain/6c139d6e362f04a1582e8a8f511f8aeab031fecf..857e9c13d8d9808fcd1ac8ff3469f6c0b90b7fea:/apt-pkg/sourcelist.cc diff --git a/apt-pkg/sourcelist.cc b/apt-pkg/sourcelist.cc index 62d5e6fcd..6b7a299d6 100644 --- a/apt-pkg/sourcelist.cc +++ b/apt-pkg/sourcelist.cc @@ -1,6 +1,6 @@ // -*- mode: cpp; mode: fold -*- // Description /*{{{*/ -// $Id: sourcelist.cc,v 1.1 1998/07/07 04:17:06 jgg Exp $ +// $Id: sourcelist.cc,v 1.3 2002/08/15 20:51:37 niemeyer Exp $ /* ###################################################################### List of Sources @@ -8,458 +8,371 @@ ##################################################################### */ /*}}}*/ // Include Files /*{{{*/ -#ifdef __GNUG__ -#pragma implementation "pkglib/sourcelist.h" -#endif +#include +#include +#include +#include +#include + +#include -#include -#include -#include -#include -#include +#include -#include +// CNC:2003-03-03 - This is needed for ReadDir stuff. +#include #include -#include +#include #include +#include /*}}}*/ -// SourceList::pkgSourceList - Constructors /*{{{*/ -// --------------------------------------------------------------------- -/* */ -pkgSourceList::pkgSourceList() -{ -} +using namespace std; -pkgSourceList::pkgSourceList(string File) +// Global list of Items supported +static pkgSourceList::Type *ItmList[10]; +pkgSourceList::Type **pkgSourceList::Type::GlobalList = ItmList; +unsigned long pkgSourceList::Type::GlobalListLen = 0; + +// Type::Type - Constructor /*{{{*/ +// --------------------------------------------------------------------- +/* Link this to the global list of items*/ +pkgSourceList::Type::Type() { - Read(File); + ItmList[GlobalListLen] = this; + GlobalListLen++; } /*}}}*/ -// SourceList::ReadMainList - Read the main source list from etc /*{{{*/ +// Type::GetType - Get a specific meta for a given type /*{{{*/ // --------------------------------------------------------------------- /* */ -bool pkgSourceList::ReadMainList() +pkgSourceList::Type *pkgSourceList::Type::GetType(const char *Type) { - return Read(PKG_DEB_CF_SOURCELIST); + for (unsigned I = 0; I != GlobalListLen; I++) + if (strcmp(GlobalList[I]->Name,Type) == 0) + return GlobalList[I]; + return 0; } /*}}}*/ -// SourceList::Read - Parse the sourcelist file /*{{{*/ +// Type::FixupURI - Normalize the URI and check it.. /*{{{*/ // --------------------------------------------------------------------- /* */ -bool pkgSourceList::Read(string File) +bool pkgSourceList::Type::FixupURI(string &URI) const { - // Open the stream for reading - ifstream F(File.c_str(),ios::in | ios::nocreate); - if (!F != 0) - return _error->Errno("ifstream::ifstream","Opening %s",File.c_str()); - - List.erase(List.begin(),List.end()); - char Buffer[300]; + if (URI.empty() == true) + return false; - int CurLine = 0; - while (F.eof() == false) - { - F.getline(Buffer,sizeof(Buffer)); - CurLine++; - _strtabexpand(Buffer,sizeof(Buffer)); - _strstrip(Buffer); - - // Comment or blank - if (Buffer[0] == '#' || Buffer[0] == 0) - continue; - - // Grok it - string Type; - string URI; - Item Itm; - char *C = Buffer; - if (ParseQuoteWord(C,Type) == false) - return _error->Error("Malformed line %u in source list %s (type)",CurLine,File.c_str()); - if (ParseQuoteWord(C,URI) == false) - return _error->Error("Malformed line %u in source list %s (URI)",CurLine,File.c_str()); - if (ParseQuoteWord(C,Itm.Dist) == false) - return _error->Error("Malformed line %u in source list %s (dist)",CurLine,File.c_str()); - if (Itm.SetType(Type) == false) - return _error->Error("Malformed line %u in source list %s (type parse)",CurLine,File.c_str()); - if (Itm.SetURI(URI) == false) - return _error->Error("Malformed line %u in source list %s (URI parse)",CurLine,File.c_str()); - - // Check for an absolute dists specification. - if (Itm.Dist.empty() == false && Itm.Dist[Itm.Dist.size() - 1] == '/') - { - if (ParseQuoteWord(C,Itm.Section) == true) - return _error->Error("Malformed line %u in source list %s (Absolute dist)",CurLine,File.c_str()); - Itm.Dist = SubstVar(Itm.Dist,"$(ARCH)",PKG_DEB_ARCH); - List.push_back(Itm); - continue; - } + if (URI.find(':') == string::npos) + return false; - // Grab the rest of the dists - if (ParseQuoteWord(C,Itm.Section) == false) - return _error->Error("Malformed line %u in source list %s (dist parse)",CurLine,File.c_str()); - - do - { - List.push_back(Itm); - } - while (ParseQuoteWord(C,Itm.Section) == true); - } + URI = SubstVar(URI,"$(ARCH)",_config->Find("APT::Architecture")); + + // Make sure that the URI is / postfixed + if (URI[URI.size() - 1] != '/') + URI += '/'; + return true; } /*}}}*/ -// SourceList::SanitizeURI - Hash the uri /*{{{*/ -// --------------------------------------------------------------------- -/* This converts a URI into a safe filename. It quotes all unsafe characters - and converts / to _ and removes the scheme identifier. */ -string pkgSourceList::SanitizeURI(string URI) -{ - string::const_iterator I = URI.begin() + URI.find(':') + 1; - for (; I < URI.end() && *I == '/'; I++); - - // "\x00-\x20{}|\\\\^\\[\\]<>\"\x7F-\xFF"; - URI = QuoteString(string(I,URI.end() - I),"\\|{}[]<>\"^~_=!@#$%^&*"); - string::iterator J = URI.begin(); - for (; J != URI.end(); J++) - if (*J == '/') - *J = '_'; - return URI; -} - /*}}}*/ -// SourceList::MatchPkgFile - Find the package file that has the ver /*{{{*/ +// Type::ParseLine - Parse a single line /*{{{*/ // --------------------------------------------------------------------- -/* This will return List.end() if it could not find the matching - file */ -pkgSourceList::const_iterator pkgSourceList::MatchPkgFile(pkgCache::VerIterator Ver) +/* This is a generic one that is the 'usual' format for sources.list + Weird types may override this. */ +bool pkgSourceList::Type::ParseLine(vector &List, + const char *Buffer, + unsigned long const &CurLine, + string const &File) const { - string Base = PKG_DEB_ST_LIST; - for (const_iterator I = List.begin(); I != List.end(); I++) + for (;Buffer != 0 && isspace(*Buffer); ++Buffer); // Skip whitespaces + + // Parse option field if it exists + // e.g.: [ option1=value1 option2=value2 ] + map Options; + if (Buffer != 0 && Buffer[0] == '[') { - string URI = I->PackagesURI(); - switch (I->Type) + ++Buffer; // ignore the [ + for (;Buffer != 0 && isspace(*Buffer); ++Buffer); // Skip whitespaces + while (*Buffer != ']') { - case Item::Deb: - if (Base + SanitizeURI(URI) == Ver.File().FileName()) - return I; - break; - }; - } - return List.end(); -} - /*}}}*/ -// SourceList::Item << - Writes the item to a stream /*{{{*/ -// --------------------------------------------------------------------- -/* This is not suitable for rebuilding the sourcelist file but it good for - debugging. */ -ostream &operator <<(ostream &O,pkgSourceList::Item &Itm) -{ - O << Itm.Type << ' ' << Itm.URI << ' ' << Itm.Dist << ' ' << Itm.Section; - return O; -} - /*}}}*/ -// SourceList::Item::SetType - Sets the distribution type /*{{{*/ -// --------------------------------------------------------------------- -/* */ -bool pkgSourceList::Item::SetType(string S) -{ - if (S == "deb") - { - Type = Deb; - return true; - } + // get one option, e.g. option1=value1 + string option; + if (ParseQuoteWord(Buffer,option) == false) + return _error->Error(_("Malformed line %lu in source list %s ([option] unparseable)"),CurLine,File.c_str()); - return true; -} - /*}}}*/ -// SourceList::Item::SetURI - Set the URI /*{{{*/ -// --------------------------------------------------------------------- -/* For simplicity we strip the scheme off the uri */ -bool pkgSourceList::Item::SetURI(string S) -{ - if (S.empty() == true) - return false; + if (option.length() < 3) + return _error->Error(_("Malformed line %lu in source list %s ([option] too short)"),CurLine,File.c_str()); - if (S.find(':') == string::npos) - return false; + size_t const needle = option.find('='); + if (needle == string::npos) + return _error->Error(_("Malformed line %lu in source list %s ([%s] is not an assignment)"),CurLine,File.c_str(), option.c_str()); + + string const key = string(option, 0, needle); + string const value = string(option, needle + 1, option.length()); + + if (key.empty() == true) + return _error->Error(_("Malformed line %lu in source list %s ([%s] has no key)"),CurLine,File.c_str(), option.c_str()); - S = SubstVar(S,"$(ARCH)",PKG_DEB_ARCH); + if (value.empty() == true) + return _error->Error(_("Malformed line %lu in source list %s ([%s] key %s has no value)"),CurLine,File.c_str(),option.c_str(),key.c_str()); + + Options[key] = value; + } + ++Buffer; // ignore the ] + for (;Buffer != 0 && isspace(*Buffer); ++Buffer); // Skip whitespaces + } + + string URI; + string Dist; + string Section; + + if (ParseQuoteWord(Buffer,URI) == false) + return _error->Error(_("Malformed line %lu in source list %s (URI)"),CurLine,File.c_str()); + if (ParseQuoteWord(Buffer,Dist) == false) + return _error->Error(_("Malformed line %lu in source list %s (dist)"),CurLine,File.c_str()); + + if (FixupURI(URI) == false) + return _error->Error(_("Malformed line %lu in source list %s (URI parse)"),CurLine,File.c_str()); - // Make sure that the URN is / postfixed - URI = S; - if (URI[URI.size() - 1] != '/') - URI += '/'; + // Check for an absolute dists specification. + if (Dist.empty() == false && Dist[Dist.size() - 1] == '/') + { + if (ParseQuoteWord(Buffer,Section) == true) + return _error->Error(_("Malformed line %lu in source list %s (absolute dist)"),CurLine,File.c_str()); + Dist = SubstVar(Dist,"$(ARCH)",_config->Find("APT::Architecture")); + return CreateItem(List, URI, Dist, Section, Options); + } + + // Grab the rest of the dists + if (ParseQuoteWord(Buffer,Section) == false) + return _error->Error(_("Malformed line %lu in source list %s (dist parse)"),CurLine,File.c_str()); + + do + { + if (CreateItem(List, URI, Dist, Section, Options) == false) + return false; + } + while (ParseQuoteWord(Buffer,Section) == true); return true; } /*}}}*/ -// SourceList::Item::PackagesURI - Returns a URI to the packages file /*{{{*/ + +// SourceList::pkgSourceList - Constructors /*{{{*/ // --------------------------------------------------------------------- /* */ -string pkgSourceList::Item::PackagesURI() const +pkgSourceList::pkgSourceList() { - string Res; - switch (Type) - { - case Deb: - if (Dist[Dist.size() - 1] == '/') - Res = URI + Dist; - else - Res = URI + "dists/" + Dist + '/' + Section + - "/binary-" + PKG_DEB_ARCH + '/'; - - Res += "Packages"; - break; - }; - return Res; +} + +pkgSourceList::pkgSourceList(string File) +{ + Read(File); } /*}}}*/ -// SourceList::Item::PackagesInfo - Shorter version of the URI /*{{{*/ +// SourceList::~pkgSourceList - Destructor /*{{{*/ // --------------------------------------------------------------------- -/* This is a shorter version that is designed to be < 60 chars or so */ -string pkgSourceList::Item::PackagesInfo() const +/* */ +pkgSourceList::~pkgSourceList() { - string Res; - switch (Type) - { - case Deb: - Res += SiteOnly(URI) + ' '; - if (Dist[Dist.size() - 1] == '/') - Res += Dist; - else - Res += Dist + '/' + Section; - - Res += " Packages"; - break; - }; - return Res; + for (const_iterator I = SrcList.begin(); I != SrcList.end(); I++) + delete *I; } /*}}}*/ -// SourceList::Item::ArchiveInfo - Shorter version of the archive spec /*{{{*/ + /*}}}*/ +// SourceList::ReadMainList - Read the main source list from etc /*{{{*/ // --------------------------------------------------------------------- -/* This is a shorter version that is designed to be < 60 chars or so */ -string pkgSourceList::Item::ArchiveInfo(pkgCache::VerIterator Ver) const +/* */ +bool pkgSourceList::ReadMainList() { - string Res; - switch (Type) - { - case Deb: - Res += SiteOnly(URI) + ' '; - if (Dist[Dist.size() - 1] == '/') - Res += Dist; - else - Res += Dist + '/' + Section; - - Res += " "; - Res += Ver.ParentPkg().Name(); - break; - }; + // CNC:2003-03-03 - Multiple sources list support. + bool Res = true; +#if 0 + Res = ReadVendors(); + if (Res == false) + return false; +#endif + + Reset(); + // CNC:2003-11-28 - Entries in sources.list have priority over + // entries in sources.list.d. + string Main = _config->FindFile("Dir::Etc::sourcelist"); + string Parts = _config->FindDir("Dir::Etc::sourceparts"); + + if (FileExists(Main) == true) + Res &= ReadAppend(Main); + else if (FileExists(Parts) == false) + // Only warn if there are no sources.list.d. + _error->WarningE("FileExists",_("Unable to read %s"),Main.c_str()); + + if (FileExists(Parts) == true) + Res &= ReadSourceDir(Parts); + else if (FileExists(Main) == false) + // Only warn if there is no sources.list file. + _error->WarningE("FileExists",_("Unable to read %s"),Parts.c_str()); + return Res; } /*}}}*/ -// SourceList::Item::ArchiveURI - Returns a URI to the given archive /*{{{*/ +// CNC:2003-03-03 - Needed to preserve backwards compatibility. +// SourceList::Reset - Clear the sourcelist contents /*{{{*/ // --------------------------------------------------------------------- /* */ -string pkgSourceList::Item::ArchiveURI(string File) const +void pkgSourceList::Reset() { - string Res; - switch (Type) - { - case Deb: - Res = URI + File; - break; - }; - return Res; + for (const_iterator I = SrcList.begin(); I != SrcList.end(); I++) + delete *I; + SrcList.erase(SrcList.begin(),SrcList.end()); } /*}}}*/ -// SourceList::Item::SiteOnly - Strip off the path part of a URI /*{{{*/ +// CNC:2003-03-03 - Function moved to ReadAppend() and Reset(). +// SourceList::Read - Parse the sourcelist file /*{{{*/ // --------------------------------------------------------------------- /* */ -string pkgSourceList::Item::SiteOnly(string URI) const +bool pkgSourceList::Read(string File) { - unsigned int Pos = URI.find(':'); - if (Pos == string::npos || Pos + 3 > URI.length()) - return URI; - if (URI[Pos + 1] != '/' || URI[Pos + 2] != '/') - return URI; - - Pos = URI.find('/',Pos + 3); - if (Pos == string::npos) - return URI; - return string(URI,0,Pos); + Reset(); + return ReadAppend(File); } /*}}}*/ - -// UpdateMeta - Update the meta information /*{{{*/ +// SourceList::ReadAppend - Parse a sourcelist file /*{{{*/ // --------------------------------------------------------------------- -/* The meta information is package files, revision information and mirror - lists. */ -bool pkgUpdateMeta(pkgSourceList &List,pkgAquire &Engine) +/* */ +bool pkgSourceList::ReadAppend(string File) { - if (Engine.OutputDir(PKG_DEB_ST_LIST) == false) - return false; + // Open the stream for reading + ifstream F(File.c_str(),ios::in /*| ios::nocreate*/); + if (!F != 0) + return _error->Errno("ifstream::ifstream",_("Opening %s"),File.c_str()); - for (pkgSourceList::const_iterator I = List.begin(); I != List.end(); I++) +#if 0 // Now Reset() does this. + for (const_iterator I = SrcList.begin(); I != SrcList.end(); I++) + delete *I; + SrcList.erase(SrcList.begin(),SrcList.end()); +#endif + // CNC:2003-12-10 - 300 is too short. + char Buffer[1024]; + + int CurLine = 0; + while (F.eof() == false) { - string URI = I->PackagesURI(); - string GetInfo = I->PackagesInfo(); - switch (I->Type) - { - case pkgSourceList::Item::Deb: - if (Engine.Get(URI + ".gz",List.SanitizeURI(URI),GetInfo) == false) - return false; - break; - }; + F.getline(Buffer,sizeof(Buffer)); + CurLine++; + _strtabexpand(Buffer,sizeof(Buffer)); + if (F.fail() && !F.eof()) + return _error->Error(_("Line %u too long in source list %s."), + CurLine,File.c_str()); + + + char *I; + // CNC:2003-02-20 - Do not break if '#' is inside []. + for (I = Buffer; *I != 0 && *I != '#'; I++) + if (*I == '[') + for (I++; *I != 0 && *I != ']'; I++); + *I = 0; + + const char *C = _strstrip(Buffer); + + // Comment or blank + if (C[0] == '#' || C[0] == 0) + continue; + + // Grok it + string LineType; + if (ParseQuoteWord(C,LineType) == false) + return _error->Error(_("Malformed line %u in source list %s (type)"),CurLine,File.c_str()); + + Type *Parse = Type::GetType(LineType.c_str()); + if (Parse == 0) + return _error->Error(_("Type '%s' is not known on line %u in source list %s"),LineType.c_str(),CurLine,File.c_str()); + + if (Parse->ParseLine(SrcList, C, CurLine, File) == false) + return false; } - return true; } /*}}}*/ -// MakeSrcCache - Generate a cache file of all the package files /*{{{*/ +// SourceList::FindIndex - Get the index associated with a file /*{{{*/ // --------------------------------------------------------------------- -/* This goes over the source list and builds a cache of all the package - files. */ -bool pkgMakeSrcCache(pkgSourceList &List) +/* */ +bool pkgSourceList::FindIndex(pkgCache::PkgFileIterator File, + pkgIndexFile *&Found) const { - // First we date check the cache - bool Bad = false; - while (Bad == false) + for (const_iterator I = SrcList.begin(); I != SrcList.end(); I++) { - if (FileExists(PKG_DEB_CA_SRCCACHE) == false) - break; - - pkgCache Cache(PKG_DEB_CA_SRCCACHE,true,true); - if (_error->PendingError() == true) + vector *Indexes = (*I)->GetIndexFiles(); + for (vector::const_iterator J = Indexes->begin(); + J != Indexes->end(); J++) { - _error->Discard(); - break; + if ((*J)->FindInCache(*File.Cache()) == File) + { + Found = (*J); + return true; + } } - - // They are certianly out of sync - if (Cache.Head().PackageFileCount != List.size()) - break; - - for (pkgCache::PkgFileIterator F(Cache); F.end() == false; F++) - { - // Search for a match in the source list - Bad = true; - for (pkgSourceList::const_iterator I = List.begin(); - I != List.end(); I++) - { - string File = string(PKG_DEB_ST_LIST) + - List.SanitizeURI(I->PackagesURI()); - if (F.FileName() == File) - { - Bad = false; - break; - } - } - - // Check if the file matches what was cached - Bad |= !F.IsOk(); - if (Bad == true) - break; - } - - if (Bad == false) - return true; } - - unlink(PKG_DEB_CA_SRCCACHE); - pkgCache::MergeState Merge(PKG_DEB_CA_SRCCACHE); - if (_error->PendingError() == true) - return false; - - for (pkgSourceList::const_iterator I = List.begin(); I != List.end(); I++) - { - string File = string(PKG_DEB_ST_LIST) + List.SanitizeURI(I->PackagesURI()); - if (Merge.MergePackageFile(File,"??","??") == false) + + return false; +} + /*}}}*/ +// SourceList::GetIndexes - Load the index files into the downloader /*{{{*/ +// --------------------------------------------------------------------- +/* */ +bool pkgSourceList::GetIndexes(pkgAcquire *Owner, bool GetAll) const +{ + for (const_iterator I = SrcList.begin(); I != SrcList.end(); I++) + if ((*I)->GetIndexes(Owner,GetAll) == false) return false; - } - return true; } /*}}}*/ -// MakeStatusCache - Generates a cache that includes the status files /*{{{*/ +// CNC:2003-03-03 - By Anton V. Denisov . +// SourceList::ReadSourceDir - Read a directory with sources files +// Based on ReadConfigDir() /*{{{*/ // --------------------------------------------------------------------- -/* This copies the package source cache and then merges the status and - xstatus files into it. */ -bool pkgMakeStatusCache() +/* */ +bool pkgSourceList::ReadSourceDir(string Dir) { - // Quickly check if the existing package cache is ok - bool Bad = false; - while (Bad == false) - { - if (FileExists(PKG_DEB_CA_PKGCACHE) == false) - break; - - /* We check the dates of the two caches. This takes care of most things - quickly and easially */ - struct stat Src; - struct stat Pkg; - if (stat(PKG_DEB_CA_PKGCACHE,&Pkg) != 0 || - stat(PKG_DEB_CA_SRCCACHE,&Src) != 0) - break; - if (difftime(Src.st_mtime,Pkg.st_mtime) > 0) - break; - - pkgCache Cache(PKG_DEB_CA_PKGCACHE,true,true); - if (_error->PendingError() == true) - { - _error->Discard(); - break; - } - - for (pkgCache::PkgFileIterator F(Cache); F.end() == false; F++) - { - if (F.IsOk() == false) - { - Bad = true; - break; - } - } - - if (Bad == false) - return true; - } + DIR *D = opendir(Dir.c_str()); + if (D == 0) + return _error->Errno("opendir",_("Unable to read %s"),Dir.c_str()); - // Check the integrity of the source cache. - { - pkgCache Cache(PKG_DEB_CA_SRCCACHE,true,true); - if (_error->PendingError() == true) - return false; - } + vector List; - // Sub scope so that merge destructs before we rename the file... - string Cache = PKG_DEB_CA_PKGCACHE ".new"; + for (struct dirent *Ent = readdir(D); Ent != 0; Ent = readdir(D)) { - if (CopyFile(PKG_DEB_CA_SRCCACHE,Cache) == false) - return false; + if (Ent->d_name[0] == '.') + continue; - pkgCache::MergeState Merge(Cache); - if (_error->PendingError() == true) - return false; + // CNC:2003-12-02 Only accept .list files as valid sourceparts + if (flExtension(Ent->d_name) != "list") + continue; - // Merge in the user status file - if (FileExists(PKG_DEB_ST_USERSTATUS) == true) - if (Merge.MergePackageFile(PKG_DEB_ST_USERSTATUS,"status","0", - pkgFLAG_NotSource) == false) - return false; + // Skip bad file names ala run-parts + const char *C = Ent->d_name; + for (; *C != 0; C++) + if (isalpha(*C) == 0 && isdigit(*C) == 0 + && *C != '_' && *C != '-' && *C != '.') + break; + if (*C != 0) + continue; - // Merge in the extra status file - if (FileExists(PKG_DEB_ST_XSTATUS) == true) - if (Merge.MergePackageFile(PKG_DEB_ST_XSTATUS,"status","0", - pkgFLAG_NotSource) == false) - return false; + // Make sure it is a file and not something else + string File = flCombine(Dir,Ent->d_name); + struct stat St; + if (stat(File.c_str(),&St) != 0 || S_ISREG(St.st_mode) == 0) + continue; - // Merge in the status file - if (Merge.MergePackageFile("/var/lib/dpkg/status","status","0", - pkgFLAG_NotSource) == false) - return false; - } - - if (rename(Cache.c_str(),PKG_DEB_CA_PKGCACHE) != 0) - return false; + List.push_back(File); + } + closedir(D); + sort(List.begin(),List.end()); + + // Read the files + for (vector::const_iterator I = List.begin(); I != List.end(); I++) + if (ReadAppend(*I) == false) + return false; return true; + } /*}}}*/ +