Skip to content

Commit d785bdc

Browse files
authored
mapexpr argument for include: transform each parsed expression (#34595)
1 parent fa50312 commit d785bdc

9 files changed

+99
-23
lines changed

NEWS.md

+2
Original file line numberDiff line numberDiff line change
@@ -51,6 +51,8 @@ New library functions
5151
`merge!` are still available for backward compatibility ([#34296]).
5252
* The new `isdisjoint` function indicates whether two collections are disjoint ([#34427]).
5353
* Add function `ismutable` and deprecate `isimmutable` to check whether something is mutable.([#34652])
54+
* `include` now accepts an optional `mapexpr` first argument to transform the parsed
55+
expressions before they are evaluated ([#34595]).
5456

5557
New library features
5658
--------------------

base/Base.jl

+7-3
Original file line numberDiff line numberDiff line change
@@ -365,8 +365,8 @@ for m in methods(include)
365365
end
366366
# These functions are duplicated in client.jl/include(::String) for
367367
# nicer stacktraces. Modifications here have to be backported there
368-
include(mod::Module, path::AbstractString) = include(mod, convert(String, path))
369-
function include(mod::Module, _path::String)
368+
include(mod::Module, _path::AbstractString) = include(identity, mod, _path)
369+
function include(mapexpr::Function, mod::Module, _path::AbstractString)
370370
path, prev = _include_dependency(mod, _path)
371371
for callback in include_callbacks # to preserve order, must come before Core.include
372372
invokelatest(callback, mod, path)
@@ -376,7 +376,11 @@ function include(mod::Module, _path::String)
376376
local result
377377
try
378378
# result = Core.include(mod, path)
379-
result = ccall(:jl_load_, Any, (Any, Any), mod, path)
379+
if mapexpr === identity
380+
result = ccall(:jl_load, Any, (Any, Cstring), mod, path)
381+
else
382+
result = ccall(:jl_load_rewrite, Any, (Any, Cstring, Any), mod, path, mapexpr)
383+
end
380384
finally
381385
if prev === nothing
382386
delete!(tls, :SOURCE_PATH)

base/client.jl

+11-5
Original file line numberDiff line numberDiff line change
@@ -423,8 +423,10 @@ end
423423
# MainInclude exists to hide Main.include and eval from `names(Main)`.
424424
baremodule MainInclude
425425
using ..Base
426-
# We inline the definition of include from loading.jl/include_relative to get one-frame stacktraces.
427-
# include(fname::AbstractString) = Main.Base.include(Main, fname)
426+
include(mapexpr::Function, fname::AbstractString) = Base.include(mapexpr, Main, fname)
427+
# We inline the definition of include from loading.jl/include_relative to get one-frame stacktraces
428+
# for the common case of include(fname). Otherwise we would use:
429+
# include(fname::AbstractString) = Base.include(Main, fname)
428430
function include(fname::AbstractString)
429431
mod = Main
430432
isa(fname, String) || (fname = Base.convert(String, fname)::String)
@@ -436,7 +438,7 @@ function include(fname::AbstractString)
436438
tls[:SOURCE_PATH] = path
437439
local result
438440
try
439-
result = ccall(:jl_load_, Any, (Any, Any), mod, path)
441+
result = ccall(:jl_load, Any, (Any, Cstring), mod, path)
440442
finally
441443
if prev === nothing
442444
Base.delete!(tls, :SOURCE_PATH)
@@ -459,16 +461,20 @@ definition of `eval`, which evaluates expressions in that module.
459461
MainInclude.eval
460462

461463
"""
462-
include(path::AbstractString)
464+
include([mapexpr::Function,] path::AbstractString)
463465
464466
Evaluate the contents of the input source file in the global scope of the containing module.
465-
Every module (except those defined with `baremodule`) has its own 1-argument
467+
Every module (except those defined with `baremodule`) has its own
466468
definition of `include`, which evaluates the file in that module.
467469
Returns the result of the last evaluated expression of the input file. During including,
468470
a task-local include path is set to the directory containing the file. Nested calls to
469471
`include` will search relative to that path. This function is typically used to load source
470472
interactively, or to combine files in packages that are broken into multiple source files.
471473
474+
The optional first argument `mapexpr` can be used to transform the included code before
475+
it is evaluated: for each parsed expression `expr` in `path`, the `include` function
476+
actually evaluates `mapexpr(expr)`. If it is omitted, `mapexpr` defaults to [`identity`](@ref).
477+
472478
Use [`Base.include`](@ref) to evaluate a file into another module.
473479
"""
474480
MainInclude.include

base/loading.jl

+24-8
Original file line numberDiff line numberDiff line change
@@ -1073,16 +1073,27 @@ end
10731073
# relative-path load
10741074

10751075
"""
1076-
include_string(m::Module, code::AbstractString, filename::AbstractString="string")
1076+
include_string([mapexpr::Function,] m::Module, code::AbstractString, filename::AbstractString="string")
10771077
10781078
Like [`include`](@ref), except reads code from the given string rather than from a file.
1079+
1080+
The optional first argument `mapexpr` can be used to transform the included code before
1081+
it is evaluated: for each parsed expression `expr` in `code`, the `include_string` function
1082+
actually evaluates `mapexpr(expr)`. If it is omitted, `mapexpr` defaults to [`identity`](@ref).
10791083
"""
1080-
include_string(m::Module, txt::String, fname::String) =
1081-
ccall(:jl_load_file_string, Any, (Ptr{UInt8}, Csize_t, Cstring, Any),
1082-
txt, sizeof(txt), fname, m)
1084+
function include_string(mapexpr::Function, m::Module, txt_::AbstractString, fname::AbstractString="string")
1085+
txt = String(txt_)
1086+
if mapexpr === identity
1087+
ccall(:jl_load_file_string, Any, (Ptr{UInt8}, Csize_t, Cstring, Any),
1088+
txt, sizeof(txt), String(fname), m)
1089+
else
1090+
ccall(:jl_load_rewrite_file_string, Any, (Ptr{UInt8}, Csize_t, Cstring, Any, Any),
1091+
txt, sizeof(txt), String(fname), m, mapexpr)
1092+
end
1093+
end
10831094

10841095
include_string(m::Module, txt::AbstractString, fname::AbstractString="string") =
1085-
include_string(m, String(txt), String(fname))
1096+
include_string(identity, m, txt, fname)
10861097

10871098
function source_path(default::Union{AbstractString,Nothing}="")
10881099
s = current_task().storage
@@ -1098,15 +1109,19 @@ function source_dir()
10981109
end
10991110

11001111
"""
1101-
Base.include([m::Module,] path::AbstractString)
1112+
Base.include([mapexpr::Function,] [m::Module,] path::AbstractString)
11021113
11031114
Evaluate the contents of the input source file in the global scope of module `m`.
1104-
Every module (except those defined with [`baremodule`](@ref)) has its own 1-argument
1105-
definition of `include`, which evaluates the file in that module.
1115+
Every module (except those defined with [`baremodule`](@ref)) has its own
1116+
definition of `include` omitting the `m` argument, which evaluates the file in that module.
11061117
Returns the result of the last evaluated expression of the input file. During including,
11071118
a task-local include path is set to the directory containing the file. Nested calls to
11081119
`include` will search relative to that path. This function is typically used to load source
11091120
interactively, or to combine files in packages that are broken into multiple source files.
1121+
1122+
The optional first argument `mapexpr` can be used to transform the included code before
1123+
it is evaluated: for each parsed expression `expr` in `path`, the `include` function
1124+
actually evaluates `mapexpr(expr)`. If it is omitted, `mapexpr` defaults to [`identity`](@ref).
11101125
"""
11111126
Base.include # defined in sysimg.jl
11121127

@@ -1122,6 +1137,7 @@ function evalfile(path::AbstractString, args::Vector{String}=String[])
11221137
:(const ARGS = $args),
11231138
:(eval(x) = $(Expr(:core, :eval))(__anon__, x)),
11241139
:(include(x) = $(Expr(:top, :include))(__anon__, x)),
1140+
:(include(mapexpr::Function, x) = $(Expr(:top, :include))(mapexpr, __anon__, x)),
11251141
:(include($path))))
11261142
end
11271143
evalfile(path::AbstractString, args::Vector) = evalfile(path, String[args...])

src/ast.c

+17-2
Original file line numberDiff line numberDiff line change
@@ -828,7 +828,8 @@ JL_DLLEXPORT jl_value_t *jl_parse_string(const char *str, size_t len,
828828
// parse and eval a whole file, possibly reading from a string (`content`)
829829
jl_value_t *jl_parse_eval_all(const char *fname,
830830
const char *content, size_t contentlen,
831-
jl_module_t *inmodule)
831+
jl_module_t *inmodule,
832+
jl_value_t *mapexpr)
832833
{
833834
jl_ptls_t ptls = jl_get_ptls_states();
834835
if (ptls->in_pure_callback)
@@ -879,9 +880,16 @@ jl_value_t *jl_parse_eval_all(const char *fname,
879880
JL_TIMING(LOWERING);
880881
if (fl_ctx->T == fl_applyn(fl_ctx, 1, symbol_value(symbol(fl_ctx, "contains-macrocall")), expression)) {
881882
form = scm_to_julia(fl_ctx, expression, inmodule);
883+
if (mapexpr)
884+
form = jl_call1(mapexpr, form);
882885
form = jl_expand_macros(form, inmodule, NULL, 0);
883886
expression = julia_to_scm(fl_ctx, form);
884887
}
888+
else if (mapexpr) {
889+
form = scm_to_julia(fl_ctx, expression, inmodule);
890+
form = jl_call1(mapexpr, form);
891+
expression = julia_to_scm(fl_ctx, form);
892+
}
885893
// expand non-final expressions in statement position (value unused)
886894
expression =
887895
fl_applyn(fl_ctx, 4,
@@ -925,10 +933,17 @@ jl_value_t *jl_parse_eval_all(const char *fname,
925933
return result;
926934
}
927935

936+
JL_DLLEXPORT jl_value_t *jl_load_rewrite_file_string(const char *text, size_t len,
937+
char *filename, jl_module_t *inmodule,
938+
jl_value_t *mapexpr)
939+
{
940+
return jl_parse_eval_all(filename, text, len, inmodule, mapexpr);
941+
}
942+
928943
JL_DLLEXPORT jl_value_t *jl_load_file_string(const char *text, size_t len,
929944
char *filename, jl_module_t *inmodule)
930945
{
931-
return jl_parse_eval_all(filename, text, len, inmodule);
946+
return jl_parse_eval_all(filename, text, len, inmodule, NULL);
932947
}
933948

934949
// returns either an expression or a thunk

src/jlfrontend.scm

+7-2
Original file line numberDiff line numberDiff line change
@@ -180,7 +180,8 @@
180180
(loc (if (and (pair? loc) (eq? (car loc) 'line))
181181
(list loc)
182182
'()))
183-
(x (if (eq? name 'x) 'y 'x)))
183+
(x (if (eq? name 'x) 'y 'x))
184+
(mex (if (eq? name 'mapexpr) 'map_expr 'mapexpr)))
184185
`(block
185186
(= (call eval ,x)
186187
(block
@@ -189,7 +190,11 @@
189190
(= (call include ,x)
190191
(block
191192
,@loc
192-
(call (top include) ,name ,x)))))
193+
(call (top include) ,name ,x)))
194+
(= (call include (:: ,mex (top Function)) ,x)
195+
(block
196+
,@loc
197+
(call (top include) ,mex ,name ,x)))))
193198
'none 0))
194199

195200
; run whole frontend on a string. useful for testing.

src/julia_internal.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -452,7 +452,8 @@ jl_value_t *jl_toplevel_eval_flex(jl_module_t *m, jl_value_t *e, int fast, int e
452452
jl_value_t *jl_eval_global_var(jl_module_t *m JL_PROPAGATES_ROOT, jl_sym_t *e);
453453
jl_value_t *jl_parse_eval_all(const char *fname,
454454
const char *content, size_t contentlen,
455-
jl_module_t *inmodule);
455+
jl_module_t *inmodule,
456+
jl_value_t *mapexpr);
456457
jl_value_t *jl_interpret_toplevel_thunk(jl_module_t *m, jl_code_info_t *src);
457458
jl_value_t *jl_interpret_toplevel_expr_in(jl_module_t *m, jl_value_t *e,
458459
jl_code_info_t *src,

src/toplevel.c

+7-2
Original file line numberDiff line numberDiff line change
@@ -863,13 +863,18 @@ JL_DLLEXPORT jl_value_t *jl_infer_thunk(jl_code_info_t *thk, jl_module_t *m)
863863
return (jl_value_t*)jl_any_type;
864864
}
865865

866-
JL_DLLEXPORT jl_value_t *jl_load(jl_module_t *module, const char *fname)
866+
JL_DLLEXPORT jl_value_t *jl_load_rewrite(jl_module_t *module, const char *fname, jl_value_t *mapexpr)
867867
{
868868
uv_stat_t stbuf;
869869
if (jl_stat(fname, (char*)&stbuf) != 0 || (stbuf.st_mode & S_IFMT) != S_IFREG) {
870870
jl_errorf("could not open file %s", fname);
871871
}
872-
return jl_parse_eval_all(fname, NULL, 0, module);
872+
return jl_parse_eval_all(fname, NULL, 0, module, mapexpr);
873+
}
874+
875+
JL_DLLEXPORT jl_value_t *jl_load(jl_module_t *module, const char *fname)
876+
{
877+
return jl_load_rewrite(module, fname, NULL);
873878
}
874879

875880
// load from filename given as a String object

test/loading.jl

+22
Original file line numberDiff line numberDiff line change
@@ -660,3 +660,25 @@ Base.ACTIVE_PROJECT[] = saved_active_project
660660
module Foo; import Libdl; end
661661
import .Foo.Libdl; import Libdl
662662
@test Foo.Libdl === Libdl
663+
664+
@testset "include with mapexpr" begin
665+
let exprs = Any[]
666+
@test 13 === include_string(@__MODULE__, "1+1\n3*4") do ex
667+
ex isa LineNumberNode || push!(exprs, ex)
668+
Meta.isexpr(ex, :call) ? :(1 + $ex) : ex
669+
end
670+
@test exprs == [:(1 + 1), :(3 * 4)]
671+
end
672+
# test using test_exec.jl, just because that is the shortest handy file
673+
for incl in (include, (mapexpr,path) -> Base.include(mapexpr, @__MODULE__, path))
674+
let exprs = Any[]
675+
incl("test_exec.jl") do ex
676+
ex isa LineNumberNode || push!(exprs, ex)
677+
Meta.isexpr(ex, :macrocall) ? :nothing : ex
678+
end
679+
@test length(exprs) == 2 && exprs[1] == :(using Test)
680+
@test Meta.isexpr(exprs[2], :macrocall) &&
681+
exprs[2].args[[1,3]] == [Symbol("@test"), :(1 == 2)]
682+
end
683+
end
684+
end

0 commit comments

Comments
 (0)