from abc import ABC, abstractmethod
from typing import Optional

import numpy as np

from shapely import LineString

from molecule_movement import Matching, Goal, Molecule

from loguru import logger

from ..logging import cond_log_and_raise

class AbstractScheduling(ABC):
    def __init__(self, molecules: list[Molecule], goals: list[Goal], matching: list[Matching]):
        self.molecules = molecules
        self.goals = goals
        self.matching = matching

    @abstractmethod
    def compute_schedule(self, seed: Optional[int] = None) -> list[Matching]:
        pass

    @property
    def cls(self) -> str:
        return type(self).__name__

    @property
    def length(self) -> float:
        cond_log_and_raise(self.matching, ValueError(""), msg="Tried to compute length of matching, but matching is None")
        return sum([LineString([m.molecule.starting_position, m.goal.position]).length for m in self.matching])

    @property
    def statistics(self) -> str:
        return f"{self.cls} computed schedule of total length: {self.length}"
