2014-05-13 17:34:30 +00:00
|
|
|
package symlink
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
func abs(t *testing.T, p string) string {
|
|
|
|
o, err := filepath.Abs(p)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return o
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFollowSymLinkNormal(t *testing.T) {
|
|
|
|
link := "testdata/fs/a/d/c/data"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/b/c/data"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-15 21:52:36 +00:00
|
|
|
func TestFollowSymLinkRelativePath(t *testing.T) {
|
|
|
|
link := "testdata/fs/i"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/fs/a"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-13 17:34:30 +00:00
|
|
|
func TestFollowSymLinkUnderLinkedDir(t *testing.T) {
|
|
|
|
dir, err := ioutil.TempDir("", "docker-fs-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2014-10-28 21:18:45 +00:00
|
|
|
defer os.RemoveAll(dir)
|
2014-05-13 17:34:30 +00:00
|
|
|
|
|
|
|
os.Mkdir(filepath.Join(dir, "realdir"), 0700)
|
|
|
|
os.Symlink("realdir", filepath.Join(dir, "linkdir"))
|
|
|
|
|
|
|
|
linkDir := filepath.Join(dir, "linkdir", "foo")
|
|
|
|
dirUnderLinkDir := filepath.Join(dir, "linkdir", "foo", "bar")
|
|
|
|
os.MkdirAll(dirUnderLinkDir, 0700)
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(dirUnderLinkDir, linkDir)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if rewrite != dirUnderLinkDir {
|
|
|
|
t.Fatalf("Expected %s got %s", dirUnderLinkDir, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFollowSymLinkRandomString(t *testing.T) {
|
|
|
|
if _, err := FollowSymlinkInScope("toto", "testdata"); err == nil {
|
|
|
|
t.Fatal("Random string should fail but didn't")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFollowSymLinkLastLink(t *testing.T) {
|
|
|
|
link := "testdata/fs/a/d"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/b"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFollowSymLinkRelativeLink(t *testing.T) {
|
|
|
|
link := "testdata/fs/a/e/c/data"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/fs/b/c/data"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFollowSymLinkRelativeLinkScope(t *testing.T) {
|
2014-10-28 21:18:45 +00:00
|
|
|
// avoid letting symlink f lead us out of the "testdata" scope
|
|
|
|
// we don't normalize because symlink f is in scope and there is no
|
|
|
|
// information leak
|
|
|
|
{
|
|
|
|
link := "testdata/fs/a/f"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/test"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// avoid letting symlink f lead us out of the "testdata/fs" scope
|
|
|
|
// we don't normalize because symlink f is in scope and there is no
|
|
|
|
// information leak
|
|
|
|
{
|
|
|
|
link := "testdata/fs/a/f"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata/fs")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/fs/test"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// avoid letting symlink g (pointed at by symlink h) take out of scope
|
|
|
|
// TODO: we should probably normalize to scope here because ../[....]/root
|
|
|
|
// is out of scope and we leak information
|
|
|
|
{
|
|
|
|
link := "testdata/fs/b/h"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/root"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// avoid letting allowing symlink e lead us to ../b
|
|
|
|
// normalize to the "testdata/fs/a"
|
|
|
|
{
|
|
|
|
link := "testdata/fs/a/e"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata/fs/a")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/fs/a"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// avoid letting symlink -> ../directory/file escape from scope
|
|
|
|
// normalize to "testdata/fs/j"
|
|
|
|
{
|
|
|
|
link := "testdata/fs/j/k"
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(link, "testdata/fs/j")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if expected := abs(t, "testdata/fs/j"); expected != rewrite {
|
|
|
|
t.Fatalf("Expected %s got %s", expected, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure we don't allow escaping to /
|
|
|
|
// normalize to dir
|
|
|
|
{
|
|
|
|
dir, err := ioutil.TempDir("", "docker-fs-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
|
|
|
|
linkFile := filepath.Join(dir, "foo")
|
|
|
|
os.Mkdir(filepath.Join(dir, ""), 0700)
|
|
|
|
os.Symlink("/", linkFile)
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(linkFile, dir)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if rewrite != dir {
|
|
|
|
t.Fatalf("Expected %s got %s", dir, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure we don't allow escaping to /
|
|
|
|
// normalize to dir
|
|
|
|
{
|
|
|
|
dir, err := ioutil.TempDir("", "docker-fs-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
|
|
|
|
linkFile := filepath.Join(dir, "foo")
|
|
|
|
os.Mkdir(filepath.Join(dir, ""), 0700)
|
|
|
|
os.Symlink("/../../", linkFile)
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(linkFile, dir)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if rewrite != dir {
|
|
|
|
t.Fatalf("Expected %s got %s", dir, rewrite)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure we stay in scope without leaking information
|
|
|
|
// this also checks for escaping to /
|
|
|
|
// normalize to dir
|
|
|
|
{
|
|
|
|
dir, err := ioutil.TempDir("", "docker-fs-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
|
|
|
|
linkFile := filepath.Join(dir, "foo")
|
|
|
|
os.Mkdir(filepath.Join(dir, ""), 0700)
|
|
|
|
os.Symlink("../../", linkFile)
|
|
|
|
|
|
|
|
rewrite, err := FollowSymlinkInScope(linkFile, dir)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if rewrite != dir {
|
|
|
|
t.Fatalf("Expected %s got %s", dir, rewrite)
|
|
|
|
}
|
2014-05-13 17:34:30 +00:00
|
|
|
}
|
|
|
|
}
|