traversessymlink.go 1.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. // Copyright (C) 2016 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at https://mozilla.org/MPL/2.0/.
  6. package osutil
  7. import (
  8. "fmt"
  9. "path/filepath"
  10. "strings"
  11. "github.com/syncthing/syncthing/lib/fs"
  12. )
  13. // TraversesSymlinkError is an error indicating symlink traversal
  14. type TraversesSymlinkError struct {
  15. path string
  16. }
  17. func (e TraversesSymlinkError) Error() string {
  18. return fmt.Sprintf("traverses symlink: %s", e.path)
  19. }
  20. // NotADirectoryError is an error indicating an expected path is not a directory
  21. type NotADirectoryError struct {
  22. path string
  23. }
  24. func (e NotADirectoryError) Error() string {
  25. return fmt.Sprintf("not a directory: %s", e.path)
  26. }
  27. // TraversesSymlink returns an error if any path component of name (including name
  28. // itself) traverses a symlink.
  29. func TraversesSymlink(filesystem fs.Filesystem, name string) error {
  30. var err error
  31. name, err = fs.Canonicalize(name)
  32. if err != nil {
  33. return err
  34. }
  35. if name == "." {
  36. // The result of calling TraversesSymlink(filesystem, filepath.Dir("foo"))
  37. return nil
  38. }
  39. var path string
  40. for _, part := range strings.Split(name, string(fs.PathSeparator)) {
  41. path = filepath.Join(path, part)
  42. info, err := filesystem.Lstat(path)
  43. if err != nil {
  44. if fs.IsNotExist(err) {
  45. return nil
  46. }
  47. return err
  48. }
  49. if info.IsSymlink() {
  50. return &TraversesSymlinkError{
  51. path: path,
  52. }
  53. }
  54. if !info.IsDir() {
  55. return &NotADirectoryError{
  56. path: path,
  57. }
  58. }
  59. }
  60. return nil
  61. }