/vendor/github.com/docker/docker/builder/dockerfile/dispatchers_windows.go

https://bitbucket.org/gollariel/dockertool · Go · 87 lines · 49 code · 8 blank · 30 comment · 18 complexity · 43b21e1944c528a1019b62070f85b842 MD5 · raw file

  1. package dockerfile
  2. import (
  3. "errors"
  4. "fmt"
  5. "os"
  6. "path/filepath"
  7. "regexp"
  8. "strings"
  9. "github.com/docker/docker/pkg/system"
  10. )
  11. var pattern = regexp.MustCompile(`^[a-zA-Z]:\.$`)
  12. // normaliseWorkdir normalises a user requested working directory in a
  13. // platform semantically consistent way.
  14. func normaliseWorkdir(current string, requested string) (string, error) {
  15. if requested == "" {
  16. return "", errors.New("cannot normalise nothing")
  17. }
  18. // `filepath.Clean` will replace "" with "." so skip in that case
  19. if current != "" {
  20. current = filepath.Clean(current)
  21. }
  22. if requested != "" {
  23. requested = filepath.Clean(requested)
  24. }
  25. // If either current or requested in Windows is:
  26. // C:
  27. // C:.
  28. // then an error will be thrown as the definition for the above
  29. // refers to `current directory on drive C:`
  30. // Since filepath.Clean() will automatically normalize the above
  31. // to `C:.`, we only need to check the last format
  32. if pattern.MatchString(current) {
  33. return "", fmt.Errorf("%s is not a directory. If you are specifying a drive letter, please add a trailing '\\'", current)
  34. }
  35. if pattern.MatchString(requested) {
  36. return "", fmt.Errorf("%s is not a directory. If you are specifying a drive letter, please add a trailing '\\'", requested)
  37. }
  38. // Target semantics is C:\somefolder, specifically in the format:
  39. // UPPERCASEDriveLetter-Colon-Backslash-FolderName. We are already
  40. // guaranteed that `current`, if set, is consistent. This allows us to
  41. // cope correctly with any of the following in a Dockerfile:
  42. // WORKDIR a --> C:\a
  43. // WORKDIR c:\\foo --> C:\foo
  44. // WORKDIR \\foo --> C:\foo
  45. // WORKDIR /foo --> C:\foo
  46. // WORKDIR c:\\foo \ WORKDIR bar --> C:\foo --> C:\foo\bar
  47. // WORKDIR C:/foo \ WORKDIR bar --> C:\foo --> C:\foo\bar
  48. // WORKDIR C:/foo \ WORKDIR \\bar --> C:\foo --> C:\bar
  49. // WORKDIR /foo \ WORKDIR c:/bar --> C:\foo --> C:\bar
  50. if len(current) == 0 || system.IsAbs(requested) {
  51. if (requested[0] == os.PathSeparator) ||
  52. (len(requested) > 1 && string(requested[1]) != ":") ||
  53. (len(requested) == 1) {
  54. requested = filepath.Join(`C:\`, requested)
  55. }
  56. } else {
  57. requested = filepath.Join(current, requested)
  58. }
  59. // Upper-case drive letter
  60. return (strings.ToUpper(string(requested[0])) + requested[1:]), nil
  61. }
  62. func errNotJSON(command, original string) error {
  63. // For Windows users, give a hint if it looks like it might contain
  64. // a path which hasn't been escaped such as ["c:\windows\system32\prog.exe", "-param"],
  65. // as JSON must be escaped. Unfortunate...
  66. //
  67. // Specifically looking for quote-driveletter-colon-backslash, there's no
  68. // double backslash and a [] pair. No, this is not perfect, but it doesn't
  69. // have to be. It's simply a hint to make life a little easier.
  70. extra := ""
  71. original = filepath.FromSlash(strings.ToLower(strings.Replace(strings.ToLower(original), strings.ToLower(command)+" ", "", -1)))
  72. if len(regexp.MustCompile(`"[a-z]:\\.*`).FindStringSubmatch(original)) > 0 &&
  73. !strings.Contains(original, `\\`) &&
  74. strings.Contains(original, "[") &&
  75. strings.Contains(original, "]") {
  76. extra = fmt.Sprintf(`. It looks like '%s' includes a file path without an escaped back-slash. JSON requires back-slashes to be escaped such as ["c:\\path\\to\\file.exe", "/parameter"]`, original)
  77. }
  78. return fmt.Errorf("%s requires the arguments to be in JSON form%s", command, extra)
  79. }