transfer.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. // Copyright (C) 2019-2022 Nicola Murino
  2. //
  3. // This program is free software: you can redistribute it and/or modify
  4. // it under the terms of the GNU Affero General Public License as published
  5. // by the Free Software Foundation, version 3.
  6. //
  7. // This program is distributed in the hope that it will be useful,
  8. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. // GNU Affero General Public License for more details.
  11. //
  12. // You should have received a copy of the GNU Affero General Public License
  13. // along with this program. If not, see <https://www.gnu.org/licenses/>.
  14. package ftpd
  15. import (
  16. "errors"
  17. "io"
  18. "github.com/eikenb/pipeat"
  19. "github.com/drakkan/sftpgo/v2/internal/common"
  20. "github.com/drakkan/sftpgo/v2/internal/vfs"
  21. )
  22. // transfer contains the transfer details for an upload or a download.
  23. // It implements the ftpserver.FileTransfer interface to handle files downloads and uploads
  24. type transfer struct {
  25. *common.BaseTransfer
  26. writer io.WriteCloser
  27. reader io.ReadCloser
  28. isFinished bool
  29. expectedOffset int64
  30. }
  31. func newTransfer(baseTransfer *common.BaseTransfer, pipeWriter *vfs.PipeWriter, pipeReader *pipeat.PipeReaderAt,
  32. expectedOffset int64) *transfer {
  33. var writer io.WriteCloser
  34. var reader io.ReadCloser
  35. if baseTransfer.File != nil {
  36. writer = baseTransfer.File
  37. reader = baseTransfer.File
  38. } else if pipeWriter != nil {
  39. writer = pipeWriter
  40. } else if pipeReader != nil {
  41. reader = pipeReader
  42. }
  43. return &transfer{
  44. BaseTransfer: baseTransfer,
  45. writer: writer,
  46. reader: reader,
  47. isFinished: false,
  48. expectedOffset: expectedOffset,
  49. }
  50. }
  51. // Read reads the contents to downloads.
  52. func (t *transfer) Read(p []byte) (n int, err error) {
  53. t.Connection.UpdateLastActivity()
  54. n, err = t.reader.Read(p)
  55. t.BytesSent.Add(int64(n))
  56. if err == nil {
  57. err = t.CheckRead()
  58. }
  59. if err != nil && err != io.EOF {
  60. t.TransferError(err)
  61. err = t.ConvertError(err)
  62. return
  63. }
  64. t.HandleThrottle()
  65. return
  66. }
  67. // Write writes the uploaded contents.
  68. func (t *transfer) Write(p []byte) (n int, err error) {
  69. t.Connection.UpdateLastActivity()
  70. n, err = t.writer.Write(p)
  71. t.BytesReceived.Add(int64(n))
  72. if err == nil {
  73. err = t.CheckWrite()
  74. }
  75. if err != nil {
  76. t.TransferError(err)
  77. err = t.ConvertError(err)
  78. return
  79. }
  80. t.HandleThrottle()
  81. return
  82. }
  83. // Seek sets the offset to resume an upload or a download
  84. func (t *transfer) Seek(offset int64, whence int) (int64, error) {
  85. t.Connection.UpdateLastActivity()
  86. if t.File != nil {
  87. ret, err := t.File.Seek(offset, whence)
  88. if err != nil {
  89. t.TransferError(err)
  90. }
  91. return ret, err
  92. }
  93. if t.reader != nil && t.expectedOffset == offset && whence == io.SeekStart {
  94. return offset, nil
  95. }
  96. t.TransferError(errors.New("seek is unsupported for this transfer"))
  97. return 0, common.ErrOpUnsupported
  98. }
  99. // Close it is called when the transfer is completed.
  100. func (t *transfer) Close() error {
  101. if err := t.setFinished(); err != nil {
  102. return err
  103. }
  104. err := t.closeIO()
  105. errBaseClose := t.BaseTransfer.Close()
  106. if errBaseClose != nil {
  107. err = errBaseClose
  108. }
  109. return t.Connection.GetFsError(t.Fs, err)
  110. }
  111. func (t *transfer) closeIO() error {
  112. var err error
  113. if t.File != nil {
  114. err = t.File.Close()
  115. } else if t.writer != nil {
  116. err = t.writer.Close()
  117. t.Lock()
  118. // we set ErrTransfer here so quota is not updated, in this case the uploads are atomic
  119. if err != nil && t.ErrTransfer == nil {
  120. t.ErrTransfer = err
  121. }
  122. t.Unlock()
  123. } else if t.reader != nil {
  124. err = t.reader.Close()
  125. }
  126. return err
  127. }
  128. func (t *transfer) setFinished() error {
  129. t.Lock()
  130. defer t.Unlock()
  131. if t.isFinished {
  132. return common.ErrTransferClosed
  133. }
  134. t.isFinished = true
  135. return nil
  136. }