-
Notifications
You must be signed in to change notification settings - Fork 731
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix: Pruning generated files for relative operations #2994
Changes from 4 commits
4ca827b
e061640
fbcf014
fa6d427
9a98154
05b9e89
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -126,6 +126,7 @@ public struct Glob { | |
var parts = pattern.components(separatedBy: String.Globstar) | ||
var firstPart = parts.removeFirst() | ||
let lastPart = parts.joined(separator: String.Globstar) | ||
let lastDirComponent = lastPart.components(separatedBy: "*").first ?? "" | ||
|
||
if isExclude { | ||
// Remove ! here otherwise the Linux glob function would not return any results. Results for | ||
|
@@ -185,7 +186,15 @@ public struct Glob { | |
} | ||
|
||
return OrderedSet<String>(directories.compactMap({ directory in | ||
var path = directory.appendingPathComponent(lastPart).standardizedFileURL.path | ||
// build directory URL up the the '*' in the pattern, and check if it is a symlink, if so skip it | ||
let symCheckURL = directory.appendingPathComponent(lastDirComponent).standardizedFileURL | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @AnthonyMDev this block can run the same symlink check for production use cases we previously had in the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Alternatively, I may be able to build the url up to the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Would look something like this although maybe cleaned up a little:
|
||
if let resourceValues = try? symCheckURL.resourceValues(forKeys: [.isSymbolicLinkKey]), | ||
let isSymLink = resourceValues.isSymbolicLink, | ||
isSymLink { | ||
return nil | ||
} | ||
|
||
var path = directory.resolvingSymlinksInPath().appendingPathComponent(lastPart).standardizedFileURL.path | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @AnthonyMDev Adding |
||
if isExclude { | ||
path.insert("!", at: path.startIndex) | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1446,6 +1446,60 @@ class ApolloCodegenTests: XCTestCase { | |
expect(ApolloFileManager.default.doesFileExist(atPath: testUserFileInChildPath)).to(beTrue()) | ||
expect(ApolloFileManager.default.doesFileExist(atPath: testUserFileInNestedChildPath)).to(beTrue()) | ||
} | ||
|
||
func test__fileDeletion__inOperationRelativeDirectory__whenSymlinkIsUsed() throws { | ||
// given | ||
try createFile(containing: schemaData, named: "schema.graphqls") | ||
|
||
let schemaDirectory = "SchemaModule" | ||
let codeDirectory = "code" | ||
let relativeSubPath = "Operations" | ||
let operationFilename = "TestQuery.graphql" | ||
|
||
try createOperationFile( | ||
type: .query, | ||
named: "TestQuery", | ||
filename: operationFilename, | ||
inDirectory: codeDirectory | ||
) | ||
|
||
let symLinkURL = directoryURL.appendingPathComponent("/\(codeDirectory)/\(relativeSubPath)/") | ||
let symLinkDestURL = directoryURL.appendingPathComponent("\(schemaDirectory)/Sources/Operations/") | ||
let fileValidationPath = symLinkDestURL.appendingPathComponent("\(operationFilename).swift").path | ||
|
||
//setup symlink folder | ||
try testFileManager.fileManager.createDirectory(at: symLinkDestURL, withIntermediateDirectories: true) | ||
try testFileManager.fileManager.createSymbolicLink(at: symLinkURL, withDestinationURL: symLinkDestURL) | ||
|
||
// when | ||
let config = ApolloCodegenConfiguration.mock( | ||
input: .init( | ||
schemaSearchPaths: ["schema*.graphqls"], | ||
operationSearchPaths: ["code/**/*.graphql"] | ||
), | ||
output: .init( | ||
schemaTypes: .init(path: schemaDirectory, | ||
moduleType: .swiftPackageManager), | ||
operations: .relative(subpath: relativeSubPath) | ||
), | ||
options: .init( | ||
pruneGeneratedFiles: true | ||
) | ||
) | ||
|
||
// then | ||
|
||
// running codegen multiple times to validate symlink related file creation/deletion bug | ||
try ApolloCodegen.build(with: config, rootURL: directoryURL) | ||
expect(ApolloFileManager.default.doesFileExist(atPath: fileValidationPath)).to(beTrue()) | ||
|
||
try ApolloCodegen.build(with: config, rootURL: directoryURL) | ||
expect(ApolloFileManager.default.doesFileExist(atPath: fileValidationPath)).to(beTrue()) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Just confirming here, without the code added in this PR, this test passes on the first expectation, fails the second, and passes the third, right? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Correct |
||
|
||
try ApolloCodegen.build(with: config, rootURL: directoryURL) | ||
expect(ApolloFileManager.default.doesFileExist(atPath: fileValidationPath)).to(beTrue()) | ||
|
||
} | ||
|
||
func test__fileDeletion__givenGeneratedFilesExist_InOperationRelativeDirectoriesWithSubPath_deletesOnlyRelativeGeneratedFilesInOperationSearchPaths() throws { | ||
// given | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@calvincestari I ended up refactoring the
Glob
code to this so that the same code in thecompactMap
can handle both production and test use cases instead of checking resource keys for production and then always still resolving symlinks. So just tweaking how the path is built allows for just resolving symlinks for every path that comes through.