dirstream.cc 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. // -*- mode: cpp; mode: fold -*-
  2. // Description /*{{{*/
  3. // $Id: dirstream.cc,v 1.3.2.1 2004/01/16 18:58:50 mdz Exp $
  4. /* ######################################################################
  5. Directory Stream
  6. This class provides a simple basic extractor that can be used for
  7. a number of purposes.
  8. ##################################################################### */
  9. /*}}}*/
  10. // Include Files /*{{{*/
  11. #include<config.h>
  12. #include <apt-pkg/dirstream.h>
  13. #include <apt-pkg/error.h>
  14. #include <fcntl.h>
  15. #include <sys/stat.h>
  16. #include <sys/time.h>
  17. #include <errno.h>
  18. #include <unistd.h>
  19. #include <apti18n.h>
  20. /*}}}*/
  21. // DirStream::DoItem - Process an item /*{{{*/
  22. // ---------------------------------------------------------------------
  23. /* This is a very simple extractor, it does not deal with things like
  24. overwriting directories with files and so on. */
  25. bool pkgDirStream::DoItem(Item &Itm,int &Fd)
  26. {
  27. switch (Itm.Type)
  28. {
  29. case Item::File:
  30. {
  31. /* Open the output file, NDELAY is used to prevent this from
  32. blowing up on device special files.. */
  33. int iFd = open(Itm.Name,O_NDELAY|O_WRONLY|O_CREAT|O_TRUNC|O_APPEND,
  34. Itm.Mode);
  35. if (iFd < 0)
  36. return _error->Errno("open",_("Failed to write file %s"),
  37. Itm.Name);
  38. // fchmod deals with umask and fchown sets the ownership
  39. if (fchmod(iFd,Itm.Mode) != 0)
  40. {
  41. close(iFd);
  42. return _error->Errno("fchmod",_("Failed to write file %s"), Itm.Name);
  43. }
  44. if (fchown(iFd,Itm.UID,Itm.GID) != 0 && errno != EPERM)
  45. {
  46. close(iFd);
  47. return _error->Errno("fchown",_("Failed to write file %s"), Itm.Name);
  48. }
  49. Fd = iFd;
  50. return true;
  51. }
  52. case Item::HardLink:
  53. case Item::SymbolicLink:
  54. case Item::CharDevice:
  55. case Item::BlockDevice:
  56. case Item::Directory:
  57. {
  58. struct stat Buf;
  59. // check if the dir is already there, if so return true
  60. if (stat(Itm.Name,&Buf) == 0)
  61. {
  62. if(S_ISDIR(Buf.st_mode))
  63. return true;
  64. // something else is there already, return false
  65. return false;
  66. }
  67. // nothing here, create the dir
  68. if(mkdir(Itm.Name,Itm.Mode) < 0)
  69. return false;
  70. return true;
  71. }
  72. case Item::FIFO:
  73. break;
  74. }
  75. return true;
  76. }
  77. /*}}}*/
  78. // DirStream::FinishedFile - Finished processing a file /*{{{*/
  79. // ---------------------------------------------------------------------
  80. /* */
  81. bool pkgDirStream::FinishedFile(Item &Itm,int Fd)
  82. {
  83. if (Fd < 0)
  84. return true;
  85. /* Set the modification times. The only way it can fail is if someone
  86. has futzed with our file, which is intolerable :> */
  87. struct timeval times[2];
  88. times[0].tv_sec = times[1].tv_sec = Itm.MTime;
  89. times[0].tv_usec = times[1].tv_usec = 0;
  90. if (utimes(Itm.Name, times) != 0)
  91. _error->Errno("utimes", "Failed to set modification time for %s",Itm.Name);
  92. if (close(Fd) != 0)
  93. return _error->Errno("close",_("Failed to close file %s"),Itm.Name);
  94. return true;
  95. }
  96. /*}}}*/
  97. // DirStream::Fail - Failed processing a file /*{{{*/
  98. // ---------------------------------------------------------------------
  99. /* */
  100. bool pkgDirStream::Fail(Item &/*Itm*/, int Fd)
  101. {
  102. if (Fd < 0)
  103. return true;
  104. close(Fd);
  105. return false;
  106. }
  107. /*}}}*/