@@ -129,18 +129,31 @@ def iter_ranges(
129
129
sel : Union [Scalar , slice , Range , List [Scalar ], np .ndarray , "pyarrow.Array" ],
130
130
sparse : bool ,
131
131
nonempty_domain : Optional [Range ] = None ,
132
+ current_domain : Optional [Range ] = None ,
132
133
) -> Iterator [Range ]:
133
134
if isinstance (sel , slice ):
134
135
if sel .step is not None :
135
136
raise ValueError ("Stepped slice ranges are not supported" )
136
137
138
+ # we always want to be inside the current domain, if this is passed,
139
+ # but we also don't want to fall out of the nonempty domain
137
140
rstart = sel .start
138
- if rstart is None and nonempty_domain :
139
- rstart = nonempty_domain [0 ]
141
+ if rstart is None :
142
+ if current_domain and nonempty_domain :
143
+ rstart = max (current_domain [0 ], nonempty_domain [0 ])
144
+ elif current_domain :
145
+ rstart = current_domain [0 ]
146
+ elif nonempty_domain :
147
+ rstart = nonempty_domain [0 ]
140
148
141
149
rend = sel .stop
142
- if rend is None and nonempty_domain :
143
- rend = nonempty_domain [1 ]
150
+ if rend is None :
151
+ if current_domain and nonempty_domain :
152
+ rend = min (current_domain [1 ], nonempty_domain [1 ])
153
+ elif current_domain :
154
+ rend = current_domain [1 ]
155
+ elif nonempty_domain :
156
+ rend = nonempty_domain [1 ]
144
157
145
158
if sparse and sel .start is None and sel .stop is None :
146
159
# don't set nonempty_domain for full-domain slices w/ sparse
@@ -188,11 +201,13 @@ def iter_label_range(sel: Union[Scalar, slice, Range, List[Scalar]]):
188
201
yield scalar , scalar
189
202
190
203
191
- def dim_ranges_from_selection (selection , nonempty_domain , is_sparse ):
204
+ def dim_ranges_from_selection (selection , nonempty_domain , current_domain , is_sparse ):
192
205
# don't try to index nonempty_domain if None
193
206
selection = selection if isinstance (selection , list ) else [selection ]
194
207
return tuple (
195
- rng for sel in selection for rng in iter_ranges (sel , is_sparse , nonempty_domain )
208
+ rng
209
+ for sel in selection
210
+ for rng in iter_ranges (sel , is_sparse , nonempty_domain , current_domain )
196
211
)
197
212
198
213
@@ -206,27 +221,23 @@ def label_ranges_from_selection(selection):
206
221
def getitem_ranges (array : Array , idx : Any ) -> Sequence [Sequence [Range ]]:
207
222
ranges : List [Sequence [Range ]] = [()] * array .schema .domain .ndim
208
223
209
- # In the case that current domain is non-empty, we need to consider it
210
- if (
211
- hasattr (array .schema , "current_domain" )
212
- and not array .schema .current_domain .is_empty
213
- ):
214
- for i in range (array .schema .domain .ndim ):
215
- ranges [i ] = [
216
- (
217
- array .schema .current_domain .ndrectangle .range (i )[0 ],
218
- array .schema .current_domain .ndrectangle .range (i )[1 ],
219
- )
220
- ]
221
-
222
- return tuple (ranges )
223
-
224
224
ned = array .nonempty_domain ()
225
+ current_domain = (
226
+ array .schema .current_domain
227
+ if hasattr (array .schema , "current_domain" )
228
+ and not array .schema .current_domain .is_empty
229
+ else None
230
+ )
225
231
if ned is None :
226
232
ned = [None ] * array .schema .domain .ndim
227
233
is_sparse = array .schema .sparse
228
234
for i , dim_sel in enumerate ([idx ] if not isinstance (idx , tuple ) else idx ):
229
- ranges [i ] = dim_ranges_from_selection (dim_sel , ned [i ], is_sparse )
235
+ ranges [i ] = dim_ranges_from_selection (
236
+ dim_sel ,
237
+ ned [i ],
238
+ current_domain .ndrectangle .range (i ) if current_domain else None ,
239
+ is_sparse ,
240
+ )
230
241
return tuple (ranges )
231
242
232
243
@@ -236,6 +247,12 @@ def getitem_ranges_with_labels(
236
247
dim_ranges : List [Sequence [Range ]] = [()] * array .schema .domain .ndim
237
248
label_ranges : Dict [str , Sequence [Range ]] = {}
238
249
ned = array .nonempty_domain ()
250
+ current_domain = (
251
+ array .schema .current_domain
252
+ if hasattr (array .schema , "current_domain" )
253
+ and not array .schema .current_domain .is_empty
254
+ else None
255
+ )
239
256
if ned is None :
240
257
ned = [None ] * array .schema .domain .ndim
241
258
is_sparse = array .schema .sparse
@@ -244,7 +261,10 @@ def getitem_ranges_with_labels(
244
261
label_ranges [labels [dim_idx ]] = label_ranges_from_selection (dim_sel )
245
262
else :
246
263
dim_ranges [dim_idx ] = dim_ranges_from_selection (
247
- dim_sel , ned [dim_idx ], is_sparse
264
+ dim_sel ,
265
+ ned [dim_idx ],
266
+ current_domain .ndrectangle .range (dim_idx ) if current_domain else None ,
267
+ is_sparse ,
248
268
)
249
269
return dim_ranges , label_ranges
250
270
0 commit comments