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
);
141 APT_CASE(CREATED_BY
);
142 APT_CASE(FALLBACK_OF
);
144 APT_CASE(DEFAULTENABLED
);
145 APT_CASE(COMPRESSIONTYPES
);
146 APT_CASE(SOURCESENTRY
);
148 APT_CASE(KEEPCOMPRESSEDAS
);
150 case FILENAME
: return _config
->FindDir("Dir::State::lists") + URItoFileName(URI
);
151 case EXISTING_FILENAME
:
152 std::string
const filename
= Option(FILENAME
);
153 std::vector
<std::string
> const types
= VectorizeString(Option(COMPRESSIONTYPES
), ' ');
154 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
158 std::string
const file
= (*t
== "uncompressed") ? filename
: (filename
+ "." + *t
);
159 if (FileExists(file
))
164 std::map
<std::string
,std::string
>::const_iterator
const M
= Options
.find(Key
);
165 if (M
== Options
.end())
170 bool IndexTarget::OptionBool(OptionKeys
const EnumKey
) const /*{{{*/
172 return StringToBool(Option(EnumKey
));
175 std::string
IndexTarget::Format(std::string format
) const /*{{{*/
177 for (std::map
<std::string
, std::string
>::const_iterator O
= Options
.begin(); O
!= Options
.end(); ++O
)
179 format
= SubstVar(format
, std::string("$(") + O
->first
+ ")", O
->second
);
181 format
= SubstVar(format
, "$(METAKEY)", MetaKey
);
182 format
= SubstVar(format
, "$(SHORTDESC)", ShortDesc
);
183 format
= SubstVar(format
, "$(DESCRIPTION)", Description
);
184 format
= SubstVar(format
, "$(URI)", URI
);
185 format
= SubstVar(format
, "$(FILENAME)", Option(IndexTarget::FILENAME
));
190 pkgDebianIndexTargetFile::pkgDebianIndexTargetFile(IndexTarget
const &Target
, bool const Trusted
) :/*{{{*/
191 pkgDebianIndexFile(Trusted
), d(NULL
), Target(Target
)
195 std::string
pkgDebianIndexTargetFile::ArchiveURI(std::string
const &File
) const/*{{{*/
197 return Target
.Option(IndexTarget::REPO_URI
) + File
;
200 std::string
pkgDebianIndexTargetFile::Describe(bool const Short
) const /*{{{*/
203 return Target
.Description
;
204 return Target
.Description
+ " (" + IndexFileName() + ")";
207 std::string
pkgDebianIndexTargetFile::IndexFileName() const /*{{{*/
209 std::string
const s
= Target
.Option(IndexTarget::FILENAME
);
213 std::vector
<std::string
> const types
= VectorizeString(Target
.Option(IndexTarget::COMPRESSIONTYPES
), ' ');
214 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
216 std::string p
= s
+ '.' + *t
;
223 unsigned long pkgDebianIndexTargetFile::Size() const /*{{{*/
225 unsigned long size
= 0;
227 /* we need to ignore errors here; if the lists are absent, just return 0 */
228 _error
->PushToStack();
230 FileFd
f(IndexFileName(), FileFd::ReadOnly
, FileFd::Extension
);
234 if (_error
->PendingError() == true)
236 _error
->RevertToStack();
241 bool pkgDebianIndexTargetFile::Exists() const /*{{{*/
243 return FileExists(IndexFileName());
246 std::string
pkgDebianIndexTargetFile::GetArchitecture() const /*{{{*/
248 return Target
.Option(IndexTarget::ARCHITECTURE
);
251 std::string
pkgDebianIndexTargetFile::GetComponent() const /*{{{*/
253 return Target
.Option(IndexTarget::COMPONENT
);
256 bool pkgDebianIndexTargetFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
258 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
259 return _error
->Error("Problem opening %s",FileName
.c_str());
263 std::string
pkgDebianIndexTargetFile::GetProgressDescription() const
265 return Target
.Description
;
268 pkgDebianIndexRealFile::pkgDebianIndexRealFile(std::string
const &pFile
, bool const Trusted
) :/*{{{*/
269 pkgDebianIndexFile(Trusted
), d(NULL
)
273 else if (pFile
== "/nonexistent/stdin")
276 File
= flAbsPath(pFile
);
279 // IndexRealFile::Size - Return the size of the index /*{{{*/
280 unsigned long pkgDebianIndexRealFile::Size() const
283 if (stat(File
.c_str(),&S
) != 0)
288 bool pkgDebianIndexRealFile::Exists() const /*{{{*/
290 return FileExists(File
);
293 std::string
pkgDebianIndexRealFile::Describe(bool const /*Short*/) const/*{{{*/
298 std::string
pkgDebianIndexRealFile::ArchiveURI(std::string
const &/*File*/) const/*{{{*/
300 return "file:" + File
;
303 std::string
pkgDebianIndexRealFile::IndexFileName() const /*{{{*/
308 std::string
pkgDebianIndexRealFile::GetProgressDescription() const
312 bool pkgDebianIndexRealFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
314 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
315 return _error
->Error("Problem opening %s",FileName
.c_str());
320 pkgDebianIndexFile::pkgDebianIndexFile(bool const Trusted
) : pkgIndexFile(Trusted
)
323 pkgDebianIndexFile::~pkgDebianIndexFile()
326 pkgCacheListParser
* pkgDebianIndexFile::CreateListParser(FileFd
&Pkg
)
328 if (Pkg
.IsOpen() == false)
330 _error
->PushToStack();
331 pkgCacheListParser
* const Parser
= new debListParser(&Pkg
);
332 bool const newError
= _error
->PendingError();
333 _error
->MergeWithStack();
342 bool pkgDebianIndexFile::Merge(pkgCacheGenerator
&Gen
,OpProgress
* const Prog
)
344 std::string
const PackageFile
= IndexFileName();
346 if (OpenListFile(Pkg
, PackageFile
) == false)
348 _error
->PushToStack();
349 std::unique_ptr
<pkgCacheListParser
> Parser(CreateListParser(Pkg
));
350 bool const newError
= _error
->PendingError();
351 _error
->MergeWithStack();
352 if (newError
== false && Parser
== nullptr)
358 Prog
->SubProgress(0, GetProgressDescription());
360 if (Gen
.SelectFile(PackageFile
, *this, GetArchitecture(), GetComponent(), GetIndexFlags()) == false)
361 return _error
->Error("Problem with SelectFile %s",PackageFile
.c_str());
363 // Store the IMS information
364 pkgCache::PkgFileIterator File
= Gen
.GetCurFile();
365 pkgCacheGenerator::Dynamic
<pkgCache::PkgFileIterator
> DynFile(File
);
366 File
->Size
= Pkg
.FileSize();
367 File
->mtime
= Pkg
.ModificationTime();
369 if (Gen
.MergeList(*Parser
) == false)
370 return _error
->Error("Problem with MergeList %s",PackageFile
.c_str());
373 pkgCache::PkgFileIterator
pkgDebianIndexFile::FindInCache(pkgCache
&Cache
) const
375 std::string
const FileName
= IndexFileName();
376 pkgCache::PkgFileIterator File
= Cache
.FileBegin();
377 for (; File
.end() == false; ++File
)
379 if (File
.FileName() == NULL
|| FileName
!= File
.FileName())
383 if (stat(File
.FileName(),&St
) != 0)
385 if (_config
->FindB("Debug::pkgCacheGen", false))
386 std::clog
<< "DebianIndexFile::FindInCache - stat failed on " << File
.FileName() << std::endl
;
387 return pkgCache::PkgFileIterator(Cache
);
389 if ((map_filesize_t
)St
.st_size
!= File
->Size
|| St
.st_mtime
!= File
->mtime
)
391 if (_config
->FindB("Debug::pkgCacheGen", false))
392 std::clog
<< "DebianIndexFile::FindInCache - size (" << St
.st_size
<< " <> " << File
->Size
393 << ") or mtime (" << St
.st_mtime
<< " <> " << File
->mtime
394 << ") doesn't match for " << File
.FileName() << std::endl
;
395 return pkgCache::PkgFileIterator(Cache
);
403 APT_CONST
pkgIndexFile::~pkgIndexFile() {}
404 APT_CONST
pkgDebianIndexTargetFile::~pkgDebianIndexTargetFile() {}
405 APT_CONST
pkgDebianIndexRealFile::~pkgDebianIndexRealFile() {}