mirror of
https://github.com/esphome/esphome.git
synced 2024-11-27 09:18:00 +01:00
commit
f0089b7940
13 changed files with 303 additions and 196 deletions
|
@ -180,7 +180,11 @@ def compile_program(args, config):
|
||||||
from esphome import platformio_api
|
from esphome import platformio_api
|
||||||
|
|
||||||
_LOGGER.info("Compiling app...")
|
_LOGGER.info("Compiling app...")
|
||||||
return platformio_api.run_compile(config, CORE.verbose)
|
rc = platformio_api.run_compile(config, CORE.verbose)
|
||||||
|
if rc != 0:
|
||||||
|
return rc
|
||||||
|
idedata = platformio_api.get_idedata(config)
|
||||||
|
return 0 if idedata is not None else 1
|
||||||
|
|
||||||
|
|
||||||
def upload_using_esptool(config, port):
|
def upload_using_esptool(config, port):
|
||||||
|
@ -458,6 +462,21 @@ def command_update_all(args):
|
||||||
return failed
|
return failed
|
||||||
|
|
||||||
|
|
||||||
|
def command_idedata(args, config):
|
||||||
|
from esphome import platformio_api
|
||||||
|
import json
|
||||||
|
|
||||||
|
logging.disable(logging.INFO)
|
||||||
|
logging.disable(logging.WARNING)
|
||||||
|
|
||||||
|
idedata = platformio_api.get_idedata(config)
|
||||||
|
if idedata is None:
|
||||||
|
return 1
|
||||||
|
|
||||||
|
print(json.dumps(idedata.raw, indent=2) + "\n")
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
PRE_CONFIG_ACTIONS = {
|
PRE_CONFIG_ACTIONS = {
|
||||||
"wizard": command_wizard,
|
"wizard": command_wizard,
|
||||||
"version": command_version,
|
"version": command_version,
|
||||||
|
@ -475,6 +494,7 @@ POST_CONFIG_ACTIONS = {
|
||||||
"clean-mqtt": command_clean_mqtt,
|
"clean-mqtt": command_clean_mqtt,
|
||||||
"mqtt-fingerprint": command_mqtt_fingerprint,
|
"mqtt-fingerprint": command_mqtt_fingerprint,
|
||||||
"clean": command_clean,
|
"clean": command_clean,
|
||||||
|
"idedata": command_idedata,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -650,6 +670,11 @@ def parse_args(argv):
|
||||||
"configuration", help="Your YAML configuration file directories.", nargs="+"
|
"configuration", help="Your YAML configuration file directories.", nargs="+"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
parser_idedata = subparsers.add_parser("idedata")
|
||||||
|
parser_idedata.add_argument(
|
||||||
|
"configuration", help="Your YAML configuration file(s).", nargs=1
|
||||||
|
)
|
||||||
|
|
||||||
# Keep backward compatibility with the old command line format of
|
# Keep backward compatibility with the old command line format of
|
||||||
# esphome <config> <command>.
|
# esphome <config> <command>.
|
||||||
#
|
#
|
||||||
|
@ -762,7 +787,7 @@ def run_esphome(argv):
|
||||||
|
|
||||||
config = read_config(dict(args.substitution) if args.substitution else {})
|
config = read_config(dict(args.substitution) if args.substitution else {})
|
||||||
if config is None:
|
if config is None:
|
||||||
return 1
|
return 2
|
||||||
CORE.config = config
|
CORE.config = config
|
||||||
|
|
||||||
if args.command not in POST_CONFIG_ACTIONS:
|
if args.command not in POST_CONFIG_ACTIONS:
|
||||||
|
|
|
@ -76,9 +76,9 @@ class ADE7953 : public i2c::I2CDevice, public PollingComponent {
|
||||||
return err;
|
return err;
|
||||||
*value = 0;
|
*value = 0;
|
||||||
*value |= ((uint32_t) recv[0]) << 24;
|
*value |= ((uint32_t) recv[0]) << 24;
|
||||||
*value |= ((uint32_t) recv[1]) << 24;
|
*value |= ((uint32_t) recv[1]) << 16;
|
||||||
*value |= ((uint32_t) recv[2]) << 24;
|
*value |= ((uint32_t) recv[2]) << 8;
|
||||||
*value |= ((uint32_t) recv[3]) << 24;
|
*value |= ((uint32_t) recv[3]);
|
||||||
return i2c::ERROR_OK;
|
return i2c::ERROR_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,6 +7,7 @@ from esphome.helpers import write_file_if_changed
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_BOARD,
|
CONF_BOARD,
|
||||||
CONF_FRAMEWORK,
|
CONF_FRAMEWORK,
|
||||||
|
CONF_SOURCE,
|
||||||
CONF_TYPE,
|
CONF_TYPE,
|
||||||
CONF_VARIANT,
|
CONF_VARIANT,
|
||||||
CONF_VERSION,
|
CONF_VERSION,
|
||||||
|
@ -53,7 +54,7 @@ def set_core_data(config):
|
||||||
elif conf[CONF_TYPE] == FRAMEWORK_ARDUINO:
|
elif conf[CONF_TYPE] == FRAMEWORK_ARDUINO:
|
||||||
CORE.data[KEY_CORE][KEY_TARGET_FRAMEWORK] = "arduino"
|
CORE.data[KEY_CORE][KEY_TARGET_FRAMEWORK] = "arduino"
|
||||||
CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION] = cv.Version.parse(
|
CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION] = cv.Version.parse(
|
||||||
config[CONF_FRAMEWORK][CONF_VERSION_HINT]
|
config[CONF_FRAMEWORK][CONF_VERSION]
|
||||||
)
|
)
|
||||||
CORE.data[KEY_ESP32][KEY_BOARD] = config[CONF_BOARD]
|
CORE.data[KEY_ESP32][KEY_BOARD] = config[CONF_BOARD]
|
||||||
CORE.data[KEY_ESP32][KEY_VARIANT] = config[CONF_VARIANT]
|
CORE.data[KEY_ESP32][KEY_VARIANT] = config[CONF_VARIANT]
|
||||||
|
@ -94,6 +95,13 @@ def _format_framework_arduino_version(ver: cv.Version) -> str:
|
||||||
return f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
return f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
||||||
|
|
||||||
|
|
||||||
|
def _format_framework_espidf_version(ver: cv.Version) -> str:
|
||||||
|
# format the given arduino (https://github.com/espressif/esp-idf/releases) version to
|
||||||
|
# a PIO platformio/framework-espidf value
|
||||||
|
# List of package versions: https://api.registry.platformio.org/v3/packages/platformio/tool/framework-espidf
|
||||||
|
return f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
||||||
|
|
||||||
|
|
||||||
# NOTE: Keep this in mind when updating the recommended version:
|
# NOTE: Keep this in mind when updating the recommended version:
|
||||||
# * New framework historically have had some regressions, especially for WiFi.
|
# * New framework historically have had some regressions, especially for WiFi.
|
||||||
# The new version needs to be thoroughly validated before changing the
|
# The new version needs to be thoroughly validated before changing the
|
||||||
|
@ -123,119 +131,97 @@ ESP_IDF_PLATFORM_VERSION = cv.Version(3, 3, 2)
|
||||||
def _arduino_check_versions(value):
|
def _arduino_check_versions(value):
|
||||||
value = value.copy()
|
value = value.copy()
|
||||||
lookups = {
|
lookups = {
|
||||||
"dev": ("https://github.com/espressif/arduino-esp32.git", cv.Version(2, 0, 0)),
|
"dev": (cv.Version(2, 0, 0), "https://github.com/espressif/arduino-esp32.git"),
|
||||||
"latest": ("", cv.Version(1, 0, 3)),
|
"latest": (cv.Version(1, 0, 6), None),
|
||||||
"recommended": (
|
"recommended": (RECOMMENDED_ARDUINO_FRAMEWORK_VERSION, None),
|
||||||
_format_framework_arduino_version(RECOMMENDED_ARDUINO_FRAMEWORK_VERSION),
|
|
||||||
RECOMMENDED_ARDUINO_FRAMEWORK_VERSION,
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
ver_value = value[CONF_VERSION]
|
|
||||||
default_ver_hint = None
|
if value[CONF_VERSION] in lookups:
|
||||||
if ver_value.lower() in lookups:
|
if CONF_SOURCE in value:
|
||||||
default_ver_hint = str(lookups[ver_value.lower()][1])
|
raise cv.Invalid(
|
||||||
ver_value = lookups[ver_value.lower()][0]
|
"Framework version needs to be explicitly specified when custom source is used."
|
||||||
|
)
|
||||||
|
|
||||||
|
version, source = lookups[value[CONF_VERSION]]
|
||||||
else:
|
else:
|
||||||
with cv.suppress_invalid():
|
version = cv.Version.parse(cv.version_number(value[CONF_VERSION]))
|
||||||
ver = cv.Version.parse(cv.version_number(value))
|
source = value.get(CONF_SOURCE, None)
|
||||||
if ver <= cv.Version(1, 0, 3):
|
|
||||||
ver_value = f"~2.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
else:
|
|
||||||
ver_value = f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
default_ver_hint = str(ver)
|
|
||||||
value[CONF_VERSION] = ver_value
|
|
||||||
|
|
||||||
if CONF_VERSION_HINT not in value and default_ver_hint is None:
|
value[CONF_VERSION] = str(version)
|
||||||
raise cv.Invalid("Needs a version hint to understand the framework version")
|
value[CONF_SOURCE] = source or _format_framework_arduino_version(version)
|
||||||
|
|
||||||
ver_hint_s = value.get(CONF_VERSION_HINT, default_ver_hint)
|
platform_version = value.get(CONF_PLATFORM_VERSION, ARDUINO_PLATFORM_VERSION)
|
||||||
value[CONF_VERSION_HINT] = ver_hint_s
|
value[CONF_PLATFORM_VERSION] = str(platform_version)
|
||||||
plat_ver = value.get(CONF_PLATFORM_VERSION, ARDUINO_PLATFORM_VERSION)
|
|
||||||
value[CONF_PLATFORM_VERSION] = str(plat_ver)
|
|
||||||
|
|
||||||
if cv.Version.parse(ver_hint_s) != RECOMMENDED_ARDUINO_FRAMEWORK_VERSION:
|
if version != RECOMMENDED_ARDUINO_FRAMEWORK_VERSION:
|
||||||
_LOGGER.warning(
|
_LOGGER.warning(
|
||||||
"The selected arduino framework version is not the recommended one"
|
"The selected Arduino framework version is not the recommended one. "
|
||||||
)
|
"If there are connectivity or build issues please remove the manual version."
|
||||||
_LOGGER.warning(
|
|
||||||
"If there are connectivity or build issues please remove the manual version"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
def _format_framework_espidf_version(ver: cv.Version) -> str:
|
|
||||||
# format the given arduino (https://github.com/espressif/esp-idf/releases) version to
|
|
||||||
# a PIO platformio/framework-espidf value
|
|
||||||
# List of package versions: https://api.registry.platformio.org/v3/packages/platformio/tool/framework-espidf
|
|
||||||
return f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
|
|
||||||
|
|
||||||
def _esp_idf_check_versions(value):
|
def _esp_idf_check_versions(value):
|
||||||
value = value.copy()
|
value = value.copy()
|
||||||
lookups = {
|
lookups = {
|
||||||
"dev": ("https://github.com/espressif/esp-idf.git", cv.Version(4, 3, 1)),
|
"dev": (cv.Version(4, 3, 1), "https://github.com/espressif/esp-idf.git"),
|
||||||
"latest": ("", cv.Version(4, 3, 0)),
|
"latest": (cv.Version(4, 3, 0), None),
|
||||||
"recommended": (
|
"recommended": (RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION, None),
|
||||||
_format_framework_espidf_version(RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION),
|
|
||||||
RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION,
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
ver_value = value[CONF_VERSION]
|
|
||||||
default_ver_hint = None
|
if value[CONF_VERSION] in lookups:
|
||||||
if ver_value.lower() in lookups:
|
if CONF_SOURCE in value:
|
||||||
default_ver_hint = str(lookups[ver_value.lower()][1])
|
raise cv.Invalid(
|
||||||
ver_value = lookups[ver_value.lower()][0]
|
"Framework version needs to be explicitly specified when custom source is used."
|
||||||
|
)
|
||||||
|
|
||||||
|
version, source = lookups[value[CONF_VERSION]]
|
||||||
else:
|
else:
|
||||||
with cv.suppress_invalid():
|
version = cv.Version.parse(cv.version_number(value[CONF_VERSION]))
|
||||||
ver = cv.Version.parse(cv.version_number(value))
|
source = value.get(CONF_SOURCE, None)
|
||||||
ver_value = f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
default_ver_hint = str(ver)
|
|
||||||
value[CONF_VERSION] = ver_value
|
|
||||||
|
|
||||||
if CONF_VERSION_HINT not in value and default_ver_hint is None:
|
if version < cv.Version(4, 0, 0):
|
||||||
raise cv.Invalid("Needs a version hint to understand the framework version")
|
raise cv.Invalid("Only ESP-IDF 4.0+ is supported.")
|
||||||
|
|
||||||
ver_hint_s = value.get(CONF_VERSION_HINT, default_ver_hint)
|
value[CONF_VERSION] = str(version)
|
||||||
value[CONF_VERSION_HINT] = ver_hint_s
|
value[CONF_SOURCE] = source or _format_framework_espidf_version(version)
|
||||||
if cv.Version.parse(ver_hint_s) < cv.Version(4, 0, 0):
|
|
||||||
raise cv.Invalid("Only ESP-IDF 4.0+ is supported")
|
platform_version = value.get(CONF_PLATFORM_VERSION, ESP_IDF_PLATFORM_VERSION)
|
||||||
if cv.Version.parse(ver_hint_s) != RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION:
|
value[CONF_PLATFORM_VERSION] = str(platform_version)
|
||||||
|
|
||||||
|
if version != RECOMMENDED_ARDUINO_FRAMEWORK_VERSION:
|
||||||
_LOGGER.warning(
|
_LOGGER.warning(
|
||||||
"The selected esp-idf framework version is not the recommended one"
|
"The selected ESP-IDF framework version is not the recommended one. "
|
||||||
|
"If there are connectivity or build issues please remove the manual version."
|
||||||
)
|
)
|
||||||
_LOGGER.warning(
|
|
||||||
"If there are connectivity or build issues please remove the manual version"
|
|
||||||
)
|
|
||||||
|
|
||||||
plat_ver = value.get(CONF_PLATFORM_VERSION, ESP_IDF_PLATFORM_VERSION)
|
|
||||||
value[CONF_PLATFORM_VERSION] = str(plat_ver)
|
|
||||||
|
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
CONF_VERSION_HINT = "version_hint"
|
|
||||||
CONF_PLATFORM_VERSION = "platform_version"
|
CONF_PLATFORM_VERSION = "platform_version"
|
||||||
|
|
||||||
ARDUINO_FRAMEWORK_SCHEMA = cv.All(
|
ARDUINO_FRAMEWORK_SCHEMA = cv.All(
|
||||||
cv.Schema(
|
cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
|
cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
|
||||||
cv.Optional(CONF_VERSION_HINT): cv.version_number,
|
cv.Optional(CONF_SOURCE): cv.string_strict,
|
||||||
cv.Optional(CONF_PLATFORM_VERSION): cv.string_strict,
|
cv.Optional(CONF_PLATFORM_VERSION): cv.string_strict,
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
_arduino_check_versions,
|
_arduino_check_versions,
|
||||||
)
|
)
|
||||||
|
|
||||||
CONF_SDKCONFIG_OPTIONS = "sdkconfig_options"
|
CONF_SDKCONFIG_OPTIONS = "sdkconfig_options"
|
||||||
ESP_IDF_FRAMEWORK_SCHEMA = cv.All(
|
ESP_IDF_FRAMEWORK_SCHEMA = cv.All(
|
||||||
cv.Schema(
|
cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
|
cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
|
||||||
cv.Optional(CONF_VERSION_HINT): cv.version_number,
|
cv.Optional(CONF_SOURCE): cv.string_strict,
|
||||||
|
cv.Optional(CONF_PLATFORM_VERSION): cv.string_strict,
|
||||||
cv.Optional(CONF_SDKCONFIG_OPTIONS, default={}): {
|
cv.Optional(CONF_SDKCONFIG_OPTIONS, default={}): {
|
||||||
cv.string_strict: cv.string_strict
|
cv.string_strict: cv.string_strict
|
||||||
},
|
},
|
||||||
cv.Optional(CONF_PLATFORM_VERSION): cv.string_strict,
|
|
||||||
cv.Optional(CONF_ADVANCED, default={}): cv.Schema(
|
cv.Optional(CONF_ADVANCED, default={}): cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Optional(CONF_IGNORE_EFUSE_MAC_CRC, default=False): cv.boolean,
|
cv.Optional(CONF_IGNORE_EFUSE_MAC_CRC, default=False): cv.boolean,
|
||||||
|
@ -293,7 +279,7 @@ async def to_code(config):
|
||||||
cg.add_build_flag("-Wno-nonnull-compare")
|
cg.add_build_flag("-Wno-nonnull-compare")
|
||||||
cg.add_platformio_option(
|
cg.add_platformio_option(
|
||||||
"platform_packages",
|
"platform_packages",
|
||||||
[f"platformio/framework-espidf @ {conf[CONF_VERSION]}"],
|
[f"platformio/framework-espidf @ {conf[CONF_SOURCE]}"],
|
||||||
)
|
)
|
||||||
add_idf_sdkconfig_option("CONFIG_PARTITION_TABLE_SINGLE_APP", False)
|
add_idf_sdkconfig_option("CONFIG_PARTITION_TABLE_SINGLE_APP", False)
|
||||||
add_idf_sdkconfig_option("CONFIG_PARTITION_TABLE_CUSTOM", True)
|
add_idf_sdkconfig_option("CONFIG_PARTITION_TABLE_CUSTOM", True)
|
||||||
|
@ -323,7 +309,7 @@ async def to_code(config):
|
||||||
cg.add_build_flag("-DUSE_ESP32_FRAMEWORK_ARDUINO")
|
cg.add_build_flag("-DUSE_ESP32_FRAMEWORK_ARDUINO")
|
||||||
cg.add_platformio_option(
|
cg.add_platformio_option(
|
||||||
"platform_packages",
|
"platform_packages",
|
||||||
[f"platformio/framework-arduinoespressif32 @ {conf[CONF_VERSION]}"],
|
[f"platformio/framework-arduinoespressif32 @ {conf[CONF_SOURCE]}"],
|
||||||
)
|
)
|
||||||
|
|
||||||
cg.add_platformio_option("board_build.partitions", "partitions.csv")
|
cg.add_platformio_option("board_build.partitions", "partitions.csv")
|
||||||
|
|
|
@ -4,6 +4,7 @@ from esphome.const import (
|
||||||
CONF_BOARD,
|
CONF_BOARD,
|
||||||
CONF_BOARD_FLASH_MODE,
|
CONF_BOARD_FLASH_MODE,
|
||||||
CONF_FRAMEWORK,
|
CONF_FRAMEWORK,
|
||||||
|
CONF_SOURCE,
|
||||||
CONF_VERSION,
|
CONF_VERSION,
|
||||||
KEY_CORE,
|
KEY_CORE,
|
||||||
KEY_FRAMEWORK_VERSION,
|
KEY_FRAMEWORK_VERSION,
|
||||||
|
@ -31,7 +32,7 @@ def set_core_data(config):
|
||||||
CORE.data[KEY_CORE][KEY_TARGET_PLATFORM] = "esp8266"
|
CORE.data[KEY_CORE][KEY_TARGET_PLATFORM] = "esp8266"
|
||||||
CORE.data[KEY_CORE][KEY_TARGET_FRAMEWORK] = "arduino"
|
CORE.data[KEY_CORE][KEY_TARGET_FRAMEWORK] = "arduino"
|
||||||
CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION] = cv.Version.parse(
|
CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION] = cv.Version.parse(
|
||||||
config[CONF_FRAMEWORK][CONF_VERSION_HINT]
|
config[CONF_FRAMEWORK][CONF_VERSION]
|
||||||
)
|
)
|
||||||
CORE.data[KEY_ESP8266][KEY_BOARD] = config[CONF_BOARD]
|
CORE.data[KEY_ESP8266][KEY_BOARD] = config[CONF_BOARD]
|
||||||
return config
|
return config
|
||||||
|
@ -70,66 +71,50 @@ ARDUINO_3_PLATFORM_VERSION = cv.Version(3, 0, 2)
|
||||||
def _arduino_check_versions(value):
|
def _arduino_check_versions(value):
|
||||||
value = value.copy()
|
value = value.copy()
|
||||||
lookups = {
|
lookups = {
|
||||||
"dev": ("https://github.com/esp8266/Arduino.git", cv.Version(3, 0, 2)),
|
"dev": (cv.Version(3, 0, 2), "https://github.com/esp8266/Arduino.git"),
|
||||||
"latest": ("", cv.Version(3, 0, 2)),
|
"latest": (cv.Version(3, 0, 2), None),
|
||||||
"recommended": (
|
"recommended": (RECOMMENDED_ARDUINO_FRAMEWORK_VERSION, None),
|
||||||
_format_framework_arduino_version(RECOMMENDED_ARDUINO_FRAMEWORK_VERSION),
|
|
||||||
RECOMMENDED_ARDUINO_FRAMEWORK_VERSION,
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
ver_value = value[CONF_VERSION]
|
|
||||||
default_ver_hint = None
|
if value[CONF_VERSION] in lookups:
|
||||||
if ver_value.lower() in lookups:
|
if CONF_SOURCE in value:
|
||||||
default_ver_hint = str(lookups[ver_value.lower()][1])
|
raise cv.Invalid(
|
||||||
ver_value = lookups[ver_value.lower()][0]
|
"Framework version needs to be explicitly specified when custom source is used."
|
||||||
|
)
|
||||||
|
|
||||||
|
version, source = lookups[value[CONF_VERSION]]
|
||||||
else:
|
else:
|
||||||
with cv.suppress_invalid():
|
version = cv.Version.parse(cv.version_number(value[CONF_VERSION]))
|
||||||
ver = cv.Version.parse(cv.version_number(value))
|
source = value.get(CONF_SOURCE, None)
|
||||||
if ver <= cv.Version(2, 4, 1):
|
|
||||||
ver_value = f"~1.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
elif ver <= cv.Version(2, 6, 2):
|
|
||||||
ver_value = f"~2.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
else:
|
|
||||||
ver_value = f"~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0"
|
|
||||||
default_ver_hint = str(ver)
|
|
||||||
|
|
||||||
value[CONF_VERSION] = ver_value
|
value[CONF_VERSION] = str(version)
|
||||||
|
value[CONF_SOURCE] = source or _format_framework_arduino_version(version)
|
||||||
|
|
||||||
if CONF_VERSION_HINT not in value and default_ver_hint is None:
|
platform_version = value.get(CONF_PLATFORM_VERSION)
|
||||||
raise cv.Invalid("Needs a version hint to understand the framework version")
|
if platform_version is None:
|
||||||
|
if version >= cv.Version(3, 0, 0):
|
||||||
ver_hint_s = value.get(CONF_VERSION_HINT, default_ver_hint)
|
platform_version = ARDUINO_3_PLATFORM_VERSION
|
||||||
value[CONF_VERSION_HINT] = ver_hint_s
|
elif version >= cv.Version(2, 5, 0):
|
||||||
plat_ver = value.get(CONF_PLATFORM_VERSION)
|
platform_version = ARDUINO_2_PLATFORM_VERSION
|
||||||
|
|
||||||
if plat_ver is None:
|
|
||||||
ver_hint = cv.Version.parse(ver_hint_s)
|
|
||||||
if ver_hint >= cv.Version(3, 0, 0):
|
|
||||||
plat_ver = ARDUINO_3_PLATFORM_VERSION
|
|
||||||
elif ver_hint >= cv.Version(2, 5, 0):
|
|
||||||
plat_ver = ARDUINO_2_PLATFORM_VERSION
|
|
||||||
else:
|
else:
|
||||||
plat_ver = cv.Version(1, 8, 0)
|
platform_version = cv.Version(1, 8, 0)
|
||||||
value[CONF_PLATFORM_VERSION] = str(plat_ver)
|
value[CONF_PLATFORM_VERSION] = str(platform_version)
|
||||||
|
|
||||||
if cv.Version.parse(ver_hint_s) != RECOMMENDED_ARDUINO_FRAMEWORK_VERSION:
|
if version != RECOMMENDED_ARDUINO_FRAMEWORK_VERSION:
|
||||||
_LOGGER.warning(
|
_LOGGER.warning(
|
||||||
"The selected arduino framework version is not the recommended one"
|
"The selected Arduino framework version is not the recommended one. "
|
||||||
)
|
"If there are connectivity or build issues please remove the manual version."
|
||||||
_LOGGER.warning(
|
|
||||||
"If there are connectivity or build issues please remove the manual version"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
CONF_VERSION_HINT = "version_hint"
|
|
||||||
CONF_PLATFORM_VERSION = "platform_version"
|
CONF_PLATFORM_VERSION = "platform_version"
|
||||||
ARDUINO_FRAMEWORK_SCHEMA = cv.All(
|
ARDUINO_FRAMEWORK_SCHEMA = cv.All(
|
||||||
cv.Schema(
|
cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
|
cv.Optional(CONF_VERSION, default="recommended"): cv.string_strict,
|
||||||
cv.Optional(CONF_VERSION_HINT): cv.version_number,
|
cv.Optional(CONF_SOURCE): cv.string_strict,
|
||||||
cv.Optional(CONF_PLATFORM_VERSION): cv.string_strict,
|
cv.Optional(CONF_PLATFORM_VERSION): cv.string_strict,
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
|
@ -167,7 +152,7 @@ async def to_code(config):
|
||||||
cg.add_build_flag("-DUSE_ESP8266_FRAMEWORK_ARDUINO")
|
cg.add_build_flag("-DUSE_ESP8266_FRAMEWORK_ARDUINO")
|
||||||
cg.add_platformio_option(
|
cg.add_platformio_option(
|
||||||
"platform_packages",
|
"platform_packages",
|
||||||
[f"platformio/framework-arduinoespressif8266 @ {conf[CONF_VERSION]}"],
|
[f"platformio/framework-arduinoespressif8266 @ {conf[CONF_SOURCE]}"],
|
||||||
)
|
)
|
||||||
cg.add_platformio_option(
|
cg.add_platformio_option(
|
||||||
"platform", f"platformio/espressif8266 @ {conf[CONF_PLATFORM_VERSION]}"
|
"platform", f"platformio/espressif8266 @ {conf[CONF_PLATFORM_VERSION]}"
|
||||||
|
|
|
@ -38,7 +38,7 @@ ModbusRegisterType_ns = modbus_controller_ns.namespace("ModbusRegisterType")
|
||||||
ModbusRegisterType = ModbusRegisterType_ns.enum("ModbusRegisterType")
|
ModbusRegisterType = ModbusRegisterType_ns.enum("ModbusRegisterType")
|
||||||
MODBUS_REGISTER_TYPE = {
|
MODBUS_REGISTER_TYPE = {
|
||||||
"coil": ModbusRegisterType.COIL,
|
"coil": ModbusRegisterType.COIL,
|
||||||
"discrete_input": ModbusRegisterType.DISCRETE,
|
"discrete_input": ModbusRegisterType.DISCRETE_INPUT,
|
||||||
"holding": ModbusRegisterType.HOLDING,
|
"holding": ModbusRegisterType.HOLDING,
|
||||||
"read": ModbusRegisterType.READ,
|
"read": ModbusRegisterType.READ,
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ from esphome.const import (
|
||||||
CONF_BRIGHTNESS,
|
CONF_BRIGHTNESS,
|
||||||
CONF_TRIGGER_ID,
|
CONF_TRIGGER_ID,
|
||||||
)
|
)
|
||||||
|
from esphome.core import CORE
|
||||||
from . import Nextion, nextion_ns, nextion_ref
|
from . import Nextion, nextion_ns, nextion_ref
|
||||||
from .base_component import (
|
from .base_component import (
|
||||||
CONF_ON_SLEEP,
|
CONF_ON_SLEEP,
|
||||||
|
@ -76,6 +76,9 @@ async def to_code(config):
|
||||||
if CONF_TFT_URL in config:
|
if CONF_TFT_URL in config:
|
||||||
cg.add_define("USE_NEXTION_TFT_UPLOAD")
|
cg.add_define("USE_NEXTION_TFT_UPLOAD")
|
||||||
cg.add(var.set_tft_url(config[CONF_TFT_URL]))
|
cg.add(var.set_tft_url(config[CONF_TFT_URL]))
|
||||||
|
if CORE.is_esp32:
|
||||||
|
cg.add_library("WiFiClientSecure", None)
|
||||||
|
cg.add_library("HTTPClient", None)
|
||||||
|
|
||||||
if CONF_TOUCH_SLEEP_TIMEOUT in config:
|
if CONF_TOUCH_SLEEP_TIMEOUT in config:
|
||||||
cg.add(var.set_touch_sleep_timeout_internal(config[CONF_TOUCH_SLEEP_TIMEOUT]))
|
cg.add(var.set_touch_sleep_timeout_internal(config[CONF_TOUCH_SLEEP_TIMEOUT]))
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#include "scd4x.h"
|
#include "scd4x.h"
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
|
@ -38,6 +39,7 @@ void SCD4XComponent::setup() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint32_t stop_measurement_delay = 0;
|
||||||
// In order to query the device periodic measurement must be ceased
|
// In order to query the device periodic measurement must be ceased
|
||||||
if (raw_read_status[0]) {
|
if (raw_read_status[0]) {
|
||||||
ESP_LOGD(TAG, "Sensor has data available, stopping periodic measurement");
|
ESP_LOGD(TAG, "Sensor has data available, stopping periodic measurement");
|
||||||
|
@ -46,68 +48,72 @@ void SCD4XComponent::setup() {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
// According to the SCD4x datasheet the sensor will only respond to other commands after waiting 500 ms after
|
||||||
|
// issuing the stop_periodic_measurement command
|
||||||
|
stop_measurement_delay = 500;
|
||||||
}
|
}
|
||||||
|
this->set_timeout(stop_measurement_delay, [this]() {
|
||||||
|
if (!this->write_command_(SCD4X_CMD_GET_SERIAL_NUMBER)) {
|
||||||
|
ESP_LOGE(TAG, "Failed to write get serial command");
|
||||||
|
this->error_code_ = COMMUNICATION_FAILED;
|
||||||
|
this->mark_failed();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!this->write_command_(SCD4X_CMD_GET_SERIAL_NUMBER)) {
|
uint16_t raw_serial_number[3];
|
||||||
ESP_LOGE(TAG, "Failed to write get serial command");
|
if (!this->read_data_(raw_serial_number, 3)) {
|
||||||
this->error_code_ = COMMUNICATION_FAILED;
|
ESP_LOGE(TAG, "Failed to read serial number");
|
||||||
this->mark_failed();
|
this->error_code_ = SERIAL_NUMBER_IDENTIFICATION_FAILED;
|
||||||
return;
|
this->mark_failed();
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
|
ESP_LOGD(TAG, "Serial number %02d.%02d.%02d", (uint16_t(raw_serial_number[0]) >> 8),
|
||||||
|
uint16_t(raw_serial_number[0] & 0xFF), (uint16_t(raw_serial_number[1]) >> 8));
|
||||||
|
|
||||||
uint16_t raw_serial_number[3];
|
if (!this->write_command_(SCD4X_CMD_TEMPERATURE_OFFSET,
|
||||||
if (!this->read_data_(raw_serial_number, 3)) {
|
(uint16_t)(temperature_offset_ * SCD4X_TEMPERATURE_OFFSET_MULTIPLIER))) {
|
||||||
ESP_LOGE(TAG, "Failed to read serial number");
|
ESP_LOGE(TAG, "Error setting temperature offset.");
|
||||||
this->error_code_ = SERIAL_NUMBER_IDENTIFICATION_FAILED;
|
|
||||||
this->mark_failed();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
ESP_LOGD(TAG, "Serial number %02d.%02d.%02d", (uint16_t(raw_serial_number[0]) >> 8),
|
|
||||||
uint16_t(raw_serial_number[0] & 0xFF), (uint16_t(raw_serial_number[1]) >> 8));
|
|
||||||
|
|
||||||
if (!this->write_command_(SCD4X_CMD_TEMPERATURE_OFFSET,
|
|
||||||
(uint16_t)(temperature_offset_ * SCD4X_TEMPERATURE_OFFSET_MULTIPLIER))) {
|
|
||||||
ESP_LOGE(TAG, "Error setting temperature offset.");
|
|
||||||
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
|
||||||
this->mark_failed();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If pressure compensation available use it
|
|
||||||
// else use altitude
|
|
||||||
if (ambient_pressure_compensation_) {
|
|
||||||
if (!this->write_command_(SCD4X_CMD_AMBIENT_PRESSURE_COMPENSATION, ambient_pressure_compensation_)) {
|
|
||||||
ESP_LOGE(TAG, "Error setting ambient pressure compensation.");
|
|
||||||
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
if (!this->write_command_(SCD4X_CMD_ALTITUDE_COMPENSATION, altitude_compensation_)) {
|
// If pressure compensation available use it
|
||||||
ESP_LOGE(TAG, "Error setting altitude compensation.");
|
// else use altitude
|
||||||
|
if (ambient_pressure_compensation_) {
|
||||||
|
if (!this->update_ambient_pressure_compensation_(ambient_pressure_)) {
|
||||||
|
ESP_LOGE(TAG, "Error setting ambient pressure compensation.");
|
||||||
|
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
||||||
|
this->mark_failed();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!this->write_command_(SCD4X_CMD_ALTITUDE_COMPENSATION, altitude_compensation_)) {
|
||||||
|
ESP_LOGE(TAG, "Error setting altitude compensation.");
|
||||||
|
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
||||||
|
this->mark_failed();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this->write_command_(SCD4X_CMD_AUTOMATIC_SELF_CALIBRATION, enable_asc_ ? 1 : 0)) {
|
||||||
|
ESP_LOGE(TAG, "Error setting automatic self calibration.");
|
||||||
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (!this->write_command_(SCD4X_CMD_AUTOMATIC_SELF_CALIBRATION, enable_asc_ ? 1 : 0)) {
|
// Finally start sensor measurements
|
||||||
ESP_LOGE(TAG, "Error setting automatic self calibration.");
|
if (!this->write_command_(SCD4X_CMD_START_CONTINUOUS_MEASUREMENTS)) {
|
||||||
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
ESP_LOGE(TAG, "Error starting continuous measurements.");
|
||||||
this->mark_failed();
|
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
||||||
return;
|
this->mark_failed();
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Finally start sensor measurements
|
initialized_ = true;
|
||||||
if (!this->write_command_(SCD4X_CMD_START_CONTINUOUS_MEASUREMENTS)) {
|
ESP_LOGD(TAG, "Sensor initialized");
|
||||||
ESP_LOGE(TAG, "Error starting continuous measurements.");
|
});
|
||||||
this->error_code_ = MEASUREMENT_INIT_FAILED;
|
|
||||||
this->mark_failed();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
initialized_ = true;
|
|
||||||
ESP_LOGD(TAG, "Sensor initialized");
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -150,6 +156,13 @@ void SCD4XComponent::update() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this->ambient_pressure_source_ != nullptr) {
|
||||||
|
float pressure = this->ambient_pressure_source_->state / 1000.0f;
|
||||||
|
if (!std::isnan(pressure)) {
|
||||||
|
set_ambient_pressure_compensation(this->ambient_pressure_source_->state / 1000.0f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check if data is ready
|
// Check if data is ready
|
||||||
if (!this->write_command_(SCD4X_CMD_GET_DATA_READY_STATUS)) {
|
if (!this->write_command_(SCD4X_CMD_GET_DATA_READY_STATUS)) {
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
|
@ -191,6 +204,28 @@ void SCD4XComponent::update() {
|
||||||
|
|
||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
}
|
}
|
||||||
|
// Note pressure in bar here. Convert to hPa
|
||||||
|
void SCD4XComponent::set_ambient_pressure_compensation(float pressure_in_bar) {
|
||||||
|
ambient_pressure_compensation_ = true;
|
||||||
|
uint16_t new_ambient_pressure = (uint16_t)(pressure_in_bar * 1000);
|
||||||
|
// remove millibar from comparison to avoid frequent updates +/- 10 millibar doesn't matter
|
||||||
|
if (initialized_ && (new_ambient_pressure / 10 != ambient_pressure_ / 10)) {
|
||||||
|
update_ambient_pressure_compensation_(new_ambient_pressure);
|
||||||
|
ambient_pressure_ = new_ambient_pressure;
|
||||||
|
} else {
|
||||||
|
ESP_LOGD(TAG, "ambient pressure compensation skipped - no change required");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SCD4XComponent::update_ambient_pressure_compensation_(uint16_t pressure_in_hpa) {
|
||||||
|
if (this->write_command_(SCD4X_CMD_AMBIENT_PRESSURE_COMPENSATION, pressure_in_hpa)) {
|
||||||
|
ESP_LOGD(TAG, "setting ambient pressure compensation to %d hPa", pressure_in_hpa);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
ESP_LOGE(TAG, "Error setting ambient pressure compensation.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
uint8_t SCD4XComponent::sht_crc_(uint8_t data1, uint8_t data2) {
|
uint8_t SCD4XComponent::sht_crc_(uint8_t data1, uint8_t data2) {
|
||||||
uint8_t bit;
|
uint8_t bit;
|
||||||
|
|
|
@ -18,10 +18,8 @@ class SCD4XComponent : public PollingComponent, public i2c::I2CDevice {
|
||||||
|
|
||||||
void set_automatic_self_calibration(bool asc) { enable_asc_ = asc; }
|
void set_automatic_self_calibration(bool asc) { enable_asc_ = asc; }
|
||||||
void set_altitude_compensation(uint16_t altitude) { altitude_compensation_ = altitude; }
|
void set_altitude_compensation(uint16_t altitude) { altitude_compensation_ = altitude; }
|
||||||
void set_ambient_pressure_compensation(float pressure) {
|
void set_ambient_pressure_compensation(float pressure_in_bar);
|
||||||
ambient_pressure_compensation_ = true;
|
void set_ambient_pressure_source(sensor::Sensor *pressure) { ambient_pressure_source_ = pressure; }
|
||||||
ambient_pressure_ = (uint16_t)(pressure * 1000);
|
|
||||||
}
|
|
||||||
void set_temperature_offset(float offset) { temperature_offset_ = offset; };
|
void set_temperature_offset(float offset) { temperature_offset_ = offset; };
|
||||||
|
|
||||||
void set_co2_sensor(sensor::Sensor *co2) { co2_sensor_ = co2; }
|
void set_co2_sensor(sensor::Sensor *co2) { co2_sensor_ = co2; }
|
||||||
|
@ -33,6 +31,7 @@ class SCD4XComponent : public PollingComponent, public i2c::I2CDevice {
|
||||||
bool read_data_(uint16_t *data, uint8_t len);
|
bool read_data_(uint16_t *data, uint8_t len);
|
||||||
bool write_command_(uint16_t command);
|
bool write_command_(uint16_t command);
|
||||||
bool write_command_(uint16_t command, uint16_t data);
|
bool write_command_(uint16_t command, uint16_t data);
|
||||||
|
bool update_ambient_pressure_compensation_(uint16_t pressure_in_hpa);
|
||||||
|
|
||||||
ERRORCODE error_code_;
|
ERRORCODE error_code_;
|
||||||
|
|
||||||
|
@ -47,6 +46,8 @@ class SCD4XComponent : public PollingComponent, public i2c::I2CDevice {
|
||||||
sensor::Sensor *co2_sensor_{nullptr};
|
sensor::Sensor *co2_sensor_{nullptr};
|
||||||
sensor::Sensor *temperature_sensor_{nullptr};
|
sensor::Sensor *temperature_sensor_{nullptr};
|
||||||
sensor::Sensor *humidity_sensor_{nullptr};
|
sensor::Sensor *humidity_sensor_{nullptr};
|
||||||
|
// used for compensation
|
||||||
|
sensor::Sensor *ambient_pressure_source_{nullptr};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace scd4x
|
} // namespace scd4x
|
||||||
|
|
|
@ -29,6 +29,7 @@ CONF_AUTOMATIC_SELF_CALIBRATION = "automatic_self_calibration"
|
||||||
CONF_ALTITUDE_COMPENSATION = "altitude_compensation"
|
CONF_ALTITUDE_COMPENSATION = "altitude_compensation"
|
||||||
CONF_AMBIENT_PRESSURE_COMPENSATION = "ambient_pressure_compensation"
|
CONF_AMBIENT_PRESSURE_COMPENSATION = "ambient_pressure_compensation"
|
||||||
CONF_TEMPERATURE_OFFSET = "temperature_offset"
|
CONF_TEMPERATURE_OFFSET = "temperature_offset"
|
||||||
|
CONF_AMBIENT_PRESSURE_COMPENSATION_SOURCE = "ambient_pressure_compensation_source"
|
||||||
|
|
||||||
CONFIG_SCHEMA = (
|
CONFIG_SCHEMA = (
|
||||||
cv.Schema(
|
cv.Schema(
|
||||||
|
@ -62,6 +63,9 @@ CONFIG_SCHEMA = (
|
||||||
),
|
),
|
||||||
cv.Optional(CONF_AMBIENT_PRESSURE_COMPENSATION): cv.pressure,
|
cv.Optional(CONF_AMBIENT_PRESSURE_COMPENSATION): cv.pressure,
|
||||||
cv.Optional(CONF_TEMPERATURE_OFFSET, default="4°C"): cv.temperature,
|
cv.Optional(CONF_TEMPERATURE_OFFSET, default="4°C"): cv.temperature,
|
||||||
|
cv.Optional(CONF_AMBIENT_PRESSURE_COMPENSATION_SOURCE): cv.use_id(
|
||||||
|
sensor.Sensor
|
||||||
|
),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
.extend(cv.polling_component_schema("60s"))
|
.extend(cv.polling_component_schema("60s"))
|
||||||
|
@ -92,7 +96,10 @@ async def to_code(config):
|
||||||
cg.add(getattr(var, funcName)(config[key]))
|
cg.add(getattr(var, funcName)(config[key]))
|
||||||
|
|
||||||
for key, funcName in SENSOR_MAP.items():
|
for key, funcName in SENSOR_MAP.items():
|
||||||
|
|
||||||
if key in config:
|
if key in config:
|
||||||
sens = await sensor.new_sensor(config[key])
|
sens = await sensor.new_sensor(config[key])
|
||||||
cg.add(getattr(var, funcName)(sens))
|
cg.add(getattr(var, funcName)(sens))
|
||||||
|
|
||||||
|
if CONF_AMBIENT_PRESSURE_COMPENSATION_SOURCE in config:
|
||||||
|
sens = await cg.get_variable(config[CONF_AMBIENT_PRESSURE_COMPENSATION_SOURCE])
|
||||||
|
cg.add(var.set_ambient_pressure_source(sens))
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
"""Constants used by esphome."""
|
"""Constants used by esphome."""
|
||||||
|
|
||||||
__version__ = "2021.10.0b3"
|
__version__ = "2021.10.0b4"
|
||||||
|
|
||||||
ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
|
ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,7 @@ from esphome.const import (
|
||||||
CONF_PLATFORMIO_OPTIONS,
|
CONF_PLATFORMIO_OPTIONS,
|
||||||
CONF_PRIORITY,
|
CONF_PRIORITY,
|
||||||
CONF_PROJECT,
|
CONF_PROJECT,
|
||||||
|
CONF_SOURCE,
|
||||||
CONF_TRIGGER_ID,
|
CONF_TRIGGER_ID,
|
||||||
CONF_TYPE,
|
CONF_TYPE,
|
||||||
CONF_VERSION,
|
CONF_VERSION,
|
||||||
|
@ -181,10 +182,12 @@ def preload_core_config(config, result):
|
||||||
if CONF_BOARD_FLASH_MODE in conf:
|
if CONF_BOARD_FLASH_MODE in conf:
|
||||||
plat_conf[CONF_BOARD_FLASH_MODE] = conf.pop(CONF_BOARD_FLASH_MODE)
|
plat_conf[CONF_BOARD_FLASH_MODE] = conf.pop(CONF_BOARD_FLASH_MODE)
|
||||||
if CONF_ARDUINO_VERSION in conf:
|
if CONF_ARDUINO_VERSION in conf:
|
||||||
plat_conf[CONF_FRAMEWORK] = {
|
plat_conf[CONF_FRAMEWORK] = {CONF_TYPE: "arduino"}
|
||||||
CONF_TYPE: "arduino",
|
try:
|
||||||
CONF_VERSION: conf.pop(CONF_ARDUINO_VERSION),
|
cv.Version.parse(conf[CONF_ARDUINO_VERSION])
|
||||||
}
|
plat_conf[CONF_FRAMEWORK][CONF_VERSION] = conf.pop(CONF_ARDUINO_VERSION)
|
||||||
|
except ValueError:
|
||||||
|
plat_conf[CONF_FRAMEWORK][CONF_SOURCE] = conf.pop(CONF_ARDUINO_VERSION)
|
||||||
if CONF_BOARD in conf:
|
if CONF_BOARD in conf:
|
||||||
plat_conf[CONF_BOARD] = conf.pop(CONF_BOARD)
|
plat_conf[CONF_BOARD] = conf.pop(CONF_BOARD)
|
||||||
# Insert generated target platform config to main config
|
# Insert generated target platform config to main config
|
||||||
|
|
|
@ -9,6 +9,7 @@ import json
|
||||||
import logging
|
import logging
|
||||||
import multiprocessing
|
import multiprocessing
|
||||||
import os
|
import os
|
||||||
|
from pathlib import Path
|
||||||
import secrets
|
import secrets
|
||||||
import shutil
|
import shutil
|
||||||
import subprocess
|
import subprocess
|
||||||
|
@ -26,7 +27,7 @@ import tornado.process
|
||||||
import tornado.web
|
import tornado.web
|
||||||
import tornado.websocket
|
import tornado.websocket
|
||||||
|
|
||||||
from esphome import const, util
|
from esphome import const, platformio_api, util
|
||||||
from esphome.helpers import mkdir_p, get_bool_env, run_system_command
|
from esphome.helpers import mkdir_p, get_bool_env, run_system_command
|
||||||
from esphome.storage_json import (
|
from esphome.storage_json import (
|
||||||
EsphomeStorageJSON,
|
EsphomeStorageJSON,
|
||||||
|
@ -398,17 +399,45 @@ class DownloadBinaryRequestHandler(BaseHandler):
|
||||||
@authenticated
|
@authenticated
|
||||||
@bind_config
|
@bind_config
|
||||||
def get(self, configuration=None):
|
def get(self, configuration=None):
|
||||||
# pylint: disable=no-value-for-parameter
|
type = self.get_argument("type", "firmware.bin")
|
||||||
storage_path = ext_storage_path(settings.config_dir, configuration)
|
|
||||||
storage_json = StorageJSON.load(storage_path)
|
if type == "firmware.bin":
|
||||||
if storage_json is None:
|
storage_path = ext_storage_path(settings.config_dir, configuration)
|
||||||
self.send_error()
|
storage_json = StorageJSON.load(storage_path)
|
||||||
|
if storage_json is None:
|
||||||
|
self.send_error(404)
|
||||||
|
return
|
||||||
|
filename = f"{storage_json.name}.bin"
|
||||||
|
path = storage_json.firmware_bin_path
|
||||||
|
|
||||||
|
else:
|
||||||
|
args = ["esphome", "idedata", settings.rel_path(configuration)]
|
||||||
|
rc, stdout, _ = run_system_command(*args)
|
||||||
|
|
||||||
|
if rc != 0:
|
||||||
|
self.send_error(404 if rc == 2 else 500)
|
||||||
|
return
|
||||||
|
|
||||||
|
idedata = platformio_api.IDEData(json.loads(stdout))
|
||||||
|
|
||||||
|
found = False
|
||||||
|
for image in idedata.extra_flash_images:
|
||||||
|
if image.path.endswith(type):
|
||||||
|
path = image.path
|
||||||
|
filename = type
|
||||||
|
found = True
|
||||||
|
break
|
||||||
|
|
||||||
|
if not found:
|
||||||
|
self.send_error(404)
|
||||||
|
return
|
||||||
|
|
||||||
|
self.set_header("Content-Type", "application/octet-stream")
|
||||||
|
self.set_header("Content-Disposition", f'attachment; filename="{filename}"')
|
||||||
|
if not Path(path).is_file():
|
||||||
|
self.send_error(404)
|
||||||
return
|
return
|
||||||
|
|
||||||
path = storage_json.firmware_bin_path
|
|
||||||
self.set_header("Content-Type", "application/octet-stream")
|
|
||||||
filename = f"{storage_json.name}.bin"
|
|
||||||
self.set_header("Content-Disposition", f'attachment; filename="{filename}"')
|
|
||||||
with open(path, "rb") as f:
|
with open(path, "rb") as f:
|
||||||
while True:
|
while True:
|
||||||
data = f.read(16384)
|
data = f.read(16384)
|
||||||
|
@ -418,6 +447,38 @@ class DownloadBinaryRequestHandler(BaseHandler):
|
||||||
self.finish()
|
self.finish()
|
||||||
|
|
||||||
|
|
||||||
|
class ManifestRequestHandler(BaseHandler):
|
||||||
|
@authenticated
|
||||||
|
@bind_config
|
||||||
|
def get(self, configuration=None):
|
||||||
|
args = ["esphome", "idedata", settings.rel_path(configuration)]
|
||||||
|
rc, stdout, _ = run_system_command(*args)
|
||||||
|
|
||||||
|
if rc != 0:
|
||||||
|
self.send_error(404 if rc == 2 else 500)
|
||||||
|
return
|
||||||
|
|
||||||
|
idedata = platformio_api.IDEData(json.loads(stdout))
|
||||||
|
|
||||||
|
firmware_offset = "0x10000" if idedata.extra_flash_images else "0x0"
|
||||||
|
flash_images = [
|
||||||
|
{
|
||||||
|
"path": f"./download.bin?configuration={configuration}&type=firmware.bin",
|
||||||
|
"offset": firmware_offset,
|
||||||
|
}
|
||||||
|
] + [
|
||||||
|
{
|
||||||
|
"path": f"./download.bin?configuration={configuration}&type={os.path.basename(image.path)}",
|
||||||
|
"offset": image.offset,
|
||||||
|
}
|
||||||
|
for image in idedata.extra_flash_images
|
||||||
|
]
|
||||||
|
|
||||||
|
self.set_header("Content-Type", "application/json")
|
||||||
|
self.write(json.dumps(flash_images))
|
||||||
|
self.finish()
|
||||||
|
|
||||||
|
|
||||||
def _list_dashboard_entries():
|
def _list_dashboard_entries():
|
||||||
files = settings.list_yaml_files()
|
files = settings.list_yaml_files()
|
||||||
return [DashboardEntry(file) for file in files]
|
return [DashboardEntry(file) for file in files]
|
||||||
|
@ -862,6 +923,7 @@ def make_app(debug=get_bool_env(ENV_DEV)):
|
||||||
(f"{rel}info", InfoRequestHandler),
|
(f"{rel}info", InfoRequestHandler),
|
||||||
(f"{rel}edit", EditRequestHandler),
|
(f"{rel}edit", EditRequestHandler),
|
||||||
(f"{rel}download.bin", DownloadBinaryRequestHandler),
|
(f"{rel}download.bin", DownloadBinaryRequestHandler),
|
||||||
|
(f"{rel}manifest.json", ManifestRequestHandler),
|
||||||
(f"{rel}serial-ports", SerialPortRequestHandler),
|
(f"{rel}serial-ports", SerialPortRequestHandler),
|
||||||
(f"{rel}ping", PingRequestHandler),
|
(f"{rel}ping", PingRequestHandler),
|
||||||
(f"{rel}delete", DeleteRequestHandler),
|
(f"{rel}delete", DeleteRequestHandler),
|
||||||
|
|
|
@ -9,7 +9,7 @@ pyserial==3.5
|
||||||
platformio==5.2.1
|
platformio==5.2.1
|
||||||
esptool==3.1
|
esptool==3.1
|
||||||
click==8.0.3
|
click==8.0.3
|
||||||
esphome-dashboard==20211011.1
|
esphome-dashboard==20211015.0
|
||||||
aioesphomeapi==9.1.5
|
aioesphomeapi==9.1.5
|
||||||
|
|
||||||
# esp-idf requires this, but doesn't bundle it by default
|
# esp-idf requires this, but doesn't bundle it by default
|
||||||
|
|
Loading…
Reference in a new issue