]> git.saurik.com Git - apt.git/commitdiff
Merge remote-tracking branch 'donkult/debian/experimental' into debian/experimental
authorMichael Vogt <mvo@debian.org>
Wed, 14 May 2014 16:04:48 +0000 (18:04 +0200)
committerMichael Vogt <mvo@debian.org>
Wed, 14 May 2014 16:04:48 +0000 (18:04 +0200)
Conflicts:
apt-pkg/acquire-item.cc
apt-pkg/acquire-item.h

1  2 
apt-pkg/acquire-item.cc
apt-pkg/acquire-item.h

diff --combined apt-pkg/acquire-item.cc
index b3d67de2cd3cec95ff2a4d55ff7ad8e3c6b0201e,ce19e5486ca15e6bb82ce8ad319b3c91d383cc2c..99013d649eb591d765a7803ca741e32912696e67
  
  using namespace std;
  
+ static void printHashSumComparision(std::string const &URI, HashStringList const &Expected, HashStringList const &Actual) /*{{{*/
+ {
+    if (_config->FindB("Debug::Acquire::HashSumMismatch", false) == false)
+       return;
+    std::cerr << std::endl << URI << ":" << std::endl << " Expected Hash: " << std::endl;
+    for (HashStringList::const_iterator hs = Expected.begin(); hs != Expected.end(); ++hs)
+       std::cerr <<  "\t- " << hs->toStr() << std::endl;
+    std::cerr << " Actual Hash: " << std::endl;
+    for (HashStringList::const_iterator hs = Actual.begin(); hs != Actual.end(); ++hs)
+       std::cerr <<  "\t- " << hs->toStr() << std::endl;
+ }
+                                                                       /*}}}*/
  // Acquire::Item::Item - Constructor                                  /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- pkgAcquire::Item::Item(pkgAcquire *Owner) : Owner(Owner), FileSize(0),
-                        PartialSize(0), Mode(0), ID(0), Complete(false), 
-                        Local(false), QueueCounter(0), 
-                        ExpectedAdditionalItems(0)
+ pkgAcquire::Item::Item(pkgAcquire *Owner, HashStringList const &ExpectedHashes) :
+    Owner(Owner), FileSize(0), PartialSize(0), Mode(0), ID(0), Complete(false),
+    Local(false), QueueCounter(0), ExpectedAdditionalItems(0),
+    ExpectedHashes(ExpectedHashes)
  {
     Owner->Add(this);
     Status = StatIdle;
@@@ -118,7 -129,7 +129,7 @@@ void pkgAcquire::Item::Start(string /*M
  // Acquire::Item::Done - Item downloaded OK                           /*{{{*/
  // ---------------------------------------------------------------------
  /* */
- void pkgAcquire::Item::Done(string Message,unsigned long long Size,string /*Hash*/,
+ void pkgAcquire::Item::Done(string Message,unsigned long long Size,HashStringList const &/*Hash*/,
                            pkgAcquire::MethodConfig * /*Cnf*/)
  {
     // We just downloaded something..
@@@ -229,8 -240,8 +240,8 @@@ void pkgAcquire::Item::ReportMirrorFail
     possibly query additional files */
  pkgAcqSubIndex::pkgAcqSubIndex(pkgAcquire *Owner, string const &URI,
                                 string const &URIDesc, string const &ShortDesc,
-                                HashString const &ExpectedHash)
-    : Item(Owner), ExpectedHash(ExpectedHash)
+                                HashStringList const &ExpectedHashes)
+    : Item(Owner, ExpectedHashes)
  {
     /* XXX: Beware: Currently this class does nothing (of value) anymore ! */
     Debug = _config->FindB("Debug::pkgAcquire::SubIndex",false);
  // AcqSubIndex::Custom600Headers - Insert custom request headers      /*{{{*/
  // ---------------------------------------------------------------------
  /* The only header we use is the last-modified header. */
- string pkgAcqSubIndex::Custom600Headers()
+ string pkgAcqSubIndex::Custom600Headers() const
  {
     string Final = _config->FindDir("Dir::State::lists");
     Final += URItoFileName(Desc.URI);
@@@ -275,7 -286,7 +286,7 @@@ void pkgAcqSubIndex::Failed(string Mess
     // No good Index is provided
  }
                                                                        /*}}}*/
- void pkgAcqSubIndex::Done(string Message,unsigned long long Size,string Md5Hash,      /*{{{*/
+ void pkgAcqSubIndex::Done(string Message,unsigned long long Size,HashStringList const &Hashes,        /*{{{*/
                           pkgAcquire::MethodConfig *Cnf)
  {
     if(Debug)
        return;
     }
  
-    Item::Done(Message,Size,Md5Hash,Cnf);
+    Item::Done(Message, Size, Hashes, Cnf);
  
     string FinalFile = _config->FindDir("Dir::State::lists")+URItoFileName(Desc.URI);
  
@@@ -343,10 -354,11 +354,10 @@@ bool pkgAcqSubIndex::ParseIndex(string 
   * the original packages file
   */
  pkgAcqDiffIndex::pkgAcqDiffIndex(pkgAcquire *Owner,
-                                  IndexTarget const *Target,
-                                HashString ExpectedHash,
 -      IndexTarget const * const Target,
 -      HashStringList const &ExpectedHashes,
 -      indexRecords *MetaIndexParser)
 -   : Item(Owner, ExpectedHashes), Target(Target),
 -     MetaIndexParser(MetaIndexParser)
++                                 IndexTarget const * const Target,
++                               HashStringList const &ExpectedHashes,
 +                                 indexRecords *MetaIndexParser)
-    : pkgAcqBaseIndex(Owner, Target, ExpectedHash, MetaIndexParser)
++   : pkgAcqBaseIndex(Owner, Target, ExpectedHashes, MetaIndexParser)
  {
     
     Debug = _config->FindB("Debug::pkgAcquire::Diffs",false);
  // AcqIndex::Custom600Headers - Insert custom request headers         /*{{{*/
  // ---------------------------------------------------------------------
  /* The only header we use is the last-modified header. */
- string pkgAcqDiffIndex::Custom600Headers()
+ string pkgAcqDiffIndex::Custom600Headers() const
  {
     string Final = _config->FindDir("Dir::State::lists");
     Final += URItoFileName(RealURI) + string(".IndexDiff");
@@@ -444,8 -456,8 +455,8 @@@ bool pkgAcqDiffIndex::ParseDiffIndex(st
            std::clog << "Package file is up-to-date" << std::endl;
         // list cleanup needs to know that this file as well as the already
         // present index is ours, so we create an empty diff to save it for us
-        new pkgAcqIndexDiffs(Owner, Target, ExpectedHash, MetaIndexParser, 
 -       new pkgAcqIndexDiffs(Owner, RealURI, Description, Desc.ShortDesc,
 -             ExpectedHashes, ServerSha1, available_patches);
++       new pkgAcqIndexDiffs(Owner, Target, ExpectedHashes, MetaIndexParser, 
 +                              ServerSha1, available_patches);
         return true;
        }
        else
         }
  
         if (pdiff_merge == false)
 -          new pkgAcqIndexDiffs(Owner, RealURI, Description, Desc.ShortDesc,
 -                ExpectedHashes, ServerSha1, available_patches);
 -       else
 +         {
-           new pkgAcqIndexDiffs(Owner, Target, ExpectedHash, MetaIndexParser,
++          new pkgAcqIndexDiffs(Owner, Target, ExpectedHashes, MetaIndexParser,
 +                                 ServerSha1, available_patches);
 +         }
 +         else
         {
            std::vector<pkgAcqIndexMergeDiffs*> *diffs = new std::vector<pkgAcqIndexMergeDiffs*>(available_patches.size());
            for(size_t i = 0; i < available_patches.size(); ++i)
 -             (*diffs)[i] = new pkgAcqIndexMergeDiffs(Owner, RealURI, Description, Desc.ShortDesc, ExpectedHashes,
 -                   available_patches[i], diffs);
 +             (*diffs)[i] = new pkgAcqIndexMergeDiffs(Owner, Target,
-                                                        ExpectedHash,
++                                                       ExpectedHashes,
 +                                                       MetaIndexParser,
 +                                                       available_patches[i],
 +                                                       diffs);
         }
  
         Complete = false;
@@@ -566,20 -573,20 +577,20 @@@ void pkgAcqDiffIndex::Failed(string Mes
        std::clog << "pkgAcqDiffIndex failed: " << Desc.URI << " with " << Message << std::endl
                << "Falling back to normal index file acquire" << std::endl;
  
-    new pkgAcqIndex(Owner, Target, ExpectedHash, MetaIndexParser);
+    new pkgAcqIndex(Owner, Target, ExpectedHashes, MetaIndexParser);
  
     Complete = false;
     Status = StatDone;
     Dequeue();
  }
                                                                        /*}}}*/
- void pkgAcqDiffIndex::Done(string Message,unsigned long long Size,string Md5Hash,     /*{{{*/
+ void pkgAcqDiffIndex::Done(string Message,unsigned long long Size,HashStringList const &Hashes,       /*{{{*/
                           pkgAcquire::MethodConfig *Cnf)
  {
     if(Debug)
        std::clog << "pkgAcqDiffIndex::Done(): " << Desc.URI << std::endl;
  
-    Item::Done(Message,Size,Md5Hash,Cnf);
+    Item::Done(Message, Size, Hashes, Cnf);
  
     string FinalFile;
     FinalFile = _config->FindDir("Dir::State::lists")+URItoFileName(RealURI);
   * for each diff and the index
   */
  pkgAcqIndexDiffs::pkgAcqIndexDiffs(pkgAcquire *Owner,
 -                                 string URI,string URIDesc,string ShortDesc,
 -                                 HashStringList const &ExpectedHashes,
 +                                   struct IndexTarget const * const Target,
-                                    HashString ExpectedHash,
++                                   HashStringList const &ExpectedHashes,
 +                                   indexRecords *MetaIndexParser,
                                   string ServerSha1,
                                   vector<DiffInfo> diffs)
-    : pkgAcqBaseIndex(Owner, Target, ExpectedHash, MetaIndexParser),
 -   : Item(Owner, ExpectedHashes), RealURI(URI),
++   : pkgAcqBaseIndex(Owner, Target, ExpectedHashes, MetaIndexParser),
       available_patches(diffs), ServerSha1(ServerSha1)
  {
     
     DestFile = _config->FindDir("Dir::State::lists") + "partial/";
 -   DestFile += URItoFileName(URI);
 +   DestFile += URItoFileName(Target->URI);
  
     Debug = _config->FindB("Debug::pkgAcquire::Diffs",false);
  
 -   Description = URIDesc;
 +   RealURI = Target->URI;
     Desc.Owner = this;
 -   Desc.ShortDesc = ShortDesc;
 +   Description = Target->Description;
 +   Desc.ShortDesc = Target->ShortDesc;
  
     if(available_patches.empty() == true)
     {
@@@ -646,7 -651,7 +657,7 @@@ void pkgAcqIndexDiffs::Failed(string Me
     if(Debug)
        std::clog << "pkgAcqIndexDiffs failed: " << Desc.URI << " with " << Message << std::endl
                << "Falling back to normal index file acquire" << std::endl;
-    new pkgAcqIndex(Owner, Target, ExpectedHash, MetaIndexParser);
 -   new pkgAcqIndex(Owner, RealURI, Description,Desc.ShortDesc, HashSums());
++   new pkgAcqIndex(Owner, Target, ExpectedHashes, MetaIndexParser);
     Finish();
  }
                                                                        /*}}}*/
@@@ -660,7 -665,7 +671,7 @@@ void pkgAcqIndexDiffs::Finish(bool allD
        DestFile = _config->FindDir("Dir::State::lists");
        DestFile += URItoFileName(RealURI);
  
-       if(!ExpectedHash.empty() && !ExpectedHash.VerifyFile(DestFile))
+       if(HashSums().usable() && !HashSums().VerifyFile(DestFile))
        {
         RenameOnError(HashSumMismatch);
         Dequeue();
@@@ -738,13 -743,13 +749,13 @@@ bool pkgAcqIndexDiffs::QueueNextDiff(
     return true;
  }
                                                                        /*}}}*/
- void pkgAcqIndexDiffs::Done(string Message,unsigned long long Size,string Md5Hash,    /*{{{*/
+ void pkgAcqIndexDiffs::Done(string Message,unsigned long long Size, HashStringList const &Hashes,     /*{{{*/
                            pkgAcquire::MethodConfig *Cnf)
  {
     if(Debug)
        std::clog << "pkgAcqIndexDiffs::Done(): " << Desc.URI << std::endl;
  
-    Item::Done(Message,Size,Md5Hash,Cnf);
+    Item::Done(Message, Size, Hashes, Cnf);
  
     string FinalFile;
     FinalFile = _config->FindDir("Dir::State::lists")+URItoFileName(RealURI);
  
        // see if there is more to download
        if(available_patches.empty() == false) {
 -       new pkgAcqIndexDiffs(Owner, RealURI, Description, Desc.ShortDesc,
 -                            HashSums(), ServerSha1, available_patches);
 +       new pkgAcqIndexDiffs(Owner, Target,
-                             ExpectedHash, MetaIndexParser,
++                            ExpectedHashes, MetaIndexParser,
 +                              ServerSha1, available_patches);
         return Finish();
        } else 
         return Finish(true);
                                                                        /*}}}*/
  // AcqIndexMergeDiffs::AcqIndexMergeDiffs - Constructor                       /*{{{*/
  pkgAcqIndexMergeDiffs::pkgAcqIndexMergeDiffs(pkgAcquire *Owner,
 -                                 string const &URI, string const &URIDesc,
 -                                 string const &ShortDesc, HashStringList const &ExpectedHashes,
 -                                 DiffInfo const &patch,
 -                                 std::vector<pkgAcqIndexMergeDiffs*> const * const allPatches)
 -   : Item(Owner, ExpectedHashes), RealURI(URI),
 -     patch(patch),allPatches(allPatches), State(StateFetchDiff)
 +                                             struct IndexTarget const * const Target,
-                                              HashString ExpectedHash,
++                                             HashStringList const &ExpectedHashes,
 +                                             indexRecords *MetaIndexParser,
 +                                             DiffInfo const &patch,
 +                                             std::vector<pkgAcqIndexMergeDiffs*> const * const allPatches)
-    : pkgAcqBaseIndex(Owner, Target, ExpectedHash, MetaIndexParser),
++   : pkgAcqBaseIndex(Owner, Target, ExpectedHashes, MetaIndexParser),
 +     patch(patch), allPatches(allPatches), State(StateFetchDiff)
  {
  
     DestFile = _config->FindDir("Dir::State::lists") + "partial/";
 -   DestFile += URItoFileName(URI);
 +   DestFile += URItoFileName(Target->URI);
  
     Debug = _config->FindB("Debug::pkgAcquire::Diffs",false);
  
 -   Description = URIDesc;
 +   RealURI = Target->URI;
     Desc.Owner = this;
 -   Desc.ShortDesc = ShortDesc;
 +   Description = Target->Description;
 +   Desc.ShortDesc = Target->ShortDesc;
  
     Desc.URI = RealURI + ".diff/" + patch.file + ".gz";
     Desc.Description = Description + " " + patch.file + string(".pdiff");
@@@ -845,16 -847,17 +856,16 @@@ void pkgAcqIndexMergeDiffs::Failed(stri
     // first failure means we should fallback
     State = StateErrorDiff;
     std::clog << "Falling back to normal index file acquire" << std::endl;
-    new pkgAcqIndex(Owner, Target, ExpectedHash, MetaIndexParser);
 -   new pkgAcqIndex(Owner, RealURI, Description,Desc.ShortDesc,
 -                 ExpectedHashes);
++   new pkgAcqIndex(Owner, Target, ExpectedHashes, MetaIndexParser);
  }
                                                                        /*}}}*/
- void pkgAcqIndexMergeDiffs::Done(string Message,unsigned long long Size,string Md5Hash,       /*{{{*/
+ void pkgAcqIndexMergeDiffs::Done(string Message,unsigned long long Size,HashStringList const &Hashes, /*{{{*/
                            pkgAcquire::MethodConfig *Cnf)
  {
     if(Debug)
        std::clog << "pkgAcqIndexMergeDiffs::Done(): " << Desc.URI << std::endl;
  
-    Item::Done(Message,Size,Md5Hash,Cnf);
+    Item::Done(Message,Size,Hashes,Cnf);
  
     string const FinalFile = _config->FindDir("Dir::State::lists") + URItoFileName(RealURI);
  
     else if (State == StateApplyDiff)
     {
        // see if we really got the expected file
-       if(!ExpectedHash.empty() && !ExpectedHash.VerifyFile(DestFile))
+       if(ExpectedHashes.usable() && !ExpectedHashes.VerifyFile(DestFile))
        {
         RenameOnError(HashSumMismatch);
         return;
     instantiated to fetch the revision file */   
  pkgAcqIndex::pkgAcqIndex(pkgAcquire *Owner,
                         string URI,string URIDesc,string ShortDesc,
-                        HashString ExpectedHash, string comprExt)
 -                       HashStringList const &ExpectedHashes, string comprExt)
 -   : Item(Owner, ExpectedHashes), RealURI(URI), Target(0),
 -     MetaIndexParser(0)
++                       HashStringList const  &ExpectedHash, string comprExt)
 +   : pkgAcqBaseIndex(Owner, NULL, ExpectedHash, NULL), RealURI(URI)
  {
     if(comprExt.empty() == true)
     {
  
     Init(URI, URIDesc, ShortDesc);
  }
 -pkgAcqIndex::pkgAcqIndex(pkgAcquire *Owner, IndexTarget const * const Target,
 -                       HashStringList const &ExpectedHashes, indexRecords *MetaIndexParser)
 -   : Item(Owner, ExpectedHashes), RealURI(Target->URI)
 +pkgAcqIndex::pkgAcqIndex(pkgAcquire *Owner, IndexTarget const *Target,
-                        HashString const &ExpectedHash, indexRecords *MetaIndexParser)
++                       HashStringList const &ExpectedHash, 
++                         indexRecords *MetaIndexParser)
 +   : pkgAcqBaseIndex(Owner, Target, ExpectedHash, MetaIndexParser), 
 +     RealURI(Target->URI)
  {
     // autoselect the compression method
     std::vector<std::string> types = APT::Configuration::getCompressionTypes();
     CompressionExtension = "";
-    if (ExpectedHash.empty() == false)
+    if (ExpectedHashes.usable())
     {
        for (std::vector<std::string>::const_iterator t = types.begin(); t != types.end(); ++t)
         if (*t == "uncompressed" || MetaIndexParser->Exists(string(Target->MetaKey).append(".").append(*t)) == true)
     else
       Verify = true;
  
 -   // we need this in Init()
 -   this->Target = Target;
 -   this->MetaIndexParser = MetaIndexParser;
 -
     Init(Target->URI, Target->Description, Target->ShortDesc);
  }
                                                                        /*}}}*/
@@@ -1018,7 -1025,7 +1030,7 @@@ void pkgAcqIndex::Init(string const &UR
  // AcqIndex::Custom600Headers - Insert custom request headers         /*{{{*/
  // ---------------------------------------------------------------------
  /* The only header we use is the last-modified header. */
- string pkgAcqIndex::Custom600Headers()
+ string pkgAcqIndex::Custom600Headers() const
  {
     string Final = _config->FindDir("Dir::State::lists");
     Final += URItoFileName(RealURI);
@@@ -1064,22 -1071,17 +1076,17 @@@ void pkgAcqIndex::Failed(string Message
     to the uncompressed version of the file. If this is so the file
     is copied into the partial directory. In all other cases the file
     is decompressed with a gzip uri. */
- void pkgAcqIndex::Done(string Message,unsigned long long Size,string Hash,
+ void pkgAcqIndex::Done(string Message,unsigned long long Size,HashStringList const &Hashes,
                       pkgAcquire::MethodConfig *Cfg)
  {
-    Item::Done(Message,Size,Hash,Cfg);
+    Item::Done(Message,Size,Hashes,Cfg);
  
     if (Decompression == true)
     {
-       if (_config->FindB("Debug::pkgAcquire::Auth", false))
-       {
-          std::cerr << std::endl << RealURI << ": Computed Hash: " << Hash;
-          std::cerr << "  Expected Hash: " << ExpectedHash.toStr() << std::endl;
-       }
-       if (!ExpectedHash.empty() && ExpectedHash.toStr() != Hash)
+       if (ExpectedHashes.usable() && ExpectedHashes != Hashes)
        {
         RenameOnError(HashSumMismatch);
+        printHashSumComparision(RealURI, ExpectedHashes, Hashes);
           return;
        }
  
  /* The Translation file is added to the queue */
  pkgAcqIndexTrans::pkgAcqIndexTrans(pkgAcquire *Owner,
                            string URI,string URIDesc,string ShortDesc) 
-   : pkgAcqIndex(Owner, URI, URIDesc, ShortDesc, HashString(), "")
+   : pkgAcqIndex(Owner, URI, URIDesc, ShortDesc, HashStringList(), "")
  {
  }
- pkgAcqIndexTrans::pkgAcqIndexTrans(pkgAcquire *Owner, IndexTarget const *Target,
-                        HashString const &ExpectedHash, indexRecords *MetaIndexParser)
-   : pkgAcqIndex(Owner, Target, ExpectedHash, MetaIndexParser)
+ pkgAcqIndexTrans::pkgAcqIndexTrans(pkgAcquire *Owner, IndexTarget const * const Target,
+                        HashStringList const &ExpectedHashes, indexRecords *MetaIndexParser)
+   : pkgAcqIndex(Owner, Target, ExpectedHashes, MetaIndexParser)
  {
     // load the filesize
     indexRecords::checkSum *Record = MetaIndexParser->Lookup(string(Target->MetaKey));
                                                                        /*}}}*/
  // AcqIndexTrans::Custom600Headers - Insert custom request headers    /*{{{*/
  // ---------------------------------------------------------------------
- string pkgAcqIndexTrans::Custom600Headers()
+ string pkgAcqIndexTrans::Custom600Headers() const
  {
     string Final = _config->FindDir("Dir::State::lists");
     Final += URItoFileName(RealURI);
@@@ -1260,7 -1262,7 +1267,7 @@@ pkgAcqMetaSig::pkgAcqMetaSig(pkgAcquir
                             string MetaIndexShortDesc,
                             const vector<IndexTarget*>* IndexTargets,
                             indexRecords* MetaIndexParser) :
-    Item(Owner), RealURI(URI), MetaIndexURI(MetaIndexURI),
+    Item(Owner, HashStringList()), RealURI(URI), MetaIndexURI(MetaIndexURI),
     MetaIndexURIDesc(MetaIndexURIDesc), MetaIndexShortDesc(MetaIndexShortDesc),
     MetaIndexParser(MetaIndexParser), IndexTargets(IndexTargets)
  {
@@@ -1313,7 -1315,7 +1320,7 @@@ pkgAcqMetaSig::~pkgAcqMetaSig()                                         /*
  // pkgAcqMetaSig::Custom600Headers - Insert custom request headers    /*{{{*/
  // ---------------------------------------------------------------------
  /* The only header we use is the last-modified header. */
- string pkgAcqMetaSig::Custom600Headers()
+ string pkgAcqMetaSig::Custom600Headers() const
  {
     struct stat Buf;
     if (stat(LastGoodSig.c_str(),&Buf) != 0)
     return "\nIndex-File: true\nLast-Modified: " + TimeRFC1123(Buf.st_mtime);
  }
  
- void pkgAcqMetaSig::Done(string Message,unsigned long long Size,string MD5,
+ void pkgAcqMetaSig::Done(string Message,unsigned long long Size, HashStringList const &Hashes,
                         pkgAcquire::MethodConfig *Cfg)
  {
-    Item::Done(Message,Size,MD5,Cfg);
+    Item::Done(Message, Size, Hashes, Cfg);
  
     string FileName = LookupTag(Message,"Filename");
     if (FileName.empty() == true)
@@@ -1405,9 -1407,9 +1412,9 @@@ void pkgAcqMetaSig::Failed(string Messa
  pkgAcqMetaIndex::pkgAcqMetaIndex(pkgAcquire *Owner,                   /*{{{*/
                                 string URI,string URIDesc,string ShortDesc,
                                 string SigFile,
-                                const vector<struct IndexTarget*>* IndexTargets,
+                                const vector<IndexTarget*>* IndexTargets,
                                 indexRecords* MetaIndexParser) :
-    Item(Owner), RealURI(URI), SigFile(SigFile), IndexTargets(IndexTargets),
+    Item(Owner, HashStringList()), RealURI(URI), SigFile(SigFile), IndexTargets(IndexTargets),
     MetaIndexParser(MetaIndexParser), AuthPass(false), IMSHit(false)
  {
     DestFile = _config->FindDir("Dir::State::lists") + "partial/";
  // pkgAcqMetaIndex::Custom600Headers - Insert custom request headers  /*{{{*/
  // ---------------------------------------------------------------------
  /* The only header we use is the last-modified header. */
- string pkgAcqMetaIndex::Custom600Headers()
+ string pkgAcqMetaIndex::Custom600Headers() const
  {
     string Final = _config->FindDir("Dir::State::lists");
     Final += URItoFileName(RealURI);
     return "\nIndex-File: true\nLast-Modified: " + TimeRFC1123(Buf.st_mtime);
  }
                                                                        /*}}}*/
- void pkgAcqMetaIndex::Done(string Message,unsigned long long Size,string Hash,        /*{{{*/
+ void pkgAcqMetaIndex::Done(string Message,unsigned long long Size,HashStringList const &Hashes,       /*{{{*/
                           pkgAcquire::MethodConfig *Cfg)
  {
-    Item::Done(Message,Size,Hash,Cfg);
+    Item::Done(Message,Size,Hashes,Cfg);
  
     // MetaIndexes are done in two passes: one to download the
     // metaindex with an appropriate method, and a second to verify it
@@@ -1614,12 -1616,12 +1621,12 @@@ void pkgAcqMetaIndex::QueueIndexes(boo
     // at this point the real Items are loaded in the fetcher
     ExpectedAdditionalItems = 0;
  
-    for (vector <struct IndexTarget*>::const_iterator Target = IndexTargets->begin();
+    for (vector <IndexTarget*>::const_iterator Target = IndexTargets->begin();
          Target != IndexTargets->end();
          ++Target)
     {
-       HashString ExpectedIndexHash;
-       indexRecords::checkSum *Record = MetaIndexParser->Lookup((*Target)->MetaKey);
+       HashStringList ExpectedIndexHashes;
+       const indexRecords::checkSum *Record = MetaIndexParser->Lookup((*Target)->MetaKey);
        bool compressedAvailable = false;
        if (Record == NULL)
        {
        }
        else
        {
-        ExpectedIndexHash = Record->Hash;
+        ExpectedIndexHashes = Record->Hashes;
         if (_config->FindB("Debug::pkgAcquire::Auth", false))
         {
-           std::cerr << "Queueing: " << (*Target)->URI << std::endl;
-           std::cerr << "Expected Hash: " << ExpectedIndexHash.toStr() << std::endl;
+           std::cerr << "Queueing: " << (*Target)->URI << std::endl
+              << "Expected Hash:" << std::endl;
+           for (HashStringList::const_iterator hs = ExpectedIndexHashes.begin(); hs != ExpectedIndexHashes.end(); ++hs)
+              std::cerr <<  "\t- " << hs->toStr() << std::endl;
            std::cerr << "For: " << Record->MetaKeyFilename << std::endl;
         }
-        if (verify == true && ExpectedIndexHash.empty() == true && (*Target)->IsOptional() == false)
+        if (verify == true && ExpectedIndexHashes.empty() == true && (*Target)->IsOptional() == false)
         {
            Status = StatAuthError;
            strprintf(ErrorText, _("Unable to find hash sum for '%s' in Release file"), (*Target)->MetaKey.c_str());
        {
         if ((*Target)->IsSubIndex() == true)
            new pkgAcqSubIndex(Owner, (*Target)->URI, (*Target)->Description,
-                               (*Target)->ShortDesc, ExpectedIndexHash);
+                               (*Target)->ShortDesc, ExpectedIndexHashes);
         else if (transInRelease == false || Record != NULL || compressedAvailable == true)
         {
            if (_config->FindB("Acquire::PDiffs",true) == true && transInRelease == true &&
                MetaIndexParser->Exists((*Target)->MetaKey + ".diff/Index") == true)
-              new pkgAcqDiffIndex(Owner, *Target, ExpectedIndexHash, MetaIndexParser);
+              new pkgAcqDiffIndex(Owner, *Target, ExpectedIndexHashes, MetaIndexParser);
            else
-              new pkgAcqIndexTrans(Owner, *Target, ExpectedIndexHash, MetaIndexParser);
+              new pkgAcqIndexTrans(Owner, *Target, ExpectedIndexHashes, MetaIndexParser);
         }
         continue;
        }
           instead, but passing the required info to it is to much hassle */
        if(_config->FindB("Acquire::PDiffs",true) == true && (verify == false ||
          MetaIndexParser->Exists((*Target)->MetaKey + ".diff/Index") == true))
-        new pkgAcqDiffIndex(Owner, *Target, ExpectedIndexHash, MetaIndexParser);
+        new pkgAcqDiffIndex(Owner, *Target, ExpectedIndexHashes, MetaIndexParser);
        else
-        new pkgAcqIndex(Owner, *Target, ExpectedIndexHash, MetaIndexParser);
+        new pkgAcqIndex(Owner, *Target, ExpectedIndexHashes, MetaIndexParser);
     }
  }
                                                                        /*}}}*/
@@@ -1834,7 -1838,7 +1843,7 @@@ pkgAcqMetaClearSig::pkgAcqMetaClearSig(
                string const &URI, string const &URIDesc, string const &ShortDesc,
                string const &MetaIndexURI, string const &MetaIndexURIDesc, string const &MetaIndexShortDesc,
                string const &MetaSigURI, string const &MetaSigURIDesc, string const &MetaSigShortDesc,
-               const vector<struct IndexTarget*>* IndexTargets,
+               const vector<IndexTarget*>* IndexTargets,
                indexRecords* MetaIndexParser) :
        pkgAcqMetaIndex(Owner, URI, URIDesc, ShortDesc, "", IndexTargets, MetaIndexParser),
        MetaIndexURI(MetaIndexURI), MetaIndexURIDesc(MetaIndexURIDesc), MetaIndexShortDesc(MetaIndexShortDesc),
@@@ -1870,7 -1874,7 +1879,7 @@@ pkgAcqMetaClearSig::~pkgAcqMetaClearSig
  // pkgAcqMetaClearSig::Custom600Headers - Insert custom request headers       /*{{{*/
  // ---------------------------------------------------------------------
  // FIXME: this can go away once the InRelease file is used widely
- string pkgAcqMetaClearSig::Custom600Headers()
+ string pkgAcqMetaClearSig::Custom600Headers() const
  {
     string Final = _config->FindDir("Dir::State::lists");
     Final += URItoFileName(RealURI);
@@@ -1919,7 -1923,7 +1928,7 @@@ void pkgAcqMetaClearSig::Failed(string 
  pkgAcqArchive::pkgAcqArchive(pkgAcquire *Owner,pkgSourceList *Sources,
                             pkgRecords *Recs,pkgCache::VerIterator const &Version,
                             string &StoreFilename) :
-                Item(Owner), Version(Version), Sources(Sources), Recs(Recs), 
+                Item(Owner, HashStringList()), Version(Version), Sources(Sources), Recs(Recs), 
                 StoreFilename(StoreFilename), Vf(Version.FileList()), 
               Trusted(false)
  {
     checking later. */
  bool pkgAcqArchive::QueueNext()
  {
-    string const ForceHash = _config->Find("Acquire::ForceHash");
     for (; Vf.end() == false; ++Vf)
     {
        // Ignore not source sources
        pkgRecords::Parser &Parse = Recs->Lookup(Vf);
        if (_error->PendingError() == true)
         return false;
-       
        string PkgFile = Parse.FileName();
-       if (ForceHash.empty() == false)
-       {
-        if(stringcasecmp(ForceHash, "sha512") == 0)
-           ExpectedHash = HashString("SHA512", Parse.SHA512Hash());
-        else if(stringcasecmp(ForceHash, "sha256") == 0)
-           ExpectedHash = HashString("SHA256", Parse.SHA256Hash());
-        else if (stringcasecmp(ForceHash, "sha1") == 0)
-           ExpectedHash = HashString("SHA1", Parse.SHA1Hash());
-        else
-           ExpectedHash = HashString("MD5Sum", Parse.MD5Hash());
-       }
-       else
-       {
-        string Hash;
-        if ((Hash = Parse.SHA512Hash()).empty() == false)
-           ExpectedHash = HashString("SHA512", Hash);
-        else if ((Hash = Parse.SHA256Hash()).empty() == false)
-           ExpectedHash = HashString("SHA256", Hash);
-        else if ((Hash = Parse.SHA1Hash()).empty() == false)
-           ExpectedHash = HashString("SHA1", Hash);
-        else
-           ExpectedHash = HashString("MD5Sum", Parse.MD5Hash());
-       }
+       ExpectedHashes = Parse.Hashes();
        if (PkgFile.empty() == true)
         return _error->Error(_("The package index files are corrupted. No Filename: "
                              "field for package %s."),
  // AcqArchive::Done - Finished fetching                                       /*{{{*/
  // ---------------------------------------------------------------------
  /* */
- void pkgAcqArchive::Done(string Message,unsigned long long Size,string CalcHash,
+ void pkgAcqArchive::Done(string Message,unsigned long long Size, HashStringList const &CalcHashes,
                         pkgAcquire::MethodConfig *Cfg)
  {
-    Item::Done(Message,Size,CalcHash,Cfg);
+    Item::Done(Message, Size, CalcHashes, Cfg);
     
     // Check the size
     if (Size != Version->Size)
        RenameOnError(SizeMismatch);
        return;
     }
-    
-    // Check the hash
     // FIXME: could this empty() check impose *any* sort of security issue?
-    if(ExpectedHash.empty() == false && ExpectedHash.toStr() != CalcHash)
+    if(ExpectedHashes.usable() && ExpectedHashes != CalcHashes)
     {
        RenameOnError(HashSumMismatch);
+       printHashSumComparision(DestFile, ExpectedHashes, CalcHashes);
        return;
     }
  
@@@ -2224,7 -2206,7 +2211,7 @@@ void pkgAcqArchive::Failed(string Messa
                                                                        /*}}}*/
  // AcqArchive::IsTrusted - Determine whether this archive comes from a trusted source /*{{{*/
  // ---------------------------------------------------------------------
- APT_PURE bool pkgAcqArchive::IsTrusted()
+ APT_PURE bool pkgAcqArchive::IsTrusted() const
  {
     return Trusted;
  }
@@@ -2243,11 -2225,11 +2230,11 @@@ void pkgAcqArchive::Finished(
  // AcqFile::pkgAcqFile - Constructor                                  /*{{{*/
  // ---------------------------------------------------------------------
  /* The file is added to the queue */
- pkgAcqFile::pkgAcqFile(pkgAcquire *Owner,string URI,string Hash,
+ pkgAcqFile::pkgAcqFile(pkgAcquire *Owner,string URI, HashStringList const &Hashes,
                       unsigned long long Size,string Dsc,string ShortDesc,
                       const string &DestDir, const string &DestFilename,
                         bool IsIndexFile) :
-                        Item(Owner), ExpectedHash(Hash), IsIndexFile(IsIndexFile)
+                        Item(Owner, Hashes), IsIndexFile(IsIndexFile)
  {
     Retries = _config->FindI("Acquire::Retries",0);
     
  // AcqFile::Done - Item downloaded OK                                 /*{{{*/
  // ---------------------------------------------------------------------
  /* */
- void pkgAcqFile::Done(string Message,unsigned long long Size,string CalcHash,
+ void pkgAcqFile::Done(string Message,unsigned long long Size,HashStringList const &CalcHashes,
                      pkgAcquire::MethodConfig *Cnf)
  {
-    Item::Done(Message,Size,CalcHash,Cnf);
+    Item::Done(Message,Size,CalcHashes,Cnf);
  
     // Check the hash
-    if(!ExpectedHash.empty() && ExpectedHash.toStr() != CalcHash)
+    if(ExpectedHashes.usable() && ExpectedHashes != CalcHashes)
     {
        RenameOnError(HashSumMismatch);
+       printHashSumComparision(DestFile, ExpectedHashes, CalcHashes);
        return;
     }
     
@@@ -2363,7 -2346,7 +2351,7 @@@ void pkgAcqFile::Failed(string Message,
  // AcqIndex::Custom600Headers - Insert custom request headers         /*{{{*/
  // ---------------------------------------------------------------------
  /* The only header we use is the last-modified header. */
- string pkgAcqFile::Custom600Headers()
+ string pkgAcqFile::Custom600Headers() const
  {
     if (IsIndexFile)
        return "\nIndex-File: true";
diff --combined apt-pkg/acquire-item.h
index 35cd78afcfa02b00d4a33a87590f162207d63213,d054b96af85e662d52ad1c2014100cd71b55edbb..3d863874cdc1e9dad16e81f570197a870e49c8f4
@@@ -46,6 -46,7 +46,7 @@@
  class indexRecords;
  class pkgRecords;
  class pkgSourceList;
+ class IndexTarget;
  
  /** \brief Represents the process by which a pkgAcquire object should {{{
   *  retrieve a file or a collection of files.
@@@ -211,12 -212,12 +212,12 @@@ class pkgAcquire::Item : public WeakPoi
      *  \param Message Data from the acquire method.  Use LookupTag()
      *  to parse it.
      *  \param Size The size of the object that was fetched.
-     *  \param Hash The HashSum of the object that was fetched.
+     *  \param Hashes The HashSums of the object that was fetched.
      *  \param Cnf The method via which the object was fetched.
      *
      *  \sa pkgAcqMethod
      */
-    virtual void Done(std::string Message,unsigned long long Size,std::string Hash,
+    virtual void Done(std::string Message, unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
  
     /** \brief Invoked when the worker starts to fetch this object.
      *  line, so they should (if nonempty) have a leading newline and
      *  no trailing newline.
      */
-    virtual std::string Custom600Headers() {return std::string();};
+    virtual std::string Custom600Headers() const {return std::string();};
  
     /** \brief A "descriptive" URI-like string.
      *
      *  \return a URI that should be used to describe what is being fetched.
      */
-    virtual std::string DescURI() = 0;
+    virtual std::string DescURI() const = 0;
     /** \brief Short item description.
      *
      *  \return a brief description of the object being fetched.
      */
-    virtual std::string ShortDesc() {return DescURI();}
+    virtual std::string ShortDesc() const {return DescURI();}
  
     /** \brief Invoked by the worker when the download is completely done. */
     virtual void Finished() {};
     
-    /** \brief HashSum 
+    /** \brief HashSums
      *
-     *  \return the HashSum of this object, if applicable; otherwise, an
-     *  empty string.
+     *  \return the HashSums of this object, if applicable; otherwise, an
+     *  empty list.
      */
-    virtual std::string HashSum() {return std::string();};
+    HashStringList HashSums() const {return ExpectedHashes;};
+    std::string HashSum() const {HashStringList const hashes = HashSums(); HashString const * const hs = hashes.find(NULL); return hs != NULL ? hs->toStr() : ""; };
  
     /** \return the acquire process with which this item is associated. */
-    pkgAcquire *GetOwner() {return Owner;};
+    pkgAcquire *GetOwner() const {return Owner;};
  
     /** \return \b true if this object is being fetched from a trusted source. */
-    virtual bool IsTrusted() {return false;};
+    virtual bool IsTrusted() const {return false;};
     
     // report mirror problems
     /** \brief Report mirror problem
      *  process, but does not place it into any fetch queues (you must
      *  manually invoke QueueURI() to do so).
      *
-     *  Initializes all fields of the item other than Owner to 0,
-     *  false, or the empty string.
-     *
      *  \param Owner The new owner of this item.
+     *  \param ExpectedHashes of the file represented by this item
      */
-    Item(pkgAcquire *Owner);
+    Item(pkgAcquire *Owner, HashStringList const &ExpectedHashes);
  
     /** \brief Remove this item from its owner's queue by invoking
      *  pkgAcquire::Remove.
      * \param state respresenting the error we encountered
      */
     bool RenameOnError(RenameOnErrorState const state);
+    /** \brief The HashSums of the item is supposed to have than done */
+    HashStringList ExpectedHashes;
+    /** \brief The item that is currently being downloaded. */
+    pkgAcquire::ItemDesc Desc;
  };
                                                                        /*}}}*/
  /** \brief Information about an index patch (aka diff). */            /*{{{*/
@@@ -335,20 -341,13 +341,13 @@@ class pkgAcqSubIndex : public pkgAcquir
     /** \brief If \b true, debugging information will be written to std::clog. */
     bool Debug;
  
-    /** \brief The item that is currently being downloaded. */
-    pkgAcquire::ItemDesc Desc;
-    /** \brief The Hash that this file should have after download
-     */
-    HashString ExpectedHash;
   public:
     // Specialized action members
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string Md5Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string DescURI() {return Desc.URI;};
-    virtual std::string Custom600Headers();
+    virtual std::string DescURI() const {return Desc.URI;};
+    virtual std::string Custom600Headers() const;
     virtual bool ParseIndex(std::string const &IndexFile);
  
     /** \brief Create a new pkgAcqSubIndex.
      *
      *  \param ShortDesc A short description of the list file to download.
      *
-     *  \param ExpectedHash The list file's MD5 signature.
+     *  \param ExpectedHashes The list file's hashsums which are expected.
      */
     pkgAcqSubIndex(pkgAcquire *Owner, std::string const &URI,std::string const &URIDesc,
-                  std::string const &ShortDesc, HashString const &ExpectedHash);
+                  std::string const &ShortDesc, HashStringList const &ExpectedHashes);
 +};
 +                                                                      /*}}}*/
 +
 +/** \brief Common base class for all classes that deal with fetching  {{{
 +           indexes
 + */
 +class pkgAcqBaseIndex : public pkgAcquire::Item
 +{
 + protected:
 +   /** \brief Pointer to the IndexTarget data
 +    */
 +   const struct IndexTarget * Target;
 +   indexRecords *MetaIndexParser;
 +
-    /** \brief The Hash that this file should have after download
-     */
-    HashString ExpectedHash;
 +   pkgAcqBaseIndex(pkgAcquire *Owner,
 +                   struct IndexTarget const * const Target,
-                    HashString ExpectedHash,
++                   HashStringList const &ExpectedHashes,
 +                   indexRecords *MetaIndexParser)
-       : Item(Owner), Target(Target), MetaIndexParser(MetaIndexParser),
-         ExpectedHash(ExpectedHash) {};
++      : Item(Owner, ExpectedHashes), Target(Target), 
++        MetaIndexParser(MetaIndexParser) {};
++
  };
                                                                        /*}}}*/
  /** \brief An item that is responsible for fetching an index file of  {{{
   *
   *  \sa pkgAcqIndexDiffs, pkgAcqIndex
   */
 -class pkgAcqDiffIndex : public pkgAcquire::Item
 +class pkgAcqDiffIndex : public pkgAcqBaseIndex
  {
   protected:
     /** \brief If \b true, debugging information will be written to std::clog. */
     bool Debug;
  
-    /** \brief The item that is currently being downloaded. */
-    pkgAcquire::ItemDesc Desc;
     /** \brief The URI of the index file to recreate at our end (either
      *  by downloading it or by applying partial patches).
      */
      */
     std::string Description;
  
 -   /** \brief Pointer to the IndexTarget data
 -    */
 -   const struct IndexTarget * Target;
 -   indexRecords *MetaIndexParser;
 -
   public:
     // Specialized action members
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string Md5Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string DescURI() {return RealURI + "Index";};
-    virtual std::string Custom600Headers();
+    virtual std::string DescURI() const {return RealURI + "Index";};
+    virtual std::string Custom600Headers() const;
  
     /** \brief Parse the Index file for a set of Packages diffs.
      *
      *
      *  \param ShortDesc A short description of the list file to download.
      *
-     *  \param ExpectedHash The list file's MD5 signature.
+     *  \param ExpectedHashes The list file's hashsums which are expected.
      */
     pkgAcqDiffIndex(pkgAcquire *Owner,
                     struct IndexTarget const * const Target,
-                    HashString ExpectedHash,
+                    HashStringList const &ExpectedHashes,
                     indexRecords *MetaIndexParser);
  };
                                                                        /*}}}*/
   *
   *  \sa pkgAcqDiffIndex, pkgAcqIndex
   */
 -class pkgAcqIndexMergeDiffs : public pkgAcquire::Item
 +class pkgAcqIndexMergeDiffs : public pkgAcqBaseIndex
  {
     protected:
  
      */
     bool Debug;
  
-    /** \brief description of the item that is currently being
-     *  downloaded.
-     */
-    pkgAcquire::ItemDesc Desc;
     /** \brief URI of the package index file that is being
      *  reconstructed.
      */
      *  outright; its arguments are ignored.
      */
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string Md5Hash,
-                    pkgAcquire::MethodConfig *Cnf);
-    virtual std::string DescURI() {return RealURI + "Index";};
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
+        pkgAcquire::MethodConfig *Cnf);
+    virtual std::string DescURI() const {return RealURI + "Index";};
  
     /** \brief Create an index merge-diff item.
      *
      *
      *  \param ShortDesc A brief description of this item.
      *
-     *  \param ExpectedHash The expected md5sum of the completely
+     *  \param ExpectedHashes The expected md5sum of the completely
      *  reconstructed package index file; the index file will be tested
      *  against this value when it is entirely reconstructed.
      *
      *  \param allPatches contains all related items so that each item can
      *  check if it was the last one to complete the download step
      */
 -   pkgAcqIndexMergeDiffs(pkgAcquire *Owner,std::string const &URI,std::string const &URIDesc,
 -                  std::string const &ShortDesc, HashStringList const &ExpectedHashes,
 -                  DiffInfo const &patch, std::vector<pkgAcqIndexMergeDiffs*> const * const allPatches);
 +   pkgAcqIndexMergeDiffs(pkgAcquire *Owner,
 +                         struct IndexTarget const * const Target,
-                          HashString ExpectedHash,
++                         HashStringList const &ExpectedHash,
 +                         indexRecords *MetaIndexParser,
 +                         DiffInfo const &patch, 
 +                         std::vector<pkgAcqIndexMergeDiffs*> const * const allPatches);
  };
                                                                        /*}}}*/
  /** \brief An item that is responsible for fetching server-merge patches {{{
   *
   *  \sa pkgAcqDiffIndex, pkgAcqIndex
   */
 -class pkgAcqIndexDiffs : public pkgAcquire::Item
 +class pkgAcqIndexDiffs : public pkgAcqBaseIndex
  {
     private:
  
     /** \brief Handle tasks that must be performed after the item
      *  finishes downloading.
      *
-     *  Dequeues the item and checks the resulting file's md5sum
-     *  against ExpectedHash after the last patch was applied.
+     *  Dequeues the item and checks the resulting file's hashsums
+     *  against ExpectedHashes after the last patch was applied.
      *  There is no need to check the md5/sha1 after a "normal" 
      *  patch because QueueNextDiff() will check the sha1 later.
      *
      */
     bool Debug;
  
-    /** \brief A description of the item that is currently being
-     *  downloaded.
-     */
-    pkgAcquire::ItemDesc Desc;
     /** \brief The URI of the package index file that is being
      *  reconstructed.
      */
      */
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
  
-    virtual void Done(std::string Message,unsigned long long Size,std::string Md5Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string DescURI() {return RealURI + "Index";};
+    virtual std::string DescURI() const {return RealURI + "Index";};
  
     /** \brief Create an index diff item.
      *
      *
      *  \param ShortDesc A brief description of this item.
      *
-     *  \param ExpectedHash The expected md5sum of the completely
+     *  \param ExpectedHashes The expected md5sum of the completely
      *  reconstructed package index file; the index file will be tested
      *  against this value when it is entirely reconstructed.
      *
      *  should be ordered so that each diff appears before any diff
      *  that depends on it.
      */
 -   pkgAcqIndexDiffs(pkgAcquire *Owner,std::string URI,std::string URIDesc,
 -                  std::string ShortDesc, HashStringList const &ExpectedHashes,
 +   pkgAcqIndexDiffs(pkgAcquire *Owner,
 +                    struct IndexTarget const * const Target,
-                     HashString ExpectedHash,
++                    HashStringList const &ExpectedHash,
 +                    indexRecords *MetaIndexParser,
                    std::string ServerSha1,
                    std::vector<DiffInfo> diffs=std::vector<DiffInfo>());
  };
   *
   *  \todo Why does pkgAcqIndex have protected members?
   */
 -class pkgAcqIndex : public pkgAcquire::Item
 +class pkgAcqIndex : public pkgAcqBaseIndex
  {
     protected:
  
     //        the downloaded file contains the expected tag
     bool Verify;
  
-    /** \brief The download request that is currently being
-     *   processed.
-     */
-    pkgAcquire::ItemDesc Desc;
     /** \brief The object that is actually being fetched (minus any
      *  compression-related extensions).
      */
      */
     std::string CompressionExtension;
  
 -   /** \brief Pointer to the IndexTarget data
 -    */
 -   const struct IndexTarget * Target;
 -   indexRecords *MetaIndexParser;
 -
     public:
     
     // Specialized action members
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string Md5Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string Custom600Headers();
-    virtual std::string DescURI() {return Desc.URI;};
-    virtual std::string HashSum() {return ExpectedHash.toStr(); };
+    virtual std::string Custom600Headers() const;
+    virtual std::string DescURI() const {return Desc.URI;};
  
     /** \brief Create a pkgAcqIndex.
      *
      *
      *  \param ShortDesc A brief description of this index file.
      *
-     *  \param ExpectedHash The expected hashsum of this index file.
+     *  \param ExpectedHashes The expected hashsum of this index file.
      *
      *  \param compressExt The compression-related extension with which
      *  this index file should be downloaded, or "" to autodetect
      *  fallback is ".gz" or none.
      */
     pkgAcqIndex(pkgAcquire *Owner,std::string URI,std::string URIDesc,
-              std::string ShortDesc, HashString ExpectedHash, 
+              std::string ShortDesc, HashStringList const &ExpectedHashes,
               std::string compressExt="");
 -   pkgAcqIndex(pkgAcquire *Owner, IndexTarget const * const Target,
 -                       HashStringList const &ExpectedHashes, indexRecords *MetaIndexParser);
 -   void Init(std::string const &URI, std::string const &URIDesc, std::string const &ShortDesc);
 +   pkgAcqIndex(pkgAcquire *Owner,
 +               struct IndexTarget const * const Target,
-                HashString const &ExpectedHash,
++               HashStringList const &ExpectedHash,
 +               indexRecords *MetaIndexParser);
 +   void Init(std::string const &URI, std::string const &URIDesc,
 +             std::string const &ShortDesc);
  };
                                                                        /*}}}*/
  /** \brief An acquire item that is responsible for fetching a         {{{
@@@ -788,7 -745,7 +764,7 @@@ class pkgAcqIndexTrans : public pkgAcqI
     public:
    
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual std::string Custom600Headers();
+    virtual std::string Custom600Headers() const;
  
     /** \brief Create a pkgAcqIndexTrans.
      *
      */
     pkgAcqIndexTrans(pkgAcquire *Owner,std::string URI,std::string URIDesc,
                    std::string ShortDesc);
-    pkgAcqIndexTrans(pkgAcquire *Owner, struct IndexTarget const * const Target,
-                   HashString const &ExpectedHash, indexRecords *MetaIndexParser);
+    pkgAcqIndexTrans(pkgAcquire *Owner, IndexTarget const * const Target,
+                   HashStringList const &ExpectedHashes, indexRecords *MetaIndexParser);
  };
                                                                        /*}}}*/
  /** \brief Information about an index file. */                                /*{{{*/
@@@ -872,9 -829,6 +848,6 @@@ class pkgAcqMetaSig : public pkgAcquire
     /** \brief The last good signature file */
     std::string LastGoodSig;
  
-    /** \brief The fetch request that is currently being processed. */
-    pkgAcquire::ItemDesc Desc;
     /** \brief The URI of the signature file.  Unlike Desc.URI, this is
      *  never modified; it is used to determine the file that is being
      *  downloaded.
      *
      *  \todo Why a list of pointers instead of a list of structs?
      */
-    const std::vector<struct IndexTarget*>* IndexTargets;
+    const std::vector<IndexTarget*>* IndexTargets;
  
     public:
     
     // Specialized action members
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string Md5Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string Custom600Headers();
-    virtual std::string DescURI() {return RealURI; };
+    virtual std::string Custom600Headers() const;
+    virtual std::string DescURI() const {return RealURI; };
  
     /** \brief Create a new pkgAcqMetaSig. */
     pkgAcqMetaSig(pkgAcquire *Owner,std::string URI,std::string URIDesc, std::string ShortDesc,
                 std::string MetaIndexURI, std::string MetaIndexURIDesc, std::string MetaIndexShortDesc,
-                const std::vector<struct IndexTarget*>* IndexTargets,
+                const std::vector<IndexTarget*>* IndexTargets,
                 indexRecords* MetaIndexParser);
     virtual ~pkgAcqMetaSig();
  };
  class pkgAcqMetaIndex : public pkgAcquire::Item
  {
     protected:
-    /** \brief The fetch command that is currently being processed. */
-    pkgAcquire::ItemDesc Desc;
     /** \brief The URI that is actually being downloaded; never
      *  modified by pkgAcqMetaIndex.
      */
     std::string SigFile;
  
     /** \brief The index files to download. */
-    const std::vector<struct IndexTarget*>* IndexTargets;
+    const std::vector<IndexTarget*>* IndexTargets;
  
     /** \brief The parser for the meta-index file. */
     indexRecords* MetaIndexParser;
     
     // Specialized action members
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size, std::string Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string Custom600Headers();
-    virtual std::string DescURI() {return RealURI; };
+    virtual std::string Custom600Headers() const;
+    virtual std::string DescURI() const {return RealURI; };
  
     /** \brief Create a new pkgAcqMetaIndex. */
     pkgAcqMetaIndex(pkgAcquire *Owner,
                   std::string URI,std::string URIDesc, std::string ShortDesc,
                   std::string SigFile,
-                  const std::vector<struct IndexTarget*>* IndexTargets,
+                  const std::vector<IndexTarget*>* IndexTargets,
                   indexRecords* MetaIndexParser);
  };
                                                                        /*}}}*/
@@@ -1040,14 -991,14 +1010,14 @@@ class pkgAcqMetaClearSig : public pkgAc
  
  public:
     void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual std::string Custom600Headers();
+    virtual std::string Custom600Headers() const;
  
     /** \brief Create a new pkgAcqMetaClearSig. */
     pkgAcqMetaClearSig(pkgAcquire *Owner,
                std::string const &URI, std::string const &URIDesc, std::string const &ShortDesc,
                std::string const &MetaIndexURI, std::string const &MetaIndexURIDesc, std::string const &MetaIndexShortDesc,
                std::string const &MetaSigURI, std::string const &MetaSigURIDesc, std::string const &MetaSigShortDesc,
-               const std::vector<struct IndexTarget*>* IndexTargets,
+               const std::vector<IndexTarget*>* IndexTargets,
                indexRecords* MetaIndexParser);
     virtual ~pkgAcqMetaClearSig();
  };
@@@ -1063,9 -1014,6 +1033,6 @@@ class pkgAcqArchive : public pkgAcquire
     /** \brief The package version being fetched. */
     pkgCache::VerIterator Version;
  
-    /** \brief The fetch command that is currently being processed. */
-    pkgAcquire::ItemDesc Desc;
     /** \brief The list of sources from which to pick archives to
      *  download this package from.
      */
      */
     pkgRecords *Recs;
  
-    /** \brief The hashsum of this package. */
-    HashString ExpectedHash;
     /** \brief A location in which the actual filename of the package
      *  should be stored.
      */
     public:
     
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string Hash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &Hashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string DescURI() {return Desc.URI;};
-    virtual std::string ShortDesc() {return Desc.ShortDesc;};
+    virtual std::string DescURI() const {return Desc.URI;};
+    virtual std::string ShortDesc() const {return Desc.ShortDesc;};
     virtual void Finished();
-    virtual std::string HashSum() {return ExpectedHash.toStr(); };
-    virtual bool IsTrusted();
+    virtual bool IsTrusted() const;
     
     /** \brief Create a new pkgAcqArchive.
      *
   */
  class pkgAcqFile : public pkgAcquire::Item
  {
-    /** \brief The currently active download process. */
-    pkgAcquire::ItemDesc Desc;
-    /** \brief The hashsum of the file to download, if it is known. */
-    HashString ExpectedHash;
     /** \brief How many times to retry the download, set from
      *  Acquire::Retries.
      */
     
     // Specialized action members
     virtual void Failed(std::string Message,pkgAcquire::MethodConfig *Cnf);
-    virtual void Done(std::string Message,unsigned long long Size,std::string CalcHash,
+    virtual void Done(std::string Message,unsigned long long Size, HashStringList const &CalcHashes,
                     pkgAcquire::MethodConfig *Cnf);
-    virtual std::string DescURI() {return Desc.URI;};
-    virtual std::string HashSum() {return ExpectedHash.toStr(); };
-    virtual std::string Custom600Headers();
+    virtual std::string DescURI() const {return Desc.URI;};
+    virtual std::string Custom600Headers() const;
  
     /** \brief Create a new pkgAcqFile object.
      *
      *
      *  \param URI The URI to download.
      *
-     *  \param Hash The hashsum of the file to download, if it is known;
-     *  otherwise "".
+     *  \param Hashes The hashsums of the file to download, if they are known;
+     *  otherwise empty list.
      *
      *  \param Size The size of the file to download, if it is known;
      *  otherwise 0.
      * is the absolute name to which the file should be downloaded.
      */
  
-    pkgAcqFile(pkgAcquire *Owner, std::string URI, std::string Hash, unsigned long long Size,
+    pkgAcqFile(pkgAcquire *Owner, std::string URI, HashStringList const &Hashes, unsigned long long Size,
              std::string Desc, std::string ShortDesc,
              const std::string &DestDir="", const std::string &DestFilename="",
              bool IsIndexFile=false);