1 package FS::part_export::shellcommands;
3 use vars qw(@ISA @saltset);
4 use String::ShellQuote;
7 @ISA = qw(FS::part_export);
9 @saltset = ( 'a'..'z' , 'A'..'Z' , '0'..'9' , '.' , '/' );
11 sub rebless { shift; }
15 $self->_export_command('useradd', @_);
20 $self->_export_command('userdel', @_);
25 $self->_export_command('suspend', @_);
28 sub _export_unsuspend {
30 $self->_export_command('unsuspend', @_);
34 my ( $self, $action, $svc_acct) = (shift, shift, shift);
35 my $command = $self->option($action);
36 return '' if $command =~ /^\s*$/;
37 my $stdin = $self->option($action."_stdin");
42 ${$_} = $svc_acct->getfield($_) foreach $svc_acct->fields;
45 foreach my $acct_snarf ( $svc_acct->acct_snarf ) {
46 ${"snarf_$_$count"} = shell_quote( $acct_snarf->get($_) )
47 foreach qw( machine username _password );
52 my $cust_pkg = $svc_acct->cust_svc->cust_pkg;
54 $email = ( grep { $_ ne 'POST' } $cust_pkg->cust_main->invoicing_list )[0];
59 $finger = shell_quote $finger;
60 $quoted_password = shell_quote $_password;
61 $domain = $svc_acct->domain;
63 #eventually should check a "password-encoding" field
64 if ( length($svc_acct->_password) == 13
65 || $svc_acct->_password =~ /^\$(1|2a?)\$/ ) {
66 $crypt_password = shell_quote $svc_acct->_password;
68 $crypt_password = crypt(
70 $saltset[int(rand(64))].$saltset[int(rand(64))]
74 $self->shellcommands_queue( $svc_acct->svcnum,
75 user => $self->option('user')||'root',
76 host => $self->machine,
77 command => eval(qq("$command")),
78 stdin_string => eval(qq("$stdin")),
83 my($self, $new, $old ) = (shift, shift, shift);
84 my $command = $self->option('usermod');
85 my $stdin = $self->option('usermod_stdin');
89 ${"old_$_"} = $old->getfield($_) foreach $old->fields;
90 ${"new_$_"} = $new->getfield($_) foreach $new->fields;
92 $new_finger = shell_quote $new_finger;
93 $quoted_new__password = shell_quote $new__password; #old, wrong?
94 $new_quoted_password = shell_quote $new__password; #new, better?
95 $old_domain = $old->domain;
96 $new_domain = $new->domain;
98 #eventuall should check a "password-encoding" field
99 if ( length($new->_password) == 13
100 || $new->_password =~ /^\$(1|2a?)\$/ ) {
101 $new_crypt_password = shell_quote $new->_password;
103 $new_crypt_password =
104 crypt( $new->_password, $saltset[int(rand(64))].$saltset[int(rand(64))]
108 if ( $self->option('usermod_pwonly') ) {
110 if ( $old_username ne $new_username ) {
111 $error ||= "can't change username";
113 if ( $old_domain ne $new_domain ) {
114 $error ||= "can't change domain";
116 if ( $old_uid != $new_uid ) {
117 $error ||= "can't change uid";
119 if ( $old_dir ne $new_dir ) {
120 $error ||= "can't change dir";
122 return $error. ' ('. $self->exporttype. ' to '. $self->machine. ')'
125 $self->shellcommands_queue( $new->svcnum,
126 user => $self->option('user')||'root',
127 host => $self->machine,
128 command => eval(qq("$command")),
129 stdin_string => eval(qq("$stdin")),
133 #a good idea to queue anything that could fail or take any time
134 sub shellcommands_queue {
135 my( $self, $svcnum ) = (shift, shift);
136 my $queue = new FS::queue {
138 'job' => "FS::part_export::shellcommands::ssh_cmd",
140 $queue->insert( @_ );
143 sub ssh_cmd { #subroutine, not method
145 &Net::SSH::ssh_cmd( { @_ } );
148 #sub shellcommands_insert { #subroutine, not method
150 #sub shellcommands_replace { #subroutine, not method
152 #sub shellcommands_delete { #subroutine, not method