]> git.saurik.com Git - apt.git/blame - apt-pkg/indexfile.cc
rred: If there were I/O errors, fail
[apt.git] / apt-pkg / indexfile.cc
CommitLineData
b2e465d6
AL
1// -*- mode: cpp; mode: fold -*-
2// Description /*{{{*/
7db98ffc 3// $Id: indexfile.cc,v 1.2.2.1 2003/12/24 23:09:17 mdz Exp $
b2e465d6
AL
4/* ######################################################################
5
6 Index File - Abstraction for an index of archive/souce file.
7
8 ##################################################################### */
9 /*}}}*/
10// Include Files /*{{{*/
ea542140
DK
11#include<config.h>
12
e3c1cfc7 13#include <apt-pkg/configuration.h>
b2e465d6
AL
14#include <apt-pkg/indexfile.h>
15#include <apt-pkg/error.h>
e3c1cfc7 16#include <apt-pkg/fileutl.h>
45df0ad2 17#include <apt-pkg/aptconfiguration.h>
453b82a3 18#include <apt-pkg/pkgcache.h>
c9443c01 19#include <apt-pkg/pkgcachegen.h>
453b82a3
DK
20#include <apt-pkg/cacheiterators.h>
21#include <apt-pkg/srcrecords.h>
d7a51997 22#include <apt-pkg/strutl.h>
c9443c01 23#include <apt-pkg/progress.h>
64b66a46 24#include <apt-pkg/deblistparser.h>
453b82a3 25#include <apt-pkg/macros.h>
a52f938b 26
64b66a46 27#include <apt-pkg/debindexfile.h>
c9443c01 28
c9443c01 29#include <sys/stat.h>
c9443c01 30
453b82a3
DK
31#include <string>
32#include <vector>
a52f938b 33#include <clocale>
4f333a8b 34#include <cstring>
c9443c01 35#include <memory>
b2e465d6
AL
36 /*}}}*/
37
38// Global list of Item supported
39static pkgIndexFile::Type *ItmList[10];
40pkgIndexFile::Type **pkgIndexFile::Type::GlobalList = ItmList;
41unsigned long pkgIndexFile::Type::GlobalListLen = 0;
42
43// Type::Type - Constructor /*{{{*/
44// ---------------------------------------------------------------------
45/* */
46pkgIndexFile::Type::Type()
47{
48 ItmList[GlobalListLen] = this;
dd688285
DK
49 GlobalListLen++;
50 Label = NULL;
b2e465d6
AL
51}
52 /*}}}*/
53// Type::GetType - Locate the type by name /*{{{*/
54// ---------------------------------------------------------------------
55/* */
56pkgIndexFile::Type *pkgIndexFile::Type::GetType(const char *Type)
57{
58 for (unsigned I = 0; I != GlobalListLen; I++)
59 if (strcmp(GlobalList[I]->Label,Type) == 0)
60 return GlobalList[I];
61 return 0;
e3c1cfc7
DK
62}
63 /*}}}*/
c9443c01 64pkgIndexFile::pkgIndexFile(bool const Trusted) : /*{{{*/
6c55f07a 65 d(NULL), Trusted(Trusted)
e3c1cfc7 66{
b2e465d6
AL
67}
68 /*}}}*/
b2e465d6 69// IndexFile::ArchiveInfo - Stub /*{{{*/
64b66a46 70std::string pkgIndexFile::ArchiveInfo(pkgCache::VerIterator const &Ver) const
b2e465d6 71{
64b66a46
DK
72 debDebPkgFileIndex const * const debfile = dynamic_cast<debDebPkgFileIndex const*>(this);
73 if (debfile != nullptr)
74 return debfile->ArchiveInfo_impl(Ver);
8f3ba4e8 75 return std::string();
b2e465d6
AL
76}
77 /*}}}*/
78// IndexFile::FindInCache - Stub /*{{{*/
b2e465d6
AL
79pkgCache::PkgFileIterator pkgIndexFile::FindInCache(pkgCache &Cache) const
80{
81 return pkgCache::PkgFileIterator(Cache);
82}
83 /*}}}*/
84// IndexFile::SourceIndex - Stub /*{{{*/
65512241
DK
85std::string pkgIndexFile::SourceInfo(pkgSrcRecords::Parser const &/*Record*/,
86 pkgSrcRecords::File const &/*File*/) const
b2e465d6 87{
8f3ba4e8 88 return std::string();
b2e465d6
AL
89}
90 /*}}}*/
45df0ad2 91// IndexFile::TranslationsAvailable - Check if will use Translation /*{{{*/
45df0ad2
DK
92bool pkgIndexFile::TranslationsAvailable() {
93 return (APT::Configuration::getLanguages().empty() != true);
a52f938b
OS
94}
95 /*}}}*/
45df0ad2 96// IndexFile::CheckLanguageCode - Check the Language Code /*{{{*/
5dd00edb 97bool pkgIndexFile::CheckLanguageCode(const char * const Lang)
a52f938b
OS
98{
99 if (strlen(Lang) == 2 || (strlen(Lang) == 5 && Lang[2] == '_'))
100 return true;
101
102 if (strcmp(Lang,"C") != 0)
103 _error->Warning("Wrong language code %s", Lang);
104
105 return false;
106}
107 /*}}}*/
45df0ad2 108// IndexFile::LanguageCode - Return the Language Code /*{{{*/
5dd00edb 109std::string pkgIndexFile::LanguageCode() {
668ae33a 110APT_IGNORE_DEPRECATED_PUSH
45df0ad2
DK
111 if (TranslationsAvailable() == false)
112 return "";
113 return APT::Configuration::getLanguages()[0];
668ae33a 114APT_IGNORE_DEPRECATED_POP
a52f938b
OS
115}
116 /*}}}*/
e3c1cfc7
DK
117
118// IndexTarget - Constructor /*{{{*/
119IndexTarget::IndexTarget(std::string const &MetaKey, std::string const &ShortDesc,
120 std::string const &LongDesc, std::string const &URI, bool const IsOptional,
653ef26c
DK
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)
e3c1cfc7
DK
124{
125}
126 /*}}}*/
001c76fe 127std::string IndexTarget::Option(OptionKeys const EnumKey) const /*{{{*/
e3c1cfc7 128{
001c76fe
DK
129 std::string Key;
130 switch (EnumKey)
131 {
132#define APT_CASE(X) case X: Key = #X; break
133 APT_CASE(SITE);
134 APT_CASE(RELEASE);
135 APT_CASE(COMPONENT);
136 APT_CASE(LANGUAGE);
137 APT_CASE(ARCHITECTURE);
138 APT_CASE(BASE_URI);
139 APT_CASE(REPO_URI);
8881b11e 140 APT_CASE(TARGET_OF);
001c76fe 141 APT_CASE(CREATED_BY);
1a3a14ac 142 APT_CASE(PDIFFS);
9adb9778 143 APT_CASE(DEFAULTENABLED);
d7a51997 144 APT_CASE(COMPRESSIONTYPES);
3090ae69 145 APT_CASE(SOURCESENTRY);
24e8f24e 146 APT_CASE(BY_HASH);
0179cfa8 147 APT_CASE(KEEPCOMPRESSEDAS);
001c76fe 148#undef APT_CASE
8881b11e 149 case FILENAME: return _config->FindDir("Dir::State::lists") + URItoFileName(URI);
3fd89e62
DK
150 case EXISTING_FILENAME:
151 std::string const filename = Option(FILENAME);
d7a51997 152 std::vector<std::string> const types = VectorizeString(Option(COMPRESSIONTYPES), ' ');
3fd89e62
DK
153 for (std::vector<std::string>::const_iterator t = types.begin(); t != types.end(); ++t)
154 {
155 if (t->empty())
156 continue;
157 std::string const file = (*t == "uncompressed") ? filename : (filename + "." + *t);
158 if (FileExists(file))
159 return file;
160 }
161 return "";
001c76fe 162 }
e3c1cfc7
DK
163 std::map<std::string,std::string>::const_iterator const M = Options.find(Key);
164 if (M == Options.end())
165 return "";
166 return M->second;
167}
168 /*}}}*/
1a3a14ac
DK
169bool IndexTarget::OptionBool(OptionKeys const EnumKey) const /*{{{*/
170{
171 return StringToBool(Option(EnumKey));
172}
173 /*}}}*/
8881b11e
DK
174std::string IndexTarget::Format(std::string format) const /*{{{*/
175{
176 for (std::map<std::string, std::string>::const_iterator O = Options.begin(); O != Options.end(); ++O)
177 {
178 format = SubstVar(format, std::string("$(") + O->first + ")", O->second);
179 }
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));
185 return format;
186}
187 /*}}}*/
e3c1cfc7 188
c9443c01
DK
189pkgDebianIndexTargetFile::pkgDebianIndexTargetFile(IndexTarget const &Target, bool const Trusted) :/*{{{*/
190 pkgDebianIndexFile(Trusted), d(NULL), Target(Target)
e3c1cfc7
DK
191{
192}
193 /*}}}*/
c9443c01 194std::string pkgDebianIndexTargetFile::ArchiveURI(std::string const &File) const/*{{{*/
e3c1cfc7 195{
001c76fe 196 return Target.Option(IndexTarget::REPO_URI) + File;
e3c1cfc7
DK
197}
198 /*}}}*/
c9443c01 199std::string pkgDebianIndexTargetFile::Describe(bool const Short) const /*{{{*/
e3c1cfc7
DK
200{
201 if (Short)
202 return Target.Description;
203 return Target.Description + " (" + IndexFileName() + ")";
204}
205 /*}}}*/
c9443c01 206std::string pkgDebianIndexTargetFile::IndexFileName() const /*{{{*/
e3c1cfc7 207{
8881b11e 208 std::string const s = Target.Option(IndexTarget::FILENAME);
e3c1cfc7
DK
209 if (FileExists(s))
210 return s;
211
d7a51997 212 std::vector<std::string> const types = VectorizeString(Target.Option(IndexTarget::COMPRESSIONTYPES), ' ');
e3c1cfc7
DK
213 for (std::vector<std::string>::const_iterator t = types.begin(); t != types.end(); ++t)
214 {
215 std::string p = s + '.' + *t;
216 if (FileExists(p))
217 return p;
218 }
219 return s;
220}
221 /*}}}*/
c9443c01 222unsigned long pkgDebianIndexTargetFile::Size() const /*{{{*/
e3c1cfc7
DK
223{
224 unsigned long size = 0;
225
226 /* we need to ignore errors here; if the lists are absent, just return 0 */
227 _error->PushToStack();
228
229 FileFd f(IndexFileName(), FileFd::ReadOnly, FileFd::Extension);
230 if (!f.Failed())
231 size = f.Size();
232
233 if (_error->PendingError() == true)
234 size = 0;
235 _error->RevertToStack();
236
237 return size;
238}
239 /*}}}*/
c9443c01 240bool pkgDebianIndexTargetFile::Exists() const /*{{{*/
e3c1cfc7
DK
241{
242 return FileExists(IndexFileName());
243}
244 /*}}}*/
c9443c01
DK
245std::string pkgDebianIndexTargetFile::GetArchitecture() const /*{{{*/
246{
247 return Target.Option(IndexTarget::ARCHITECTURE);
248}
249 /*}}}*/
250std::string pkgDebianIndexTargetFile::GetComponent() const /*{{{*/
251{
252 return Target.Option(IndexTarget::COMPONENT);
253}
254 /*}}}*/
255bool pkgDebianIndexTargetFile::OpenListFile(FileFd &Pkg, std::string const &FileName)/*{{{*/
256{
257 if (Pkg.Open(FileName, FileFd::ReadOnly, FileFd::Extension) == false)
258 return _error->Error("Problem opening %s",FileName.c_str());
259 return true;
260}
261 /*}}}*/
262std::string pkgDebianIndexTargetFile::GetProgressDescription() const
263{
264 return Target.Description;
265}
266
7f58427b
DK
267pkgDebianIndexRealFile::pkgDebianIndexRealFile(std::string const &pFile, bool const Trusted) :/*{{{*/
268 pkgDebianIndexFile(Trusted), d(NULL)
c9443c01 269{
a249b3e6
DK
270 if (pFile.empty())
271 ;
272 else if (pFile == "/nonexistent/stdin")
7f58427b
DK
273 File = pFile;
274 else
275 File = flAbsPath(pFile);
c9443c01
DK
276}
277 /*}}}*/
278// IndexRealFile::Size - Return the size of the index /*{{{*/
279unsigned long pkgDebianIndexRealFile::Size() const
280{
281 struct stat S;
282 if (stat(File.c_str(),&S) != 0)
283 return 0;
284 return S.st_size;
285}
286 /*}}}*/
287bool pkgDebianIndexRealFile::Exists() const /*{{{*/
288{
289 return FileExists(File);
290}
291 /*}}}*/
292std::string pkgDebianIndexRealFile::Describe(bool const /*Short*/) const/*{{{*/
293{
294 return File;
295}
296 /*}}}*/
297std::string pkgDebianIndexRealFile::ArchiveURI(std::string const &/*File*/) const/*{{{*/
298{
299 return "file:" + File;
300}
301 /*}}}*/
302std::string pkgDebianIndexRealFile::IndexFileName() const /*{{{*/
303{
304 return File;
305}
306 /*}}}*/
307std::string pkgDebianIndexRealFile::GetProgressDescription() const
308{
309 return File;
310}
311bool pkgDebianIndexRealFile::OpenListFile(FileFd &Pkg, std::string const &FileName)/*{{{*/
312{
313 if (Pkg.Open(FileName, FileFd::ReadOnly, FileFd::None) == false)
314 return _error->Error("Problem opening %s",FileName.c_str());
315 return true;
316}
317 /*}}}*/
318
319pkgDebianIndexFile::pkgDebianIndexFile(bool const Trusted) : pkgIndexFile(Trusted)
320{
321}
322pkgDebianIndexFile::~pkgDebianIndexFile()
323{
324}
325pkgCacheListParser * pkgDebianIndexFile::CreateListParser(FileFd &Pkg)
326{
327 if (Pkg.IsOpen() == false)
328 return NULL;
329 _error->PushToStack();
7f8c0eed 330 pkgCacheListParser * const Parser = new debListParser(&Pkg);
c9443c01
DK
331 bool const newError = _error->PendingError();
332 _error->MergeWithStack();
333 return newError ? NULL : Parser;
334}
335bool pkgDebianIndexFile::Merge(pkgCacheGenerator &Gen,OpProgress * const Prog)
336{
337 std::string const PackageFile = IndexFileName();
338 FileFd Pkg;
339 if (OpenListFile(Pkg, PackageFile) == false)
340 return false;
341 _error->PushToStack();
342 std::unique_ptr<pkgCacheListParser> Parser(CreateListParser(Pkg));
343 bool const newError = _error->PendingError();
95278287 344 _error->MergeWithStack();
c9443c01
DK
345 if (newError == false && Parser == nullptr)
346 return true;
347 if (Parser == NULL)
348 return false;
349
350 if (Prog != NULL)
351 Prog->SubProgress(0, GetProgressDescription());
352
353 if (Gen.SelectFile(PackageFile, *this, GetArchitecture(), GetComponent(), GetIndexFlags()) == false)
354 return _error->Error("Problem with SelectFile %s",PackageFile.c_str());
355
356 // Store the IMS information
357 pkgCache::PkgFileIterator File = Gen.GetCurFile();
358 pkgCacheGenerator::Dynamic<pkgCache::PkgFileIterator> DynFile(File);
359 File->Size = Pkg.FileSize();
360 File->mtime = Pkg.ModificationTime();
361
362 if (Gen.MergeList(*Parser) == false)
363 return _error->Error("Problem with MergeList %s",PackageFile.c_str());
364 return true;
365}
366pkgCache::PkgFileIterator pkgDebianIndexFile::FindInCache(pkgCache &Cache) const
367{
368 std::string const FileName = IndexFileName();
369 pkgCache::PkgFileIterator File = Cache.FileBegin();
370 for (; File.end() == false; ++File)
371 {
372 if (File.FileName() == NULL || FileName != File.FileName())
373 continue;
374
375 struct stat St;
376 if (stat(File.FileName(),&St) != 0)
377 {
378 if (_config->FindB("Debug::pkgCacheGen", false))
379 std::clog << "DebianIndexFile::FindInCache - stat failed on " << File.FileName() << std::endl;
380 return pkgCache::PkgFileIterator(Cache);
381 }
382 if ((map_filesize_t)St.st_size != File->Size || St.st_mtime != File->mtime)
383 {
384 if (_config->FindB("Debug::pkgCacheGen", false))
385 std::clog << "DebianIndexFile::FindInCache - size (" << St.st_size << " <> " << File->Size
386 << ") or mtime (" << St.st_mtime << " <> " << File->mtime
387 << ") doesn't match for " << File.FileName() << std::endl;
388 return pkgCache::PkgFileIterator(Cache);
389 }
390 return File;
391 }
392
393 return File;
394}
c8a4ce6c
DK
395
396APT_CONST pkgIndexFile::~pkgIndexFile() {}
c9443c01
DK
397APT_CONST pkgDebianIndexTargetFile::~pkgDebianIndexTargetFile() {}
398APT_CONST pkgDebianIndexRealFile::~pkgDebianIndexRealFile() {}