LDAP authentication process modification #824
3 changed files with 60 additions and 32 deletions
|
@ -17,9 +17,11 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
|
|||
|
||||
## Fixed
|
||||
- Meilisearch: order of results returned from our REST API now actually matches how Meilisearch ranks results
|
||||
- LDAP authentication did not work in Docker because the `eldap` module was missing
|
||||
|
||||
## Changed
|
||||
- Refactored Rich Media to cache the content in the database. Fetching operations that could block status rendering have been eliminated.
|
||||
- Modified the LDAP authentication process, to allow logging against servers that require _simple authentication_ instead of _anonymous_ one to access the directory
|
||||
|
||||
## 2024.04.1 (Security)
|
||||
|
||||
|
|
|
@ -65,7 +65,38 @@ defp ldap_user(name, password) do
|
|||
end
|
||||
end
|
||||
|
||||
bind_user(connection, ldap, name, password)
|
||||
#TODO
|
||||
# Make optional bind mode between simple and anonymous
|
||||
# through a select option in admin-fe interface.
|
||||
|
||||
ldap_username = System.get_env("LDAP_READONLY_USER_USERNAME") || "readonly"
|
||||
ldap_pwd = System.get_env("LDAP_READONLY_USER_PASSWORD")
|
||||
base = Keyword.get(ldap, :base)
|
||||
base_parts = String.split(base, ",")
|
||||
root_base = Enum.join(Enum.drop_while(base_parts, fn x -> !String.starts_with?(x, "dc") end), ",")
|
||||
dn = "cn=#{ldap_username},#{root_base}"
|
||||
|
||||
case :eldap.simple_bind(
|
||||
connection,
|
||||
dn,
|
||||
ldap_pwd
|
||||
) do
|
||||
:ok ->
|
||||
|
||||
scope = :eldap.wholeSubtree()
|
||||
filter = :eldap.equalityMatch("uid", name)
|
||||
deref = :eldap.neverDerefAliases()
|
||||
|
||||
case :eldap.search(connection, base: base, scope: scope, deref: deref, filter: filter, timeout: @search_timeout) do
|
||||
{:ok, {:eldap_search_result, [{:eldap_entry, _, attributes}], _, _}} ->
|
||||
|
||||
bind_user(connection, ldap, attributes, name, password)
|
||||
|
||||
error ->
|
||||
Logger.error("Binding error: #{inspect(error)}")
|
||||
end
|
||||
end
|
||||
|
||||
after
|
||||
:eldap.close(connection)
|
||||
end
|
||||
|
@ -76,18 +107,21 @@ defp ldap_user(name, password) do
|
|||
end
|
||||
end
|
||||
|
||||
defp bind_user(connection, ldap, name, password) do
|
||||
uid = Keyword.get(ldap, :uid, "cn")
|
||||
defp bind_user(connection, ldap, attributes, name, password) do
|
||||
#uid = Keyword.get(ldap, :uid, "cn")
|
||||
base = Keyword.get(ldap, :base)
|
||||
cn_attr = List.keyfind(attributes, ~c"cn", 0)
|
||||
cn = Enum.at(Tuple.to_list(cn_attr),1)
|
||||
attr = "cn"
|
||||
|
||||
case :eldap.simple_bind(connection, "#{uid}=#{name},#{base}", password) do
|
||||
case :eldap.simple_bind(connection, "#{attr}=#{cn},#{base}", password) do
|
||||
:ok ->
|
||||
case fetch_user(name) do
|
||||
%User{} = user ->
|
||||
user
|
||||
|
||||
|
||||
_ ->
|
||||
register_user(connection, base, uid, name)
|
||||
register_user(attributes, name)
|
||||
end
|
||||
|
||||
error ->
|
||||
|
@ -95,35 +129,26 @@ defp bind_user(connection, ldap, name, password) do
|
|||
end
|
||||
end
|
||||
|
||||
defp register_user(connection, base, uid, name) do
|
||||
case :eldap.search(connection, [
|
||||
{:base, to_charlist(base)},
|
||||
{:filter, :eldap.equalityMatch(to_charlist(uid), to_charlist(name))},
|
||||
{:scope, :eldap.wholeSubtree()},
|
||||
{:timeout, @search_timeout}
|
||||
]) do
|
||||
{:ok, {:eldap_search_result, [{:eldap_entry, _, attributes}], _, _}} ->
|
||||
params = %{
|
||||
name: name,
|
||||
nickname: name,
|
||||
password: nil
|
||||
}
|
||||
defp register_user(attributes, name) do
|
||||
|
||||
params =
|
||||
case List.keyfind(attributes, ~c"mail", 0) do
|
||||
{_, [mail]} -> Map.put_new(params, :email, :erlang.list_to_binary(mail))
|
||||
_ -> params
|
||||
end
|
||||
params = %{
|
||||
name: name,
|
||||
nickname: name,
|
||||
password: nil
|
||||
}
|
||||
|
||||
changeset = User.register_changeset_ldap(%User{}, params)
|
||||
params =
|
||||
case List.keyfind(attributes, ~c"mail", 0) do
|
||||
{_, [mail]} -> Map.put_new(params, :email, :erlang.list_to_binary(mail))
|
||||
_ -> params
|
||||
end
|
||||
|
||||
case User.register(changeset) do
|
||||
{:ok, user} -> user
|
||||
error -> error
|
||||
end
|
||||
changeset = User.register_changeset_ldap(%User{}, params)
|
||||
|
||||
error ->
|
||||
error
|
||||
case User.register(changeset) do
|
||||
{:ok, user} -> user
|
||||
error -> error
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
|
|
3
mix.exs
3
mix.exs
|
@ -79,7 +79,8 @@ def application do
|
|||
:fast_sanitize,
|
||||
:os_mon,
|
||||
:ssl,
|
||||
:recon
|
||||
:recon,
|
||||
:eldap
|
||||
],
|
||||
included_applications: [:ex_syslogger]
|
||||
]
|
||||
|
|
Loading…
Reference in a new issue