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
);
150 APT_CASE(ALLOW_INSECURE
);
151 APT_CASE(ALLOW_WEAK
);
152 APT_CASE(ALLOW_DOWNGRADE_TO_INSECURE
);
154 case FILENAME
: return _config
->FindDir("Dir::State::lists") + URItoFileName(URI
);
155 case EXISTING_FILENAME
:
156 std::string
const filename
= Option(FILENAME
);
157 std::vector
<std::string
> const types
= VectorizeString(Option(COMPRESSIONTYPES
), ' ');
158 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
162 std::string
const file
= (*t
== "uncompressed") ? filename
: (filename
+ "." + *t
);
163 if (FileExists(file
))
168 std::map
<std::string
,std::string
>::const_iterator
const M
= Options
.find(Key
);
169 if (M
== Options
.end())
174 bool IndexTarget::OptionBool(OptionKeys
const EnumKey
) const /*{{{*/
176 return StringToBool(Option(EnumKey
), false);
179 std::string
IndexTarget::Format(std::string format
) const /*{{{*/
181 for (std::map
<std::string
, std::string
>::const_iterator O
= Options
.begin(); O
!= Options
.end(); ++O
)
183 format
= SubstVar(format
, std::string("$(") + O
->first
+ ")", O
->second
);
185 format
= SubstVar(format
, "$(METAKEY)", MetaKey
);
186 format
= SubstVar(format
, "$(SHORTDESC)", ShortDesc
);
187 format
= SubstVar(format
, "$(DESCRIPTION)", Description
);
188 format
= SubstVar(format
, "$(URI)", URI
);
189 format
= SubstVar(format
, "$(FILENAME)", Option(IndexTarget::FILENAME
));
194 pkgDebianIndexTargetFile::pkgDebianIndexTargetFile(IndexTarget
const &Target
, bool const Trusted
) :/*{{{*/
195 pkgDebianIndexFile(Trusted
), d(NULL
), Target(Target
)
199 std::string
pkgDebianIndexTargetFile::ArchiveURI(std::string
const &File
) const/*{{{*/
201 return Target
.Option(IndexTarget::REPO_URI
) + File
;
204 std::string
pkgDebianIndexTargetFile::Describe(bool const Short
) const /*{{{*/
207 return Target
.Description
;
208 return Target
.Description
+ " (" + IndexFileName() + ")";
211 std::string
pkgDebianIndexTargetFile::IndexFileName() const /*{{{*/
213 std::string
const s
= Target
.Option(IndexTarget::FILENAME
);
217 std::vector
<std::string
> const types
= VectorizeString(Target
.Option(IndexTarget::COMPRESSIONTYPES
), ' ');
218 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
220 std::string p
= s
+ '.' + *t
;
227 unsigned long pkgDebianIndexTargetFile::Size() const /*{{{*/
229 unsigned long size
= 0;
231 /* we need to ignore errors here; if the lists are absent, just return 0 */
232 _error
->PushToStack();
234 FileFd
f(IndexFileName(), FileFd::ReadOnly
, FileFd::Extension
);
238 if (_error
->PendingError() == true)
240 _error
->RevertToStack();
245 bool pkgDebianIndexTargetFile::Exists() const /*{{{*/
247 return FileExists(IndexFileName());
250 std::string
pkgDebianIndexTargetFile::GetArchitecture() const /*{{{*/
252 return Target
.Option(IndexTarget::ARCHITECTURE
);
255 std::string
pkgDebianIndexTargetFile::GetComponent() const /*{{{*/
257 return Target
.Option(IndexTarget::COMPONENT
);
260 bool pkgDebianIndexTargetFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
262 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
263 return _error
->Error("Problem opening %s",FileName
.c_str());
267 std::string
pkgDebianIndexTargetFile::GetProgressDescription() const
269 return Target
.Description
;
272 pkgDebianIndexRealFile::pkgDebianIndexRealFile(std::string
const &pFile
, bool const Trusted
) :/*{{{*/
273 pkgDebianIndexFile(Trusted
), d(NULL
)
277 else if (pFile
== "/nonexistent/stdin")
280 File
= flAbsPath(pFile
);
283 // IndexRealFile::Size - Return the size of the index /*{{{*/
284 unsigned long pkgDebianIndexRealFile::Size() const
287 if (stat(File
.c_str(),&S
) != 0)
292 bool pkgDebianIndexRealFile::Exists() const /*{{{*/
294 return FileExists(File
);
297 std::string
pkgDebianIndexRealFile::Describe(bool const /*Short*/) const/*{{{*/
302 std::string
pkgDebianIndexRealFile::ArchiveURI(std::string
const &/*File*/) const/*{{{*/
304 return "file:" + File
;
307 std::string
pkgDebianIndexRealFile::IndexFileName() const /*{{{*/
312 std::string
pkgDebianIndexRealFile::GetProgressDescription() const
316 bool pkgDebianIndexRealFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
318 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
319 return _error
->Error("Problem opening %s",FileName
.c_str());
324 pkgDebianIndexFile::pkgDebianIndexFile(bool const Trusted
) : pkgIndexFile(Trusted
)
327 pkgDebianIndexFile::~pkgDebianIndexFile()
330 pkgCacheListParser
* pkgDebianIndexFile::CreateListParser(FileFd
&Pkg
)
332 if (Pkg
.IsOpen() == false)
334 _error
->PushToStack();
335 pkgCacheListParser
* const Parser
= new debListParser(&Pkg
);
336 bool const newError
= _error
->PendingError();
337 _error
->MergeWithStack();
346 bool pkgDebianIndexFile::Merge(pkgCacheGenerator
&Gen
,OpProgress
* const Prog
)
348 std::string
const PackageFile
= IndexFileName();
350 if (OpenListFile(Pkg
, PackageFile
) == false)
352 _error
->PushToStack();
353 std::unique_ptr
<pkgCacheListParser
> Parser(CreateListParser(Pkg
));
354 bool const newError
= _error
->PendingError();
355 _error
->MergeWithStack();
356 if (newError
== false && Parser
== nullptr)
362 Prog
->SubProgress(0, GetProgressDescription());
364 if (Gen
.SelectFile(PackageFile
, *this, GetArchitecture(), GetComponent(), GetIndexFlags()) == false)
365 return _error
->Error("Problem with SelectFile %s",PackageFile
.c_str());
367 // Store the IMS information
368 pkgCache::PkgFileIterator File
= Gen
.GetCurFile();
369 pkgCacheGenerator::Dynamic
<pkgCache::PkgFileIterator
> DynFile(File
);
370 File
->Size
= Pkg
.FileSize();
371 File
->mtime
= Pkg
.ModificationTime();
373 if (Gen
.MergeList(*Parser
) == false)
374 return _error
->Error("Problem with MergeList %s",PackageFile
.c_str());
377 pkgCache::PkgFileIterator
pkgDebianIndexFile::FindInCache(pkgCache
&Cache
) const
379 std::string
const FileName
= IndexFileName();
380 pkgCache::PkgFileIterator File
= Cache
.FileBegin();
381 for (; File
.end() == false; ++File
)
383 if (File
.FileName() == NULL
|| FileName
!= File
.FileName())
387 if (stat(File
.FileName(),&St
) != 0)
389 if (_config
->FindB("Debug::pkgCacheGen", false))
390 std::clog
<< "DebianIndexFile::FindInCache - stat failed on " << File
.FileName() << std::endl
;
391 return pkgCache::PkgFileIterator(Cache
);
393 if ((map_filesize_t
)St
.st_size
!= File
->Size
|| St
.st_mtime
!= File
->mtime
)
395 if (_config
->FindB("Debug::pkgCacheGen", false))
396 std::clog
<< "DebianIndexFile::FindInCache - size (" << St
.st_size
<< " <> " << File
->Size
397 << ") or mtime (" << St
.st_mtime
<< " <> " << File
->mtime
398 << ") doesn't match for " << File
.FileName() << std::endl
;
399 return pkgCache::PkgFileIterator(Cache
);
407 APT_CONST
pkgIndexFile::~pkgIndexFile() {}
408 APT_CONST
pkgDebianIndexTargetFile::~pkgDebianIndexTargetFile() {}
409 APT_CONST
pkgDebianIndexRealFile::~pkgDebianIndexRealFile() {}