@@ -123,6 +123,12 @@ const softscope! = softscope
123
123
124
124
const repl_ast_transforms = Any[softscope] # defaults for new REPL backends
125
125
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
+
126
132
function eval_user_input (@nospecialize (ast), backend:: REPLBackend )
127
133
lasterr = nothing
128
134
Base. sigatomic_begin ()
@@ -133,6 +139,9 @@ function eval_user_input(@nospecialize(ast), backend::REPLBackend)
133
139
put! (backend. response_channel, Pair {Any, Bool} (lasterr, true ))
134
140
else
135
141
backend. in_eval = true
142
+ if ! isempty (install_packages_hooks)
143
+ check_for_missing_packages_and_run_hooks (ast)
144
+ end
136
145
for xf in backend. ast_transforms
137
146
ast = Base. invokelatest (xf, ast)
138
147
end
@@ -155,6 +164,34 @@ function eval_user_input(@nospecialize(ast), backend::REPLBackend)
155
164
nothing
156
165
end
157
166
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
+
158
195
"""
159
196
start_repl_backend(repl_channel::Channel, response_channel::Channel)
160
197
0 commit comments