Skip to content

Commit

Permalink
fix(git): fix file list caching bug for repo mode
Browse files Browse the repository at this point in the history
This fixes an error that would come up e.g. when an action config in the
project root excluded a subdirectory containing another action config.

In that setup, the file list for the action in the subdirectory would be
empty because the file list had been cached using the exclude config
applied when computing the file list for the action config at the root
level (which included no files for the excluded subdirectory).

This was fixed by adding another level of caching to `GitRepoHandler`
that accounts for different includes/excludes and filter functions for
the same path.
  • Loading branch information
thsig committed Feb 6, 2024
1 parent 54c71a4 commit 66d56ee
Show file tree
Hide file tree
Showing 5 changed files with 210 additions and 71 deletions.
50 changes: 35 additions & 15 deletions core/src/vcs/git-repo.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@ import fsExtra from "fs-extra"
import { pathToCacheContext } from "../cache.js"
import { FileTree } from "./file-tree.js"
import { normalize, sep } from "path"
import { stableStringify } from "../util/string.js"
import { hashString } from "../util/util.js"

const { pathExists } = fsExtra

Expand Down Expand Up @@ -62,10 +64,7 @@ export class GitRepoHandler extends GitHandler {
override async getFiles(params: GetFilesParams): Promise<VcsFile[]> {
const { log, path, pathDescription, filter, failOnPrompt = false } = params

if (params.include && params.include.length === 0) {
// No need to proceed, nothing should be included
return []
}
let scanRoot = params.scanRoot || path

if (!(await pathExists(path))) {
log.warn(`${pathDescription} ${path} could not be found.`)
Expand All @@ -77,12 +76,32 @@ export class GitRepoHandler extends GitHandler {
return []
}

let scanRoot = params.scanRoot || path

if (!params.scanRoot && params.pathDescription !== "submodule") {
scanRoot = await this.getRepoRoot(log, path, failOnPrompt)
}

const scanFromProjectRoot = scanRoot === this.garden?.projectRoot
const { augmentedExcludes, augmentedIncludes } = await getIncludeExcludeFiles({ ...params, scanFromProjectRoot })

const hashedFilterParams = hashString(
stableStringify({
filter: filter ? filter.toString() : undefined, // We hash the source code of the filter function if provided.
augmentedIncludes,
augmentedExcludes,
})
)
const filteredFilesCacheKey = ["git-repo-files", path, hashedFilterParams]

const cached = this.cache.get(log, filteredFilesCacheKey) as VcsFile[] | undefined
if (cached) {
return cached
}

if (params.include && params.include.length === 0) {
// No need to proceed, nothing should be included
return []
}

const fileTree = await this.scanRepo({
log,
path: scanRoot,
Expand All @@ -96,23 +115,20 @@ export class GitRepoHandler extends GitHandler {
// Thus, some excluded files can appear in the resulting fileTree.
// It happens because the new "repo" mode does not use `--glob-pathspecs` flag
// and does some explicit glob patterns augmentation that misses some edge-cases.
exclude: params.exclude,
// exclude: params.exclude,
})

const moduleFiles = fileTree.getFilesAtPath(path)

const scanFromProjectRoot = scanRoot === this.garden?.projectRoot
const { augmentedExcludes, augmentedIncludes } = await getIncludeExcludeFiles({ ...params, scanFromProjectRoot })
const filesAtPath = fileTree.getFilesAtPath(path)

log.debug(
`Found ${moduleFiles.length} files in module path, filtering by ${augmentedIncludes.length} include and ${augmentedExcludes.length} exclude globs`
`Found ${filesAtPath.length} files in path, filtering by ${augmentedIncludes.length} include and ${augmentedExcludes.length} exclude globs`
)
log.silly(() => `Include globs: ${augmentedIncludes.join(", ")}`)
log.silly(() =>
augmentedExcludes.length > 0 ? `Exclude globs: ${augmentedExcludes.join(", ")}` : "No exclude globs"
)

const filtered = moduleFiles.filter(({ path: p }) => {
const filtered = filesAtPath.filter(({ path: p }) => {
if (filter && !filter(p)) {
return false
}
Expand All @@ -122,11 +138,15 @@ export class GitRepoHandler extends GitHandler {
// but that caused issues with the glob matching on windows due to backslashes
const relativePath = p.replace(`${path}${sep}`, "")
log.silly(() => `Checking if ${relativePath} matches include/exclude globs`)
return matchPath(relativePath, augmentedIncludes, augmentedExcludes)
const matches = matchPath(relativePath, augmentedIncludes, augmentedExcludes)
log.debug(`rel path: ${relativePath}, matches: ${matches}`)
return matches
})

log.debug(`Found ${filtered.length} files in module path after glob matching`)

this.cache.set(log, filteredFilesCacheKey, filtered, pathToCacheContext(path))

return filtered
}

Expand All @@ -140,7 +160,7 @@ export class GitRepoHandler extends GitHandler {
const { log, path } = params

const key = ["git-repo-files", path]
let existing = this.cache.get(log, key) as FileTree
let existing = this.cache.get(log, key) as FileTree | undefined

if (existing) {
params.log.silly(() => `Found cached repository match at ${path}`)
Expand Down
6 changes: 3 additions & 3 deletions core/src/vcs/git.ts
Original file line number Diff line number Diff line change
Expand Up @@ -200,13 +200,12 @@ export class GitHandler extends VcsHandler {
}

const { log, path, pathDescription = "directory", filter, failOnPrompt = false } = params
const { exclude, hasIncludes, include } = await getIncludeExcludeFiles(params)

const gitLog = log
.createLog({ name: "git" })
.debug(
`Scanning ${pathDescription} at ${path}\n → Includes: ${include || "(none)"}\n → Excludes: ${
exclude || "(none)"
`Scanning ${pathDescription} at ${path}\n → Includes: ${params.include || "(none)"}\n → Excludes: ${
params.exclude || "(none)"
}`
)

Expand All @@ -225,6 +224,7 @@ export class GitHandler extends VcsHandler {
throw err
}
}
const { exclude, hasIncludes, include } = await getIncludeExcludeFiles(params)

let files: VcsFile[] = []

Expand Down
Loading

0 comments on commit 66d56ee

Please sign in to comment.