1 // -*- mode: cpp; mode: fold -*-
3 // $Id: indexfile.cc,v 1.2.2.1 2003/12/24 23:09:17 mdz Exp $
4 /* ######################################################################
6 Index File - Abstraction for an index of archive/souce file.
8 ##################################################################### */
10 // Include Files /*{{{*/
13 #include <apt-pkg/configuration.h>
14 #include <apt-pkg/indexfile.h>
15 #include <apt-pkg/error.h>
16 #include <apt-pkg/fileutl.h>
17 #include <apt-pkg/aptconfiguration.h>
18 #include <apt-pkg/pkgcache.h>
19 #include <apt-pkg/pkgcachegen.h>
20 #include <apt-pkg/cacheiterators.h>
21 #include <apt-pkg/srcrecords.h>
22 #include <apt-pkg/strutl.h>
23 #include <apt-pkg/progress.h>
24 #include <apt-pkg/deblistparser.h>
25 #include <apt-pkg/macros.h>
27 #include <apt-pkg/debindexfile.h>
38 // Global list of Item supported
39 static pkgIndexFile::Type
*ItmList
[10];
40 pkgIndexFile::Type
**pkgIndexFile::Type::GlobalList
= ItmList
;
41 unsigned long pkgIndexFile::Type::GlobalListLen
= 0;
43 // Type::Type - Constructor /*{{{*/
44 // ---------------------------------------------------------------------
46 pkgIndexFile::Type::Type()
48 ItmList
[GlobalListLen
] = this;
53 // Type::GetType - Locate the type by name /*{{{*/
54 // ---------------------------------------------------------------------
56 pkgIndexFile::Type
*pkgIndexFile::Type::GetType(const char *Type
)
58 for (unsigned I
= 0; I
!= GlobalListLen
; I
++)
59 if (strcmp(GlobalList
[I
]->Label
,Type
) == 0)
64 pkgIndexFile::pkgIndexFile(bool const Trusted
) : /*{{{*/
65 d(NULL
), Trusted(Trusted
)
69 // IndexFile::ArchiveInfo - Stub /*{{{*/
70 std::string
pkgIndexFile::ArchiveInfo(pkgCache::VerIterator
const &Ver
) const
72 debDebPkgFileIndex
const * const debfile
= dynamic_cast<debDebPkgFileIndex
const*>(this);
73 if (debfile
!= nullptr)
74 return debfile
->ArchiveInfo_impl(Ver
);
78 // IndexFile::FindInCache - Stub /*{{{*/
79 pkgCache::PkgFileIterator
pkgIndexFile::FindInCache(pkgCache
&Cache
) const
81 return pkgCache::PkgFileIterator(Cache
);
84 // IndexFile::SourceIndex - Stub /*{{{*/
85 std::string
pkgIndexFile::SourceInfo(pkgSrcRecords::Parser
const &/*Record*/,
86 pkgSrcRecords::File
const &/*File*/) const
91 // IndexFile::TranslationsAvailable - Check if will use Translation /*{{{*/
92 bool pkgIndexFile::TranslationsAvailable() {
93 return (APT::Configuration::getLanguages().empty() != true);
96 // IndexFile::CheckLanguageCode - Check the Language Code /*{{{*/
97 bool pkgIndexFile::CheckLanguageCode(const char * const Lang
)
99 if (strlen(Lang
) == 2 || (strlen(Lang
) == 5 && Lang
[2] == '_'))
102 if (strcmp(Lang
,"C") != 0)
103 _error
->Warning("Wrong language code %s", Lang
);
108 // IndexFile::LanguageCode - Return the Language Code /*{{{*/
109 std::string
pkgIndexFile::LanguageCode() {
110 APT_IGNORE_DEPRECATED_PUSH
111 if (TranslationsAvailable() == false)
113 return APT::Configuration::getLanguages()[0];
114 APT_IGNORE_DEPRECATED_POP
118 // IndexTarget - Constructor /*{{{*/
119 IndexTarget::IndexTarget(std::string
const &MetaKey
, std::string
const &ShortDesc
,
120 std::string
const &LongDesc
, std::string
const &URI
, bool const IsOptional
,
121 bool const KeepCompressed
, std::map
<std::string
, std::string
> const &Options
) :
122 URI(URI
), Description(LongDesc
), ShortDesc(ShortDesc
), MetaKey(MetaKey
),
123 IsOptional(IsOptional
), KeepCompressed(KeepCompressed
), Options(Options
)
127 std::string
IndexTarget::Option(OptionKeys
const EnumKey
) const /*{{{*/
132 #define APT_CASE(X) case X: Key = #X; break
137 APT_CASE(ARCHITECTURE
);
140 APT_CASE(IDENTIFIER
);
142 APT_CASE(CREATED_BY
);
143 APT_CASE(FALLBACK_OF
);
145 APT_CASE(DEFAULTENABLED
);
146 APT_CASE(COMPRESSIONTYPES
);
147 APT_CASE(SOURCESENTRY
);
149 APT_CASE(KEEPCOMPRESSEDAS
);
151 case FILENAME
: return _config
->FindDir("Dir::State::lists") + URItoFileName(URI
);
152 case EXISTING_FILENAME
:
153 std::string
const filename
= Option(FILENAME
);
154 std::vector
<std::string
> const types
= VectorizeString(Option(COMPRESSIONTYPES
), ' ');
155 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
159 std::string
const file
= (*t
== "uncompressed") ? filename
: (filename
+ "." + *t
);
160 if (FileExists(file
))
165 std::map
<std::string
,std::string
>::const_iterator
const M
= Options
.find(Key
);
166 if (M
== Options
.end())
171 bool IndexTarget::OptionBool(OptionKeys
const EnumKey
) const /*{{{*/
173 return StringToBool(Option(EnumKey
));
176 std::string
IndexTarget::Format(std::string format
) const /*{{{*/
178 for (std::map
<std::string
, std::string
>::const_iterator O
= Options
.begin(); O
!= Options
.end(); ++O
)
180 format
= SubstVar(format
, std::string("$(") + O
->first
+ ")", O
->second
);
182 format
= SubstVar(format
, "$(METAKEY)", MetaKey
);
183 format
= SubstVar(format
, "$(SHORTDESC)", ShortDesc
);
184 format
= SubstVar(format
, "$(DESCRIPTION)", Description
);
185 format
= SubstVar(format
, "$(URI)", URI
);
186 format
= SubstVar(format
, "$(FILENAME)", Option(IndexTarget::FILENAME
));
191 pkgDebianIndexTargetFile::pkgDebianIndexTargetFile(IndexTarget
const &Target
, bool const Trusted
) :/*{{{*/
192 pkgDebianIndexFile(Trusted
), d(NULL
), Target(Target
)
196 std::string
pkgDebianIndexTargetFile::ArchiveURI(std::string
const &File
) const/*{{{*/
198 return Target
.Option(IndexTarget::REPO_URI
) + File
;
201 std::string
pkgDebianIndexTargetFile::Describe(bool const Short
) const /*{{{*/
204 return Target
.Description
;
205 return Target
.Description
+ " (" + IndexFileName() + ")";
208 std::string
pkgDebianIndexTargetFile::IndexFileName() const /*{{{*/
210 std::string
const s
= Target
.Option(IndexTarget::FILENAME
);
214 std::vector
<std::string
> const types
= VectorizeString(Target
.Option(IndexTarget::COMPRESSIONTYPES
), ' ');
215 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
217 std::string p
= s
+ '.' + *t
;
224 unsigned long pkgDebianIndexTargetFile::Size() const /*{{{*/
226 unsigned long size
= 0;
228 /* we need to ignore errors here; if the lists are absent, just return 0 */
229 _error
->PushToStack();
231 FileFd
f(IndexFileName(), FileFd::ReadOnly
, FileFd::Extension
);
235 if (_error
->PendingError() == true)
237 _error
->RevertToStack();
242 bool pkgDebianIndexTargetFile::Exists() const /*{{{*/
244 return FileExists(IndexFileName());
247 std::string
pkgDebianIndexTargetFile::GetArchitecture() const /*{{{*/
249 return Target
.Option(IndexTarget::ARCHITECTURE
);
252 std::string
pkgDebianIndexTargetFile::GetComponent() const /*{{{*/
254 return Target
.Option(IndexTarget::COMPONENT
);
257 bool pkgDebianIndexTargetFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
259 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
260 return _error
->Error("Problem opening %s",FileName
.c_str());
264 std::string
pkgDebianIndexTargetFile::GetProgressDescription() const
266 return Target
.Description
;
269 pkgDebianIndexRealFile::pkgDebianIndexRealFile(std::string
const &pFile
, bool const Trusted
) :/*{{{*/
270 pkgDebianIndexFile(Trusted
), d(NULL
)
274 else if (pFile
== "/nonexistent/stdin")
277 File
= flAbsPath(pFile
);
280 // IndexRealFile::Size - Return the size of the index /*{{{*/
281 unsigned long pkgDebianIndexRealFile::Size() const
284 if (stat(File
.c_str(),&S
) != 0)
289 bool pkgDebianIndexRealFile::Exists() const /*{{{*/
291 return FileExists(File
);
294 std::string
pkgDebianIndexRealFile::Describe(bool const /*Short*/) const/*{{{*/
299 std::string
pkgDebianIndexRealFile::ArchiveURI(std::string
const &/*File*/) const/*{{{*/
301 return "file:" + File
;
304 std::string
pkgDebianIndexRealFile::IndexFileName() const /*{{{*/
309 std::string
pkgDebianIndexRealFile::GetProgressDescription() const
313 bool pkgDebianIndexRealFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
315 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
316 return _error
->Error("Problem opening %s",FileName
.c_str());
321 pkgDebianIndexFile::pkgDebianIndexFile(bool const Trusted
) : pkgIndexFile(Trusted
)
324 pkgDebianIndexFile::~pkgDebianIndexFile()
327 pkgCacheListParser
* pkgDebianIndexFile::CreateListParser(FileFd
&Pkg
)
329 if (Pkg
.IsOpen() == false)
331 _error
->PushToStack();
332 pkgCacheListParser
* const Parser
= new debListParser(&Pkg
);
333 bool const newError
= _error
->PendingError();
334 _error
->MergeWithStack();
343 bool pkgDebianIndexFile::Merge(pkgCacheGenerator
&Gen
,OpProgress
* const Prog
)
345 std::string
const PackageFile
= IndexFileName();
347 if (OpenListFile(Pkg
, PackageFile
) == false)
349 _error
->PushToStack();
350 std::unique_ptr
<pkgCacheListParser
> Parser(CreateListParser(Pkg
));
351 bool const newError
= _error
->PendingError();
352 _error
->MergeWithStack();
353 if (newError
== false && Parser
== nullptr)
359 Prog
->SubProgress(0, GetProgressDescription());
361 if (Gen
.SelectFile(PackageFile
, *this, GetArchitecture(), GetComponent(), GetIndexFlags()) == false)
362 return _error
->Error("Problem with SelectFile %s",PackageFile
.c_str());
364 // Store the IMS information
365 pkgCache::PkgFileIterator File
= Gen
.GetCurFile();
366 pkgCacheGenerator::Dynamic
<pkgCache::PkgFileIterator
> DynFile(File
);
367 File
->Size
= Pkg
.FileSize();
368 File
->mtime
= Pkg
.ModificationTime();
370 if (Gen
.MergeList(*Parser
) == false)
371 return _error
->Error("Problem with MergeList %s",PackageFile
.c_str());
374 pkgCache::PkgFileIterator
pkgDebianIndexFile::FindInCache(pkgCache
&Cache
) const
376 std::string
const FileName
= IndexFileName();
377 pkgCache::PkgFileIterator File
= Cache
.FileBegin();
378 for (; File
.end() == false; ++File
)
380 if (File
.FileName() == NULL
|| FileName
!= File
.FileName())
384 if (stat(File
.FileName(),&St
) != 0)
386 if (_config
->FindB("Debug::pkgCacheGen", false))
387 std::clog
<< "DebianIndexFile::FindInCache - stat failed on " << File
.FileName() << std::endl
;
388 return pkgCache::PkgFileIterator(Cache
);
390 if ((map_filesize_t
)St
.st_size
!= File
->Size
|| St
.st_mtime
!= File
->mtime
)
392 if (_config
->FindB("Debug::pkgCacheGen", false))
393 std::clog
<< "DebianIndexFile::FindInCache - size (" << St
.st_size
<< " <> " << File
->Size
394 << ") or mtime (" << St
.st_mtime
<< " <> " << File
->mtime
395 << ") doesn't match for " << File
.FileName() << std::endl
;
396 return pkgCache::PkgFileIterator(Cache
);
404 APT_CONST
pkgIndexFile::~pkgIndexFile() {}
405 APT_CONST
pkgDebianIndexTargetFile::~pkgDebianIndexTargetFile() {}
406 APT_CONST
pkgDebianIndexRealFile::~pkgDebianIndexRealFile() {}