Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add RangeIndex #10076

Open
wants to merge 24 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
dcc8f5b
add RangeIndex
benbovy Feb 25, 2025
04827db
Merge branch 'main' into range-index
benbovy Mar 20, 2025
adbd3d4
refactor RangeIndex
benbovy Mar 21, 2025
e7f6476
fix error raised during coords diff formatting
benbovy Mar 21, 2025
fb1b10b
assert_allclose: add support for Coordinates
benbovy Mar 21, 2025
76f58c0
add tests (wip)
benbovy Mar 21, 2025
53a02c8
assert invariants: skip check IndexVariable ...
benbovy Mar 14, 2025
9eaa530
more tests and fixes
benbovy Mar 21, 2025
e6709a1
no support for set_xindex (error msg)
benbovy Mar 21, 2025
e18725c
add public API documentation
benbovy Mar 21, 2025
cc3601f
fix doctests
benbovy Mar 21, 2025
b5c5207
add docstring examples
benbovy Mar 21, 2025
b48ed9d
[skip-ci] update whats new
benbovy Mar 21, 2025
2a27198
[skip-ci] doc: add RangeIndex factories to API (hidden)
benbovy Mar 21, 2025
f819846
add repr + start, stop, step properties
benbovy Mar 21, 2025
cd0a396
add support for rename vars and/or dims
benbovy Mar 21, 2025
4655934
step: attr -> property
benbovy Mar 24, 2025
ee11665
Merge branch 'main' into range-index
benbovy Mar 27, 2025
212a9ae
doc: list RangeIndex constructors in API ref
benbovy Mar 31, 2025
8ad8349
CoordinateTransformIndex.rename: deep copy transform
benbovy Mar 31, 2025
efe3e81
update arange and linspace signatures
benbovy Mar 31, 2025
38c8a6c
RangeIndex repr: show size, coord_name and dim props
benbovy Mar 31, 2025
24c6b55
fix doctests
benbovy Mar 31, 2025
911d76c
Merge branch 'main' into range-index
benbovy Mar 31, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions doc/api-hidden.rst
Original file line number Diff line number Diff line change
Expand Up @@ -530,6 +530,10 @@
Index.rename
Index.copy

indexes.RangeIndex.start
indexes.RangeIndex.stop
indexes.RangeIndex.step

backends.NetCDF4DataStore.close
backends.NetCDF4DataStore.encode
backends.NetCDF4DataStore.encode_attribute
Expand Down
3 changes: 3 additions & 0 deletions doc/api.rst
Original file line number Diff line number Diff line change
Expand Up @@ -1573,6 +1573,7 @@ Custom Indexes
:toctree: generated/

CFTimeIndex
indexes.RangeIndex

Creating custom indexes
-----------------------
Expand All @@ -1582,6 +1583,8 @@ Creating custom indexes
cftime_range
date_range
date_range_like
indexes.RangeIndex.arange
indexes.RangeIndex.linspace

Tutorial
========
Expand Down
4 changes: 3 additions & 1 deletion doc/whats-new.rst
Original file line number Diff line number Diff line change
Expand Up @@ -53,9 +53,11 @@ Andrecho, Deepak Cherian, Ian Hunt-Isaak, Karl Krauth, Mathias Hauser, Maximilia

New Features
~~~~~~~~~~~~

- Allow setting a ``fill_value`` for Zarr format 3 arrays. Specify ``fill_value`` in ``encoding`` as usual.
(:issue:`10064`). By `Deepak Cherian <https://github.com/dcherian>`_.
- Added :py:class:`indexes.RangeIndex` as an alternative, memory saving Xarray index representing
a 1-dimensional bounded interval with evenly spaced floating values (:issue:`8473`, :pull:`10076`).
By `Benoit Bovy <https://github.com/benbovy>`_.

Breaking changes
~~~~~~~~~~~~~~~~
Expand Down
6 changes: 4 additions & 2 deletions xarray/core/formatting.py
Original file line number Diff line number Diff line change
Expand Up @@ -312,7 +312,7 @@ def inline_variable_array_repr(var, max_width):
def summarize_variable(
name: Hashable,
var: Variable,
col_width: int,
col_width: int | None = None,
max_width: int | None = None,
is_index: bool = False,
):
Expand All @@ -327,7 +327,9 @@ def summarize_variable(
max_width = max_width_options

marker = "*" if is_index else " "
first_col = pretty_print(f" {marker} {name} ", col_width)
first_col = f" {marker} {name} "
if col_width is not None:
first_col = pretty_print(first_col, col_width)

if variable.dims:
dims_str = "({}) ".format(", ".join(map(str, variable.dims)))
Expand Down
17 changes: 15 additions & 2 deletions xarray/core/indexes.py
Original file line number Diff line number Diff line change
Expand Up @@ -1500,8 +1500,21 @@ def rename(
name_dict: Mapping[Any, Hashable],
dims_dict: Mapping[Any, Hashable],
) -> Self:
# TODO: maybe update self.transform coord_names, dim_size and dims attributes
return self
coord_names = self.transform.coord_names
dims = self.transform.dims
dim_size = self.transform.dim_size

if not set(coord_names) & set(name_dict) and not set(dims) & set(dims_dict):
return self

new_transform = copy.deepcopy(self.transform)
new_transform.coord_names = tuple(name_dict.get(n, n) for n in coord_names)
new_transform.dims = tuple(str(dims_dict.get(d, d)) for d in dims)
new_transform.dim_size = {
str(dims_dict.get(d, d)): v for d, v in dim_size.items()
}

return type(self)(new_transform)


def create_default_index_implicit(
Expand Down
12 changes: 6 additions & 6 deletions xarray/core/indexing.py
Original file line number Diff line number Diff line change
Expand Up @@ -240,16 +240,16 @@ def expanded_indexer(key, ndim):
return tuple(new_key)


def _normalize_slice(sl: slice, size: int) -> slice:
def normalize_slice(sl: slice, size: int) -> slice:
"""
Ensure that given slice only contains positive start and stop values
(stop can be -1 for full-size slices with negative steps, e.g. [-10::-1])

Examples
--------
>>> _normalize_slice(slice(0, 9), 10)
>>> normalize_slice(slice(0, 9), 10)
slice(0, 9, 1)
>>> _normalize_slice(slice(0, -1), 10)
>>> normalize_slice(slice(0, -1), 10)
slice(0, 9, 1)
"""
return slice(*sl.indices(size))
Expand All @@ -266,7 +266,7 @@ def _expand_slice(slice_: slice, size: int) -> np.ndarray[Any, np.dtype[np.integ
>>> _expand_slice(slice(0, -1), 10)
array([0, 1, 2, 3, 4, 5, 6, 7, 8])
"""
sl = _normalize_slice(slice_, size)
sl = normalize_slice(slice_, size)
return np.arange(sl.start, sl.stop, sl.step)


Expand All @@ -275,14 +275,14 @@ def slice_slice(old_slice: slice, applied_slice: slice, size: int) -> slice:
index it with another slice to return a new slice equivalent to applying
the slices sequentially
"""
old_slice = _normalize_slice(old_slice, size)
old_slice = normalize_slice(old_slice, size)

size_after_old_slice = len(range(old_slice.start, old_slice.stop, old_slice.step))
if size_after_old_slice == 0:
# nothing left after applying first slice
return slice(0)

applied_slice = _normalize_slice(applied_slice, size_after_old_slice)
applied_slice = normalize_slice(applied_slice, size_after_old_slice)

start = old_slice.start + applied_slice.start * old_slice.step
if start < 0:
Expand Down
3 changes: 2 additions & 1 deletion xarray/indexes/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,5 +8,6 @@
PandasIndex,
PandasMultiIndex,
)
from xarray.indexes.range_index import RangeIndex

__all__ = ["Index", "PandasIndex", "PandasMultiIndex"]
__all__ = ["Index", "PandasIndex", "PandasMultiIndex", "RangeIndex"]
Loading
Loading