fix RADIUS usergroup editing
[freeside.git] / httemplate / edit / part_svc.cgi
1 <%
2 my $part_svc;
3 my $clone = '';
4 if ( $cgi->param('clone') && $cgi->param('clone') =~ /^(\d+)$/ ) {#clone
5   #$cgi->param('clone') =~ /^(\d+)$/ or die "malformed query: $query";
6   $part_svc = qsearchs('part_svc', { 'svcpart'=>$1 } )
7     or die "unknown svcpart: $1";
8   $clone = $part_svc->svcpart;
9   $part_svc->svcpart('');
10 } elsif ( $cgi->keywords ) { #edit
11   my($query) = $cgi->keywords;
12   $query =~ /^(\d+)$/ or die "malformed query: $query";
13   $part_svc=qsearchs('part_svc', { 'svcpart'=>$1 } )
14     or die "unknown svcpart: $1";
15 } else { #adding
16   $part_svc = new FS::part_svc {};
17 }
18
19 my $action = $part_svc->svcpart ? 'Edit' : 'Add';
20 my $hashref = $part_svc->hashref;
21 #   my $p_svcdb = $part_svc->svcdb || 'svc_acct';
22
23
24            #" onLoad=\"visualize()\""
25 %>
26 <%= header("$action Service Definition",
27            menubar( 'Main Menu'         => $p,
28                     'View all service definitions' => "${p}browse/part_svc.cgi"
29                   ),
30            )
31 %>
32
33 <FORM NAME="dummy">
34
35       Service Part #<%= $part_svc->svcpart ? $part_svc->svcpart : "(NEW)" %>
36 <BR><BR>
37 Service  <INPUT TYPE="text" NAME="svc" VALUE="<%= $hashref->{svc} %>"><BR>
38 Disable new orders <INPUT TYPE="checkbox" NAME="disabled" VALUE="Y"<%= $hashref->{disabled} eq 'Y' ? ' CHECKED' : '' %>><BR>
39 <INPUT TYPE="hidden" NAME="svcpart" VALUE="<%= $hashref->{svcpart} %>">
40 <BR>
41 Services are items you offer to your customers.
42 <UL><LI>svc_acct - Shell accounts, POP mailboxes, SLIP/PPP and ISDN accounts
43     <LI>svc_domain - Domains
44     <LI>svc_forward - mail forwarding
45     <LI>svc_www - Virtual domain website
46     <LI>svc_broadband - Broadband/High-speed Internet service
47     <LI>svc_external - Externally-tracked service
48 <!--   <LI>svc_charge - One-time charges (Partially unimplemented)
49        <LI>svc_wo - Work orders (Partially unimplemented)
50 -->
51 </UL>
52 For the selected table, you can give fields default or fixed (unchangable)
53 values.  For example, a SLIP/PPP account may have a default (or perhaps fixed)
54 <B>slipip</B> of <B>0.0.0.0</B>, while a POP mailbox will probably have a fixed
55 blank <B>slipip</B> as well as a fixed shell something like <B>/bin/true</B> or
56 <B>/usr/bin/passwd</B>.
57 <BR><BR>
58
59 <%
60
61 my %vfields;
62
63 #these might belong somewhere else for other user interfaces 
64 #pry need to eventually create stuff that's shared amount UIs
65 my $conf = new FS::Conf;
66 my %defs = (
67   'svc_acct' => {
68     'dir'       => 'Home directory',
69     'uid'       => 'UID (set to fixed and blank for no UIDs)',
70     'slipip'    => 'IP address',
71 #    'popnum'    => qq!<A HREF="$p/browse/svc_acct_pop.cgi/">POP number</A>!,
72     'popnum'    => {
73                      desc => 'Access number',
74                      type => 'select',
75                      select_table => 'svc_acct_pop',
76                      select_key   => 'popnum',
77                      select_label => 'city',
78                    },
79     'username'  => {
80                       desc => 'Username',
81                       type => 'disabled',
82                    },
83     'quota'     => '',
84     '_password' => 'Password',
85     'gid'       => 'GID (when blank, defaults to UID)',
86     'shell'     => {
87                      desc =>'Shell (all service definitions should have a default or fixed shell that is present in the <b>shells</b> configuration file, set to blank for no shell tracking)',
88                      type =>'select',
89                      select_list => [ $conf->config('shells') ],
90                    },
91     'finger'    => 'GECOS',
92     'domsvc'    => {
93                      desc =>'svcnum from svc_domain',
94                      type =>'select',
95                      select_table => 'svc_domain',
96                      select_key   => 'svcnum',
97                      select_label => 'domain',
98                    },
99     'usergroup' => {
100                      desc =>'RADIUS groups',
101                      type =>'radius_usergroup_selector',
102                    },
103   },
104   'svc_domain' => {
105     'domain'    => 'Domain',
106   },
107   'svc_forward' => {
108     'srcsvc'    => 'service from which mail is to be forwarded',
109     'dstsvc'    => 'service to which mail is to be forwarded',
110     'dst'       => 'someone@another.domain.com to use when dstsvc is 0',
111   },
112 #  'svc_charge' => {
113 #    'amount'    => 'amount',
114 #  },
115 #  'svc_wo' => {
116 #    'worker'    => 'Worker',
117 #    '_date'      => 'Date',
118 #  },
119   'svc_www' => {
120     #'recnum' => '',
121     #'usersvc' => '',
122   },
123   'svc_broadband' => {
124     'speed_down' => 'Maximum download speed for this service in Kbps.  0 denotes unlimited.',
125     'speed_up' => 'Maximum upload speed for this service in Kbps.  0 denotes unlimited.',
126     'ip_addr' => 'IP address.  Leave blank for automatic assignment.',
127     'blocknum' => 'Address block.',
128   },
129   'svc_external' => {
130     #'id' => '',
131     #'title' => '',
132   },
133 );
134
135   foreach my $svcdb (grep dbdef->table($_), keys %defs ) {
136     my $self = "FS::$svcdb"->new;
137     $vfields{$svcdb} = {};
138     foreach my $field ($self->virtual_fields) { # svc_Common::virtual_fields with a null svcpart returns all of them
139       my $pvf = $self->pvf($field);
140       my @list = $pvf->list;
141       if (scalar @list) {
142         $defs{$svcdb}->{$field} = { desc        => $pvf->label,
143                                     type        => 'select',
144                                     select_list => \@list };
145       } else {
146         $defs{$svcdb}->{$field} = $pvf->label;
147       } #endif
148       $vfields{$svcdb}->{$field} = $pvf;
149       warn "\$vfields{$svcdb}->{$field} = $pvf";
150     } #next $field
151   } #next $svcdb
152   
153   my @dbs = $hashref->{svcdb}
154              ? ( $hashref->{svcdb} )
155              : qw( svc_acct svc_domain svc_forward svc_www svc_broadband svc_external );
156
157   tie my %svcdb, 'Tie::IxHash', map { $_=>$_ } grep dbdef->table($_), @dbs;
158   my $widget = new HTML::Widgets::SelectLayers(
159     #'selected_layer' => $p_svcdb,
160     'selected_layer' => $hashref->{svcdb} || 'svc_acct',
161     'options'        => \%svcdb,
162     'form_name'      => 'dummy',
163     #'form_action'    => 'process/part_svc.cgi',
164     'form_action'    => 'part_svc.cgi', #self
165     'form_text'      => [ qw( svc svcpart ) ],
166     'form_checkbox'  => [ 'disabled' ],
167     'layer_callback' => sub {
168       my $layer = shift;
169       
170       my $html = qq!<INPUT TYPE="hidden" NAME="svcdb" VALUE="$layer">!;
171
172       my $columns = 3;
173       my $count = 0;
174       my @part_export =
175         map { qsearch( 'part_export', {exporttype => $_ } ) }
176           keys %{FS::part_export::export_info($layer)};
177       $html .= '<BR><BR>'. table().
178                table(). "<TR><TH COLSPAN=$columns>Exports</TH></TR><TR>";
179       foreach my $part_export ( @part_export ) {
180         $html .= '<TD><INPUT TYPE="checkbox"'.
181                  ' NAME="exportnum'. $part_export->exportnum. '"  VALUE="1" ';
182         $html .= 'CHECKED'
183           if ( $clone || $part_svc->svcpart ) #null svcpart search causing error
184               && qsearchs( 'export_svc', {
185                                    exportnum => $part_export->exportnum,
186                                    svcpart   => $clone || $part_svc->svcpart });
187         $html .= '>'. $part_export->exportnum. ': '. $part_export->exporttype.
188                  ' to '. $part_export->machine. '</TD>';
189         $count++;
190         $html .= '</TR><TR>' unless $count % $columns;
191       }
192       $html .= '</TR></TABLE><BR><BR>';
193
194       $html .=  table(). "<TH>Field</TH><TH COLSPAN=2>Modifier</TH>";
195       #yucky kludge
196       my @fields = defined( dbdef->table($layer) )
197                       ? grep { $_ ne 'svcnum' } fields($layer)
198                       : ();
199       push @fields, 'usergroup' if $layer eq 'svc_acct'; #kludge
200       $part_svc->svcpart($clone) if $clone; #haha, undone below
201       foreach my $field (@fields) {
202         my $part_svc_column = $part_svc->part_svc_column($field);
203         my $value = $part_svc_column->columnvalue;
204         my $flag = $part_svc_column->columnflag;
205         my $def = $defs{$layer}{$field};
206         my $desc = ref($def) ? $def->{desc} : $def;
207         
208         $html .= "<TR><TD>$field";
209         $html .= "- <FONT SIZE=-1>$desc</FONT>" if $desc;
210         $html .=  "</TD>";
211         $flag = '' if ref($def) && $def->{type} eq 'disabled';
212         $html .=
213           qq!<TD><INPUT TYPE="radio" NAME="${layer}__${field}_flag" VALUE=""!.
214           ' CHECKED'x($flag eq ''). ">Off</TD>".
215           '<TD>';
216         unless ( ref($def) && $def->{type} eq 'disabled' ) {
217           $html .= 
218             qq!<INPUT TYPE="radio" NAME="${layer}__${field}_flag" VALUE="D"!.
219             ' CHECKED'x($flag eq 'D'). ">Default ".
220             qq!<INPUT TYPE="radio" NAME="${layer}__${field}_flag" VALUE="F"!.
221             ' CHECKED'x($flag eq 'F'). ">Fixed ";
222           $html .= '<BR>';
223         }
224         if ( ref($def) ) {
225           if ( $def->{type} eq 'select' ) {
226             $html .= qq!<SELECT NAME="${layer}__${field}">!;
227             $html .= '<OPTION> </OPTION>' unless $value;
228             if ( $def->{select_table} ) {
229               foreach my $record ( qsearch( $def->{select_table}, {} ) ) {
230                 my $rvalue = $record->getfield($def->{select_key});
231                 $html .= qq!<OPTION VALUE="$rvalue"!.
232                          ( $rvalue==$value ? ' SELECTED>' : '>' ).
233                          $record->getfield($def->{select_label}). '</OPTION>';
234               } #next $record
235             } else { # select_list
236               foreach my $item ( @{$def->{select_list}} ) {
237                 $html .= qq!<OPTION VALUE="$item"!.
238                          ( $item eq $value ? ' SELECTED>' : '>' ).
239                          $item. '</OPTION>';
240               } #next $item
241             } #endif
242             $html .= '</SELECT>';
243           } elsif ( $def->{type} eq 'radius_usergroup_selector' ) {
244             $html .= FS::svc_acct::radius_usergroup_selector(
245               [ split(',', $value) ], "${layer}__${field}" );
246           } elsif ( $def->{type} eq 'disabled' ) {
247             $html .=
248               qq!<INPUT TYPE="hidden" NAME="${layer}__${field}" VALUE="">!;
249           } else {
250             $html .= '<font color="#ff0000">unknown type'. $def->{type};
251           }
252         } else {
253           $html .=
254             qq!<INPUT TYPE="text" NAME="${layer}__${field}" VALUE="$value">!;
255         }
256
257         if($vfields{$layer}->{$field}) {
258           $html .= qq!<BR><INPUT TYPE="radio" NAME="${layer}__${field}_flag" VALUE="X"!.
259           ' CHECKED'x($flag eq 'X'). ">Excluded ";
260         }
261         $html .= "</TD></TR>\n";
262       }
263       $part_svc->svcpart('') if $clone; #undone
264       $html .= "</TABLE>";
265
266       $html .= include('/elements/progress-init.html',
267                          $layer, #form name
268                          [ qw(svc svcpart disabled exportnum), @fields ],
269                          'process/part_svc.cgi',
270                          $p.'browse/part_svc.cgi',
271                          $layer,
272                       );
273       $html .= '<BR><INPUT NAME="submit" TYPE="button" VALUE="'.
274                ($hashref->{svcpart} ? 'Apply changes' : 'Add service'). '" '.
275                ' onClick="document.'. "$layer.submit.disabled=true; ".
276                "fixup(document.$layer); $layer". 'process();">';
277
278       #$html .= '<BR><INPUT TYPE="submit" VALUE="'.
279       #         ($hashref->{svcpart} ? 'Apply changes' : 'Add service'). '">';
280
281       $html;
282
283     },
284   );
285
286 %>
287 Table <%= $widget->html %>
288   </BODY>
289 </HTML>
290