Skip to content

Commit

Permalink
WIP: Publish extra active_route fields to mqtt output
Browse files Browse the repository at this point in the history
This refactors the existing code to make it more maintainable.

Nil values are published as "nil" string. This ensures that they will
get published, and ensures that MQTT doesn't drop the retained data.

This is WIP because it hasn't had any testing yet, and is likely to fail
even the CI tests :-)

Fixes #3748
  • Loading branch information
brianmay committed Mar 28, 2024
1 parent 4c56d39 commit 712de83
Show file tree
Hide file tree
Showing 2 changed files with 93 additions and 41 deletions.
123 changes: 83 additions & 40 deletions lib/teslamate/mqtt/pubsub/vehicle_subscriber.ex
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,9 @@ defmodule TeslaMate.Mqtt.PubSub.VehicleSubscriber do

alias TeslaMate.Mqtt.Publisher
alias TeslaMate.Vehicles.Vehicle.Summary
alias TeslaMate.Locations.GeoFence
alias TeslaMate.Vehicles

defstruct [:car_id, :last_summary, :deps, :namespace]
defstruct [:car_id, :last_values, :deps, :namespace]
alias __MODULE__, as: State

def child_spec(arg) do
Expand Down Expand Up @@ -39,7 +38,7 @@ defmodule TeslaMate.Mqtt.PubSub.VehicleSubscriber do
end

@impl true
def handle_info(summary, %State{last_summary: summary} = state) do
def handle_info(_summary, %State{} = state) do
{:noreply, state}
end

Expand All @@ -48,18 +47,19 @@ defmodule TeslaMate.Mqtt.PubSub.VehicleSubscriber do
time_to_full_charge shift_state geofence trim_badging)a

def handle_info(%Summary{} = summary, state) do
summary
|> Map.from_struct()
|> Map.drop([:car])
values =
summary
|> Map.from_struct()
|> Map.drop([:car])
|> add_car_latitude_longitude(summary)
|> add_geofence(summary)
|> add_active_route(summary)

values
|> Stream.reject(&match?({_key, :unknown}, &1))
|> Stream.filter(fn {key, value} ->
(key in @always_published or value != nil) and
(state.last_summary == nil or Map.get(state.last_summary, key) != value)
end)
|> Stream.map(fn
{key = :geofence, %GeoFence{name: name}} -> {key, name}
{key = :geofence, nil} -> {key, Application.get_env(:teslamate, :default_geofence)}
{key, val} -> {key, val}
(state.last_values == nil or Map.get(state.last_values, key) != value)
end)
|> Task.async_stream(&publish(&1, state),
max_concurrency: 10,
Expand All @@ -74,37 +74,80 @@ defmodule TeslaMate.Mqtt.PubSub.VehicleSubscriber do
nil
end)

if state.last_summary == nil or
state.last_summary.latitude != summary.latitude or
state.last_summary.longitude != summary.longitude do
lat_lng =
case {summary.latitude, summary.longitude} do
{nil, _} -> nil
{_, nil} -> nil
{%Decimal{} = lat, %Decimal{} = lon} -> {Decimal.to_float(lat), Decimal.to_float(lon)}
{lat, lon} -> {lat, lon}
end

case lat_lng do
nil ->
nil

{lat, lon} ->
location =
%{
latitude: lat,
longitude: lon
}
|> Jason.encode!()

case publish({"location", location}, state) do
:ok -> nil
{:error, reason} -> Logger.warning("Failed to publish location: #{inspect(reason)}")
end
{:noreply, %State{state | last_values: values}}
end

defp add_car_latitude_longitude(map, %Summary{} = summary) do
lat_lng =
case {summary.latitude, summary.longitude} do
{nil, _} -> nil
{_, nil} -> nil
{%Decimal{} = lat, %Decimal{} = lon} -> {Decimal.to_float(lat), Decimal.to_float(lon)}
{lat, lon} -> {lat, lon}
end

case lat_lng do
nil ->
map

{lat, lon} ->
location =
%{
latitude: lat,
longitude: lon
}
|> Jason.encode!()

Map.put(map, "location", location)
end
end

defp add_geofence(map, %Summary{} = summary) do
# This overwrites the existing geofence value in map.
case summary.geofence do
nil ->
Map.put(map, "geofence", Application.get_env(:teslamate, :default_geofence))

geofence ->
Map.put(map, "geofence", geofence.name)
end
end

{:noreply, %State{state | last_summary: summary}}
defp add_active_route(map, %Summary{active_route_destination: nil}) do
# This overwrites the existing values in map.
%{
map
| "active_route_destination" => "nil",
"active_route_latitude" => "nil",
"active_route_longitude" => "nil",
"active_route_energy_at_arrival" => "nil",
"active_route_miles_to_arrival" => "nil",
"active_route_minutes_to_arrival" => "nil",
"active_route_traffic_minutes_delay" => "nil",
"active_route_location" => "nil"
}
end

defp add_active_route(map, %Summary{} = summary) do
# This overwrites the existing values in map.
location =
%{
latitude: summary.active_route_latitude,
longitude: summary.active_route_longitude
}
|> Jason.encode!()

%{
map
| "active_route_destination" => summary.active_route_destination,
"active_route_latitude" => summary.active_route_latitude,
"active_route_longitude" => summary.active_route_longitude,
"active_route_energy_at_arrival" => summary.active_route_energy_at_arrival,
"active_route_miles_to_arrival" => summary.active_route_miles_to_arrival,
"active_route_minutes_to_arrival" => summary.active_route_minutes_to_arrival,
"active_route_traffic_minutes_delay" => summary.active_route_traffic_minutes_delay,
"active_route_location" => location
}
end

defp publish({key, value}, %State{car_id: car_id, namespace: namespace, deps: deps}) do
Expand Down
11 changes: 10 additions & 1 deletion lib/teslamate/vehicles/vehicle/summary.ex
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,8 @@ defmodule TeslaMate.Vehicles.Vehicle.Summary do
model trim_badging exterior_color wheel_type spoiler_type trunk_open frunk_open elevation power
charge_current_request charge_current_request_max tpms_pressure_fl tpms_pressure_fr tpms_pressure_rl tpms_pressure_rr
tpms_soft_warning_fl tpms_soft_warning_fr tpms_soft_warning_rl tpms_soft_warning_rr climate_keeper_mode
active_route_destination active_route_latitude active_route_longitude
active_route_destination active_route_latitude active_route_longitude active_route_energy_at_arrival
active_route_miles_to_arrival active_route_minutes_to_arrival active_route_traffic_minutes_delay
)a

def into(nil, %{state: :start, healthy?: healthy?, car: car}) do
Expand Down Expand Up @@ -79,6 +80,14 @@ defmodule TeslaMate.Vehicles.Vehicle.Summary do
active_route_destination: get_in_struct(vehicle, [:drive_state, :active_route_destination]),
active_route_latitude: get_in_struct(vehicle, [:drive_state, :active_route_latitude]),
active_route_longitude: get_in_struct(vehicle, [:drive_state, :active_route_longitude]),
active_route_energy_at_arrival:
get_in_struct(vehicle, [:drive_state, :active_route_energy_at_arrival]),
active_route_miles_to_arrival:
get_in_struct(vehicle, [:drive_state, :active_route_miles_to_arrival]),
active_route_minutes_to_arrival:
get_in_struct(vehicle, [:drive_state, :active_route_minutes_to_arrival]),
active_route_traffic_minutes_delay:
get_in_struct(vehicle, [:drive_state, :active_route_traffic_minutes_delay]),
latitude: get_in_struct(vehicle, [:drive_state, :latitude]),
longitude: get_in_struct(vehicle, [:drive_state, :longitude]),
power: get_in_struct(vehicle, [:drive_state, :power]),
Expand Down

0 comments on commit 712de83

Please sign in to comment.