Skip to content

Commit 6f50980

Browse files
IanButterworthjohanmon
authored andcommitted
Enable REPL to offer to install missing packages if install hooks are provided (JuliaLang#39026)
1 parent 1adc4e5 commit 6f50980

File tree

2 files changed

+64
-0
lines changed

2 files changed

+64
-0
lines changed

stdlib/REPL/src/REPL.jl

+37
Original file line numberDiff line numberDiff line change
@@ -123,6 +123,12 @@ const softscope! = softscope
123123

124124
const repl_ast_transforms = Any[softscope] # defaults for new REPL backends
125125

126+
# Allows an external package to add hooks into the code loading.
127+
# The hook should take a Vector{Symbol} of package names and
128+
# return true if all packages could be installed, false if not
129+
# to e.g. install packages on demand
130+
const install_packages_hooks = Any[]
131+
126132
function eval_user_input(@nospecialize(ast), backend::REPLBackend)
127133
lasterr = nothing
128134
Base.sigatomic_begin()
@@ -133,6 +139,9 @@ function eval_user_input(@nospecialize(ast), backend::REPLBackend)
133139
put!(backend.response_channel, Pair{Any, Bool}(lasterr, true))
134140
else
135141
backend.in_eval = true
142+
if !isempty(install_packages_hooks)
143+
check_for_missing_packages_and_run_hooks(ast)
144+
end
136145
for xf in backend.ast_transforms
137146
ast = Base.invokelatest(xf, ast)
138147
end
@@ -155,6 +164,34 @@ function eval_user_input(@nospecialize(ast), backend::REPLBackend)
155164
nothing
156165
end
157166

167+
function check_for_missing_packages_and_run_hooks(ast)
168+
mods = modules_to_be_loaded(ast)
169+
filter!(mod -> isnothing(Base.identify_package(String(mod))), mods) # keep missing modules
170+
if !isempty(mods)
171+
for f in install_packages_hooks
172+
Base.invokelatest(f, mods) && return
173+
end
174+
end
175+
end
176+
177+
function modules_to_be_loaded(ast, mods = Symbol[])
178+
if ast.head in [:using, :import]
179+
for arg in ast.args
180+
if first(arg.args) isa Symbol # i.e. `Foo`
181+
if first(arg.args) != :. # don't include local imports
182+
push!(mods, first(arg.args))
183+
end
184+
else # i.e. `Foo: bar`
185+
push!(mods, first(first(arg.args).args))
186+
end
187+
end
188+
end
189+
for arg in ast.args
190+
arg isa Expr && modules_to_be_loaded(arg, mods)
191+
end
192+
return mods
193+
end
194+
158195
"""
159196
start_repl_backend(repl_channel::Channel, response_channel::Channel)
160197

stdlib/REPL/test/repl.jl

+27
Original file line numberDiff line numberDiff line change
@@ -1295,3 +1295,30 @@ Base.wait(frontend_task)
12951295
macro throw_with_linenumbernode(err)
12961296
Expr(:block, LineNumberNode(42, Symbol("test.jl")), :(() -> throw($err)))
12971297
end
1298+
1299+
@testset "Install missing packages via hooks" begin
1300+
@testset "Parse AST for packages" begin
1301+
mods = REPL.modules_to_be_loaded(Meta.parse("using Foo"))
1302+
@test mods == [:Foo]
1303+
mods = REPL.modules_to_be_loaded(Meta.parse("import Foo"))
1304+
@test mods == [:Foo]
1305+
mods = REPL.modules_to_be_loaded(Meta.parse("using Foo, Bar"))
1306+
@test mods == [:Foo, :Bar]
1307+
mods = REPL.modules_to_be_loaded(Meta.parse("import Foo, Bar"))
1308+
@test mods == [:Foo, :Bar]
1309+
1310+
mods = REPL.modules_to_be_loaded(Meta.parse("if false using Foo end"))
1311+
@test mods == [:Foo]
1312+
mods = REPL.modules_to_be_loaded(Meta.parse("if false if false using Foo end end"))
1313+
@test mods == [:Foo]
1314+
mods = REPL.modules_to_be_loaded(Meta.parse("if false using Foo, Bar end"))
1315+
@test mods == [:Foo, :Bar]
1316+
mods = REPL.modules_to_be_loaded(Meta.parse("if false using Foo: bar end"))
1317+
@test mods == [:Foo]
1318+
1319+
mods = REPL.modules_to_be_loaded(Meta.parse("import Foo.bar as baz"))
1320+
@test mods == [:Foo]
1321+
mods = REPL.modules_to_be_loaded(Meta.parse("using .Foo"))
1322+
@test mods == []
1323+
end
1324+
end

0 commit comments

Comments
 (0)