Revision 100 (by dpavlin, 2010/09/02 12:36:37) apply regex on hrEduPersonGroupMember

this is used to rename group
#!/usr/bin/perl
# Copyright (c) 2006 Hans Klunder <hans.klunder@bigfoot.com>. All rights reserved.
# This program is free software; you can redistribute it and/or
# modify it under the same terms as Perl itself.

# It's modified by Dobrica Pavlinusic <dpavlin@rot13.org> to include following:
#
# * rewrite LDAP bind request cn: username@domain.com -> uid=username,dc=domain,dc=com
# * rewrite search responses:
# ** expand key:value pairs from hrEduPersonUniqueNumber into hrEduPersonUniqueNumber_key
# ** augment response with yaml/dn.yaml data (for external data import)

use strict;
use warnings;

use IO::Select;
use IO::Socket;
use IO::Socket::SSL;
use warnings;
use Data::Dump qw/dump/;
use Convert::ASN1 qw(asn_read);
use Net::LDAP::ASN qw(LDAPRequest LDAPResponse);
our $VERSION = '0.3';
use fields qw(socket target);
use YAML qw/LoadFile/;

my $debug = 0;

my $config = {
	yaml_dir => './yaml/',
	listen => shift @ARGV || 'localhost:1389',
	upstream_ldap => 'ldap.ffzg.hr',
	upstream_ssl => 1,
	overlay_prefix => 'ffzg-',
#	log_file => 'log/ldap-rewrite.log',

};

my $log_fh;

sub log {
	return unless $config->{log_file};

	if ( ! $log_fh ) {
		open($log_fh, '>>', $config->{log_file}) || die "can't open ", $config->{log_file},": $!";
		print $log_fh "# " . time;
	}
	$log_fh->autoflush(1);
	print $log_fh join("\n", @_),"\n";
}

BEGIN {
	$SIG{'__WARN__'} = sub { warn @_; main::log(@_); }
}


if ( ! -d $config->{yaml_dir} ) {
	warn "DISABLE ", $config->{yaml_dir}," data overlay";
}

warn "# config = ",dump( $config );

sub handle {
 	my $clientsocket=shift;
	my $serversocket=shift;

	# read from client
	asn_read($clientsocket, my $reqpdu);
	if ( ! $reqpdu ) {
		warn "client closed connection\n";
		return 0;
	}
	$reqpdu = log_request($reqpdu);

	# send to server
	print $serversocket $reqpdu or die "Could not send PDU to server\n ";
	
	# read from server
	my $ready;
	my $sel = IO::Select->new($serversocket);
	for( $ready = 1 ; $ready ; $ready = $sel->can_read(0)) {
		asn_read($serversocket, my $respdu);
		if ( ! $respdu ) {
			warn "server closed connection\n";
			return 0;
		}
		$respdu = log_response($respdu);
		# and send the result to the client
		print $clientsocket $respdu || return 0;
	}

	return 1;
}


sub log_request {
	my $pdu=shift;

	die "empty pdu" unless $pdu;

#	print '-' x 80,"\n";
#	print "Request ASN 1:\n";
#	Convert::ASN1::asn_hexdump(\*STDOUT,$pdu);
#	print "Request Perl:\n";
	my $request = $LDAPRequest->decode($pdu);
	warn "## request = ",dump($request);

	if ( defined $request->{bindRequest} ) {
		if ( $request->{bindRequest}->{name} =~ m{@} ) {
			my $old = $request->{bindRequest}->{name};
			$request->{bindRequest}->{name} =~ s/[@\.]/,dc=/g;
			$request->{bindRequest}->{name} =~ s/^/uid=/;
			warn "rewrite bind cn $old -> ", $request->{bindRequest}->{name};
			Convert::ASN1::asn_hexdump(\*STDOUT,$pdu) if $debug;
			$pdu = $LDAPRequest->encode($request);
			Convert::ASN1::asn_hexdump(\*STDOUT,$pdu) if $debug;
		}
	}

	return $pdu;
}

sub log_response {
	my $pdu=shift;
	die "empty pdu" unless $pdu;

#	print '-' x 80,"\n";
#	print "Response ASN 1:\n";
#	Convert::ASN1::asn_hexdump(\*STDOUT,$pdu);
#	print "Response Perl:\n";
	my $response = $LDAPResponse->decode($pdu);

	if ( defined $response->{protocolOp}->{searchResEntry} ) {
		my $uid = $response->{protocolOp}->{searchResEntry}->{objectName};
		warn "## objectName $uid";

		my @attrs;

		foreach my $attr ( @{ $response->{protocolOp}->{searchResEntry}->{attributes} } ) {
			if ( $attr->{type} =~ m/date/i ) {
				foreach my $i ( 0 .. $#{ $attr->{vals} } ) {
					$attr->{vals}->[$i] = "$1-$2-$3" if $attr->{vals}->[$i] =~ m/^([12]\d\d\d)([01]\d+)([0123]\d+)$/;
				}
			} elsif ( $attr->{type} eq 'hrEduPersonUniqueNumber' ) {
				foreach my $val ( @{ $attr->{vals} } ) {
					next if $val !~ m{.+:.+};
					my ( $n, $v ) = split(/\s*:\s*/, $val );
					push @attrs, { type => $attr->{type} . '_' . $n, vals => [ $v ] };
				}
			} elsif ( $attr->{type} eq 'hrEduPersonGroupMember' ) {
				foreach my $i ( 0 .. $#{ $attr->{vals} } ) {
					$attr->{vals}->[$i] =~ s/^u2010/p2010/gs && warn "FIXME group";
				}
			}
		}

		warn "# ++ attrs ",dump( @attrs );

		push @{ $response->{protocolOp}->{searchResEntry}->{attributes} }, $_ foreach @attrs;

		my @additional_yamls = ( $uid );
		foreach my $attr ( @{ $response->{protocolOp}->{searchResEntry}->{attributes} } ) {
			foreach my $v ( @{ $attr->{vals} } ) {
				push @additional_yamls, $attr->{type} . '/' . $v;
			}
		}

		#warn "# additional_yamls ",dump( @additional_yamls );

		foreach my $path ( @additional_yamls ) {
			my $full_path = $config->{yaml_dir} . '/' . $path . '.yaml';
			next unless -e $full_path;

			my $data = LoadFile( $full_path );
			warn "# $full_path yaml = ",dump($data);

			foreach my $type ( keys %$data ) {

				my $vals = $data->{$type};

				push @{ $response->{protocolOp}->{searchResEntry}->{attributes} }, {
					type => $config->{overlay_prefix} . $type,
					vals => ref($vals) eq 'ARRAY' ? $vals : [ $vals ],
				};
			}
		}

		$pdu = $LDAPResponse->encode($response);
	}

	warn "## response = ", dump($response);

	return $pdu;
}


my $listenersock = IO::Socket::INET->new(
	Listen => 5,
	Proto => 'tcp',
	Reuse => 1,
	LocalAddr => $config->{listen},
) || die "can't open listen socket: $!";

our $server_sock;

sub connect_to_server {
	my $sock;
	if ( $config->{upstream_ssl} ) {
		$sock = IO::Socket::SSL->new( $config->{upstream_ldap} . ':ldaps' );
	} else {
		$sock = IO::Socket::INET->new(
			Proto => 'tcp',
			PeerAddr => $config->{upstream_ldap},
			PeerPort => 389,
		);
	}
	die "can't open ", $config->{upstream_ldap}, " $!\n" unless $sock;
	warn "## connected to ", $sock->peerhost, ":", $sock->peerport, "\n";
	return $sock;
}

my $sel = IO::Select->new($listenersock);
while (my @ready = $sel->can_read) {
	foreach my $fh (@ready) {
		if ($fh == $listenersock) {
			# let's create a new socket
			my $psock = $listenersock->accept;
			$sel->add($psock);
			warn "## add $psock " . time;
		} else {
			$server_sock->{$fh} ||= connect_to_server;
			if ( ! handle($fh,$server_sock->{$fh}) ) {
				warn "## remove $fh " . time;
				$sel->remove($server_sock->{$fh});
				$server_sock->{$fh}->close;
				delete $server_sock->{$fh};
				# we have finished with the socket
				$sel->remove($fh);
				$fh->close;
			}
		}
	}
}

1;