.. toctree:: :maxdepth: 2 :caption: Contents: Creating the Application ------------------------ Now that we're set up to create a Ligare application, let's find out how to actually create one. We will follow a structure that lets us write the application as a single Python "module," similar to how `python-guide.org `_ demonstrates. First create the module. .. code-block:: shell-session user@: my-ligare-app $ mkdir app user@: my-ligare-app $ touch app/__init__.py app/__main__.py Now we need to add some code to ``__main__.py``. First, let's add some imports that our application depends on. .. code-block:: python from injector import inject from Ligare.programming.application import ApplicationBase, ApplicationBuilder from typing_extensions import override We're going to write an application that displays a message stored in a configuration file. To do this, we need to do a few things: 1. Create an ``Application`` class 2. Create a ``Config`` class and file 3. Configure the application Creating the Application Class ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ All Ligare applications are extensions of ``ApplicationBase``. These classes must implement a ``run`` method, which can use the Ligare dependency injection system. .. code-block:: python class Application(ApplicationBase): @override def run(self): print("Hello world!") input("\nPress anything to exit. ") Here, we have created a class called ``Application`` that extends ``ApplicationBase``. Although we could run our application by instantiating ``Application`` and calling ``run``, we would lose out on what Ligare care do for us. To take advantage of that functionality, we will use ``ApplicationBuilder``. .. code-block:: python builder = ApplicationBuilder(Application) This gives us a builder for our ``Application`` class with which we can configure the application, and then start it. This is the minimum required to set up our application. We can now "build" it, and then start it. .. code-block:: python application = builder.build() if __name__ == "__main__": application.run() Our completed application looks like this. .. code-block:: python from injector import inject from Ligare.programming.application import ApplicationBase, ApplicationBuilder from typing_extensions import override class Application(ApplicationBase): @inject @override def run(self): print("Hello, world!") input("\nPress anything to exit. ") builder = ApplicationBuilder(Application) application = builder.build() if __name__ == "__main__": application.run() When we run the application, we see the expected output. .. code-block:: shell-session user@: my-ligare-app $ python -m app Hello, world! Press anything to exit. Creating the Config Class ^^^^^^^^^^^^^^^^^^^^^^^^^ Now let's see how to create the ``Config`` class and make better use of Ligare. Here we will see: * How to use Ligare's dependency injection system * How to set up application configuration with a file * How to use configuration values from a configuration file Start by importing Ligare's base ``Config`` class. .. code-block:: python from Ligare.programming.config import Config To create a ``Config`` class for our application, we need to extend ``Config`` and add any fields to it that we expect to be in a configuration file, and that we want the application to be able to read. .. code-block:: python class AppConfig(Config): message: str Because we want to display a message from a configuration file, we just add a "message" field to our class. We will not set this value directly; this class is a Pydantic dataclass whose values Ligare will set from a TOML file. Registering the Config Class ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ For Ligare to know about this class, and to tell Ligare where our configuration file is, we use the ``use_configuration`` method from the ``ApplicationBuilder`` instance to "register" the class with Ligare's dependency injection system. This must occur before ``builder.build()`` is called. .. code-block:: python builder.use_configuration( lambda config_builder: config_builder \ .with_config_type(AppConfig) \ .with_config_filename("app/config.toml") ) ``use_configuration`` expects to receive a method that takes a single parameter. That parameter type is itself an ``ApplicationConfigBuilder`` that is used to set options specific to the configuration of a Ligare application. `It's builders all the way down! `_ Using the Config Class ~~~~~~~~~~~~~~~~~~~~~~ Now we need to adjust our run method. .. code-block:: python class Application(ApplicationBase): @inject # 1 @override def run(self, config: "AppConfig"): # 2 print(config.message) # 3 input("\nPress anything to exit. ") There are three notable changes here: 1. The ``@inject`` decorator was added to the run method 2. The ``config`` parameter was added to the run method 3. The ``config`` object is used to print a field value called ``message`` Ligare supports the use of `Dependency Injection `_ so you can use certain objects throughout your application without having to instantiate them yourself. In this case, ``run(config)`` is called automatically by Ligare. Because the ``@inject`` decorator is present (#1), the value of ``config`` is created automatically, and it is passed into your run method. We then need to specify what the type of the parameter is (#2); in this case, the type is ``AppConfig``. This is accomplished with Python's `type annotation `_ system, by writing ``config: "AppConfig"``. Finally, we use the value of ``config`` to access the field ``message``, and pass that into ``print``. (#3) With these changes, your application should resemble the Ligare example CLI application. .. literalinclude:: ../../../../../examples/cli-app/app/__main__.py :language: python :caption: :example:`cli-app/app/__main__.py` Creating the Application Configuration File ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ There is one last thing to do, which is to create the configuration file and store our message in it. .. code-block:: shell-session user@: my-ligare-app $ cat > app/config.toml <