from typing import Iterable, Tuple, Optional, List
import math
import numpy as np
from shapely.geometry import Point, MultiPoint

def points_along_direction(
    start: Tuple[float, float],
    direction_through: Tuple[float, float],
    spacing: float,
    *,
    n: Optional[int] = None,
    length: Optional[float] = None,
    include_start: bool = True,
) -> List[Point]:
    """
    Return a list of Shapely Points starting at `start`, moving toward `direction_through`,
    such that consecutive points are exactly `spacing` apart.

    Exactly one of `n` (number of points) or `length` (total ray length to cover) must be given.
    """
    if spacing <= 0:
        raise ValueError("spacing must be > 0")

    x0, y0 = map(float, start)
    x1, y1 = map(float, direction_through)

    # Unit direction vector from start toward direction_through
    dx, dy = x1 - x0, y1 - y0
    norm = math.hypot(dx, dy)
    if norm == 0.0:
        raise ValueError("direction_through must differ from start to define a direction")
    ux, uy = dx / norm, dy / norm

    if (n is None) == (length is None):
        raise ValueError("Provide exactly one of n or length")

    if length is not None:
        if length < 0:
            raise ValueError("length must be >= 0")
        # number of points including the start
        n = int(math.floor(length / spacing)) + 1

    k0 = 0 if include_start else 1  # skip the start if requested
    return [
        Point(x0 + k * spacing * ux, y0 + k * spacing * uy)
        for k in range(k0, k0 + int(n))
    ]

def multipoint_along_direction(
    start: Tuple[float, float],
    direction_through: Tuple[float, float],
    spacing: float,
    *,
    n: Optional[int] = None,
    length: Optional[float] = None,
    include_start: bool = True,
) -> MultiPoint:
    """Convenience wrapper returning a MultiPoint."""
    return MultiPoint(points_along_direction(
        start, direction_through, spacing,
        n=n, length=length, include_start=include_start
    ))
