Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow one encryption counter reset per 100 messages to allow changing… #117

Closed
wants to merge 8 commits into from
33 changes: 31 additions & 2 deletions src/bthome_ble/parser.py
Original file line number Diff line number Diff line change
Expand Up @@ -140,6 +140,8 @@
# advertisements is increasing, to have some replay protection. We always
# start at zero allow the first message after a restart.
self.encryption_counter = 0.0
self.reset_counter = 0 # Tracks the number of resets
self.message_since_last_reset = 0 # Tracks messages since the last reset

# The packet_id is used to filter duplicate messages in BTHome V2.
self.packet_id: float | None = None
Expand Down Expand Up @@ -645,13 +647,28 @@
raise ValueError

# filter advertisements with decreasing encryption counter.
self.message_since_last_reset += 1 # Increment messages since last reset
if (
new_encryption_counter < last_encryption_counter
and self.bindkey_verified is True
):
if new_encryption_counter < 100 and last_encryption_counter >= 4294967195:
# the counter has (most likely) restarted from 0 after reaching the highest number.
self.reset_counter += 1 # Increment the reset counter
# Replay attack protection is two-edged sword: if you don't implement it, you allow attacker to replay a whole bunch of measurements.
# If you don't give some wiggle room for allowing resets changing batteries becomes a challenge
# Beware: If attacker manages to record a message with high encryption counter number they can
# DoS all of your actual measurements until encryption counter reaches even higher value
if ((new_encryption_counter < 1000 and last_encryption_counter >= 4294967195) or
(self.reset_counter <= 1 and self.message_since_last_reset >= 100)):
# Counter reset logic: either it's a legitimate overflow reset, or a reset is allowed per policy
self.encryption_counter = new_encryption_counter
self.message_since_last_reset = 0 # Reset the message counter since the last reset
_LOGGER.warning(
"%s: The new encryption counter (%i) is lower than the previous value (%i). "
"Treating as a legitimate reset.",
self.title,
new_encryption_counter,
last_encryption_counter,
)
else:
# in all other cases, we assume the data has been comprimised and skip the
# advertisement
Expand All @@ -662,9 +679,21 @@
new_encryption_counter,
last_encryption_counter,
)
self.message_since_last_reset = 0 # Reset the message counter since the last reset
raise ValueError
else:
self.encryption_counter = new_encryption_counter


# Reset the reset_counter if a hundred messages have been received since the last reset
if self.message_since_last_reset >= 100 and self.reset_counter >= 1:
if self.reset_counter <= 1:
# There has been only one reset. No big deal. Clear it out.
self.reset_counter = 0

Check warning on line 692 in src/bthome_ble/parser.py

View check run for this annotation

Codecov / codecov/patch

src/bthome_ble/parser.py#L692

Added line #L692 was not covered by tests
if self.reset_counter > 1:
# We're under seige, there has been multiple resets - carry this knowledge over into next hundred block.
self.reset_counter = 1
self.message_since_last_reset = 0

Check warning on line 696 in src/bthome_ble/parser.py

View check run for this annotation

Codecov / codecov/patch

src/bthome_ble/parser.py#L695-L696

Added lines #L695 - L696 were not covered by tests

# decrypt the data
try:
Expand Down
Loading