Skip to content

Commit 1e70913

Browse files
LilithHafnerc42f
authored andcommitted
Add public keyword (#50105)
Co-authored-by: Claire Foster <[email protected]>
1 parent 1175944 commit 1e70913

File tree

30 files changed

+432
-90
lines changed

30 files changed

+432
-90
lines changed

base/docs/basedocs.jl

+11-1
Original file line numberDiff line numberDiff line change
@@ -52,13 +52,23 @@ kw"import"
5252
"""
5353
export
5454
55-
`export` is used within modules to tell Julia which functions should be
55+
`export` is used within modules to tell Julia which names should be
5656
made available to the user. For example: `export foo` makes the name
5757
`foo` available when [`using`](@ref) the module.
5858
See the [manual section about modules](@ref modules) for details.
5959
"""
6060
kw"export"
6161

62+
"""
63+
public
64+
65+
`public` is used within modules to tell Julia which names are part of the
66+
public API of the module . For example: `public foo` indicates that the name
67+
`foo` is public, without making it available available when [`using`](@ref)
68+
the module. See the [manual section about modules](@ref modules) for details.
69+
"""
70+
kw"public"
71+
6272
"""
6373
as
6474

base/exports.jl

+81
Original file line numberDiff line numberDiff line change
@@ -1063,3 +1063,84 @@ export
10631063
@view,
10641064
@views,
10651065
@static
1066+
1067+
# TODO: use normal syntax once JuliaSyntax.jl becomes available at this point in bootstrapping
1068+
eval(Expr(:public,
1069+
# Modules
1070+
:Checked,
1071+
:Filesystem,
1072+
:Order,
1073+
:Sort,
1074+
1075+
# Types
1076+
:AbstractLock,
1077+
:AsyncCondition,
1078+
:CodeUnits,
1079+
:Event,
1080+
:Fix1,
1081+
:Fix2,
1082+
:Generator,
1083+
:ImmutableDict,
1084+
:OneTo,
1085+
:UUID,
1086+
1087+
# Semaphores
1088+
:Semaphore,
1089+
:acquire,
1090+
:release,
1091+
1092+
# collections
1093+
:IteratorEltype,
1094+
:IteratorSize,
1095+
:to_index,
1096+
:vect,
1097+
:isdone,
1098+
:front,
1099+
:rest,
1100+
:split_rest,
1101+
:tail,
1102+
:checked_length,
1103+
1104+
# Loading
1105+
:DL_LOAD_PATH,
1106+
:load_path,
1107+
:active_project,
1108+
1109+
# Reflection and introspection
1110+
:isambiguous,
1111+
:isexpr,
1112+
:isidentifier,
1113+
:issingletontype,
1114+
:identify_package,
1115+
:locate_package,
1116+
:moduleroot,
1117+
:jit_total_bytes,
1118+
:summarysize,
1119+
:isexported,
1120+
:ispublic,
1121+
1122+
# Opperators
1123+
:operator_associativity,
1124+
:operator_precedence,
1125+
:isbinaryoperator,
1126+
:isoperator,
1127+
:isunaryoperator,
1128+
1129+
# C interface
1130+
:cconvert,
1131+
:unsafe_convert,
1132+
1133+
# Error handling
1134+
:exit_on_sigint,
1135+
:windowserror,
1136+
1137+
# Macros
1138+
Symbol("@assume_effects"),
1139+
Symbol("@constprop"),
1140+
Symbol("@locals"),
1141+
Symbol("@propagate_inbounds"),
1142+
1143+
# misc
1144+
:notnothing,
1145+
:runtests,
1146+
:text_colors))

base/reflection.jl

+81-5
Original file line numberDiff line numberDiff line change
@@ -75,24 +75,100 @@ end
7575
"""
7676
names(x::Module; all::Bool = false, imported::Bool = false)
7777
78-
Get an array of the names exported by a `Module`, excluding deprecated names.
79-
If `all` is true, then the list also includes non-exported names defined in the module,
78+
Get an array of the public names of a `Module`, excluding deprecated names.
79+
If `all` is true, then the list also includes non-public names defined in the module,
8080
deprecated names, and compiler-generated names.
8181
If `imported` is true, then names explicitly imported from other modules
8282
are also included.
8383
84-
As a special case, all names defined in `Main` are considered \"exported\",
85-
since it is not idiomatic to explicitly export names from `Main`.
84+
As a special case, all names defined in `Main` are considered \"public\",
85+
since it is not idiomatic to explicitly mark names from `Main` as public.
8686
87-
See also: [`@locals`](@ref Base.@locals), [`@__MODULE__`](@ref).
87+
See also: [`isexported`](@ref), [`ispublic`](@ref), [`@locals`](@ref Base.@locals), [`@__MODULE__`](@ref).
8888
"""
8989
names(m::Module; all::Bool = false, imported::Bool = false) =
9090
sort!(unsorted_names(m; all, imported))
9191
unsorted_names(m::Module; all::Bool = false, imported::Bool = false) =
9292
ccall(:jl_module_names, Array{Symbol,1}, (Any, Cint, Cint), m, all, imported)
9393

94+
"""
95+
isexported(m::Module, s::Symbol) -> Bool
96+
97+
Returns whether a symbol is exported from a module.
98+
99+
See also: [`ispublic`](@ref), [`names`](@ref)
100+
101+
```jldoctest
102+
julia> module Mod
103+
export foo
104+
public bar
105+
end
106+
Mod
107+
108+
julia> Base.isexported(Mod, :foo)
109+
true
110+
111+
julia> Base.isexported(Mod, :bar)
112+
false
113+
114+
julia> Base.isexported(Mod, :baz)
115+
false
116+
```
117+
"""
94118
isexported(m::Module, s::Symbol) = ccall(:jl_module_exports_p, Cint, (Any, Any), m, s) != 0
119+
120+
"""
121+
ispublic(m::Module, s::Symbol) -> Bool
122+
123+
Returns whether a symbol is marked as public in a module.
124+
125+
Exported symbols are considered public.
126+
127+
See also: [`isexported`](@ref), [`names`](@ref)
128+
129+
```jldoctest
130+
julia> module Mod
131+
export foo
132+
public bar
133+
end
134+
Mod
135+
136+
julia> Base.ispublic(Mod, :foo)
137+
true
138+
139+
julia> Base.ispublic(Mod, :bar)
140+
true
141+
142+
julia> Base.ispublic(Mod, :baz)
143+
false
144+
```
145+
"""
146+
ispublic(m::Module, s::Symbol) = ccall(:jl_module_public_p, Cint, (Any, Any), m, s) != 0
147+
148+
# TODO: this is vaguely broken because it only works for explicit calls to
149+
# `Base.deprecate`, not the @deprecated macro:
95150
isdeprecated(m::Module, s::Symbol) = ccall(:jl_is_binding_deprecated, Cint, (Any, Any), m, s) != 0
151+
152+
"""
153+
isbindingresolved(m::Module, s::Symbol) -> Bool
154+
155+
Returns whether the binding of a symbol in a module is resolved.
156+
157+
See also: [`isexported`](@ref), [`ispublic`](@ref), [`isdeprecated`](@ref)
158+
159+
```jldoctest
160+
julia> module Mod
161+
foo() = 17
162+
end
163+
Mod
164+
165+
julia> Base.isbindingresolved(Mod, :foo)
166+
true
167+
168+
julia> Base.isbindingresolved(Mod, :bar)
169+
false
170+
```
171+
"""
96172
isbindingresolved(m::Module, var::Symbol) = ccall(:jl_binding_resolved_p, Cint, (Any, Any), m, var) != 0
97173

98174
function binding_module(m::Module, s::Symbol)

base/show.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -2226,7 +2226,7 @@ function show_unquoted(io::IO, ex::Expr, indent::Int, prec::Int, quote_level::In
22262226
print(io, head, ' ')
22272227
show_list(io, args, ", ", indent, 0, quote_level)
22282228

2229-
elseif head === :export
2229+
elseif head in (:export, :public)
22302230
print(io, head, ' ')
22312231
show_list(io, mapany(allow_macroname, args), ", ", indent)
22322232

deps/JuliaSyntax.version

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
11
JULIASYNTAX_BRANCH = main
2-
JULIASYNTAX_SHA1 = 045d156c44dbb87769c7416d049a7c08908539d4
2+
JULIASYNTAX_SHA1 = a9110fa8ecbe79943bb9525b4ccd99a3976cfcb7
33
JULIASYNTAX_GIT_URL := https://github.com/JuliaLang/JuliaSyntax.jl.git
44
JULIASYNTAX_TAR_URL = https://api.github.com/repos/JuliaLang/JuliaSyntax.jl/tarball/$1

deps/checksums/JuliaSyntax-045d156c44dbb87769c7416d049a7c08908539d4.tar.gz/md5

-1
This file was deleted.

deps/checksums/JuliaSyntax-045d156c44dbb87769c7416d049a7c08908539d4.tar.gz/sha512

-1
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
a42bbb42babbbd727556f6bc01455826
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
d542766e72b57418b9b4e17743f89d8535c1f36497346b57538bd0cb451e64af9493015692179f80ec4ee8cf18349c1e0888f5710db895e19f9bb0322f0f7464

doc/src/base/base.md

+4
Original file line numberDiff line numberDiff line change
@@ -60,13 +60,15 @@ However, you can create variables with names:
6060
Finally:
6161
`where` is parsed as an infix operator for writing parametric method and type definitions;
6262
`in` and `isa` are parsed as infix operators;
63+
`public` is parsed as a keyword when beginning a toplevel statement;
6364
`outer` is parsed as a keyword when used to modify the scope of a variable in an iteration specification of a `for` loop;
6465
and `as` is used as a keyword to rename an identifier brought into scope by `import` or `using`.
6566
Creation of variables named `where`, `in`, `isa`, `outer` and `as` is allowed, though.
6667

6768
```@docs
6869
module
6970
export
71+
public
7072
import
7173
using
7274
as
@@ -451,6 +453,8 @@ Base.@__DIR__
451453
Base.@__LINE__
452454
Base.fullname
453455
Base.names
456+
Base.isexported
457+
Base.ispublic
454458
Base.nameof(::Function)
455459
Base.functionloc(::Any, ::Any)
456460
Base.functionloc(::Method)

doc/src/base/reflection.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,9 @@ Julia provides a variety of runtime reflection capabilities.
44

55
## Module bindings
66

7-
The exported names for a `Module` are available using [`names(m::Module)`](@ref), which will return
8-
an array of [`Symbol`](@ref) elements representing the exported bindings. `names(m::Module, all = true)`
9-
returns symbols for all bindings in `m`, regardless of export status.
7+
The public names for a `Module` are available using [`names(m::Module)`](@ref), which will return
8+
an array of [`Symbol`](@ref) elements representing the public bindings. `names(m::Module, all = true)`
9+
returns symbols for all bindings in `m`, regardless of public status.
1010

1111
## DataType fields
1212

doc/src/manual/faq.md

+22-5
Original file line numberDiff line numberDiff line change
@@ -22,11 +22,28 @@ On the other hand, language *interoperability* is extremely useful: we want to e
2222

2323
### How does Julia define its public API?
2424

25-
Julia `Base` and standard library functionality described in the
26-
[the documentation](https://docs.julialang.org/) that is not marked as unstable
27-
(e.g. experimental and internal) is covered by [SemVer](https://semver.org/).
28-
Functions, types, and constants are not part of the public API if they are not
29-
included in the documentation, _even if they have docstrings_.
25+
Julia's public [API](https://en.wikipedia.org/wiki/API) is the behavior described in
26+
documentation of public symbols from `Base` and the standard libraries. Functions,
27+
types, and constants are not part of the public API if they are not public, even if
28+
they have docstrings or are described in the documentation. Further, only the documented
29+
behavior of public symbols is part of the public API. Undocumented behavior of public
30+
symbols is internal.
31+
32+
Public symbols are those marked with either `public foo` or `export foo`.
33+
34+
In other words:
35+
36+
- Documented behavior of public symbols is part of the public API.
37+
- Undocumented behavior of public symbols is not part of the public API.
38+
- Documented behavior of private symbols is not part of the public API.
39+
- Undocumented behavior of private symbols is not part of the public API.
40+
41+
You can get a complete list of the public symbols from a module with `names(MyModule)`.
42+
43+
Package authors are encouraged to define their public API similarly.
44+
45+
Anything in Julia's Public API is covered by [SemVer](https://semver.org/) and therefore
46+
will not be removed or receive meaningful breaking changes before Julia 2.0.
3047

3148
### There is a useful undocumented function/type/constant. Can I use it?
3249

doc/src/manual/modules.md

+7-2
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@ Modules in Julia help organize code into coherent units. They are delimited synt
77
allows the same name to be used for different functions or global variables without conflict, as long as they are in separate modules.
88

99
2. Modules have facilities for detailed namespace management: each defines a set of names it
10-
`export`s, and can import names from other modules with `using` and `import` (we explain these below).
10+
`export`s and marks as `public`, and can import names from other modules with `using` and
11+
`import` (we explain these below).
1112

1213
3. Modules can be precompiled for faster loading, and may contain code for runtime initialization.
1314

@@ -16,7 +17,7 @@ Typically, in larger Julia packages you will see module code organized into file
1617
```julia
1718
module SomeModule
1819

19-
# export, using, import statements are usually here; we discuss these below
20+
# export, public, using, import statements are usually here; we discuss these below
2021

2122
include("file1.jl")
2223
include("file2.jl")
@@ -103,6 +104,10 @@ Also, some modules don't export names at all. This is usually done if they use c
103104
words, such as `derivative`, in their API, which could easily clash with the export lists of other
104105
modules. We will see how to manage name clashes below.
105106

107+
To mark a name as public without exporting it into the namespace of folks who call `using NiceStuff`,
108+
one can use `public` instead of `export`. This marks the public name(s) as part of the public API,
109+
but does not have any namespace implications.
110+
106111
### Standalone `using` and `import`
107112

108113
Possibly the most common way of loading a module is `using ModuleName`. This [loads](@ref

doc/src/manual/noteworthy-differences.md

+4-3
Original file line numberDiff line numberDiff line change
@@ -413,15 +413,16 @@ For users coming to Julia from R, these are some noteworthy differences:
413413
file are `include`d only once (No `#ifdef` confusion).
414414

415415
### Julia &hArr; C/C++: Module interface
416-
* C++ exposes interfaces using "public" `.h`/`.hpp` files whereas Julia `module`s `export`
417-
symbols that are intended for their users.
416+
* C++ exposes interfaces using "public" `.h`/`.hpp` files whereas Julia `module`s mark
417+
specific symbols that are intended for their users as `public`or `export`ed.
418418
* Often, Julia `module`s simply add functionality by generating new "methods" to existing
419419
functions (ex: `Base.push!`).
420420
* Developers of Julia packages therefore cannot rely on header files for interface
421421
documentation.
422422
* Interfaces for Julia packages are typically described using docstrings, README.md,
423423
static web pages, ...
424-
* Some developers choose not to `export` all symbols required to use their package/module.
424+
* Some developers choose not to `export` all symbols required to use their package/module,
425+
but should still mark unexported user facing symbols as `public`.
425426
* Users might be expected to access these components by qualifying functions/structs/...
426427
with the package/module name (ex: `MyModule.run_this_task(...)`).
427428

src/ast.c

+2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ JL_DLLEXPORT jl_sym_t *jl_top_sym;
2828
JL_DLLEXPORT jl_sym_t *jl_module_sym;
2929
JL_DLLEXPORT jl_sym_t *jl_slot_sym;
3030
JL_DLLEXPORT jl_sym_t *jl_export_sym;
31+
JL_DLLEXPORT jl_sym_t *jl_public_sym;
3132
JL_DLLEXPORT jl_sym_t *jl_import_sym;
3233
JL_DLLEXPORT jl_sym_t *jl_toplevel_sym;
3334
JL_DLLEXPORT jl_sym_t *jl_quote_sym;
@@ -304,6 +305,7 @@ void jl_init_common_symbols(void)
304305
jl_lambda_sym = jl_symbol("lambda");
305306
jl_module_sym = jl_symbol("module");
306307
jl_export_sym = jl_symbol("export");
308+
jl_public_sym = jl_symbol("public");
307309
jl_import_sym = jl_symbol("import");
308310
jl_using_sym = jl_symbol("using");
309311
jl_assign_sym = jl_symbol("=");

src/builtins.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -1939,7 +1939,7 @@ static void add_intrinsic(jl_module_t *inm, const char *name, enum intrinsic f)
19391939
jl_value_t *i = jl_permbox32(jl_intrinsic_type, 0, (int32_t)f);
19401940
jl_sym_t *sym = jl_symbol(name);
19411941
jl_set_const(inm, sym, i);
1942-
jl_module_export(inm, sym);
1942+
jl_module_public(inm, sym, 1);
19431943
}
19441944

19451945
void jl_init_intrinsic_properties(void) JL_GC_DISABLED

0 commit comments

Comments
 (0)