package FS::part_event; use strict; use base qw( FS::m2name_Common FS::option_Common ); use vars qw( $DEBUG ); use Carp qw(confess); use FS::Record qw( dbh qsearch qsearchs ); use FS::Conf; use FS::Cursor; use FS::part_event_option; use FS::part_event_condition; use FS::cust_event; $DEBUG = 0; =head1 NAME FS::part_event - Object methods for part_event records =head1 SYNOPSIS use FS::part_event; $record = new FS::part_event \%hash; $record = new FS::part_event { 'column' => 'value' }; $error = $record->insert( { 'option' => 'value' } ); $error = $record->insert( \%options ); $error = $new_record->replace($old_record); $error = $record->delete; $error = $record->check; $error = $record->do_event( $direct_object ); =head1 DESCRIPTION An FS::part_event object represents an event definition - a billing, collection or other callback which is triggered when certain customer, invoice, package or other conditions are met. FS::part_event inherits from FS::Record. The following fields are currently supported: =over 4 =item eventpart - primary key =item agentnum - Optional agentnum (see L) =item event - event name =item eventtable - table name against which this event is triggered: one of "cust_main", "cust_bill", "cust_statement", "cust_pkg", "svc_acct". =item check_freq - how often events of this type are checked; currently "1d" (daily) and "1m" (monthly) are recognized. Note that the apprioriate freeside-daily and/or freeside-monthly cron job needs to be in place. =item weight - ordering for events =item action - event action (like part_bill_event.plan - eventcode plan) =item disabled - Disabled flag, empty or `Y' =back =head1 METHODS =over 4 =item new HASHREF Creates a new invoice event definition. To add the invoice event definition to the database, see L<"insert">. Note that this stores the hash reference, not a distinct copy of the hash it points to. You can ask the object for a copy with the I method. =cut # the new method can be inherited from FS::Record, if a table method is defined sub table { 'part_event'; } =item insert [ HASHREF ] Adds this record to the database. If there is an error, returns the error, otherwise returns false. If a list or hash reference of options is supplied, part_export_option records are created (see L). =cut # the insert method can be inherited from FS::Record =item delete Delete this record from the database. =cut # the delete method can be inherited from FS::Record =item replace OLD_RECORD [ HASHREF | OPTION => VALUE ... ] Replaces the OLD_RECORD with this one in the database. If there is an error, returns the error, otherwise returns false. If a list or hash reference of options is supplied, part_event_option records are created or modified (see L). =cut # the replace method can be inherited from FS::Record =item check Checks all fields to make sure this is a valid invoice event definition. If there is an error, returns the error, otherwise returns false. Called by the insert and replace methods. =cut # the check method should currently be supplied - FS::Record contains some # data checking routines sub check { my $self = shift; $self->weight(0) unless $self->weight; my $error = $self->ut_numbern('eventpart') || $self->ut_text('event') || $self->ut_enum('eventtable', [ $self->eventtables ] ) || $self->ut_enum('check_freq', [ '1d', '1m' ]) || $self->ut_number('weight') || $self->ut_alpha('action') || $self->ut_enum('disabled', [ '', 'Y' ] ) || $self->ut_agentnum_acl('agentnum', 'Edit global billing events') ; return $error if $error; #XXX check action to make sure a module exists? # well it'll die in _rebless... $self->SUPER::check; } =item _rebless Reblesses the object into the FS::part_event::Action::ACTION class, where ACTION is the object's I field. =cut sub _rebless { my $self = shift; my $action = $self->action or return $self; #my $class = ref($self). "::$action"; my $class = "FS::part_event::Action::$action"; eval "use $class"; die $@ if $@; bless($self, $class); # unless $@; $self; } =item part_event_condition Returns the conditions associated with this event, as FS::part_event_condition objects (see L) =item new_cust_event OBJECT, [ OPTION => VALUE ] Creates a new customer event (see L) for the provided object. The only option allowed is 'time', to set the "current" time for the event. =cut sub new_cust_event { my( $self, $object, %opt ) = @_; confess "**** $object is not a ". $self->eventtable if ref($object) ne "FS::". $self->eventtable; my $pkey = $object->primary_key; new FS::cust_event { 'eventpart' => $self->eventpart, 'tablenum' => $object->$pkey(), #'_date' => time, #i think we always want the real "now" here. '_date' => ($opt{'time'} || time), 'status' => 'new', }; } #surely this doesn't work sub reasontext { confess "part_event->reasontext deprecated"; } #=item reasontext # #Returns the text of any reason associated with this event. # #=cut # #sub reasontext { # my $self = shift; # my $r = qsearchs('reason', { 'reasonnum' => $self->reason }); # if ($r){ # $r->reason; # }else{ # ''; # } #} =item agent Returns the associated agent for this event, if any, as an FS::agent object. =item templatename Returns the alternate invoice template name, if any, or false if there is no alternate template for this event. =cut sub templatename { my $self = shift; if ( $self->action =~ /^cust_bill_send_(alternate|agent)$/ && ( $self->option('agent_templatename') || $self->option('templatename') ) ) { $self->option('agent_templatename') || $self->option('templatename'); } else { ''; } } =item targets OPTIONS Returns all objects (of type C, for this object's C) eligible for processing under this event, as of right now. The L object used to test event conditions will be included in each object as the 'cust_event' pseudo-field. This is not used in normal event processing (which is done on a per-customer basis to control timing of pre- and post-billing events) but can be useful when configuring events. =cut sub targets { # may want to cursor this also my $self = shift; my %opt = @_; my $time = $opt{'time'} ||= time; my $query = $self->_target_query(%opt); my @objects = qsearch($query); my @tested_objects; foreach my $object ( @objects ) { my $cust_event = $self->new_cust_event($object, 'time' => $time); next unless $cust_event->test_conditions; $object->set('cust_event', $cust_event); push @tested_objects, $object; } @tested_objects; } sub _target_query { my $self = shift; my %opt = @_; my $time = $opt{'time'}; my $eventpart = $self->eventpart; $eventpart =~ /^\d+$/ or die "bad eventpart $eventpart"; my $eventtable = $self->eventtable; # find all objects that meet the conditions for this part_event my $linkage = ''; # this is the 'object' side of the FROM clause if ( $eventtable ne 'cust_main' ) { $linkage = ($self->eventtables_cust_join->{$eventtable} || '') . ' LEFT JOIN cust_main USING (custnum) '; } # this is the 'event' side my $join = FS::part_event_condition->join_conditions_sql( $eventtable, 'time' => $time ); my $where = FS::part_event_condition->where_conditions_sql( $eventtable, 'time' => $time ); $join = $linkage . " INNER JOIN part_event ON ( part_event.eventpart = $eventpart ) $join"; $where .= ' AND cust_main.agentnum = '.$self->agentnum if $self->agentnum; # don't enforce check_freq since this is a special, out-of-order check # and don't enforce disabled because we want to be able to see targets # for a disabled event { table => $eventtable, hashref => {}, addl_from => $join, extra_sql => "WHERE $where", }; } =item initialize PARAMS Identify all objects eligible for this event and create L records for each of them, as of the present time, with status "initial". When combined with conditions that prevent an event from running more than once (at all or within some period), this will exclude any objects that met the conditions before the event was created. If an L object needs to be initialized, it should be created in a disabled state to avoid running the event prematurely for any existing objects. C will enable it once all the cust_event records have been created. This may take some time, so it should be run from the job queue. =cut sub initialize { my $self = shift; my $error; my $time = time; local $FS::UID::AutoCommit = 1; my $cursor = FS::Cursor->new( $self->_target_query('time' => $time) ); while (my $object = $cursor->fetch) { my $cust_event = $self->new_cust_event($object, 'time' => $time); next unless $cust_event->test_conditions; $cust_event->status('initial'); $error = $cust_event->insert; die $error if $error; } # on successful completion only, re-enable the event if ( $self->disabled ) { $self->disabled(''); $error = $self->replace; die $error if $error; } return; } =cut =back =head1 CLASS METHODS =over 4 =item eventtable_labels Returns a hash reference of labels for eventtable values, i.e. 'cust_main'=>'Customer' =cut sub eventtable_labels { #my $class = shift; tie my %hash, 'Tie::IxHash', 'cust_pkg' => 'Package', 'cust_bill' => 'Invoice', 'cust_main' => 'Customer', 'cust_pay' => 'Payment', 'cust_pay_batch' => 'Batch payment', 'cust_statement' => 'Statement', #too general a name here? "Invoice group"? 'svc_acct' => 'Account service (svc_acct)', ; \%hash } =item eventtable_pkey_sql Returns a hash reference of full SQL primary key names for eventtable values, i.e. 'cust_main'=>'cust_main.custnum' =cut sub eventtable_pkey_sql { my $class = shift; my $hashref = $class->eventtable_pkey; my %hash = map { $_ => "$_.". $hashref->{$_} } keys %$hashref; \%hash; } =item eventtable_pkey Returns a hash reference of full SQL primary key names for eventtable values, i.e. 'cust_main'=>'custnum' =cut sub eventtable_pkey { #my $class = shift; { 'cust_main' => 'custnum', 'cust_bill' => 'invnum', 'cust_pkg' => 'pkgnum', 'cust_pay' => 'paynum', 'cust_pay_batch' => 'paybatchnum', 'cust_statement' => 'statementnum', 'svc_acct' => 'svcnum', }; } =item eventtables Returns a list of eventtable values (default ordering; suited for display). =cut sub eventtables { my $class = shift; my $eventtables = $class->eventtable_labels; keys %$eventtables; } =item eventtables_runorder Returns a list of eventtable values (run order). =cut sub eventtables_runorder { shift->eventtables; #same for now } =item eventtables_cust_join Returns a hash reference of SQL expressions to join each eventtable to a table with a 'custnum' field. =cut sub eventtables_cust_join { my %hash = ( 'svc_acct' => 'LEFT JOIN cust_svc USING (svcnum) LEFT JOIN cust_pkg USING (pkgnum)', ); \%hash; } =item eventtables_custnum Returns a hash reference of SQL expressions for the 'custnum' field when I is in effect. The default is "$eventtable.custnum". =cut sub eventtables_custnum { my %hash = ( map({ $_, "$_.custnum" } shift->eventtables), 'svc_acct' => 'cust_pkg.custnum' ); \%hash; } =item check_freq_labels Returns a hash reference of labels for check_freq values, i.e. '1d'=>'daily' =cut sub check_freq_labels { #my $class = shift; #Tie::IxHash?? { '1d' => 'daily', '1m' => 'monthly', }; } =item actions [ EVENTTABLE ] Return information about the available actions. If an eventtable is specified, only return information about actions available for that eventtable. Information is returned as key-value pairs. Keys are event names. Values are hashrefs with the following keys: =over 4 =item description =item eventtable_hashref =item option_fields =item default_weight =item deprecated =back =head1 ADDING NEW EVENTTABLES To add an eventtable, you must: =over 4 =item Add the table to "eventtable_labels" (with a label) and to "eventtable_pkey" (with its primary key). =item If the table doesn't have a "custnum" field of its own (such as a svc_x table), add a suitable join expression to eventtables_cust_join and an expression for the final custnum field to eventtables_custnum. =item Create a method named FS::cust_main->$eventtable(): a wrapper around qsearch() to return all records in the new table belonging to the cust_main object. This method must accept 'addl_from' and 'extra_sql' arguments in the way qsearch() does. For svc_ tables, wrap the svc_x() method. =item Add it to FS::cust_event->join_sql and search_sql_where so that search/cust_event.html will find it. =item Create a UI link/form to search for events linked to objects in the new eventtable, using search/cust_event.html. Place this somewhere appropriate to the eventtable. =back See L for more information. =cut #false laziness w/part_event_condition.pm #some false laziness w/part_export & part_pkg my %actions; foreach my $INC ( @INC ) { foreach my $file ( glob("$INC/FS/part_event/Action/*.pm") ) { warn "attempting to load Action from $file\n" if $DEBUG; $file =~ /\/(\w+)\.pm$/ or do { warn "unrecognized file in $INC/FS/part_event/Action/: $file\n"; next; }; my $mod = $1; eval "use FS::part_event::Action::$mod;"; if ( $@ ) { die "error using FS::part_event::Action::$mod (skipping): $@\n" if $@; #warn "error using FS::part_event::Action::$mod (skipping): $@\n" if $@; #next; } $actions{$mod} = { ( map { $_ => "FS::part_event::Action::$mod"->$_() } qw( description eventtable_hashref default_weight deprecated will_send_invoice ) #option_fields_hashref ), 'option_fields' => [ "FS::part_event::Action::$mod"->option_fields() ], }; } } sub actions { my( $class, $eventtable ) = @_; ( map { $_ => $actions{$_} } sort { $actions{$a}->{'default_weight'} <=> $actions{$b}->{'default_weight'} || $actions{$a}->{'description'} cmp $actions{$b}->{'description'} } $class->all_actions( $eventtable ) ); } =item all_actions [ EVENTTABLE ] Returns a list of just the action names =cut sub all_actions { my ( $class, $eventtable ) = @_; grep { !$eventtable || $actions{$_}->{'eventtable_hashref'}{$eventtable} } keys %actions } =item invoice_send_actions Returns a list of action names that send invoices =cut sub invoice_send_actions { my ( $class ) = @_; grep { $actions{$_}->{'will_send_invoice'} } keys %actions } =item process_initialize 'eventpart' => EVENTPART Job queue wrapper for "initialize". EVENTPART identifies the L object to initialize. =cut sub process_initialize { my %opt = @_; my $part_event = qsearchs('part_event', { eventpart => $opt{'eventpart'}}) or die "eventpart '$opt{eventpart}' not found!\n"; $part_event->initialize; } sub _upgrade_data { #class method my ($class, %opts) = @_; foreach my $part_event ( qsearch('part_event', { 'action' => 'cust_bill_realtime_card' }), qsearch('part_event', { 'action' => 'cust_bill_realtime_check' }), ) { $part_event->action('realtime_auto'); my $error = $part_event->replace; die $error if $error; } } =back =head1 SEE ALSO L, L, L, L, L, L, L, L, schema.html from the base documentation. =cut 1;