X-Git-Url: https://git.saurik.com/apt.git/blobdiff_plain/be4401bfa4a240bbc894e1bfeb1e1e8d63fc7b18..91414dd77ec7efe1d3fc053e32cbe4d82263e015:/methods/copy.cc?ds=sidebyside diff --git a/methods/copy.cc b/methods/copy.cc index 94e88e503..e515b2def 100644 --- a/methods/copy.cc +++ b/methods/copy.cc @@ -1,6 +1,6 @@ // -*- mode: cpp; mode: fold -*- // Description /*{{{*/ -// $Id: copy.cc,v 1.5 1998/11/01 05:27:40 jgg Exp $ +// $Id: copy.cc,v 1.7.2.1 2004/01/16 18:58:50 mdz Exp $ /* ###################################################################### Copy URI - This method takes a uri like a file: uri and copies it @@ -9,22 +9,30 @@ ##################################################################### */ /*}}}*/ // Include Files /*{{{*/ +#include + #include +#include #include #include +#include +#include +#include "aptmethod.h" +#include #include -#include -#include +#include + +#include /*}}}*/ -class CopyMethod : public pkgAcqMethod +class CopyMethod : public aptMethod { - virtual bool Fetch(FetchItem *Itm); - + virtual bool Fetch(FetchItem *Itm) APT_OVERRIDE; + public: - - CopyMethod() : pkgAcqMethod("1.0",SingleInstance) {}; + + CopyMethod() : aptMethod("copy", "1.0",SingleInstance | SendConfig) {}; }; // CopyMethod::Fetch - Fetch a file /*{{{*/ @@ -32,47 +40,54 @@ class CopyMethod : public pkgAcqMethod /* */ bool CopyMethod::Fetch(FetchItem *Itm) { - URI Get = Itm->Uri; - string File = Get.Path; + // this ensures that relative paths work in copy + std::string const File = Itm->Uri.substr(Itm->Uri.find(':')+1); + + // Stat the file and send a start message + struct stat Buf; + if (stat(File.c_str(),&Buf) != 0) + return _error->Errno("stat",_("Failed to stat")); + + // Forumulate a result and send a start message + FetchResult Res; + Res.Size = Buf.st_size; + Res.Filename = Itm->DestFile; + Res.LastModified = Buf.st_mtime; + Res.IMSHit = false; + URIStart(Res); + + // just calc the hashes if the source and destination are identical + if (File == Itm->DestFile || Itm->DestFile == "/dev/null") + { + CalculateHashes(Itm, Res); + URIDone(Res); + return true; + } // See if the file exists FileFd From(File,FileFd::ReadOnly); - FileFd To(Itm->DestFile,FileFd::WriteEmpty); + FileFd To(Itm->DestFile,FileFd::WriteAtomic); To.EraseOnFailure(); - if (_error->PendingError() == true) - return false; - + // Copy the file if (CopyFile(From,To) == false) + { + To.OpFail(); return false; + } From.Close(); To.Close(); - + // Transfer the modification times - struct stat Buf; - if (stat(File.c_str(),&Buf) != 0) - { - To.OpFail(); - return _error->Errno("stat","Failed to stat"); - } - - struct utimbuf TimeBuf; - TimeBuf.actime = Buf.st_atime; - TimeBuf.modtime = Buf.st_mtime; - if (utime(Itm->DestFile.c_str(),&TimeBuf) != 0) - { - To.OpFail(); - return _error->Errno("utime","Failed to set modification time"); - } - - // Forumulate a result - FetchResult Res; - Res.Size = Buf.st_size; - Res.Filename = Itm->DestFile; - Res.LastModified = Buf.st_mtime; - Res.IMSHit = false; - + struct timeval times[2]; + times[0].tv_sec = Buf.st_atime; + times[1].tv_sec = Buf.st_mtime; + times[0].tv_usec = times[1].tv_usec = 0; + if (utimes(Res.Filename.c_str(), times) != 0) + return _error->Errno("utimes",_("Failed to set modification time")); + + CalculateHashes(Itm, Res); URIDone(Res); return true; } @@ -80,6 +95,9 @@ bool CopyMethod::Fetch(FetchItem *Itm) int main() { + setlocale(LC_ALL, ""); + CopyMethod Mth; + return Mth.Run(); }