emqttd_auth_ldap.erl 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. %%%-----------------------------------------------------------------------------
  2. %%% Copyright (c) 2012-2015 eMQTT.IO, All Rights Reserved.
  3. %%%
  4. %%% Permission is hereby granted, free of charge, to any person obtaining a copy
  5. %%% of this software and associated documentation files (the "Software"), to deal
  6. %%% in the Software without restriction, including without limitation the rights
  7. %%% to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. %%% copies of the Software, and to permit persons to whom the Software is
  9. %%% furnished to do so, subject to the following conditions:
  10. %%%
  11. %%% The above copyright notice and this permission notice shall be included in all
  12. %%% copies or substantial portions of the Software.
  13. %%%
  14. %%% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. %%% IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. %%% FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. %%% AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. %%% LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. %%% OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  20. %%% SOFTWARE.
  21. %%%-----------------------------------------------------------------------------
  22. %%% @doc
  23. %%% LDAP Authentication Module.
  24. %%%
  25. %%% @end
  26. %%%-----------------------------------------------------------------------------
  27. -module(emqttd_auth_ldap).
  28. -author("Feng Lee <feng@emqtt.io>").
  29. -include_lib("emqttd/include/emqttd.hrl").
  30. -import(proplists, [get_value/2, get_value/3]).
  31. -behaviour(emqttd_auth_mod).
  32. -export([init/1, check/3, description/0]).
  33. -record(state, {servers, user_dn, options}).
  34. init(Opts) ->
  35. Servers = get_value(servers, Opts, ["localhost"]),
  36. Port = get_value(port, Opts, 389),
  37. Timeout = get_value(timeout, Opts, 30),
  38. UserDn = get_value(user_dn, Opts),
  39. LdapOpts =
  40. case get_value(ssl, Opts, false) of
  41. true ->
  42. SslOpts = get_value(sslopts, Opts),
  43. [{port, Port}, {timeout, Timeout}, {sslopts, SslOpts}];
  44. false ->
  45. [{port, Port}, {timeout, Timeout}]
  46. end,
  47. {ok, #state{servers = Servers, user_dn = UserDn, options = LdapOpts}}.
  48. check(#mqtt_client{username = undefined}, _Password, _State) ->
  49. {error, "Username undefined"};
  50. check(_Client, undefined, _State) ->
  51. {error, "Password undefined"};
  52. check(_Client, <<>>, _State) ->
  53. {error, "Password undefined"};
  54. check(#mqtt_client{username = Username}, Password,
  55. #state{servers = Servers, user_dn = UserDn, options = Options}) ->
  56. case eldap:open(Servers, Options) of
  57. {ok, LDAP} ->
  58. UserDn1 = fill(binary_to_list(Username), UserDn),
  59. ldap_bind(LDAP, UserDn1, binary_to_list(Password));
  60. {error, Reason} ->
  61. {error, Reason}
  62. end.
  63. ldap_bind(LDAP, UserDn, Password) ->
  64. case catch eldap:simple_bind(LDAP, UserDn, Password) of
  65. ok ->
  66. ok;
  67. {error, invalidCredentials} ->
  68. {error, "LDAP Invalid Credentials"};
  69. {error, Error} ->
  70. {error, Error};
  71. {'EXIT', Reason} ->
  72. {error, Reason}
  73. end.
  74. fill(Username, UserDn) ->
  75. re:replace(UserDn, "\\$u", Username, [global, {return, list}]).
  76. description() ->
  77. "LDAP Authentication Module".