1 # Copyright (c) 2004 Ivan Kohler <ivan-rt@420.am>
2 # Copyright (c) 2008 Freeside Internet Services, Inc.
4 # This work is made available to you under the terms of Version 2 of
5 # the GNU General Public License. A copy of that license should have
6 # been provided with this software, but in any event can be snarfed
9 # This work is distributed in the hope that it will be useful, but
10 # WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 # General Public License for more details.
16 RT::Interface::Web_Vendor
22 Freeside vendor overlay for RT::Interface::Web.
26 use_ok(RT::Interface::Web_Vendor);
32 #package RT::Interface::Web;
35 package HTML::Mason::Commands;
37 no warnings qw(redefine);
39 =head2 ProcessTicketCustomers
43 sub ProcessTicketCustomers {
52 my $Ticket = $args{'TicketObj'};
53 my $ARGSRef = $args{'ARGSRef'};
54 my $Debug = $args{'Debug'};
55 my $me = 'ProcessTicketCustomers';
57 ### false laziness w/RT::Interface::Web::ProcessTicketLinks
58 # Delete links that are gone gone gone.
59 foreach my $arg ( keys %$ARGSRef ) {
60 if ( $arg =~ /DeleteLink-(.*?)-(DependsOn|MemberOf|RefersTo)-(.*)$/ ) {
66 "Trying to delete: Base: $base Target: $target Type $type";
67 my ( $val, $msg ) = $Ticket->DeleteLink( Base => $base,
82 my @svcnums = map { /^Ticket-AddService-(\d+)$/; $1 }
83 grep { /^Ticket-AddService-(\d+)$/ && $ARGSRef->{$_} }
87 foreach my $svcnum (@svcnums) {
88 my @link = ( 'Type' => 'MemberOf',
89 'Target' => "freeside://freeside/cust_svc/$svcnum",
92 my( $val, $msg ) = $Ticket->AddLink(@link);
102 push @custnums, map { /^Ticket-AddCustomer-(\d+)$/; $1 }
103 grep { /^Ticket-AddCustomer-(\d+)$/ && $ARGSRef->{$_} }
106 #my @delete_custnums =
107 # map { /^Ticket-AddCustomer-(\d+)$/; $1 }
108 # grep { /^Ticket-AddCustomer-(\d+)$/ && $ARGSRef->{$_} }
112 #figure out if we're going to auto-link requestors, and find them if so
115 my $num_cur_cust = $Ticket->Customers->Count;
116 my $num_new_cust = scalar(@custnums);
117 warn "$me: $num_cur_cust current customers / $num_new_cust new customers\n"
120 #if we're linking the first ticket to one customer
121 my $link_requestors = ( $num_cur_cust == 0 && $num_new_cust == 1 );
122 warn "$me: adding a single customer to a previously customerless".
123 " ticket, so linking customers to requestor too\n"
124 if $Debug && $link_requestors;
127 if ( $link_requestors ) {
129 #find any requestors without customers
131 grep { ! $_->Customers->Count }
132 @{ $Ticket->Requestors->UserMembersObj->ItemsArrayRef };
134 warn "$me: found ". scalar(@Requestors). " requestors without".
135 " customers; linking them\n"
141 #remove any declared non-customer addresses
144 my $exclude_regexp = RT->Config->Get('NonCustomerEmailRegexp');
145 @Requestors = grep { not $_->EmailAddress =~ $exclude_regexp } @Requestors
146 if defined $exclude_regexp;
149 #link ticket (and requestors) to customers
152 foreach my $custnum ( @custnums ) {
154 my @link = ( 'Type' => 'MemberOf',
155 'Target' => "freeside://freeside/cust_main/$custnum",
158 my( $val, $msg ) = $Ticket->AddLink(@link);
161 #add customer links to requestors
162 foreach my $Requestor ( @Requestors ) {
163 my( $val, $msg ) = $Requestor->AddLink(@link);
165 warn "$me: linking requestor to custnum $custnum: $msg\n"
175 #false laziness w/above... eventually it should go away in favor of this
176 sub ProcessObjectCustomers {
184 my $Object = $args{'Object'};
185 my $ARGSRef = $args{'ARGSRef'};
187 ### false laziness w/RT::Interface::Web::ProcessTicketLinks
188 # Delete links that are gone gone gone.
189 foreach my $arg ( keys %$ARGSRef ) {
190 if ( $arg =~ /DeleteLink-(.*?)-(DependsOn|MemberOf|RefersTo)-(.*)$/ ) {
196 "Trying to delete: Base: $base Target: $target Type $type";
197 my ( $val, $msg ) = $Object->DeleteLink( Base => $base,
208 #my @delete_custnums =
209 # map { /^Object-AddCustomer-(\d+)$/; $1 }
210 # grep { /^Object-AddCustomer-(\d+)$/ && $ARGSRef->{$_} }
213 my @custnums = map { /^Object-AddCustomer-(\d+)$/; $1 }
214 grep { /^Object-AddCustomer-(\d+)$/ && $ARGSRef->{$_} }
217 foreach my $custnum ( @custnums ) {
219 $Object->AddLink( 'Type' => 'MemberOf',
220 'Target' => "freeside://freeside/cust_main/$custnum",
229 =head2 ProcessTicketBasics ( TicketObj => $Ticket, ARGSRef => \%ARGS );
231 Updates all core ticket fields except Status, and returns an array of results
236 sub ProcessTicketBasics {
244 my $TicketObj = $args{'TicketObj'};
245 my $ARGSRef = $args{'ARGSRef'};
247 # {{{ Set basic fields
260 # the UI for editing WillResolve through Ticket Basics should allow
262 if ( exists $ARGSRef->{'WillResolve_Date'} ) {
263 my $to_date = delete($ARGSRef->{'WillResolve_Date'});
264 my $DateObj = RT::Date->new($session{'CurrentUser'});
266 $DateObj->Set(Format => 'unknown', Value => $to_date);
267 if ( $DateObj->Unix > time ) {
268 $ARGSRef->{'WillResolve'} = $DateObj->ISO;
270 warn "Ticket ".$TicketObj->Id.": WillResolve date '$to_date' not accepted.\n";
271 # and then don't set it in ARGSRef
273 } elsif ( $TicketObj and $TicketObj->WillResolveObj->Unix > 0 ) {
274 $DateObj->Set(Value => 0);
275 $ARGSRef->{'WillResolve'} = $DateObj->ISO;
279 if ( $ARGSRef->{'Queue'} and ( $ARGSRef->{'Queue'} !~ /^(\d+)$/ ) ) {
280 my $tempqueue = RT::Queue->new($RT::SystemUser);
281 $tempqueue->Load( $ARGSRef->{'Queue'} );
282 if ( $tempqueue->id ) {
283 $ARGSRef->{'Queue'} = $tempqueue->id;
287 # RT core _will_ allow Set transactions that change these
288 # fields to empty strings, but internally change the values
289 # to zero. This is sloppy and causes some problems.
290 foreach my $field (qw(TimeWorked TimeEstimated TimeLeft)) {
291 if (exists $ARGSRef->{$field}) {
292 $ARGSRef->{$field} =~ s/\s//g;
293 $ARGSRef->{$field} ||= 0;
297 my @results = UpdateRecordObject(
298 AttributesRef => \@attribs,
299 Object => $TicketObj,
303 # We special case owner changing, so we can use ForceOwnerChange
304 if ( $ARGSRef->{'Owner'} && ( $TicketObj->Owner != $ARGSRef->{'Owner'} ) ) {
306 if ( $ARGSRef->{'ForceOwnerChange'} ) {
307 $ChownType = "Force";
312 my ( $val, $msg ) = $TicketObj->SetOwner( $ARGSRef->{'Owner'}, $ChownType );
313 push( @results, $msg );
319 =head2 ProcessTicketDates (TicketObj => RT::Ticket, ARGSRef => {})
321 Process updates to the Starts, Started, Told, Resolved, and WillResolve
326 sub ProcessTicketDates {
333 my $Ticket = $args{'TicketObj'};
334 my $ARGSRef = $args{'ARGSRef'};
338 # {{{ Set date fields
339 my @date_fields = qw(
348 #Run through each field in this list. update the value if apropriate
349 foreach my $field (@date_fields) {
350 next unless exists $ARGSRef->{ $field . '_Date' };
351 next if $ARGSRef->{ $field . '_Date' } eq '';
355 my $DateObj = RT::Date->new( $session{'CurrentUser'} );
358 Value => $ARGSRef->{ $field . '_Date' }
361 if ( $field eq 'WillResolve'
362 and $DateObj->Unix > 0
363 and $DateObj->Unix <= time ) {
364 push @results, "Can't set WillResolve date in the past.";
368 my $obj = $field . "Obj";
369 if ( ( defined $DateObj->Unix )
370 and ( $DateObj->Unix != $Ticket->$obj()->Unix() ) )
372 my $method = "Set$field";
373 my ( $code, $msg ) = $Ticket->$method( $DateObj->ISO );
374 push @results, "$msg";
382 =head2 ProcessTicketStatus (TicketObj => RT::Ticket, ARGSRef => {})
384 Process updates to the 'Status' field of the ticket. If the new value
385 of Status is 'resolved', this will check required custom fields before
390 sub ProcessTicketStatus {
397 my $TicketObj = $args{'TicketObj'};
398 my $ARGSRef = $args{'ARGSRef'};
401 return () if !$ARGSRef->{'Status'};
403 if ( lc( $ARGSRef->{'Status'} ) eq 'resolved' ) {
404 foreach my $field ( $TicketObj->MissingRequiredFields ) {
405 push @results, loc('Missing required field: [_1]', $field->Name);
409 $m->notes('RedirectToBasics' => 1);
413 return UpdateRecordObject(
414 AttributesRef => [ 'Status' ],
415 Object => $TicketObj,
420 sub default_FormatDate { $_[0]->AsString }
422 sub ProcessColumnMapValue {
424 my %args = ( Arguments => [],
426 FormatDate => \&default_FormatDate,
430 if ( ref $value eq 'RT::Date' ) {
431 return $args{FormatDate}->($value);
432 } elsif ( UNIVERSAL::isa( $value, 'CODE' ) ) {
433 my @tmp = $value->( @{ $args{'Arguments'} } );
434 return ProcessColumnMapValue( ( @tmp > 1 ? \@tmp : $tmp[0] ), %args );
435 } elsif ( UNIVERSAL::isa( $value, 'ARRAY' ) ) {
436 return join '', map ProcessColumnMapValue( $_, %args ), @$value;
437 } elsif ( UNIVERSAL::isa( $value, 'SCALAR' ) ) {
442 return $m->interp->apply_escapes( $value, 'h' ) if $args{'Escape'};