bugfixes from Jeff Finucane <jeff@cmh.net>, thanks!
[freeside.git] / bin / svc_acct_sm.import
1 #!/usr/bin/perl -Tw
2 #
3 # $Id: svc_acct_sm.import,v 1.8 2000-12-03 15:14:00 ivan Exp $
4 #
5 # ivan@sisd.com 98-mar-9
6 #
7 # generalized svcparts ivan@sisd.com 98-mar-23
8
9 # You really need to enable ssh into a shell machine as this needs to rename
10 # .qmail-extension files.
11 #
12 # now an interactive script ivan@sisd.com 98-jun-30
13 #
14 # has an (untested) section for sendmail, s/warn/die/g and generates a program
15 # to run on your mail machine _later_ instead of ssh'ing for each user
16 # ivan@sisd.com 98-jul-13
17 #
18 # $Log: svc_acct_sm.import,v $
19 # Revision 1.8  2000-12-03 15:14:00  ivan
20 # bugfixes from Jeff Finucane <jeff@cmh.net>, thanks!
21 #
22 # Revision 1.7  2000/06/29 10:51:52  ivan
23 # oops, silly mistake
24 #
25 # Revision 1.6  2000/06/29 10:48:25  ivan
26 # make svc_acct_sm skip blank lines in sendmail import
27 #
28 # Revision 1.5  2000/02/03 05:16:52  ivan
29 # beginning of DNS and Apache support
30 #
31 # Revision 1.4  1999/03/25 08:42:20  ivan
32 # import stuff uses Term::Query and spits out (some kinds of) nonsensical input
33 #
34 # Revision 1.3  1999/03/24 00:51:55  ivan
35 # die if no relevant services... cvspain
36 #
37 # Revision 1.2  1998/12/10 07:23:18  ivan
38 # use FS::Conf, need user (for datasrc)
39 #
40
41 use strict;
42 use vars qw(%d_part_svc %m_part_svc);
43 use Term::Query qw(query);
44 use FS::SSH qw(iscp);
45 use FS::UID qw(adminsuidsetup datasrc);
46 use FS::Record qw(qsearch qsearchs);
47 use FS::svc_acct_sm;
48 use FS::svc_domain;
49 use FS::svc_acct;
50 use FS::part_svc;
51
52 my $user = shift or die &usage;
53 adminsuidsetup $user;
54
55 my($spooldir)="/usr/local/etc/freeside/export.". datasrc;
56
57 my(%mta) = (
58   1 => "qmail",
59   2 => "sendmail",
60 );
61
62 ###
63
64 %d_part_svc =
65   map { $_->svcpart, $_ } qsearch('part_svc',{'svcdb'=>'svc_domain'});
66 %m_part_svc =
67   map { $_->svcpart, $_ } qsearch('part_svc',{'svcdb'=>'svc_acct_sm'});
68
69 die "No services with svcdb svc_domain!\n" unless %d_part_svc;
70 die "No services with svcdb svc_svc_acct_sm!\n" unless %m_part_svc;
71
72 print "\n\n", 
73       ( join "\n", map "$_: ".$d_part_svc{$_}->svc, sort keys %d_part_svc ),
74       "\n\n";
75 $^W=0; #Term::Query isn't -w-safe
76 my $domain_svcpart = 
77   query "Enter part number for domains: ", 'irk', [ keys %d_part_svc ];
78 $^W=1;
79
80 print "\n\n", 
81       ( join "\n", map "$_: ".$m_part_svc{$_}->svc, sort keys %m_part_svc ),
82       "\n\n";
83 $^W=0; #Term::Query isn't -w-safe
84 my $mailalias_svcpart = 
85   query "Enter part number for mail aliases: ", 'irk', [ keys %m_part_svc ];
86 $^W=1;
87
88 print "\n\n", <<END;
89 Select your MTA from the following list.
90 END
91 print join "\n", map "$_: $mta{$_}", sort keys %mta;
92 print "\n\n";
93 $^W=0; #Term::Query isn't -w-safe
94 my $mta = query ":", 'irk', [ keys %mta ];
95 $^W=1;
96
97 if ( $mta{$mta} eq "qmail" ) {
98
99   print "\n\n", <<END;
100 Enter the location and name of your qmail control directory, for example
101 "mail.isp.com:/var/qmail/control"
102 END
103   my($control)=&getvalue(":");
104   iscp("root\@$control/rcpthosts","$spooldir/rcpthosts.import");
105 #  iscp("root\@$control/recipientmap","$spooldir/recipientmap.import");
106   iscp("root\@$control/virtualdomains","$spooldir/virtualdomains.import");
107
108 #  print "\n\n", <<END;
109 #Enter the name of the machine with your user .qmail files, for example
110 #"mail.isp.com"
111 #END
112 #  print ":";
113 #  my($shellmachine)=&getvalue;
114
115 } elsif ( $mta{$mta} eq "sendmail" ) {
116
117   print "\n\n", <<END;
118 Enter the location and name of your sendmail virtual user table, for example
119 "mail.isp.com:/etc/virtusertable"
120 END
121   my($virtusertable)=&getvalue(":");
122   iscp("root\@$virtusertable","$spooldir/virtusertable.import");
123
124   print "\n\n", <<END;
125 Enter the location and name of your sendmail.cw file, for example
126 "mail.isp.com:/etc/sendmail.cw"
127 END
128   my($sendmail_cw)=&getvalue(":");
129   iscp("root\@$sendmail_cw","$spooldir/sendmail.cw.import");
130
131 } else {
132   die "Unknown MTA!\n";
133 }
134
135 sub getvalue {
136   my $prompt = shift;
137   $^W=0; #Term::Query isn't -w-safe
138   my $data = query $prompt, '';
139   $^W=1;
140   $data;
141 }
142
143 print "\n\n";
144
145 ###
146
147 $FS::svc_domain::whois_hack=1;
148 $FS::svc_acct_sm::nossh_hack=1;
149
150 if ( $mta{$mta} eq "qmail" ) {
151   open(RCPTHOSTS,"<$spooldir/rcpthosts.import")
152     or die "Can't open $spooldir/rcpthosts.import: $!";
153 } elsif ( $mta{$mta} eq "sendmail" ) {
154   open(RCPTHOSTS,"<$spooldir/sendmail.cw.import")
155     or die "Can't open $spooldir/sendmail.cw.import: $!";
156 } else {
157   die "Unknown MTA!\n";
158 }
159
160 my(%svcnum);
161
162 while (<RCPTHOSTS>) {
163   next if /^(#|$)/;
164   next if $mta{$mta} eq 'sendmail' && /^\s*$/; #blank lines
165   /^\.?([\w\-\.]+)$/
166     #or do { warn "Strange rcpthosts/sendmail.cw line: $_"; next; };
167     or die "Strange rcpthosts/sendmail.cw line: $_";
168   my $domain = $1;
169   my($svc_domain);
170   unless ( $svc_domain = qsearchs('svc_domain', {'domain'=>$domain} ) ) {
171     $svc_domain = new FS::svc_domain ({
172       'domain'  => $domain,
173       'svcpart' => $domain_svcpart,
174       'action'  => 'N',
175     });
176     my $error = $svc_domain->insert;
177     #warn $error if $error;
178     die $error if $error;
179   }
180   $svcnum{$domain}=$svc_domain->svcnum;
181 }
182 close RCPTHOSTS; 
183
184 #these two loops have enough similar parts they should probably be merged
185 if ( $mta{$mta} eq "qmail" ) {
186
187   open(VD_FIX,">$spooldir/virtualdomains.FIX");
188   print VD_FIX "#!/usr/bin/perl\n";
189
190   open(VIRTUALDOMAINS,"<$spooldir/virtualdomains.import")
191     or die "Can't open $spooldir/virtualdomains.import: $!";
192   while (<VIRTUALDOMAINS>) {
193     next if /^#/;
194     /^\.?([\w\-\.]+):(\w+)(\-([\w\-\.]+))?$/
195       #or do { warn "Strange virtualdomains line: $_"; next; };
196       or die "Strange virtualdomains line: $_";
197     my($domain,$username,$dash_ext,$extension)=($1,$2,$3,$4);
198     $dash_ext ||= '';
199     $extension ||= '';
200     my($svc_acct)=qsearchs('svc_acct',{'username'=>$username});
201     unless ( $svc_acct ) {
202       #warn "Unknown user $username in virtualdomains; skipping\n";
203       #die "Unknown user $username in virtualdomains; skipping\n";
204       next;
205     }
206     if ( $domain ne $extension ) {
207       #warn "virtualdomains line $domain:$username$dash_ext changed to $domain:$username-$domain\n";
208       my($dir)=$svc_acct->dir;
209       my($qdomain)=$domain;
210       $qdomain =~ s/\./:/g; #see manpage for 'dot-qmail': EXTENSION ADDRESSES
211       #example to move .qmail files for virtual domains to their new location 
212       #dry run
213       #issh("root\@$shellmachine",'perl -e \'foreach $a (<'. $dir. '/.qmail'. $dash_ext. '-*>) { $old=$a; $a =~ s/\\.qmail'. $dash_ext. '\\-/\\.qmail\\-'. $qdomain. '\\-/; print " $old -> $a\n"; }\'');
214       #the real thing
215       #issh("root\@$shellmachine",'perl -e \'foreach $a (<'. $dir. '/.qmail'. $dash_ext. '-*>) { $old=$a; $a =~ s/\\.qmail'. $dash_ext. '\\-/\\.qmail\\-'. $qdomain. '\\-/; rename $old, $a; }\'');
216       print VD_FIX <<END;
217 foreach \$file (<$dir/.qmail$dash_ext-*>) {
218   \$old = \$file;
219   \$file =~ s/\.qmail$dash_ext\-/\.qmail\-$qdomain\-/;
220   rename \$old, \$file;
221 }
222 END
223     }
224
225     unless ( exists $svcnum{$domain} ) {
226       my($svc_domain) = new FS::svc_domain ({
227         'domain'  => $domain,
228         'svcpart' => $domain_svcpart,
229         'action'  => 'N',
230       });
231       my $error = $svc_domain->insert;
232       #warn $error if $error;
233       die $error if $error;
234       $svcnum{$domain}=$svc_domain->svcnum;
235     }
236
237     my($svc_acct_sm)=new FS::svc_acct_sm ({
238       'domsvc'  => $svcnum{$domain},
239       'domuid'  => $svc_acct->uid,
240       'domuser' => '*',
241       'svcpart' => $mailalias_svcpart,
242     });
243     my($error)='';
244     $error=$svc_acct_sm->insert;
245     #warn $error if $error;
246     die $error, ", domain $domain" if $error;
247   }
248   close VIRTUALDOMAINS;
249   close VD_FIX;
250
251 } elsif ( $mta{$mta} eq "sendmail" ) {
252
253   open(VIRTUSERTABLE,"<$spooldir/virtusertable.import")
254     or die "Can't open $spooldir/virtusertable.import: $!";
255   while (<VIRTUSERTABLE>) {
256     next if /^#/; #comments?
257     next if /^\s*$/; #blank lines
258     /^([\w\-\.]+)?\@([\w\-\.]+)\t+([\w\-\.]+)$/
259       #or do { warn "Strange virtusertable line: $_"; next; };
260       or die "Strange virtusertable line: $_";
261     my($domuser,$domain,$username)=($1,$2,$3);
262     my($svc_acct)=qsearchs('svc_acct',{'username'=>$username});
263     unless ( $svc_acct ) {
264       #warn "Unknown user $username in virtusertable";
265       die "Unknown user $username in virtusertable";
266       next;
267     }
268     my($svc_acct_sm)=new FS::svc_acct_sm ({
269       'domsvc'  => $svcnum{$domain},
270       'domuid'  => $svc_acct->uid,
271       'domuser' => $domuser || '*',
272       'svcpart' => $mailalias_svcpart,
273     });
274     my($error)='';
275     $error=$svc_acct_sm->insert;
276     #warn $error if $error;
277     die $error if $error;
278   }
279   close VIRTUSERTABLE;
280
281 } else {
282   die "Unknown MTA!\n";
283 }
284
285 #open(RECIPIENTMAP,"<$spooldir/recipientmap.import");
286 #close RECIPIENTMAP;
287
288 print "\n\n", <<END if $mta{$mta} eq "qmail";
289 Don\'t forget to run $spooldir/virtualdomains.FIX before using
290 $spooldir/virtualdomains !
291 END
292
293 #
294
295 sub usage {
296   die "Usage:\n\n  svc_acct_sm.import user\n";
297 }
298