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

Validateur NeTEx : quelques metadata #4160

Merged
merged 4 commits into from
Sep 5, 2024
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
31 changes: 19 additions & 12 deletions apps/transport/lib/validators/enroute_chouette_valid_client.ex
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,9 @@ defmodule Transport.EnRouteChouetteValidClient.Wrapper do
@callback create_a_validation(Path.t()) :: binary()
@callback get_a_validation(binary()) ::
{:pending, integer()}
| {:successful, binary()}
| :warning
| :failed
| {:successful, binary(), integer()}
| {:warning, integer()}
| {:failed, integer()}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Je me suis demandé initialement si ce tuple où on ajoute un entier en bout de chaîne ne commençait pas à devenir trop long, ne risquait pas de poser de problème quand on fait évoluer cette structure etc.

Mais au final ma compréhension du code actuel est que c'est une structure éphémère non persistée en base et qui pourra donc être revue sans souci (ex: besoin de passer à une map avec des clés plus explicites).

Je consigne juste cette note pour ma compréhension !

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

En effet, c'est bien ça.

Disons que c'est une manifestation de mon habitude de manipuler des types algébriques en entrée ou sortie de toute fonction et d'apprécier d'avoir une documentation implicite via les types des embranchements possibles.

Le fait que l'entier que je rajoute dans cette PR ne soit pas identifié comme une durée en seconde est une limite de cette syntaxe, et on pourrait arguer que le spec ainsi obtenu devient dangereusement verbeux.

(I miss my algebraic types.)

| :unexpected_validation_status
| :unexpected_datetime_format
@callback get_messages(binary()) :: {binary(), map()}
Expand Down Expand Up @@ -48,28 +48,35 @@ defmodule Transport.EnRouteChouetteValidClient do

case response |> Map.fetch!("user_status") do
"pending" ->
case {get_datetime(response, "created_at"), get_datetime(response, "updated_at")} do
{{:ok, created_at, _}, {:ok, updated_at, _}} ->
{:pending, DateTime.diff(updated_at, created_at)}

_ ->
:unexpected_datetime_format
case get_elapsed(response) do
nil -> :unexpected_datetime_format
elapsed -> {:pending, elapsed}
end

"successful" ->
{:successful, url}
{:successful, url, get_elapsed(response)}

"warning" ->
:warning
{:warning, get_elapsed(response)}

"failed" ->
:failed
{:failed, get_elapsed(response)}

_ ->
:unexpected_validation_status
end
end

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pour ci-dessous : j'ai tendance à préférer des appels en bang "!" qui lèvent une erreur, pour réduire le code, et du coup réduire les potentialités de bug dans le code qui analyse les retours.

Par exemple ici j'aurais opté pour quelque chose comme:

defp get_elapsed!(response) do
  created_at = get_datetime!(response, "created_at")
  updated_at = get_datetime!(response, "updated_at")
  DateTime.diff(updated_at, created_at)
end

Et (voir https://hexdocs.pm/elixir/1.17.2/Date.html#from_iso8601!/2)

defp get_datetime!(map, key) do
  map 
  |> Map.fetch!(key)
  |> DateTime.from_iso8601!()
end

C'est d'ailleurs en phase avec le fait que Map.fetch! va de toute façon lever une exception en cas de clé manquante, et adopte donc plutôt un style "fail fast" aussi.

(Et du coup, gérer l'erreur dans l'appelant de get_elapsed!)

Après ça reste aussi une opinion, mais perso je trouve que ça fait au final beaucoup moins de code sur une codebase quand c'est applicable (il faut que l'appelant puisse intercepter l'erreur, par contre !)

Copy link
Contributor Author

@ptitfred ptitfred Sep 5, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

il faut que l'appelant puisse intercepter l'erreur, par contre !

Oui, et tu peux avoir des cas où la clef est manquante/invalide sans que ça soit grave, pas forcément la responsabilité du code appelant.

En effet si j'étais cohérent j'aurais pas utilisé Map.fetch!. En pratique je peux faire confiance au fait que le datetime est bien formatté. Ce cas d'erreur spécifique m'ennuie.

Je dois revoir cette partie de toute façon, j'ai identifié un problème en retestant avec la validation OnDemand.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Et (voir https://hexdocs.pm/elixir/1.17.2/Date.html#from_iso8601!/2)

Ce helper est fourni pour Date mais pas pour DateTime ; je vais l'émuler.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

J'imagine que c'est lié au fait que DateTime.from_iso8601 renvoie et le datetime et l'offset, et donc qu'il faudrait l'interpréter également (ce que je ne fais pas, je fais l'hypothèse que l'offset est constant).

defp get_elapsed(response) do
case {get_datetime(response, "created_at"), get_datetime(response, "updated_at")} do
{{:ok, created_at, _}, {:ok, updated_at, _}} ->
DateTime.diff(updated_at, created_at)

_ ->
nil
end
end

@impl Transport.EnRouteChouetteValidClient.Wrapper
def get_messages(validation_id) do
url = Path.join([validation_url(validation_id), "messages"])
Expand Down
52 changes: 33 additions & 19 deletions apps/transport/lib/validators/netex_validator.ex
Original file line number Diff line number Diff line change
Expand Up @@ -27,19 +27,26 @@ defmodule Transport.Validators.NeTEx do

def validate_resource_history(resource_history, filepath) do
case validate_with_enroute(filepath) do
{:ok, result_url} ->
insert_validation_results(resource_history.id, result_url)
{:ok, %{url: result_url, elapsed_seconds: elapsed_seconds, retries: retries}} ->
insert_validation_results(resource_history.id, result_url, %{elapsed_seconds: elapsed_seconds, retries: retries})

:ok

{:error, {result_url, errors}} ->
insert_validation_results(resource_history.id, result_url, errors)
{:error, %{details: {result_url, errors}, elapsed_seconds: elapsed_seconds, retries: retries}} ->
insert_validation_results(
resource_history.id,
result_url,
%{elapsed_seconds: elapsed_seconds, retries: retries},
errors
)

:ok

{:error, :unexpected_validation_status} ->
Logger.error("Invalid API call to enRoute Chouette Valid")
{:error, "enRoute Chouette Valid: Unexpected validation status"}

{:error, :timeout} ->
{:error, %{message: :timeout, retries: _retries}} ->
Logger.error("Timeout while fetching result on enRoute Chouette Valid")
{:error, "enRoute Chouette Valid: Timeout while fetching results"}
end
Expand All @@ -58,23 +65,29 @@ defmodule Transport.Validators.NeTEx do
def validate(url, opts \\ []) do
with_url(url, fn filepath ->
case validate_with_enroute(filepath, opts) do
{:ok, result_url} ->
{:ok, %{url: result_url, elapsed_seconds: elapsed_seconds, retries: retries}} ->
# result_url in metadata?
Logger.info("Result URL: #{result_url}")
{:ok, %{"validations" => index_messages([]), "metadata" => %{}}}

{:error, {result_url, errors}} ->
{:ok,
%{"validations" => index_messages([]), "metadata" => %{elapsed_seconds: elapsed_seconds, retries: retries}}}

{:error, %{details: {result_url, errors}, elapsed_seconds: elapsed_seconds, retries: retries}} ->
Logger.info("Result URL: #{result_url}")
# result_url in metadata?
{:ok, %{"validations" => index_messages(errors), "metadata" => %{}}}
{:ok,
%{
"validations" => index_messages(errors),
"metadata" => %{elapsed_seconds: elapsed_seconds, retries: retries}
}}

{:error, :unexpected_validation_status} ->
Logger.error("Invalid API call to enRoute Chouette Valid")
{:error, "enRoute Chouette Valid: Unexpected validation status"}
{:error, %{message: "enRoute Chouette Valid: Unexpected validation status"}}

{:error, :timeout} ->
{:error, %{message: :timeout, retries: retries}} ->
Logger.error("Timeout while fetching result on enRoute Chouette Valid")
{:error, "enRoute Chouette Valid: Timeout while fetching results"}
{:error, %{message: "enRoute Chouette Valid: Timeout while fetching results", retries: retries}}
end
end)
end
Expand Down Expand Up @@ -110,7 +123,7 @@ defmodule Transport.Validators.NeTEx do
System.tmp_dir!() |> Path.join("enroute_validation_netex_#{Ecto.UUID.generate()}")
end

def insert_validation_results(resource_history_id, result_url, errors \\ []) do
def insert_validation_results(resource_history_id, result_url, metadata, errors \\ []) do
result = index_messages(errors)

%DB.MultiValidation{
Expand All @@ -120,7 +133,8 @@ defmodule Transport.Validators.NeTEx do
resource_history_id: resource_history_id,
validator_version: validator_version(),
command: result_url,
max_error: get_max_severity_error(result)
max_error: get_max_severity_error(result),
metadata: %DB.ResourceMetadata{metadata: metadata}
}
|> DB.Repo.insert!()
end
Expand Down Expand Up @@ -201,7 +215,7 @@ defmodule Transport.Validators.NeTEx do
defp fetch_validation_results(validation_id, retries, opts) do
case client().get_a_validation(validation_id) do
{:pending, elapsed_seconds} when elapsed_seconds > @timeout ->
{:error, :timeout}
{:error, %{message: :timeout, retries: retries}}

{:pending, _elapsed_seconds} ->
if Keyword.get(opts, :graceful_retry, true) do
Expand All @@ -210,11 +224,11 @@ defmodule Transport.Validators.NeTEx do

fetch_validation_results(validation_id, retries + 1, opts)

{:successful, url} ->
{:ok, url}
{:successful, url, elapsed_seconds} ->
{:ok, %{url: url, elapsed_seconds: elapsed_seconds, retries: retries}}

value when value in [:warning, :failed] ->
{:error, client().get_messages(validation_id)}
{value, elapsed_seconds} when value in [:warning, :failed] ->
{:error, %{details: client().get_messages(validation_id), elapsed_seconds: elapsed_seconds, retries: retries}}

:unexpected_validation_status ->
{:error, :unexpected_validation_status}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,7 @@ defmodule Transport.EnRouteChouetteValidClientTest do
"started_at": "2024-07-05T14:41:20.680Z",
"ended_at": "2024-07-05T14:41:20.685Z",
"created_at": "2024-07-05T14:41:19.933Z",
"updated_at": "2024-07-05T14:41:19.933Z"
"updated_at": "2024-07-05T14:41:24.933Z"
}
"""

Expand All @@ -89,7 +89,7 @@ defmodule Transport.EnRouteChouetteValidClientTest do
%HTTPoison.Response{status_code: 200, body: response_body}
end)

assert {:successful, url} == EnRouteChouetteValidClient.get_a_validation(validation_id)
assert {:successful, url, 5} == EnRouteChouetteValidClient.get_a_validation(validation_id)
end

test "warning" do
Expand All @@ -104,7 +104,7 @@ defmodule Transport.EnRouteChouetteValidClientTest do
"started_at": "2024-07-05T14:41:20.680Z",
"ended_at": "2024-07-05T14:41:20.685Z",
"created_at": "2024-07-05T14:41:19.933Z",
"updated_at": "2024-07-05T14:41:19.933Z"
"updated_at": "2024-07-05T14:41:23.933Z"
}
"""

Expand All @@ -115,7 +115,7 @@ defmodule Transport.EnRouteChouetteValidClientTest do
%HTTPoison.Response{status_code: 200, body: response_body}
end)

assert :warning == EnRouteChouetteValidClient.get_a_validation(validation_id)
assert {:warning, 4} == EnRouteChouetteValidClient.get_a_validation(validation_id)
end

test "failed" do
Expand All @@ -130,7 +130,7 @@ defmodule Transport.EnRouteChouetteValidClientTest do
"started_at": "2024-07-05T14:41:20.680Z",
"ended_at": "2024-07-05T14:41:20.685Z",
"created_at": "2024-07-05T14:41:19.933Z",
"updated_at": "2024-07-05T14:41:19.933Z"
"updated_at": "2024-07-05T14:41:27.933Z"
}
"""

Expand All @@ -141,7 +141,7 @@ defmodule Transport.EnRouteChouetteValidClientTest do
%HTTPoison.Response{status_code: 200, body: response_body}
end)

assert :failed == EnRouteChouetteValidClient.get_a_validation(validation_id)
assert {:failed, 8} == EnRouteChouetteValidClient.get_a_validation(validation_id)
end
end

Expand Down
39 changes: 24 additions & 15 deletions apps/transport/test/transport/validators/netex_validator_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -39,33 +39,35 @@ defmodule Transport.Validators.NeTExTest do
{resource, resource_history} = mk_netex_resource()

validation_id = expect_create_validation()
expect_successful_validation(validation_id)
expect_successful_validation(validation_id, 12)

assert :ok == NeTEx.validate_and_save(resource)

multi_validation = DB.MultiValidation |> DB.Repo.get_by(resource_history_id: resource_history.id)
multi_validation = load_multi_validation(resource_history.id)

assert multi_validation.command == "http://localhost:9999/chouette-valid/#{validation_id}"
assert multi_validation.validator == "enroute-chouette-netex-validator"
assert multi_validation.validator_version == "saas-production"
assert multi_validation.result == %{}
assert multi_validation.metadata.metadata == %{"retries" => 0, "elapsed_seconds" => 12}
end

test "invalid NeTEx" do
{resource, resource_history} = mk_netex_resource()

validation_id = expect_create_validation()
expect_failed_validation(validation_id)
expect_failed_validation(validation_id, 31)

expect_get_messages(validation_id, @sample_error_messages)

assert :ok == NeTEx.validate_and_save(resource)

multi_validation = DB.MultiValidation |> DB.Repo.get_by(resource_history_id: resource_history.id)
multi_validation = load_multi_validation(resource_history.id)

assert multi_validation.command == "http://localhost:9999/chouette-valid/#{validation_id}/messages"
assert multi_validation.validator == "enroute-chouette-netex-validator"
assert multi_validation.validator_version == "saas-production"
assert multi_validation.metadata.metadata == %{"retries" => 0, "elapsed_seconds" => 31}

assert multi_validation.result == %{
"uic-operating-period" => [
Expand All @@ -91,24 +93,30 @@ defmodule Transport.Validators.NeTExTest do
]
}
end

defp load_multi_validation(resource_history_id) do
DB.MultiValidation
|> DB.Repo.get_by(resource_history_id: resource_history_id)
|> DB.Repo.preload(:metadata)
end
end

describe "raw URL" do
test "valid NeTEx" do
resource_url = mk_raw_netex_resource()

validation_id = expect_create_validation()
expect_successful_validation(validation_id)
expect_successful_validation(validation_id, 9)

assert {:ok, %{"validations" => %{}, "metadata" => %{}}} ==
assert {:ok, %{"validations" => %{}, "metadata" => %{retries: 0, elapsed_seconds: 9}}} ==
NeTEx.validate(resource_url)
end

test "invalid NeTEx" do
resource_url = mk_raw_netex_resource()

validation_id = expect_create_validation()
expect_failed_validation(validation_id)
expect_failed_validation(validation_id, 25)

expect_get_messages(validation_id, @sample_error_messages)

Expand Down Expand Up @@ -136,7 +144,8 @@ defmodule Transport.Validators.NeTExTest do
]
}

assert {:ok, %{"validations" => validation_result, "metadata" => %{}}} == NeTEx.validate(resource_url)
assert {:ok, %{"validations" => validation_result, "metadata" => %{retries: 0, elapsed_seconds: 25}}} ==
NeTEx.validate(resource_url)
end

test "retries" do
Expand All @@ -146,7 +155,7 @@ defmodule Transport.Validators.NeTExTest do
expect_pending_validation(validation_id, 10)
expect_pending_validation(validation_id, 20)
expect_pending_validation(validation_id, 30)
expect_failed_validation(validation_id)
expect_failed_validation(validation_id, 35)

expect_get_messages(validation_id, @sample_error_message)

Expand All @@ -162,7 +171,7 @@ defmodule Transport.Validators.NeTExTest do

# Let's disable graceful retry as we are mocking the API, otherwise the
# test would take almost a minute.
assert {:ok, %{"validations" => validation_result, "metadata" => %{}}} ==
assert {:ok, %{"validations" => validation_result, "metadata" => %{retries: 3, elapsed_seconds: 35}}} ==
NeTEx.validate(resource_url, graceful_retry: false)
end

Expand All @@ -175,7 +184,7 @@ defmodule Transport.Validators.NeTExTest do

{result, log} = with_log(fn -> NeTEx.validate(resource_url, graceful_retry: false) end)

assert result == {:error, "enRoute Chouette Valid: Timeout while fetching results"}
assert result == {:error, %{message: "enRoute Chouette Valid: Timeout while fetching results", retries: 1}}
assert log =~ "[error] Timeout while fetching result on enRoute Chouette Valid"
end
end
Expand All @@ -190,14 +199,14 @@ defmodule Transport.Validators.NeTExTest do
expect(Transport.EnRouteChouetteValidClient.Mock, :get_a_validation, fn ^validation_id -> {:pending, elapsed} end)
end

defp expect_successful_validation(validation_id) do
defp expect_successful_validation(validation_id, elapsed) do
expect(Transport.EnRouteChouetteValidClient.Mock, :get_a_validation, fn ^validation_id ->
{:successful, "http://localhost:9999/chouette-valid/#{validation_id}"}
{:successful, "http://localhost:9999/chouette-valid/#{validation_id}", elapsed}
end)
end

defp expect_failed_validation(validation_id) do
expect(Transport.EnRouteChouetteValidClient.Mock, :get_a_validation, fn ^validation_id -> :failed end)
defp expect_failed_validation(validation_id, elapsed) do
expect(Transport.EnRouteChouetteValidClient.Mock, :get_a_validation, fn ^validation_id -> {:failed, elapsed} end)
end

defp expect_get_messages(validation_id, result) do
Expand Down