Skip to content

Commit 822b6db

Browse files
committed
loading: add special case for getting moduleroot from a submodule (or itself) without dot
Also clean up and fix some related manual errata that had become incorrect over time.
1 parent 22b71d0 commit 822b6db

File tree

5 files changed

+36
-26
lines changed

5 files changed

+36
-26
lines changed

HISTORY.md

+4
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,10 @@ Language changes
5252
* Errors during `getfield` now raise a new `FieldError` exception type instead of the generic
5353
`ErrorException` ([#54504]).
5454
* Macros in function-signature-position no longer require parentheses. E.g. `function @main(args) ... end` is now permitted, whereas `function (@main)(args) ... end` was required in prior Julia versions.
55+
* Calling `using` on a package name inside of that package of that name (especially relevant
56+
for a submodule) now explicitly uses that package without examining the Manifest and
57+
environment, which is identical to the behavior of `..Name`. This appears to better match
58+
how users expect this to behave in the wild. ([#57727])
5559

5660
Compiler/Runtime improvements
5761
-----------------------------

Makefile

+1
Original file line numberDiff line numberDiff line change
@@ -177,6 +177,7 @@ release-candidate: release testall
177177
@echo 14. Push to Juliaup (https://github.com/JuliaLang/juliaup/wiki/Adding-a-Julia-version)
178178
@echo 15. Announce on mailing lists
179179
@echo 16. Change master to release-0.X in base/version.jl and base/version_git.sh as in 4cb1e20
180+
@echo 17. Move NEWS.md contents to HISTORY.md
180181
@echo
181182

182183
$(build_man1dir)/julia.1: $(JULIAHOME)/doc/man/julia.1 | $(build_man1dir)

base/loading.jl

+4
Original file line numberDiff line numberDiff line change
@@ -2390,6 +2390,10 @@ function __require(into::Module, mod::Symbol)
23902390
error("`using/import $mod` outside of a Module detected. Importing a package outside of a module \
23912391
is not allowed during package precompilation.")
23922392
end
2393+
topmod = moduleroot(into)
2394+
if nameof(topmod) === mod
2395+
return topmod
2396+
end
23932397
@lock require_lock begin
23942398
LOADING_CACHE[] = LoadingCache()
23952399
try

doc/src/manual/modules.md

+14-16
Original file line numberDiff line numberDiff line change
@@ -371,7 +371,7 @@ There are three important standard modules:
371371

372372
Modules can contain *submodules*, nesting the same syntax `module ... end`. They can be used to introduce separate namespaces, which can be helpful for organizing complex codebases. Note that each `module` introduces its own [scope](@ref scope-of-variables), so submodules do not automatically “inherit” names from their parent.
373373

374-
It is recommended that submodules refer to other modules within the enclosing parent module (including the latter) using *relative module qualifiers* in `using` and `import` statements. A relative module qualifier starts with a period (`.`), which corresponds to the current module, and each successive `.` leads to the parent of the current module. This should be followed by modules if necessary, and eventually the actual name to access, all separated by `.`s.
374+
It is recommended that submodules refer to other modules within the enclosing parent module (including the latter) using *relative module qualifiers* in `using` and `import` statements. A relative module qualifier starts with a period (`.`), which corresponds to the current module, and each successive `.` leads to the parent of the current module. This should be followed by modules if necessary, and eventually the actual name to access, all separated by `.`s. As a special case, however, referring to the module root can be written without `.`, avoiding the need to count the depth to reach that module.
375375

376376
Consider the following example, where the submodule `SubA` defines a function, which is then extended in its “sibling” module:
377377

@@ -386,19 +386,24 @@ julia> module ParentModule
386386
export add_D # export it from ParentModule too
387387
module SubB
388388
import ..SubA: add_D # relative path for a “sibling” module
389+
# import ParentModule.SubA: add_D # when in a package, such as when this is loaded by using or import, this would be equivalent to the previous import, but not at the REPL
389390
struct Infinity end
390391
add_D(x::Infinity) = x
391392
end
392393
end;
393394
394395
```
395396

396-
You may see code in packages, which, in a similar situation, uses
397+
You may see code in packages, which, in a similar situation, uses import without the `.`:
398+
```jldoctest
399+
julia> import ParentModule.SubA: add_D
400+
ERROR: ArgumentError: Package ParentModule not found in current path.
401+
```
402+
However, since this operates through [code loading](@ref code-loading), it only works if `ParentModule` is in a package in a file. If `ParentModule` was defined at the REPL, it is necessary to use use relative paths:
397403
```jldoctest module_manual
398404
julia> import .ParentModule.SubA: add_D
399405
400406
```
401-
However, this operates through [code loading](@ref code-loading), and thus only works if `ParentModule` is in a package. It is better to use relative paths.
402407

403408
Note that the order of definitions also matters if you are evaluating values. Consider
404409

@@ -491,8 +496,12 @@ In particular, if you define a `function __init__()` in a module, then Julia wil
491496
immediately *after* the module is loaded (e.g., by `import`, `using`, or `require`) at runtime
492497
for the *first* time (i.e., `__init__` is only called once, and only after all statements in the
493498
module have been executed). Because it is called after the module is fully imported, any submodules
494-
or other imported modules have their `__init__` functions called *before* the `__init__` of the
495-
enclosing module.
499+
or other imported modules have their `__init__` functions called *before* the `__init__` of
500+
the enclosing module. This is also synchronized across threads, so that code can safely rely upon
501+
this ordering of effects, such that all `__init__` will have run, in dependency ordering,
502+
before the `using` result is completed. They may run concurrently with other `__init__`
503+
methods which are not dependencies however, so be careful when accessing any shared state
504+
outside the current module to use locks when needed.
496505

497506
Two typical uses of `__init__` are calling runtime initialization functions of external C libraries
498507
and initializing global constants that involve pointers returned by external libraries. For example,
@@ -524,17 +533,6 @@ pointer value must be called at runtime for precompilation to work ([`Ptr`](@ref
524533
null pointers unless they are hidden inside an [`isbits`](@ref) object). This includes the return values
525534
of the Julia functions [`@cfunction`](@ref) and [`pointer`](@ref).
526535

527-
Dictionary and set types, or in general anything that depends on the output of a `hash(key)` method,
528-
are a trickier case. In the common case where the keys are numbers, strings, symbols, ranges,
529-
`Expr`, or compositions of these types (via arrays, tuples, sets, pairs, etc.) they are safe to
530-
precompile. However, for a few other key types, such as `Function` or `DataType` and generic
531-
user-defined types where you haven't defined a `hash` method, the fallback `hash` method depends
532-
on the memory address of the object (via its `objectid`) and hence may change from run to run.
533-
If you have one of these key types, or if you aren't sure, to be safe you can initialize this
534-
dictionary from within your `__init__` function. Alternatively, you can use the [`IdDict`](@ref)
535-
dictionary type, which is specially handled by precompilation so that it is safe to initialize
536-
at compile-time.
537-
538536
When using precompilation, it is important to keep a clear sense of the distinction between the
539537
compilation phase and the execution phase. In this mode, it will often be much more clearly apparent
540538
that Julia is a compiler which allows execution of arbitrary Julia code, not a standalone interpreter

test/precompile.jl

+13-10
Original file line numberDiff line numberDiff line change
@@ -1595,25 +1595,28 @@ precompile_test_harness("Issue #26028") do load_path
15951595
"""
15961596
module Foo26028
15971597
module Bar26028
1598+
using Foo26028: Foo26028 as InnerFoo1
1599+
using ..Foo26028: Foo26028 as InnerFoo2
15981600
x = 0
15991601
y = 0
16001602
end
16011603
function __init__()
1602-
include(joinpath(@__DIR__, "Baz26028.jl"))
1603-
end
1604+
Baz = @eval module Baz26028
1605+
using Test
1606+
public @test_throws
1607+
import Foo26028.Bar26028.y as y1
1608+
import ..Foo26028.Bar26028.y as y2
1609+
end
1610+
@eval Base \$Baz.@test_throws(ConcurrencyViolationError("deadlock detected in loading Foo26028 using Foo26028"),
1611+
import Foo26028.Bar26028.x)
16041612
end
1605-
""")
1606-
write(joinpath(load_path, "Baz26028.jl"),
1607-
"""
1608-
module Baz26028
1609-
using Test
1610-
@test_throws(ConcurrencyViolationError("deadlock detected in loading Foo26028 using Foo26028"),
1611-
@eval import Foo26028.Bar26028.x)
1612-
import ..Foo26028.Bar26028.y
16131613
end
16141614
""")
16151615
Base.compilecache(Base.PkgId("Foo26028"))
16161616
@test_nowarn @eval using Foo26028
1617+
invokelatest() do
1618+
@test Foo26028 === Foo26028.Bar26028.InnerFoo1 === Foo26028.Bar26028.InnerFoo2
1619+
end
16171620
end
16181621

16191622
precompile_test_harness("Issue #29936") do load_path

0 commit comments

Comments
 (0)