Skip to content

Commit dbfdca7

Browse files
committed
Revert to an error for getindex
1 parent f105fe6 commit dbfdca7

16 files changed

+124
-438
lines changed

docs/src/manual/containers.md

+30-133
Original file line numberDiff line numberDiff line change
@@ -259,6 +259,22 @@ julia> DataFrames.DataFrame(table)
259259
4 │ 2 B (2, :B)
260260
```
261261

262+
### Keyword indexing
263+
264+
If all axes are named, you can use keyword indexing:
265+
266+
```jldoctest containers_dense
267+
julia> x[i = 2, j = :A]
268+
(2, :A)
269+
270+
julia> x[i = :, j = :B]
271+
1-dimensional DenseAxisArray{Tuple{Int64, Symbol},1,...} with index sets:
272+
Dimension 1, Base.OneTo(2)
273+
And data, a 2-element Vector{Tuple{Int64, Symbol}}:
274+
(1, :B)
275+
(2, :B)
276+
```
277+
262278
## SparseAxisArray
263279

264280
A [`Containers.SparseAxisArray`](@ref) is created when the index sets are
@@ -352,6 +368,20 @@ julia> DataFrames.DataFrame(table)
352368
4 │ 3 A (3, :A)
353369
```
354370

371+
### Keyword indexing
372+
373+
If all axes are named, you can use keyword indexing:
374+
375+
```jldoctest containers_sparse
376+
julia> x[i = 2, j = :A]
377+
(2, :A)
378+
379+
julia> x[i = :, j = :B]
380+
JuMP.Containers.SparseAxisArray{Tuple{Int64, Symbol}, 1, Tuple{Int64}} with 2 entries:
381+
[2] = (2, :B)
382+
[3] = (3, :B)
383+
```
384+
355385
## Forcing the container type
356386

357387
Pass `container = T` to use `T` as the container. For example:
@@ -485,136 +515,3 @@ JuMP.Containers.SparseAxisArray{Int64, 2, Tuple{Int64, Int64}} with 2 entries:
485515
[1, 2] = 3
486516
[1, 4] = 5
487517
```
488-
489-
## Keyword indexing
490-
491-
JuMP v1.10.0 added experimental support for keyword indexing of
492-
[`Containers.DenseAxisArray`](@ref) and [`Containers.SparseAxisArray`](@ref).
493-
Keyword indexing lets you uses the named indices of [`Containers.DenseAxisArray`](@ref)
494-
and [`Containers.SparseAxisArray`](@ref) as keyword arguments when indexing a
495-
container.
496-
497-
### The container macro
498-
499-
For containers constructed using the [`Containers.@container`](@ref) macro,
500-
opt-in to the keyword indexing syntax by passing
501-
`enable_keyword_indexing = true`:
502-
503-
```jldoctest
504-
julia> Containers.@container(
505-
x[i=2:4, j=1:3],
506-
i + j,
507-
enable_keyword_indexing = true,
508-
)
509-
2-dimensional DenseAxisArray{Int64,2,...} with index sets:
510-
Dimension 1, 2:4
511-
Dimension 2, Base.OneTo(3)
512-
And data, a 3×3 Matrix{Int64}:
513-
3 4 5
514-
4 5 6
515-
5 6 7
516-
517-
julia> x[i = 2, j = 1]
518-
3
519-
520-
julia> Containers.@container(
521-
y[i=2:4, j=1:3; i > j],
522-
i + j,
523-
enable_keyword_indexing = true,
524-
)
525-
JuMP.Containers.SparseAxisArray{Int64, 2, Tuple{Int64, Int64}} with 6 entries:
526-
[2, 1] = 3
527-
[3, 1] = 4
528-
[3, 2] = 5
529-
[4, 1] = 5
530-
[4, 2] = 6
531-
[4, 3] = 7
532-
533-
julia> y[i = 2:4, j = 1]
534-
JuMP.Containers.SparseAxisArray{Int64, 1, Tuple{Int64}} with 3 entries:
535-
[2] = 3
536-
[3] = 4
537-
[4] = 5
538-
```
539-
540-
### The JuMP macros
541-
542-
For containers constructed using the JuMP macros like [`@variable`](@ref),
543-
opt-in using [`enable_keyword_indexing`](@ref):
544-
545-
```jldoctest
546-
julia> using JuMP
547-
548-
julia> model = Model();
549-
550-
julia> @variable(model, x[i = 2:3])
551-
1-dimensional DenseAxisArray{VariableRef,1,...} with index sets:
552-
Dimension 1, 2:3
553-
And data, a 2-element Vector{VariableRef}:
554-
x[2]
555-
x[3]
556-
557-
julia> x[i = 2]
558-
ERROR: Keyword indexing is disabled. To enable, pass `enable_keyword_indexing = true` to the `Containers.@container` macro, or call `JuMP.enable_keyword_indexing(model, true)` before calling any JuMP macros like `@variable`.
559-
Stacktrace:
560-
[...]
561-
562-
julia> model = Model();
563-
564-
julia> enable_keyword_indexing(model, true)
565-
566-
julia> @variable(model, x[i = 2:3])
567-
1-dimensional DenseAxisArray{VariableRef,1,...} with index sets:
568-
Dimension 1, 2:3
569-
And data, a 2-element Vector{VariableRef}:
570-
x[2]
571-
x[3]
572-
573-
julia> x[i = 2]
574-
x[2]
575-
```
576-
577-
### Limitations
578-
579-
The keyword indexing syntax is currently opt-in because it does not work for
580-
`Array`:
581-
582-
```jldoctest
583-
julia> Containers.@container(
584-
x[i=1:4, j=1:3],
585-
i + j,
586-
enable_keyword_indexing = true,
587-
)
588-
ERROR: Keyword indexing is not supported with Array.
589-
Stacktrace:
590-
[...]
591-
```
592-
593-
Work-around this limitation by forcing the container type:
594-
595-
```jldoctest
596-
julia> model = Model();
597-
598-
julia> enable_keyword_indexing(model, true)
599-
600-
julia> @variable(model, x[i = 1:3])
601-
3-element Vector{VariableRef}:
602-
x[1]
603-
x[2]
604-
x[3]
605-
606-
julia> x[i = 2]
607-
ERROR: MethodError: no method matching getindex(::Vector{VariableRef}; i=2)
608-
[...]
609-
610-
julia> @variable(model, y[i = 1:3], container = DenseAxisArray)
611-
1-dimensional DenseAxisArray{VariableRef,1,...} with index sets:
612-
Dimension 1, Base.OneTo(3)
613-
And data, a 3-element Vector{VariableRef}:
614-
y[1]
615-
y[2]
616-
y[3]
617-
618-
julia> y[i = 2]
619-
y[2]
620-
```

docs/src/reference/models.md

-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,6 @@ object_dictionary
3333
unregister
3434
latex_formulation
3535
set_string_names_on_creation
36-
enable_keyword_indexing
3736
```
3837

3938
## Working with attributes

src/Containers/Containers.jl

-11
Original file line numberDiff line numberDiff line change
@@ -25,17 +25,6 @@ Base.IndexStyle(::IndexAnyCartesian, ::IndexAnyCartesian) = IndexAnyCartesian()
2525

2626
export DenseAxisArray, SparseAxisArray
2727

28-
_check_keyword_indexing_allowed(::Any) = nothing
29-
30-
function _check_keyword_indexing_allowed(::Nothing)
31-
return error(
32-
"Keyword indexing is disabled. To enable, pass " *
33-
"`enable_keyword_indexing = true` to the `Containers.@container` " *
34-
"macro, or call `JuMP.enable_keyword_indexing(model, true)` " *
35-
"before calling any JuMP macros like `@variable`.",
36-
)
37-
end
38-
3928
include("DenseAxisArray.jl")
4029
include("SparseAxisArray.jl")
4130

src/Containers/DenseAxisArray.jl

+3-8
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ struct DenseAxisArray{T,N,Ax,L<:NTuple{N,_AxisLookup}} <: AbstractArray{T,N}
2222
data::Array{T,N}
2323
axes::Ax
2424
lookup::L
25-
names::Union{Nothing,NTuple{N,Symbol}}
25+
names::NTuple{N,Symbol}
2626
end
2727

2828
function Base.Array{T,N}(x::DenseAxisArray) where {T,N}
@@ -208,6 +208,7 @@ function DenseAxisArray(
208208
) where {T,N}
209209
@assert length(axes) == N
210210
new_axes = _abstract_vector.(axes) # Force all axes to be AbstractVector!
211+
names = something(names, ntuple(n -> Symbol("#$n"), N))
211212
return DenseAxisArray(data, new_axes, build_lookup.(new_axes), names)
212213
end
213214

@@ -355,7 +356,6 @@ _is_range(::Any) = false
355356
_is_range(::Union{Vector{Int},Colon}) = true
356357

357358
function _kwargs_to_args(A::DenseAxisArray{T,N}; kwargs...) where {T,N}
358-
_check_keyword_indexing_allowed(A.names)
359359
return ntuple(N) do i
360360
kw = keys(kwargs)[i]
361361
if A.names[i] != kw
@@ -381,11 +381,7 @@ function Base.getindex(A::DenseAxisArray{T,N}, args...; kwargs...) where {T,N}
381381
return A.data[new_indices...]::T
382382
end
383383
new_axes = _getindex_recurse(A.axes, new_indices, _is_range)
384-
names = if A.names === nothing
385-
nothing
386-
else
387-
A.names[findall(_is_range, new_indices)]
388-
end
384+
names = A.names[findall(_is_range, new_indices)]
389385
return DenseAxisArray(A.data[new_indices...], new_axes...; names = names)
390386
end
391387

@@ -747,7 +743,6 @@ function _fixed_indices(view_axes::Tuple, axes::Tuple)
747743
end
748744

749745
function _kwargs_to_args(A::DenseAxisArrayView{T,N}; kwargs...) where {T,N}
750-
_check_keyword_indexing_allowed(A.data.names)
751746
non_default_indices = _fixed_indices(A.axes, A.data.axes)
752747
return ntuple(N) do i
753748
kw = keys(kwargs)[i]

src/Containers/SparseAxisArray.jl

+3-4
Original file line numberDiff line numberDiff line change
@@ -37,11 +37,11 @@ julia> array[:b, 3]
3737
"""
3838
struct SparseAxisArray{T,N,K<:NTuple{N,Any}} <: AbstractArray{T,N}
3939
data::Dict{K,T}
40-
names::Union{Nothing,NTuple{N,Symbol}}
40+
names::NTuple{N,Symbol}
4141
end
4242

4343
function SparseAxisArray(d::Dict{K,T}) where {T,N,K<:NTuple{N,Any}}
44-
return SparseAxisArray(d, nothing)
44+
return SparseAxisArray(d, ntuple(n -> Symbol("#$n"), N))
4545
end
4646

4747
Base.length(sa::SparseAxisArray) = length(sa.data)
@@ -99,7 +99,6 @@ function Base.setindex!(
9999
end
100100

101101
function _kwargs_to_args(d::SparseAxisArray{T,N}; kwargs...) where {T,N}
102-
_check_keyword_indexing_allowed(d.names)
103102
if length(kwargs) != N
104103
throw(BoundsError(d, kwargs))
105104
end
@@ -284,7 +283,7 @@ function Base.copy(
284283
# type information to call SparseAxisArray(dict). As a work-around, we
285284
# explicitly construct the type of the resulting SparseAxisArray.
286285
# For more, see JuMP issue #2867.
287-
return SparseAxisArray{Any,N,K}(dict, nothing)
286+
return SparseAxisArray{Any,N,K}(dict, ntuple(n -> Symbol("#$n"), N))
288287
end
289288
return SparseAxisArray(dict)
290289
end

src/Containers/container.jl

-26
Original file line numberDiff line numberDiff line change
@@ -71,36 +71,10 @@ function container(f::Function, indices, D, names)
7171
return container(f, indices, D)
7272
end
7373

74-
struct _ForceEnableKeywordIndexing{T}
75-
names::T
76-
end
77-
7874
function container(f::Function, indices, ::Type{AutoContainerType}, names)
7975
return container(f, indices, default_container(indices), names)
8076
end
8177

82-
function container(
83-
f::Function,
84-
indices,
85-
::Type{AutoContainerType},
86-
names::_ForceEnableKeywordIndexing,
87-
)
88-
return container(f, indices, default_container(indices), names)
89-
end
90-
91-
function container(f::Function, indices, D, names::_ForceEnableKeywordIndexing)
92-
return container(f, indices, D, names.names)
93-
end
94-
95-
function container(
96-
::Function,
97-
::Any,
98-
::Type{Array},
99-
::_ForceEnableKeywordIndexing,
100-
)
101-
return error("Keyword indexing is not supported with Array.")
102-
end
103-
10478
function container(f::Function, indices)
10579
return container(f, indices, default_container(indices))
10680
end

src/Containers/macro.jl

+6-29
Original file line numberDiff line numberDiff line change
@@ -239,8 +239,7 @@ end
239239
index_vars::Vector{Any},
240240
indices::Expr,
241241
code,
242-
requested_container::Union{Symbol,Expr};
243-
keyword_indexing::Union{Bool,Expr},
242+
requested_container::Union{Symbol,Expr},
244243
)
245244
246245
Used in macros to construct a call to [`container`](@ref). This should be used
@@ -289,8 +288,7 @@ function container_code(
289288
index_vars::Vector{Any},
290289
indices::Expr,
291290
code,
292-
requested_container::Union{Symbol,Expr};
293-
keyword_indexing::Union{Bool,Symbol,Expr} = false,
291+
requested_container::Union{Symbol,Expr},
294292
)
295293
if isempty(index_vars)
296294
return code
@@ -310,19 +308,9 @@ function container_code(
310308
else
311309
# This is a symbol or expression from outside JuMP, so we need to escape
312310
# it.
313-
keyword_indexing = missing
314311
esc(requested_container)
315312
end
316-
return quote
317-
names = if $keyword_indexing === missing
318-
$index_vars
319-
elseif $keyword_indexing === true
320-
$_ForceEnableKeywordIndexing($index_vars)
321-
else
322-
nothing
323-
end
324-
$container($f, $indices, $container_type, names)
325-
end
313+
return Expr(:call, container, f, indices, container_type, index_vars)
326314
end
327315

328316
"""
@@ -383,23 +371,12 @@ JuMP.Containers.SparseAxisArray{Int64,2,Tuple{Int64,Int64}} with 6 entries:
383371
```
384372
"""
385373
macro container(args...)
386-
args, kwargs, requested_container = _extract_kw_args(args)
374+
args, kw_args, requested_container = _extract_kw_args(args)
387375
@assert length(args) == 2
388-
enable_keyword_indexing = false
389-
for kw in kwargs
390-
@assert Meta.isexpr(kw, :(=), 2)
391-
@assert kw.args[1] == :enable_keyword_indexing
392-
enable_keyword_indexing = esc(kw.args[2])
393-
end
376+
@assert isempty(kw_args)
394377
var, value = args
395378
index_vars, indices = build_ref_sets(error, var)
396-
code = container_code(
397-
index_vars,
398-
indices,
399-
esc(value),
400-
requested_container;
401-
keyword_indexing = enable_keyword_indexing,
402-
)
379+
code = container_code(index_vars, indices, esc(value), requested_container)
403380
if isexpr(var, :vect) || isexpr(var, :vcat)
404381
return code
405382
else

0 commit comments

Comments
 (0)