// -*- mode: cpp; mode: fold -*- // Description /*{{{*/ /* ###################################################################### File URI method for APT This simply checks that the file specified exists, if so the relevant information is returned. If a .gz filename is specified then the file name with .gz removed will also be checked and information about it will be returned in Alt-* ##################################################################### */ /*}}}*/ // Include Files /*{{{*/ #include <config.h> #include "aptmethod.h" #include <apt-pkg/aptconfiguration.h> #include <apt-pkg/error.h> #include <apt-pkg/fileutl.h> #include <apt-pkg/hashes.h> #include <apt-pkg/strutl.h> #include <string> #include <sys/stat.h> #include <apti18n.h> /*}}}*/ class FileMethod : public aptMethod { virtual bool Fetch(FetchItem *Itm) APT_OVERRIDE; public: FileMethod() : aptMethod("file", "1.0", SingleInstance | SendConfig | LocalOnly) { SeccompFlags = aptMethod::BASE; } }; // FileMethod::Fetch - Fetch a file /*{{{*/ // --------------------------------------------------------------------- /* */ bool FileMethod::Fetch(FetchItem *Itm) { URI Get(Itm->Uri); std::string File = Get.Path; FetchResult Res; if (Get.Host.empty() == false) return _error->Error(_("Invalid URI, local URIS must not start with //")); struct stat Buf; // deal with destination files which might linger around if (lstat(Itm->DestFile.c_str(), &Buf) == 0) { if ((Buf.st_mode & S_IFREG) != 0) { if (Itm->LastModified == Buf.st_mtime && Itm->LastModified != 0) { HashStringList const hsl = Itm->ExpectedHashes; if (Itm->ExpectedHashes.VerifyFile(File)) { Res.Filename = Itm->DestFile; Res.IMSHit = true; } } } } if (Res.IMSHit != true) RemoveFile("file", Itm->DestFile); int olderrno = 0; // See if the file exists if (stat(File.c_str(),&Buf) == 0) { Res.Size = Buf.st_size; Res.Filename = File; Res.LastModified = Buf.st_mtime; Res.IMSHit = false; if (Itm->LastModified == Buf.st_mtime && Itm->LastModified != 0) { unsigned long long const filesize = Itm->ExpectedHashes.FileSize(); if (filesize != 0 && filesize == Res.Size) Res.IMSHit = true; } CalculateHashes(Itm, Res); } else olderrno = errno; if (Res.IMSHit == false) URIStart(Res); // See if the uncompressed file exists and reuse it FetchResult AltRes; AltRes.Filename.clear(); std::vector<std::string> extensions = APT::Configuration::getCompressorExtensions(); for (std::vector<std::string>::const_iterator ext = extensions.begin(); ext != extensions.end(); ++ext) { if (APT::String::Endswith(File, *ext) == true) { std::string const unfile = File.substr(0, File.length() - ext->length()); if (stat(unfile.c_str(),&Buf) == 0) { AltRes.Size = Buf.st_size; AltRes.Filename = unfile; AltRes.LastModified = Buf.st_mtime; AltRes.IMSHit = false; if (Itm->LastModified == Buf.st_mtime && Itm->LastModified != 0) AltRes.IMSHit = true; break; } // no break here as we could have situations similar to '.gz' vs '.tar.gz' here } } if (AltRes.Filename.empty() == false) URIDone(Res,&AltRes); else if (Res.Filename.empty() == false) URIDone(Res); else { errno = olderrno; return _error->Errno(File.c_str(), _("File not found")); } return true; } /*}}}*/ int main() { return FileMethod().Run(); }