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
);
143 APT_CASE(DEFAULTENABLED
);
144 APT_CASE(COMPRESSIONTYPES
);
145 APT_CASE(SOURCESENTRY
);
147 APT_CASE(KEEPCOMPRESSEDAS
);
149 case FILENAME
: return _config
->FindDir("Dir::State::lists") + URItoFileName(URI
);
150 case EXISTING_FILENAME
:
151 std::string
const filename
= Option(FILENAME
);
152 std::vector
<std::string
> const types
= VectorizeString(Option(COMPRESSIONTYPES
), ' ');
153 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
157 std::string
const file
= (*t
== "uncompressed") ? filename
: (filename
+ "." + *t
);
158 if (FileExists(file
))
163 std::map
<std::string
,std::string
>::const_iterator
const M
= Options
.find(Key
);
164 if (M
== Options
.end())
169 bool IndexTarget::OptionBool(OptionKeys
const EnumKey
) const /*{{{*/
171 return StringToBool(Option(EnumKey
));
174 std::string
IndexTarget::Format(std::string format
) const /*{{{*/
176 for (std::map
<std::string
, std::string
>::const_iterator O
= Options
.begin(); O
!= Options
.end(); ++O
)
178 format
= SubstVar(format
, std::string("$(") + O
->first
+ ")", O
->second
);
180 format
= SubstVar(format
, "$(METAKEY)", MetaKey
);
181 format
= SubstVar(format
, "$(SHORTDESC)", ShortDesc
);
182 format
= SubstVar(format
, "$(DESCRIPTION)", Description
);
183 format
= SubstVar(format
, "$(URI)", URI
);
184 format
= SubstVar(format
, "$(FILENAME)", Option(IndexTarget::FILENAME
));
189 pkgDebianIndexTargetFile::pkgDebianIndexTargetFile(IndexTarget
const &Target
, bool const Trusted
) :/*{{{*/
190 pkgDebianIndexFile(Trusted
), d(NULL
), Target(Target
)
194 std::string
pkgDebianIndexTargetFile::ArchiveURI(std::string
const &File
) const/*{{{*/
196 return Target
.Option(IndexTarget::REPO_URI
) + File
;
199 std::string
pkgDebianIndexTargetFile::Describe(bool const Short
) const /*{{{*/
202 return Target
.Description
;
203 return Target
.Description
+ " (" + IndexFileName() + ")";
206 std::string
pkgDebianIndexTargetFile::IndexFileName() const /*{{{*/
208 std::string
const s
= Target
.Option(IndexTarget::FILENAME
);
212 std::vector
<std::string
> const types
= VectorizeString(Target
.Option(IndexTarget::COMPRESSIONTYPES
), ' ');
213 for (std::vector
<std::string
>::const_iterator t
= types
.begin(); t
!= types
.end(); ++t
)
215 std::string p
= s
+ '.' + *t
;
222 unsigned long pkgDebianIndexTargetFile::Size() const /*{{{*/
224 unsigned long size
= 0;
226 /* we need to ignore errors here; if the lists are absent, just return 0 */
227 _error
->PushToStack();
229 FileFd
f(IndexFileName(), FileFd::ReadOnly
, FileFd::Extension
);
233 if (_error
->PendingError() == true)
235 _error
->RevertToStack();
240 bool pkgDebianIndexTargetFile::Exists() const /*{{{*/
242 return FileExists(IndexFileName());
245 std::string
pkgDebianIndexTargetFile::GetArchitecture() const /*{{{*/
247 return Target
.Option(IndexTarget::ARCHITECTURE
);
250 std::string
pkgDebianIndexTargetFile::GetComponent() const /*{{{*/
252 return Target
.Option(IndexTarget::COMPONENT
);
255 bool pkgDebianIndexTargetFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
257 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::Extension
) == false)
258 return _error
->Error("Problem opening %s",FileName
.c_str());
262 std::string
pkgDebianIndexTargetFile::GetProgressDescription() const
264 return Target
.Description
;
267 pkgDebianIndexRealFile::pkgDebianIndexRealFile(std::string
const &pFile
, bool const Trusted
) :/*{{{*/
268 pkgDebianIndexFile(Trusted
), d(NULL
)
270 if (pFile
== "/nonexistent/stdin")
273 File
= flAbsPath(pFile
);
276 // IndexRealFile::Size - Return the size of the index /*{{{*/
277 unsigned long pkgDebianIndexRealFile::Size() const
280 if (stat(File
.c_str(),&S
) != 0)
285 bool pkgDebianIndexRealFile::Exists() const /*{{{*/
287 return FileExists(File
);
290 std::string
pkgDebianIndexRealFile::Describe(bool const /*Short*/) const/*{{{*/
295 std::string
pkgDebianIndexRealFile::ArchiveURI(std::string
const &/*File*/) const/*{{{*/
297 return "file:" + File
;
300 std::string
pkgDebianIndexRealFile::IndexFileName() const /*{{{*/
305 std::string
pkgDebianIndexRealFile::GetProgressDescription() const
309 bool pkgDebianIndexRealFile::OpenListFile(FileFd
&Pkg
, std::string
const &FileName
)/*{{{*/
311 if (Pkg
.Open(FileName
, FileFd::ReadOnly
, FileFd::None
) == false)
312 return _error
->Error("Problem opening %s",FileName
.c_str());
317 pkgDebianIndexFile::pkgDebianIndexFile(bool const Trusted
) : pkgIndexFile(Trusted
)
320 pkgDebianIndexFile::~pkgDebianIndexFile()
323 pkgCacheListParser
* pkgDebianIndexFile::CreateListParser(FileFd
&Pkg
)
325 if (Pkg
.IsOpen() == false)
327 _error
->PushToStack();
328 pkgCacheListParser
* const Parser
= new debListParser(&Pkg
);
329 bool const newError
= _error
->PendingError();
330 _error
->MergeWithStack();
331 return newError
? NULL
: Parser
;
333 bool pkgDebianIndexFile::Merge(pkgCacheGenerator
&Gen
,OpProgress
* const Prog
)
335 std::string
const PackageFile
= IndexFileName();
337 if (OpenListFile(Pkg
, PackageFile
) == false)
339 _error
->PushToStack();
340 std::unique_ptr
<pkgCacheListParser
> Parser(CreateListParser(Pkg
));
341 bool const newError
= _error
->PendingError();
342 _error
->MergeWithStack();
343 if (newError
== false && Parser
== nullptr)
349 Prog
->SubProgress(0, GetProgressDescription());
351 if (Gen
.SelectFile(PackageFile
, *this, GetArchitecture(), GetComponent(), GetIndexFlags()) == false)
352 return _error
->Error("Problem with SelectFile %s",PackageFile
.c_str());
354 // Store the IMS information
355 pkgCache::PkgFileIterator File
= Gen
.GetCurFile();
356 pkgCacheGenerator::Dynamic
<pkgCache::PkgFileIterator
> DynFile(File
);
357 File
->Size
= Pkg
.FileSize();
358 File
->mtime
= Pkg
.ModificationTime();
360 if (Gen
.MergeList(*Parser
) == false)
361 return _error
->Error("Problem with MergeList %s",PackageFile
.c_str());
364 pkgCache::PkgFileIterator
pkgDebianIndexFile::FindInCache(pkgCache
&Cache
) const
366 std::string
const FileName
= IndexFileName();
367 pkgCache::PkgFileIterator File
= Cache
.FileBegin();
368 for (; File
.end() == false; ++File
)
370 if (File
.FileName() == NULL
|| FileName
!= File
.FileName())
374 if (stat(File
.FileName(),&St
) != 0)
376 if (_config
->FindB("Debug::pkgCacheGen", false))
377 std::clog
<< "DebianIndexFile::FindInCache - stat failed on " << File
.FileName() << std::endl
;
378 return pkgCache::PkgFileIterator(Cache
);
380 if ((map_filesize_t
)St
.st_size
!= File
->Size
|| St
.st_mtime
!= File
->mtime
)
382 if (_config
->FindB("Debug::pkgCacheGen", false))
383 std::clog
<< "DebianIndexFile::FindInCache - size (" << St
.st_size
<< " <> " << File
->Size
384 << ") or mtime (" << St
.st_mtime
<< " <> " << File
->mtime
385 << ") doesn't match for " << File
.FileName() << std::endl
;
386 return pkgCache::PkgFileIterator(Cache
);
394 APT_CONST
pkgIndexFile::~pkgIndexFile() {}
395 APT_CONST
pkgDebianIndexTargetFile::~pkgDebianIndexTargetFile() {}
396 APT_CONST
pkgDebianIndexRealFile::~pkgDebianIndexRealFile() {}