blob: fbb05cb22ae062480dc783c9662704ac7e67906c [file]
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
"""Tests for the read-only SystemTable base class."""
import types
import unittest
from pypaimon.common.identifier import Identifier
from pypaimon.schema.data_types import AtomicType, DataField, RowType
from pypaimon.table.system.system_table import SystemTable
_DUMMY_ROW_TYPE = RowType(False, [
DataField(0, "key", AtomicType("STRING", nullable=False)),
DataField(1, "value", AtomicType("STRING", nullable=True)),
])
class _DummySystemTable(SystemTable):
"""Concrete SystemTable used only by these unit tests.
Subclasses are normally per-system-table (snapshots, schemas, ...).
This stub exercises the abstract base contract independently of any
real metadata source.
"""
def system_table_name(self) -> str:
return "dummy"
def row_type(self) -> RowType:
return _DUMMY_ROW_TYPE
def _build_arrow_table(self): # pragma: no cover - not exercised in this test
import pyarrow as pa
return pa.table({"key": [], "value": []})
def _fake_base(database: str = "db", table: str = "t", branch=None):
"""Construct a minimal stand-in for FileStoreTable.
SystemTable only touches ``identifier``, ``file_io`` and ``table_path``
on its base, so a SimpleNamespace covers the surface without dragging
in catalog/schema bootstrap.
"""
identifier = Identifier.create(database, table, branch=branch)
return types.SimpleNamespace(
identifier=identifier,
file_io=object(),
table_path="/tmp/" + database + "/" + table,
)
class SystemTableTest(unittest.TestCase):
def test_identifier_encodes_system_table_suffix(self):
sys_table = _DummySystemTable(_fake_base())
self.assertTrue(sys_table.identifier.is_system_table())
self.assertEqual("dummy", sys_table.identifier.get_system_table_name())
self.assertEqual("t", sys_table.identifier.get_table_name())
self.assertEqual("db", sys_table.identifier.get_database_name())
self.assertEqual("t$dummy", sys_table.identifier.get_object_name())
def test_identifier_preserves_branch_segment(self):
sys_table = _DummySystemTable(_fake_base(branch="dev"))
self.assertEqual("dev", sys_table.identifier.get_branch_name())
self.assertEqual("dummy", sys_table.identifier.get_system_table_name())
self.assertEqual("t$branch_dev$dummy", sys_table.identifier.get_object_name())
def test_base_table_handles_are_exposed(self):
base = _fake_base()
sys_table = _DummySystemTable(base)
self.assertIs(base, sys_table.base_table)
self.assertIs(base.file_io, sys_table.file_io)
self.assertEqual(base.table_path, sys_table.table_path)
def test_row_type_and_primary_keys_defaults(self):
sys_table = _DummySystemTable(_fake_base())
self.assertIs(_DUMMY_ROW_TYPE, sys_table.row_type())
self.assertEqual([], sys_table.primary_keys())
def test_write_and_search_builders_are_read_only(self):
sys_table = _DummySystemTable(_fake_base())
for method_name in (
"new_stream_read_builder",
"new_batch_write_builder",
"new_stream_write_builder",
"new_full_text_search_builder",
"new_vector_search_builder",
):
with self.assertRaises(NotImplementedError) as ctx:
getattr(sys_table, method_name)()
self.assertIn("read-only", str(ctx.exception).lower(),
"method {}: {}".format(method_name, ctx.exception))
if __name__ == "__main__":
unittest.main()