From d46c3fa54a6e49e96b52260a510e1b61769773bd Mon Sep 17 00:00:00 2001 From: kotva006 Date: Fri, 3 May 2024 11:30:57 -0500 Subject: [PATCH 1/4] fix(Schedules): Fix alerts not showing up for stops with inbound trains --- lib/predictions/repo.ex | 33 ++++++++++++++++++++++----------- 1 file changed, 22 insertions(+), 11 deletions(-) diff --git a/lib/predictions/repo.ex b/lib/predictions/repo.ex index 1d66bb2f22..ca18cc0c98 100644 --- a/lib/predictions/repo.ex +++ b/lib/predictions/repo.ex @@ -24,7 +24,7 @@ defmodule Predictions.Repo do opts |> add_all_optional_params() |> cache_fetch() - |> filter_by_min_time(Keyword.get(opts, :min_time)) + |> filter_predictions(Keyword.get(opts, :min_time)) |> load_from_other_repos end @@ -32,6 +32,7 @@ defmodule Predictions.Repo do opts |> add_all_optional_params() |> fetch() + |> filter_predictions(nil) |> load_from_other_repos end @@ -53,6 +54,18 @@ defmodule Predictions.Repo do end end + @spec filter_predictions([Parser.record()] | {:error, any}, DateTime.t() | nil) :: + [Parser.record()] | {:error, any} + defp filter_predictions({:error, error}, _) do + {:error, error} + end + + defp filter_predictions(predictions, min_time) do + Enum.filter(predictions, fn prediction -> + has_departure_time?(prediction) && after_min_time?(prediction, min_time) + end) + end + defp fetch(params) do _ = Logger.info("predictions_repo_all_cache=cache_miss") @@ -98,19 +111,17 @@ defmodule Predictions.Repo do [] end - @spec filter_by_min_time([Parser.record()] | {:error, any}, DateTime.t() | nil) :: - [Parser.record()] | {:error, any} - defp filter_by_min_time({:error, error}, _) do - {:error, error} + defp has_departure_time?( + {_id, _trip_id, _stop_id, _route_id, _direction_id, _arrival, departure, _time, + _stop_sequence, _schedule_relationship, _track, _status, _departing?, + _vehicle_id} = _prediction + ) do + departure != nil end - defp filter_by_min_time(predictions, nil) do - predictions - end + defp has_departure_time?(_), do: false - defp filter_by_min_time(predictions, %DateTime{} = min_time) do - Enum.filter(predictions, &after_min_time?(&1, min_time)) - end + defp after_min_time?(_, nil), do: true defp after_min_time?( { From 74bae7d75f99b9d696829183bc2ece618cc47ff5 Mon Sep 17 00:00:00 2001 From: kotva006 Date: Fri, 3 May 2024 13:12:30 -0500 Subject: [PATCH 2/4] Added a test --- test/predictions/repo_test.exs | 62 ++++++++++++++++++++++++++++++++++ 1 file changed, 62 insertions(+) diff --git a/test/predictions/repo_test.exs b/test/predictions/repo_test.exs index 3c5840a3bf..d8263f9901 100644 --- a/test/predictions/repo_test.exs +++ b/test/predictions/repo_test.exs @@ -2,6 +2,7 @@ defmodule Predictions.RepoTest do use ExUnit.Case, async: false @moduletag :external + import Mock import Mox alias Predictions.Repo @@ -58,6 +59,67 @@ defmodule Predictions.RepoTest do end end + test "filtes out predictions with no departure" do + five_minutes_in_future = DateTime.add(Timex.now(), 5, :minute) + + {:ok, five_minutes_in_future_string} = + Timex.format(five_minutes_in_future, "{ISO:Extended:Z}") + + expect(MBTA.Api.Mock, :get_json, fn _, _ -> + %JsonApi{ + data: [ + %JsonApi.Item{ + attributes: %{ + "departure_time" => nil, + "arrival_time" => five_minutes_in_future_string, + "direction_id" => 1 + }, + relationships: %{ + "route" => [ + %{ + id: "Red" + } + ], + "trip" => [], + "vehicle" => [], + "stop" => [ + %{id: "StopID"} + ] + } + }, + %JsonApi.Item{ + attributes: %{ + "departure_time" => five_minutes_in_future_string, + "arrival_time" => five_minutes_in_future_string, + "direction_id" => 1 + }, + relationships: %{ + "route" => [ + %{ + id: "Red" + } + ], + "trip" => [], + "vehicle" => [], + "stop" => [ + %{id: "StopID"} + ] + } + } + ] + } + end) + + with_mocks([ + {Stops.Repo, [:passthrough], get_parent: fn _ -> %Stops.Stop{id: "Parent"} end}, + {Routes.Repo, [:passthrough], get: fn _ -> Routes.MockRepoApi.get("Red") end} + ]) do + predictions = Repo.all(route: "39") + + assert Kernel.length(predictions) == 1 + end + end + test "returns a list even if the server is down" do expect(MBTA.Api.Mock, :get_json, fn _, _ -> {:error, %HTTPoison.Error{reason: :econnrefused}} From bda0facce709686dede05dd0143afed21192e9f4 Mon Sep 17 00:00:00 2001 From: kotva006 Date: Tue, 7 May 2024 08:14:15 -0500 Subject: [PATCH 3/4] Fixed test to use only mox --- test/predictions/repo_test.exs | 54 +++++++++++++++++++++++++++------- 1 file changed, 44 insertions(+), 10 deletions(-) diff --git a/test/predictions/repo_test.exs b/test/predictions/repo_test.exs index d8263f9901..aa0c075940 100644 --- a/test/predictions/repo_test.exs +++ b/test/predictions/repo_test.exs @@ -2,7 +2,6 @@ defmodule Predictions.RepoTest do use ExUnit.Case, async: false @moduletag :external - import Mock import Mox alias Predictions.Repo @@ -62,8 +61,8 @@ defmodule Predictions.RepoTest do test "filtes out predictions with no departure" do five_minutes_in_future = DateTime.add(Timex.now(), 5, :minute) - {:ok, five_minutes_in_future_string} = - Timex.format(five_minutes_in_future, "{ISO:Extended:Z}") + five_minutes_in_future_string = + Timex.format!(five_minutes_in_future, "{ISO:Extended:Z}") expect(MBTA.Api.Mock, :get_json, fn _, _ -> %JsonApi{ @@ -110,14 +109,49 @@ defmodule Predictions.RepoTest do } end) - with_mocks([ - {Stops.Repo, [:passthrough], get_parent: fn _ -> %Stops.Stop{id: "Parent"} end}, - {Routes.Repo, [:passthrough], get: fn _ -> Routes.MockRepoApi.get("Red") end} - ]) do - predictions = Repo.all(route: "39") + # Route for calculating display time + expect(MBTA.Api.Mock, :get_json, fn _, _ -> + %JsonApi{ + data: [ + %JsonApi.Item{ + id: "Red", + attributes: %{ + "type" => "1", + "long_name" => "Red Test Subway", + "direction_names" => ["North Test Name", "South Test Name"], + "direction_destinations" => ["North Test", "South Test"] + }, + relationships: %{} + } + ] + } + end) - assert Kernel.length(predictions) == 1 - end + # Parent Stop for prediction + expect(MBTA.Api.Mock, :get_json, fn _, _ -> + %JsonApi{ + data: [ + %JsonApi.Item{ + id: "place-subway", + attributes: %{ + "name" => "Test Subway Stop", + "location_type" => 0, + "platform_name" => "Test Subway Platform", + "platform_code" => "Test Code", + "description" => "Test Description" + }, + relationships: %{ + "facilities" => %{}, + "zone" => "Test Zone" + } + } + ] + } + end) + + predictions = Repo.all(route: "39") + + assert Kernel.length(predictions) == 1 end test "returns a list even if the server is down" do From 4996beb909741455fcb27e279f9d475bae3a184e Mon Sep 17 00:00:00 2001 From: kotva006 Date: Wed, 8 May 2024 12:02:50 -0500 Subject: [PATCH 4/4] Rewokred test, and addressed feedback --- lib/predictions/repo.ex | 4 +- test/predictions/repo_test.exs | 158 ++++++++++++++++++--------------- 2 files changed, 88 insertions(+), 74 deletions(-) diff --git a/lib/predictions/repo.ex b/lib/predictions/repo.ex index ca18cc0c98..c6f7a9d90a 100644 --- a/lib/predictions/repo.ex +++ b/lib/predictions/repo.ex @@ -32,7 +32,7 @@ defmodule Predictions.Repo do opts |> add_all_optional_params() |> fetch() - |> filter_predictions(nil) + |> filter_predictions() |> load_from_other_repos end @@ -56,6 +56,8 @@ defmodule Predictions.Repo do @spec filter_predictions([Parser.record()] | {:error, any}, DateTime.t() | nil) :: [Parser.record()] | {:error, any} + defp filter_predictions(predictions, min_time \\ nil) + defp filter_predictions({:error, error}, _) do {:error, error} end diff --git a/test/predictions/repo_test.exs b/test/predictions/repo_test.exs index aa0c075940..019e6ce5c4 100644 --- a/test/predictions/repo_test.exs +++ b/test/predictions/repo_test.exs @@ -1,6 +1,5 @@ defmodule Predictions.RepoTest do use ExUnit.Case, async: false - @moduletag :external import Mox @@ -21,6 +20,7 @@ defmodule Predictions.RepoTest do setup :verify_on_exit! describe "all/1" do + @tag :external test "returns a list" do expect(MBTA.Api.Mock, :get_json, fn _, _ -> %JsonApi{data: []} @@ -31,6 +31,7 @@ defmodule Predictions.RepoTest do assert is_list(predictions) end + @tag :external test "can filter by trip" do expect(MBTA.Api.Mock, :get_json, fn _, _ -> %JsonApi{data: []} @@ -43,6 +44,7 @@ defmodule Predictions.RepoTest do end end + @tag :external test "filters by min_time" do min_time = Util.now() |> Timex.shift(minutes: 15) @@ -58,102 +60,104 @@ defmodule Predictions.RepoTest do end end - test "filtes out predictions with no departure" do + test "filters out predictions with no departure" do five_minutes_in_future = DateTime.add(Timex.now(), 5, :minute) five_minutes_in_future_string = Timex.format!(five_minutes_in_future, "{ISO:Extended:Z}") + # Set as constant to take advantage of caching (less calls to get_json) + route_id = Faker.Pizza.cheese() + + prediction_json = fn %{departure_time: departure_time, arrival_time: arrival_time} -> + %JsonApi.Item{ + attributes: %{ + "departure_time" => departure_time, + "arrival_time" => arrival_time, + "direction_id" => Faker.random_between(0, 1) + }, + relationships: %{ + "route" => [ + %{ + id: route_id + } + ], + "trip" => [], + "vehicle" => [], + "stop" => [ + %{id: Faker.Pizza.topping()} + ] + } + } + end + expect(MBTA.Api.Mock, :get_json, fn _, _ -> %JsonApi{ data: [ - %JsonApi.Item{ - attributes: %{ - "departure_time" => nil, - "arrival_time" => five_minutes_in_future_string, - "direction_id" => 1 - }, - relationships: %{ - "route" => [ - %{ - id: "Red" - } - ], - "trip" => [], - "vehicle" => [], - "stop" => [ - %{id: "StopID"} - ] - } - }, - %JsonApi.Item{ - attributes: %{ - "departure_time" => five_minutes_in_future_string, - "arrival_time" => five_minutes_in_future_string, - "direction_id" => 1 - }, - relationships: %{ - "route" => [ - %{ - id: "Red" - } - ], - "trip" => [], - "vehicle" => [], - "stop" => [ - %{id: "StopID"} - ] - } - } + prediction_json.(%{departure_time: nil, arrival_time: five_minutes_in_future_string}), + prediction_json.(%{ + departure_time: five_minutes_in_future_string, + arrival_time: five_minutes_in_future_string + }) ] } end) + test_stop_data = %JsonApi{ + data: [ + %JsonApi.Item{ + id: Faker.Pizza.cheese(), + attributes: %{ + "name" => Faker.Pizza.combo(), + "location_type" => Faker.random_between(0, 1), + "platform_name" => Faker.Pizza.company(), + "platform_code" => Faker.Pizza.company(), + "description" => Faker.Pizza.topping() + }, + relationships: %{ + "facilities" => %{}, + "zone" => Faker.Pizza.topping() + } + } + ] + } + + test_route_data = %JsonApi{ + data: [ + %JsonApi.Item{ + id: Faker.Pizza.cheese(), + attributes: %{ + "type" => "1", + "long_name" => Faker.Pizza.topping(), + "direction_names" => [Faker.Pizza.meat(), Faker.Pizza.meat()], + "direction_destinations" => [Faker.Pizza.company(), Faker.Pizza.company()] + }, + relationships: %{} + } + ] + } + # Route for calculating display time expect(MBTA.Api.Mock, :get_json, fn _, _ -> - %JsonApi{ - data: [ - %JsonApi.Item{ - id: "Red", - attributes: %{ - "type" => "1", - "long_name" => "Red Test Subway", - "direction_names" => ["North Test Name", "South Test Name"], - "direction_destinations" => ["North Test", "South Test"] - }, - relationships: %{} - } - ] - } + test_route_data end) # Parent Stop for prediction expect(MBTA.Api.Mock, :get_json, fn _, _ -> - %JsonApi{ - data: [ - %JsonApi.Item{ - id: "place-subway", - attributes: %{ - "name" => "Test Subway Stop", - "location_type" => 0, - "platform_name" => "Test Subway Platform", - "platform_code" => "Test Code", - "description" => "Test Description" - }, - relationships: %{ - "facilities" => %{}, - "zone" => "Test Zone" - } - } - ] - } + test_stop_data + end) + + # Route for generating struct + expect(MBTA.Api.Mock, :get_json, fn _, _ -> + test_route_data end) - predictions = Repo.all(route: "39") + predictions = Repo.all(route: Faker.Pizza.cheese()) assert Kernel.length(predictions) == 1 end + @tag :external test "returns a list even if the server is down" do expect(MBTA.Api.Mock, :get_json, fn _, _ -> {:error, %HTTPoison.Error{reason: :econnrefused}} @@ -162,6 +166,7 @@ defmodule Predictions.RepoTest do assert Repo.all(route: "test_down_server") == [] end + @tag :external test "returns valid entries even if some don't parse" do expect(MBTA.Api.Mock, :get_json, fn _, _, _ -> %JsonApi{data: []} @@ -177,6 +182,7 @@ defmodule Predictions.RepoTest do refute Repo.all(route: "Red", trip: "made_up_trip") == [] end + @tag :external test "caches trips that are retrieved", %{cache: cache} do expect(MBTA.Api.Mock, :get_json, fn _, _ -> %JsonApi{data: []} @@ -187,6 +193,7 @@ defmodule Predictions.RepoTest do assert {:ok, %Schedules.Trip{id: "trip"}} = cache.get({:trip, "trip"}) end + @tag :external test "returns an empty list if the API returns an error" do # make sure it's cached expect(MBTA.Api.Mock, :get_json, fn _, _, _ -> @@ -218,6 +225,7 @@ defmodule Predictions.RepoTest do end describe "load_from_other_repos/1" do + @tag :external test "turns a list of records into structs" do prediction = { "prediction_id", @@ -255,6 +263,7 @@ defmodule Predictions.RepoTest do end end + @tag :external test "drops prediction if stop_id is nil" do prediction = { "prediction_id", @@ -276,6 +285,7 @@ defmodule Predictions.RepoTest do assert Predictions.Repo.load_from_other_repos([prediction]) == [] end + @tag :external test "drops prediction if stop doesn't exist" do prediction = { "prediction_id", @@ -302,6 +312,7 @@ defmodule Predictions.RepoTest do assert log =~ "Discarding prediction" end + @tag :external test "drops subway prediction if it is in the past" do prediction_in_the_past = { "past_prediction", @@ -359,6 +370,7 @@ defmodule Predictions.RepoTest do assert Enum.count(total_predictions) == 2 end + @tag :external test "does not drop prediction though it is in the past" do # predictions in the past are only dropped for subway bus_prediction_in_the_past = {