pyspiral 0.5.0__cp310-abi3-macosx_11_0_arm64.whl → 0.6.1__cp310-abi3-macosx_11_0_arm64.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (86) hide show
  1. {pyspiral-0.5.0.dist-info → pyspiral-0.6.1.dist-info}/METADATA +7 -3
  2. pyspiral-0.6.1.dist-info/RECORD +99 -0
  3. {pyspiral-0.5.0.dist-info → pyspiral-0.6.1.dist-info}/WHEEL +1 -1
  4. spiral/__init__.py +11 -3
  5. spiral/_lib.abi3.so +0 -0
  6. spiral/adbc.py +6 -6
  7. spiral/api/__init__.py +8 -2
  8. spiral/api/client.py +1 -1
  9. spiral/api/key_space_indexes.py +23 -0
  10. spiral/api/projects.py +15 -0
  11. spiral/api/text_indexes.py +1 -1
  12. spiral/cli/__init__.py +15 -6
  13. spiral/cli/admin.py +2 -4
  14. spiral/cli/app.py +4 -2
  15. spiral/cli/fs.py +5 -6
  16. spiral/cli/iceberg.py +97 -0
  17. spiral/cli/key_spaces.py +89 -0
  18. spiral/cli/login.py +6 -7
  19. spiral/cli/orgs.py +7 -8
  20. spiral/cli/printer.py +3 -3
  21. spiral/cli/projects.py +5 -6
  22. spiral/cli/tables.py +131 -0
  23. spiral/cli/telemetry.py +3 -4
  24. spiral/cli/text.py +115 -0
  25. spiral/cli/types.py +3 -4
  26. spiral/cli/workloads.py +7 -8
  27. spiral/client.py +111 -8
  28. spiral/core/authn/__init__.pyi +27 -0
  29. spiral/core/client/__init__.pyi +152 -63
  30. spiral/core/table/__init__.pyi +17 -27
  31. spiral/core/table/metastore/__init__.pyi +0 -4
  32. spiral/core/table/spec/__init__.pyi +0 -2
  33. spiral/{tables/dataset.py → dataset.py} +13 -7
  34. spiral/{tables/debug → debug}/manifests.py +15 -6
  35. spiral/{tables/debug → debug}/scan.py +3 -3
  36. spiral/expressions/base.py +3 -3
  37. spiral/expressions/udf.py +1 -1
  38. spiral/{iceberg/client.py → iceberg.py} +1 -3
  39. spiral/key_space_index.py +44 -0
  40. spiral/project.py +171 -18
  41. spiral/protogen/_/arrow/flight/protocol/sql/__init__.py +1668 -1110
  42. spiral/protogen/_/google/protobuf/__init__.py +2190 -0
  43. spiral/protogen/_/message_pool.py +3 -0
  44. spiral/protogen/_/py.typed +0 -0
  45. spiral/protogen/_/scandal/__init__.py +138 -126
  46. spiral/protogen/_/spfs/__init__.py +72 -0
  47. spiral/protogen/_/spql/__init__.py +61 -0
  48. spiral/protogen/_/substrait/__init__.py +5256 -2459
  49. spiral/protogen/_/substrait/extensions/__init__.py +103 -49
  50. spiral/{tables/scan.py → scan.py} +38 -44
  51. spiral/settings.py +14 -3
  52. spiral/snapshot.py +55 -0
  53. spiral/streaming_/__init__.py +3 -0
  54. spiral/streaming_/reader.py +131 -0
  55. spiral/streaming_/stream.py +146 -0
  56. spiral/substrait_.py +9 -9
  57. spiral/table.py +259 -0
  58. spiral/text_index.py +17 -0
  59. spiral/{tables/transaction.py → transaction.py} +11 -15
  60. pyspiral-0.5.0.dist-info/RECORD +0 -103
  61. spiral/cli/iceberg/__init__.py +0 -7
  62. spiral/cli/iceberg/namespaces.py +0 -47
  63. spiral/cli/iceberg/tables.py +0 -60
  64. spiral/cli/indexes/__init__.py +0 -40
  65. spiral/cli/indexes/args.py +0 -39
  66. spiral/cli/indexes/workers.py +0 -59
  67. spiral/cli/tables/__init__.py +0 -88
  68. spiral/cli/tables/args.py +0 -42
  69. spiral/core/index/__init__.pyi +0 -7
  70. spiral/iceberg/__init__.py +0 -3
  71. spiral/indexes/__init__.py +0 -5
  72. spiral/indexes/client.py +0 -137
  73. spiral/indexes/index.py +0 -28
  74. spiral/indexes/scan.py +0 -22
  75. spiral/protogen/_/spiral/table/__init__.py +0 -22
  76. spiral/protogen/substrait/__init__.py +0 -3399
  77. spiral/protogen/substrait/extensions/__init__.py +0 -115
  78. spiral/tables/__init__.py +0 -12
  79. spiral/tables/client.py +0 -133
  80. spiral/tables/maintenance.py +0 -12
  81. spiral/tables/snapshot.py +0 -78
  82. spiral/tables/table.py +0 -145
  83. {pyspiral-0.5.0.dist-info → pyspiral-0.6.1.dist-info}/entry_points.txt +0 -0
  84. /spiral/{protogen/_/spiral → debug}/__init__.py +0 -0
  85. /spiral/{tables/debug → debug}/metrics.py +0 -0
  86. /spiral/{tables/debug → protogen/_/google}/__init__.py +0 -0
@@ -0,0 +1,27 @@
1
+ from spiral.api.types import OrgId
2
+
3
+ class Token:
4
+ def __init__(self, value: str): ...
5
+ def expose_secret(self) -> str: ...
6
+
7
+ class Authn:
8
+ @staticmethod
9
+ def from_token(token: Token) -> Authn: ...
10
+ @staticmethod
11
+ def from_fallback() -> Authn: ...
12
+ @staticmethod
13
+ def from_device() -> Authn: ...
14
+ def token(self) -> Token | None: ...
15
+
16
+ class DeviceCodeAuth:
17
+ @staticmethod
18
+ def default() -> DeviceCodeAuth:
19
+ """Return the static device code instance."""
20
+ ...
21
+ def authenticate(self, force: bool = False, org_id: OrgId | None = None) -> Token:
22
+ """Authenticate using device code flow."""
23
+ ...
24
+
25
+ def logout(self) -> None:
26
+ """Logout from the device authentication session."""
27
+ ...
@@ -1,34 +1,11 @@
1
- from spiral.api.types import DatasetName, IndexName, OrgId, ProjectId, RootUri, TableName
2
- from spiral.core.index import SearchScan, TextIndex
3
- from spiral.core.table import Table, TableMaintenance, TableScan, TableSnapshot, TableTransaction
4
- from spiral.core.table.spec import Schema
5
- from spiral.expressions import Expr
6
-
7
- class Token:
8
- def __init__(self, value: str): ...
9
- def expose_secret(self) -> str: ...
10
-
11
- class Authn:
12
- @staticmethod
13
- def from_token(token: Token) -> Authn: ...
14
- @staticmethod
15
- def from_fallback() -> Authn: ...
16
- @staticmethod
17
- def from_device() -> Authn: ...
18
- def token(self) -> Token | None: ...
19
-
20
- class DeviceCodeAuth:
21
- @staticmethod
22
- def default() -> DeviceCodeAuth:
23
- """Return the static device code instance."""
24
- ...
25
- def authenticate(self, force: bool = False, org_id: OrgId | None = None) -> Token:
26
- """Authenticate using device code flow."""
27
- ...
1
+ from typing import Any, Literal
28
2
 
29
- def logout(self) -> None:
30
- """Logout from the device authentication session."""
31
- ...
3
+ import pyarrow as pa
4
+ from spiral.api.types import DatasetName, IndexName, ProjectId, RootUri, TableName
5
+ from spiral.core.authn import Authn
6
+ from spiral.core.table import ColumnGroupState, KeyRange, KeySpaceState, Scan, Snapshot, Table, Transaction
7
+ from spiral.core.table.spec import ColumnGroup, Schema
8
+ from spiral.expressions import Expr
32
9
 
33
10
  class Spiral:
34
11
  """A client for Spiral database"""
@@ -43,6 +20,39 @@ class Spiral:
43
20
  def authn(self) -> Authn:
44
21
  """Get the current authentication context."""
45
22
  ...
23
+
24
+ def scan(
25
+ self,
26
+ projection: Expr,
27
+ filter: Expr | None = None,
28
+ asof: int | None = None,
29
+ exclude_keys: bool = False,
30
+ ) -> Scan:
31
+ """Construct a table scan."""
32
+ ...
33
+
34
+ def transaction(self, table: Table, format: str | None = None) -> Transaction:
35
+ """Being a table transaction."""
36
+ ...
37
+
38
+ def search(
39
+ self,
40
+ top_k: int,
41
+ rank_by: Expr,
42
+ *,
43
+ filters: Expr | None = None,
44
+ freshness_window_s: int | None = None,
45
+ ) -> pa.RecordBatchReader:
46
+ """Search an index.
47
+
48
+ Searching an index returns a stream of record batches that match table's key schema + float score column.
49
+ """
50
+ ...
51
+
52
+ def table(self, table_id: str) -> Table:
53
+ """Get a table."""
54
+ ...
55
+
46
56
  def create_table(
47
57
  self,
48
58
  project_id: ProjectId,
@@ -56,63 +66,142 @@ class Spiral:
56
66
  """Create a new table in the specified project."""
57
67
  ...
58
68
 
59
- def get_table(self, table_id: str) -> Table:
60
- """Get and open table."""
61
-
62
- def open_table(self, table_id: str, key_schema: Schema, root_uri: RootUri) -> Table:
63
- """Open a table. This does not make any network calls."""
69
+ def text_index(self, index_id: str) -> TextIndex:
70
+ """Get a text index."""
64
71
  ...
65
72
 
66
- def open_table_scan(
73
+ def create_text_index(
67
74
  self,
75
+ project_id: ProjectId,
76
+ name: IndexName,
68
77
  projection: Expr,
69
78
  filter: Expr | None = None,
70
- asof: int | None = None,
71
- exclude_keys: bool = False,
72
- ) -> TableScan:
73
- """Construct a table scan."""
79
+ *,
80
+ root_uri: RootUri | None = None,
81
+ exist_ok: bool = False,
82
+ ) -> TextIndex:
83
+ """Create a new index in the specified project."""
74
84
  ...
75
85
 
76
- def open_transaction(self, table: Table, format: str | None = None) -> TableTransaction:
77
- """Being transaction."""
86
+ def key_space_index(self, index_id: str) -> KeySpaceIndex:
87
+ """Get a key space index."""
78
88
  ...
79
89
 
80
- def open_maintenance(self, table: Table, format: str | None = None) -> TableMaintenance:
81
- """Access maintenance operations for a table."""
82
- ...
83
- def create_text_index(
90
+ def create_key_space_index(
84
91
  self,
85
92
  project_id: ProjectId,
86
93
  name: IndexName,
94
+ granularity: int,
87
95
  projection: Expr,
88
96
  filter: Expr | None = None,
89
97
  *,
90
98
  root_uri: RootUri | None = None,
91
99
  exist_ok: bool = False,
92
- ) -> TextIndex:
93
- """Create a new index in the specified project."""
100
+ ) -> KeySpaceIndex:
101
+ """Create a new key space index in the specified project."""
94
102
  ...
95
103
 
96
- def get_text_index(self, index_id: str) -> TextIndex:
97
- """Get a text-based index."""
104
+ def _ops(self, *, format: str | None = None) -> Operations:
105
+ """Access maintenance operations.
106
+
107
+ IMPORTANT: This API is internal and is currently exposed for development & testing.
108
+ Maintenance operations are run by SpiralDB.
109
+ """
98
110
  ...
99
111
 
100
- def open_search_scan(
112
+ class TextIndex:
113
+ id: str
114
+
115
+ class KeySpaceIndex:
116
+ id: str
117
+ table_id: str
118
+ granularity: int
119
+ projection: Expr
120
+ filter: Expr
121
+ asof: int
122
+
123
+ class Shard:
124
+ key_range: KeyRange
125
+ cardinality: int
126
+
127
+ def __init__(self, key_range: KeyRange, cardinality: int): ...
128
+
129
+ class ShuffleStrategy:
130
+ # Results are buffered in a pool of `shuffle_buffer_size` rows and shuffled again.
131
+ shuffle_buffer_size: int
132
+
133
+ # All randomness is derived from this seed. If None, a random seed is generated from the OS.
134
+ seed: int | None
135
+
136
+ # Externally provided shards to shuffle before reading rows.
137
+ shards: list[Shard] | None
138
+
139
+ def __init__(
140
+ self,
141
+ seed: int | None = None,
142
+ shuffle_buffer_size: int | None = None,
143
+ shards: list[Shard] | None = None,
144
+ ): ...
145
+
146
+ class Operations:
147
+ def flush_wal(self, table: Table) -> None:
148
+ """
149
+ Flush the write-ahead log of the table.
150
+ """
151
+ ...
152
+ def compact_key_space(
101
153
  self,
102
- rank_by: Expr,
103
- top_k: int,
104
- # NOTE(marko): Required for now.
105
- freshness_window_s: int,
106
154
  *,
107
- filter: Expr | None = None,
108
- ) -> SearchScan:
109
- """Query an index."""
155
+ table: Table,
156
+ mode: Literal["plan", "read", "write"] | None = None,
157
+ partition_bytes_min: int | None = None,
158
+ ):
159
+ """
160
+ Compact the key space of the table.
161
+ """
110
162
  ...
111
-
112
- def _sync_snapshot(self, index_id: str, snapshot: TableSnapshot) -> None:
163
+ def compact_column_group(
164
+ self,
165
+ table: Table,
166
+ column_group: ColumnGroup,
167
+ *,
168
+ mode: Literal["plan", "read", "write"] | None = None,
169
+ partition_bytes_min: int | None = None,
170
+ ):
113
171
  """
114
- Synchronize an index with a table snapshot.
115
-
116
- IMPORTANT: This is only exposed for testing purposes and should not be used.
172
+ Compact a column group in the table.
173
+ """
174
+ ...
175
+ def update_text_index(self, index: TextIndex, snapshot: Snapshot) -> None:
176
+ """
177
+ Index table changes up to the given snapshot.
178
+ """
179
+ ...
180
+ def update_key_space_index(self, index: KeySpaceIndex, snapshot: Snapshot) -> None:
181
+ """
182
+ Index table changes up to the given snapshot.
183
+ """
184
+ ...
185
+ def key_space_state(self, snapshot: Snapshot) -> KeySpaceState:
186
+ """
187
+ The key space state for the table.
188
+ """
189
+ ...
190
+ def column_group_state(
191
+ self, snapshot: Snapshot, key_space_state: KeySpaceState, column_group: ColumnGroup
192
+ ) -> ColumnGroupState:
193
+ """
194
+ The state the column group of the table.
195
+ """
196
+ ...
197
+ def column_groups_states(self, snapshot: Snapshot, key_space_state: KeySpaceState) -> list[ColumnGroupState]:
198
+ """
199
+ The state of each column group of the table.
200
+ """
201
+ ...
202
+ def compute_shards(self, index: KeySpaceIndex) -> list[Shard]:
203
+ """
204
+ Compute the scan shards from a key space index.
117
205
  """
118
206
  ...
207
+ def metrics(self) -> dict[str, Any]: ...
@@ -1,6 +1,7 @@
1
- from typing import Any, Literal
1
+ from typing import Any
2
2
 
3
3
  import pyarrow as pa
4
+ from spiral.core.client import ShuffleStrategy
4
5
  from spiral.expressions import Expr
5
6
 
6
7
  from .manifests import FragmentManifest
@@ -42,22 +43,24 @@ class Table:
42
43
 
43
44
  def get_wal(self, *, asof: int | None) -> WriteAheadLog: ...
44
45
  def get_schema(self, *, asof: int | None) -> Schema: ...
45
- def get_snapshot(self, *, asof: int | None) -> TableSnapshot: ...
46
+ def get_snapshot(self, *, asof: int | None) -> Snapshot: ...
46
47
 
47
- class TableSnapshot:
48
+ class Snapshot:
48
49
  """A snapshot of a table at a specific point in time."""
49
50
 
50
51
  asof: int
51
52
  table: Table
52
53
  wal: WriteAheadLog
53
54
 
54
- class TableScan:
55
+ class Scan:
55
56
  def key_schema(self) -> Schema: ...
56
57
  def schema(self) -> Schema: ...
57
58
  def is_empty(self) -> bool: ...
58
- def split(self) -> list[KeyRange]: ...
59
+ def splits(self) -> list[KeyRange]: ...
59
60
  def table_ids(self) -> list[str]: ...
60
61
  def column_groups(self) -> list[ColumnGroup]: ...
62
+ def column_group_state(self, column_group: ColumnGroup) -> ColumnGroupState: ...
63
+ def key_space_state(self, table_id: str) -> KeySpaceState: ...
61
64
  def to_record_batches(
62
65
  self,
63
66
  key_table: pa.Table | pa.RecordBatch | None = None,
@@ -65,13 +68,16 @@ class TableScan:
65
68
  ) -> pa.RecordBatchReader: ...
66
69
  def to_shuffled_record_batches(
67
70
  self,
71
+ strategy: ShuffleStrategy | None = None,
68
72
  batch_readahead: int | None = None,
69
- shuffle_buffer_size: int | None = None,
70
- shuffle_pool_num_rows: int | None = None,
71
73
  ) -> pa.RecordBatchReader: ...
72
- def column_group_state(self, column_group: ColumnGroup) -> ColumnGroupState: ...
73
- def key_space_state(self, table_id: str) -> KeySpaceState: ...
74
74
  def metrics(self) -> dict[str, Any]: ...
75
+ def _prepare_shard(
76
+ self,
77
+ output_path: str,
78
+ key_range: KeyRange,
79
+ expected_cardinality: int | None = None,
80
+ ) -> None: ...
75
81
 
76
82
  class KeySpaceState:
77
83
  manifest: FragmentManifest
@@ -80,30 +86,14 @@ class KeySpaceState:
80
86
 
81
87
  class ColumnGroupState:
82
88
  manifest: FragmentManifest
89
+ column_group: ColumnGroup
83
90
 
84
91
  def schema(self) -> Schema: ...
85
92
 
86
- class TableTransaction:
93
+ class Transaction:
87
94
  status: str
88
95
 
89
96
  def write(self, expr: Expr, *, partition_size_bytes: int | None = None): ...
90
97
  def commit(self): ...
91
98
  def abort(self): ...
92
99
  def metrics(self) -> dict[str, Any]: ...
93
-
94
- class TableMaintenance:
95
- def flush_wal(self): ...
96
- def compact_key_space(
97
- self,
98
- *,
99
- mode: Literal["plan", "read", "write"] | None = None,
100
- partition_bytes_min: int | None = None,
101
- ): ...
102
- def compact_column_group(
103
- self,
104
- column_group: ColumnGroup,
105
- *,
106
- mode: Literal["plan", "read", "write"] | None = None,
107
- partition_bytes_min: int | None = None,
108
- ): ...
109
- def metrics(self) -> dict[str, Any]: ...
@@ -56,7 +56,3 @@ class PyMetastore:
56
56
  authn: Authn,
57
57
  ) -> PyMetastore:
58
58
  """Construct a PyMetastore backed by an HTTP metastore service."""
59
-
60
- @staticmethod
61
- def test(table_id: str, root_uri: str, key_schema: Schema) -> PyMetastore:
62
- """Construct a PyMetastore backed by an in-memory mock metastore service."""
@@ -101,8 +101,6 @@ class FragmentLevel:
101
101
  ...
102
102
 
103
103
  class Key:
104
- key: bytes
105
-
106
104
  def __init__(self, key: bytes): ...
107
105
  def __bytes__(self): ...
108
106
  def step(self) -> Key:
@@ -4,14 +4,15 @@ import pyarrow as pa
4
4
  import pyarrow.compute as pc
5
5
  import pyarrow.dataset as ds
6
6
 
7
- from spiral.tables import Scan, Snapshot
7
+ from spiral.scan import Scan
8
+ from spiral.snapshot import Snapshot
8
9
 
9
10
 
10
- class TableDataset(ds.Dataset):
11
+ class Dataset(ds.Dataset):
11
12
  def __init__(self, snapshot: Snapshot):
12
13
  self._snapshot = snapshot
13
14
  self._table = snapshot.table
14
- self._schema: pa.Schema = self._snapshot._snapshot.table.get_schema(asof=self._snapshot.asof).to_arrow()
15
+ self._schema: pa.Schema = self._snapshot.schema().to_arrow()
15
16
 
16
17
  # We don't actually initialize a Dataset, we just implement enough of the API
17
18
  # to fool both DuckDB and Polars.
@@ -43,7 +44,7 @@ class TableDataset(ds.Dataset):
43
44
  memory_pool,
44
45
  ).count_rows()
45
46
 
46
- def filter(self, expression: pc.Expression) -> "TableDataset":
47
+ def filter(self, expression: pc.Expression) -> "Dataset":
47
48
  raise NotImplementedError("filter not implemented")
48
49
 
49
50
  def get_fragments(self, filter: pc.Expression | None = None):
@@ -89,7 +90,7 @@ class TableDataset(ds.Dataset):
89
90
  def join_asof(self, right_dataset, on, by, tolerance, right_on=None, right_by=None):
90
91
  raise NotImplementedError("join_asof not implemented")
91
92
 
92
- def replace_schema(self, schema: pa.Schema) -> "TableDataset":
93
+ def replace_schema(self, schema: pa.Schema) -> "Dataset":
93
94
  raise NotImplementedError("replace_schema not implemented")
94
95
 
95
96
  def scanner(
@@ -112,13 +113,18 @@ class TableDataset(ds.Dataset):
112
113
  )
113
114
 
114
115
  scan = (
115
- self._snapshot.scan(
116
+ self._table.spiral.scan(
116
117
  {c: self._table[c] for c in columns},
117
118
  where=filter,
119
+ asof=self._snapshot.asof,
118
120
  exclude_keys=True,
119
121
  )
120
122
  if columns
121
- else self._snapshot.scan(where=filter)
123
+ else self._table.spiral.scan(
124
+ self._table,
125
+ where=filter,
126
+ asof=self._snapshot.asof,
127
+ )
122
128
  )
123
129
  self._last_scan = scan
124
130
 
@@ -1,23 +1,32 @@
1
1
  from spiral import datetime_
2
- from spiral.core.table import TableScan
2
+ from spiral.core.table import Scan
3
3
  from spiral.core.table.manifests import FragmentManifest
4
- from spiral.tables.debug.metrics import _format_bytes
4
+ from spiral.core.table.spec import ColumnGroup
5
+ from spiral.debug.metrics import _format_bytes
5
6
 
6
7
 
7
- def display_manifests(scan: TableScan):
8
+ def display_scan_manifests(scan: Scan):
8
9
  """Display all manifests in a scan."""
9
10
  if len(scan.table_ids()) != 1:
10
11
  raise NotImplementedError("Multiple table scans are not supported.")
11
12
  table_id = scan.table_ids()[0]
13
+ key_space_manifest = scan.key_space_state(table_id).manifest
14
+ column_group_manifests = [
15
+ (column_group, scan.column_group_state(column_group).manifest) for column_group in scan.column_groups()
16
+ ]
12
17
 
13
- key_space_manifest: FragmentManifest = scan.key_space_state(table_id).manifest
18
+ display_manifests(key_space_manifest, column_group_manifests)
19
+
20
+
21
+ def display_manifests(
22
+ key_space_manifest: FragmentManifest, column_group_manifests: list[tuple[ColumnGroup, FragmentManifest]]
23
+ ):
14
24
  _table_of_fragments(
15
25
  key_space_manifest,
16
26
  title="Key Space manifest",
17
27
  )
18
28
 
19
- for column_group in scan.column_groups():
20
- column_group_manifest: FragmentManifest = scan.column_group_state(column_group).manifest
29
+ for column_group, column_group_manifest in column_group_manifests:
21
30
  _table_of_fragments(
22
31
  column_group_manifest,
23
32
  title=f"Column Group manifest for {str(column_group)}",
@@ -1,12 +1,12 @@
1
1
  from datetime import datetime
2
2
 
3
- from spiral.core.table import TableScan
3
+ from spiral.core.table import Scan
4
4
  from spiral.core.table.manifests import FragmentFile, FragmentManifest
5
5
  from spiral.core.table.spec import Key
6
6
  from spiral.types_ import Timestamp
7
7
 
8
8
 
9
- def show_scan(scan: TableScan):
9
+ def show_scan(scan: Scan):
10
10
  """Displays a scan in a way that is useful for debugging."""
11
11
  table_ids = scan.table_ids()
12
12
  if len(table_ids) > 1:
@@ -14,7 +14,7 @@ def show_scan(scan: TableScan):
14
14
  table_id = table_ids[0]
15
15
  column_groups = scan.column_groups()
16
16
 
17
- splits = scan.split()
17
+ splits = scan.splits()
18
18
  key_space_state = scan.key_space_state(table_id)
19
19
 
20
20
  # Collect all key bounds from all manifests. This makes sure all visualizations are aligned.
@@ -132,7 +132,7 @@ class Expr:
132
132
  packed = packed.select(exclude=exclude)
133
133
  return packed
134
134
 
135
- if not paths:
135
+ if not paths and not exclude:
136
136
  return self
137
137
 
138
138
  return se.select(self, names=list(paths), exclude=exclude)
@@ -145,5 +145,5 @@ class Expr:
145
145
  return Expr(_lib.expr.binary(op, self.__expr__, rhs.__expr__))
146
146
 
147
147
 
148
- ScalarLike: TypeAlias = bool | int | float | str | list | datetime.datetime | None
149
- ExprLike: TypeAlias = Expr | dict | ScalarLike
148
+ ScalarLike: TypeAlias = bool | int | float | str | list["ScalarLike"] | datetime.datetime | None
149
+ ExprLike: TypeAlias = Expr | dict["ExprLike", "ExprLike"] | ScalarLike
spiral/expressions/udf.py CHANGED
@@ -38,7 +38,7 @@ class RefUDF(BaseUDF):
38
38
  super().__init__(_lib.expr.udf.create(name, return_type=self.return_type, invoke=self.invoke, scope="ref"))
39
39
 
40
40
  @abc.abstractmethod
41
- def invoke(self, fp: _lib.FileObject, *input_args: pa.Array) -> pa.Array:
41
+ def invoke(self, fp, *input_args: pa.Array) -> pa.Array:
42
42
  """Invoke the UDF with the given arguments.
43
43
 
44
44
  NOTE: The first argument is always the ref cell. All array input args will be sliced to the appropriate row.
@@ -13,10 +13,8 @@ class Iceberg:
13
13
  and ACID transactions to your warehouse.
14
14
  """
15
15
 
16
- def __init__(self, spiral: "Spiral", *, project_id: str | None = None):
16
+ def __init__(self, spiral: "Spiral"):
17
17
  self._spiral = spiral
18
- self._project_id = project_id
19
-
20
18
  self._api = self._spiral.config.api
21
19
 
22
20
  def catalog(self) -> "Catalog":
@@ -0,0 +1,44 @@
1
+ from spiral.core.client import KeySpaceIndex as CoreKeySpaceIndex
2
+ from spiral.expressions import Expr
3
+ from spiral.types_ import Timestamp
4
+
5
+
6
+ class KeySpaceIndex:
7
+ """
8
+ KeysIndex represents an optionally materialized key space, defined by a projection and a filter over a table.
9
+ It can be used to efficiently and precisely shard the table for parallel processing or distributed training.
10
+
11
+ An index is defined by:
12
+ - A granularity that defines the target size of key ranges in the index.
13
+ IMPORTANT: Actual key ranges may be smaller, but will not exceed twice the granularity.
14
+ - A projection expression that defines which columns are included in the resulting key space.
15
+ - An optional filter expression that defines which rows are included in the index.
16
+ """
17
+
18
+ def __init__(self, core: CoreKeySpaceIndex, *, name: str | None = None):
19
+ self.core = core
20
+ self._name = name
21
+
22
+ @property
23
+ def index_id(self) -> str:
24
+ return self.core.id
25
+
26
+ @property
27
+ def table_id(self) -> str:
28
+ return self.core.table_id
29
+
30
+ @property
31
+ def name(self) -> str:
32
+ return self._name or self.index_id
33
+
34
+ @property
35
+ def asof(self) -> Timestamp:
36
+ return self.core.asof
37
+
38
+ @property
39
+ def projection(self) -> Expr:
40
+ return Expr(self.core.projection)
41
+
42
+ @property
43
+ def filter(self) -> Expr | None:
44
+ return Expr(self.core.filter) if self.core.filter is not None else None