From d98d9ede9cbf3794cd476f3d558a18525c2d1c1c Mon Sep 17 00:00:00 2001 From: Stefanos Mousafeiris Date: Thu, 14 Nov 2024 13:46:00 +0200 Subject: [PATCH] fix: Root table validation should return 400 (#1975) When root table is missing it should return 400 with the correct message. Currently it is returning 500 which would lead to clients retrying even through the request is malformed. --- .changeset/dirty-wolves-wash.md | 5 ++++ .../lib/electric/plug/serve_shape_plug.ex | 2 ++ .../electric/plug/serve_shape_plug_test.exs | 27 ++++++++++++++++--- 3 files changed, 31 insertions(+), 3 deletions(-) create mode 100644 .changeset/dirty-wolves-wash.md diff --git a/.changeset/dirty-wolves-wash.md b/.changeset/dirty-wolves-wash.md new file mode 100644 index 0000000000..4ffbb89725 --- /dev/null +++ b/.changeset/dirty-wolves-wash.md @@ -0,0 +1,5 @@ +--- +"@core/sync-service": patch +--- + +Fix root table parameter validation to return 400 when missing diff --git a/packages/sync-service/lib/electric/plug/serve_shape_plug.ex b/packages/sync-service/lib/electric/plug/serve_shape_plug.ex index 7fca9241f7..bad8e694b8 100644 --- a/packages/sync-service/lib/electric/plug/serve_shape_plug.ex +++ b/packages/sync-service/lib/electric/plug/serve_shape_plug.ex @@ -153,6 +153,8 @@ defmodule Electric.Plug.ServeShapePlug do end end + def cast_root_table(%Ecto.Changeset{valid?: false} = changeset, _), do: changeset + def cast_root_table(%Ecto.Changeset{} = changeset, opts) do table = fetch_change!(changeset, :table) where = fetch_field!(changeset, :where) diff --git a/packages/sync-service/test/electric/plug/serve_shape_plug_test.exs b/packages/sync-service/test/electric/plug/serve_shape_plug_test.exs index 0b98c8c1bb..cf395216a5 100644 --- a/packages/sync-service/test/electric/plug/serve_shape_plug_test.exs +++ b/packages/sync-service/test/electric/plug/serve_shape_plug_test.exs @@ -152,22 +152,43 @@ defmodule Electric.Plug.ServeShapePlugTest do :with_tenant_id ] - test "returns 400 for invalid params", ctx do + test "returns 400 for invalid table", ctx do conn = ctx - |> conn(:get, %{"table" => ".invalid_shape"}, "?offset=invalid") + |> conn(:get, %{"table" => ".invalid_shape"}, "?offset=-1") |> ServeShapePlug.call([]) assert conn.status == 400 assert Jason.decode!(conn.resp_body) == %{ - "offset" => ["has invalid format"], "table" => [ "Invalid zero-length delimited identifier" ] } end + test "returns 400 for invalid offset", ctx do + conn = + ctx + |> conn(:get, %{"table" => "foo"}, "?offset=invalid") + |> ServeShapePlug.call([]) + + assert conn.status == 400 + + assert Jason.decode!(conn.resp_body) == %{"offset" => ["has invalid format"]} + end + + test "returns 400 when table param is missing", ctx do + conn = + ctx + |> conn(:get, %{}, "?offset=-1") + |> ServeShapePlug.call([]) + + assert conn.status == 400 + + assert %{"table" => ["can't be blank"]} = Jason.decode!(conn.resp_body) + end + test "returns 400 when table does not exist", ctx do # this will pass table name validation # but will fail to find the table