}
// Handle the root first
- fileInfo, err := lstatIfOs(fs, root)
+ fileInfo, realPath, err := getRealFileInfo(fs, root)
if err != nil {
return walker(root, nil, err)
}
if !fileInfo.IsDir() {
- return nil
+ return fmt.Errorf("Cannot walk regular file %s", root)
}
- if err := walker(root, fileInfo, err); err != nil && err != filepath.SkipDir {
+ if err := walker(realPath, fileInfo, err); err != nil && err != filepath.SkipDir {
return err
}
}
+func getRealFileInfo(fs afero.Fs, path string) (os.FileInfo, string, error) {
+ fileInfo, err := lstatIfOs(fs, path)
+ realPath := path
+
+ if err != nil {
+ return nil, "", err
+ }
+
+ if fileInfo.Mode()&os.ModeSymlink == os.ModeSymlink {
+ link, err := filepath.EvalSymlinks(path)
+ if err != nil {
+ return nil, "", fmt.Errorf("Cannot read symbolic link '%s', error was: %s", path, err)
+ }
+ fileInfo, err = lstatIfOs(fs, link)
+ if err != nil {
+ return nil, "", fmt.Errorf("Cannot stat '%s', error was: %s", link, err)
+ }
+ realPath = link
+ }
+ return fileInfo, realPath, nil
+}
+
+// GetRealPath returns the real file path for the given path, whether it is a
+// symlink or not.
+func GetRealPath(fs afero.Fs, path string) (string, error) {
+ _, realPath, err := getRealFileInfo(fs, path)
+
+ if err != nil {
+ return "", err
+ }
+
+ return realPath, nil
+}
+
// Code copied from Afero's path.go
// if the filesystem is OsFs use Lstat, else use fs.Stat
func lstatIfOs(fs afero.Fs, path string) (info os.FileInfo, err error) {
"testing"
"time"
+ "github.com/stretchr/testify/assert"
+
"github.com/spf13/afero"
"github.com/spf13/viper"
)
}
}
+func TestGetRealPath(t *testing.T) {
+ d1, err := ioutil.TempDir("", "d1")
+ defer os.Remove(d1)
+ fs := afero.NewOsFs()
+
+ rp1, err := GetRealPath(fs, d1)
+ assert.NoError(t, err)
+ assert.Equal(t, d1, rp1)
+
+ sym := filepath.Join(os.TempDir(), "d1sym")
+ err = os.Symlink(d1, sym)
+ defer os.Remove(sym)
+ assert.NoError(t, err)
+
+ rp2, err := GetRealPath(fs, sym)
+ assert.NoError(t, err)
+
+ // On OS X, the temp folder is itself a symbolic link (to /private...)
+ // This has to do for now.
+ assert.True(t, strings.HasSuffix(rp2, d1))
+
+}
+
func TestMakePathRelative(t *testing.T) {
type test struct {
inPath, path1, path2, output string
logger := helpers.NewDistinctFeedbackLogger()
for _, ev := range events {
- // Need to re-read source
- if strings.HasPrefix(ev.Name, s.absContentDir()) {
+ if s.isContentDirEvent(ev) {
logger.Println("Source changed", ev.Name)
sourceChanged = append(sourceChanged, ev)
}
- if strings.HasPrefix(ev.Name, s.absLayoutDir()) || strings.HasPrefix(ev.Name, s.absThemeDir()) {
+ if s.isLayoutDirEvent(ev) || s.isThemeDirEvent(ev) {
logger.Println("Template changed", ev.Name)
tmplChanged = append(tmplChanged, ev)
}
- if strings.HasPrefix(ev.Name, s.absDataDir()) {
+ if s.isDataDirEvent(ev) {
logger.Println("Data changed", ev.Name)
dataChanged = append(dataChanged, ev)
}
// so we do this first to prevent races.
if ev.Op&fsnotify.Remove == fsnotify.Remove {
//remove the file & a create will follow
- path, _ := helpers.GetRelativePath(ev.Name, s.absContentDir())
+ path, _ := helpers.GetRelativePath(ev.Name, s.getContentDir(ev.Name))
s.removePageByPath(path)
continue
}
if ev.Op&fsnotify.Rename == fsnotify.Rename {
// If the file is still on disk, it's only been updated, if it's not, it's been moved
if ex, err := afero.Exists(hugofs.Source(), ev.Name); !ex || err != nil {
- path, _ := helpers.GetRelativePath(ev.Name, s.absContentDir())
+ path, _ := helpers.GetRelativePath(ev.Name, s.getContentDir(ev.Name))
s.removePageByPath(path)
continue
}
return helpers.AbsPathify(viper.GetString("I18nDir"))
}
+func (s *Site) isDataDirEvent(e fsnotify.Event) bool {
+ return s.getDataDir(e.Name) != ""
+}
+
+func (s *Site) getDataDir(path string) string {
+ return getRealDir(s.absDataDir(), path)
+}
+
func (s *Site) absThemeDir() string {
return helpers.AbsPathify(viper.GetString("themesDir") + "/" + viper.GetString("theme"))
}
+func (s *Site) isThemeDirEvent(e fsnotify.Event) bool {
+ return s.getThemeDir(e.Name) != ""
+}
+
+func (s *Site) getThemeDir(path string) string {
+ return getRealDir(s.absThemeDir(), path)
+}
+
func (s *Site) absLayoutDir() string {
return helpers.AbsPathify(viper.GetString("LayoutDir"))
}
+func (s *Site) isLayoutDirEvent(e fsnotify.Event) bool {
+ return s.getLayoutDir(e.Name) != ""
+}
+
+func (s *Site) getLayoutDir(path string) string {
+ return getRealDir(s.absLayoutDir(), path)
+}
+
func (s *Site) absContentDir() string {
return helpers.AbsPathify(viper.GetString("ContentDir"))
}
+func (s *Site) isContentDirEvent(e fsnotify.Event) bool {
+ return s.getContentDir(e.Name) != ""
+}
+
+func (s *Site) getContentDir(path string) string {
+ return getRealDir(s.absContentDir(), path)
+}
+
+// getRealDir gets the base path of the given path, also handling the case where
+// base is a symlinked folder.
+func getRealDir(base, path string) string {
+
+ if strings.HasPrefix(path, base) {
+ return base
+ }
+
+ realDir, err := helpers.GetRealPath(hugofs.Source(), base)
+
+ if err != nil {
+ if !os.IsNotExist(err) {
+ jww.ERROR.Printf("Failed to get real path for %s: %s", path, err)
+ }
+ return ""
+ }
+
+ if strings.HasPrefix(path, realDir) {
+ return realDir
+ }
+
+ return ""
+}
+
func (s *Site) absPublishDir() string {
return helpers.AbsPathify(viper.GetString("PublishDir"))
}
if err != nil {
return nil, err
}
- file, err = source.NewFileFromAbs(s.absContentDir(), absFilePath, reader)
+ file, err = source.NewFileFromAbs(s.getContentDir(absFilePath), absFilePath, reader)
if err != nil {
return nil, err