from abc import ABC, abstractmethod

from torch_geometric.data import Data


class GraphGeneratorConfig(ABC):
    """
    For each graph generator, a subclass of this class should exist that can hold the parameters for the graph
    generator.
    Instances of that subclass can then directly be used to call the graph generator.

    Subclasses of this class should be annotated with `@yaml_object(YAML())` and `@dataclass()`, e.g.
    ```python
    from ruamel.yaml import YAML, yaml_object

    @yaml_object(YAML())
    @dataclass()
    class MyConfig(GraphGeneratorConfig):
        ...
    ```
    """

    @abstractmethod
    def generate_graph(self) -> Data:
        """
        Generates a graph based on this configuration.

        Subclasses can implement this by simply assigning the graph generator function that this config class is
        intended to configure:
        ```python
        class MyConfig(GraphGeneratorConfig):
            generate_graph = my_graph_generator_function
        ```
        The graph generator function should take an instance of this config class as an argument, and no other
        arguments.
        It should generate and return a single graph.

        The simple assignment works because `self` (i.e. the config) will implicitly be passed to the function when it
        is called as
        ```python
        my_config = MyConfig(...)
        graph = my_config.generate_graph()
        ```

        It is important that the function is assigned without a type hint, since the `@dataclass` decorator would
        otherwise add an argument to the generated constructor.
        This would allow the function to accidentally be changed.
        """
        pass

    @abstractmethod
    def validate(self):
        """
        Checks whether the configuration is valid.
        If not, a `ValueError` is raised.
        """
        pass
