import numpy as np

# Constants
SUCC = 0
FAIL = -1
FALSE = 0
TRUE = 1

# pgpr_vector class
class pgpr_vector:
    def __init__(self, n=0, value=None, dtype=float):
        """
        Initialize a vector.
        :param n: Size of the vector.
        :param value: Optional value to initialize all elements.
        :param dtype: Data type of the vector elements (default: float).
        """
        self.nn = n
        self.dtype = dtype
        val = dtype(value) if value is not None else dtype()
        self.v = [val] * n

    def __getitem__(self, i):
        if isinstance(i, slice):
            return self.v[i]
        if i < 0 or i >= self.nn:
            raise IndexError("Index out of bounds.")
        return self.v[i]


    def __setitem__(self, i, val):
        if isinstance(i, slice):
            indices = range(*i.indices(self.nn))
            if len(indices) != len(val):
                raise ValueError("Mismatched lengths for slice assignment.")
            for idx, v in zip(indices, val):
                self.v[idx] = v
        else:
            if i < 0 or i >= self.nn:
                raise IndexError("Index out of bounds.")
            self.v[i] = val

    def __sub__(self, other):
        """
        Subtract two pgpr_vector objects element-wise.
        :param other: Another pgpr_vector object.
        :return: A new pgpr_vector object with the result of the subtraction.
        """
        if not isinstance(other, pgpr_vector):
            raise TypeError("Subtraction is only supported between pgpr_vector objects.")
        if self.size() != other.size():
            raise ValueError("Vectors must have the same size for subtraction.")
        result = pgpr_vector(self.size())
        for i in range(self.size()):
            result[i] = self[i] - other[i]
        return result
    
    def __pow__(self, power):
        """
        Raise each element of the pgpr_vector to the given power.
        :param power: The exponent to raise each element to.
        :return: A new pgpr_vector with the result of the exponentiation.
        """
        if not isinstance(power, (int, float)):
            raise TypeError("Power must be an integer or float.")
        result = pgpr_vector(self.size())
        for i in range(self.size()):
            result[i] = self[i] ** power
        return result

    def size(self):
        return self.nn

    def __len__(self):
        return self.nn

    def resize(self, newn, value=None, dtype=None):
        """
        Resize the vector.
        :param newn: New size.
        :param value: Optional value to initialize new elements.
        :param dtype: Optional data type override.
        """
        if dtype is not None:
            self.dtype = dtype
        val = self.dtype(value) if value is not None else self.dtype()
        self.v = [val for _ in range(newn)]
        self.nn = newn

    def assign(self, newn, value, dtype=None):
        self.resize(newn, value, dtype)

    def data(self):
        return self.v


# pgpr_matrix class
class pgpr_matrix:
    def __init__(self, n=0, m=0, value=None, dtype=float):
        """
        Initialize a matrix.
        :param n: Number of rows.
        :param m: Number of columns.
        :param value: Optional value to initialize all elements.
        """
        self.nn = n
        self.mm = m
        self.dtype = type(value) if value is not None else float
        self.v = np.full((n, m), value, dtype=self.dtype) if value is not None else np.zeros((n, m), dtype=self.dtype)

    def __getitem__(self, i):
        """
        Get the i-th row.
        """
        if i < 0 or i >= self.nn:
            raise IndexError("pgpr_matrix subscript out of bounds")
        return self.v[i]

    def __setitem__(self, i, value):
        """
        Set the i-th row.
        """
        if i < 0 or i >= self.nn:
            raise IndexError("pgpr_matrix subscript out of bounds")
        self.v[i] = value

    def nrows(self):
        """
        Get the number of rows.
        """
        return self.nn

    def ncols(self):
        """
        Get the number of columns.
        """
        return self.mm

    def resize(self, newn, newm, value=None, dtype=None):
        """
        Resize the matrix.
        :param newn: New number of rows.
        :param newm: New number of columns.
        :param value: Optional value to initialize new elements.
        :param dtype: Optional dtype override.
        """
        if dtype is not None:
            self.dtype = dtype
        self.v = np.full((newn, newm), value, dtype=self.dtype) if value is not None else np.zeros((newn, newm), dtype=self.dtype)
        self.nn = newn
        self.mm = newm

    def assign(self, newn, newm, value, dtype=None):
        """
        Resize and assign a constant value to all elements.
        :param newn: New number of rows.
        :param newm: New number of columns.
        :param value: Value to assign.
        :param dtype: Optional dtype override.
        """
        self.resize(newn, newm, value, dtype)
    
# Type aliases for vectors and matrices
Vbool = lambda n=0, value=False: pgpr_vector(n, value, dtype=bool)
Vint = lambda n=0, value=0: pgpr_vector(n, value, dtype=int)
Vuint = lambda n=0, value=0: pgpr_vector(n, value, dtype=np.uint32)
Vdoub = lambda n=0, value=0.0: pgpr_vector(n, value, dtype=float)

Mint = lambda n=0, m=0, value=0: pgpr_matrix(n, m, value, dtype=int)
Muint = lambda n=0, m=0, value=0: pgpr_matrix(n, m, value, dtype=np.uint32)
Mdoub = lambda n=0, m=0, value=0.0: pgpr_matrix(n, m, value, dtype=float)