source: trunk/LATMOS-Accounts/lib/LATMOS/Accounts/Bases/Ldap/User.pm @ 2475

Last change on this file since 2475 was 2475, checked in by nanardon, 3 years ago

start samba support: manage sambaSID

File size: 9.2 KB
Line 
1package LATMOS::Accounts::Bases::Ldap::User;
2
3use 5.010000;
4use strict;
5use warnings;
6
7use base qw(LATMOS::Accounts::Bases::Ldap::objects);
8use Net::LDAP;
9use Net::LDAPS;
10use Net::LDAP::Entry;
11use Net::LDAP::Control::Paged;
12use Net::LDAP::Constant qw( LDAP_CONTROL_PAGED ); 
13use Net::LDAP::Util     qw( escape_filter_value );
14use LATMOS::Accounts::Log;
15use LATMOS::Accounts::Bases::Ldap::Group;
16use LATMOS::Accounts::Utils;
17use Crypt::SmbHash;
18
19our $VERSION = (q$Rev: 649 $ =~ /^Rev: (\d+) /)[0];
20
21=head1 NAME
22
23LATMOS::Ldap - Perl extension for blah blah blah
24
25=head1 SYNOPSIS
26
27  use LATMOS::Ldap;
28  blah blah blah
29
30=head1 DESCRIPTION
31
32Stub documentation for LATMOS::Ldap, created by h2xs. It looks like the
33author of the extension was negligent enough to leave the stub
34unedited.
35
36Blah blah blah.
37
38=head1 FUNCTIONS
39
40=cut
41
42sub _class_filter { '(&(ObjectClass=posixAccount))' }
43
44sub _key_attr { 'cn' } 
45
46sub _my_ldap_classes { qw(
47    top
48    inetOrgPerson
49    organizationalPerson
50    posixAccount
51    shadowAccount
52    sambaSamAccount
53) }
54
55sub _computeSSID {
56    my ($value) = @_;
57    $value * 2 + 1000
58}
59
60sub _get_attr_schema {
61    my ($class, $base) = @_;
62    $class->SUPER::_get_attr_schema($base,
63    {
64        displayName => { },
65        givenName => { },
66        homePhone => { },
67        homePostalAddress => { },
68        initials => { },
69        mail => { },
70        sn => { },
71        mobile => { },
72        o => { },
73        uid => { mandatory => 1 },
74        manager => {
75            delayed => 1,
76            can_values => sub { $base->list_objects('user') },
77        },
78        facsimileTelephoneNumber => { },
79        uidNumber => {
80            uniq => 1,
81            post => sub {
82                my ($self, $value) = @_;
83                $self->object->set_fields(sambaSID => $self->object->base->sambaSID(_computeSSID($value)));
84            },
85        },
86        gidNumber => {
87            reference => 'group',
88            mandatory => 1,
89            can_values => sub {
90                map { $base->get_object('group',
91                        $_)->get_attributes('gidNumber') }
92                $base->list_objects('group')
93            },
94            display => sub {
95                my ($self, $val) = @_;
96                my ($gr) = $self->base->search_objects('group', "gidNumber=$val")
97                    or return;
98                return $gr;
99            },
100
101        },
102        homeDirectory => { },
103        loginShell => { },
104        gecos => { },
105        description => { },
106        shadowLastChange => { },
107        shadowMin => { },
108        shadowMax => { },
109        shadowWarning => { },
110        shadowInactive => { },
111        shadowExpire => { },
112        shadowFlag => { },
113        street => { },
114        postOfficeBox => { },
115        postalCode => { },
116        postalAddress => { },
117        streetAddress => { },
118        physicalDeliveryOfficeName => { },
119        ou => { },
120        st => { },
121        l => { },
122        telephoneNumber => { },
123        memberOf => {
124            delayed => 1,
125            multiple => 1,
126            reference => 'group',
127        },
128        cn => { ro => 1, },
129        dn => { ro => 1, },
130        objectClass => { ro => 1, },
131        pwdAccountLockedTime => {},
132        pwdPolicySubentry => {},
133        pwdChangedTime => { ro => 1 },
134        labeledURI => {},
135        userPassword    => { readable => 0, },
136        sambaLMPassword => { readable => 0, },
137        sambaNTPassword => { readable => 0, },
138        sambaSID => { },
139    }
140    );
141}
142
143sub _update_class {
144    my ($self) = @_;
145
146    $self->SUPER::_update_class(
147        sambaSID => $self->base->sambaSID(
148            _computeSSID($self->_get_attributes('uidNumber'))
149        )
150    );
151}
152
153sub _create {
154    my ($class, $base, $id, %data) = @_;
155
156    my $entry = Net::LDAP::Entry->new();
157
158    $entry->dn(join(',',
159        sprintf('%s=%s',
160            $class->_dn_attribute($base),
161            escape_filter_value($id)),
162            $base->object_base_dn($class->type),
163        )
164    );
165    $entry->replace(objectClass =>
166        [ $class->_my_ldap_classes ],);
167    $data{cn} ||= $data{sn} || $id; # cn is mandatory
168    $data{sn} ||= $id; # sn is mandatory
169    $data{uid} ||= $id; # uid is mandatory
170    $data{sambaSID} ||= $base->sambaSID(_computeSSID($data{uidNumber}));
171    $data{homeDirectory} ||= '/dev/null'; # homeDirectory is mandatory
172    $data{$class->_key_attribute($base)} = $id;
173    foreach (keys %data) {
174        $class->_populate_entry($entry,
175            $_, $data{$_}, $base);
176    }
177
178    my $msg = $base->ldap->add($entry);
179    if ($msg->code) {
180        $base->log(LA_ERR, "Cannot create user: %s", $msg->error);
181        return 0;
182    }
183    return 1;
184}
185
186sub _rename {
187    my ($class, $base, $uid, $newuid) = @_;
188
189    $class->SUPER::_rename($base, $uid, $newuid) or return;
190}
191
192sub get_field {
193    my ($self, $field) = @_;
194
195    $field eq 'streetAddress' and $field = 'street';
196    $field eq 'memberOf' and do {
197        my @res;
198        $self->base->_unlimited_search(
199            base => $self->base->object_base_dn('group'),
200            filter => sprintf(
201                '(&(objectClass=posixGroup)(memberUID=%s))',
202                escape_filter_value($self->id),
203            ),
204            callback => sub {
205                my ($mesg, $entry) = @_;
206                ref $entry eq 'Net::LDAP::Entry' or return;
207                push(@res, $entry->get_value(
208                    LATMOS::Accounts::Bases::Ldap::Group->_key_attribute($self->base),
209                ));
210                1;
211            },
212        );
213        return [ sort(@res) ];
214    };
215    $field eq 'manager' and do {
216        my $dn = $self->SUPER::get_field($field) or return;
217        my $manager = $self->base->_get_object_from_dn($dn) or return;
218        return $manager->get_value(__PACKAGE__->_key_attribute($self->base));
219    };
220    $self->SUPER::get_field($field);
221}
222
223sub _populate_entry {
224    my ($self, $entry, $f, $val, $base) = @_;
225    $base ||= $self->base;
226    for ($f) {
227        /^sn$/ and $val ||= $entry->get_value($self->_key_attribute);
228        /^memberOf$/ and do {
229            my %users;
230            $users{$_}{e} = 1 foreach (ref $self
231                ? $self->get_attributes('memberOf')
232                : ());
233            $val = [ $val ] unless(ref $val);
234            $users{$_}{n} = 1 foreach (grep { $_ } @{ $val || []});
235            foreach (keys %users) {
236                $users{$_}{e} && $users{$_}{n} and next;
237                my $group = $base->get_object('group', $_) or next;
238                if ($users{$_}{e}) {
239                    $group->{entry}->delete(memberUID =>
240                        $entry->get_value(
241                            LATMOS::Accounts::Bases::Ldap::User->_key_attribute($base),
242                        )
243                    );
244                } elsif ($users{$_}{n}) {
245                    $group->{entry}->add(memberUID =>
246                        $entry->get_value(
247                            LATMOS::Accounts::Bases::Ldap::User->_key_attribute($base),
248                        )
249                    );
250                } # else {} # can't happen
251                my $mesg = $group->{entry}->update($base->ldap);
252                if ($mesg->code) {
253                    $base->log(LA_ERR, "Cannot set attributes: %s", $mesg->error);
254                    return;
255                }
256            }
257            return 1;
258        };
259        /^manager$/ && $val and do {
260            my $user = $base->get_object('user', $val) or do {
261                $val = undef;
262                next;
263            };
264            $val = $user->get_field('dn');
265            next;
266        };
267        /^gecos$/ and do {
268            $val = to_ascii($val);
269        };
270        /^homeDirectory$/ and $val ||= '/dev/null';
271    }
272    $self->SUPER::_populate_entry($entry, $f, $val, $base);
273}
274
275sub _set_password {
276    my ($self, $clear_pass) = @_;
277
278    # openldap use prefix to identify encryption passwd
279    # {CRYPT} is system dependant, eg use crypt from system
280    # As we run openldap on UNIX, this should not be a problem
281    # as we use perl crypt() which does the same
282    # This code will have to be changed if we use openldap on other UNIX
283    my $crypt = '{CRYPT}' . $self->base->passCrypt($clear_pass);
284
285    my ($lm, $nt) = ntlmgen $clear_pass;
286
287    my $res = $self->set_fields(
288        userPassword    => $crypt,
289        sambaLMPassword => $lm,
290        sambaNTPassword => $nt,
291    );
292    $self->base->log(LA_NOTICE, 'Mot de passe changé pour %s', $self->id)
293    if($res);
294    return $res;
295}
296
297=head2 _InjectCryptPasswd($cryptpasswd)
298
299Inject a password encrypted using standard UNIX method.
300
301Works only for unix authentification method inside LDAP
302
303=cut
304
305sub _InjectCryptPasswd {
306    my ($self, $cryptpasswd) = @_;
307
308    my $res = $self->set_fields(
309        userPassword => '{CRYPT}' . $cryptpasswd,
310    );
311
312    if ($res) {
313        $self->base->log(LA_NOTICE, 'Crypted password injected for %s', $self->id);
314        return 1;
315    } else {
316        $self->base->log(LA_ERR, 'Cannot inject crypted password for %s', $self->id);
317        return 0;
318    }
319}
320
3211;
322
323__END__
324
325=head1 SEE ALSO
326
327=head1 AUTHOR
328
329Olivier Thauvin, E<lt>olivier.thauvin@aerov.jussieu.frE<gt>
330
331=head1 COPYRIGHT AND LICENSE
332
333Copyright (C) 2008 CNRS SA/CETP/LATMOS
334
335This library is free software; you can redistribute it and/or modify
336it under the same terms as Perl itself, either Perl version 5.10.0 or,
337at your option, any later version of Perl 5 you may have available.
338
339
340=cut
Note: See TracBrowser for help on using the repository browser.