123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- // -*- mode: cpp; mode: fold -*-
- // Description /*{{{*/
- /* ######################################################################
- Store method - Takes a file URI and stores its content (for which it will
- calculate the hashes) in the given destination. The input file will be
- extracted based on its file extension (or with the given compressor if
- called with one of the compatible symlinks) and potentially recompressed
- based on the file extension of the destination filename.
- ##################################################################### */
- /*}}}*/
- // Include Files /*{{{*/
- #include <config.h>
- #include <apt-pkg/configuration.h>
- #include <apt-pkg/acquire-method.h>
- #include <apt-pkg/error.h>
- #include <apt-pkg/fileutl.h>
- #include <apt-pkg/hashes.h>
- #include <apt-pkg/strutl.h>
- #include <apt-pkg/aptconfiguration.h>
- #include "aptmethod.h"
- #include <string.h>
- #include <sys/stat.h>
- #include <sys/time.h>
- #include <string>
- #include <vector>
- #include <apti18n.h>
- /*}}}*/
- class StoreMethod : public aptMethod
- {
- std::string const Prog;
- virtual bool Fetch(FetchItem *Itm) APT_OVERRIDE;
- public:
- explicit StoreMethod(std::string const &pProg) : aptMethod(pProg.c_str(),"1.2",SingleInstance | SendConfig), Prog(pProg) {};
- };
- static bool OpenFileWithCompressorByName(FileFd &fileFd, std::string const &Filename, unsigned int const Mode, std::string const &Name)
- {
- if (Name == "store")
- return fileFd.Open(Filename, Mode, FileFd::Extension);
- std::vector<APT::Configuration::Compressor> const compressors = APT::Configuration::getCompressors();
- std::vector<APT::Configuration::Compressor>::const_iterator compressor = compressors.begin();
- for (; compressor != compressors.end(); ++compressor)
- if (compressor->Name == Name)
- break;
- if (compressor == compressors.end())
- return _error->Error("Extraction of file %s requires unknown compressor %s", Filename.c_str(), Name.c_str());
- return fileFd.Open(Filename, Mode, *compressor);
- }
- /*}}}*/
- bool StoreMethod::Fetch(FetchItem *Itm) /*{{{*/
- {
- URI Get = Itm->Uri;
- std::string Path = Get.Host + Get.Path; // To account for relative paths
-
- FetchResult Res;
- Res.Filename = Itm->DestFile;
- URIStart(Res);
- // Open the source and destination files
- FileFd From;
- if (_config->FindB("Method::Compress", false) == false)
- {
- if (OpenFileWithCompressorByName(From, Path, FileFd::ReadOnly, Prog) == false)
- return false;
- if(From.IsCompressed() && From.FileSize() == 0)
- return _error->Error(_("Empty files can't be valid archives"));
- }
- else
- From.Open(Path, FileFd::ReadOnly, FileFd::Extension);
- if (From.IsOpen() == false || From.Failed() == true)
- return false;
- FileFd To;
- if (Itm->DestFile != "/dev/null" && Itm->DestFile != Path)
- {
- if (_config->FindB("Method::Compress", false) == false)
- To.Open(Itm->DestFile, FileFd::WriteOnly | FileFd::Create | FileFd::Atomic, FileFd::Extension);
- else if (OpenFileWithCompressorByName(To, Itm->DestFile, FileFd::WriteOnly | FileFd::Create | FileFd::Empty, Prog) == false)
- return false;
- if (To.IsOpen() == false || To.Failed() == true)
- return false;
- To.EraseOnFailure();
- }
- // Read data from source, generate checksums and write
- Hashes Hash(Itm->ExpectedHashes);
- bool Failed = false;
- Res.Size = 0;
- while (1)
- {
- unsigned char Buffer[4*1024];
- unsigned long long Count = 0;
- if (!From.Read(Buffer,sizeof(Buffer),&Count))
- {
- if (To.IsOpen())
- To.OpFail();
- return false;
- }
- if (Count == 0)
- break;
- Res.Size += Count;
- Hash.Add(Buffer,Count);
- if (To.IsOpen() && To.Write(Buffer,Count) == false)
- {
- Failed = true;
- break;
- }
- }
- From.Close();
- To.Close();
- if (Failed == true)
- return false;
- // Transfer the modification times
- if (Itm->DestFile != "/dev/null")
- {
- struct stat Buf;
- if (stat(Path.c_str(),&Buf) != 0)
- return _error->Errno("stat",_("Failed to stat"));
- struct timeval times[2];
- times[0].tv_sec = Buf.st_atime;
- Res.LastModified = times[1].tv_sec = Buf.st_mtime;
- times[0].tv_usec = times[1].tv_usec = 0;
- if (utimes(Itm->DestFile.c_str(), times) != 0)
- return _error->Errno("utimes",_("Failed to set modification time"));
- }
- // Return a Done response
- Res.TakeHashes(Hash);
- URIDone(Res);
- return true;
- }
- /*}}}*/
- int main(int, char *argv[])
- {
- setlocale(LC_ALL, "");
- StoreMethod Mth(flNotDir(argv[0]));
- return Mth.Run();
- }
|