mirror of
https://github.com/docker/compose.git
synced 2026-02-09 01:59:22 +08:00
Most files already grouped imports into "stdlib -> other -> local",
but some files didn't. The gci formatter is similar to goimports, but
has better options to make sure imports are grouped in the expected
order (and to make sure no additional groups are present).
This formatter has a 'fix' function, so code can be re-formatted auto-
matically;
golangci-lint run -v --fix
Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
172 lines
4.2 KiB
Go
172 lines
4.2 KiB
Go
/*
|
|
Copyright 2020 Docker Compose CLI authors
|
|
|
|
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 watch
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"slices"
|
|
"strings"
|
|
|
|
"github.com/compose-spec/compose-go/v2/types"
|
|
"github.com/moby/patternmatcher"
|
|
"github.com/moby/patternmatcher/ignorefile"
|
|
|
|
"github.com/docker/compose/v5/internal/paths"
|
|
)
|
|
|
|
type dockerPathMatcher struct {
|
|
repoRoot string
|
|
matcher *patternmatcher.PatternMatcher
|
|
}
|
|
|
|
func (i dockerPathMatcher) Matches(f string) (bool, error) {
|
|
if !filepath.IsAbs(f) {
|
|
f = filepath.Join(i.repoRoot, f)
|
|
}
|
|
return i.matcher.MatchesOrParentMatches(f)
|
|
}
|
|
|
|
func (i dockerPathMatcher) MatchesEntireDir(f string) (bool, error) {
|
|
matches, err := i.Matches(f)
|
|
if !matches || err != nil {
|
|
return matches, err
|
|
}
|
|
|
|
// We match the dir, but we might exclude files underneath it.
|
|
if i.matcher.Exclusions() {
|
|
for _, pattern := range i.matcher.Patterns() {
|
|
if !pattern.Exclusion() {
|
|
continue
|
|
}
|
|
if paths.IsChild(f, pattern.String()) {
|
|
// Found an exclusion match -- we don't match this whole dir
|
|
return false, nil
|
|
}
|
|
}
|
|
return true, nil
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
func LoadDockerIgnore(build *types.BuildConfig) (PathMatcher, error) {
|
|
if build == nil {
|
|
return EmptyMatcher{}, nil
|
|
}
|
|
repoRoot := build.Context
|
|
absRoot, err := filepath.Abs(repoRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// first try Dockerfile-specific ignore-file
|
|
f, err := os.Open(filepath.Join(repoRoot, build.Dockerfile+".dockerignore"))
|
|
if os.IsNotExist(err) {
|
|
// defaults to a global .dockerignore
|
|
f, err = os.Open(filepath.Join(repoRoot, ".dockerignore"))
|
|
if os.IsNotExist(err) {
|
|
return NewDockerPatternMatcher(repoRoot, nil)
|
|
}
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func() { _ = f.Close() }()
|
|
|
|
patterns, err := readDockerignorePatterns(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return NewDockerPatternMatcher(absRoot, patterns)
|
|
}
|
|
|
|
// Make all the patterns use absolute paths.
|
|
func absPatterns(absRoot string, patterns []string) []string {
|
|
absPatterns := make([]string, 0, len(patterns))
|
|
for _, p := range patterns {
|
|
// The pattern parsing here is loosely adapted from fileutils' NewPatternMatcher
|
|
p = strings.TrimSpace(p)
|
|
if p == "" {
|
|
continue
|
|
}
|
|
p = filepath.Clean(p)
|
|
|
|
pPath := p
|
|
isExclusion := false
|
|
if p[0] == '!' {
|
|
pPath = p[1:]
|
|
isExclusion = true
|
|
}
|
|
|
|
if !filepath.IsAbs(pPath) {
|
|
pPath = filepath.Join(absRoot, pPath)
|
|
}
|
|
absPattern := pPath
|
|
if isExclusion {
|
|
absPattern = fmt.Sprintf("!%s", pPath)
|
|
}
|
|
absPatterns = append(absPatterns, absPattern)
|
|
}
|
|
return absPatterns
|
|
}
|
|
|
|
func NewDockerPatternMatcher(repoRoot string, patterns []string) (*dockerPathMatcher, error) {
|
|
absRoot, err := filepath.Abs(repoRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Check if "*" is present in patterns
|
|
hasAllPattern := slices.Contains(patterns, "*")
|
|
if hasAllPattern {
|
|
// Remove all non-exclusion patterns (those that don't start with '!')
|
|
patterns = slices.DeleteFunc(patterns, func(p string) bool {
|
|
return p != "" && p[0] != '!' // Only keep exclusion patterns
|
|
})
|
|
}
|
|
|
|
pm, err := patternmatcher.New(absPatterns(absRoot, patterns))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &dockerPathMatcher{
|
|
repoRoot: absRoot,
|
|
matcher: pm,
|
|
}, nil
|
|
}
|
|
|
|
func readDockerignorePatterns(r io.Reader) ([]string, error) {
|
|
patterns, err := ignorefile.ReadAll(r)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error reading .dockerignore: %w", err)
|
|
}
|
|
return patterns, nil
|
|
}
|
|
|
|
func DockerIgnoreTesterFromContents(repoRoot string, contents string) (*dockerPathMatcher, error) {
|
|
patterns, err := ignorefile.ReadAll(strings.NewReader(contents))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error reading .dockerignore: %w", err)
|
|
}
|
|
|
|
return NewDockerPatternMatcher(repoRoot, patterns)
|
|
}
|