- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 5.7k
Add support for "package extensions" to code loading #47695
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
Conversation
53d34f2    to
    a2b2714      
    Compare
  
    | Pushed a bump to Pkg here to include JuliaLang/Pkg.jl#3264 to run CI of these together. | 
| Does this have downsides/a tradeoff compared to #47040? | 
| Some are discussed in #47040 (comment) | 
| Gotcha, I just thought there may be some other ones. Thinking about the description of this in the Pkg PR, what happens when there's glue needed between glue, e.g. when you have weak dependencies  | 
| 
 Yes. Another question is, what if you want to activate extra functionality inside a glue package when some other package is loaded. Basically glue^2. The reason I think about this is that I saw this line: in Plots, where a  [gluedeps]
PGFPlotsX = "..."
Measurements = "..."
[gluepkgs]
PGFPlotsGlue = "PGFPlotsX"
PGFPlotsMeasurementGlue = ["PGFPlotsGlue", "Measurements"]And  | 
| 
 And that's exactly the reason why I'm more in favor of the other proposal :D No risk of accidentally going quadratic in terms of complexity, at the cost of penalizing larger environments a bit (which probably should just use a large sysimage anyway?). | 
| @nanosoldier  | 
0ca5dbf    to
    3c4559e      
    Compare
  
    | 
 The weak deps idea has some advantages and is more "powerful" since you have access to all information during precompile time, yes. But after thinking about this for quite some time I do think that the suggestion here is what should be tried first. Some reasons: 
 | 
| Your package evaluation job has completed - possible new issues were detected. A full report can be found here. | 
| PkgEval looks good. | 
        
          
                doc/src/manual/code-loading.md
              
                Outdated
          
        
      | 2. Packages in non-primary environments can end up using incompatible versions of their dependencies even if their own environments are entirely compatible. This can happen when one of their dependencies is shadowed by a version in an earlier environment in the stack (either by graph or path, or both). | ||
|  | ||
| Since the primary environment is typically the environment of a project you're working on, while environments later in the stack contain additional tools, this is the right trade-off: it's better to break your development tools but keep the project working. When such incompatibilities occur, you'll typically want to upgrade your dev tools to versions that are compatible with the main project. | ||
| ### "Glue" packages and dependencies | 
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.
Obviously there are more changes to like this, but just to mark it as I think this is where the discussion settled
| ### "Glue" packages and dependencies | |
| ### "Weak" dependencies and "Glue" packages | 
79a554d    to
    714933b      
    Compare
  
    | For those following along, it is likely that "glue packages" will be renamed to "package extensions". | 
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.
Awesome.
* Add support for "glue packages" to code loading This allows packages to define "glue packages" which are modules that are automatically loaded when a set of other packages are loaded into the Julia session. (cherry picked from commit 495a004)
| else | ||
| require(extid.id) | ||
| @debug "Extension $(extid.id.name) of $(extid.parentid.name) loaded" | ||
| end | 
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.
Not sure if this is a deliberate design so wanna ask it here. Does it sound like a good idea to make Plotting.ContourExt a "normal" submodule? And if not loaded, Plotting.ContourExt keeps to be nothing.
This makes it easy and intuitive to get objects inside the glue module, for instance:
Base.get_extension(Plotting, :ContourExt).Contourbecomes
Plotting.ContourExt.ContourOne hack/workaround is to insert
module ContourExt
using Plotting
function __init__()
    setproperty!(Plotting, :ContourExt, Base.get_extension(Plotting, :ContourExt))
end
endThere 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.
The extensions should in my opinion be "private" and their behavior should almost only be visible to users by new dispatch rules being defined. You shouldn't really need to reach into the extension module itself. The get_extension is thus mostly for debugging.
This is an implementation of the proposal in #43119 (with some tweaks) that aims to replace Requires.jl for loading code based on the presence of other packages. It is a different implementation to #47040 and should work better in for example big environments where you expect to only load a small subset of packages.
In spirit, it is very similar to Requires.jl in that given a set of packages and a file, you load the file when those packages have been loaded in the julia session. The differences to Requires.jl are:
__init__(which is done when using Requries.jl).The added docs in code loading as well as the added docs in the accompanying Pkg PR (JuliaLang/Pkg.jl#3264) can be used to get a more detail description of its usage.
A lot of the diff in this PR is due to the code loading being pretty awkward and would have been easier to be built on top of #46690 but I don't want to couple those PRs.
For reviewers: going through all the uuid lookups etc in the code loading is probably fairly uninteresting since it is mostly mechanical, the interesting parts are those around line 1080 - 1200 in
loading.jlwhich deals with the insertion of gluepkgs "callbacks" and the loading of them.An example of a real package moving from Requires.jl to using this system can be seen in https://github.com/KristofferC/PGFPlotsX.jl/compare/kc/glue. It should be pretty easy to be backwards compatible with Requires. You just put the
@requireinside a version check and include the file for the conditional file in the@requireblock.One TODO that is left is that Pkg should precompile these "glue packages" during
Pkg.precompile(which is often run automatically) in the case of them having a chance to be loaded in the current environment.