summaryrefslogtreecommitdiff
path: root/FS/FS/part_export/cyrus.pm
blob: 84c9e5a30bd4ba83add688e71c979d8cc31b38bc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
package FS::part_export::cyrus;

use vars qw(@ISA %info);
use Tie::IxHash;
use FS::part_export;

@ISA = qw(FS::part_export);

tie my %options, 'Tie::IxHash',
  'server'   => { label=>'IMAP server' },
  'username' => { label=>'Admin username' },
  'password' => { label=>'Admin password' },
;

%info = ( 
  'svc'      => 'svc_acct',
  'desc'     => 'Real-time export to Cyrus IMAP server',
  'options'  => \%options,
  'nodomain' => 'Y',
  'notes'    => <<'END'
Integration with
<a href="http://asg.web.cmu.edu/cyrus/imapd/">Cyrus IMAP Server</a>.
Cyrus::IMAP::Admin should be installed locally and the connection to the
server secured.  <B>svc_acct.quota</B>, if available, is used to set the
Cyrus quota.
END
);

sub rebless { shift; }

sub _export_insert {
  my($self, $svc_acct) = (shift, shift);
  $self->cyrus_queue( $svc_acct->svcnum, 'insert',
    $svc_acct->username, $svc_acct->quota );
}

sub _export_replace {
  my( $self, $new, $old ) = (shift, shift, shift);
  return "can't change username using Cyrus"
    if $old->username ne $new->username;
  return '';
#  #return '' unless $old->_password ne $new->_password;
#  $self->cyrus_queue( $new->svcnum,
#    'replace', $new->username, $new->_password );
}

sub _export_delete {
  my( $self, $svc_acct ) = (shift, shift);
  $self->cyrus_queue( $svc_acct->svcnum, 'delete',
    $svc_acct->username );
}

#a good idea to queue anything that could fail or take any time
sub cyrus_queue {
  my( $self, $svcnum, $method ) = (shift, shift, shift);
  my $queue = new FS::queue {
    'svcnum' => $svcnum,
    'job'    => "FS::part_export::cyrus::cyrus_$method",
  };
  $queue->insert(
    $self->option('server'),
    $self->option('username'),
    $self->option('password'),
    @_
  );
}

sub cyrus_insert { #subroutine, not method
  my $client = cyrus_connect(shift, shift, shift);
  my( $username, $quota ) = @_;
  my $rc = $client->create("user.$username");
  my $error = $client->error;
  die "creating user.$username: $error" if $error;

  $rc = $client->setacl("user.$username", $username => 'all' );
  $error = $client->error;
  die "setacl user.$username: $error" if $error;

  if ( $quota ) {
    $rc = $client->setquota("user.$username", 'STORAGE' => $quota );
    $error = $client->error;
    die "setquota user.$username: $error" if $error;
  }

}

sub cyrus_delete { #subroutine, not method
  my ( $server, $admin_username, $password_username, $username ) = @_;
  my $client = cyrus_connect($server, $admin_username, $password_username);

  my $rc = $client->setacl("user.$username", $admin_username => 'all' );
  my $error = $client->error;
  die $error if $error;

  $rc = $client->delete("user.$username");
  $error = $client->error;
  die $error if $error;
}

sub cyrus_connect {

  my( $server, $admin_username, $admin_password ) = @_;

  eval "use Cyrus::IMAP::Admin;";

  my $client = Cyrus::IMAP::Admin->new($server);
  $client->authenticate(
    -user      => $admin_username,
    -mechanism => "login",       
    -password  => $admin_password,
  );
  $client;

}

#sub cyrus_replace { #subroutine, not method
#}

1;