Skip to content
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: Attempt to work around PackagePlugin.Path deprecation #125

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

paulgessinger
Copy link

Currently, the build of this package (and dependees) produces a number of warnings like:

warning: 'appending' is deprecated: Use `URL` type instead of `Path`.
 51 | 
 52 |         // Setup folder
 53 |         let genFolder = context.pluginWorkDirectory.appending(["ProtocolGen"])
    |                                                     `- warning: 'appending' is deprecated: Use `URL` type instead of `Path`.
 54 |         try FileManager.default.createDirectory(
 55 |             atPath: genFolder.string, withIntermediateDirectories: true

This (to my understanding) is because the type PackagePlugin.Path is deprecated in favor of Foundation.URL, but the replacements accessor properties are not universally implemented until (I think) version 6.1.

In this PR, I'm attempting to remove these warnings. In Plugin.swift this is relatively straightforward, but in SwiftPackageTarget.swift it's a bit tricky:

The procotol SourceModuleTarget does not currently include the needed var directoryURL: URL, even though both (and I think those are the only two types that conform to this protocol) ClangSourceModuleTarget and SwiftSourceModuleTarget have this property. I'm adding here a workaround that tries to downcast the incoming SourceModuleTarget to either of these types, and then packages it in a temporary helper protocol CompatSourceModuleTarget that exposes the directoryURL property.

I'm not sure if this is the best way to this, but I thought I'd open a PR to discuss it. I can't run the tests included in the repo (even on main they fail for me), so I don't know if the changes are fully correct, aside from compiling without warnings.

Let me know what you think!

@@ -50,32 +50,32 @@ struct MetaProtocolCodable: BuildToolPlugin {
let (allTargets, imports) = config.scanInput(for: target, in: context)

// Setup folder
let genFolder = context.pluginWorkDirectory.appending(["ProtocolGen"])
let genFolder = context.pluginWorkDirectoryURL
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wouldn't this change the path for generated source files?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Overlooked this, yes. I'll change it

/// See: https://github.com/swiftlang/swift-package-manager/blob/735ddd97fa651729921315c8e46bd790429362cb/Sources/PackagePlugin/PackageModel.swift#L184-L186///
/// The workaround defines a custom protocol that adds the missing property, and then introduces
/// a new initializer that accepts the actual target protocol and attempts to downcast.
protocol CompatSourceModuleTarget: SourceModuleTarget {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not use an extension to define this property for older versions instead? Won't that be much simpler than current implementation?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You mean an extension that adds it to SourceModuleTarget and then does the downcast to call into the underlying property? Otherwise, I'm not sure I'm following what you're suggesting.

Copy link
Contributor

@soumyamahunt soumyamahunt Mar 18, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was thinking an implementation of directoryURL that is used for older swift versions. This could live as an extension while restricted with compiler directive.

This would avoid breaking change for older versions.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hm, but wouldn't I need this in the protocol that's being used internally? The underlying types expose directoryURL already in Swift 6, but I don't know how I could access the implementation without downcasting.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@paulgessinger I was thinking extension with compiler directive like this:

#if swift(<6.1)
extension SourceModuleTarget {
    var directoryURL: URL {
        // create directory url for older versions
    }
}
#endif

This way newer properties can be used providing backwards compatibility.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But that would still require downcasting inside the implementation, since I can't convert the Path type to URL without triggering the deprecation warning, wouldn't it?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

since I can't convert the Path type to URL without triggering the deprecation warning

The warning will only be generated for Swift 6 version if I am not wrong. For other versions I don't think warning will be generated. For me that is an acceptable solution than dropping compatibility for Swift 6 and below.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right. This PR is an attempt to avoid the warning in Swift 6

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@paulgessinger may be you can do something like this that handles warning with Swift 6 version as well:

#if swift(<6.1)
extension SourceModuleTarget {
    var directoryURL: URL {
        #if swift(<6)
        // create directory url for older versions convert the Path type to URL
        #else
        // create directory url with down-casting
        #endif
    }
}
#endif

var buildCommands = allTargets.flatMap { target in
return target.sourceFiles(withSuffix: "swift").map { file in
let moduleName = target.moduleName
let fileName = file.path.stem
let fileName = file.url.lastPathComponent
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wouldn't this change break compatibility with older Swift version?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes I think so. That would be a tradeoff. Maybe it's possible to partition the implementation into different functions that are called based on the Swift version to avoid the deprecated behavior only in Swift 6.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

None yet

2 participants