From eb001559476fd8d2e8e35b5914b5cb20286949ca Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Fri, 23 Aug 2024 05:23:34 +0000 Subject: [PATCH 1/7] Update contributing.rst --- guides/contributing.rst | 539 +++++++++++++++++++++++----------------- 1 file changed, 308 insertions(+), 231 deletions(-) diff --git a/guides/contributing.rst b/guides/contributing.rst index d6dbee0fca..6e6a0c49a1 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -5,75 +5,81 @@ Contributing :description: Getting started guide for contributing to the ESPHome project :image: github-circle.svg -Contributions to the ESPHome suite are very welcome! All the code for the projects -is hosted on GitHub and you can find the sources here: +We welcome contributions to the ESPHome suite of code and documentation! All the code for the projects is hosted on +GitHub: - `ESPHome `__ (Project Source Code) - `ESPHome-Docs `__ (The documentation which you're reading here) -Just clone the repository locally, do the changes for your new feature/bug fix and submit -a pull request. +Just clone the repository locally, make changes for your new feature/bug fix and submit a pull request. -Contributing to ESPHome-Docs ----------------------------- +Contributing to ``esphome-docs`` +-------------------------------- .. image:: /images/logo-docs.svg :align: center :width: 60.0% :class: dark-invert -One of the areas of ESPHome that can always be improved is the documentation. -If you see an issue somewhere, a spelling mistakes or if you want to share your awesome -setup, please feel free to submit a pull request. +Our documentation can always be improved. We rely on contributions from our users to do so. If you notice an issue (for +example, spelling/grammar mistakes) or if you want to share your awesome new setup, we encourage you to submit a pull +request (PR). The ESPHome documentation is built using `sphinx `__ and uses -`reStructuredText `__ for all source files. +`reStructuredText `__ for all source (``.rst``) files. -If you're not familiar with writing rST, see :ref:`rst-syntax` for a quick refresher. +If you're not familiar with rST, see :ref:`rst-syntax` for a quick refresher. -Through Github +Through GitHub ************** -This guide essentially goes over the same material found in `GitHub's Editing files in another user's repository `__. You may also find that page helpful to read. +This guide essentially goes over the same material found in +`GitHub's Editing files in another user's repository `__. +You may also find that page helpful to read. -At the bottom of each page in the docs, there is a "Edit this page on GitHub" link. Click this link and you'll see something like this: +At the bottom of each page in the docs, there is a "Edit this page on GitHub" link. Click this link and you'll see +something like this: .. figure:: images/docs_ghedit_1.png :align: center :width: 80.0% :alt: a screenshot of an rST file opened in GitHub, with the edit button circled -Click the edit button to start making changes. If you're not sure about some element of syntax, see the quick-start :ref:`rst-syntax` guide. +Click the edit button to start making changes. If you're not sure about some element of syntax, see the quick-start +:ref:`rst-syntax` guide. -Once you've made your changes, give them a useful name and press "Propose changes". At this point, you've made the changes on your own personal copy of the docs in GitHub, but you still need to submit them to us. +Once you've made your changes, give them a useful name and press "Propose changes". At this point, you've made the +changes on your own personal copy of the docs in GitHub, but you still need to submit them to us. .. figure:: images/docs_ghedit_2.png :align: center :width: 80.0% :alt: the commit creation screen in GitHub, with the commit title and "Propose changes" button circled -To do that, you need to create a "Pull request": +To do that, you need to create a "Pull request" (PR): .. figure:: images/docs_ghedit_3.png :align: center :width: 80.0% :alt: the pull request prompt screen in GitHub with the "Create pull request" button circled -Fill out the new pull request form, replacing the ``[ ]`` with ``[x]`` to indicate that you have followed the instructions. +Fill out the new pull request form, replacing the ``[ ]`` with ``[x]`` to indicate that you have followed the +instructions. .. figure:: images/docs_ghedit_4.png :align: center :width: 80.0% :alt: the pull request creation screen in GitHub with the "Create pull request" button circled -After waiting a while, you might see a green or a red mark next to your commit in your pull request: +After a few minutes, you'll see either a green check ✅ or a red ❌ next to your commit in your pull request: .. figure:: images/docs_ghedit_ci_failed.png :align: center :width: 80.0% :alt: the pull request with a commit with a red x next to it -This means that there is some error stopping your pull request from being fully processed. Click on the X, click on "Details" next to the lint step, and look and see what's causing your change to fail. +If you see the red ❌, there is at least one error preventing your pull request from being fully processed. Click on the +❌, then click on "Details" next to the lint step to determine what's wrong with your change(s). .. figure:: images/docs_ghedit_ci_details.png :align: center @@ -85,9 +91,15 @@ This means that there is some error stopping your pull request from being fully :width: 80.0% :alt: log messages showing reason for failed build -For example, in this case, you'd want to go to line 136 of ``pzemac.rst`` and adjust the number of ``===`` so that it completely underlines the section heading. +In this example, you need to go to line 136 of ``pzemac.rst`` and adjust the number of ``===`` so that it completely +underlines the section heading. -Once you make that change, the pull request will be built again, and hopefully this time where will be no other errors. +Once you make that change, the pull request will be tested & built again; ideally, this time where will be no remaining +errors. If, however, more errors are discovered, simply repeat the process above to correct them. + +.. note:: + + All tests must be passing before we will review (and merge) a pull request. Build ***** @@ -100,7 +112,8 @@ Build docker run --rm -v "${PWD}/":/workspaces/esphome-docs -p 8000:8000 -it ghcr.io/esphome/esphome-docs - With ``PWD`` referring to the root of the ``esphome-docs`` git repository. Then go to ``:8000`` in your browser. + With ``PWD`` referring to the root of the ``esphome-docs`` git repository. Then go to ``:8000`` in + your browser. This way, you don't have to install the dependencies to build the documentation. @@ -121,16 +134,21 @@ Then, use the provided Makefile to build the changes and start a live-updating w Notes ***** -Some notes about the docs: - - Use the English language (duh...) -- An image tells a thousand words, please use them wherever possible. But also don't forget to shrink them, for example - I often use https://tinypng.com/ -- Try to use examples as often as possible (also while it's great to use highly accurate, - and domain-specific lingo, it should not interfere with new users understanding the content) -- Fixes/improvements for the docs themselves should go to the ``current`` branch of the - esphomedocs repository. New features should be added against the ``next`` branch. -- Always create new branches in your fork for each pull request. +- An image is worth a thousand words, please use them wherever possible. Regardless, don't forget to optimize them so + pages load quickly; you might try using https://tinypng.com/ +- Try to use examples as often as possible. +- When using highly accurate, domain-specific terminology, be sure that it does not interfere with new users + understanding the content. +- Be sure to target the correct **base branch** of the ``esphome-docs`` repository: + + - **Fixes/improvements** for documentation must target the ``current`` branch. + - **New features** must target the ``next`` branch. + +- **Create new branches in your fork** for each pull request; to avoid confusion (and other potential issues), do not + make changes directly in the ``next`` and ``current`` branches in your fork. +- Wrap lines in all files at no more than 120 characters. This makes reviewing PRs faster and easier. Exceptions should + be made only for lines which contain long links or other specific content/metadata that cannot be wrapped. .. _rst-syntax: @@ -139,10 +157,12 @@ Syntax Here's a quick RST primer: -Title hierarchy is based on order of occurrence, not on type of character used to underline it. This -documents establish the following character order for better consistency. +Title hierarchy is based on order of occurrence, not on type of character used to underline it. For consistency, we +adhere to the following order: -- **Headers**: You can write titles like this: +- **Headers**: + + You can write titles like this: .. code-block:: rst @@ -153,20 +173,27 @@ documents establish the following character order for better consistency. .. code-block:: rst - My Sub Section - -------------- + My Section + ---------- and sub-section headers like this: .. code-block:: rst - My Sub-sub Section - ****************** + My Sub-section + ************** + + and sub-sub-section headers like this: + + .. code-block:: rst + + My Sub-sub-section + ^^^^^^^^^^^^^^^^^^ .. note:: - The length of the bar below the text **must** match the title Text length. - Also, titles should be in Title Case + - The length of the bar below the text **must** match the title text length. + - Section titles should use Title Case. - **Links**: To create a link to an external resource (for example https://www.google.com), use ``\`Link text \`__``. For example: @@ -177,8 +204,8 @@ documents establish the following character order for better consistency. `Google.com `__ -- **References**: To reference another document, use the ``:doc:`` and ``:ref:`` roles (references - are set up globally and can be used between documents): +- **References**: To reference another document, use the ``:doc:`` and ``:ref:`` roles (references are set up globally + and can be used between documents): .. code-block:: rst @@ -223,9 +250,9 @@ documents establish the following character order for better consistency. .. note:: - Please note the empty line after the ``code-block`` directive. That is necessary. + Note that a blank line is *required* after every ``code-block`` directive. -- **Images**: To show images, use the ``figure`` directive: +- **Images**: Use the ``figure`` directive to display an image: .. code-block:: rst @@ -243,13 +270,11 @@ documents establish the following character order for better consistency. .. note:: - All images in the documentation need to be as small as possible to ensure - fast page load times. For normal figures the maximum size should be at most - about 1000x800 px or so. Additionally, please use online tools like + All images in the documentation need to be as small as possible to minimize page load times. Typically, the + maximum size should be roughly 1000x800 px or so. Additionally, please use online tools like https://tinypng.com/ or https://tinyjpg.com/ to further compress images. -- **Notes and warnings**: You can create simple notes and warnings using the ``note`` and ``warning`` - directives: +- **Notes and warnings**: You can create simple notes and warnings using the ``note`` and ``warning`` directives: .. code-block:: rst @@ -305,28 +330,30 @@ documents establish the following character order for better consistency. 2. Ordered item #2 - **imgtable**: ESPHome uses a custom RST directive to show the table on the front page (see - `index.rst `__). - New pages need to be added to the ``imgtable`` list. The syntax is CSV with , (without RST), - (in top-level images/ directory), (optional - short text to describe the component). The aspect ratio of these images should be 8:10 (or 10:8) but exceptions are possible. + `index.rst `__). New pages need to be added to the + ``imgtable`` list. The syntax is CSV with , (without RST), (in the top-level + ``images/`` directory), (optional; short text to describe the component). The aspect ratio of these images + should be 8:10 (or 10:8) but exceptions are possible. Because these images are served on the main page, they need to be compressed heavily. SVGs are preferred over JPGs - and JPGs should be max. 300x300px. + and JPGs should be no more than 300x300px. + If you have imagemagick installed, you can use this command to convert the thumbnail: .. code-block:: bash convert -sampling-factor 4:2:0 -strip -interlace Plane -quality 80% -resize 300x300 in.jpg out.jpg -reStructured text can do a lot more than this, so if you're looking for a more complete guide -please have a look at the `Sphinx reStructuredText Primer `__. +reStructured text can do a lot more than this; if you're looking for a more complete guide, please have a look at the +`Sphinx reStructuredText Primer `__. .. _setup_dev_env: -Setting Up Development Environment ----------------------------------- +Setting Up a Development Environment +------------------------------------ -For developing new features to ESPHome, you will first need to set up a development environment. -This is only possible for ``pip`` installs. +If you want to develop (a) new feature(s) for ESPHome, you need to set up a development environment. +Note that ``pip`` must be installed before running the ``setup`` script. .. code-block:: bash @@ -341,31 +368,29 @@ This is only possible for ``pip`` installs. git checkout -b my-new-feature cd .. -The environment is now ready for use, but you need to activate the Python virtual environment -every time you are using it. +The environment is now ready for use, but you'll need to activate the Python virtual environment (venv) each time you +wish to use it. .. code-block:: bash # Activate venv source venv/bin/activate -Now you can open ESPHome in your IDE of choice (mine is CLion) with the PlatformIO -addons (see PlatformIO docs for more info). Then develop the new feature with the -guidelines below. +Now you can open ESPHome in your IDE of choice (many of us are using `VSCode `__) +with the PlatformIO addons (see PlatformIO docs for more info) and develop the new feature with the guidelines below. -All PRs are automatically checked for some basic formatting/code mistakes with Github Actions. -These checks *must* pass for your PR to be mergeable. +All PRs are automatically checked and tested for some common formatting/code errors with Github Actions. *These checks* +**must all pass** *before we will review (and eventually merge) your PR.* Setting Up Git Environment -------------------------- -ESPHome's code-base is hosted on GitHub, and contributing is done exclusively through -"Pull Requests" (PRs) in the GitHub interface. So you need to set up your git environment -first. +ESPHome's codebase is hosted on GitHub; contributing is done exclusively through "Pull Requests" (PRs) in the +GitHub interface. To use this, you need to set up your ``git`` environment first. -When you want to create a patch for ESPHome, first go to the repository you want to contribute to -(esphome, etc) and click fork in the top right corner. This will create -a fork of the repository that you can modify and create git branches on. +If you want to contribute changes/fixes you've made back to ESPHome, first, go to the repository you want to contribute +to (``esphome``, for example) and click "fork" in the top right corner. This will create a fork of the repository that +you can modify and create branches on. .. code-block:: bash @@ -396,20 +421,96 @@ a fork of the repository that you can modify and create git branches on. git push -u origin # For example: git push -u origin gpio-switch-fix -Then go to your repository fork in GitHub and wait for a create pull request message to show -up in the top (alternatively go to branches and create it from there). Fill out the -Pull Request template outlining your changes; if your PR is not ready to merge yet please -mark it as a draft PR in the dropdown of the green "create PR" button. +Submitting a Pull Request +************************* + +After you have pushed your changes to GitHub, go to your repository fork and look for a "create pull request" button +near the top of the page (or, alternatively, go to branches and create it from there). As you create the PR: + +- Complete the Pull Request template: + + - Include a brief (but complete) summary of your changes. + - PRs without a descrption/summary of the changes will not be reviewed or merged, although exceptions may + occasionally be made for small PRs and/or PRs made by frequent contributors/maintainers. + - **Do not delete the template.** + +- **Mark your PR as a draft** if it is not ready to be reviewed or merged yet. Your PR should be considered a draft if: + + - You still plan to make more changes to the code/documentation. + - Changes have been requested to the PR but you have not completed making (or discussing) the requested changes yet. + - You are waiting on feedback from the community and/or maintainers to complete your changes. + + This lets reviewers know that the PR may continue to change so they will not spend valuable time reviewing it until + it is ready. We do this because, if a PR is reviewed and then it changes, it must be re-reviewed. Reviewing a single + PR multiple times is not a productive use of time and we try as much as possible to avoid doing so. + +Review Process +************** + +ESPHome's maintainers work hard to maintain a high standard for its code, so reviews can take some time. At the bottom +of each pull request you will see the "Github Actions" continuous integration (CI) checks which will automatically +analyize all code changed in your branch. These checks try to spot (and suggest corrections for) errors. If any CI +check fails, please look at the Github Actions log and fix all errors that appear there. + +**All automated checks must be passing** before a given PR will be reviewed and (eventually) merged! + +**When will my PR be reviewed/merged?** + +ESPHome is a big project; we encourage everybody to test, review and comment on PRs. Despite this, reviews can (and +often do) take some time. + +**But howwww looonnnggg???** + +Small PRs are easier to review and are often reviewed first. If you want your PR to be reviewed (and merged) quickly, +here are some tips: + +- We would rather review ten ten-line PRs than one 100-line PR. +- Be sure to follow all :ref:`codebase_standards` as you make changes -- when reviewers have to spend time + commenting on/correcting your PR because you didn't name variables correctly or didn't prefix member variable + accesses with ``this->``, it wastes time we could be using to review other PRs which *do* follow the standards. +- If you wish to take on a big project, such as refactoring a substantial section of the codebase or integrating + another open source project with ESPHome, please discuss this with us on `Discord `__ or + `create a discussion on GitHub `__ **before** you do all the work and + attempt to submit a massive PR. +- While we realize it's not *always* possible, avoid submitting PRs which are thousands of lines in size. Such PRs are + simply too complex and take excessive amounts of time to review. Break your work into multiple, smaller PRs to make + the changes more tenable for reviewers. +- If you are not sure about how you should proceed with some changes, **please** + `discuss it with us on Discord `__ before you go do a bunch of work that we can't (for + whatever reason) accept...and then you have to go back and re-do it all to get your PR merged. It's easier to make + corrections early-on -- and we want to help you! + +Why Was My PR was Marked as a Draft? +************************************ + +If your PR was reviewed and changes were requested, our bot will automatically mark your PR as a draft. This means that +the PR is not ready to be merged or further reviewed for the moment. + +When a PR is marked as a draft, it tells other reviewers that this particular PR is a work-in-progress and it doesn't +require their attention yet. + +Once you have made the requested changes, you can mark the PR as ready for review again by clicking the "Ready for +review" button: + +.. figure:: images/pr-draft-ready.png + :align: center + :width: 100.0% + :alt: The ready for review button in the bottom of a PR in draft mode + +Before you click the "Ready for review" button, ensure that: -**Review Process:** ESPHome's code base tries to have a high code standard. At the bottom -of the Pull Request you will be able to see the "Github Actions" continuous integration check which -will automatically go through your patch and try to spot errors. If the CI check fails, -please see the Github Actions log and fix all errors that appear there. Only PRs that pass the automated -checks can be merged! +- You have addressed all requested changes +- There are no merge conflicts +- All CI jobs and checks are passing successfully -**Catching up with reality**: Sometimes other commits have been made to the same files -you edited. Then your changes need to be re-applied on top of the latest changes with -a "rebase". More info `here `__. +Once you've clicked the "Ready for review" button, the PR will return to a normal state again and our bot will +automatically notify the reviewers who requested the changes that the PR is ready to go! + +Catching Up with Reality +************************ + +Sometimes other commits have been made to the same files you edited. Your changes need to be re-applied on top of the +latest changes with a "rebase". More info `here `__. .. code-block:: bash @@ -427,21 +528,19 @@ Contributing to ESPHome :width: 60.0% :class: dark-invert -This is a guide to contributing to the ESPHome codebase. ESPHome uses two languages for its project: -Python and C++. +This is a guide to contributing to the ESPHome codebase. ESPHome uses two languages for its project: Python and C++. -The user configuration is read, validated and transformed into a custom firmware -with the Python side of the firmware. +The Python side of ESPHome reads a YAML configuration file, validates it and transforms it into a custom firmware which +includes only the code needed to perform as defined in the configuration file. -The C++ codebase is what's actually running on the ESP and called the "runtime". This part of -the codebase should first set up the communication interface to a sensor/component/etc. and -communicate with the ESPHome core via the defined interfaces (like Sensor, BinarySensor, Switch). +The C++ part of the codebase is what's actually running on the microcontroller; this is called the "runtime". This part +of the codebase should first set up the communication interface to a sensor/component/etc. and then communicate with +the ESPHome core via the defined interfaces (like ``Sensor``, ``BinarySensor`` and ``Switch``, among others). -1. Directory Structure -********************** +Directory Structure +******************* -After you've :ref:`set up development environment `, you will have a folder structure -like this: +After you've :ref:`set up a development environment `, you will have a folder structure like this: .. code-block:: text @@ -464,10 +563,10 @@ like this: │ │ ├── switch.py │ ... -As you can see, all components are in the "components" folder. Each component is in its own -subfolder which contains the Python code (.py) and the C++ code (.h and .cpp). +All components are in the "components" folder. Each component is in its own subfolder which contains the Python code +(``.py``) and the C++ code (``.h`` and ``.cpp``). -Suppose the user types in the following: +Consider a YAML configuration file containing the following: .. code-block:: yaml @@ -476,34 +575,35 @@ Suppose the user types in the following: sensor: - platform: hello2 -In both cases, ESPHome will automatically look for corresponding entries in the "components" -folder and find the directory with the given name. For example the first entry will make ESPHome -look at the ``esphome/components/hello1/__init__.py`` file and the second entry will result in +In both cases, ESPHome will automatically look for corresponding entries in the "components" folder and find the +directory with the given name. In this example, the first entry causes ESPHome to look for the +``esphome/components/hello1/__init__.py`` file and the second entry tells ESPHome to look for ``esphome/components/hello2/sensor.py``. -Let's leave what's written in those files for (2.), but for now you should also know that -whenever a component is loaded, all the C++ source files in the folder of the component -are automatically copied into the generated PlatformIO project. So you just need to add the C++ -source files in the folder and the ESPHome core will copy them with no additional code required -by the component developer. +Let's leave what's written in those files for :ref:`the next section `, but for now you should also +know that, whenever a component is loaded, all the C++ source files in the folder of the component are automatically +copied into the generated PlatformIO project. All you need to do is add the C++ source files in the component's folder +and the ESPHome core will copy them with no additional code required by the component developer. .. note:: - For testing you can use :doc:`/components/external_components`. + For testing, you can use :doc:`/components/external_components`. ESPHome also has a ``custom_components`` mechanism like `Home Assistant does - `__. However this is - discouraged in favor of :doc:`/components/external_components`. + `__. Note, however, that + **custom componenets are deprecated** in favor of :doc:`/components/external_components`. -2. Config Validation -******************** +.. _config_validation: -The first thing ESPHome does is read and validate the user config. For this ESPHome has a powerful -"config validation" mechanism. Each component defines a config schema that is validated against -the user config. +Config Validation +***************** -To do this, all ESPHome Python modules that can be configured by the user have a special field -called ``CONFIG_SCHEMA``. An example of such a schema is shown below: +The first task ESPHome performs is to read and validate the provided YAML configuration file. ESPHome has a powerful +"config validation" mechanism for this purpose. Each component defines a config schema which is used to validate the +provided configuration file. + +To do this, all ESPHome Python modules that can be configured by the user define a special variable named +``CONFIG_SCHEMA``. An example of such a schema is shown below: .. code-block:: python @@ -517,31 +617,32 @@ called ``CONFIG_SCHEMA``. An example of such a schema is shown below: cv.Optional(CONF_MY_OPTIONAL_KEY, default=10): cv.int_, }).extend(cv.COMPONENT_SCHEMA) -This variable is automatically loaded by the ESPHome core and validated against. +This variable is automatically loaded by the ESPHome core and is used to validate the provided configuration. The underlying system ESPHome uses for this is `voluptuous `__. -Going into how to validate is out of scope for this guide, but the best way to learn is to look -at examples of how similar components validate user input. +How validation works is out of scope for this guide; the easiest way to learn is to look at how similar components +validate user input. -A few point on validation: +A few notes on validation: -- ESPHome puts a lot of effort into **strict validation** - If possible, all validation methods should be as strict - as possible and detect wrong user input at the validation stage (and not later). -- All default values should be defined in the schema (and not in C++ codebase or other code parts). -- Config keys should be descriptive - If the meaning of a key is not immediately obvious you should - always prefer long_but_descriptive_keys. +- ESPHome puts a lot of effort into **strict validation**. All validation methods should be as strict as possible and + detect incorrect user input at the validation stage, mitigating compiler warnings and/or errors. +- All default values should be defined in the schema -- not in C++ codebase. +- Prefer naming configuration keys in a way which is descriptive instead of short. Put another way, if the meaning of a + key is not immediately obvious, don't be afraid to use ``long_but_descriptive_keys``. There is no reason to use + obscure shorthand. As an example, ``scrn_btn_inpt`` is indeed shorter but more difficult to understand, particularly + for new users; avoid naming keys and variables in this way. -3. Code Generation -****************** +Code Generation +*************** -After the user input has been successfully validated, the last step of the Python codebase -is called: Code generation. +The last step the Python codebase performs is called *code generation*. This runs only after the user input has been +successfully validated. -As you may know, ESPHome converts the user's configuration into C++ code (you can see the generated -code under ``/src/main.cpp``). Each component must define its own ``to_code`` method -that converts the user input to C++ code. +As you may know, ESPHome "converts" the user's YAML configuration into C++ code (you can see the generated code under +``/src/main.cpp``). Each component must define its own ``to_code`` method that "converts" the user input to +C++ code. -This method is also automatically loaded and invoked by the ESPHome core. An example of -such a method can be seen below: +This method is also automatically loaded and invoked by the ESPHome core. Here's an example of such a method: .. code-block:: python @@ -553,34 +654,34 @@ such a method can be seen below: cg.add(var.set_my_required_key(config[CONF_MY_REQUIRED_KEY])) -Again, going into all the details of ESPHome code generation would be out-of-scope. However, -ESPHome's code generation is 99% syntactic sugar - and again it's probably best to study other -components and just copy what they do. +The details of ESPHome code generation is out-of-scope for this document. However, ESPHome's code generation is 99% +syntactic sugar - and (again) it's probably best to study similar components and just copy what they do. There's one important concept for the ``to_code`` method: coroutines with ``yield``. -First the problem that leads to coroutines: In ESPHome, components can declare (via ``cg.Pvariable``) and access variables -(``cg.get_variable()``) - but sometimes when one part of the code base requests a variable -it has not been declared yet because the code for the component creating the variable has not run yet. -To allow for ID references, ESPHome uses so-called ``coroutines``. When you see a ``yield`` statement -in a ``to_code`` method, ESPHome will call the provided method - and if that method needs to wait -for a variable to be declared first, ``yield`` will wait until that variable has been declared. -After that, ``yield`` returns and the method will execute on the next line. +The problem that necessitates coroutines is this: in ESPHome, components can declare (via ``cg.Pvariable``) and access +variables (``cg.get_variable()``) -- but sometimes, when one part of the codebase requests a variable, it has not been +declared yet because the code for the component creating the variable has not yet run. + +To allow for ID references, ESPHome uses so-called ``coroutines``. When you see a ``yield`` statement in a ``to_code`` +method, ESPHome will call the provided method and, if that method needs to wait for a variable to be declared first, +``yield`` will wait until that variable has been declared. After that, ``yield`` returns and the method will execute on +the next line. -Next, there's a special method - ``cg.add`` - that you will often use. ``cg.add()`` does a very simple -thing: Any C++ declared in the parentheses of ``cg.add()`` will be added to the generated code. -If you do not call "add" a piece of code explicitly, it will not be added to the main.cpp file! +Next, there's a special method - ``cg.add`` - that you will often use. ``cg.add()`` performs a very simple task: Any +C++ declared in the parentheses of ``cg.add()`` will be added to the generated code. Note that, if you do not call +"add" to insert a piece of code explicitly, it will not be added to the ``main.cpp`` file! -4. Runtime -********** +Runtime +******* -Okay, the Python part of the codebase is now complete - now let's talk about the C++ part of -creating a new component. +At this point, the Python part of the codebase has completed its work. Let's move on and discuss the C++ part of +components. -The two major parts of any component roughly are: +Most components consist of two primary parts/steps: - - Setup Phase - - Run Phase +- Setup Phase +- Run Phase When you create a new component, your new component will inherit from :apiclass:`Component`. That class has a special ``setup()`` method that will be called once to set up the component - @@ -592,83 +693,83 @@ if communication works (if not, it should call ``mark_failed()``). Again, look at examples of other components to learn more. -The next thing that will be called with your component is ``loop()`` (or ``update()`` for a -:apiclass:`PollingComponent`). In these methods you should retrieve the latest data from the -component and publish them with the provided methods. One thing to note in these methods -is that anything in ``loop()`` or ``setup()`` **should not block**. Specifically methods like -``delay(10)`` should be avoided and delays above ~10ms are not permitted. The reason for this -is that ESPHome uses a central single-threaded loop for all components - if your component -blocks the whole loop will be slowed down. +The next method that will be called with your component is ``loop()`` (or ``update()`` for a +:apiclass:`PollingComponent`). These methods should retrieve the latest data from your component and publish them with +the provided methods. -Finally, your component should have a ``dump_config`` method that prints the user configuration. +.. note:: + + **Code in** ``loop()``, ``update()`` **and** ``setup()`` **must not block**. Specifically, methods like ``delay()`` + should be avoided and **delays longer than 10 ms are not permitted**. The reason for this is that ESPHome uses a + single-threaded loop for all components -- if your component blocks, it will delay the whole loop, negatively + impacting other components. This can result in a variety of problems such as network connections being lost. -5. Extras -********* +Finally, your component must have a ``dump_config`` method that prints the complete user configuration. + +Extras +****** .. note:: - This serves as documentation for some of ESPHome's internals and is not necessarily part of the - development guide. + This serves as documentation for some of ESPHome's internals and is not necessarily part of the development guide. -All Python modules have some magic symbols that will automatically be loaded by the ESPHome -loader. These are: +All Python modules have some magic symbols that will automatically be loaded by the ESPHome loader. These are: - ``CONFIG_SCHEMA``: The configuration schema to validate the user config against. -- ``to_code``: The function that will be called with the validated configuration and should - create the necessary C++ source code. -- ``DEPENDENCIES``: Mark the component to depend on other components. If the user hasn't explicitly - added these components in their configuration, a validation error will be generated. +- ``to_code``: The function that will be called with the validated configuration and should create the necessary C++ + source code. +- ``DEPENDENCIES``: Mark the component to depend on other components. If the user hasn't explicitly added these + components in their configuration, a validation error will be generated. - ``AUTO_LOAD``: Automatically load a component if the user hasn't added it manually. -- ``MULTI_CONF``: Mark this component to accept an array of configurations. If this is an - integer instead of a boolean, validation will only permit the given number of entries. -- ``CONFLICTS_WITH``: Mark a list of components as conflicting with this component. If the user - has one of them in the config, a validation error will be generated. - +- ``MULTI_CONF``: Mark this component to accept an array of configurations. If this is an integer instead of a boolean, + validation will only permit the given number of entries. +- ``CONFLICTS_WITH``: Mark a list of components as conflicting with this component. If the user has one of them in + their config, a validation error will be generated. - ``ESP_PLATFORMS``: Provide a list of allowed ESP types this component works with. -- ``CODEOWNERS``: GitHub usernames or team names of people that are responsible for this component. - You should add at least your GitHub username here, as well as anyone who helped you to write code - that is being included. +- ``CODEOWNERS``: GitHub usernames or team names of people that are responsible for this component. You should add at + least your GitHub username here, as well as anyone who helped you to write code that is being included. + +.. _codebase_standards: Codebase Standards ------------------ -Standard for the esphome-core codebase: +ESPHome's maintainers work hard to maintain a high standard for its code. We try our best to adhere to these standards: - The C++ code style is based on the `Google C++ Style Guide `__ with a few modifications. - - function, method and variable names are ``lower_snake_case`` - - class/struct/enum names should be ``UpperCamelCase`` - - constants should be ``UPPER_SNAKE_CASE`` - - fields should be ``protected`` and ``lower_snake_case_with_trailing_underscore_`` (DO NOT use private) + - Function, method and variable names are ``lower_snake_case`` + - Class/struct/enum names should be ``UpperCamelCase`` + - Constants should be ``UPPER_SNAKE_CASE`` + - Fields should be ``protected`` and ``lower_snake_case_with_trailing_underscore_`` (DO NOT use ``private``) - It's preferred to use long variable/function names over short and non-descriptive ones. - - All uses of class members and member functions should be prefixed with - ``this->`` to distinguish them from global functions in code review. + - All uses of class members and member functions should be prefixed with ``this->`` to distinguish them from global + functions/variables. - Use two spaces, not tabs. - - Using ``#define`` s is discouraged and should be replaced with constants. + - Using ``#define`` is discouraged and should be replaced with constants or enums (if appropriate). - Use ``using type_t = int;`` instead of ``typedef int type_t;`` + - Wrap lines in all files at no more than 120 characters. This makes reviewing PRs faster and easier. Exceptions + should be made only for lines where wrapping them would result in a syntax issue. -- New components should dump their configuration using ``ESP_LOGCONFIG`` - at startup in ``dump_config()`` +- Components should dump their configuration using ``ESP_LOGCONFIG`` at startup in ``dump_config()``. - ESPHome uses a unified formatting tool for all source files (but this tool can be difficult to install). When creating a new PR in GitHub, see the Github Actions output to see what formatting needs to be changed and what potential problems are detected. - -- The number of external libraries should be kept to a minimum. If the component you're developing has a simple - communication interface, please consider implementing the library natively in ESPHome. - - - This depends on the communication interface of course - if the library is directly working - with pins or doesn't do any I/O itself, it's ok. However if it's something like I²C, then ESPHome's - own communication abstractions should be used. Especially if the library accesses a global variable/state - like ``Wire`` there's a problem because then the component may not modular (i.e. not possible - to create two instances of a component on one ESP) - -- Components **must** use the provided abstractions like ``sensor``, ``switch`` etc. - Components should specifically **not** directly access other components like for example - publish to MQTT topics. - -- Implementations for new devices should contain reference links for the datasheet and other sample - implementations. +- Use of external libraries should be kept to a minimum: + + - If the component you're developing has a simple communication interface, please consider implementing the library + natively in ESPHome. + - Libraries which directly manipulate pins or don't do any I/O generally do not cause problems. + - Libraries which use hardware interfaces (I²C, for example), should be configured/wrapped to use ESPHome's own + communication abstractions. + - If the library accesses a global variable/state (``Wire`` is a good example) then there's likely a problem because + the component may not be modular. Put another way, this approach may mean that it's not possible to create multiple + instances of the component for use wihtin ESPHome. + +- Components **must** use the provided abstractions like ``sensor``, ``switch``, etc. Components specifically should + **not** directly access other components -- for example, to publish to MQTT topics. +- Implementations for new devices should contain reference links for the datasheet and other sample implementations. - Please test your changes :) .. note:: @@ -694,30 +795,6 @@ Standard for the esphome-core codebase: # Run lint only over changed files from powershell docker run --rm -v "$($current_dir):/esphome" -it ghcr.io/esphome/esphome-lint script/quicklint - -PRs are being drafted when changes are needed ---------------------------------------------- - -If there have been changes requested to your PR, our bot will automatically mark your PR as a draft. -This means that the PR is not ready to be merged or further reviewed for the moment. - -Draft PRs tell other reviewers that look at the list of all PRs that this PR is currently in progress and doesn't require their attention yet. - -Once you have made the requested changes, you can mark the PR as ready for review again by clicking the "Ready for review button": - -.. figure:: images/pr-draft-ready.png - :align: center - :width: 100.0% - :alt: The ready for review button in the bottom of a PR in draft mode - -Before you click the "Ready for review" button, ensure you have addressed all requested changes, -there are no merge conflicts, and that all our CI jobs and checks are passing successfully. - -Once you've clicked the "Ready for review" button, the PR will return to a normal state again, -and our bot will automatically notify the reviewers who requested the changes that the PR is ready to go! - - - See Also -------- From 904959c7e0763970474537a7df869d3d7338e45f Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Mon, 26 Aug 2024 17:53:08 -0500 Subject: [PATCH 2/7] Update guides/contributing.rst Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com> --- guides/contributing.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/guides/contributing.rst b/guides/contributing.rst index 6e6a0c49a1..7d07605170 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -578,7 +578,7 @@ Consider a YAML configuration file containing the following: In both cases, ESPHome will automatically look for corresponding entries in the "components" folder and find the directory with the given name. In this example, the first entry causes ESPHome to look for the ``esphome/components/hello1/__init__.py`` file and the second entry tells ESPHome to look for -``esphome/components/hello2/sensor.py``. +``esphome/components/hello2/sensor.py`` or ``esphome/components/hello2/sensor/__init__.py``. Let's leave what's written in those files for :ref:`the next section `, but for now you should also know that, whenever a component is loaded, all the C++ source files in the folder of the component are automatically From 510cd0db90a7b2e2fce78c4e2d5a816bd4b0f763 Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Mon, 26 Aug 2024 17:53:38 -0500 Subject: [PATCH 3/7] Remove reference to `ESP_PLATFORMS` Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com> --- guides/contributing.rst | 1 - 1 file changed, 1 deletion(-) diff --git a/guides/contributing.rst b/guides/contributing.rst index 7d07605170..38f472a3a3 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -725,7 +725,6 @@ All Python modules have some magic symbols that will automatically be loaded by validation will only permit the given number of entries. - ``CONFLICTS_WITH``: Mark a list of components as conflicting with this component. If the user has one of them in their config, a validation error will be generated. -- ``ESP_PLATFORMS``: Provide a list of allowed ESP types this component works with. - ``CODEOWNERS``: GitHub usernames or team names of people that are responsible for this component. You should add at least your GitHub username here, as well as anyone who helped you to write code that is being included. From 33d48387bcba798c6ea072ee5db0a5ebed32f61d Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Tue, 27 Aug 2024 00:37:05 +0000 Subject: [PATCH 4/7] Update examples, add more `git` tips --- guides/contributing.rst | 96 ++++++++++++++++++++++++++------ guides/images/update_branch.png | Bin 0 -> 57057 bytes 2 files changed, 80 insertions(+), 16 deletions(-) create mode 100644 guides/images/update_branch.png diff --git a/guides/contributing.rst b/guides/contributing.rst index 38f472a3a3..5162b7eb27 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -506,18 +506,68 @@ Before you click the "Ready for review" button, ensure that: Once you've clicked the "Ready for review" button, the PR will return to a normal state again and our bot will automatically notify the reviewers who requested the changes that the PR is ready to go! -Catching Up with Reality -************************ +Updating Your Branches +********************** -Sometimes other commits have been made to the same files you edited. Your changes need to be re-applied on top of the -latest changes with a "rebase". More info `here `__. +Sometimes you'll want (or need) to bring changes that were made in ESPHome's ``dev`` branch back into your (local copy +of a) branch. + +The examples that follow in this section assume that you have: + +- already used ``git remote`` to add ``upstream`` as shown earlier, and +- your feature branch (the branch from which you created your PR) currently checked out + +.. _feature_branches: + +Feature Branches +^^^^^^^^^^^^^^^^ + +There are a couple of ways you can update your (local) feature branch. The easiest is by clicking the "Update branch" +button in GitHub: + +.. image:: images/update_branch.png + :align: center + :width: 80.0% + :class: light-invert + +...then run ``git pull`` to pull these changes back down from GitHub. + +If you prefer to do it the command-line/terminal way, you can do this, instead: .. code-block:: bash - # Fetch the latest upstream changes and apply them + # Fetch the latest upstream changes + git fetch upstream dev + # Merge in the changes we fetched above + git merge upstream/dev + +Your Local Copy of ``dev`` +^^^^^^^^^^^^^^^^^^^^^^^^^^ + +As you create new branches for your work, you'll want to be sure they include all of the latest changes from ESPHome's +``dev`` branch -- it's not a good practice to create a new feature branch from an outdated ``dev`` branch. + +For this reason, you'll periodically want to update your local ``dev`` branch. A more detailed explanation can be found +`here `__, but here's the TL;DR: + +.. code-block:: bash + + # Fetch the latest upstream changes git fetch upstream dev git rebase upstream/dev +Note that you can use this procedure for other branches, too, such as ``next`` or ``current`` from ``esphome-docs``. + +.. warning:: + + Using ``git rebase`` will result in your changes having to be *force-pushed* back up to GitHub. + + **Do not force-push** your branch once your PR is being reviewed; GitHub allows reviewers to mark files a "viewed" + and, when you force-push, this history **is lost**, forcing your reviewer to re-review files they may have already + reviewed! + + If you must update your branch, use a method described in :ref:`feature_branches`, instead. + .. _contributing_to_esphome: Contributing to ESPHome @@ -648,24 +698,24 @@ This method is also automatically loaded and invoked by the ESPHome core. Here's import esphome.codegen as cg - def to_code(config): + async def to_code(config): var = cg.new_Pvariable(config[CONF_ID]) - yield cg.register_component(var) + await cg.register_component(var, config) cg.add(var.set_my_required_key(config[CONF_MY_REQUIRED_KEY])) The details of ESPHome code generation is out-of-scope for this document. However, ESPHome's code generation is 99% syntactic sugar - and (again) it's probably best to study similar components and just copy what they do. -There's one important concept for the ``to_code`` method: coroutines with ``yield``. +There's one important concept for the ``to_code`` method: coroutines with ``await``. The problem that necessitates coroutines is this: in ESPHome, components can declare (via ``cg.Pvariable``) and access variables (``cg.get_variable()``) -- but sometimes, when one part of the codebase requests a variable, it has not been declared yet because the code for the component creating the variable has not yet run. -To allow for ID references, ESPHome uses so-called ``coroutines``. When you see a ``yield`` statement in a ``to_code`` +To allow for ID references, ESPHome uses so-called ``coroutines``. When you see an ``await`` statement in a ``to_code`` method, ESPHome will call the provided method and, if that method needs to wait for a variable to be declared first, -``yield`` will wait until that variable has been declared. After that, ``yield`` returns and the method will execute on +``await`` will wait until that variable has been declared. After that, ``await`` returns and the method will execute on the next line. Next, there's a special method - ``cg.add`` - that you will often use. ``cg.add()`` performs a very simple task: Any @@ -697,14 +747,28 @@ The next method that will be called with your component is ``loop()`` (or ``upda :apiclass:`PollingComponent`). These methods should retrieve the latest data from your component and publish them with the provided methods. -.. note:: +Finally, your component must have a ``dump_config`` method that prints the complete user configuration. - **Code in** ``loop()``, ``update()`` **and** ``setup()`` **must not block**. Specifically, methods like ``delay()`` - should be avoided and **delays longer than 10 ms are not permitted**. The reason for this is that ESPHome uses a - single-threaded loop for all components -- if your component blocks, it will delay the whole loop, negatively - impacting other components. This can result in a variety of problems such as network connections being lost. +A Note About Delays in Code +*************************** -Finally, your component must have a ``dump_config`` method that prints the complete user configuration. +**Code in** ``loop()``, ``update()`` **and** ``setup()`` **must not block**. + +Methods like ``delay()`` should be avoided and **delays longer than 10 ms are not permitted**. Because ESPHome uses a +single-threaded loop for all components, if your component blocks, it will delay the whole loop, negatively impacting +other components. This can result in a variety of problems such as network connections being lost. + +If your code **must** wait for something to happen (for example, your sensor requires hundreds of milliseconds to +initialize and/or take a reading), then you'll need to implement a state machine to facilitate this. For example, your +code can send the "take reading" command, return, and, when the next iteration of ``loop()`` or ``update()`` is called, +it then attempts to read back the measurement from the sensor. + +``loop()`` is called every 16 ms (assuming no other components delay this, which may happen from time to time) and +``update()`` is called at an interval defined in the user configuration for the component, but only for +:apiclass:`PollingComponent`. + +For any :apiclass:`Component` (which is nearly everything), the well-known ``set_timeout`` method is also available; +this can be a handy alternative to implemeting a state machine. Extras ****** diff --git a/guides/images/update_branch.png b/guides/images/update_branch.png new file mode 100644 index 0000000000000000000000000000000000000000..b5cc406acb85b3439ffbe2a473d139a08db1aeaf GIT binary patch literal 57057 zcmeFYWnA3J(lCkzClCn1f?FWC1b26LcX#*T1h?QC2<|qx26qoOI0W|@oIBaw^PK%X z_dR#d$NS+O{=@WiSG81CS9i&LQIwZJL&isjf`USmk`z^jf`TuEl$R0VA^*ESkk>;& zp$b@uh$u>lh>$2cJD6M9nn6KHe)+0_ps6zaE=Nb1v;hi6S=b@S6-E*%|5dy`904g3 zhIDjbp0NZDwr-ndcUV>A2Y12a2xFr+$9}bWc@jPzhrTF z>;ZY8N^-X%`&&xjp}fAqNa@`YWM)YJcSoD zJyAnQUl4eBv6j5;|}c}`wr2VWSxT4scZ)oaSkrFyqa zUBbNDzOvE?hpCaCVt?P z#0L&q<>6urL=jGmKI|ZQ?01fsBn!cw`6L3I`d0N?EHv^5D3zn}HCocoyc%TSk5@N> z8A(IRhg}nxj>-3F+DhueEI(8u=#fLElrYMLj?;3ni1Qop)ROyS@f#8(w@oi}1m9C* zgu>Ha*Y=`HNoLJ-vM~lYcK+O>Bc+!9g8tT&GKA(bnEk5x@^(|V#yv3ND`yG$5HS`P zCr&5H2Tf*=3LO&ZXpKW(3>>%tnt?4aT?$I;@HegreZCVWN^_f@ISp47l{^f?&IxuE9o$2{$quEC zll8?1ZDPBPZ<8iJ_&mZfdrGRAPf3&Rj%H*!59=`egG<7*34`Oul7m=;txiQk+U?e6 zXvglgJmPn!=zMJ=DV5PWDcA!F%K@1^=n5iM5>DUpl8us~BF87m=a1)B!&OaNo|y$A zHD|*w$B7Q_mINNckFSgfF6t?pFnA&w4U210elk}OG1Vk#!Y)J-rf%lX--vd$!gur! zZx07@_XgY&_}10Q?(xiX-f;RUU1K8_jR2=V^GWf_k&TVf8rE#!2z75nc*c_BGRQO7 zLk)E@MlLR9YTkLkPTM_Aj6c9attY|4!b)_xY*Iu2JVOG!B%EcTVfk~bLU9I!P=9?( z%c<*5+6@TdUbVXK#!Z7I;3US`fc6!HLWJKW!Ke8AmI+!s_tjzxoGdhJx2_`0NH{e}SmOE()(^~D30(}T zt#CN<1j+Y_L4~C05r_$7tnprwcy%yZakfHb2~R_BcG#Su4+QPW(G!Sv*e{X#71dM; z*o7;n5jM$!DR z-E!SB-h$uK-=Z-_xL~f$6(u`JI+p4zw8TtR6{IDtBtN0brqCmGOW==J8=^ABaYp3& zN=Js1G&xbOZ3{{&lM)trAxojCd znab}u-+=|4N=i!1Di_LM)ZUbhDQT70DeSh7hP;Q=X?R<55*81*-dr z6iioGLN};tL)#%$X1IreK7)i}s`g3-l(x&q)yoyjRocFRl(WCtmrf|>=IIyeSNSVG z)p=LLi}gtdSN6=OeOJnN65TEm@1oM1)ne6VRdA`@V;D!tSktDU|5#~A-?`vhS*dN& zKs3wT%9Sm*KfONAhS_JrkxZ`p3yk6WADo?D7r#f{cA_Q>_nxK+LN z{hIFD_f`Z~hhP5q=4{dIvP`zqHOodCVI7MERXBOnQy=G7%02YXlK4dgGzBCATKqen z{a{zSCrB0yBYVVy>)|{48p9t@_t6*${S9vkMdqd^m?j)sH`V*p`~1mG3KYp}r@0GQ z#XAOC2cQR*2FjxqaKUL)a*whhROPbr8N3$1m^7nx2eidm3nu5L*v3??Tl*Wvuj|tr zwZ3W9F6q}l?5VmGd4=84UQu1~qhLi%zhkAdraRSCrb|!JO0l8SpQh{r zU20tlu5564aB^fqN7*$pY$A?q7vUE*>_PT0fEatg>cMHj8cs7uOTf+fb-d@C_qC_C zXOWk>H_+6Y)g&z{eZK=yKTyBEBi76B8g$NZ&cE@L8W*e(JIRoHsN+i1~?m2dXP_G91xs-${`9q-id4 zZ;)`XbntReV-QyGB7y)SGs-Y(DbfLo8+ta8H*GY=yEjd|$5ew6ecP%o_q}|57GWF` zCE`CMD&jofTH(-qv3(yC`s6ux0AKFj14if4(duXOTH;$ zD@;^QQU(-*;?RiNylodb!A_Lh_R#<$0k_9cBzP=m>i4-N{0Tf4hWMoPTy`4EIQDFS z$d%j)scPA??OHDzPsUs5N3}3mwxTDw{u!cC4H=W6bn zprT_i*EAQ8GrV(>3wST#YUVOFqCu4F+n&@wQjz>2#!8gTkL;ul>gt?T>M!xOlMvJ7g z#Y4BKGq+C<9h1(TXIt?FJud0F2=5$`;?P5Sq^Lg6OV&kA7&|ogRMquCX3P#3{RY_qdJjk4iP4mHGJGla zy5hQ2+U_PTFPGUL*)-l(dbT@XkF=^qt9>sAR^>F`3%BUp6|v8;&#-6PkS~6!sMiG8 zuBF~+9aU?kJ0R(o)o&TLV|KKKcc9mDDKtUJ`g zN?(zZwC^`b+_vGjIV85%e=kFndZ@jogZ^^*{IK~cuBml2CzW8iW0`*&%s2pC*6E$0FA27Mh^RVkq0X8JCtDfKI@@o|DA$9 zMt{0mY3n@WE{-B`opgS7l7f7ym^ho6*}GUexQaD$IYTOt93{0} zprGDS{(eJCDN~$7LBagAQq^?Tl#}H)aj;`BGIcOEWALb8&GoGO;kSu+T#w=v}<*U5z~H?On+K?Brkl zh?=>WI9oZoS~=L0{O;Gt*ul+}kBsd1K>zdlbDn0NR{tK!-sP`lK^DmPyM>XNfr;_| zgyw2x{vXhOxBQ9r$GZL;j`w$G+=^D7X11E5R(23lL#D>h%)!LO`^Px{ucm*`^iNbZ z7c*xO2RjI+EC0Wl^%wHr8~-=rA7g6$drTH)&c6@&x0b(Q{$2yOvYCs6t=n%Js@Yq) z^0V+V{-3Jy1(Fg z8GrNsZ?f>Gp#4z`ku-i}UdI2CT7G1gDykMJC_yMGQ6W`N=;LgJthZ_l{bv;BVMsfq z`*cDWqGF0nMia>57z^*2s><7rT2C{WP60L(ZRT@o8n1-lu$eGKhqatz`*yC^uFT7d zTU&jKwq?O%Esq(})%qRa>ax=ek3%3q>Jmk{Y*}s9nok<04#2jwh=L;t3Mu}do;foZ zyX7YKHJvu^uj^~k;5G`B#DBv1-Kii3+y}{5&@pZQI06LVR^f9{@=JwgBAn#se|ks| zVcr%7eIC&JXVA}ZkJN%fBLV7kQZN*P|MXzM1^y5OSLp5)$;%#&Qj@2Ih*a|Dtb_RwUOGAo<j0mlv>1yDAgMtAe3`~B1-9Il?5C$46 zj|7{E<@6t^0s+84gho~loHEwJB!T{?C-5_pVmvY~p2t6fLSrI^K)oszw5+B3XEBTy z?EWoJ(a}8rjXc2#e-2V5Y0&&{xJY97n{46(g#Qg6VMqmqXfd?uWd2E*ev^#}LN;jf z?brVeKLRlz{6M5nq5MY*L&%l}A=~E!Cxm}u2Bfb*K|}~YLf(!3H=7nt7DBe}p=Q|s zh94xq`5_!-CiRay_A?v|6@+a6SB!tF#s4eD|B~?kzlbp=ERF8si1{aL3b_onBr3T~ z=>qW>N_sw#nVFeFpYRUX-3V_O%KrXD14WWocX!H)dBoW%$;qOA zZLn5`9ax1KfT6?%yJhL}1v`q5<=-i&i&cj|a&c*mQpkvl7p2&o&_V_5$`l+7uevWg0&-r&#kFLA{%yYk^O5a63^L1E)>g_{NPtOFyWb@k{Y zK)$x`n*C+1hBw0T(vp(#uL*fF6&_v_&Ob=4w<1oHm^M1BPx~UHpqR}Lpw&6`d0&;< zl$k}8H1qriwIRqyJyqglVl~(N+BB<}u5)%4Wka@UeKy@M* zB7n}SXP;1;TrMsyVCZ3UJ)bD3sHDWri^Sb!QuTN|Mo6)Xjn@v_N#?)XcpR)YX1b~h z(JI3SmfHYM_v(ejN^fiyhguuU$=x^?{;rd`a6u;x8rpH$KivL8thR8$=CV1uh+kp zbXvi`=$+VqL8ES#i-b?i6xs1;H9EH(uhw?ZKCg=Z$AiKx1?or1-grF)BjaTBG`6Mx zTl_!hCwK!#BEP%0Ck;DJ9vWXfp249Z)FtTb;#{`QB`q^2B~<2->pxTOVPG{=C{<|~ z4h7F#A*TLOtJe1mok44}`f^Y3f7JDYc5fu*>^{WcTSYJse(dwa>n~a6#^Pmw=p{O= z4;2`fq|!ZaEM~*j=SCYSsM=JE;b?v+AFnukzCKtHpp}P*t2GIGojNuqUT1niHq7Pv zfp)oH{69;7AyT1*xnyEkhHx5xq*6lB_&wyXIAnp2Q2v7S zTi=_THAQ5w7k8`b9DeUAMRK@!m9iSiSN>>$842lHSDrdN@KG-OOE6-o9w=8OzO|ns@#E} zpPwGg4wFkWNH_ok%gej>9K24AeD&Er}Z$G)h9NZm}8-FEPV<|bj!c0^z56VH}SaL)qJi| zU-;<9>W;mpyRupnv(qdt`=jQKOenm2bF9O6b!cBnz|?;VN8l8a>+zfzW)ddlgJDzH z$w#CP;f3Esx`M-Jx3|4_6f|HcdL==^z#z{;z|OL%xLB=+rAukF{1JEULlIx$J%|9q z#rW+6J6yx+s_xyz!;%sA&Ai4{E&#X?AEsBvs44fQx=10Fpt`FwLpjjy)KAlagBjTe zo!>bEqWACtr`^^j2luj{7nRre9tPuZNncc%X}P#sPTdpvT*$j0me3Gz{=Q2NaRVEg zS%xDRlUD0ExirgaQxEe zcbEHwH*RcX6EUx+U<#w&`>^G*YO?$-(!BdbwN%Lng_2f53>}=8`|{(Z4v7qQhrMN= ziQ}^m;w_er0|qedDP`0*;BcZulQ_aX+Gh3j!rQgmi>h7A>=x;jUoz3c&Z|K|L6&yj zLno2*F7K=dpE`02(iPYM^mA8zPk+Bc3X@N9=F5s(-N&V5B>8*kf1w+XN{Z=sQqF@> z(_^-&nZxI;{P<)n@aM9Fs4;w8x zk^(Qo`SNOUzNn1!q5W;RCa;A(^~a%C0cZP`dsyRkD3=Y69Dloi2Qt0semL>Nry41h zn=zF!s;wBIPyXs9Ml1Vu9j)mmbOyEJ@19!Ojt51Er&2R83P0KYBpJ@|UCnH?o*yx}0iCSyEg1EuRO+-> zp}SGSf4z{Blq{Kr9nE*%dY;GHXjJ7Z(EK>v>E|OgIpk;qN;Q!G>ap%M2-)$6M9{!V zx3jD|iNALHAA5cmu8v*SKwhep!e;0@0}a}kMMD#po>gsa~>=>h7cR?+=!5u`_TRwT8R>IEgR>Rqc*?vARgHZzB_nF9y8D; zuzS6#V*xai73L0wZFvq?i;zAEn+^RX$i{JkvPjCeBJYBdZxHM#9Ki=`eR9!tvCx6m zeFOA$u6MLHYMMprx&As`3y2?$N*hu>a^0ty?A*Q~-$^ToOUUW)vC-`1t22D|)v7TJ z^nG4>F$JHl5z#+x=RQ+ZuBe8+c{p8h*zNFGzxl98_EBbn3#@xkc>_9}*J*VxEkNBkh{3o%hn6S=?&!d?#$@f4Sn&HGDWSnXM{qn3)q>yR8!1a)l zkwJ?ta9@J=*wEDRw4k#k6Ycf*g_@GGaR%#hR)Igk%D|rn7U6X=l7@j@8K-ZzU^y+T z-C}3<|&NSKa zsGwI}iAB8|mSJos@d2Nrm7E%$?mK3&UT@cpGNu+>w>4LWS%su($;-#uuC%J{J+UbO zy^rLyI%v<=8`kw)Vo-6Jy8EGs8Uq40pcm`dU8ehSGS^{j@Vq)tr=}|8c{W3^b{|m~ z9V-W(F2dJefWCl-*4)0HK{vr{ql``xfK`jnhF|QX{MG8+T zEqt;^dzg0W!;`-4JI@x$?KaFRaDbnbBnauPtgPy6msqq)CQIvYIFKgu*#6SAB}g!c z-2NqnvX?hWDH}N)(gvi76@{W#c*#msE69AlEcsYXtn-u|J%B!SrOsD$e}>x@UlQ-B z?#ormm8P@g!5n{nG9YJtXAY4QBaOjE&`rGeOE9`?cayUsX24L_&0gl^4;6(w?&J!k z&0mSkCckev_GX>gd=e1D;XpgQr*_4mW4CSEzFh|M)2^%S;xN8lP`7!3k$o5Rj4259jppEywGuz z)3!LakOEWr`KQaqMFU2TRnGznO3I}uud!fojP?Q-C%^kDXN$%w{RqEGg5WMa;hW!2CwsapAZ#+6HPL zlPqm5gTsNr#s9XaQ<{qJGNS$Ra@78)2Fl5&MMT$g_j{^oBC<9H{PXja2+@P5&PDT{ zL+~%m*GJ2Z<-Av`S2J1&J>~VO`kvD}A$2|{Ph~JJ&d$zmZ5byWKG%p*CFrDwr#`1j zIY0&cs*5%?y7ZED1zaY8ZLEKI46z0coQdBDu*-*i)J32uwtrRIbC`8?M?lweEAQob z-+Uyjx&U*(pU<9_MT`0-KOURo=g2*Lg&*d`zSjl^i|c&g#dO-}jaM*jd@NRp$SM16 z>btvp!wJE$Ok|bS4_uo3>=$}(kKp25)%$8X^@rE??Utz3y{axqAo|&R^vK1&3^>lg@xKmkI2{d_g7c$nCWq zo2^GXRDSYY6>(>}4d4UTWI*aSTARK1f0@=Haos;{ESv<*hV- z%YA6Pe>q#>`tt|01B)1HqMu#sxu>oUZ)h_J<41qesL(fGbDn~M>ZofN;uV_SU#?B>Fy@qAtj`%9@#j# zN_<_^w?c(MP)W8hTHT>vi5tr81Ll%_ez>6KMg&sEi+CoVtUWNz5*rx5b(fTsjql>S z&08@nlMuuH$k6;t_)5I0`b6_(ZuW%taTl1*_Q8YmIvM4II#tXTTAqgO*CXZ52WF`}94+(`m)cG|@u{yOl!y;* zl(KO{^_=|AnvdjNVmnmTcFP8d0)*}7gsKmJd{CW%-1fW-UXAvlSHJZvui%CM1e)&Y z96m(ajKd8*)%81Jzq}TBu2%PPZ}FxDUJd&AMJidouX4)L3FtH+a}&@&K+<o48wJIP7~s~$U;e~E->L^r9MlQ0oY0KJHT@kwe5**^PZ zvW@CcYl~at>MEH$!x&wHKAJQIdle+=uNnJSB*8jVa12qIo^4fz2h9(gMnp1mR-Wjf z%zpu}KANAUd)3o!_93`#^V2n&U$j}UM4$i=71s6q;sYANel@A>(uv3F0dsS!`3?Tv z#?vtVrA;h}Y3dpYSqu^1Kqu{nSIF#Du|Uafj?c@76c)nsKw5V}+;$Z- zU5IrZ;`lC_H@wjSW){RO;D{^!+SuTm1?q8Ve=#k^&}SPaGNCYyZ*H1@XsmgSI#AK;FzE*~On)r#|dFud5C-m62KDz|c~ zxd-^Kp-1WYRHu}HiIsQ4MwaCR)k}^{U^kM0d!?jFB1>w2n{w8MgAAMM@+w@wUlBe* zO+>-?5j4U0y4R7^cAjd4NlS_hR_Qd00QuA}mH0!A~6P)IT)#xKwp5~7dKBm7U5z!YLkA0e5HvXf|ra2$anZ$3bmNk!^sz;Ui$^P<}dTys5-@={tbb*9|kUepmALyXi8I*1@M10mU7&45{ zf2Fr^=bF`WecA7^;q%+E0>C4utdILo*GanW z!;~``R}L|}r<~;7oTKlZe!Q_SAF#apl;JaAk2v{?NCKT}rUC7)I_vJ;hLaikf!X>x zKk=zezdoS)u3&C^)E>8_>%9GHHzm^{vF)sNw8sBd2z>62K|Hars`ZT9E4Oy@(8d=z zk;`E)pzE5zazrf>hffx8ca_UGVKykAZF-$1!wW)l>z|`B7}|Wg&zOvh#rT}bVdriq zlI}52z@XZu_^?#z!vU}+5#@7ydAfh{N=D^v?`i>M+x5K02&J122+=RLoL$0h`F4!G zrK{z~EiMuqN-IhFBaOpBG6J|g7@MSWP1OF<`C$6gdIK`Q%|gHNf4!GQYP(H*+Ok+6 zS6#Ze72gxQgNl~QaT^hM86QUKwpH2tnwsN%1<0R#3l1m~CjZa)2 z*Qq^=)3|WZ>v*OyN>CAfe&o7;m1C~_?ox)~>#RH;CAzh|Sja2aWlyw~@GR$)Q1A7; z97kT(xw#J0o0Szd%`o6$FE_uBYGM!gHl(~N5&G$VbXtOlrn>81dv_p`YgOQuulP^j zU%X%;l4-e%s(k4Skuq?3R*<-ar-&^+DYJ02IVEOiU;)j{coYtAu|YLfo<4!<;@(mG zF9%@Y2T}}wja)>x3bl`5p$kVMDoRQ%ms?a1H7>saz~P;y?`m98_bw&TRdlFy@0hw7Tqf?O?6F8uh*4sf%2PPfxI}lpF{1I8>Bj)_6VdF z;)Bm=YgQ!3VMB$+%Z{TUj;*%SW|7FKD85KPR-2Uf4PM2D=MxOCO>Isy&;`=yAnu_} z_u~b-vMvsF8~gmDsuuCrgxsdFZEi>EEpk`3a(uQKI-Jz)a_gFW??4(qJO&81>{-Z{f z~DHzEw<=6Mo;zH#_s(dK+cW?-8VH07~UQ?;y5* z&cewJ0Pcj;-#6Q@X={~uVMC%KQSQ*lWK5l`76IUF8vNLYfoDHsLV+oZu#EQ9icj3d zJuI#Z!DbMdR_-~wfO|S!sq?DWN>5Gwh%RuQ=k*1KtC+6OZCTIdR~5uftldxZuQNX6 z^P58HUTWO$HpxN!_%qc2|FEOcR2%lixsi4Z%!(haSYz}b$qd%z1W)5IqTf#9x{F6LD>#v!;HX&VJ4L{Gc`RV z(K(7>CiGfWB=oemJY28$$Q^rf3LLDhIlbMKkf|?{lhrzAHCvXAbIz|2B%WS8KAsXW z51ols1o_>3)!=m>?7kX^A&|>?Z({;+SHPw(a2w3J*(`e5-uLl=(`tlqO=}asiM;Kh z&Ic`e)wM?O24bx3pE;axHj5|*HW9~vnl3v_Z?@v&&;ZS#p-$(coOJnsisA7 zW??WGs(I3|pX0SaDZT{gxE~p{4|>}qBzmb6uWa-hNS7i6XF1zcX}ea}AHrDqSRm)R zrp}=tAsd!4E)2Ew!yvlXX>`XS|@El)y zRjL?((04JWm;_GD4rM1s7Ab#iGh%@rC3<%~T58bxM*Na;3 zMWTn^R$nJ1TonRw&>ABxvc~-=f;is$lDA$LL)YpO z>QG*&n@^FBsfd$ScC3jLEGeU@`)io;8foBE`DU}sxaHyal|$}*`pWu z>gUtEZ*$dF(%K2P@@X}A0o?2sox~P>x`QrI_in8Sqvs^-Dr7x-t<7Uagojkps#AyThVL5BKWBc{cl__(HXk&q2on@qVYr&7yyXxbR>Q zWYpqk&i4`;DEv-O<5#`)EM56Rkj|h7%V#4@%iN0@vjuFTm6sY5e+7s%~wB`pH&34?2Dkc)xERh zSs2ZXj6mN_KYPfv&Q~4083#SS*9RH|`vTUnRsjW5Y-Lnc10@H11j3NmaUa}>AG6JE zULX+zYrRrl+pSusRy~#Tn36>Eh(G_0O)7ZsO*r`L>yza)^gW|<)H~l1R1L4=DDGaX zghLz76-@#+KQ6bjRo~f|pq`^L)Xk&#m4k5rFvX~kwj;G{d2q3*o%am*M*Z&5S0RC( zW*RVXG+IAYR&DU|e0fovi?5LF7^$TG6ey2{#rvR)quazB?|&0=ahYCM#Q*7iJ~u_e#Ir@mQbw`8{jK2nD4&vW$9LS!?QsAF={L8>M$? zUnk${6|m}tmk|tjYF4{J+O^*HZ;J8;HQn=^#fYdqaCfA`qtZ>1yES+VYJrZbyV4Y6 zUCbwI)$t}%{h#iKH)#!*>~Wk(}A_A{f1}Y zM|n#kFs_v`<}|g@Gr&56mw_lJD<#KF*1JItul|wy_E7l&bn|x%6g9mxQZIS3d|BlU@Ke=S$~?K_fV@DEVmbM*u=@s-z_UA*9M3rw~B-o zAtc}79n!NK?;r+xNE8eJ`yE2A2n39Rc)`c@^X|jLX!)EvHXqmxa$E>`+%8LIwZc0- zGuBdZarNud(OuP>PrTg;D4Q6$_Uk0%@yIX+E(3>Uh%32uN+h(b`&4bFEeP#jm1!at zjX!{WfB?PnA(zyueEoWDbG2#YgyrVkj@p3Ro^anLtQNBgosMr?+I-HTrTD-I9$XO&<=~x&bb{o zxf~q}5vin`OU*9(4D}rq6{-t7&e+~-(}N>bdTn<<$p$`7++FVSkFQy2ah_!==HRt6 zNAEz4#Py7@-l?*_bt}&F2cNOB5ib)1jU2wRLk_^7R^glj4A*z#UJ&Kt%*SoSgGC0e zVTh)mgZNCv>ZJLK)zf@c%<6&OK1{~#j&PY+c(0+P8e?rd~Fa~sbatL}8 zyQkqHbbHQB5c2H!$RQ}Lp)8aXNKEFiSWX!ZkI^vOManJ#{!#u7IeasCW*|Mvj&gOudGs>++`d2OAtsF zLDCP$A8`)zPO`DqYF1o`Gr;1ti_co%*!Fjqox9_owMlIAZ}A`*QmNya999zt4wTe` ztf7$r7iCRN!^;A&8bb^L$3g?K&F}czFKh94%EjUV0!MxLaeds<4ae&mN6k#BbjqKV_~ zd^WGLU20uBdn%{J{nDxK)~>nQ%?)`X!&G@vlnWmE@730WW(Wp5I_ zEc0CX0N4>uXV5uz^xWYMzSN8(-Y~3a36tQiP-vmfX%ui)jcL1-M#J%%(l#r~ZgWk! zVr=fV)1!Fje#StY)v&lG7cl3GcWKdRvk1%zZoc&tdd3R*6lLDmHwKBpSn+!z9;~)R zW%*4kIZyW&eD^}}V;?g|6xoX-X)HHZTx=@$!ab*ygt(&D^ykLT$}l{5KmCe)$KSi; zcKmZ%=OTWA?~|2s_ruw;8iVYT$B({v#w&L@ukyad=ZTx1hNTs#62QX>w>l*0N2^}H zxY14E47i(qWpS9yNajww#dFyG49h@@qffbAg?~0+dYW~K;oD`Y{|WK$7)%VPn&VQ7`~pPJeu`1H89y3-VmlU3cdDSogt>QXU6d)wH# z^E5Kb*rJSyI{ny%V|9;GOvvx}c>}hO-(3iq(UL6JXP-eYt1W$3hM`S2=WbtMR|4Bu zxVG>9Iw8}(vmcRd!du+Hxs>2<7~%*xLIKTYXO3N#s3?5>I1tOq_+teHl@Fv@-z;H3mVG@$u9ov6 z`gD5xM10C@k_X}z&vJz6^-Hgf=e66o zcp~=AnCP+MXeeXL2rjHfd??}OT18>l3LnHQ(A8Zz?m62}?81*>lxtJn20U^L5xNVM zPz^&9TOwTXC@B_hDXBMj#cGlEqvyUUnIky2POZ@ z(O&COwa3S((EhIH!&-*KkKI1iO)K`Ew;db3C@1A(yugtAz7DWw1)w_O`sw&6%c~6p zNhxsknO)r@ts@dJ4mrZIOsi8y`*<1Ld0+vk)rfJSCnJ7;DcggQ@#b5EnP&o&gM(Js zk?(W6*MhO6ih-|i8JjH6qUcJyp9e9|vF7k~!WF8WFp&31`xkIcg#x|sfn0%8Da?jQ zw0QFtu?`W+?`l{i-N#-F0NHKIIIw1-m~D+}BHxJX!qt z`-Z@KDU!CgM9TLDZ}GyYGY$=(H&D?I7RgkqL?pY^hn<5vRouhEu8)#cX{BPBqd!>t zSF)M4i`QRJiEG7h-89UqHsG4kx5_a>A|d6tX4u@?FaAM@+oSO~UNa_f8Z_cY>sWII zN3lF-muhm2M;9F1*L#kir!=Nwn#0NG^yjzL1QQ`OXQA(hbP17(J^f)$$1;LJXDqX)u*YmokwEV#lwQXw82qlnr{5m+WOThBm z{m;dfeNs>3)&6ptQ5(0_S4u^e6d3fswYTu)m7_MGD!Wi)@{$9@oENsL6o?P%>13J< zZNkKPvvPkx80=dGuWeNxLkoVPcK~3(**!RwbtKig!S4dq( zo^4x@xYUll-&MC2s|CD2p_ZGVri=DN?5TkH?PkE^Ff|{qN7h5Ai`U^Cjs5(=SsE|s z7HekaqCfx=rFN6?c)56h9Fwb{n6Ot7=^z`VE5^>py1}>rBcZ^zr8z9oZoZzB3fs4XcaGOBC!rc}Md2BxF>ce716)I(CNEK|Or#9l zoq_etxgNjG0$rTQ^KNwjd+_xFn57-*8MF}$i6IcHnQeCFKJi)17Ry~VgHE6Oh-Hb7 zPhRqO~`5 zgEQ^`toQKx)Cm6#10;!NB;aGO+d3qkod4;Rl4su#Z~5rlf?s#+iA$Zt>~I+(-hTfU z?+wK-xag&pCw||apQhAB?qe*os5F|!k_lD{hkvAeVac~DeZN>Es!fp``C!mEy_}R` zr7@BJetVit;zeEf@Kqb>w-A+*9vR{N?10XWwBy-dJnIK8myfyriN0v->e<=u_joXy zeRcGi9HYg=0d!{_SFv83vpJDlh}cy48qrSlX#xsBuOs^O{B2x!ojT82tvI`dpYnI> z=Ny?euIfu`ZSKR#lLCv|?etOaQo5oe>~Jlbk51dzZH)-ZExTtMB5^D(dXE-a%@LPP z&k&l{zi;98-m6(~Qg>O;#V+B?PenGrrDM$2TFcrlHgx4e{dP zw52cyJB1QBbmHZ_uDT)ce~{yUSTANdI;xgq!?Y1`mafxv@Bg~+ZIWgnE}rozYLXx8 zdWq$Iz+8z+rI@@_-U1TEf;WnFKD2vWkr#^p^eS~u{_y1$iXMIaq|V1$bP(U+&}HFq z$<={HdmB*u!4dy`AmjBQnnR*ySYXCruD3G(G_l5GJW10bQ%ocGo-!C9#xxa5T%l+; z_C0l|-(;Xi*W!k4e^fP?mp z=HHwi;Ky{JXt=alz&j^CJ&Xgcq4z%u*_uY!X&AV~4C=5&tQ^OY{@6k|Q%I3vaOom~ z2qJvTd2duSj=HaJ(EI!xC8XGR)bkPV{S(0I>w4cMXSDsFgLZ3v;B`jGuj^%vl=9i; z;)_7f&Dhsiz(9rg@ukH~;kMM%PX zd9(eyLlM&i*Xj3iz)Pkla6uegjuE@vS=U3=^Bc#&C#@;opTZ@OTot=Zy3KLgmHKa! zRACSPAn@-K29mnXo(pu@YrEv|YYwIUsrba;ggn#?3VH*xueuX)3EwwDY-Z=epmW4r zSw`f;8q|B` z3hz};w^HU6b^0(%EH~s@A>!E@#n7qAP=iF~$7Tix$NCua|Iqc8L2<5Kn|6XbjZ1>N z6Wrb12@b)6yIXLAyGw9)8iEIRcN%wh*Xca_*)w0w_fA!QcB<&U*R}2?=XqRVs8`R~ z?pH$^SY~pDPYq0ZBdNl(sm9R+N7|ukkbZ6ZsJ3xF{UtaH%~)IbQ2DGky9+4>-d#Y& zU#xb=vWa({ipxMNDpY}&5}{xIU3Bo)3~@s`uj@x6#Y}wfnRIfw>Xcv2=HJF?y04ZD z#`|QC{roPBewMq#FX5S5fH$Bk`q3c=cX^F7EO0;W1ur;L7Cz~p9T!c}XEHJ9>}HXW zT*hy0Kjj(9SeDUj(e>s5{d_V-$lH|8qYO_1ht7`|4Q`^mU}jR57mFQlS~;T6_P?MV z!5VCB*1Y+6$qt~L%BhPV`7paY(O1tgzL@?*`mFR8==%;W{{%x zR^kpSZgDxF8_W8@rw|)4$mmJji`yoanN51fFE~BvkilV7SB>CFNQ%40d~a%39=2f@ zFKy?tv$vN%se#MffCRY?34_8$WThvscxe~_o9p%FUJc5<1b{yalF<9ZE&9;AgcU6C-m7dLLCQ4!Vn#ToL;02oeCx83T^T>+DxKH$BZt z4!wr9HZozlC!6j5&k4cF;fFB&bM=qrz$M|s3o(9gl5_})ooTz=ym z&SVND591^`zxq>g;h@GXOj;eMrof`6h8;AGdja3$!#3j=gnfu9uDA2UV!7T}iKtH* zGc%=5RII?a`gaDP4~QeC7M8D`Kld@7ScZaKs?OCZW6ftFtIT4vBC(iENCu*HOsRJJ z1(DJ5Cs`Jc3rj#^yPV6lB#}>~P8|4@t;k1q_usGi3u_}n6&zW0s zgP%B$*K?TKAKF$5nCx}t!L4>q047xTezs3PR&f6^?2J8&PNQ>a2KbW>C*l#lu7wug zr$|R4ZMM^es71U=ZpL^*u9oSTj5$*_KX(*-Tgy3rj0SnQn7b|$WNH4mYi(tNaR6bYK!W=BcX|%$n)#03G zuJ{z<@_$@I|N}+frz)rImn+l z9#mHnyCAn0>8b#Hi4VpXHD1=!(vOt~%RKj+Lb}|3JboxE7p( z_v^`_Y~x=rx2_2 zVLHb9_h*%A{;+Yo&Ra8ey(&jg=8_X%^K^uu4FqQOmiuM?Z|gTlQlk_-+yzBXag-z5m;g4APAL`y8;Q6e`GdV z%=FV5JO#Ud%!MTSN{5`bns2U17V??bIu&VpkU6DWqcNc6lxXL( zWDdP(V(u&gDuGp2=ZsdpnJ2?jd&k3f)KyrAv6R=cINQvvOm&XTQST_!4QPdti-0OD z5rXdcRXW>_5C!KDuK`~`S=C%;MpZWRnsJTSv!J&ViLETjj*e$40R4jD*u-@Yly_mh$FODn%!CZ+0iwL6^Mf{Ls%N zrRhqjF59?*qlmsWI6jpYqFWx%pMEztCpk!zTOz_(orHLLYenaeuY4ZpK$)7-;uNXW zbMJa|%ts)wx9dkmX>&da5gx-DNf}{%e@P*Blm4~GN4f02K;VXJ7pFtHcC@)v=P_gE zA>QM$i43sIwrL*j$>u{dx-Vlv8jZs9seRdfdxo~lCT7bvb#Df{vzT@)#zU8*-#!7`g_@3uTmqJ!ZjT#A-L4u! zLXjw&kXLpFN{*S@9%6qrIHimej`cypuI>440~cmR&&*>^=flCP9R0pCu7u-KoqNK& zee5V{AEoT+V~7Qk=!E#P>IL<$(c6T5mVCeT3m=OBFxPnzk|RR5-4Uv_9YUQLwRT4S z_f~yntg$pOm}tumE9#nQD^^mrM-5T5H|@own>z20F$=JEiISt%x)}}*NnX4YR=dd? z`}Wz@op;(!=lkQAPeL!RnazaFWq{#{1tR#l)d{Ze z#NL?;^5mTndAT;#$2yLHUjtE{G8oi3TThUKzI7KmA)My|seDVPsJvcy2G?tEUnK-; zSQUBI&pP%Axi1S6#@teXg1xm&9SU`eL@itMJ@pn1%2`S2$lgNY^g zJBPP%uAI34EzcG5A&V{W3l*|<^k6?!5^JGV!o!+r2;$O<#*mwFp5)eh@2K0WU74gm zR@?uy_+IkUAr|^+yP2m+$c$hxp&BJ|D7+i$@*FsUd;U9SPxBU?4vJ!g7-`dvtpMX- zwmAW~EXQfPve9U5@MnFT-|M!n+A(8*YY7@hkG$QB2-`jpwJF@2{;NlE#n@oNX-pUE zD9B06<=J-S>(Hn^wlR;pLetwg&?|Lm#AxjiBGbw1cdE^5OP>}{YkcVPt=ncB!H(Oi zSpE~IR=u#i55aq^=jl`#m0^^z#y*Ysmom@fx8$%9>-fSRZ*?6_f)Bx!dhKOL0cgvf zD`6Py3T6|rmWtCWeD`ZE^=!6@7DtcXM+>x4KKu3KRav~lOXk_%8vUM@*tA;L?qj;8 z$wEi)in9%eyRD9L2i{7DUF`*XtTd_`sS3)0v zaShm7TGJI4tk;wy(F7p*H*$l(lFjduU~41=E1Bx>4G|$xlSY+^+5MnGueHB84CkcQK|tqNN9NMuDKxAj zDS$cA9Yittv0Bch{%ol}W#cqmFMR|Bwc@TR5W|5?9cC*QGz8M&_Mbnjy(17lE$o~CX_Smj``5a#KeK4Sroxj>lvQ;@ zh1x}EvZ_f%S;yb1%H6D_(u+ndME&x%kwJH~eU4zbr3GwdLO?zH$Nmf;gmBQ7uY=Fk z#FLm{WGgJ+FzKa)lcyyuI9@lD_+AD`rR}49&5&83IH-5?sU^VAJ0A!HFI)UcA89!C z?~**vTiLU5%W!Z`5$it zbwT=mx9ZoVwN}A4W;^gygTRIL4ejfL9@I3Pfy`qn3n{C%*~76|r-L>`+8#Ty_*e>e zV(Vx21wFgBwd=-er`30XN3k`}Yr(3yL6Tr^(d724Y1yq&lG_zTCYgjH`ASk>MX>sY zgY`~%73J(TwwlV8_vIr6=X=e$a!F)b(v%Hw$TUe}89Shj=6mw5?jJ@%Q^ zxkRq*)YVGa*KgpICuB$OzzM6&ACJ!4l+BxIu~kQ1q4YauK7BN>aN+Ao#rdPqG8zV< z3M?vN%@gj8^TQ+wxGp7m=*YSjO>z;AJepL{qn=}qx$x;gazQ5YEbw$NO*P*5YT(DZ zpm|BV5Sm}DPWUU@y7vR0SV=k?oPHd3`^jsU(5uGfTXJx_uKa{(jWC3Q7y5X*t?2I< zSY_lKsdVlfPX$q6Ab#T)TvQ>jJ_vW`2LDo>Bk`82e;A8ylq%8F($XM|9WZ(U9~yk^ zi?WlMOWzH(C<+~vFF2(u`8V>2z>x>1$o{$-CD>@c4r{g9V%H5;^61>FlpsNx5;5-i z6E?M6xm+&gw+J}>WAqpL4z=ZSA97(SQ!+Yd;=v~TJwQ^`Gk6lps8rW|x)ebcnT{`O zcz>HhYav%4T|#ZDGa-2&b8rFn%Cx_WMN21$6==0tt(&8_b%;bGp3P>uPFncPl&bI7BX6r5wBZxX>y?Nl#I$JU)6np%-+79l_0<8dJs0fsJYyVgeUtuJdyq!XBW` zqEruFj(HM-&{gfGscS;-8U`=*OjQ3LOp1RWsmOtu+6g{Jz^pHFA9i@5C2kppB$AW* zL(iN@r4d!|Q$?mkGy^KJ1_Wn%^^G2#n~wp~Vn-XgO?M4p_$jQL8syIBUPkC*sZq6ipD; zO~?Xm%a!)SHV334_TJq%?Yf}x^Olx|D z$qy&B<`=WVa9r-^e_~?}E9xiW-hG~aN}3aY&|}|pqxEKUun}FnE>od#KA-9d`j+ts zeEzzjoKDvEz8TtjBwR(~zTPuQ7U79Yex=Q7~Z@Xr@aiwc=nGaAc z(ay^Yo(Uo3ooy}~-c`_til>mp9@&J1O?{0O*#OU==lQmXJZA%$9ag;(i}jWYHa9c* z_*^e)A-JpKa$L{X+HEPSI`*`iOoK>JXGy#0V>yJxL`%BZ#PTfySMT2Hk?I;skrT0k$ zNP-9piv|twC%PDZQ%&96Lr#{z z7X3P-9=ibDZoOI7zjn{$ZOLfoK%}l;JQ)|Q<6Es{n-w z+j8e6^aA^rpG|9w;KDQfv>u|`~jD5Mc?!LqfEAJnm}&lj=n=%eG+T|#881aGZ9RkWkw&} zWWZKXic;an)bHv^C#p;ZB3LK*@lmPjV1?4_)Ldw$=u`k1GgW`DY~dy^#n@C{vx~1!O8&o67{|wl zZnjx~Cj*8|p_fYO=QtJ80e1R0)@fN2=X~_E$p$rB3w>HQv!o#chV9*Xjh~LP$0p6C zU^mxdS+4FnkDg$fqfLgyRqK4zPF;nBqwTdEce8X0(wH@MXmPPj)2xX_CFEY(k-Es2 zvc`$uQgP8%;Q4awy{qX6D+qCSI}n~$qHlE42>9Xte4QzV|0zu9$K?Xp15O{Km|kEU z53PQ7b=@7TZacx9n19P>*?qHw*qw0rbAS?0c(7wlz(8A@w)3TJuw$4#Cm~@QO#>-F zc|?Oj?;GZmQj!thbp(^ZJvBsJIhY)f>?7z_)pq%0HJ_K3htv(v(^qo8<{_yh@Kxvc zuOplnoD!U6*aRwAbsw%&Xct3}f)t45FfP<|#O%J-Ff3i+3j+O``>8r;EtA53krKd4 zr~)qP6UQ<*q`{&TNYW0EdQSoZ_Q4aUGp&56oOBi^YeCeIj@+1kb$kC8+$D_sW1Clw zhB_So_86vScJ+`ut9r;79cr%T*<2976WJL&%!F(fkFPUdl_FODwIxzj6xEflY3k(y|A6e8KPkYUh%eo5I_3ir_buqivAq{vs%W7InB;ajqt!%=M zg#~|oqIg-xcDZY&+Zv6GL$H*2cfGn3?6Pli+qI`R~KAZp-v=;ff zr?Y+i)`79Nr=079t#+Lh31HVF00wnWURSg#GmF>N5?s7Hp8zv3cW|V@%Pl&SoF(OV zQKIWBM|qUeI|81sX5D#1s{WkE}_ra>CSN) zdlHV1@mS@k1?g~7fJ9=14P93oSM!SH*3_df{L!nm5G42Z0w*vr#jx*hKXjd^`W#Pc zj2Kbb6NRsgqCo$b|%QBSjVD;ImX)in#Y=dHYzfP9>w!6hto0iaXu(tcc8UIy`xqedSif(6F z`t7Z=R+B5xz*H;jf5s)Yp$Wn9?^x9BThUI;sHIiIbEnCTQ(`4nyvUhZCbMfMgew6u z(CJ_r+X+ICSKt#5e%NKBknWfKNwg~3%~OpIS8Q%Mi$zH^CIq1-g=zJ(=?s*(Bt_aL zlbf@R*jG2eBooJibu@`-&(}NUfu?n3%)$*?MqjM>(6$uV@Evc7O|-#!??QjiBcsPy zPnOMKhaFEx1j0Lk1`Cizw)YcJmHPWrr{Cboc?atVUiyRpKYtG#51b2*R0;b|gXMHV zsgQGkq^5#`q0urJPDU+HinJ1U3UXO{u?8r|HPvukoy= zmV6rm5s~h^TQ8Dj_eaZ^zS_oH{9`YHSl8ajP|S_@M&HVQm{Udq9@i@uM!tDbz|+EY z^~jQWCMkc5OLAnSa}3L-Zd{g+lt!f%Q;x5|nB5n!>vLUhy?Y5|Ui^PzVkx=75!-8* zg-ix6nm|zzx(QqII;LA;RV7&`hm6`Jze>wu=2rt*b; zRM<@{b$Y?wz7hZF7L4qPoCHL)zct(&CUOZ<_r)egBmtkh(gplHF|j%Y>7!~j#lyo@ zTkX+er(MnlhIAk`oDoAjHi}tpBkE|A?HsiM8+r*W)Hj*ffL7ABuw~%6;p{iRi3wq> zpKIRtWPT9~-?_^b6`IEIZJ<#9M<$q(6%;eqHhYkM@S@t;*bZ5czPJ>>JA2_FY;!6f zMU3{bwi*%c$A%3OleSd)YDyQ)GR$mZuF^@9pkhZRs_j*ipYlAxlR=8v2=ax(@e7ub z18N>jKxBf!Ks5kC)j{HGz;6?Q7NlC5B$68l z9-qD{faisEUH7imW5x=&yWt^F5%1rfGl430%!MaZ2&Uz*FiZatP4wLe<4Cc+5&XHauVfJL+Gna=6eayvm1q&p8B(xVhB4E-?u+-9r4Bi{yVZ!ThyJJ}iA8xrr?w+OaCTFmpuAEk2&6?cVd) zvj*o5s3{t)5iy035-dJ6Uh3?xdL%ah#)&I?mh0pT@Y7Svu9g-((E66JyQH=`mzY1Z zh3{K+3k?&1`!v0x=?_>SqNj;7K;0eDJi@vi%jH_q%{1hH4E^f#y3$SzX1%fV>Feq~ zW@Th-8=yQsNk_C({slM5KzyN*I$i9L*Pu6`1?wmEmlAL@!HgiBf9s5(3?cB)paAoy zqFkv4pW{zi#!YD*)p}1=;A&_rcJWW^ZH)L^)O{8jhVhhX($^9$@&5xs^S8&4nl{gE zK&Tzsu~spn5go{yp%bH`3(`03pu9D6bm!=rfzFmkKAHHsY5Sd^9F;o}(1`Mw4v?r; zu@+h6s|Yk#edKFPt2sf>P@5D!qXhp?UpSH*OGvt9Fp>eZLDNXmsno%)uUuI7n^^x( z2sXYyznAOCgM+bXd63uN1-TGEIT+@IIX-h$7(u~W9|wj~2~sM8rYSnp1$s@km%jhw z6)K^Ojh$6O$7j1aLdKdyzOQ!xOroQ!s%KgfQv^4wsPs0Wy^0Nd`>%;v=^Wk=S70e< zGEfsktcWFa5^4&bpekU1CZGsSnNTj>SERO@EmS*nZdF!78k9}JvD9-w zB8dS--Q+D#JNd^7P6S!3_#{3Y;7MDAnLNku$8v*#)NSz`E^8v}nk~0(Kwa;&Wg_{L zw(Q^1qeL9y3!RkzQhZ-T0hszR-7MRrEr$D$yHKrTyiE9zy%PA}@Ghf_og|j;D`~K7 z&nv5<;Arf&q9x(hYj56COCA0hB~hRBziCZqWIuB6X2UN$D8YF`@^5%CGHZ!oKoy1S z%43n-(&r8#QY?uH^bBVCJk5mlg&4C-XVlzcLcV6K-0NY{BXv&> zVI}`i)yIaF^21-A>|L$!MmGS;s*3Dz`L_?D)Ho=prCH*AS2M!y;#foX&6jmeiTcWK z7v^~X1WP&zU?u<)a5|S-8G${{+RV7z6l1Fl7N;W@ReopvH^FpEf7m=mk5}o@GL@SP z?np_(W-V5e%VOvFmmfSlqfxXddO)ETH0iW!1rGPIeG`!b@W4wmVYU8mhZ|nCP?6Ak zMD-CIXunN7SUoV(!IcGav3U?>&}4o0KDWpL))%U{oLL{7!0)hYUD_SlBUsJQS3pnk zu|dWJ_9Lvq;)s9W3(Dfed(bqKD?qDa`qcY;Oe}2al&UGb4!hN5_N!wi#Hb(G>Y5)u ziT?AOQo;376-w!PXFX`49(n&dH&oJA!JU0u`T1`RMEnA^kuo(Y3!XZ0)|QGGPS<$s z{!L9#xuq)ZmcKwW?2m5I#1@!B-&wNdu#9eI>r45ZvlZlV+a2TJn=|NZOZgO4#XBgm zB#C2pj*2@%Ca+^$A7r$^N7aCMfoIrm?y}|_b!VsnvX4cONq<7md1sM!&Wf4vzRHs# zx%9)os6~GsR!8XSd=J4fu+vTVaHh#mEo-Sz)|Z_-AGe69OabwqZ{!m4p)G$4BmP*< zFW?jo_Xo6isUf_h;D2957db+Z1YagfF06-@rf5XxsJu_UsWww3gv&@+)8eG#_iKj9 zflq>p%O}x;#|Z+Gn=3V?GRzFVV9Fq$z`;aSG!dpx+4BAQf4?G@Nc9hrn-^FEND4&t zYA`8z0tn+9lGNL*AD_TvaLN>$mX7yLR5?JOWXia3xx-PIRv((?$x3`8%}5p;@!!%u zi234*y?<@sA(Q$j0F^`3nZp2{`^M_GKg)nQvX-}*C!i)paJD8LNnxy-bG_IAihe-V zqsr!!A?iRk>9IS$(f+Ezi(E^A)JOYD%|>BGLZ(A^sqJ zkB%ZP?g9~vzdtyCx{QcauZLYM+`GuSMp@lV{|jlFOODwM1B{YCxjPGVzDgrFK~`X^dgj>tk}X1;Z2u1;ViW|CUMhCU@8$SjwW7&1#^U zjvx=n-sn>9@w1e~lYh?$yibvbYe%RprOhp6F;55?p9xJLX7+S&6zbXEEj>_@xq9n~ zsg3Wim#haX7e$1P%yfru} zjSD#x_PjfxpefyjKmWpYTumG-ylONM6KjH%ZwP9%{xb9QW}q>m;rJ4UffVwX-edjR z0=&hRW~&S9=a>T_7KHHnz6Iy&eN)rd&#usFmzl57QtzoOS?cMd2zuY=6YWdSuxnX% zHo4k~F}+Q9VF#5G$LygZ_EtA_N~?Xv>}U#?$ab$|vs)8!WDmO7&++|i!V~~vh+p@L zGr`uiYsUGH^Xo^}hoD>-N^im-3gjGf0epdKk?-BJ?kwQ?OTs~@gK~Qc9IiW-Wf#^W zaP7V6zU4D(X}B3oQ~L-(6C=RjDb_W){&MN|j7T{3xN2!=N_UP&gj~%*fae$2yPkF3 zAdTMJow4v+ZJitg>h+TaU1eBVQw!lX;*)P|pWX;X7X{!{;GZ~UrKMlPWB_CPK{nv3 zD$EAL9gQ&V2Ox+2d7~@!MD9VdCJzZRh$Kk%_*Vd?02sRDjI0(?UWz-r+~5#st@Z^W zCm!8rI8tPUiP|1s1dP94 z<@*I-3H)}c(5~gPt~{(_@@udGDS#2fmb$UwH~0DlslmtsbpN4+^2i7gq>!uk70Jnd zulp&t`G zg7~r-ivs&qUmkWYpGHS8^hQ;#y67mkJ_wDD%afSEkd_uT`c+q-5SoUUX#77F;x8x_mDwe+b@3m#SW<9rLNORaS1$Uo5PWcA z#|xETa`y1+s=#8E=`xp3L&KJ(BC;|{-kYv%PWvcJ9X91)ES|XQ@ed7pJQ)0#&yEdp z8BgNCF#*THFLTL8|0(^$U?3i%o2>h;3U5Utr%_l9qh035oV0dDu$(S_^@r{Weq_1M z!IJ&USN_onbph+~_W);H)OxY0hsy(!PqtM0_c+jYm)j zMw8VDgluA0gk)r83xo`ki~bqELn(YHb&kjXo(?g`6I(^CQK=87a+d(P376BCPDfg= zOa>-Hr(I%N&X&~n1IJ<;8rXcS$dI3U}+$0=?^mp)phqVIkd{4mL2N4h7`h{@e28ue4 zUA*XA@F%y{JA%K=We&S#>gT&=FMDAj>!q?JZnx07dik&YVDjyZfVtipepXd&giZ+{ zt!n=%Hd|2_aYTeIQ&1n`}48zbh68G}LDL2Ke`Ve$u!}PVT9*oPAdmV$#YhB=U73v^uQzB_yXc-jtStkb zig)(+JwS6d1mj*aW@B<((C*P2uqgLTXYGlzn8-~^D!cp`Lr%x@N&^sE`y36cp^K4| zfO#Jap8-Q38JTeM<6<%6>TRzNAx@x1JujO1^V}r;HdFf;(UWeVW^9gKM~3V9)$8^- z9}zq}{9Y@v%?i*s^kk_XbgJ83kGSwvTe}6k{a3#`TiR}1WvdH+A9hv>&Qs$iK6eYF z;tR?Z$JNnYcja`mTaYYEwkqxny?oO9aowTW*_m}uTIEBI&0clw$u+98U{KpJm7d@< zSdZ31IcYVbLy_xyzS#=YdybAzOf2ba&+u-X!(lbg*zjL1Q~&g;$COBbFR(Pepv4ip zJ3AY)hxehm^8zVkAJMK;zGMdEQE?_v9d)kz{3uxC*ZJ6vyu-lhH=#$I`S+p}Vm>H< z7#+CB5`EyKi_`~~ih6f&xy2I*cQqb^Gl^#1$9;$*2Na8XDLpg2JUWQo>AQ#LyfC*; zdJqXXUD8cj`tpDK8FP#vN?CA@=6^SEE3KDWss-PJX@y4UUPBNpe$~_Eo#y&W%6B*_ z#iy|La6Pp+x8jo1;(AZ(38`{D>gn^@FVeeUygn=&)O1v=Fh)`Nb8X7O7? z-Se$8U>f%-E%_}KP2`bx9E$K9Fz2>DnrLS>e({t)SEp{pz2AlyNga5Ve5!W2imrlI*BtSe=*Iy5oTB|$Eyi_|sLSnrc9$-vz{t1b~t<)aA4okT|_Iqjf z{5H+%X$aA{DH!?oYVd}N4cu%A8F`b>XoB;gtht;WN|;t0OXh~WhjvAIJGPA>Wg&5B zI%^ksVjShZKO7(``$SrjZ`b34I5mtLbf;*huyAdV=7J`88Ee*~9K2sXADkFS1vOwsma*dLc*zRXna|_*a>~6m@H@e`?oym-_Iu zTQ?bNOnB(uG)vVpTL--`#&z`h-c4y5*9_z2_*C?j%Hd$^ZZ@5EC-vNdPhc(DA3ym- z*W$^91$W{+_#FSRqCe&bFLJI07H$_Fh;m(z>TH6wX=bjUj&pV^f^!ZX4}6uK0@^NrFXy{1A|Bt3_Zqt3zIhP{482%GgC`dg$SGMZ=j%x-s zxyZJtjf?-F_h#$+JT+?`U%MPn$xa}faV#r{Rs`nnUs7eOl-?d^=w&LNfEu(A9kP=& zz?PNd;{3HG?jz{PJVfB^EKHd^=#E5z>f|HUCowME%s1PJpXIHw)SYDCE_OBjDkP6v zQT=mTTg2@6-lyp-k9OQ{GA+)0+B(f^!hi&pVMG-T7&3R#{vH@1ACfZjbAdEE zGAo%x`rMZL69#3Rq*+PUBGVINJrz2%DB`f+K=_&fso zgD2H!d!9ECQSD5_7}T!>0aNZu;twBYpO9y_F|re-$`Q3cJDvY{TS$!hK2?Fti2oj9 z_4#Q~Xs*V{2d@(bks7PH@193A__9e6QX9Imp3s;w3nK5GX2nKL?NgH!Q;qGn z)Q4H8C%mR6V?3MpQKxm)_vhAinZzu&9o4}vRj^chUUUUYf(E1N!Gx@~F9}cDA`9X9 zLT5|Tih|W^9@j2e*^5hOdXZ}{C2-5B2_qvHydSz3DkcOqJ}p#VFH>x_D97+uS;|7? zgC?o&Qv8`Y$)B8Fm)S`&)>XXq86FKtjv3g2CTL0WBIEXBDkFSEv@RXd_M&u{3NK{{+c3q zc*B3@NGw63Z8XMN-uKZ%NfJ4e_mfm;FdRwb?~u(+(OJwO23=SQ7q{azRxl z0zT@@l;c@7po(RAawUj5j1};u)%kc6PQ~|Y?=lKNp5*NhM}jq0v0MCV?SQ&Nq`~ zNh5j`)CB6g#j9~LbyoL;EuTL(Ns%I=67u~z2BUp32*e?C^Mc{}WGR+mX>M0EfR^e$ zehk_dNj&reYYFCcfnxs7$p<12VoGmK<(ga;_kiF4I1(}mk}cyb<{&r~pf>wlnTyV? z(u^6Js@t9hsHJ-DE!x*S;l|44XjV19KYAtufE5hKgG;K`hIBuEX)L+B7{!C%C6+B* z5rS*cOD)!0q4ONY<9HBTMn2(my~m(l&S3*Ac_|3b}gk=-qNt@q0rjy`zjum$$LtPT=;QDWNSVl8=! zp|}D5=ms)DlqWHr?{)+v91hd<@oTkZEevRee!i#ghc>R3Q|_xeEgKtxwRc{CdHeOs zonSL1GCw_d_ZZc19!h;B`e2a}Z>)4iY=ho?iGCT`Snbzwa@%pZ6*` zL`(pFYS26BcAB#5j!MN}ZNM}zW~)-z55ha3R`Zm7!w7d^*4WPx^4Jroiy|=u{?Qqe z;>j5&4rynM>Fwa6scWfR9Lx-vpVaC>0g|&#qXduqM+K8ky+uL=GwKX%!rlRkm~y!| zk!ArGP_>7Sp2Wdx>S*b<>!>drF;6=yaZp!y{uz!r)oo~lnX!@6NMfa+_@ulFxtSk= zaHoU5N0Xij@*#a9hxhn#Jw!lOHf)tfib9Na$Cn~j1tkCz2_=i$ zAShZTw5LF{_(%SJ-izVl_Lt)DQkV5NNM{s7Z^?B2f=N~w$pv}oU|4g^X&jfMW%;o; zJ|}WOZwYEq+kRShnQW5~zw%x{bqA94Hmu~8k;4{oK!2F|-pWJJB8(Ao2)9;I+jAoq z5W*-bbpayIfT{fo@-)b5rm9`L!7zKfWC7OW1(F2GRK3)(YY74N^SpT7gy7?#L$q<{ zu58M8b9z#JLx3-{WNhs}NB<|c%Z8BHaNUGJ_Tea}^W6!a!{C{c-!rP8r7~zli(GDi0^2?)#s&kW@WGo$meh4jWp-&g#?`s(U}drjW%;} zJo1E=;p%DhoFI;wp-QaZc9)@#`;mGb-ZUfo=stfUEzXa*PevUc#7qeDsEopX2%8WO z2j-qH0*za3QNw_#swnp##3-7K7*k8n+ZnD!vL-V&ye0TazTsuuy|qS@5rAq1vpK_Z z0TNEb+Cb~GFy&%}EM=!XUy21+iTgQzPiN&UUSN#-(E#D6(mrZ9WrNWtmEM^@?azj% znCzCb%xUG-k@&o3K~FO|>j*UF$BhKqu_DVIU_N9CgHB_pwcV2v#w!6Cx!cJiRvklA zr{M+U87f>a^C&@R=lS~uRm6`W_ zFZ2+B5BE=sZyM96GEhuwvQxM)Ia2op$mbjw$jEW7g<*X)A*5wJlbRXLM7QD8Z^G+O zQ<13T{AM9SKX3fpJ6diJN}t&aPJ8QjOgFQy+YX$j%)cppM55s>mh%OrB>-M~=NU4b zb0YjdczzJEuL*u`*_v}bb9~H#x}y8|+g3JW5)+UC5s-kLF4p>|I6(w;O8&|hIJ1{# zS}@VWQ80<+4Q|h> zh0*VjJjcpr0E2?@BcJAwEvLy3nzs$ZK4R*ua2XgGY&yzrOx_I>J70Yw@QxK5YWjRl zopW{V@Uar4+g2ih;Y?5bgt}W7_Jj|n`sc2-L+s8jTP^Vi7Kl*|;tGGTJ0Y(Ax-=IEK z1R8T?LZXA}jf=(y$Bq;nLLFzKiDnKH%QPyBvgQC0D-%E0T?NU}n*B^E91mVbAl z6~xL+jM#T>EV+U>nw#Vh6vW(0ei?8Uy7sejyWQcWEn_VNHAnhBDRxdO7`QcuV9f8Q zGy`ej&-fU1~jAPk3NY`3jnw#JB0@&h|ECj zL(oVhzqNr!q{*6%oP93bg|5Hs+8Gik{HRN><&L=d&LeP`RCi4~T8rdAd1cpoKRPO4 z&Vbuwf-eZXL`lwm5!g)XWRghg;M-AB4qC5RemY1Nwx56`f~l9hf8bQ_9Htn_WI)*YqYeidXf)s)X}aGO zuw>>J;x&HVGLc0U=zc-y^a;iw6SEfJdiZVc^ea(yqo@`ex|B0SUoW>-RV)REg+C27 zn@703zq3b}yQ9hhG+9Qf2-DNNLP~=cXCP@pT0Q*X6)~SFoV@>FQ8_8IxJ+(n7;4dE zL^l6DxqpDjMp1!YKhZy{+FyofzlhaqmU(w9(I}t?&`48aJrUNU(%86^4C5#nGLvXj zHHsN}T;XWV1jz5-0QrVeUjpHfo9d2T2h+FX2;s+>7i*0r$ByK3Bx61gH=xkPj*oqn zFD05rXs!$qi@@z+7Ym1BeSUmJ(!3w(Lc67|{@oB*>+Zn5`}i7!fqo50N^Ax)Ys32i z@>xdLYYI;qAV25ZbC5!yIDI8yL7-{aI1r@BA6g|&>}@s}TB6WJ{499%nwffljA@unLk~hD9H2lL8QG|jPy=M6 zHtjYw0*_<72C%tKZ5sbzM$=^8fRn0{%hCLdb(@$p+@Woc!=c-Z`sP-!=>C4KYwveq zI2Gj#c*UPj9}erS<|n<*ms%&#@=toe=CRKjv1M842b|Ek%V%hP4OR>KE_M$-XsZa) z4O+JVVTAofN@bOHFE`EGqxrb7wD{iUUjao#+(!>XPYz zM8dwqYu*1w)xG_F;~|s1Vm{NbOsq(m9$6iS-IR(T2CkvQ#UFvga#ZPO{d+#Y2D%tR zfs*bM@Ifln6ZBnQjbhulr~BR~O83bLci64^+O+h)((a6)?H4lken^3%5A zJs$KyItVQC?o!`cud?Kqn+d5bs?r*Q|Q5idOsc->v&UB8w?>@IAZ?rQ|Di z3*)Ce{CeDY&@ngO;ESLPasvqQqGTmn7$cKW%u~&~8#l=$KnAP~IA~vvKBDn3cZ1<( ztdD1Juj4#ZO(wpn*2e3UlR}Zg{_lEuQh&)iRo_XP(oY zeK^Uzno*hG@YyvemB3r)beqAH>}1$h{+hdh73+IC_MLnw_N}JyATIoL(Zf-1gwwt)ozlwJzOxDNP#G(B~4m( zJJ{jsj-#ZCZ-&ha{Ujevjl`wM7=FDwo&|~vM_LMTw&=HZ7_MN=Nv&BK*$897uCiT> zelmJ)u8-UIXEEKqDOR`fDYJKpNh;TT9Z>X^0bl8E{96zJ{UMaT`{0=K`*)7Vw^0RN zVuK@IyN;^=#nx9xMfI(JD;kGKBud%84Uig{gIcjMjv;b#JyVkN` z*>8JT-q`;r0vY?;gwhL4$(`@oU$Qpf({hQ8A(`kYlP^Xww?@TCSKA78Dl)TZsuwR8 zd1Wveb{}J1N0^#A@MG~twe)Ms8Z%3Uj-H*K5}-veu3gkibf0-GBRDmf;F;cMuo;^| z#=f6mOc%w2!r?9oM9@`k*l&K%v*fNyX><#u*YV|c+tEu#-^JYCUmti?E$cvuBwniS z)g63@hE0!GO#Td{%Z!uc@WX%5kbw&gJ=>kx-yZTQZ9T{wHD_bJ(dB$n@C{n2ubg&jMv)!BOrfKJEYIpG-7=}ZCguP>Mz9E_*`7nNzHEtpr zNHMu!w~E;yZd6A;JL!e3?Z3hr|M3v}sDUtVM-*${*RI`7wZ)oQzS{oqSIR2AFHJQ8 zY}MtQ{Ya)k&2#HU5k}4#iDo){bBDL>B|yM-LOenWBkH~rUBghMW96ql-4!4bux*l; zkKxj~?sQBoNqefoNWu}{pmfI&9oS$flkg(*Q3R_qEbLfWVw~X-4_8Y zaQ&IpS}dnvYDVlL8tTx6^@i5=ltJ_^8`vww+hoZ5Hr8t14q2olUboiRF6Q)!@3_gVpI|%B~eVtrxR{}eWT;f2c>Cu-QAx{~)Bv843 zW6%V7O#GZs0W~{#sa!89eY9L%f1|k7Rh^rfL#BE6KsBim$BRNM;n!G^xFSO}hpH>X3RC?~ckAM*w_*cI+sF4LcH(|EzO5&IBap$z zinE3kAX6|lfe;T>=)ww|cneBR78{LZoB40?(MvQSTs?do*F!C4Fp0HDZ{6Q^x+TPF zIwy~h8Kh$G*)5OOF0DSHtVGBfnO z-CcI4X+LCKe|D8es`hWAH=R^nsJCHHTfea!?Z=A=bV@C0jy?~`t~|$m8|IO^sL}2# ze#tm`q|dxQhXt(*)bczjX|x)yFPT(&*ecL|SB58<0HJGz|9XGDDhr|21N$kWgRlpD zf;^_#Vm>d-yY0iauRH7VbJm>sT_>Be%|Gsr2y=j)vUqJYFEe zRc9bt901v|eXteJXGGreLBr4xz2R?H5$l=>F#4bjX+Qs8HX5!3y}6|yz!e2pz<0{tn* znB%Uk=v2FBHJYnhXDt$2Q%`HfqB77^sx^xR7CwkoX|D4;+1++gcRiZV7JB?e+pLW? zrn}35tqC04&Z}D4OZet)J>O8q<3~4`q14d!MyUe&W4aGKV2-dDqTHjwi2_Aa-lsl& zoknf2f|V&9Hpe!#-ErzbwG<9h_4>@#E;8aT@DpW>Hc`UBtb7OH9N@L034{i+v9D2bEe9jg?p0W0Zcdyu7mFblKgAdG(1;{B~ zECp;>aI44-sEiMmnhoB5oQ4z}ln~*=stC#Ho^6=Mg_)_GP|sIN4r|}EnrQa6X|DU& znbxmV@9mmv2fR@DpV_rP4f-_)rg|O3iDUzL8slr-Qk;}p%u1bUtoA)3v4+iE z9rEGqw4SajP7)y;vKoQHf@4c1vuUG5L7v#m7m|shpkfYTHHEx=Homo{T^)O2ke=ZO?yOdhR|92U+sPJtYMd|;7Ka{24KxQOY8;%<>X^Z4bj)% zqMr4HkMUKZVEOWQ_3Uq4auzhspU<%(WU+%E*s93me-g6li@miPhm10r)p_}teW#gr z6}ic~hxDVITO1^Bm|8cl{ELl6+(=VymMZ+emx+t$oYH&no;M1lxW|ESMETuu4Uc zdI5a1WaP6gMpu{WXltRw4VMfe+8Dtlq;>MPt=7p=BNY@dcB|HF%798VXanUWCL{_DVMRYpCn4 zWvOq~9-sQu%!a(0k%;V$JjcnRlFrMYb{VM!${lmwFCk(c`#)Oj*8&wUp-nMFf*NFx zdsDB&;SdrvEyZP>xr0mRXC_zAb5NHHWD_+u7!*~QzW9_ylkvg6i^75=tPF{6KKppN z?wPhfMiKF92s=rA$Sls0h%BVXejDGO-$Vq7XL(=KVHd~9db0@l*aJv^5`0CH2!#s( zovlvuk0v{VZs}y%Vufa)%Fvtn-3>PxjVBbOkHPT|4K)~^SDwdRrC4jOxP9sfZjkUm zeoRyk>l-; zF#V@F-M`f8{szDS>JC?~>wZmIO6_NCc6UaJ4wJI#7zx=HrM!od;2SI*uqKfIT=clu zOZgcE5qxdJ4z^>nYG_8$hmx_oZeNhsT5WQ?;kds7>>Nt5uJ0U22#eFBzJ{t|SlUOK z3@xK7PQ(bOmzV4IORJmNU203hUPykJDOHy<;GwDg&0{{jtt|E4Bk|+(CbFgz%TQUZ zOmGu7jU+YS^HcBDQKIyrg72aRs*<-}@41+bEuW{m9p_wI&F1FTOpxUyfb#b}$@pE> zw%E`={ep0BYm)wcr~sZvuQ5NeKEqoe86&KU90 zlgZ7#I1|t40{in+VZU%s$8|8MJhI5w$E&!_$J5G)Kw0_6bm`nz7Ckg!x0L|IFi1x` zJ&ggePv2N>Z=%d|i-F&_mRmngymVVGRw`{utX`1q_zP8UeUm!BKVI;AB;3a5b@{We zIx+4z8BP2+Z?Y>_uFm%z%^7tiy*a%$7#=$nYpORrCl7l1o6ML({8Bgc1W(fEW(;G+ z@`=Mp3*3qD&6R9o(h6;Z65DEyPE6*_=$1d!5%MhUdVb%gl6iwiZOUpYL;S$hoQ_bM z>M4qDJQ9+$jb@I099a?OdWCJO1wF5lq*^?nsml;k`Vz-Cno`j(o9!0=`pgmbN$b_W zEc^dVIuS>_uT+VP2c(WJ^q+9j#cUT|#z$RJvS~@B8RFN^++j)EsLDT>t1Ql%8akhG zIV@lLxchhA;ne97R3eTc;`r%^ALjwq^b zl3%D?k5)s4Do_@N(WN3AOUgK0kapL#a%hw01}jPb3Zm@xi2Z?SIjLQ1%cMBBuFb`6 zYREqzRU_B1*X;MG|3m1r=ZHSGyw1bPyeRZj zs)Z<;>DT2t)9(nkKjyJWr^^z<;S-aq*)edB9xBhS)nEu$2X}b#jpuQod{k~xJj<&t z;Rc3$d0(CLc(pj!c|Kiqq;5>y&a(q}A~hYv*xD6i<`BH1Q_-F;8Ra1_MJq}cy!P;( zX^c-A{d&glF(&1($Rd;OmZBy}AtN^0_JrTg;+P=!%wdJ*q!%_HNQ8mtR=?@G&@^i2 zS6_j-OV&EHk7wJ=i)?eZ2VAomTl45MF;N{_LMjb5o6Rl^r)wng_qT@ z%$$nkO_$_;w+6R%Zrxo+TFq4@S^>5XhuvarCMy8XxI{5ye)oJ%aVAU!He@lh_`JbS z8Qa9n2X@;%*PH8uGj@`Kf=X`ckDy|d<6XbzUAd@p=(xC|Yzw=@z2I@_rCVtxye1a) zaJ?U{v^C?gTUN8XyT(2tHT-Hdm0za#XbBH3d<7u@BKR0o8HKB3W(s%-;k2aO zC2!3$gDZm?vyeXhJyw+3M3I7?4?|Dt@@YO(eG3zgd20(!KM>_c;l6{g` zZ0c^*HJKM6--NNy&O=+KnXK2Ciq!L7O?&hd=_y7WVdW5^kqJaic9$by?6ld{8XHtQ zB39Ae)`u93#+2et*_27e<;~GgiGOB92}a7@wqSfpo8lPXFh;GhIu! z?@xVp*Cj@8*-AL>tiyhXo+X8#hqJv5e6F&$8BBezYsJP0@47zq6ncH>(Qzc@ckjlR zw}J|KZt%EqwA2h<0rtxbyzSozNKGdky&B=TbS@+^;0Qd+heTsKrs6IMxet*DISKX| zNvu+oudNiJ=GC8vo<1|8a8jNN!X6T8Joh*{eB-jc^?i{~ujh2t_bF+MU6)`tu-Sdj zU5Rve0CDHEL)55m+OiAT!2Gvnkx{Mncp%lNkgY&<3AdU6QS#${~;-n(@;ZD7z!Rw)i zfGsO!#LPujF+Qe7Mzchm-o65yXv2zZK+t81BpkBo~LRJOlcxpR+ZDPag1#=Uag zn;vE^`K)@WA#l^m%tUGS{4D7zvVKh{s3Q^Yi;hou^C9ol!{KT2(OR%_bw<|9*cqg8 zTRFaACLZ7 zQl6E>@TX1lXeP6{)c;DAEex90Fi3Sza{O3PvIr$j*kvy3H&r`3k1g1SL?h~7-`(N)gL4m82$hR#EL8giv3u`tcWET`3Wzozxz|AdK@tGg z@V`H?d%*PEB3VlPuUo&Of0B*t7ZSPnE+hg!Y19}P`J6_Ix6h8H#p=!;gC2KKShfRS zemCs+#o4f*Xp1xU4rlH#Z$NKup%(|LXtu4bF3*JXwEINrkW6;_eV4+^&&lrGDOl zsD4NB<6eep6y@cOAZfYpTIQJj9_`cU&1C6K2b~Ml;E$L4N$JgW_)Hq|?Q&>@BllU9 z5|ISzE_->!G!hY4Z9D1i^v16_znFL%Fk|=CBXuopG;BK@^*A1F>kyTK|2FO-Vz{qT zV{47`)@7eWw@ZFXD-;VsyJoxucua7>jI&t#oCH2ad7mhBk-=l28HPc9&?QhgUSQdRe zi!|7C8%uwmRpRoot-ls+b+O`Zl>x15IlPAJ{N`hO%*`V2v3!q;8FMehCl2JwteRh1 ztlnHFD^pY^oA4_8I*(j9AFu^0Ib3O{T{O*-%>g^R05`Ft|S%n+`C|WNsNKyUXlv4eDR8D~I zSz?RLp6+`9mYT$i&va;$YBq$xy1vA2nyYyMuF4szg0BYgVPs^w((e%B4(DVO86&|u zdU*vassI;vB%TRHv)z8wqTxJ_k#dkdZeXKs9}i{bu<}c~<|PN&)>Y88M^cYx#)U(k zT#^R1T2Zm4dWEvMRODit&%#`uQ7%2Mpo&(8kc+HdgZ=XRn2^7os~AhqHerqWO4##4 zeIJT1u7loe$OYb4oob-ua$F8v^G^Bw;4YG9x<1omKAq+EEa+Q3g6-^Vo4|br#AP$ zCvR*83V;}C4!_VbW#^dv==|*f=WISRGm~NgHAt#Vr&)3;0W@WgY_s@{MaT5m5VsHW z&+u>`%$Eq=-|Si4ooqkisDBV?*&{3RJr*Jylp{kpw!MTjBsrzA8%$L0Ej z?u|UP4I5n7#9m>l+0O6n(cpWljhkP$3b?xzqo{|>@&99+Fs2p}WhAh=-X{XT_B|gJ z9#s1Xn-FI#Jl!5CT6r9`$mHEEJSdnsq^wC z@g58GK=Z39MOE`%eJT~j=W!C8Dr8#Kw#H(be760Kw(Sos#VQZ>3Y@bTIw=G;n5)`cOcQ8NdZVK; z<@+rf=2g7jxTL?^rVo4F_3FgFSEKySU$*G}Yi3e+0wM$H0DDo{kmoJl0y`mEIU-(S zZ}%!&Fm0t-8NLBtmS!B6Rs$i;6yf<`xa&YT$C3?nuZz!%!@WAjXEfv^;zXwYZR$;~ z04wKqJKEk%Hpw$yZOK_vvB1)+ z%PI6paI&WHy_$jdO(m_Q5nWE+miKW=b-7G+luu==SIja%;5<_)(O}k8UG^d^sa?_} zc2fMV&7-Ojlj&1K6oPU@7fDd)wtQ8cxiL2d13S+ebL*6-^M2HII`UJrkC$5(+aNS0R&opVa~buF>iu zhMfJME1HGx#!4N9Dp@my+uJqM5gPt`o&WPX)*Pv&Rlq}oJQF;IiqE18`9yKs5LjId z>Bntq;;~zN$M0sXS>1l6#$P=2wxj=MF_Lw!qT>F+Z*q5{V2J);mGZ2QZK}%jW5!Ld zb%XbHl`FGPy-~*d#Lex~mD{%BH#sA2q%$SCQqiwq>P(c*hg}YJ*8Rq{-gCGI2^1GQ zuN;ggod*rDj-H6YNLNQ7CY5Ue178osITcc?uK%eWeFeZg_ zkF(ZxxPGLu@`<@t`vYPv=kQQ}z()bct8eplF7eu|mbmvMWPCxC#O3h3e7>+8`KSm3 zs&iiF-~QFh%>D|1vPP}-29qoR;j(kBJ8Ux4<>Ta`9{+Iph`C~oYRlkRleyQ$VF?8V zTVEp<^W9z2V{tnrpZmMk+vz>(g+&8>&&{x#%ph^+ttdDvx&Wj?fUky%cQxd zrvZ2O!VhBex<&7^{E8brr$+a)@Uq;Om8HPRkuOGZ+}PcQtR1AYDl;ve9^6-Q-F-dJ zcMJI34p@wpES*SPHf~&i3^}X=qIR##a^TpW9~d4fqI@f8sa7LpIfNH{%s;>1F#L zBPI`hudW)COl+jUabbTXN;TG-DE@|lTpsA#e!G&OW3Tx9;mP1|rCpzgS4TLvzCpRS z1&?Rw_bj5aT2T5rOcD-xpl_)+N_b}Q2y)kF02CR;D_;YKjB3<}(}mQJqRojhJ_Udt z3<1ttXw8OiwT$152N1otvw@BjYi{8FUW!sw-eW__55>bTr_drx0ZPQ$eq*u}Lni2` z70DrOX^xT8&m?>oPOx}+I9Hah=qqJcl~o%_h!Cf|MRe-1yH5883kd8!)=jn1tk74_ z@;%EIu=A=hG@D+?s+`6&71=+W;`!;u5?fekr%`4f5vui+{jBTSsp+12YxM{)F@)c{ zt*Dq#W_?ZN{+Nx7MN(X6J)f{kY{O_t&hNa>eAIJu^B~rFWy4{2$h?%bI@d}+bGfi` zH_vO{vLcah$ks)sd-ciWP!+}Ad^CORocEhw)||SZ=h6l>|J5Dt;8Bp)jl}Kc#9qaH z1qs)OLIU{k+k!lD&8(61Y(KYiANQ|7Q!&SPR{R5s<>xq5dZSY$;k4uHs4~|< zyljf;(&8>Pozz#qCPvL=IfLVx4q)leDN9SJ_?B8cy2)@Imyg#W5+4#5><|7VUx6h^ zL#+dMeikd}ocn|4hR!?IcIpCxD3zBy&(Pj~;!xe2nem%ll}bR^mKpufrkH@rez3c( zKPi?gxPnZRvwBuV{sqi>7)1Y>05v+#@~FK;0$j>tR?d`pfPciH;89rpo&^ST=pveb z!^H3SD;8i0L^w^@V3NFb)iCk^sDdMxeCH~)c*IQ65kH@Ex;V1#ed7H+3bYOA(5kfa zKU2rbr=LaLM1lwG78@w41eY%h9lnL8L)r|n(&`TQ3I`HPpz+tkI8O^aA~mdr8jhp8 zPcF(9#@q2{R!Dg~qMP~N;oD&E`g+Xdtim1{cLKDR%&m1edK`OYB~MAr{DuG@OP>g z+e{d0jpqP*I5XAyw~HWbxQle0`NEb0>5K6-y=K=b{R~Uq>89+jQlX!I8jtnkF-)e~ zY3x=}Lq3t+CsVMQAUUJ_#EJRst5<8B66pTc>4xh(zWx6BFj2c+n_la95$Uqh#t$Xu z!xz<0+ZBF%@>7{#hv{#&g$-T0x2?c=EMB&YXS&?5s}iQ8kK!VCcynza664eiTRXyS z;2YLZBkC_J&%p}J_c?ki$4c@lFPK6kJEuPfaN=coYLF*+VH(%lSh&HmB&Cjz!zgFd z8sz_0r^WgsMbH~&XyqZ%KQSlHG5F%sDxLF0ao72^{S)c~wLy(CQ;w@=@Wr`mPz4ph zFP4hZuD{SB4+&CUO?LJk+E6MirWtTPoL8X#`lyYRl?EED;^{JSUc2)+|87oMf@f)| zeMbD&ES*t5r@{4yk%7EcJN3?))Ca`$c6(hTNDT;v&pNK;G(oGFmzsPJ&s71hNNtv;1YdYVhKq&5(a&MMQ*Z7g> zXeB#$Be^B;N0-i;X_()tw$d>^Q+%Yxr2~vYH3TEbEWPY8qC;XE+}FHoo~Yyd>Ol1^17HV@jKl!u#oa-`Zh3d zAv5`u#>igDP)%5}uDhd3qilX;?Gs5Dn#wu+85ul@MeoE}n*aIMDgvO#0kP<_8AyRN z2%c7p1LpF&{ucbUT@Z2yLmHNKpO`M{NGQQ2R|>v0~r~x>RVu) z-kl9H%zN=pyF#3cLAlYW@tdjrpbWE8uTi0Bi$f1~3MP=L(|V7Cha%ud^_<_hCL3kt z$L~AwJMj-Br3ugtoM%Qr)i|BC_Rv|4VSY9GaINV{2qhSNi9itQJ8X08{=8|BYlU58 zJ%v#Jb75}8&C+va+{=WdYY%S1k*orEKPs%O0bSFHzQDRIB9(ob@3M-BO z)=7_nS(wjJUL-lz6!K0CMUv8}^j&tBCuruIeol-FtV%ZvAfAg6`j8CtRu(Q$T(<^2 zl*OUU;aHXo`VjQ5U$ek(?fr&{&Ltp>ho(^0Pb9)HT4)F5efS$`iQ=>@o60*Ok?YEf zNY^DQ$ug}vwKBbG?H6xU)G%=X(?A}QW!@`(rgjIj{v!dDRA{i5JQAcuDXre3Nazt} z(YK1!tjqF$+P(jH$-P7l=?b0BYvXqfRsV{x&t z-hjQSP<+4#J_kKLD2Fub(BCHZ^M>VU;CxVH4ZsX|Q5-J+(FCgNI-QjsYfRO9kleDBfUZ^v;yesxK^03IJM zXTHKT7Cj`Hs%#9S(KYG;(_Apn7)6CrG3zFi0EO+n0GXNUik}pXx~+wjcA6`B*9%Zp zn{s^qd}9#MKWCY?es^*0wC#M8+vtRV*Czh+spD=t(U4 z2?>rLYd6T(38F3EtY;*BFUi=ia>|ic6&h#*G{J2v!%aYyyzAns{kD4eKKynJ(wAI_ z2?WHGi4I5jGgl#C?_-%?y1G^9|408{gr@ z$3qlxm_P_I`y|0B;#kO2R~&U3;$>m?%uj2rq1)ts`PS8#S-?V?-%Qy=qY)Fl?YeYE z6M(*sU-9T}3>y-Kd4Y;l!a>#@o?M!q-~OB)2`nlZFcuL!0BnjkW`@l zj1tELeHAJa;)q4b_fq-uBNKwC$Or7#R0FJ)lbGv7!NfHgp#50wkrpQa%*cN@U3rwa zpYeN0YiUOuQ$2)}33_T(6u%0xj-r-;$%NeIwLG@HuN8}Qb`RTVeVhZ5up{RdZaMXj zXasI|OUo8ciqqvBsejv)3?4VgD7(7Cpq@MO2D}T)YMCu&e(xz_|<9j?k z+`);~28Bw@YYEgF1{d-W1w)3I{>vQkFYHp<8bBX$n|*N0UHc}ET)VmfX6;i9+Jfq; zNE&@2aX2G?EqQtmv?<_sym}L*YUp+FSJRl^KWYi21Hmmb9$-68{`E0Ba=4~v*)ZJt4k2gHj3{WVlQpL{?q54>L-)U4gjc$p(w}ebRqXOgx!Pib+f@Q*N-}P7d4F+(H?heD zzG{ZyF>FjJUNi!eyZLUPx!W?r9LX#v^V{O9WHaljh|MO!xRY;nkmgRd(d3jpYdZoE z#f?ANjroKd>C5@w?$W#NwgkxJ{Gb|?j#F&B;1qR$JYJ_+=;@lA2nV&V^KP0qyI zU~uZvsn!}83b$)NHb3v2?9U8@vT$7~F?;WCw0;4KBNQ@Ix=kt!`JaCCp`r--_ff4e zGaNn4MB~IXX!xP8P?>`WX@(5zs7y0CA;(=saC zp~K9qV^6~DL+C2RVDrV(kj5=ToZNwg{&b4jYDXarl1yJ;%AR@FrV=2K?n|ut)teA$ z86w|mqbE@|&;*W9>DPamdH$!5AYzOMef&1i)22Vol7^(%=Z$F_RK401^*f9c_w|P= z4ip9-x>~&PPM4j@_XU`N_RbK=7UN0Q94ce>c(h@mX-K{y`M6u-!54^*MKzg*q;$Z$ z3jl9u#*RB8Rz{a@tTS1dmcLD&RK7q~WtpbV`p_dL8`v@t4wI%7^2aA=~K#h0z>WYohK{N&R2*oxt{g;zOW z&(}36u8tKr@|orcbbKG8to53~0=;G|+TvhEPKV=Ez$Z%z=GNx;zS$YLDp8&u*X53~ zAa$|BLqWT5Kj!bK#qc`Btfw1>BzW5sY-EPGV@P?*ga>mVgy``s+NDWI?=eO@gVW-S zaYrAhiDmpm9)7pXJGBExl3isP{u$wr=Yjc1el~Z!i3FdGyEcivBwnOQ3#<~ zVmWJxlb21*pGkJOQ@N3@o+XPP9sb!#!;?9<*{>y2!3j)*8RuDECm?6$k4O{3YB0h) zpWRNvZyV@aVB%Yp$n*tn@e-0J!hE!t*PPjlQpAZ)$byesX)&_BJ)yMp2O5l&gjBHO z5j)*Df3@S-tI%(yp^;2IQ_YU!%#%%=Zk}yNPj}0qvrl#$+C~MT4f~p9+!G(Ufr9lH zkK7pyiGyC?d^c55nct4yu4=j(qnl%@iph(n+8D^+KAjiKck@7x+bI}#n z(fd=T zA|JHbBN*XjF*)0N-XeQ(U9J72#SU5sCrWQd@4}~V?Opl9?mg?)G#GL1QE9&=H5Y<| z4zEQM2y!+e2w7QdGC^}DaNA70Q^7BzTGtRh&C&yl8@z;19=$F5IcN92m_F+Ph_`2g z{@VyeqNK>V=pez6SH$Vqr>s+WpiL+&j&m&Jecs-w4&gp!jGH0s^n#L0Ydns;&ru{G z%X}gon}&lXlwRIjwxUY}cZhVyG{|%narwP8CG@iP4cphz`Kn$1x1!^h)HV!T?Ao^n zJSj8@tpk|#BT7p@?V2#pgV$2NVIC1wpwPs75?sbqb~(0wvp+IwRTzif)M~BRC5K@- zIj8nnwQmoPJZyPj&9r+Yl7DR9QU8+Va?$c(T8UR!!rtJ&`py4*+Xd*p1i*Siqf<3v zq}9dLyOA`sQ?rgnDChuNa8iC$F!(wx)8mltym#fXe!K4=O2yF(^;ClD06kqrm7#sl z<1-CFb3AL_{fx!v$<8n=>k~S*sGqy~F+}F6yD0nvMo|Y_Skl7kSr;Xl*BH2YR5n2_ zg`nP6AR`Qi$ZphmP)fA5d1Dt7U4kX%R>ZnzH9xRoin4k+W zq!;6|)PcyBO8~xT#jKT;WHvtDxc5YP5;os+iZB@N<@}qro17gU1rZYQ2fOY=!Toj? z*;tmnEn7;WlM~}~k!Rk{(6c9}k{pl3Iqu(tw;T{IuTQsjnr@dKTLhl}pbFLnXT7%& zg2w0VhPY)f2N7O?j}6$n+v;Ho`d<2WFUDKe)JUMk$;%2B2&tpd#<< zD76fFI1#Ap#mi?oAeG-k;^46Uy)pKo17uE9qw8SUM#-yonZ+@+Su>2d_C}*C8!p?f z_@)sK{v`CP1qRXWg9kjUuAL6$t_i?P)CWuP-UcH=ifWBs4_seR*YM&vj{&p z9&>McM;MkZiZ2ruopBB|1naUXN!pUqR8u50db25w2i$vYxO#%y*p%?cW&f!6jWUrS zsYF5ISRBuGKEJ`iWj+6?iVVlH=zHtXc;jprR>4X}Wb4a9*<8MLKwHK?c}`TNGg7 zw?tqchxRTub2Mg#!?Dm$X**TalC2l*k9dy4YrF&Shrk^qDd7h;Z1s$$^FPf> z*)%CebM7Cnl6`zfoW<7UFmg zJ0U)Oen|zvgpJicmrh_-SGha_V;M_>o1GX;bSypct4x47vCa0{;p0sC2FwyNNw8di z^XmtoH;Wu|U%?{bm@Lw0okYeJ1%9&f6P*k7h17tQiDhqe zWvbv1MG+7%j3wdSV$aANEyq={rap*aAHs_#{6Y{VB*rCIz_P@lE*hgtA*>vuZ~K#t zjYfDvk%DP;fB8+cFj2a;$p>F7u*KSwkW=KMem1e?!BwI*cj{PV^!Oo@6Z`r<=1ViE zxqG?n8}2OnIquMj5wb|oFV@G^UikMU1NArBwbLar&&d_-cf3B0*e>TJ#w-3zup1aW z%Bs6E%6WdL8}Cre<6BsiiHF)1O#aj~N08!B_+6x+bioT-MRs9eyKJ+WuhXoRNUy~d zn*`jsGSnrZ&`Uvh^x3!TjbHZj;t7sAvIp}`ejd)hh!_UC_k*q**^Y3A&RBgfdn=`h z%~Xsp_LnrNCGu?wZ#z1lO1XDj`{~+AQkv&TdqS^t`(I>0H!^)6B+kc~gx@I%UgprE z&6=idYF)3or{&%1N7s_ab&U=6PN=F>%#SgF8MgL!c?_utd`NGEXoWC|z*>~;Zf;b3 zXa-oaXDB@(jOxYEodtb6p3$_2&)gzb8nBqTOb?6KES6JI*1I0EdQ4AJn;>^ms!FzXiG=R&-ddPWly04`%1GtxXXyE^k23b9POgs$Qm{ zC&=L)Opurg;4Bf(CW1W@jVcysv6dzZ5KhejSW}M@ds|?jcagtaGHSfJyT1mg*IHXN zRZKDfBO}deIThh^ykn#hVX`0&wv}fX(U(U(sLArK*C(dijyD4?wDUjYh@4#=-;l+= zA_uooZ*`(Zp>Xi$WjpMAd99;C8D2=C1GayW#AKj6+MAUOn|IkMRAo0-d(Z-@D9DOV z&M9gVj(XuEbdS+Wow?AYX19Xdk1>p+a{-^~*C>kmuBSf~9H!l3@A-FmZE>EV93!u* zH)CbIl&}ziEZ1`WY`?V* z*qg_vb-90Am|pv*0Ogj4C_6NmT`YIb_9Etj3n#>|q1WMP)hA(LOa({XOfjoi0))~Y z^T|(a%i>hA2);enbB`VH%rxhKko8J7GEDjrP5dP@O@j48BP|YjaAD0+>y62q5h_RQ zd%|(Ub@It#1*!w`Wu1bu#^im+D(&bz+gDDl5RLAe3(Bt&w9F>@B`FgHGH1scyE+vz zj}BTs=bL2ba}1p44Dj#(>gN|ne|om^DYq|ryJEG^Bb5+y2*z!q0jIT_IpIf@2Vc6; zGsqib(y_Ppkdgb3hvw-DGuV8Jx!!X|{#RdeuGg+2Cj=4aR&VmNu?!>eYzf%luen;e&AD~pj;LS%vkrinaK067|{G&x-N7vibHf**Y~Uid!-HRcFu1G z;00zMDO4HC#Y$B|ay`?F?sTYJyFM($2BW9Xvb*^iqGT*Zt#1(W$)#-HYj&wGHR*>x<2c!)9^78P{~sKjpyYrj6jU$-@eh{hKs}Igq$*J zpOnBdqHD5tp`e5fcHcjs9@HR~K0YrHzEi~XafWqy^GTy3GJw*5*wd8bcEX!6_l90q z-*IiTyL`j`Hj}raP|ve#TD_EDnOHHL^ri0epOBVv4v;2(&u`w*Ox-6f<9M@XZ7N#XwpTAzJOLlMjZZd#MC z%C#Nh?e^&^dMi=xv>KbuCnoNFF~9mvoPU_m0X>?1feOFvaU^Y?H=Qw>gER1PeyU_% zrp2M#lR>*vXoQV>NW9(@jB@{nj=;5Nl_ z3{d;b`eKVR>fY}+g)T|;)L0C$d~5^%SSyv}B18IM0$ol8It1at%|;r$S;s9>Yh;pV zgB{c{;7ejfXEmg>#%dwWm}rvR5bWF3%6mX0mEeO_LLqqD=dG8R&ZPsr;D70azNW=L z-g)!@eva|F{hi_ly@CW7T_9$sm+f&>G`vfz&AIY$2EohxuKB|lf{RqBZ7~EjqvUs+k3DamncMFP}z7<{1pZFXI~}Ps2}%&|BL%e8VJ~xSg2) z$H~|+>mT}#XXc71ekaJC_)_{<EzNblG8hH(GI;kDo|ClT`96A+J7l@xJJoHAJ(d`T zm`^Oc|A4Cc@{<{H+|y0ER*fcb7mI$&>aO%96hNfjS%PU!1jog~|sLjD&#Nc(|2Jj$ceyY6XiO&Bu#N%)jwV z%UzG4eyNtj?gT=S)M*vVSUUf-IcI9sfscI_m7w)nSIA4;%a#Hno1-k9LFwn=3ANDm zz*y`x$AxL&Mb^w1#DgZ|MN|7`%d$uCYPk4?Q3+sm_*R|gt-765Pll%7@!e%xMeZHe z{z?J;U`Ny<+hV-?m*3$#!mP^l{O2MIT{MZLdw*yl-}iMRZ)YvahkY>G|K1OSC{YnC zK?Bv!yc-!%Kcqd`+UQ(V{>HjJ^tmve;_QTM?m`?{XkAQ+}&df|t}tP4=PqsT0e7Jd0*;w2g$MW?zRxQ`ygbTpsRt1y^%4 zfF)AyBQ=$+YTim-z~W1J3)8uC#UNgfmTXC|LT ziJ^&@%+RG0uQ{683YYdSDgth`X43OY)x!T{{z`wP`eVKntDQEJYKJP4@Tag-R~V;k zJZJ*4xUSA}~e@{YYx9wZ#!nexGxo0guwP;Vzt>0j!>m(&4(55}%^ z2~haJuZc$ig0o_VK=!t^eAnP6-Txrvi^wAxgJ2Oqoareb2rpybAV^O=e@^&d@o$tP zT>Ss-!Lzf{uU*y@RAUu3Eki$Q9ff*;%{>QVg$LRdRd%*~Xf+KLFxYvcY z#H|Q1T>2CuL1VAeradXeVlUhQ5Qak&XsPuA-osf~{E4o1lK@byj_%3Vn2YbxMH*RD zfx~`|(rs6f55_@P1sx(x0j=kajGA=a5u?G1u_Thy3z&YJ%y?EKZ==Uh1&N81PC&Jr zwy3O!U6F++?1ZKOwC*-frMQ71I-3DmNwpQ--@MmkE{^LZma=qiQJd(6o=6)!SuI^ z8EBK~UAr~1gc~YPs(}rPPqo6Im~aIyLR?`ZsO9?RTjE=U7-sVTBBF3#_{aZaH+Mws S;8>r|00f?{elF{r5}E+qn&T<} literal 0 HcmV?d00001 From 7b1652f23cfd708179559879aa5037b748a74f04 Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Tue, 27 Aug 2024 01:13:42 +0000 Subject: [PATCH 5/7] Minor copy tweaks, thanks coderabbitai --- guides/contributing.rst | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/guides/contributing.rst b/guides/contributing.rst index 5162b7eb27..482ec14d87 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -5,8 +5,7 @@ Contributing :description: Getting started guide for contributing to the ESPHome project :image: github-circle.svg -We welcome contributions to the ESPHome suite of code and documentation! All the code for the projects is hosted on -GitHub: +We welcome contributions to the ESPHome suite of code and documentation! All of the project code is hosted on GitHub: - `ESPHome `__ (Project Source Code) - `ESPHome-Docs `__ (The documentation which you're reading here) @@ -45,8 +44,8 @@ something like this: :width: 80.0% :alt: a screenshot of an rST file opened in GitHub, with the edit button circled -Click the edit button to start making changes. If you're not sure about some element of syntax, see the quick-start -:ref:`rst-syntax` guide. +Click the edit button to start making changes. If you're unsure about syntax, see the quick-start :ref:`rst-syntax` +guide. Once you've made your changes, give them a useful name and press "Propose changes". At this point, you've made the changes on your own personal copy of the docs in GitHub, but you still need to submit them to us. @@ -112,8 +111,8 @@ Build docker run --rm -v "${PWD}/":/workspaces/esphome-docs -p 8000:8000 -it ghcr.io/esphome/esphome-docs - With ``PWD`` referring to the root of the ``esphome-docs`` git repository. Then go to ``:8000`` in - your browser. + ...with ``PWD`` referring to the root of the ``esphome-docs`` git repository. Then, to see the preview, go to + ``:8000`` in your browser. This way, you don't have to install the dependencies to build the documentation. From 5f23bfb021cdef225b4c7648f5d4d1f52a578c33 Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Mon, 26 Aug 2024 20:59:10 -0500 Subject: [PATCH 6/7] Update guides/contributing.rst Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com> --- guides/contributing.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/guides/contributing.rst b/guides/contributing.rst index 482ec14d87..1ff2b5d2c8 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -112,7 +112,7 @@ Build docker run --rm -v "${PWD}/":/workspaces/esphome-docs -p 8000:8000 -it ghcr.io/esphome/esphome-docs ...with ``PWD`` referring to the root of the ``esphome-docs`` git repository. Then, to see the preview, go to - ``:8000`` in your browser. + ``:8000`` in your browser. This way, you don't have to install the dependencies to build the documentation. From 6a2e65ec515185bab8cf98f9e6e749cfe0029fbf Mon Sep 17 00:00:00 2001 From: Keith Burzinski Date: Tue, 27 Aug 2024 02:15:57 +0000 Subject: [PATCH 7/7] Add anchor --- guides/contributing.rst | 2 ++ 1 file changed, 2 insertions(+) diff --git a/guides/contributing.rst b/guides/contributing.rst index 1ff2b5d2c8..4a71f82f96 100644 --- a/guides/contributing.rst +++ b/guides/contributing.rst @@ -479,6 +479,8 @@ here are some tips: whatever reason) accept...and then you have to go back and re-do it all to get your PR merged. It's easier to make corrections early-on -- and we want to help you! +.. _prs-are-being-drafted-when-changes-are-needed: + Why Was My PR was Marked as a Draft? ************************************