Learn more  » Push, build, and install  RubyGems npm packages Python packages Maven artifacts PHP packages Go Modules Bower components Debian packages RPM packages NuGet packages

arrow-nightlies / nanoarrow   python

Repository URL to install this package:

/ tests / test_c_buffer_view.py

# 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.

import pytest

import nanoarrow as na


def test_buffer_view_bool_():
    bool_array_view = na.c_array([1, 0, 0, 1], na.bool_()).view()
    view = bool_array_view.buffer(1)

    assert view.element_size_bits == 1
    assert view.size_bytes == 1
    assert view.data_type_id == na.Type.BOOL.value
    assert view.data_type == "bool"
    assert view.format == "B"

    # Check item interface
    assert len(view) == 1
    assert view[0] == 0b1001
    assert list(view) == [0b1001]

    # Check against buffer protocol
    mv = memoryview(view)
    assert len(mv) == len(view)
    assert mv[0] == view[0]
    assert list(mv) == list(view)

    # Check element interface
    assert view.n_elements == 8
    assert list(view.elements()) == [True, False, False, True] + [False] * 4
    assert [view.element(i) for i in range(8)] == list(view.elements())

    # Check element slices
    assert list(view.elements(0, 4)) == [True, False, False, True]
    assert list(view.elements(1, 3)) == [False, False, True]

    msg = "do not describe a valid slice"
    with pytest.raises(IndexError, match=msg):
        view.elements(-1, None)
    with pytest.raises(IndexError, match=msg):
        view.elements(0, -1)
    with pytest.raises(IndexError, match=msg):
        view.elements(0, 9)

    # Check repr
    assert "10010000" in repr(view)


def test_buffer_view_bool_unpack():
    from array import array

    bool_array_view = na.c_array([1, 0, 0, 1], na.bool_()).view()
    view = bool_array_view.buffer(1)

    # Check unpacking
    unpacked_all = view.unpack_bits()
    assert len(unpacked_all) == view.n_elements
    assert unpacked_all.data_type == "uint8"
    assert unpacked_all.format == "?"
    assert list(unpacked_all) == [1, 0, 0, 1, 0, 0, 0, 0]

    unpacked_some = view.unpack_bits(1, 4)
    assert len(unpacked_some) == 4
    assert list(unpacked_some) == [0, 0, 1, 0]

    # Check with non-zero destination offset
    out = bytearray([255] * 10)
    assert view.unpack_bits_into(out, dest_offset=2) == 8
    assert list(out) == [255, 255, 1, 0, 0, 1, 0, 0, 0, 0]

    # Check error requesting out-of-bounds dest_offset
    with pytest.raises(IndexError, match="Can't unpack"):
        view.unpack_bits_into(out, dest_offset=-1)

    # Check errors from requesting out-of-bounds slices
    msg = "do not describe a valid slice"
    with pytest.raises(IndexError, match=msg):
        view.unpack_bits(-1, None)
    with pytest.raises(IndexError, match=msg):
        view.unpack_bits(0, -1)
    with pytest.raises(IndexError, match=msg):
        view.unpack_bits(0, 9)

    # Check errors from an output buffer of insufficient length
    out = bytearray()
    msg = "Can't unpack 8 elements into buffer of size 0"
    with pytest.raises(IndexError, match=msg):
        view.unpack_bits_into(out)

    # Check errors from an output buffer with the wrong data type
    out = array("i", [0, 0, 0, 0])
    msg = "Destination buffer must have itemsize == 1"
    with pytest.raises(ValueError, match=msg):
        view.unpack_bits_into(out)


def test_buffer_view_non_bool():
    array_view = na.c_array([1, 2, 3, 5], na.int32()).view()
    view = array_view.buffer(1)

    assert view.element_size_bits == 32
    assert view.size_bytes == 4 * 4
    assert view.data_type_id == na.Type.INT32.value
    assert view.data_type == "int32"
    assert view.format == "i"

    # Check item interface
    assert len(view) == 4
    assert list(view) == [1, 2, 3, 5]
    assert [view[i] for i in range(4)] == list(view)

    # Check against buffer protocol
    mv = memoryview(view)
    assert len(mv) == len(view)
    assert mv[0] == view[0]
    assert [mv[i] for i in range(4)] == [view[i] for i in range(4)]

    # Check element interface
    assert view.n_elements == len(view)
    assert list(view.elements()) == list(view)
    assert [view.element(i) for i in range(4)] == list(view.elements())

    # Check element slices
    assert list(view.elements(0, 3)) == [1, 2, 3]
    assert list(view.elements(1, 3)) == [2, 3, 5]

    with pytest.raises(IndexError, match="do not describe a valid slice"):
        view.elements(-1, None)
    with pytest.raises(IndexError, match="do not describe a valid slice"):
        view.elements(0, -1)
    with pytest.raises(IndexError, match="do not describe a valid slice"):
        view.elements(1, 4)

    # Check that unpacking will error
    with pytest.raises(ValueError, match="Can't unpack non-boolean buffer"):
        view.unpack_bits()

    # Check repr
    assert "1 2 3 5" in repr(view)


def test_buffer_view_copy():
    from array import array

    array_view = na.c_array([1, 2, 3, 4], na.int32()).view()
    view = array_view.buffer(1)

    # Check copying
    copied_all = view.copy()
    assert len(copied_all) == view.n_elements
    assert copied_all.data_type == "int32"
    assert list(copied_all) == [1, 2, 3, 4]

    copied_some = view.copy(1, 3)
    assert len(copied_some) == 3
    assert list(copied_some) == [2, 3, 4]

    # Check with non-zero destination offset
    out = array(view.format, [0, 0, 0, 0, 0, 0])
    assert view.copy_into(out, dest_offset=2) == 16
    assert list(out) == [0, 0, 1, 2, 3, 4]

    # Check error requesting out-of-bounds dest_offset
    with pytest.raises(IndexError, match="Can't unpack"):
        view.copy_into(out, dest_offset=-1)

    # Check errors from requesting out-of-bounds slices
    msg = "do not describe a valid slice"
    with pytest.raises(IndexError, match=msg):
        view.copy(-1, None)
    with pytest.raises(IndexError, match=msg):
        view.copy(0, -1)
    with pytest.raises(IndexError, match=msg):
        view.copy(0, 9)

    # Check errors from an output buffer of insufficient length
    out = array("i")
    msg = "Can't unpack 4 elements into buffer of size 0"
    with pytest.raises(IndexError, match=msg):
        view.copy_into(out)

    # Check errors from an output buffer with the wrong data type
    out = array("d", [0, 0, 0, 0])
    msg = "Destination buffer must have itemsize == 1 or itemsize == 4"
    with pytest.raises(ValueError, match=msg):
        view.copy_into(out)