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