123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- # rails autoloading issue
- require 'ldap'
- require 'ldap/user'
- require 'tcr/net/ldap'
- RSpec.describe Ldap::User do
- let(:mocked_ldap) { double }
- describe '.uid_attribute' do
- it 'responds to .uid_attribute' do
- expect(described_class).to respond_to(:uid_attribute)
- end
- it 'returns uid attribute string from given attribute strucutre' do
- attributes = {
- objectguid: 'TEST',
- custom: 'value',
- }
- expect(described_class.uid_attribute(attributes)).to eq('objectguid')
- end
- it 'returns nil if no attribute could be found' do
- attributes = {
- custom: 'value',
- }
- expect(described_class.uid_attribute(attributes)).to be_nil
- end
- end
- # required as 'let' to perform test based
- # expectations and reuse it in 'let' instance
- # as additional parameter
- describe 'initialization config parameters' do
- it 'reuses given Ldap instance if given' do
- expect(Ldap).not_to receive(:new)
- described_class.new(create(:ldap_source), ldap: mocked_ldap)
- end
- it 'takes optional filter' do
- filter = '(objectClass=custom)'
- config = {
- filter: filter
- }
- instance = described_class.new(config, ldap: mocked_ldap)
- expect(instance.filter).to eq(filter)
- end
- it 'takes optional uid_attribute' do
- uid_attribute = 'objectguid'
- config = {
- uid_attribute: uid_attribute
- }
- instance = described_class.new(config, ldap: mocked_ldap)
- expect(instance.uid_attribute).to eq(uid_attribute)
- end
- it 'creates own Ldap instance if none given' do
- expect(Ldap).to receive(:new)
- described_class.new(create(:ldap_source))
- end
- end
- describe 'instance methods' do
- let(:initialization_config) do
- {
- uid_attribute: 'objectguid',
- filter: '(objectClass=user)',
- }
- end
- let(:instance) do
- described_class.new(initialization_config, ldap: mocked_ldap)
- end
- describe '#valid?' do
- shared_examples 'validates credentials' do
- it 'validates username and password' do
- connection = double
- allow(mocked_ldap).to receive(:connection).and_return(connection)
- build(:ldap_entry)
- allow(mocked_ldap).to receive(:base_dn)
- allow(connection).to receive(:bind_as).and_return(true)
- expect(instance.valid?('example_username', 'password')).to be true
- end
- it 'fails for invalid credentials' do
- connection = double
- allow(mocked_ldap).to receive(:connection).and_return(connection)
- build(:ldap_entry)
- allow(mocked_ldap).to receive(:base_dn)
- allow(connection).to receive(:bind_as).and_return(false)
- expect(instance.valid?('example_username', 'wrong_password')).to be false
- end
- end
- it 'responds to #valid?' do
- expect(instance).to respond_to(:valid?)
- end
- it_behaves_like 'validates credentials'
- context 'with a user_filter inside of the config' do
- let(:initialization_config) do
- {
- uid_attribute: 'objectguid',
- filter: '(objectClass=user)',
- user_filter: '(cn=example)'
- }
- end
- it_behaves_like 'validates credentials'
- end
- end
- describe '#attributes' do
- it 'responds to #attributes' do
- expect(instance).to respond_to(:attributes)
- end
- it 'lists user attributes with example values' do
- ldap_entry = build(:ldap_entry)
- # selectable attribute
- ldap_entry['mail'] = 'test@example.com'
- # filtered attribute
- ldap_entry['lastlogon'] = DateTime.current
- allow(mocked_ldap).to receive(:search).and_yield(ldap_entry)
- attributes = instance.attributes
- expected_attributes = {
- dn: String,
- mail: String,
- }
- expect(attributes).to include(expected_attributes)
- expect(attributes).not_to include(:lastlogon)
- end
- end
- describe '#filter' do
- let(:initialization_config) do
- {
- uid_attribute: 'objectguid',
- }
- end
- it 'responds to #filter' do
- expect(instance).to respond_to(:filter)
- end
- it 'tries filters and returns first one with entries' do
- allow(mocked_ldap).to receive(:entries?).and_return(true)
- expect(instance.filter).to be_a(String)
- end
- it 'fails if no filter found entries' do
- allow(mocked_ldap).to receive(:entries?).and_return(false)
- expect(instance.filter).to be_nil
- end
- end
- describe '#uid_attribute' do
- let(:initialization_config) do
- {
- filter: '(objectClass=user)',
- }
- end
- it 'responds to #uid_attribute' do
- expect(instance).to respond_to(:uid_attribute)
- end
- it 'tries to find uid attribute in example attributes' do
- ldap_entry = build(:ldap_entry)
- # selectable attribute
- ldap_entry['objectguid'] = 'f742b361-32c6-4a92-baaa-eaae7df657ee'
- allow(mocked_ldap).to receive(:search).and_yield(ldap_entry)
- expect(instance.uid_attribute).to be_a(String)
- end
- it 'fails if no uid attribute could be found' do
- expect(mocked_ldap).to receive(:search)
- expect(instance.uid_attribute).to be_nil
- end
- end
- end
- # Each of these test cases depends on
- # sample TCP transmission data recorded with TCR,
- # stored in test/data/tcr_cassettes.
- describe 'on mocked LDAP connections' do
- around do |example|
- cassette_name = example.description.gsub(%r{[^0-9A-Za-z.-]+}, '_')
- begin
- original_tcr_format = TCR.configuration.format
- TCR.configuration.format = 'marshal'
- TCR.use_cassette("lib/ldap/user/#{cassette_name}") { example.run }
- ensure
- TCR.configuration.format = original_tcr_format
- end
- end
- describe 'attributes' do
- subject(:user) { described_class.new(config, ldap: ldap) }
- let(:ldap) { Ldap.new(config) }
- let(:config) do
- { 'host' => 'localhost',
- 'ssl' => 'off',
- 'options' => { 'dc=example,dc=org' => 'dc=example,dc=org' },
- 'option' => 'dc=example,dc=org',
- 'base_dn' => 'dc=example,dc=org',
- 'bind_user' => 'cn=admin,dc=example,dc=org',
- 'bind_pw' => 'admin' }.with_indifferent_access
- end
- # see https://github.com/zammad/zammad/issues/2140
- #
- # This method grabs sample values of user attributes on the LDAP server.
- # It used to coerce ALL values to Unicode strings, including binary attributes
- # (e.g., usersmimecertificate / msexchmailboxsecuritydescriptor),
- # which led to valid Unicode gibberish (e.g., "\u0001\u0001\u0004...")
- #
- # When saving these values to the database,
- # ActiveRecord::Store would convert them to binary (ASCII-8BIT) strings,
- # which would then break #to_json with an Encoding::UndefinedConversion error.
- it 'skips binary attributes (#2140)' do
- source = create(:ldap_source)
- source.update(preferences: user.attributes)
- expect { source.preferences.to_json }
- .not_to raise_error
- end
- end
- end
- end
|