handler.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package httpd
  2. import (
  3. "io"
  4. "net/http"
  5. "os"
  6. "path"
  7. "strings"
  8. "github.com/drakkan/sftpgo/v2/common"
  9. "github.com/drakkan/sftpgo/v2/dataprovider"
  10. "github.com/drakkan/sftpgo/v2/logger"
  11. "github.com/drakkan/sftpgo/v2/util"
  12. )
  13. // Connection details for a HTTP connection used to inteact with an SFTPGo filesystem
  14. type Connection struct {
  15. *common.BaseConnection
  16. request *http.Request
  17. }
  18. // GetClientVersion returns the connected client's version.
  19. func (c *Connection) GetClientVersion() string {
  20. if c.request != nil {
  21. return c.request.UserAgent()
  22. }
  23. return ""
  24. }
  25. // GetRemoteAddress return the connected client's address
  26. func (c *Connection) GetRemoteAddress() string {
  27. if c.request != nil {
  28. return c.request.RemoteAddr
  29. }
  30. return ""
  31. }
  32. // Disconnect closes the active transfer
  33. func (c *Connection) Disconnect() (err error) {
  34. return c.SignalTransfersAbort()
  35. }
  36. // GetCommand returns the request method
  37. func (c *Connection) GetCommand() string {
  38. if c.request != nil {
  39. return strings.ToUpper(c.request.Method)
  40. }
  41. return ""
  42. }
  43. // Stat returns a FileInfo describing the named file/directory, or an error,
  44. // if any happens
  45. func (c *Connection) Stat(name string, mode int) (os.FileInfo, error) {
  46. c.UpdateLastActivity()
  47. name = util.CleanPath(name)
  48. if !c.User.HasPerm(dataprovider.PermListItems, path.Dir(name)) {
  49. return nil, c.GetPermissionDeniedError()
  50. }
  51. fi, err := c.DoStat(name, mode)
  52. if err != nil {
  53. c.Log(logger.LevelDebug, "error running stat on path %#v: %+v", name, err)
  54. return nil, err
  55. }
  56. return fi, err
  57. }
  58. // ReadDir returns a list of directory entries
  59. func (c *Connection) ReadDir(name string) ([]os.FileInfo, error) {
  60. c.UpdateLastActivity()
  61. name = util.CleanPath(name)
  62. return c.ListDir(name)
  63. }
  64. func (c *Connection) getFileReader(name string, offset int64, method string) (io.ReadCloser, error) {
  65. c.UpdateLastActivity()
  66. name = util.CleanPath(name)
  67. if !c.User.HasPerm(dataprovider.PermDownload, path.Dir(name)) {
  68. return nil, c.GetPermissionDeniedError()
  69. }
  70. if !c.User.IsFileAllowed(name) {
  71. c.Log(logger.LevelWarn, "reading file %#v is not allowed", name)
  72. return nil, c.GetPermissionDeniedError()
  73. }
  74. fs, p, err := c.GetFsAndResolvedPath(name)
  75. if err != nil {
  76. return nil, err
  77. }
  78. if method != http.MethodHead {
  79. if err := common.ExecutePreAction(&c.User, common.OperationPreDownload, p, name, c.GetProtocol(), 0, 0); err != nil {
  80. c.Log(logger.LevelDebug, "download for file %#v denied by pre action: %v", name, err)
  81. return nil, c.GetPermissionDeniedError()
  82. }
  83. }
  84. file, r, cancelFn, err := fs.Open(p, offset)
  85. if err != nil {
  86. c.Log(logger.LevelWarn, "could not open file %#v for reading: %+v", p, err)
  87. return nil, c.GetFsError(fs, err)
  88. }
  89. baseTransfer := common.NewBaseTransfer(file, c.BaseConnection, cancelFn, p, p, name, common.TransferDownload,
  90. 0, 0, 0, false, fs)
  91. return newHTTPDFile(baseTransfer, r), nil
  92. }