transfer.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. // Copyright (C) 2019 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/drakkan/sftpgo/v2/internal/common"
  19. "github.com/drakkan/sftpgo/v2/internal/vfs"
  20. )
  21. // transfer contains the transfer details for an upload or a download.
  22. // It implements the ftpserver.FileTransfer interface to handle files downloads and uploads
  23. type transfer struct {
  24. *common.BaseTransfer
  25. writer io.WriteCloser
  26. reader io.ReadCloser
  27. isFinished bool
  28. expectedOffset int64
  29. }
  30. func newTransfer(baseTransfer *common.BaseTransfer, pipeWriter vfs.PipeWriter, pipeReader vfs.PipeReader,
  31. expectedOffset int64) *transfer {
  32. var writer io.WriteCloser
  33. var reader io.ReadCloser
  34. if baseTransfer.File != nil {
  35. writer = baseTransfer.File
  36. reader = baseTransfer.File
  37. } else if pipeWriter != nil {
  38. writer = pipeWriter
  39. } else if pipeReader != nil {
  40. reader = pipeReader
  41. }
  42. return &transfer{
  43. BaseTransfer: baseTransfer,
  44. writer: writer,
  45. reader: reader,
  46. isFinished: false,
  47. expectedOffset: expectedOffset,
  48. }
  49. }
  50. // Read reads the contents to downloads.
  51. func (t *transfer) Read(p []byte) (n int, err error) {
  52. t.Connection.UpdateLastActivity()
  53. n, err = t.reader.Read(p)
  54. t.BytesSent.Add(int64(n))
  55. if err == nil {
  56. err = t.CheckRead()
  57. }
  58. if err != nil && err != io.EOF {
  59. t.TransferError(err)
  60. err = t.ConvertError(err)
  61. return
  62. }
  63. t.HandleThrottle()
  64. return
  65. }
  66. // Write writes the uploaded contents.
  67. func (t *transfer) Write(p []byte) (n int, err error) {
  68. t.Connection.UpdateLastActivity()
  69. n, err = t.writer.Write(p)
  70. t.BytesReceived.Add(int64(n))
  71. if err == nil {
  72. err = t.CheckWrite()
  73. }
  74. if err != nil {
  75. t.TransferError(err)
  76. err = t.ConvertError(err)
  77. return
  78. }
  79. t.HandleThrottle()
  80. return
  81. }
  82. // Seek sets the offset to resume an upload or a download
  83. func (t *transfer) Seek(offset int64, whence int) (int64, error) {
  84. t.Connection.UpdateLastActivity()
  85. if t.File != nil {
  86. ret, err := t.File.Seek(offset, whence)
  87. if err != nil {
  88. t.TransferError(err)
  89. }
  90. return ret, err
  91. }
  92. if (t.reader != nil || t.writer != nil) && t.expectedOffset == offset && whence == io.SeekStart {
  93. return offset, nil
  94. }
  95. t.TransferError(errors.New("seek is unsupported for this transfer"))
  96. return 0, common.ErrOpUnsupported
  97. }
  98. // Close it is called when the transfer is completed.
  99. func (t *transfer) Close() error {
  100. if err := t.setFinished(); err != nil {
  101. return err
  102. }
  103. err := t.closeIO()
  104. errBaseClose := t.BaseTransfer.Close()
  105. if errBaseClose != nil {
  106. err = errBaseClose
  107. }
  108. return t.Connection.GetFsError(t.Fs, err)
  109. }
  110. func (t *transfer) closeIO() error {
  111. var err error
  112. if t.File != nil {
  113. err = t.File.Close()
  114. } else if t.writer != nil {
  115. err = t.writer.Close()
  116. t.Lock()
  117. // we set ErrTransfer here so quota is not updated, in this case the uploads are atomic
  118. if err != nil && t.ErrTransfer == nil {
  119. t.ErrTransfer = err
  120. }
  121. t.Unlock()
  122. } else if t.reader != nil {
  123. err = t.reader.Close()
  124. if metadater, ok := t.reader.(vfs.Metadater); ok {
  125. t.BaseTransfer.SetMetadata(metadater.Metadata())
  126. }
  127. }
  128. return err
  129. }
  130. func (t *transfer) setFinished() error {
  131. t.Lock()
  132. defer t.Unlock()
  133. if t.isFinished {
  134. return common.ErrTransferClosed
  135. }
  136. t.isFinished = true
  137. return nil
  138. }