Merge branch 'webfinger-regex' into 'develop'

Enforce an exact domain match for WebFinger resolution

See merge request pleroma/pleroma!4380
This commit is contained in:
feld 2025-07-03 19:51:11 +00:00
commit 977097e870
4 changed files with 139 additions and 58 deletions

View file

@ -0,0 +1 @@
Enforce an exact domain match for WebFinger resolution

View file

@ -35,9 +35,9 @@ defmodule Pleroma.Web.WebFinger do
regex = regex =
if webfinger_domain = Pleroma.Config.get([__MODULE__, :domain]) do if webfinger_domain = Pleroma.Config.get([__MODULE__, :domain]) do
~r/(acct:)?(?<username>[a-z0-9A-Z_\.-]+)@(#{host}|#{webfinger_domain})/ ~r/(acct:)?(?<username>[a-z0-9A-Z_\.-]+)@(#{host}|#{webfinger_domain})$/
else else
~r/(acct:)?(?<username>[a-z0-9A-Z_\.-]+)@#{host}/ ~r/(acct:)?(?<username>[a-z0-9A-Z_\.-]+)@#{host}$/
end end
with %{"username" => username} <- Regex.named_captures(regex, resource), with %{"username" => username} <- Regex.named_captures(regex, resource),

View file

@ -33,47 +33,122 @@ defmodule Pleroma.Web.WebFinger.WebFingerControllerTest do
assert match?(^response_xml, expected_xml) assert match?(^response_xml, expected_xml)
end end
test "Webfinger JRD" do describe "Webfinger" do
user = test "JRD" do
insert(:user, clear_config([Pleroma.Web.Endpoint, :url, :host], "hyrule.world")
ap_id: "https://hyrule.world/users/zelda", clear_config([Pleroma.Web.WebFinger, :domain], "hyrule.world")
also_known_as: ["https://mushroom.kingdom/users/toad"]
)
response = user =
build_conn() insert(:user,
|> put_req_header("accept", "application/jrd+json") ap_id: "https://hyrule.world/users/zelda"
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@localhost") )
|> json_response(200)
assert response["subject"] == "acct:#{user.nickname}@localhost" response =
build_conn()
|> put_req_header("accept", "application/jrd+json")
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@hyrule.world")
|> json_response(200)
assert response["aliases"] == [ assert response["subject"] == "acct:#{user.nickname}@hyrule.world"
"https://hyrule.world/users/zelda",
"https://mushroom.kingdom/users/toad" assert response["aliases"] == [
] "https://hyrule.world/users/zelda"
]
end
test "XML" do
clear_config([Pleroma.Web.Endpoint, :url, :host], "hyrule.world")
clear_config([Pleroma.Web.WebFinger, :domain], "hyrule.world")
user =
insert(:user,
ap_id: "https://hyrule.world/users/zelda"
)
response =
build_conn()
|> put_req_header("accept", "application/xrd+xml")
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@localhost")
|> response(200)
assert response =~ "<Alias>https://hyrule.world/users/zelda</Alias>"
end
end end
test "Webfinger defaults to JSON when no Accept header is provided" do test "Webfinger defaults to JSON when no Accept header is provided" do
clear_config([Pleroma.Web.Endpoint, :url, :host], "hyrule.world")
clear_config([Pleroma.Web.WebFinger, :domain], "hyrule.world")
user = user =
insert(:user, insert(:user,
ap_id: "https://hyrule.world/users/zelda", ap_id: "https://hyrule.world/users/zelda"
also_known_as: ["https://mushroom.kingdom/users/toad"]
) )
response = response =
build_conn() build_conn()
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@localhost") |> get("/.well-known/webfinger?resource=acct:#{user.nickname}@hyrule.world")
|> json_response(200) |> json_response(200)
assert response["subject"] == "acct:#{user.nickname}@localhost" assert response["subject"] == "acct:#{user.nickname}@hyrule.world"
assert response["aliases"] == [ assert response["aliases"] == [
"https://hyrule.world/users/zelda", "https://hyrule.world/users/zelda"
"https://mushroom.kingdom/users/toad"
] ]
end end
describe "Webfinger returns also_known_as / aliases in the response" do
test "JSON" do
clear_config([Pleroma.Web.Endpoint, :url, :host], "hyrule.world")
clear_config([Pleroma.Web.WebFinger, :domain], "hyrule.world")
user =
insert(:user,
ap_id: "https://hyrule.world/users/zelda",
also_known_as: [
"https://mushroom.kingdom/users/toad",
"https://luigi.mansion/users/kingboo"
]
)
response =
build_conn()
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@hyrule.world")
|> json_response(200)
assert response["subject"] == "acct:#{user.nickname}@hyrule.world"
assert response["aliases"] == [
"https://hyrule.world/users/zelda",
"https://mushroom.kingdom/users/toad",
"https://luigi.mansion/users/kingboo"
]
end
test "XML" do
clear_config([Pleroma.Web.Endpoint, :url, :host], "hyrule.world")
clear_config([Pleroma.Web.WebFinger, :domain], "hyrule.world")
user =
insert(:user,
ap_id: "https://hyrule.world/users/zelda",
also_known_as: [
"https://mushroom.kingdom/users/toad",
"https://luigi.mansion/users/kingboo"
]
)
response =
build_conn()
|> put_req_header("accept", "application/xrd+xml")
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@localhost")
|> response(200)
assert response =~ "<Alias>https://hyrule.world/users/zelda</Alias>"
assert response =~ "<Alias>https://mushroom.kingdom/users/toad</Alias>"
assert response =~ "<Alias>https://luigi.mansion/users/kingboo</Alias>"
end
end
test "reach user on tld, while pleroma is running on subdomain" do test "reach user on tld, while pleroma is running on subdomain" do
clear_config([Pleroma.Web.Endpoint, :url, :host], "sub.example.com") clear_config([Pleroma.Web.Endpoint, :url, :host], "sub.example.com")
@ -91,44 +166,32 @@ defmodule Pleroma.Web.WebFinger.WebFingerControllerTest do
assert response["aliases"] == ["https://sub.example.com/users/#{user.nickname}"] assert response["aliases"] == ["https://sub.example.com/users/#{user.nickname}"]
end end
test "it returns 404 when user isn't found (JSON)" do describe "it returns 404 when user isn't found" do
result = test "JSON" do
build_conn() result =
|> put_req_header("accept", "application/jrd+json") build_conn()
|> get("/.well-known/webfinger?resource=acct:jimm@localhost") |> put_req_header("accept", "application/jrd+json")
|> json_response(404) |> get("/.well-known/webfinger?resource=acct:jimm@localhost")
|> json_response(404)
assert result == "Couldn't find user" assert result == "Couldn't find user"
end end
test "Webfinger XML" do test "XML" do
user = result =
insert(:user, build_conn()
ap_id: "https://hyrule.world/users/zelda", |> put_req_header("accept", "application/xrd+xml")
also_known_as: ["https://mushroom.kingdom/users/toad"] |> get("/.well-known/webfinger?resource=acct:jimm@localhost")
) |> response(404)
response = assert result == "Couldn't find user"
build_conn() end
|> put_req_header("accept", "application/xrd+xml")
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@localhost")
|> response(200)
assert response =~ "<Alias>https://hyrule.world/users/zelda</Alias>"
assert response =~ "<Alias>https://mushroom.kingdom/users/toad</Alias>"
end
test "it returns 404 when user isn't found (XML)" do
result =
build_conn()
|> put_req_header("accept", "application/xrd+xml")
|> get("/.well-known/webfinger?resource=acct:jimm@localhost")
|> response(404)
assert result == "Couldn't find user"
end end
test "Returns JSON when format is not supported" do test "Returns JSON when format is not supported" do
clear_config([Pleroma.Web.Endpoint, :url, :host], "hyrule.world")
clear_config([Pleroma.Web.WebFinger, :domain], "hyrule.world")
user = user =
insert(:user, insert(:user,
ap_id: "https://hyrule.world/users/zelda", ap_id: "https://hyrule.world/users/zelda",
@ -138,10 +201,10 @@ defmodule Pleroma.Web.WebFinger.WebFingerControllerTest do
response = response =
build_conn() build_conn()
|> put_req_header("accept", "text/html") |> put_req_header("accept", "text/html")
|> get("/.well-known/webfinger?resource=acct:#{user.nickname}@localhost") |> get("/.well-known/webfinger?resource=acct:#{user.nickname}@hyrule.world")
|> json_response(200) |> json_response(200)
assert response["subject"] == "acct:#{user.nickname}@localhost" assert response["subject"] == "acct:#{user.nickname}@hyrule.world"
assert response["aliases"] == [ assert response["aliases"] == [
"https://hyrule.world/users/zelda", "https://hyrule.world/users/zelda",

View file

@ -39,6 +39,23 @@ defmodule Pleroma.Web.WebFingerTest do
end end
end end
test "requires exact match for Endpoint host or WebFinger domain" do
clear_config([Pleroma.Web.WebFinger, :domain], "pleroma.dev")
user = insert(:user)
assert {:error, "Couldn't find user"} ==
WebFinger.webfinger("#{user.nickname}@#{Pleroma.Web.Endpoint.host()}xxxx", "JSON")
assert {:error, "Couldn't find user"} ==
WebFinger.webfinger("#{user.nickname}@pleroma.devxxxx", "JSON")
assert {:ok, _} =
WebFinger.webfinger("#{user.nickname}@#{Pleroma.Web.Endpoint.host()}", "JSON")
assert {:ok, _} =
WebFinger.webfinger("#{user.nickname}@pleroma.dev", "JSON")
end
describe "fingering" do describe "fingering" do
test "returns error for nonsensical input" do test "returns error for nonsensical input" do
assert {:error, _} = WebFinger.finger("bliblablu") assert {:error, _} = WebFinger.finger("bliblablu")