- implement filelists.xml processing for file dependencies
[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    // XXXX eh, this doesn't look sane .. 
638    if (Type == "repomd")
639       Res += "repodata/primary.xml";
640    else
641       Res += "repodata/" + Type;
642    return Res;
643 }
644                                                                         /*}}}*/
645 bool rpmRepomdIndex::GetReleases(pkgAcquire *Owner) const
646 {
647    if (!Repository->Acquire)
648       return true;
649    Repository->Acquire = false;
650    // ignore for now - we need to parse the file for checksum and 
651    // optimally for location of other xml files as well
652    new pkgAcqIndexRel(Owner,Repository,ReleaseURI("repomd.xml"),
653                       ReleaseInfo("repomd.xml"), "repomd.xml", true);
654    return true;
655 }
656
657 bool rpmRepomdIndex::GetIndexes(pkgAcquire *Owner) const
658 {
659    new pkgAcqIndex(Owner,Repository,IndexURI("primary.xml"),
660                    Info("primary.xml"), "primary.xml");
661    new pkgAcqIndex(Owner,Repository,IndexURI("filelists.xml"),
662                    Info("filelists.xml"), "filelists.xml");
663 #if 0
664    new pkgAcqIndex(Owner,Repository,IndexURI("other.xml"),
665                    Info("other.xml"), "other.xml");
666 #endif
667    return true;
668 }
669
670 string rpmRepomdIndex::Describe(bool Short) const
671 {
672    char S[300];
673    if (Short == true)
674       snprintf(S,sizeof(S),"%s",Info(MainType()).c_str());
675    else
676       snprintf(S,sizeof(S),"%s (%s)",Info(MainType()).c_str(),
677          IndexFile(MainType()).c_str());
678    return S;
679 }
680
681 string rpmRepomdIndex::IndexFile(string Type) const
682 {
683    return _config->FindDir("Dir::State::lists")+URItoFileName(IndexURI(Type));
684 };
685
686
687 bool rpmRepomdIndex::Exists() const
688 {
689    return FileExists(IndexPath());
690 }
691
692 string rpmRepomdIndex::ReleasePath() const
693 {
694    return _config->FindDir("Dir::State::lists")+URItoFileName(ReleaseURI("release.xml"));
695 }
696
697
698 bool rpmRepomdIndex::Merge(pkgCacheGenerator &Gen,OpProgress &Prog) const
699 {
700    //cout << "repomd MERGE" << endl;
701    string PackageFile = IndexPath();
702    RPMHandler *Handler = CreateHandler();
703
704    Prog.SubProgress(0,Info(MainType()));
705    ::URI Tmp(URI);
706    if (Gen.SelectFile(PackageFile,Tmp.Host,*this) == false)
707    {
708       delete Handler;
709       return _error->Error(_("Problem with SelectFile %s"),PackageFile.c_str());
710    }
711
712    // Store the IMS information
713    pkgCache::PkgFileIterator File = Gen.GetCurFile();
714    //cout << "merge pkgfile " << PackageFile << endl;
715    struct stat St;
716    if (stat(PackageFile.c_str(),&St) != 0) 
717    {
718       delete Handler;
719       return _error->Errno("stat",_("Failed to stat %s"), PackageFile.c_str());
720    }
721    File->Size = St.st_size;
722    File->mtime = St.st_mtime;
723    
724    rpmRepomdParser Parser(Handler);
725    if (_error->PendingError() == true) 
726    {
727       delete Handler;
728       return _error->Error(_("Problem opening %s"),PackageFile.c_str());
729    }
730    
731    if (Gen.MergeList(Parser) == false)
732    {
733       delete Handler;
734       return _error->Error(_("Problem with MergeList %s"),PackageFile.c_str());
735    }
736    
737    delete Handler;
738
739    // Check the release file
740    string RelFile = ReleasePath();
741    if (FileExists(RelFile) == true)
742    {
743       Parser.LoadReleaseInfo(File,RelFile);
744    }
745
746    return true;
747 }
748
749 bool rpmRepomdIndex::MergeFileProvides(pkgCacheGenerator &Gen,
750                                         OpProgress &Prog) const
751 {
752    string PackageFile = IndexPath();
753    RPMHandler *Handler = CreateHandler();
754    rpmListParser Parser(Handler);
755    if (_error->PendingError() == true) {
756       delete Handler;
757       return _error->Error(_("Problem opening %s"),PackageFile.c_str());
758    }
759    // We call SubProgress with Size(), since we won't call SelectFile() here.
760    Prog.SubProgress(Size(),Info("pkglist"));
761    if (Gen.MergeFileProvides(Parser) == false)
762       return _error->Error(_("Problem with MergeFileProvides %s"),
763                            PackageFile.c_str());
764    delete Handler;
765    return true;
766 }
767
768 pkgSrcRecords::Parser *rpmRepomdSrcIndex::CreateSrcParser() const
769 {
770    return new rpmSrcRecordParser(IndexPath(), this);
771 }
772
773 string rpmRepomdSrcIndex::SourceInfo(pkgSrcRecords::Parser const &Record,
774                                      pkgSrcRecords::File const &File) const
775 {
776    string Res;
777    Res = ::URI::SiteOnly(URI) + ' ';
778    if (Dist[Dist.size() - 1] == '/')
779    {
780       if (Dist != "/")
781          Res += Dist;
782    }      
783    else
784       Res += Dist + '/' + Section;
785    
786    Res += " ";
787    Res += Record.Package();
788    Res += " ";
789    Res += Record.Version();
790    if (File.Type.empty() == false)
791       Res += " (" + File.Type + ")";
792    return Res;
793 }
794
795 // DatabaseIndex::rpmDatabaseIndex - Constructor                        /*{{{*/
796 // ---------------------------------------------------------------------
797 /* */
798 rpmDatabaseIndex::rpmDatabaseIndex()
799 {
800 }
801                                                                         /*}}}*/
802 // DatabaseIndex::Size - Return the size of the index                   /*{{{*/
803 // ---------------------------------------------------------------------
804 /* */
805 unsigned long rpmDatabaseIndex::Size() const
806 {
807    return rpmSys.GetDBHandler()->Size();
808 }
809                                                                         /*}}}*/
810 // DatabaseIndex::CreateHandler - Create a RPMHandler for this file     /*{{{*/
811 // ---------------------------------------------------------------------
812 RPMHandler *rpmDatabaseIndex::CreateHandler() const
813 {
814    return rpmSys.GetDBHandler();
815 }
816                                                                         /*}}}*/
817 // DatabaseIndex::Merge - Load the index file into a cache              /*{{{*/
818 // ---------------------------------------------------------------------
819 /* */
820 bool rpmDatabaseIndex::Merge(pkgCacheGenerator &Gen,OpProgress &Prog) const
821 {
822    RPMDBHandler *Handler = rpmSys.GetDBHandler();
823    rpmListParser Parser(Handler);
824    if (_error->PendingError() == true)
825       return _error->Error(_("Problem opening RPM database"));
826    
827    Prog.SubProgress(0,"RPM Database");
828    if (Gen.SelectFile(Handler->DataPath(false),string(),*this,pkgCache::Flag::NotSource) == false)
829       return _error->Error(_("Problem with SelectFile RPM Database"));
830
831    // Store the IMS information
832    pkgCache::PkgFileIterator CFile = Gen.GetCurFile();
833    struct stat St;
834    if (stat(Handler->DataPath(false).c_str(),&St) != 0)
835       return _error->Errno("fstat",_("Failed to stat %s"), Handler->DataPath(false).c_str());
836    CFile->Size = St.st_size;
837    CFile->mtime = Handler->Mtime();
838    
839    if (Gen.MergeList(Parser) == false)
840       return _error->Error(_("Problem with MergeList %s"),
841                            Handler->DataPath(false).c_str());
842    return true;
843 }
844                                                                         /*}}}*/
845 // DatabaseIndex::MergeFileProvides - Process file dependencies if any  /*{{{*/
846 // ---------------------------------------------------------------------
847 /* */
848 bool rpmDatabaseIndex::MergeFileProvides(pkgCacheGenerator &Gen,
849                                          OpProgress &Prog) const
850 {
851    RPMDBHandler *Handler = rpmSys.GetDBHandler();
852    rpmListParser Parser(Handler);
853    if (_error->PendingError() == true)
854       return _error->Error(_("Problem opening RPM database"));
855    // We call SubProgress with Size(), since we won't call SelectFile() here.
856    Prog.SubProgress(Size(),"RPM Database");
857    if (Gen.MergeFileProvides(Parser) == false)
858       return _error->Error(_("Problem with MergeFileProvides %s"),
859                            Handler->DataPath(false).c_str());
860    return true;
861 }
862                                                                         /*}}}*/
863 // DatabaseIndex::FindInCache - Find this index                         /*{{{*/
864 // ---------------------------------------------------------------------
865 /* */
866 pkgCache::PkgFileIterator rpmDatabaseIndex::FindInCache(pkgCache &Cache) const
867 {
868    pkgCache::PkgFileIterator File = Cache.FileBegin();
869    for (; File.end() == false; File++)
870    {
871       if (rpmSys.GetDBHandler()->DataPath(false) != File.FileName())
872          continue;
873       struct stat St;
874       if (stat(File.FileName(),&St) != 0)
875          return pkgCache::PkgFileIterator(Cache);
876       if ((unsigned)St.st_size != File->Size || St.st_mtime != File->mtime)
877          return pkgCache::PkgFileIterator(Cache);
878       return File;
879    }   
880    return File;
881 }
882                                                                         /*}}}*/
883
884 // Source List types for rpm                                            /*{{{*/
885
886 class rpmSLTypeGen : public pkgSourceList::Type
887 {
888    public:
889
890    rpmSLTypeGen()
891    {
892       Name = "rpm";
893       Label = "Standard RPM source tree";
894    }   
895
896    pkgRepository *FindRepository(string URI,string Dist,
897                                  const pkgSourceList::Vendor *Vendor) const
898    {
899       for (vector<pkgRepository *>::const_iterator iter = RepList.begin();
900            iter != RepList.end(); iter++) 
901       {
902          if ((*iter)->URI == URI && (*iter)->Dist == Dist) 
903          {       
904             if (Vendor != NULL)
905                (*iter)->FingerPrint = Vendor->FingerPrint;
906             return *iter;
907          }
908       }
909       return NULL;
910    }
911
912    pkgRepository *GetRepository(string URI,string Dist,
913                                 const pkgSourceList::Vendor *Vendor) const
914    {
915       pkgRepository *Rep = FindRepository(URI,Dist,Vendor);
916       if (Rep != NULL)
917          return Rep;
918
919       string BaseURI;
920       if (Dist[Dist.size() - 1] == '/')
921       {
922          if (Dist != "/")
923             BaseURI = URI + Dist;
924          else 
925             BaseURI = URI + '/';
926       }
927       else
928          BaseURI = URI + Dist + '/';
929
930       Rep = new pkgRepository(URI,Dist,Vendor,BaseURI);
931       RepList.push_back(Rep);
932       return Rep;
933    }
934 };
935
936
937 class rpmSLTypeRpm : 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 rpmPkgListIndex(URI,Dist,Section,Rep));
947       return true;
948    };
949
950    rpmSLTypeRpm()
951    {
952       Name = "rpm";
953       Label = "Standard RPM binary tree";
954    }   
955 };
956
957 class rpmSLTypeSrpm : 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 rpmSrcListIndex(URI,Dist,Section,Rep));
967       return true;
968    };  
969    
970    rpmSLTypeSrpm()
971    {
972       Name = "rpm-src";
973       Label = "Standard RPM source tree";
974    }   
975 };
976
977 class rpmSLTypeRpmDir : 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 rpmPkgDirIndex(URI,Dist,Section,Rep));
987       return true;
988    };
989
990    rpmSLTypeRpmDir()
991    {
992       Name = "rpm-dir";
993       Label = "Local RPM directory tree";
994    }   
995 };
996
997 class rpmSLTypeSrpmDir : public rpmSLTypeGen
998 {
999    public:
1000
1001    bool CreateItem(vector<pkgIndexFile *> &List,
1002                    string URI, string Dist, string Section,
1003                    pkgSourceList::Vendor const *Vendor) const 
1004    {
1005       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
1006       List.push_back(new rpmSrcDirIndex(URI,Dist,Section,Rep));
1007       return true;
1008    };
1009
1010    rpmSLTypeSrpmDir()
1011    {
1012       Name = "rpm-src-dir";
1013       Label = "Local SRPM directory tree";
1014    }   
1015 };
1016
1017 class rpmSLTypeRepomd : public rpmSLTypeGen
1018 {
1019    public:
1020
1021    pkgRepository *GetRepository(string URI,string Dist,
1022                                 const pkgSourceList::Vendor *Vendor) const
1023    {
1024       pkgRepository *Rep = FindRepository(URI,Dist,Vendor);
1025       if (Rep != NULL)
1026          return Rep;
1027
1028       string BaseURI;
1029       if (Dist[Dist.size() - 1] == '/')
1030       {
1031          if (Dist != "/")
1032             BaseURI = URI + Dist;
1033          else 
1034             BaseURI = URI + '/';
1035       }
1036       else
1037          BaseURI = URI + Dist + '/';
1038
1039       Rep = new repomdRepository(URI,Dist,Vendor,BaseURI);
1040       RepList.push_back(Rep);
1041       return Rep;
1042    }
1043
1044    bool CreateItem(vector<pkgIndexFile *> &List,
1045                    string URI, string Dist, string Section,
1046                    pkgSourceList::Vendor const *Vendor) const 
1047    {
1048       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
1049       List.push_back(new rpmRepomdPkgIndex(URI,Dist,Section,Rep));
1050       return true;
1051    };
1052
1053    bool ParseLine(vector<pkgIndexFile *> &List,
1054                   pkgSourceList::Vendor const *Vendor,
1055                   const char *Buffer,
1056                   unsigned long CurLine,string File) const
1057    {
1058       string URI;
1059       string Dist;
1060       if (ParseQuoteWord(Buffer,URI) == false)
1061          return _error->Error(_("Malformed line %lu in source list %s (URI)"),CurLine,File.c_str());
1062       if (ParseQuoteWord(Buffer,Dist) == false)
1063          return _error->Error(_("Malformed line %lu in source list %s (dist)"),CurLine,File.c_str());
1064
1065       if (FixupURI(URI) == false)
1066          return _error->Error(_("Malformed line %lu in source list %s (URI parse)"),CurLine,File.c_str());
1067       
1068       Dist = SubstVar(Dist,"$(ARCH)",_config->Find("APT::Architecture"));
1069       Dist = SubstVar(Dist,"$(VERSION)",_config->Find("APT::DistroVersion"));
1070
1071       if (CreateItem(List,URI,Dist,"",Vendor) == false)
1072          return false;
1073
1074       return true;
1075    };
1076
1077
1078    rpmSLTypeRepomd()
1079    {
1080       Name = "repomd";
1081       Label = "RepoMD tree";
1082    }   
1083 };
1084 class rpmSLTypeRepomdSrc : public rpmSLTypeRepomd
1085 {
1086    public:
1087
1088    bool CreateItem(vector<pkgIndexFile *> &List,
1089                    string URI, string Dist, string Section,
1090                    pkgSourceList::Vendor const *Vendor) const 
1091    {
1092       pkgRepository *Rep = GetRepository(URI,Dist,Vendor);
1093       List.push_back(new rpmRepomdSrcIndex(URI,Dist,Section,Rep));
1094       return true;
1095    };
1096
1097    rpmSLTypeRepomdSrc()
1098    {
1099       Name = "repomd-src";
1100       Label = "RepoMD src tree";
1101    }   
1102 };
1103
1104 rpmSLTypeRpm _apt_rpmType;
1105 rpmSLTypeSrpm _apt_rpmSrcType;
1106 rpmSLTypeRpmDir _apt_rpmDirType;
1107 rpmSLTypeSrpmDir _apt_rpmSrcDirType;
1108 rpmSLTypeRepomd _apt_repomdType;
1109 rpmSLTypeRepomdSrc _apt_repomdSrcType;
1110                                                                         /*}}}*/
1111 // Index File types for rpm                                             /*{{{*/
1112 class rpmIFTypeSrc : public pkgIndexFile::Type
1113 {
1114    public:
1115    
1116    rpmIFTypeSrc() {Label = "RPM Source Index";};
1117 };
1118 class rpmIFTypePkg : public pkgIndexFile::Type
1119 {
1120    public:
1121    
1122    virtual pkgRecords::Parser *CreatePkgParser(pkgCache::PkgFileIterator File) const
1123    {
1124       return new rpmRecordParser(File.FileName(),*File.Cache());
1125    };
1126    rpmIFTypePkg() {Label = "RPM Package Index";};
1127 };
1128 class rpmIFTypeDatabase : public pkgIndexFile::Type
1129 {
1130    public:
1131    
1132    virtual pkgRecords::Parser *CreatePkgParser(pkgCache::PkgFileIterator File) const
1133    {
1134       return new rpmRecordParser(File.FileName(),*File.Cache());
1135    };
1136    rpmIFTypeDatabase() {Label = "RPM Database";};
1137 };
1138 static rpmIFTypeSrc _apt_Src;
1139 static rpmIFTypePkg _apt_Pkg;
1140 static rpmIFTypeDatabase _apt_DB;
1141
1142 const pkgIndexFile::Type *rpmSrcListIndex::GetType() const
1143 {
1144    return &_apt_Src;
1145 }
1146 const pkgIndexFile::Type *rpmPkgListIndex::GetType() const
1147 {
1148    return &_apt_Pkg;
1149 }
1150 const pkgIndexFile::Type *rpmSrcDirIndex::GetType() const
1151 {
1152    return &_apt_Src;
1153 }
1154 const pkgIndexFile::Type *rpmPkgDirIndex::GetType() const
1155 {
1156    return &_apt_Pkg;
1157 }
1158 const pkgIndexFile::Type *rpmSinglePkgIndex::GetType() const
1159 {
1160    return &_apt_Pkg;
1161 }
1162 const pkgIndexFile::Type *rpmSingleSrcIndex::GetType() const
1163 {
1164    return &_apt_Src;
1165 }
1166 const pkgIndexFile::Type *rpmDatabaseIndex::GetType() const
1167 {
1168    return &_apt_DB;
1169 }
1170 const pkgIndexFile::Type *rpmRepomdPkgIndex::GetType() const
1171 {
1172    return &_apt_Pkg;
1173 }
1174 const pkgIndexFile::Type *rpmRepomdSrcIndex::GetType() const
1175 {
1176    return &_apt_Src;
1177 }
1178
1179                                                                         /*}}}*/
1180 #endif /* HAVE_RPM */
1181
1182 // vim:sts=3:sw=3