- treat zero epochs from repomd as no epoch
[apt.git] / apt-pkg / rpm / rpmindexfile.cc
1 // -*- mode: cpp; mode: fold -*-
2 // Description                                                          /*{{{*/
3 // $Id: rpmindexfile.cc,v 1.4 2002/11/27 16:22:40 niemeyer Exp $
4 /* ######################################################################
5
6    RPM Specific sources.list types and the three sorts of RPM
7    index files.
8    
9    ##################################################################### */
10                                                                         /*}}}*/
11 // Include Files                                                        /*{{{*/
12 #ifdef __GNUG__
13 #pragma implementation "apt-pkg/rpmindexfile.h"
14 #endif
15
16 #include <config.h>
17
18 #ifdef HAVE_RPM
19
20 #include <apt-pkg/rpmindexfile.h>
21 #include <apt-pkg/rpmsrcrecords.h>
22 #include <apt-pkg/rpmlistparser.h>
23 #include <apt-pkg/rpmrecords.h>
24 #include <apt-pkg/rpmsystem.h>
25 #include <apt-pkg/rpmhandler.h>
26 #include <apt-pkg/rpmpackagedata.h>
27 #include <apt-pkg/sourcelist.h>
28 #include <apt-pkg/configuration.h>
29 #include <apt-pkg/progress.h>
30 #include <apt-pkg/error.h>
31 #include <apt-pkg/strutl.h>
32 #include <apt-pkg/acquire-item.h>
33 #include <apt-pkg/repomd.h>
34
35 #include <apti18n.h>
36
37 #include <sys/stat.h>
38                                                                         /*}}}*/
39 vector<pkgRepository *> RepList;
40
41 // rpmListIndex::Release* - Return the URI to the release file          /*{{{*/
42 // ---------------------------------------------------------------------
43 /* */
44 inline string rpmListIndex::ReleaseFile(string Type) const
45 {
46    return URItoFileName(ReleaseURI(Type));
47 }
48
49 string rpmListIndex::ReleaseURI(string Type) const
50 {
51    RPMPackageData *rpmdata = RPMPackageData::Singleton();
52    string Res;
53    if (Dist[Dist.size() - 1] == '/')
54    {
55       if (Dist != "/")
56          Res = URI + Dist;
57       else
58          Res = URI;
59    }
60    else
61       Res = URI + Dist + "/base/";
62
63    Res += Type;
64    
65    if (rpmdata->HasIndexTranslation() == true)
66    {
67       map<string,string> Dict;
68       Dict["uri"] = URI;
69       Dict["dist"] = Dist;
70       Dict["sect"] = "";
71       Dict["type"] = Type;
72       rpmdata->TranslateIndex(Res, Dict);
73    }
74
75    return Res;
76 }
77                                                                         /*}}}*/
78 // rpmListIndex::ReleaseInfo - One liner describing the index URI       /*{{{*/
79 // ---------------------------------------------------------------------
80 /* */
81 string rpmListIndex::ReleaseInfo(string Type) const 
82 {
83    string Info = ::URI::SiteOnly(URI) + ' ';
84    if (Dist[Dist.size() - 1] == '/')
85    {
86       if (Dist != "/")
87          Info += Dist;
88    }
89    else
90       Info += Dist;   
91    Info += " ";
92    Info += Type;
93    return Info;
94 }
95                                                                         /*}}}*/
96 // rpmListIndex::GetReleases - Fetch the index files                    /*{{{*/
97 // ---------------------------------------------------------------------
98 /* */
99 bool rpmListIndex::GetReleases(pkgAcquire *Owner) const
100 {
101    if (!Repository->Acquire)
102       return true;
103    Repository->Acquire = false;
104    new pkgAcqIndexRel(Owner,Repository,ReleaseURI("release"),
105                       ReleaseInfo("release"), "release", true);
106    return true;
107 }
108                                                                         /*}}}*/
109 // rpmListIndex::Info - One liner describing the index URI              /*{{{*/
110 // ---------------------------------------------------------------------
111 /* */
112 string rpmListIndex::Info(string Type) const 
113 {
114    string Info = ::URI::SiteOnly(URI) + ' ';
115    if (Dist[Dist.size() - 1] == '/')
116    {
117       if (Dist != "/")
118          Info += Dist;
119    }
120    else
121       Info += Dist + '/' + Section;   
122    Info += " ";
123    Info += Type;
124    return Info;
125 }
126                                                                         /*}}}*/
127 // rpmListIndex::Index* - Return the URI to the index files             /*{{{*/
128 // ---------------------------------------------------------------------
129 /* */
130 inline string rpmListIndex::IndexFile(string Type) const
131 {
132    return _config->FindDir("Dir::State::lists")+URItoFileName(IndexURI(Type));
133 }
134
135
136 string rpmListIndex::IndexURI(string Type) const
137 {
138    RPMPackageData *rpmdata = RPMPackageData::Singleton();
139    string Res;
140    if (Dist[Dist.size() - 1] == '/')
141    {
142       if (Dist != "/")
143          Res = URI + Dist;
144       else 
145          Res = URI;
146    }
147    else
148       Res = URI + Dist + "/base/";
149    
150    Res += Type + '.' + Section;
151
152    if (rpmdata->HasIndexTranslation() == true)
153    {
154       map<string,string> Dict;
155       Dict["uri"] = URI;
156       Dict["dist"] = Dist; 
157       Dict["sect"] = Section;
158       Dict["type"] = Type;
159       rpmdata->TranslateIndex(Res, Dict);
160    }
161
162    return Res;
163 }
164                                                                         /*}}}*/
165 // rpmListIndex::Exists - Check if the index is available               /*{{{*/
166 // ---------------------------------------------------------------------
167 /* */
168 bool rpmListIndex::Exists() const
169 {
170    return FileExists(IndexPath());
171 }
172                                                                         /*}}}*/
173 // rpmListIndex::Size - Return the size of the index                    /*{{{*/
174 // ---------------------------------------------------------------------
175 /* */
176 unsigned long rpmListIndex::Size() const
177 {
178    struct stat S;
179    if (stat(IndexPath().c_str(),&S) != 0)
180       return 0;
181    return S.st_size;
182 }
183                                                                         /*}}}*/
184 // rpmListIndex::Describe - Give a descriptive path to the index        /*{{{*/
185 // ---------------------------------------------------------------------
186 /* */
187 string rpmListIndex::Describe(bool Short) const
188 {
189    char S[300];
190    if (Short == true)
191       snprintf(S,sizeof(S),"%s",Info(MainType()).c_str());
192    else
193       snprintf(S,sizeof(S),"%s (%s)",Info(MainType()).c_str(),
194          IndexFile(MainType()).c_str());
195    return S;
196 }
197                                                                         /*}}}*/
198
199 // SrcListIndex::SourceInfo - Short 1 liner describing a source         /*{{{*/
200 // ---------------------------------------------------------------------
201 string rpmSrcListIndex::SourceInfo(pkgSrcRecords::Parser const &Record,
202                                    pkgSrcRecords::File const &File) const
203 {
204    string Res;
205    Res = ::URI::SiteOnly(URI) + ' ';
206    if (Dist[Dist.size() - 1] == '/')
207    {
208       if (Dist != "/")
209          Res += Dist;
210    }      
211    else
212       Res += Dist + '/' + Section;
213    
214    Res += " ";
215    Res += Record.Package();
216    Res += " ";
217    Res += Record.Version();
218    if (File.Type.empty() == false)
219       Res += " (" + File.Type + ")";
220    return Res;
221 }
222                                                                         /*}}}*/
223 // SrcListIndex::ArchiveURI - URI for the archive                       /*{{{*/
224 // ---------------------------------------------------------------------
225 string rpmSrcListIndex::ArchiveURI(string File) const
226 {
227    RPMPackageData *rpmdata = RPMPackageData::Singleton();
228    string Res;
229    
230    if (Dist[Dist.size() - 1] == '/')
231    {
232       if (Dist != "/")
233          Res = URI + Dist;
234       else
235          Res = URI;
236    }
237    else
238       Res = URI + Dist;
239    
240    if (File.find("/") != string::npos)
241       Res += '/' + File;
242    else
243       Res += "/SRPMS."+Section + '/' + File;
244
245    if (rpmdata->HasSourceTranslation() == true)
246    {
247       map<string,string> Dict;
248       Dict["uri"] = URI;
249       Dict["dist"] = Dist; 
250       Dict["sect"] = Section;
251       string::size_type pos = File.rfind("/");
252       if (pos != string::npos)
253          Dict["file"] = string(File, pos+1);
254       else
255          Dict["file"] = File;
256       
257       rpmdata->TranslateSource(Res, Dict);
258    }
259          
260    return Res;
261 }
262                                                                         /*}}}*/
263 // SrcListIndex::CreateSrcParser - Get a parser for the source files    /*{{{*/
264 // ---------------------------------------------------------------------
265 /* */
266 pkgSrcRecords::Parser *rpmSrcListIndex::CreateSrcParser() const
267 {
268    return new rpmSrcRecordParser(IndexPath(), this);
269 }
270                                                                         /*}}}*/
271 // SrcListIndex::GetIndexes - Fetch the index files                     /*{{{*/
272 // ---------------------------------------------------------------------
273 /* */
274 bool rpmSrcListIndex::GetIndexes(pkgAcquire *Owner) const
275 {
276    // Ignore indexes for repositories that could not be authenticated
277    if (Repository->IsAuthenticated() == true && 
278        Repository->HasRelease() == false)
279       return true;
280    new pkgAcqIndex(Owner,Repository,IndexURI("srclist"),Info("srclist"),
281                    "srclist");
282    return true;
283 }
284                                                                         /*}}}*/
285
286 // PkgListIndex::ArchiveInfo - Short version of the archive url         /*{{{*/
287 // ---------------------------------------------------------------------
288 /* This is a shorter version that is designed to be < 60 chars or so */
289 string rpmPkgListIndex::ArchiveInfo(pkgCache::VerIterator Ver) const
290 {
291    string Res = ::URI::SiteOnly(URI) + ' ';
292    if (Dist[Dist.size() - 1] == '/')
293    {
294       if (Dist != "/")
295          Res += Dist;
296    }
297    else
298       Res += Dist + '/' + Section;
299    
300    Res += " ";
301    Res += Ver.ParentPkg().Name();
302    Res += " ";
303    Res += Ver.VerStr();
304    return Res;
305 }
306                                                                         /*}}}*/
307 // PkgListIndex::ArchiveURI - URI for the archive                       /*{{{*/
308 // ---------------------------------------------------------------------
309 string rpmPkgListIndex::ArchiveURI(string File) const
310 {
311    RPMPackageData *rpmdata = RPMPackageData::Singleton();
312    string Res;
313    if (Dist[Dist.size() - 1] == '/')
314    {
315       if (Dist != "/")
316          Res = URI + Dist;
317       else
318          Res = URI;
319    }
320    else
321       Res = URI + Dist;
322
323    if (File.find("/") != string::npos)
324       Res += '/' + File;
325    else
326       Res += "/RPMS." + Section + '/' + File;
327
328    if (rpmdata->HasBinaryTranslation() == true)
329    {
330       map<string,string> Dict;
331       Dict["uri"] = URI;
332       Dict["dist"] = Dist; 
333       Dict["sect"] = Section;
334       string::size_type pos = File.rfind("/");
335       if (pos != string::npos)
336          Dict["file"] = string(File, pos+1);
337       else
338          Dict["file"] = File;
339       rpmdata->TranslateBinary(Res, Dict);
340    }
341          
342    return Res;
343 }
344                                                                         /*}}}*/
345 // PkgListIndex::GetIndexes - Fetch the index files                     /*{{{*/
346 // ---------------------------------------------------------------------
347 /* */
348 bool rpmPkgListIndex::GetIndexes(pkgAcquire *Owner) const
349 {
350    // Ignore indexes for repositories that could not be authenticated
351    if (Repository->IsAuthenticated() == true && 
352        Repository->HasRelease() == false)
353       return true;
354    new pkgAcqIndex(Owner,Repository,IndexURI("pkglist"),Info("pkglist"),
355                    "pkglist");
356    new pkgAcqIndexRel(Owner,Repository,IndexURI("release"),Info("release"),
357                       "release");
358    return true;
359 }
360                                                                         /*}}}*/
361 // PkgListIndex::Merge - Load the index file into a cache               /*{{{*/
362 // ---------------------------------------------------------------------
363 /* */
364 bool rpmPkgListIndex::Merge(pkgCacheGenerator &Gen,OpProgress &Prog) const
365 {
366    string PackageFile = IndexPath();
367    RPMHandler *Handler = CreateHandler();
368
369    Prog.SubProgress(0,Info(MainType()));
370    ::URI Tmp(URI);
371    if (Gen.SelectFile(PackageFile,Tmp.Host,*this) == false)
372    {
373       delete Handler;
374       return _error->Error(_("Problem with SelectFile %s"),PackageFile.c_str());
375    }
376
377    // Store the IMS information
378    pkgCache::PkgFileIterator File = Gen.GetCurFile();
379    struct stat St;
380    if (stat(PackageFile.c_str(),&St) != 0) 
381    {
382       delete Handler;
383       return _error->Errno("stat",_("Failed to stat %s"), PackageFile.c_str());
384    }
385    File->Size = St.st_size;
386    File->mtime = St.st_mtime;
387    
388    rpmListParser Parser(Handler);
389    if (_error->PendingError() == true) 
390    {
391       delete Handler;
392       return _error->Error(_("Problem opening %s"),PackageFile.c_str());
393    }
394    
395    if (Gen.MergeList(Parser) == false)
396    {
397       delete Handler;
398       return _error->Error(_("Problem with MergeList %s"),PackageFile.c_str());
399    }
400    
401    delete Handler;
402
403    // Check the release file
404    string RelFile = ReleasePath();
405    if (FileExists(RelFile) == true)
406    {
407       FileFd Rel(RelFile,FileFd::ReadOnly);
408       if (_error->PendingError() == true)
409          return false;
410       Parser.LoadReleaseInfo(File,Rel);
411       Rel.Seek(0);
412    }
413
414    return true;
415 }
416                                                                         /*}}}*/
417 // PkgListIndex::MergeFileProvides - Process file dependencies if any   /*{{{*/
418 // ---------------------------------------------------------------------
419 /* */
420 bool rpmPkgListIndex::MergeFileProvides(pkgCacheGenerator &Gen,
421                                         OpProgress &Prog) const
422 {
423    string PackageFile = IndexPath();
424    RPMHandler *Handler = CreateHandler();
425    rpmListParser Parser(Handler);
426    if (_error->PendingError() == true) {
427       delete Handler;
428       return _error->Error(_("Problem opening %s"),PackageFile.c_str());
429    }
430    // We call SubProgress with Size(), since we won't call SelectFile() here.
431    Prog.SubProgress(Size(),Info("pkglist"));
432    if (Gen.MergeFileProvides(Parser) == false)
433       return _error->Error(_("Problem with MergeFileProvides %s"),
434                            PackageFile.c_str());
435    delete Handler;
436    return true;
437 }
438                                                                         /*}}}*/
439 // PkgListIndex::FindInCache - Find this index                          /*{{{*/
440 // ---------------------------------------------------------------------
441 /* */
442 pkgCache::PkgFileIterator rpmPkgListIndex::FindInCache(pkgCache &Cache) const
443 {
444    string FileName = IndexPath();
445    pkgCache::PkgFileIterator File = Cache.FileBegin();
446    for (; File.end() == false; File++)
447    {
448       if (FileName != File.FileName())
449          continue;
450       
451       struct stat St;
452       if (stat(File.FileName(),&St) != 0)
453          return pkgCache::PkgFileIterator(Cache);
454
455       if ((unsigned)St.st_size != File->Size || St.st_mtime != File->mtime)
456          return pkgCache::PkgFileIterator(Cache);
457       return File;
458    }
459    
460    return File;
461 }
462                                                                         /*}}}*/
463
464 // PkgDirIndex::Index* - Return the URI to the index files              /*{{{*/
465 // ---------------------------------------------------------------------
466 /* */
467 string rpmPkgDirIndex::IndexPath() const
468 {
469    return ::URI(ArchiveURI("")).Path;
470 }
471                                                                         /*}}}*/
472 // PkgDirIndex::Release* - Return the URI to the index files            /*{{{*/
473 // ---------------------------------------------------------------------
474 /* */
475 string rpmPkgDirIndex::ReleasePath() const
476 {
477    return ::URI(IndexURI("release")).Path;
478 }
479                                                                         /*}}}*/
480 // PkgDirIndex::Size - Return the size of the index                     /*{{{*/
481 // ---------------------------------------------------------------------
482 /* This is really only used for progress reporting. */
483 unsigned long rpmPkgDirIndex::Size() const
484 {
485    // XXX: Must optimize this somehow.
486    RPMHandler *Handler = CreateHandler();
487    unsigned long Res = Handler->Size();
488    delete Handler;
489    return Res;
490 }
491                                                                         /*}}}*/
492
493 // SrcDirIndex::Index* - Return the URI to the index files              /*{{{*/
494 // ---------------------------------------------------------------------
495 /* */
496 string rpmSrcDirIndex::IndexPath() const
497 {
498    return ::URI(ArchiveURI("")).Path;
499 }
500                                                                         /*}}}*/
501 // SrcDirIndex::Size - Return the size of the index                     /*{{{*/
502 // ---------------------------------------------------------------------
503 /* This is really only used for progress reporting. */
504 unsigned long rpmSrcDirIndex::Size() const
505 {
506    // XXX: Must optimize this somehow.
507    RPMHandler *Handler = CreateHandler();
508    unsigned long Res = Handler->Size();
509    delete Handler;
510    return Res;
511 }
512
513 // SinglePkgIndex::ArchiveURI - URI for the archive                     /*{{{*/
514 // ---------------------------------------------------------------------
515 string rpmSinglePkgIndex::ArchiveURI(string File) const
516 {
517    char *cwd = getcwd(NULL,0);
518    if (File[0] == '.' && File[1] == '/')
519       File = string(File, 2);
520    string URI = "file://"+flCombine(cwd, File);
521    free(cwd);
522    return URI;
523 }
524                                                                         /*}}}*/
525 // SinglePkgIndex::ArchiveURI - URI for the archive                     /*{{{*/
526 // ---------------------------------------------------------------------
527 string rpmSingleSrcIndex::ArchiveURI(string File) const
528 {
529    char *cwd = getcwd(NULL,0);
530    if (File[0] == '.' && File[1] == '/')
531       File = string(File, 2);
532    string URI = "file://"+flCombine(cwd, File);
533    free(cwd);
534    return URI;
535 }
536
537 string rpmRepomdIndex::ArchiveURI(string File) const
538 {
539    RPMPackageData *rpmdata = RPMPackageData::Singleton();
540    string Res;
541
542    //cout << Dist << File << endl;
543
544    Res += URI + '/' + Dist + '/' + File;
545    //cout << "repomd archiveuri " << " " << Res << endl;
546
547    return Res;
548 }
549
550 string rpmRepomdIndex::ArchiveInfo(pkgCache::VerIterator Ver) const
551 {
552    string Res = ::URI::SiteOnly(URI) + ' ';
553    if (Dist[Dist.size() - 1] == '/')
554    {
555       if (Dist != "/")
556          Res += Dist;
557    }
558    else
559       Res += Dist + '/';
560    
561    Res += " ";
562    Res += Ver.ParentPkg().Name();
563    Res += " ";
564    Res += Ver.VerStr();
565    return Res;
566 }
567 pkgCache::PkgFileIterator rpmRepomdIndex::FindInCache(pkgCache &Cache) const
568 {
569    string FileName = IndexPath();
570    pkgCache::PkgFileIterator File = Cache.FileBegin();
571    for (; File.end() == false; File++)
572    {
573       if (FileName != File.FileName())
574          continue;
575       
576       struct stat St;
577       if (stat(File.FileName(),&St) != 0)
578          return pkgCache::PkgFileIterator(Cache);
579
580       if ((unsigned)St.st_size != File->Size || St.st_mtime != File->mtime)
581          return pkgCache::PkgFileIterator(Cache);
582       return File;
583    }
584    
585    return File;
586 }
587 string rpmRepomdIndex::ReleaseURI(string Type) const
588 {
589    RPMPackageData *rpmdata = RPMPackageData::Singleton();
590    string Res;
591    Res = URI + Dist + "/repodata/" + "repomd.xml";
592
593    //cout << "XXXXX repomd releaseuri " << Res << endl;
594    
595    return Res;
596 }
597
598 string rpmRepomdIndex::ReleaseInfo(string Type) const
599 {
600    string Info = ::URI::SiteOnly(URI) + ' ';
601    if (Dist[Dist.size() - 1] == '/')
602    {
603       if (Dist != "/")
604          Info += Dist;
605    }
606    else
607       Info += Dist;   
608    Info += " ";
609    Info += Type;
610    //cout << "repomd releaseinfo " << Info << endl;
611    return Info;
612 };
613
614 string rpmRepomdIndex::Info(string Type) const 
615 {
616    string Info = ::URI::SiteOnly(URI) + ' ';
617    if (Dist[Dist.size() - 1] == '/')
618    {
619       if (Dist != "/")
620          Info += Dist;
621    }
622    else
623       Info += Dist + '/' ;   
624    Info += " ";
625    Info += Type;
626    //cout << "repomd info " << Info << endl;
627    return Info;
628 }
629
630 string rpmRepomdIndex::IndexURI(string Type) const
631 {
632    RPMPackageData *rpmdata = RPMPackageData::Singleton();
633    string Res = URI + Dist;
634    if (Dist[Dist.size() - 1] != '/') {
635          Res += "/";
636    }
637    Res += "repodata/primary.xml";
638    return Res;
639 }
640                                                                         /*}}}*/
641 bool rpmRepomdIndex::GetReleases(pkgAcquire *Owner) const
642 {
643    if (!Repository->Acquire)
644       return true;
645    Repository->Acquire = false;
646    // ignore for now - we need to parse the file for checksum and 
647    // optimally for location of other xml files as well
648    new pkgAcqIndexRel(Owner,Repository,ReleaseURI("repomd.xml"),
649                       ReleaseInfo("repomd.xml"), "repomd.xml", true);
650    return true;
651 }
652
653 bool rpmRepomdIndex::GetIndexes(pkgAcquire *Owner) const
654 {
655    new pkgAcqIndex(Owner,Repository,IndexURI("primary.xml"),
656                    Info("primary.xml"), "primary.xml");
657 // wont be needing these for a while...
658 #if 0
659    new pkgAcqIndex(Owner,Repository,IndexURI("filelists.xml"),
660                    Info("filelists.xml"), "filelists.xml");
661    new pkgAcqIndex(Owner,Repository,IndexURI("other.xml"),
662                    Info("other.xml"), "other.xml");
663 #endif
664    return true;
665 }
666
667 string rpmRepomdIndex::Describe(bool Short) const
668 {
669    char S[300];
670    if (Short == true)
671       snprintf(S,sizeof(S),"%s",Info(MainType()).c_str());
672    else
673       snprintf(S,sizeof(S),"%s (%s)",Info(MainType()).c_str(),
674          IndexFile(MainType()).c_str());
675    return S;
676 }
677
678 string rpmRepomdIndex::IndexFile(string Type) const
679 {
680    //cout << "XXX indexfile " << _config->FindDir("Dir::State::lists")+URItoFileName(IndexURI(Type)) << endl;
681    return _config->FindDir("Dir::State::lists")+URItoFileName(IndexURI(Type));
682 };
683
684
685 bool rpmRepomdIndex::Exists() const
686 {
687    return FileExists(IndexPath());
688 }
689
690 string rpmRepomdIndex::ReleasePath() const
691 {
692    return _config->FindDir("Dir::State::lists")+URItoFileName(ReleaseURI("release.xml"));
693 }
694
695
696 bool rpmRepomdIndex::Merge(pkgCacheGenerator &Gen,OpProgress &Prog) const
697 {
698    //cout << "repomd MERGE" << endl;
699    string PackageFile = IndexPath();
700    RPMHandler *Handler = CreateHandler();
701
702    Prog.SubProgress(0,Info(MainType()));
703    ::URI Tmp(URI);
704    if (Gen.SelectFile(PackageFile,Tmp.Host,*this) == false)
705    {
706       delete Handler;
707       return _error->Error(_("Problem with SelectFile %s"),PackageFile.c_str());
708    }
709
710    // Store the IMS information
711    pkgCache::PkgFileIterator File = Gen.GetCurFile();
712    //cout << "merge pkgfile " << PackageFile << endl;
713    struct stat St;
714    if (stat(PackageFile.c_str(),&St) != 0) 
715    {
716       delete Handler;
717       return _error->Errno("stat",_("Failed to stat %s"), PackageFile.c_str());
718    }
719    File->Size = St.st_size;
720    File->mtime = St.st_mtime;
721    
722    rpmRepomdParser Parser(Handler);
723    if (_error->PendingError() == true) 
724    {
725       delete Handler;
726       return _error->Error(_("Problem opening %s"),PackageFile.c_str());
727    }
728    
729    if (Gen.MergeList(Parser) == false)
730    {
731       delete Handler;
732       return _error->Error(_("Problem with MergeList %s"),PackageFile.c_str());
733    }
734    
735    delete Handler;
736
737    // Check the release file
738    string RelFile = ReleasePath();
739    if (FileExists(RelFile) == true)
740    {
741       Parser.LoadReleaseInfo(File,RelFile);
742    }
743
744    return true;
745 }
746
747 pkgSrcRecords::Parser *rpmRepomdSrcIndex::CreateSrcParser() const
748 {
749    return new rpmSrcRecordParser(IndexPath(), this);
750 }
751
752 string rpmRepomdSrcIndex::SourceInfo(pkgSrcRecords::Parser const &Record,
753                                      pkgSrcRecords::File const &File) const
754 {
755    string Res;
756    Res = ::URI::SiteOnly(URI) + ' ';
757    if (Dist[Dist.size() - 1] == '/')
758    {
759       if (Dist != "/")
760          Res += Dist;
761    }      
762    else
763       Res += Dist + '/' + Section;
764    
765    Res += " ";
766    Res += Record.Package();
767    Res += " ";
768    Res += Record.Version();
769    if (File.Type.empty() == false)
770       Res += " (" + File.Type + ")";
771    return Res;
772 }
773
774 // DatabaseIndex::rpmDatabaseIndex - Constructor                        /*{{{*/
775 // ---------------------------------------------------------------------
776 /* */
777 rpmDatabaseIndex::rpmDatabaseIndex()
778 {
779 }
780                                                                         /*}}}*/
781 // DatabaseIndex::Size - Return the size of the index                   /*{{{*/
782 // ---------------------------------------------------------------------
783 /* */
784 unsigned long rpmDatabaseIndex::Size() const
785 {
786    return rpmSys.GetDBHandler()->Size();
787 }
788                                                                         /*}}}*/
789 // DatabaseIndex::CreateHandler - Create a RPMHandler for this file     /*{{{*/
790 // ---------------------------------------------------------------------
791 RPMHandler *rpmDatabaseIndex::CreateHandler() const
792 {
793    return rpmSys.GetDBHandler();
794 }
795                                                                         /*}}}*/
796 // DatabaseIndex::Merge - Load the index file into a cache              /*{{{*/
797 // ---------------------------------------------------------------------
798 /* */
799 bool rpmDatabaseIndex::Merge(pkgCacheGenerator &Gen,OpProgress &Prog) const
800 {
801    RPMDBHandler *Handler = rpmSys.GetDBHandler();
802    rpmListParser Parser(Handler);
803    if (_error->PendingError() == true)
804       return _error->Error(_("Problem opening RPM database"));
805    
806    Prog.SubProgress(0,"RPM Database");
807    if (Gen.SelectFile(Handler->DataPath(false),string(),*this,pkgCache::Flag::NotSource) == false)
808       return _error->Error(_("Problem with SelectFile RPM Database"));
809
810    // Store the IMS information
811    pkgCache::PkgFileIterator CFile = Gen.GetCurFile();
812    struct stat St;
813    if (stat(Handler->DataPath(false).c_str(),&St) != 0)
814       return _error->Errno("fstat",_("Failed to stat %s"), Handler->DataPath(false).c_str());
815    CFile->Size = St.st_size;
816    CFile->mtime = Handler->Mtime();
817    
818    if (Gen.MergeList(Parser) == false)
819       return _error->Error(_("Problem with MergeList %s"),
820                            Handler->DataPath(false).c_str());
821    return true;
822 }
823                                                                         /*}}}*/
824 // DatabaseIndex::MergeFileProvides - Process file dependencies if any  /*{{{*/
825 // ---------------------------------------------------------------------
826 /* */
827 bool rpmDatabaseIndex::MergeFileProvides(pkgCacheGenerator &Gen,
828                                          OpProgress &Prog) const
829 {
830    RPMDBHandler *Handler = rpmSys.GetDBHandler();
831    rpmListParser Parser(Handler);
832    if (_error->PendingError() == true)
833       return _error->Error(_("Problem opening RPM database"));
834    // We call SubProgress with Size(), since we won't call SelectFile() here.
835    Prog.SubProgress(Size(),"RPM Database");
836    if (Gen.MergeFileProvides(Parser) == false)
837       return _error->Error(_("Problem with MergeFileProvides %s"),
838                            Handler->DataPath(false).c_str());
839    return true;
840 }
841                                                                         /*}}}*/
842 // DatabaseIndex::FindInCache - Find this index                         /*{{{*/
843 // ---------------------------------------------------------------------
844 /* */
845 pkgCache::PkgFileIterator rpmDatabaseIndex::FindInCache(pkgCache &Cache) const
846 {
847    pkgCache::PkgFileIterator File = Cache.FileBegin();
848    for (; File.end() == false; File++)
849    {
850       if (rpmSys.GetDBHandler()->DataPath(false) != File.FileName())
851          continue;
852       struct stat St;
853       if (stat(File.FileName(),&St) != 0)
854          return pkgCache::PkgFileIterator(Cache);
855       if ((unsigned)St.st_size != File->Size || St.st_mtime != File->mtime)
856          return pkgCache::PkgFileIterator(Cache);
857       return File;
858    }   
859    return File;
860 }
861                                                                         /*}}}*/
862
863 // Source List types for rpm                                            /*{{{*/
864
865 class rpmSLTypeGen : public pkgSourceList::Type
866 {
867    public:
868
869    rpmSLTypeGen()
870    {
871       Name = "rpm";
872       Label = "Standard RPM source tree";
873    }   
874
875    pkgRepository *FindRepository(string URI,string Dist,
876                                  const pkgSourceList::Vendor *Vendor) const
877    {
878       for (vector<pkgRepository *>::const_iterator iter = RepList.begin();
879            iter != RepList.end(); iter++) 
880       {
881          if ((*iter)->URI == URI && (*iter)->Dist == Dist) 
882          {       
883             if (Vendor != NULL)
884                (*iter)->FingerPrint = Vendor->FingerPrint;
885             return *iter;
886          }
887       }
888       return NULL;
889    }
890
891    pkgRepository *GetRepository(string URI,string Dist,
892                                 const pkgSourceList::Vendor *Vendor) const
893    {
894       pkgRepository *Rep = FindRepository(URI,Dist,Vendor);
895       if (Rep != NULL)
896          return Rep;
897
898       string BaseURI;
899       if (Dist[Dist.size() - 1] == '/')
900       {
901          if (Dist != "/")
902             BaseURI = URI + Dist;
903          else 
904             BaseURI = URI + '/';
905       }
906       else
907          BaseURI = URI + Dist + '/';
908
909       //cout << "XXXX GetRepo " << URI << " " << Dist << " " << BaseURI << endl;
910       Rep = new pkgRepository(URI,Dist,Vendor,BaseURI);
911       RepList.push_back(Rep);
912       return Rep;
913    }
914 };
915
916
917 class rpmSLTypeRpm : public rpmSLTypeGen
918 {
919    public:
920
921    bool CreateItem(vector<pkgIndexFile *> &List,
922                    string URI, string Dist, string Section,
923                    pkgSourceList::Vendor const *Vendor) const 
924    {
925       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
926       List.push_back(new rpmPkgListIndex(URI,Dist,Section,Rep));
927       return true;
928    };
929
930    rpmSLTypeRpm()
931    {
932       Name = "rpm";
933       Label = "Standard RPM binary tree";
934    }   
935 };
936
937 class rpmSLTypeSrpm : public rpmSLTypeGen
938 {
939    public:
940
941    bool CreateItem(vector<pkgIndexFile *> &List,
942                    string URI, string Dist, string Section,
943                    pkgSourceList::Vendor const *Vendor) const 
944    {
945       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
946       List.push_back(new rpmSrcListIndex(URI,Dist,Section,Rep));
947       return true;
948    };  
949    
950    rpmSLTypeSrpm()
951    {
952       Name = "rpm-src";
953       Label = "Standard RPM source tree";
954    }   
955 };
956
957 class rpmSLTypeRpmDir : public rpmSLTypeGen
958 {
959    public:
960
961    bool CreateItem(vector<pkgIndexFile *> &List,
962                    string URI, string Dist, string Section,
963                    pkgSourceList::Vendor const *Vendor) const 
964    {
965       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
966       List.push_back(new rpmPkgDirIndex(URI,Dist,Section,Rep));
967       return true;
968    };
969
970    rpmSLTypeRpmDir()
971    {
972       Name = "rpm-dir";
973       Label = "Local RPM directory tree";
974    }   
975 };
976
977 class rpmSLTypeSrpmDir : public rpmSLTypeGen
978 {
979    public:
980
981    bool CreateItem(vector<pkgIndexFile *> &List,
982                    string URI, string Dist, string Section,
983                    pkgSourceList::Vendor const *Vendor) const 
984    {
985       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
986       List.push_back(new rpmSrcDirIndex(URI,Dist,Section,Rep));
987       return true;
988    };
989
990    rpmSLTypeSrpmDir()
991    {
992       Name = "rpm-src-dir";
993       Label = "Local SRPM directory tree";
994    }   
995 };
996
997 class rpmSLTypeRepomd : public rpmSLTypeGen
998 {
999    public:
1000
1001    pkgRepository *GetRepository(string URI,string Dist,
1002                                 const pkgSourceList::Vendor *Vendor) const
1003    {
1004       pkgRepository *Rep = FindRepository(URI,Dist,Vendor);
1005       if (Rep != NULL)
1006          return Rep;
1007
1008       string BaseURI;
1009       if (Dist[Dist.size() - 1] == '/')
1010       {
1011          if (Dist != "/")
1012             BaseURI = URI + Dist;
1013          else 
1014             BaseURI = URI + '/';
1015       }
1016       else
1017          BaseURI = URI + Dist + '/';
1018
1019       Rep = new repomdRepository(URI,Dist,Vendor,BaseURI);
1020       RepList.push_back(Rep);
1021       return Rep;
1022    }
1023
1024    bool CreateItem(vector<pkgIndexFile *> &List,
1025                    string URI, string Dist, string Section,
1026                    pkgSourceList::Vendor const *Vendor) const 
1027    {
1028       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
1029       List.push_back(new rpmRepomdPkgIndex(URI,Dist,Section,Rep));
1030       return true;
1031    };
1032
1033    bool ParseLine(vector<pkgIndexFile *> &List,
1034                   pkgSourceList::Vendor const *Vendor,
1035                   const char *Buffer,
1036                   unsigned long CurLine,string File) const
1037    {
1038       string URI;
1039       string Dist;
1040       if (ParseQuoteWord(Buffer,URI) == false)
1041          return _error->Error(_("Malformed line %lu in source list %s (URI)"),CurLine,File.c_str());
1042       if (ParseQuoteWord(Buffer,Dist) == false)
1043          return _error->Error(_("Malformed line %lu in source list %s (dist)"),CurLine,File.c_str());
1044
1045       if (FixupURI(URI) == false)
1046          return _error->Error(_("Malformed line %lu in source list %s (URI parse)"),CurLine,File.c_str());
1047       
1048       Dist = SubstVar(Dist,"$(ARCH)",_config->Find("APT::Architecture"));
1049       Dist = SubstVar(Dist,"$(VERSION)",_config->Find("APT::DistroVersion"));
1050
1051       if (CreateItem(List,URI,Dist,"",Vendor) == false)
1052          return false;
1053
1054       return true;
1055    };
1056
1057
1058    rpmSLTypeRepomd()
1059    {
1060       Name = "repomd";
1061       Label = "RepoMD tree";
1062    }   
1063 };
1064 class rpmSLTypeRepomdSrc : public rpmSLTypeRepomd
1065 {
1066    public:
1067
1068    bool CreateItem(vector<pkgIndexFile *> &List,
1069                    string URI, string Dist, string Section,
1070                    pkgSourceList::Vendor const *Vendor) const 
1071    {
1072       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
1073       List.push_back(new rpmRepomdSrcIndex(URI,Dist,Section,Rep));
1074       return true;
1075    };
1076
1077    rpmSLTypeRepomdSrc()
1078    {
1079       Name = "repomd-src";
1080       Label = "RepoMD src tree";
1081    }   
1082 };
1083
1084 rpmSLTypeRpm _apt_rpmType;
1085 rpmSLTypeSrpm _apt_rpmSrcType;
1086 rpmSLTypeRpmDir _apt_rpmDirType;
1087 rpmSLTypeSrpmDir _apt_rpmSrcDirType;
1088 rpmSLTypeRepomd _apt_repomdType;
1089 rpmSLTypeRepomdSrc _apt_repomdSrcType;
1090                                                                         /*}}}*/
1091 // Index File types for rpm                                             /*{{{*/
1092 class rpmIFTypeSrc : public pkgIndexFile::Type
1093 {
1094    public:
1095    
1096    rpmIFTypeSrc() {Label = "RPM Source Index";};
1097 };
1098 class rpmIFTypePkg : public pkgIndexFile::Type
1099 {
1100    public:
1101    
1102    virtual pkgRecords::Parser *CreatePkgParser(pkgCache::PkgFileIterator File) const
1103    {
1104       return new rpmRecordParser(File.FileName(),*File.Cache());
1105    };
1106    rpmIFTypePkg() {Label = "RPM Package Index";};
1107 };
1108 class rpmIFTypeDatabase : public pkgIndexFile::Type
1109 {
1110    public:
1111    
1112    virtual pkgRecords::Parser *CreatePkgParser(pkgCache::PkgFileIterator File) const
1113    {
1114       return new rpmRecordParser(File.FileName(),*File.Cache());
1115    };
1116    rpmIFTypeDatabase() {Label = "RPM Database";};
1117 };
1118 static rpmIFTypeSrc _apt_Src;
1119 static rpmIFTypePkg _apt_Pkg;
1120 static rpmIFTypeDatabase _apt_DB;
1121
1122 const pkgIndexFile::Type *rpmSrcListIndex::GetType() const
1123 {
1124    return &_apt_Src;
1125 }
1126 const pkgIndexFile::Type *rpmPkgListIndex::GetType() const
1127 {
1128    return &_apt_Pkg;
1129 }
1130 const pkgIndexFile::Type *rpmSrcDirIndex::GetType() const
1131 {
1132    return &_apt_Src;
1133 }
1134 const pkgIndexFile::Type *rpmPkgDirIndex::GetType() const
1135 {
1136    return &_apt_Pkg;
1137 }
1138 const pkgIndexFile::Type *rpmSinglePkgIndex::GetType() const
1139 {
1140    return &_apt_Pkg;
1141 }
1142 const pkgIndexFile::Type *rpmSingleSrcIndex::GetType() const
1143 {
1144    return &_apt_Src;
1145 }
1146 const pkgIndexFile::Type *rpmDatabaseIndex::GetType() const
1147 {
1148    return &_apt_DB;
1149 }
1150 const pkgIndexFile::Type *rpmRepomdPkgIndex::GetType() const
1151 {
1152    return &_apt_Pkg;
1153 }
1154 const pkgIndexFile::Type *rpmRepomdSrcIndex::GetType() const
1155 {
1156    return &_apt_Src;
1157 }
1158
1159                                                                         /*}}}*/
1160 #endif /* HAVE_RPM */
1161
1162 // vim:sts=3:sw=3