mirror of
				https://github.com/gohugoio/hugo.git
				synced 2024-05-11 05:54:58 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			159 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2017-present The Hugo Authors. All rights reserved.
 | 
						|
//
 | 
						|
// Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
// you may not use this file except in compliance with the License.
 | 
						|
// You may obtain a copy of the License at
 | 
						|
// http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
//
 | 
						|
// Unless required by applicable law or agreed to in writing, software
 | 
						|
// distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
// See the License for the specific language governing permissions and
 | 
						|
// limitations under the License.
 | 
						|
 | 
						|
package source
 | 
						|
 | 
						|
import (
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"regexp"
 | 
						|
	"runtime"
 | 
						|
 | 
						|
	"github.com/gohugoio/hugo/hugofs/glob"
 | 
						|
 | 
						|
	"github.com/gohugoio/hugo/langs"
 | 
						|
	"github.com/spf13/afero"
 | 
						|
 | 
						|
	"github.com/gohugoio/hugo/helpers"
 | 
						|
	"github.com/spf13/cast"
 | 
						|
)
 | 
						|
 | 
						|
// SourceSpec abstracts language-specific file creation.
 | 
						|
// TODO(bep) rename to Spec
 | 
						|
type SourceSpec struct {
 | 
						|
	*helpers.PathSpec
 | 
						|
 | 
						|
	SourceFs afero.Fs
 | 
						|
 | 
						|
	shouldInclude func(filename string) bool
 | 
						|
 | 
						|
	Languages              map[string]interface{}
 | 
						|
	DefaultContentLanguage string
 | 
						|
	DisabledLanguages      map[string]bool
 | 
						|
}
 | 
						|
 | 
						|
// NewSourceSpec initializes SourceSpec using languages the given filesystem and PathSpec.
 | 
						|
func NewSourceSpec(ps *helpers.PathSpec, inclusionFilter *glob.FilenameFilter, fs afero.Fs) *SourceSpec {
 | 
						|
	cfg := ps.Cfg
 | 
						|
	defaultLang := cfg.GetString("defaultContentLanguage")
 | 
						|
	languages := cfg.GetStringMap("languages")
 | 
						|
 | 
						|
	disabledLangsSet := make(map[string]bool)
 | 
						|
 | 
						|
	for _, disabledLang := range cfg.GetStringSlice("disableLanguages") {
 | 
						|
		disabledLangsSet[disabledLang] = true
 | 
						|
	}
 | 
						|
 | 
						|
	if len(languages) == 0 {
 | 
						|
		l := langs.NewDefaultLanguage(cfg)
 | 
						|
		languages[l.Lang] = l
 | 
						|
		defaultLang = l.Lang
 | 
						|
	}
 | 
						|
 | 
						|
	ignoreFiles := cast.ToStringSlice(cfg.Get("ignoreFiles"))
 | 
						|
	var regexps []*regexp.Regexp
 | 
						|
	if len(ignoreFiles) > 0 {
 | 
						|
		for _, ignorePattern := range ignoreFiles {
 | 
						|
			re, err := regexp.Compile(ignorePattern)
 | 
						|
			if err != nil {
 | 
						|
				helpers.DistinctErrorLog.Printf("Invalid regexp %q in ignoreFiles: %s", ignorePattern, err)
 | 
						|
			} else {
 | 
						|
				regexps = append(regexps, re)
 | 
						|
			}
 | 
						|
 | 
						|
		}
 | 
						|
	}
 | 
						|
	shouldInclude := func(filename string) bool {
 | 
						|
		if !inclusionFilter.Match(filename, false) {
 | 
						|
			return false
 | 
						|
		}
 | 
						|
		for _, r := range regexps {
 | 
						|
			if r.MatchString(filename) {
 | 
						|
				return false
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return true
 | 
						|
	}
 | 
						|
 | 
						|
	return &SourceSpec{shouldInclude: shouldInclude, PathSpec: ps, SourceFs: fs, Languages: languages, DefaultContentLanguage: defaultLang, DisabledLanguages: disabledLangsSet}
 | 
						|
}
 | 
						|
 | 
						|
// IgnoreFile returns whether a given file should be ignored.
 | 
						|
func (s *SourceSpec) IgnoreFile(filename string) bool {
 | 
						|
	if filename == "" {
 | 
						|
		if _, ok := s.SourceFs.(*afero.OsFs); ok {
 | 
						|
			return true
 | 
						|
		}
 | 
						|
		return false
 | 
						|
	}
 | 
						|
 | 
						|
	base := filepath.Base(filename)
 | 
						|
 | 
						|
	if len(base) > 0 {
 | 
						|
		first := base[0]
 | 
						|
		last := base[len(base)-1]
 | 
						|
		if first == '.' ||
 | 
						|
			first == '#' ||
 | 
						|
			last == '~' {
 | 
						|
			return true
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if !s.shouldInclude(filename) {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
 | 
						|
	if runtime.GOOS == "windows" {
 | 
						|
		// Also check the forward slash variant if different.
 | 
						|
		unixFilename := filepath.ToSlash(filename)
 | 
						|
		if unixFilename != filename {
 | 
						|
			if !s.shouldInclude(unixFilename) {
 | 
						|
				return true
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
// IsRegularSourceFile returns whether filename represents a regular file in the
 | 
						|
// source filesystem.
 | 
						|
func (s *SourceSpec) IsRegularSourceFile(filename string) (bool, error) {
 | 
						|
	fi, err := helpers.LstatIfPossible(s.SourceFs, filename)
 | 
						|
	if err != nil {
 | 
						|
		return false, err
 | 
						|
	}
 | 
						|
 | 
						|
	if fi.IsDir() {
 | 
						|
		return false, nil
 | 
						|
	}
 | 
						|
 | 
						|
	if fi.Mode()&os.ModeSymlink == os.ModeSymlink {
 | 
						|
		link, err := filepath.EvalSymlinks(filename)
 | 
						|
		if err != nil {
 | 
						|
			return false, err
 | 
						|
		}
 | 
						|
 | 
						|
		fi, err = helpers.LstatIfPossible(s.SourceFs, link)
 | 
						|
		if err != nil {
 | 
						|
			return false, err
 | 
						|
		}
 | 
						|
 | 
						|
		if fi.IsDir() {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return true, nil
 | 
						|
}
 |