2018-12-05 21:22:06 +01:00
|
|
|
import logging
|
2018-05-14 11:50:56 +02:00
|
|
|
import math
|
2018-12-05 21:22:06 +01:00
|
|
|
import os
|
2018-06-02 22:22:20 +02:00
|
|
|
import re
|
2021-06-17 21:54:14 +02:00
|
|
|
from typing import TYPE_CHECKING, Dict, List, Optional, Set, Tuple
|
2018-05-14 11:50:56 +02:00
|
|
|
|
2021-03-07 20:03:16 +01:00
|
|
|
from esphome.const import (
|
|
|
|
CONF_ARDUINO_VERSION,
|
|
|
|
CONF_COMMENT,
|
|
|
|
CONF_ESPHOME,
|
|
|
|
CONF_USE_ADDRESS,
|
2021-05-06 17:31:42 +02:00
|
|
|
CONF_ETHERNET,
|
2021-03-07 20:03:16 +01:00
|
|
|
CONF_WIFI,
|
2021-05-17 07:14:15 +02:00
|
|
|
SOURCE_FILE_EXTENSIONS,
|
2021-03-07 20:03:16 +01:00
|
|
|
)
|
2021-05-17 07:14:15 +02:00
|
|
|
from esphome.coroutine import FakeAwaitable as _FakeAwaitable
|
|
|
|
from esphome.coroutine import FakeEventLoop as _FakeEventLoop
|
|
|
|
|
|
|
|
# pylint: disable=unused-import
|
|
|
|
from esphome.coroutine import coroutine, coroutine_with_priority # noqa
|
2019-03-03 16:50:06 +01:00
|
|
|
from esphome.helpers import ensure_unique_string, is_hassio
|
2019-04-17 12:06:00 +02:00
|
|
|
from esphome.util import OrderedDict
|
2019-01-02 14:11:11 +01:00
|
|
|
|
2020-07-16 10:03:11 +02:00
|
|
|
if TYPE_CHECKING:
|
2021-05-07 20:02:17 +02:00
|
|
|
from ..cpp_generator import MockObj, MockObjClass, Statement
|
2021-06-17 21:54:14 +02:00
|
|
|
from ..types import ConfigType
|
2020-07-16 10:03:11 +02:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2019-02-13 16:54:02 +01:00
|
|
|
class EsphomeError(Exception):
|
|
|
|
"""General ESPHome exception occurred."""
|
2018-04-07 01:23:03 +02:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class HexInt(int):
|
2018-04-07 01:23:03 +02:00
|
|
|
def __str__(self):
|
2020-07-14 13:11:58 +02:00
|
|
|
value = self
|
|
|
|
sign = "-" if value < 0 else ""
|
|
|
|
value = abs(value)
|
|
|
|
if 0 <= value <= 255:
|
|
|
|
return f"{sign}0x{value:02X}"
|
|
|
|
return f"{sign}0x{value:X}"
|
2018-04-07 01:23:03 +02:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class IPAddress:
|
2018-04-07 01:23:03 +02:00
|
|
|
def __init__(self, *args):
|
|
|
|
if len(args) != 4:
|
2020-07-14 13:11:58 +02:00
|
|
|
raise ValueError("IPAddress must consist of 4 items")
|
2018-04-07 01:23:03 +02:00
|
|
|
self.args = args
|
|
|
|
|
|
|
|
def __str__(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return ".".join(str(x) for x in self.args)
|
2018-04-18 18:43:13 +02:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class MACAddress:
|
2018-05-14 11:50:56 +02:00
|
|
|
def __init__(self, *parts):
|
|
|
|
if len(parts) != 6:
|
2019-12-07 18:28:55 +01:00
|
|
|
raise ValueError("MAC Address must consist of 6 items")
|
2018-05-14 11:50:56 +02:00
|
|
|
self.parts = parts
|
|
|
|
|
|
|
|
def __str__(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return ":".join(f"{part:02X}" for part in self.parts)
|
2018-05-14 11:50:56 +02:00
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
@property
|
2018-06-11 10:01:54 +02:00
|
|
|
def as_hex(self):
|
2019-02-13 16:54:02 +01:00
|
|
|
from esphome.cpp_generator import RawExpression
|
2018-06-11 10:01:54 +02:00
|
|
|
|
2021-03-07 20:03:16 +01:00
|
|
|
num = "".join(f"{part:02X}" for part in self.parts)
|
|
|
|
return RawExpression(f"0x{num}ULL")
|
2018-06-11 10:01:54 +02:00
|
|
|
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
def is_approximately_integer(value):
|
2019-12-07 18:28:55 +01:00
|
|
|
if isinstance(value, int):
|
2018-05-14 11:50:56 +02:00
|
|
|
return True
|
|
|
|
return abs(value - round(value)) < 0.001
|
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class TimePeriod:
|
2021-03-07 20:03:16 +01:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
microseconds=None,
|
|
|
|
milliseconds=None,
|
|
|
|
seconds=None,
|
|
|
|
minutes=None,
|
|
|
|
hours=None,
|
|
|
|
days=None,
|
|
|
|
):
|
2018-05-14 11:50:56 +02:00
|
|
|
if days is not None:
|
|
|
|
if not is_approximately_integer(days):
|
|
|
|
frac_days, days = math.modf(days)
|
|
|
|
hours = (hours or 0) + frac_days * 24
|
|
|
|
self.days = int(round(days))
|
|
|
|
else:
|
|
|
|
self.days = None
|
|
|
|
|
|
|
|
if hours is not None:
|
|
|
|
if not is_approximately_integer(hours):
|
|
|
|
frac_hours, hours = math.modf(hours)
|
|
|
|
minutes = (minutes or 0) + frac_hours * 60
|
|
|
|
self.hours = int(round(hours))
|
|
|
|
else:
|
|
|
|
self.hours = None
|
|
|
|
|
|
|
|
if minutes is not None:
|
|
|
|
if not is_approximately_integer(minutes):
|
|
|
|
frac_minutes, minutes = math.modf(minutes)
|
|
|
|
seconds = (seconds or 0) + frac_minutes * 60
|
|
|
|
self.minutes = int(round(minutes))
|
|
|
|
else:
|
|
|
|
self.minutes = None
|
|
|
|
|
|
|
|
if seconds is not None:
|
|
|
|
if not is_approximately_integer(seconds):
|
|
|
|
frac_seconds, seconds = math.modf(seconds)
|
|
|
|
milliseconds = (milliseconds or 0) + frac_seconds * 1000
|
|
|
|
self.seconds = int(round(seconds))
|
|
|
|
else:
|
|
|
|
self.seconds = None
|
|
|
|
|
|
|
|
if milliseconds is not None:
|
|
|
|
if not is_approximately_integer(milliseconds):
|
|
|
|
frac_milliseconds, milliseconds = math.modf(milliseconds)
|
|
|
|
microseconds = (microseconds or 0) + frac_milliseconds * 1000
|
|
|
|
self.milliseconds = int(round(milliseconds))
|
|
|
|
else:
|
|
|
|
self.milliseconds = None
|
|
|
|
|
|
|
|
if microseconds is not None:
|
|
|
|
if not is_approximately_integer(microseconds):
|
|
|
|
raise ValueError("Maximum precision is microseconds")
|
|
|
|
self.microseconds = int(round(microseconds))
|
|
|
|
else:
|
|
|
|
self.microseconds = None
|
|
|
|
|
|
|
|
def as_dict(self):
|
|
|
|
out = OrderedDict()
|
|
|
|
if self.microseconds is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
out["microseconds"] = self.microseconds
|
2018-05-14 11:50:56 +02:00
|
|
|
if self.milliseconds is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
out["milliseconds"] = self.milliseconds
|
2018-05-14 11:50:56 +02:00
|
|
|
if self.seconds is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
out["seconds"] = self.seconds
|
2018-05-14 11:50:56 +02:00
|
|
|
if self.minutes is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
out["minutes"] = self.minutes
|
2018-05-14 11:50:56 +02:00
|
|
|
if self.hours is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
out["hours"] = self.hours
|
2018-05-14 11:50:56 +02:00
|
|
|
if self.days is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
out["days"] = self.days
|
2018-05-14 11:50:56 +02:00
|
|
|
return out
|
|
|
|
|
2018-10-16 20:45:24 +02:00
|
|
|
def __str__(self):
|
|
|
|
if self.microseconds is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.total_microseconds}us"
|
2018-10-16 20:45:24 +02:00
|
|
|
if self.milliseconds is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.total_milliseconds}ms"
|
2018-10-16 20:45:24 +02:00
|
|
|
if self.seconds is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.total_seconds}s"
|
2018-10-16 20:45:24 +02:00
|
|
|
if self.minutes is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.total_minutes}min"
|
2018-10-16 20:45:24 +02:00
|
|
|
if self.hours is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.total_hours}h"
|
2018-10-16 20:45:24 +02:00
|
|
|
if self.days is not None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.total_days}d"
|
|
|
|
return "0s"
|
2018-10-16 20:45:24 +02:00
|
|
|
|
2020-03-12 22:27:22 +01:00
|
|
|
def __repr__(self):
|
|
|
|
return f"TimePeriod<{self.total_microseconds}>"
|
|
|
|
|
2018-05-14 11:50:56 +02:00
|
|
|
@property
|
|
|
|
def total_microseconds(self):
|
|
|
|
return self.total_milliseconds * 1000 + (self.microseconds or 0)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def total_milliseconds(self):
|
|
|
|
return self.total_seconds * 1000 + (self.milliseconds or 0)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def total_seconds(self):
|
|
|
|
return self.total_minutes * 60 + (self.seconds or 0)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def total_minutes(self):
|
|
|
|
return self.total_hours * 60 + (self.minutes or 0)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def total_hours(self):
|
|
|
|
return self.total_days * 24 + (self.hours or 0)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def total_days(self):
|
|
|
|
return self.days or 0
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, TimePeriod):
|
|
|
|
return self.total_microseconds == other.total_microseconds
|
|
|
|
return NotImplemented
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
def __ne__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, TimePeriod):
|
|
|
|
return self.total_microseconds != other.total_microseconds
|
|
|
|
return NotImplemented
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
def __lt__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, TimePeriod):
|
|
|
|
return self.total_microseconds < other.total_microseconds
|
|
|
|
return NotImplemented
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
def __gt__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, TimePeriod):
|
|
|
|
return self.total_microseconds > other.total_microseconds
|
|
|
|
return NotImplemented
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
def __le__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, TimePeriod):
|
|
|
|
return self.total_microseconds <= other.total_microseconds
|
|
|
|
return NotImplemented
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
def __ge__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, TimePeriod):
|
|
|
|
return self.total_microseconds >= other.total_microseconds
|
|
|
|
return NotImplemented
|
2018-05-14 11:50:56 +02:00
|
|
|
|
|
|
|
|
|
|
|
class TimePeriodMicroseconds(TimePeriod):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class TimePeriodMilliseconds(TimePeriod):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class TimePeriodSeconds(TimePeriod):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2019-03-06 12:39:52 +01:00
|
|
|
class TimePeriodMinutes(TimePeriod):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2021-03-07 20:03:16 +01:00
|
|
|
LAMBDA_PROG = re.compile(r"id\(\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\)(\.?)")
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class Lambda:
|
2021-02-06 16:09:15 +01:00
|
|
|
def __init__(self, value):
|
2019-04-22 21:56:30 +02:00
|
|
|
# pylint: disable=protected-access
|
|
|
|
if isinstance(value, Lambda):
|
|
|
|
self._value = value._value
|
|
|
|
else:
|
|
|
|
self._value = value
|
2018-12-05 21:22:06 +01:00
|
|
|
self._parts = None
|
|
|
|
self._requires_ids = None
|
2018-05-20 12:41:52 +02:00
|
|
|
|
2020-04-06 03:14:49 +02:00
|
|
|
# https://stackoverflow.com/a/241506/229052
|
|
|
|
def comment_remover(self, text):
|
|
|
|
def replacer(match):
|
|
|
|
s = match.group(0)
|
2021-03-07 20:03:16 +01:00
|
|
|
if s.startswith("/"):
|
2020-04-06 03:14:49 +02:00
|
|
|
return " " # note: a space and not an empty string
|
|
|
|
return s
|
2021-03-07 20:03:16 +01:00
|
|
|
|
2020-04-06 03:14:49 +02:00
|
|
|
pattern = re.compile(
|
|
|
|
r'//.*?$|/\*.*?\*/|\'(?:\\.|[^\\\'])*\'|"(?:\\.|[^\\"])*"',
|
2021-03-07 20:03:16 +01:00
|
|
|
re.DOTALL | re.MULTILINE,
|
2020-04-06 03:14:49 +02:00
|
|
|
)
|
|
|
|
return re.sub(pattern, replacer, text)
|
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
@property
|
|
|
|
def parts(self):
|
|
|
|
if self._parts is None:
|
2020-04-06 03:14:49 +02:00
|
|
|
self._parts = re.split(LAMBDA_PROG, self.comment_remover(self._value))
|
2018-12-05 21:22:06 +01:00
|
|
|
return self._parts
|
2018-05-20 12:41:52 +02:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
@property
|
|
|
|
def requires_ids(self):
|
|
|
|
if self._requires_ids is None:
|
2021-03-07 20:03:16 +01:00
|
|
|
self._requires_ids = [
|
|
|
|
ID(self.parts[i]) for i in range(1, len(self.parts), 3)
|
|
|
|
]
|
2018-12-05 21:22:06 +01:00
|
|
|
return self._requires_ids
|
2018-05-20 12:41:52 +02:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
@property
|
|
|
|
def value(self):
|
|
|
|
return self._value
|
2018-06-02 22:22:20 +02:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
@value.setter
|
|
|
|
def value(self, value):
|
|
|
|
self._value = value
|
|
|
|
self._parts = None
|
|
|
|
self._requires_ids = None
|
2018-06-02 22:22:20 +02:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
def __str__(self):
|
|
|
|
return self.value
|
2018-06-02 22:22:20 +02:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
def __repr__(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"Lambda<{self.value}>"
|
2018-06-02 22:22:20 +02:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class ID:
|
2019-04-22 21:56:30 +02:00
|
|
|
def __init__(self, id, is_declaration=False, type=None, is_manual=None):
|
2018-06-02 22:22:20 +02:00
|
|
|
self.id = id
|
2019-04-22 21:56:30 +02:00
|
|
|
if is_manual is None:
|
|
|
|
self.is_manual = id is not None
|
|
|
|
else:
|
|
|
|
self.is_manual = is_manual
|
2018-06-02 22:22:20 +02:00
|
|
|
self.is_declaration = is_declaration
|
2021-03-07 20:03:16 +01:00
|
|
|
self.type: Optional["MockObjClass"] = type
|
2018-06-02 22:22:20 +02:00
|
|
|
|
|
|
|
def resolve(self, registered_ids):
|
2019-02-13 16:54:02 +01:00
|
|
|
from esphome.config_validation import RESERVED_IDS
|
2019-02-10 16:41:12 +01:00
|
|
|
|
2018-06-02 22:22:20 +02:00
|
|
|
if self.id is None:
|
2021-03-07 20:03:16 +01:00
|
|
|
base = str(self.type).replace("::", "_").lower()
|
|
|
|
name = "".join(c for c in base if c.isalnum() or c == "_")
|
2019-02-10 16:41:12 +01:00
|
|
|
used = set(registered_ids) | set(RESERVED_IDS)
|
|
|
|
self.id = ensure_unique_string(name, used)
|
2018-06-02 22:22:20 +02:00
|
|
|
return self.id
|
|
|
|
|
|
|
|
def __str__(self):
|
2018-08-13 19:11:33 +02:00
|
|
|
if self.id is None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return ""
|
2018-06-02 22:22:20 +02:00
|
|
|
return self.id
|
|
|
|
|
|
|
|
def __repr__(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return (
|
|
|
|
f"ID<{self.id} declaration={self.is_declaration}, "
|
|
|
|
f"type={self.type}, manual={self.is_manual}>"
|
|
|
|
)
|
2018-06-02 22:22:20 +02:00
|
|
|
|
|
|
|
def __eq__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, ID):
|
|
|
|
return self.id == other.id
|
|
|
|
return NotImplemented
|
2018-06-02 22:22:20 +02:00
|
|
|
|
|
|
|
def __hash__(self):
|
|
|
|
return hash(self.id)
|
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
def copy(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return ID(
|
|
|
|
self.id,
|
|
|
|
is_declaration=self.is_declaration,
|
|
|
|
type=self.type,
|
|
|
|
is_manual=self.is_manual,
|
|
|
|
)
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class DocumentLocation:
|
2020-01-13 23:35:55 +01:00
|
|
|
def __init__(self, document: str, line: int, column: int):
|
|
|
|
self.document: str = document
|
|
|
|
self.line: int = line
|
|
|
|
self.column: int = column
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_mark(cls, mark):
|
2021-03-07 20:03:16 +01:00
|
|
|
return cls(mark.name, mark.line, mark.column)
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
def __str__(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.document} {self.line}:{self.column}"
|
2019-04-22 21:56:30 +02:00
|
|
|
|
2021-01-24 00:17:15 +01:00
|
|
|
@property
|
|
|
|
def as_line_directive(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
document_path = str(self.document).replace("\\", "\\\\")
|
2021-03-02 02:16:36 +01:00
|
|
|
return f'#line {self.line + 1} "{document_path}"'
|
2021-01-24 00:17:15 +01:00
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class DocumentRange:
|
2020-01-13 23:35:55 +01:00
|
|
|
def __init__(self, start_mark: DocumentLocation, end_mark: DocumentLocation):
|
|
|
|
self.start_mark: DocumentLocation = start_mark
|
|
|
|
self.end_mark: DocumentLocation = end_mark
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_marks(cls, start_mark, end_mark):
|
|
|
|
return cls(
|
2021-03-07 20:03:16 +01:00
|
|
|
DocumentLocation.from_mark(start_mark), DocumentLocation.from_mark(end_mark)
|
2019-04-22 21:56:30 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def __str__(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"[{self.start_mark} - {self.end_mark}]"
|
2019-04-22 21:56:30 +02:00
|
|
|
|
2018-06-02 22:22:20 +02:00
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class Define:
|
2019-04-17 12:06:00 +02:00
|
|
|
def __init__(self, name, value=None):
|
|
|
|
self.name = name
|
|
|
|
self.value = value
|
|
|
|
|
|
|
|
@property
|
|
|
|
def as_build_flag(self):
|
|
|
|
if self.value is None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"-D{self.name}"
|
|
|
|
return f"-D{self.name}={self.value}"
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def as_macro(self):
|
|
|
|
if self.value is None:
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"#define {self.name}"
|
|
|
|
return f"#define {self.name} {self.value}"
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def as_tuple(self):
|
|
|
|
return self.name, self.value
|
|
|
|
|
|
|
|
def __hash__(self):
|
|
|
|
return hash(self.as_tuple)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, Define):
|
|
|
|
return self.as_tuple == other.as_tuple
|
|
|
|
return NotImplemented
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class Library:
|
2021-07-26 10:50:45 +02:00
|
|
|
def __init__(self, name, version, repository=None):
|
2019-04-17 12:06:00 +02:00
|
|
|
self.name = name
|
|
|
|
self.version = version
|
2021-07-26 10:50:45 +02:00
|
|
|
self.repository = repository
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.as_lib_dep
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def as_lib_dep(self):
|
2021-07-26 10:50:45 +02:00
|
|
|
if self.repository is not None:
|
|
|
|
if self.name is not None:
|
|
|
|
return f"{self.name}={self.repository}"
|
|
|
|
return self.repository
|
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
if self.version is None:
|
|
|
|
return self.name
|
2021-03-07 20:03:16 +01:00
|
|
|
return f"{self.name}@{self.version}"
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def as_tuple(self):
|
2021-07-26 10:50:45 +02:00
|
|
|
return self.name, self.version, self.repository
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
def __hash__(self):
|
|
|
|
return hash(self.as_tuple)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
2020-01-13 23:35:55 +01:00
|
|
|
if isinstance(other, Library):
|
|
|
|
return self.as_tuple == other.as_tuple
|
|
|
|
return NotImplemented
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
def find_source_files(file):
|
|
|
|
files = set()
|
|
|
|
directory = os.path.abspath(os.path.dirname(file))
|
|
|
|
for f in os.listdir(directory):
|
|
|
|
if not os.path.isfile(os.path.join(directory, f)):
|
|
|
|
continue
|
|
|
|
_, ext = os.path.splitext(f)
|
|
|
|
if ext.lower() not in SOURCE_FILE_EXTENSIONS:
|
|
|
|
continue
|
|
|
|
files.add(f)
|
|
|
|
return files
|
2019-04-09 14:30:12 +02:00
|
|
|
|
|
|
|
|
2019-02-28 10:11:05 +01:00
|
|
|
# pylint: disable=too-many-instance-attributes,too-many-public-methods
|
2019-12-07 18:28:55 +01:00
|
|
|
class EsphomeCore:
|
2018-12-05 21:22:06 +01:00
|
|
|
def __init__(self):
|
2018-12-18 19:31:43 +01:00
|
|
|
# True if command is run from dashboard
|
|
|
|
self.dashboard = False
|
2019-04-22 21:56:30 +02:00
|
|
|
# True if command is run from vscode api
|
|
|
|
self.vscode = False
|
2019-05-11 11:41:09 +02:00
|
|
|
self.ace = False
|
2018-12-05 21:22:06 +01:00
|
|
|
# The name of the node
|
2020-01-13 23:35:55 +01:00
|
|
|
self.name: Optional[str] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The relative path to the configuration YAML
|
2020-01-13 23:35:55 +01:00
|
|
|
self.config_path: Optional[str] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The relative path to where all build files are stored
|
2020-01-13 23:35:55 +01:00
|
|
|
self.build_path: Optional[str] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The platform (ESP8266, ESP32) of this device
|
2020-01-13 23:35:55 +01:00
|
|
|
self.esp_platform: Optional[str] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The board that's used (for example nodemcuv2)
|
2020-01-13 23:35:55 +01:00
|
|
|
self.board: Optional[str] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The full raw configuration
|
2021-06-17 21:54:14 +02:00
|
|
|
self.raw_config: Optional["ConfigType"] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The validated configuration, this is None until the config has been validated
|
2021-06-17 21:54:14 +02:00
|
|
|
self.config: Optional["ConfigType"] = None
|
2018-12-05 21:22:06 +01:00
|
|
|
# The pending tasks in the task queue (mostly for C++ generation)
|
2019-04-17 12:06:00 +02:00
|
|
|
# This is a priority queue (with heapq)
|
|
|
|
# Each item is a tuple of form: (-priority, unique number, task)
|
2021-05-17 07:14:15 +02:00
|
|
|
self.event_loop = _FakeEventLoop()
|
2019-04-17 12:06:00 +02:00
|
|
|
# Task counter for pending tasks
|
|
|
|
self.task_counter = 0
|
2018-12-05 21:22:06 +01:00
|
|
|
# The variable cache, for each ID this holds a MockObj of the variable obj
|
2021-03-07 20:03:16 +01:00
|
|
|
self.variables: Dict[str, "MockObj"] = {}
|
2019-04-17 12:06:00 +02:00
|
|
|
# A list of statements that go in the main setup() block
|
2021-03-07 20:03:16 +01:00
|
|
|
self.main_statements: List["Statement"] = []
|
2019-04-17 12:06:00 +02:00
|
|
|
# A list of statements to insert in the global block (includes and global variables)
|
2021-03-07 20:03:16 +01:00
|
|
|
self.global_statements: List["Statement"] = []
|
2019-04-17 12:06:00 +02:00
|
|
|
# A set of platformio libraries to add to the project
|
2020-01-13 23:35:55 +01:00
|
|
|
self.libraries: List[Library] = []
|
2019-04-17 12:06:00 +02:00
|
|
|
# A set of build flags to set in the platformio project
|
2020-01-13 23:35:55 +01:00
|
|
|
self.build_flags: Set[str] = set()
|
2019-04-17 12:06:00 +02:00
|
|
|
# A set of defines to set for the compile process in esphome/core/defines.h
|
2021-03-07 20:03:16 +01:00
|
|
|
self.defines: Set["Define"] = set()
|
2019-04-24 23:49:02 +02:00
|
|
|
# A set of strings of names of loaded integrations, used to find namespace ID conflicts
|
|
|
|
self.loaded_integrations = set()
|
2019-05-21 12:23:38 +02:00
|
|
|
# A set of component IDs to track what Component subclasses are declared
|
|
|
|
self.component_ids = set()
|
2019-10-19 14:04:14 +02:00
|
|
|
# Whether ESPHome was started in verbose mode
|
|
|
|
self.verbose = False
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
self.dashboard = False
|
|
|
|
self.name = None
|
|
|
|
self.config_path = None
|
|
|
|
self.build_path = None
|
|
|
|
self.esp_platform = None
|
|
|
|
self.board = None
|
|
|
|
self.raw_config = None
|
|
|
|
self.config = None
|
2021-05-17 07:14:15 +02:00
|
|
|
self.event_loop = _FakeEventLoop()
|
2019-04-17 12:06:00 +02:00
|
|
|
self.task_counter = 0
|
|
|
|
self.variables = {}
|
|
|
|
self.main_statements = []
|
|
|
|
self.global_statements = []
|
2019-06-28 11:29:37 +02:00
|
|
|
self.libraries = []
|
2019-04-17 12:06:00 +02:00
|
|
|
self.build_flags = set()
|
|
|
|
self.defines = set()
|
2019-04-24 23:49:02 +02:00
|
|
|
self.loaded_integrations = set()
|
2019-05-21 12:23:38 +02:00
|
|
|
self.component_ids = set()
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
@property
|
2020-01-13 23:35:55 +01:00
|
|
|
def address(self) -> Optional[str]:
|
2020-07-15 14:00:02 +02:00
|
|
|
if self.config is None:
|
|
|
|
raise ValueError("Config has not been loaded yet")
|
|
|
|
|
2021-03-07 20:03:16 +01:00
|
|
|
if "wifi" in self.config:
|
2019-02-10 23:55:13 +01:00
|
|
|
return self.config[CONF_WIFI][CONF_USE_ADDRESS]
|
2019-01-05 20:47:33 +01:00
|
|
|
|
2021-05-06 17:31:42 +02:00
|
|
|
if CONF_ETHERNET in self.config:
|
|
|
|
return self.config[CONF_ETHERNET][CONF_USE_ADDRESS]
|
2019-01-05 20:47:33 +01:00
|
|
|
|
2019-02-13 21:14:03 +01:00
|
|
|
return None
|
2018-12-05 21:22:06 +01:00
|
|
|
|
2019-10-14 11:27:07 +02:00
|
|
|
@property
|
2020-01-13 23:35:55 +01:00
|
|
|
def comment(self) -> Optional[str]:
|
2020-07-15 14:00:02 +02:00
|
|
|
if self.config is None:
|
|
|
|
raise ValueError("Config has not been loaded yet")
|
|
|
|
|
2019-10-14 11:27:07 +02:00
|
|
|
if CONF_COMMENT in self.config[CONF_ESPHOME]:
|
|
|
|
return self.config[CONF_ESPHOME][CONF_COMMENT]
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
@property
|
2020-01-13 23:35:55 +01:00
|
|
|
def arduino_version(self) -> str:
|
2020-07-15 14:00:02 +02:00
|
|
|
if self.config is None:
|
|
|
|
raise ValueError("Config has not been loaded yet")
|
|
|
|
|
2019-02-13 16:54:02 +01:00
|
|
|
return self.config[CONF_ESPHOME][CONF_ARDUINO_VERSION]
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
@property
|
|
|
|
def config_dir(self):
|
|
|
|
return os.path.dirname(self.config_path)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def config_filename(self):
|
|
|
|
return os.path.basename(self.config_path)
|
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
def relative_config_path(self, *path):
|
2020-05-24 01:33:58 +02:00
|
|
|
# pylint: disable=no-value-for-parameter
|
2018-12-05 21:22:06 +01:00
|
|
|
path_ = os.path.expanduser(os.path.join(*path))
|
|
|
|
return os.path.join(self.config_dir, path_)
|
|
|
|
|
|
|
|
def relative_build_path(self, *path):
|
2020-05-24 01:33:58 +02:00
|
|
|
# pylint: disable=no-value-for-parameter
|
2018-12-05 21:22:06 +01:00
|
|
|
path_ = os.path.expanduser(os.path.join(*path))
|
|
|
|
return os.path.join(self.build_path, path_)
|
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
def relative_src_path(self, *path):
|
2021-03-07 20:03:16 +01:00
|
|
|
return self.relative_build_path("src", *path)
|
2019-04-17 12:06:00 +02:00
|
|
|
|
2019-03-03 16:50:06 +01:00
|
|
|
def relative_pioenvs_path(self, *path):
|
|
|
|
if is_hassio():
|
2021-03-07 20:03:16 +01:00
|
|
|
return os.path.join("/data", self.name, ".pioenvs", *path)
|
|
|
|
return self.relative_build_path(".pioenvs", *path)
|
2019-03-03 16:50:06 +01:00
|
|
|
|
|
|
|
def relative_piolibdeps_path(self, *path):
|
|
|
|
if is_hassio():
|
2021-03-07 20:03:16 +01:00
|
|
|
return os.path.join("/data", self.name, ".piolibdeps", *path)
|
|
|
|
return self.relative_build_path(".piolibdeps", *path)
|
2019-03-03 16:50:06 +01:00
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
@property
|
|
|
|
def firmware_bin(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return self.relative_pioenvs_path(self.name, "firmware.bin")
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_esp8266(self):
|
|
|
|
if self.esp_platform is None:
|
2020-01-13 23:35:55 +01:00
|
|
|
raise ValueError("No platform specified")
|
2021-03-07 20:03:16 +01:00
|
|
|
return self.esp_platform == "ESP8266"
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_esp32(self):
|
|
|
|
if self.esp_platform is None:
|
2020-01-13 23:35:55 +01:00
|
|
|
raise ValueError("No platform specified")
|
2021-03-07 20:03:16 +01:00
|
|
|
return self.esp_platform == "ESP32"
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
def add_job(self, func, *args, **kwargs):
|
2021-05-17 07:14:15 +02:00
|
|
|
self.event_loop.add_job(func, *args, **kwargs)
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
def flush_tasks(self):
|
2021-05-17 07:14:15 +02:00
|
|
|
try:
|
|
|
|
self.event_loop.flush_tasks()
|
|
|
|
except RuntimeError as e:
|
|
|
|
raise EsphomeError(str(e)) from e
|
2019-04-22 21:56:30 +02:00
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
def add(self, expression):
|
|
|
|
from esphome.cpp_generator import Expression, Statement, statement
|
2018-12-05 21:22:06 +01:00
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
if isinstance(expression, Expression):
|
|
|
|
expression = statement(expression)
|
|
|
|
if not isinstance(expression, Statement):
|
2021-03-07 20:03:16 +01:00
|
|
|
raise ValueError(
|
|
|
|
"Add '{}' must be expression or statement, not {}"
|
|
|
|
"".format(expression, type(expression))
|
|
|
|
)
|
2018-12-05 21:22:06 +01:00
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
self.main_statements.append(expression)
|
2018-12-05 21:22:06 +01:00
|
|
|
_LOGGER.debug("Adding: %s", expression)
|
|
|
|
return expression
|
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
def add_global(self, expression):
|
|
|
|
from esphome.cpp_generator import Expression, Statement, statement
|
|
|
|
|
|
|
|
if isinstance(expression, Expression):
|
|
|
|
expression = statement(expression)
|
|
|
|
if not isinstance(expression, Statement):
|
2021-03-07 20:03:16 +01:00
|
|
|
raise ValueError(
|
|
|
|
"Add '{}' must be expression or statement, not {}"
|
|
|
|
"".format(expression, type(expression))
|
|
|
|
)
|
2019-04-17 12:06:00 +02:00
|
|
|
self.global_statements.append(expression)
|
|
|
|
_LOGGER.debug("Adding global: %s", expression)
|
|
|
|
return expression
|
|
|
|
|
|
|
|
def add_library(self, library):
|
|
|
|
if not isinstance(library, Library):
|
2021-03-07 20:03:16 +01:00
|
|
|
raise ValueError(
|
|
|
|
"Library {} must be instance of Library, not {}"
|
|
|
|
"".format(library, type(library))
|
|
|
|
)
|
2019-06-28 11:29:37 +02:00
|
|
|
for other in self.libraries[:]:
|
|
|
|
if other.name != library.name:
|
|
|
|
continue
|
2021-07-26 10:50:45 +02:00
|
|
|
if other.repository is not None:
|
|
|
|
if library.repository is None or other.repository == library.repository:
|
|
|
|
# Other is using a/the same repository, takes precendence
|
|
|
|
break
|
|
|
|
raise ValueError(
|
|
|
|
"Adding named Library with repository failed! Libraries {} and {} "
|
|
|
|
"requested with conflicting repositories!"
|
|
|
|
"".format(library, other)
|
|
|
|
)
|
|
|
|
|
|
|
|
if library.repository is not None:
|
|
|
|
# This is more specific since its using a repository
|
|
|
|
self.libraries.remove(other)
|
|
|
|
continue
|
|
|
|
|
2019-06-28 11:29:37 +02:00
|
|
|
if library.version is None:
|
|
|
|
# Other requirement is more specific
|
|
|
|
break
|
|
|
|
if other.version is None:
|
|
|
|
# Found more specific version requirement
|
|
|
|
self.libraries.remove(other)
|
|
|
|
continue
|
|
|
|
if other.version == library.version:
|
|
|
|
break
|
|
|
|
|
2021-03-07 20:03:16 +01:00
|
|
|
raise ValueError(
|
|
|
|
"Version pinning failed! Libraries {} and {} "
|
|
|
|
"requested with conflicting versions!"
|
|
|
|
"".format(library, other)
|
|
|
|
)
|
2019-06-28 11:29:37 +02:00
|
|
|
else:
|
2021-07-26 10:50:45 +02:00
|
|
|
_LOGGER.debug("Adding library: %s", library)
|
2019-06-28 11:29:37 +02:00
|
|
|
self.libraries.append(library)
|
2019-04-17 12:06:00 +02:00
|
|
|
return library
|
|
|
|
|
|
|
|
def add_build_flag(self, build_flag):
|
|
|
|
self.build_flags.add(build_flag)
|
|
|
|
_LOGGER.debug("Adding build flag: %s", build_flag)
|
|
|
|
return build_flag
|
|
|
|
|
|
|
|
def add_define(self, define):
|
2019-12-07 18:28:55 +01:00
|
|
|
if isinstance(define, str):
|
2019-04-17 12:06:00 +02:00
|
|
|
define = Define(define)
|
|
|
|
elif isinstance(define, Define):
|
|
|
|
pass
|
|
|
|
else:
|
2021-03-07 20:03:16 +01:00
|
|
|
raise ValueError(
|
|
|
|
"Define {} must be string or Define, not {}"
|
|
|
|
"".format(define, type(define))
|
|
|
|
)
|
2019-04-17 12:06:00 +02:00
|
|
|
self.defines.add(define)
|
|
|
|
_LOGGER.debug("Adding define: %s", define)
|
|
|
|
return define
|
|
|
|
|
2021-05-17 07:14:15 +02:00
|
|
|
def _get_variable_generator(self, id):
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
return self.variables[id]
|
|
|
|
except KeyError:
|
|
|
|
_LOGGER.debug("Waiting for variable %s (%r)", id, id)
|
|
|
|
yield
|
|
|
|
|
|
|
|
async def get_variable(self, id) -> "MockObj":
|
2019-04-17 12:06:00 +02:00
|
|
|
if not isinstance(id, ID):
|
2019-12-07 18:28:55 +01:00
|
|
|
raise ValueError(f"ID {id!r} must be of type ID!")
|
2021-05-17 07:14:15 +02:00
|
|
|
# Fast path, check if already registered without awaiting
|
|
|
|
if id in self.variables:
|
|
|
|
return self.variables[id]
|
|
|
|
return await _FakeAwaitable(self._get_variable_generator(id))
|
2018-12-05 21:22:06 +01:00
|
|
|
|
2021-05-17 07:14:15 +02:00
|
|
|
def _get_variable_with_full_id_generator(self, id):
|
2018-12-05 21:22:06 +01:00
|
|
|
while True:
|
|
|
|
if id in self.variables:
|
2019-01-02 14:11:11 +01:00
|
|
|
for k, v in self.variables.items():
|
2018-12-05 21:22:06 +01:00
|
|
|
if k == id:
|
2021-05-17 07:14:15 +02:00
|
|
|
return (k, v)
|
2018-12-05 21:22:06 +01:00
|
|
|
_LOGGER.debug("Waiting for variable %s", id)
|
2021-05-17 07:14:15 +02:00
|
|
|
yield
|
|
|
|
|
|
|
|
async def get_variable_with_full_id(self, id: ID) -> Tuple[ID, "MockObj"]:
|
|
|
|
if not isinstance(id, ID):
|
|
|
|
raise ValueError(f"ID {id!r} must be of type ID!")
|
|
|
|
return await _FakeAwaitable(self._get_variable_with_full_id_generator(id))
|
2018-12-05 21:22:06 +01:00
|
|
|
|
|
|
|
def register_variable(self, id, obj):
|
2019-01-29 17:29:21 +01:00
|
|
|
if id in self.variables:
|
2019-12-07 18:28:55 +01:00
|
|
|
raise EsphomeError(f"ID {id} is already registered")
|
2018-12-05 21:22:06 +01:00
|
|
|
_LOGGER.debug("Registered variable %s of type %s", id.id, id.type)
|
|
|
|
self.variables[id] = obj
|
|
|
|
|
2019-01-29 17:29:21 +01:00
|
|
|
def has_id(self, id):
|
|
|
|
return id in self.variables
|
|
|
|
|
2019-04-17 12:06:00 +02:00
|
|
|
@property
|
|
|
|
def cpp_main_section(self):
|
|
|
|
from esphome.cpp_generator import statement
|
|
|
|
|
|
|
|
main_code = []
|
|
|
|
for exp in self.main_statements:
|
2019-12-07 18:28:55 +01:00
|
|
|
text = str(statement(exp))
|
2019-04-17 12:06:00 +02:00
|
|
|
text = text.rstrip()
|
|
|
|
main_code.append(text)
|
2021-03-07 20:03:16 +01:00
|
|
|
return "\n".join(main_code) + "\n\n"
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def cpp_global_section(self):
|
|
|
|
from esphome.cpp_generator import statement
|
|
|
|
|
|
|
|
global_code = []
|
|
|
|
for exp in self.global_statements:
|
2019-12-07 18:28:55 +01:00
|
|
|
text = str(statement(exp))
|
2019-04-17 12:06:00 +02:00
|
|
|
text = text.rstrip()
|
|
|
|
global_code.append(text)
|
2021-03-07 20:03:16 +01:00
|
|
|
return "\n".join(global_code) + "\n"
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
class AutoLoad(OrderedDict):
|
2019-04-17 12:06:00 +02:00
|
|
|
pass
|
|
|
|
|
2018-12-05 21:22:06 +01:00
|
|
|
|
2019-12-07 18:28:55 +01:00
|
|
|
class EnumValue:
|
2019-04-22 21:56:30 +02:00
|
|
|
"""Special type used by ESPHome to mark enum values for cv.enum."""
|
2021-03-07 20:03:16 +01:00
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
@property
|
|
|
|
def enum_value(self):
|
2021-03-07 20:03:16 +01:00
|
|
|
return getattr(self, "_enum_value", None)
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
@enum_value.setter
|
|
|
|
def enum_value(self, value):
|
2021-03-07 20:03:16 +01:00
|
|
|
setattr(self, "_enum_value", value)
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
|
2019-02-13 16:54:02 +01:00
|
|
|
CORE = EsphomeCore()
|