X-Git-Url: http://git.freeside.biz/gitweb/?p=freeside.git;a=blobdiff_plain;f=FS%2FFS%2Fcust_location.pm;h=21bf92feba0e75dcb246dc7faf57ca7b21483910;hp=fdc2cf8da8544356d1e742a95c9a1a297db1447a;hb=20f03d52cc6c930f610c0b4466eeeeda54fdbb40;hpb=cf54023e010df76e0c39ac70902877d7c4c94c6e diff --git a/FS/FS/cust_location.pm b/FS/FS/cust_location.pm index fdc2cf8da..21bf92feb 100644 --- a/FS/FS/cust_location.pm +++ b/FS/FS/cust_location.pm @@ -14,6 +14,12 @@ use FS::cust_main_county; use FS::part_export; use FS::GeocodeCache; +# Essential fields. Can't be modified in place, will be considered in +# deciding if a location is "new", and (because of that) can't have +# leading/trailing whitespace. +my @essential = (qw(custnum address1 address2 city county state zip country + location_number location_type location_kind disabled)); + $import = 0; $DEBUG = 0; @@ -44,8 +50,9 @@ FS::cust_location - Object methods for cust_location records =head1 DESCRIPTION -An FS::cust_location object represents a customer location. FS::cust_location -inherits from FS::Record. The following fields are currently supported: +An FS::cust_location object represents a customer (or prospect) location. +FS::cust_location inherits from FS::Record. The following fields are currently +supported: =over 4 @@ -55,7 +62,15 @@ primary key =item custnum -custnum +Customer (see L). + +=item prospectnum + +Prospect (see L). + +=item locationname + +Optional location name. =item address1 @@ -89,6 +104,23 @@ Country (see L) Geocode +=item latitude + +=item longitude + +=item coord_auto + +Flag indicating whether coordinates were obtained automatically or manually +entered + +=item addr_clean + +Flag indicating whether address has been normalized + +=item censustract + +=item censusyear + =item district Tax district code (optional) @@ -148,9 +180,6 @@ sub find_or_insert { warn "find_or_insert:\n".Dumper($self) if $DEBUG; - my @essential = (qw(custnum address1 address2 city county state zip country - location_number location_type location_kind disabled)); - if ($conf->exists('cust_main-no_city_in_address')) { warn "Warning: passed city to find_or_insert when cust_main-no_city_in_address is configured, ignoring it" if $self->get('city'); @@ -236,8 +265,15 @@ sub insert { return $error; } - #false laziness with cust_main, will go away eventually - if ( !$import and $conf->config('tax_district_method') ) { + # If using tax_district_method, for rows in state of Washington, + # without a tax district already specified, queue a job to find + # the tax district + if ( + !$import + && !$self->district + && lc $self->state eq 'wa' + && $conf->config('tax_district_method') + ) { my $queue = new FS::queue { 'job' => 'FS::geocode_Mixin::process_district_update' @@ -351,9 +387,9 @@ sub check { return '' if $self->disabled; # so that disabling locations never fails - # maybe should just do all fields in the table? - # or in every table? - $self->trim_whitespace(qw(district city county state country)); + # whitespace in essential fields leads to problems figuring out if a + # record is "new"; get rid of it. + $self->trim_whitespace(@essential); my $error = $self->ut_numbern('locationnum') @@ -407,6 +443,26 @@ sub check { && $conf->exists('prospect_main-alt_address_format') && ! $self->location_kind; + # Do not allow bad tax district values in cust_location when + # using Washington State district sales tax calculation - would result + # in incorrect or missing sales tax on invoices. + my $tax_district_method = FS::Conf->new->config('tax_district_method'); + if ( + $tax_district_method + && $tax_district_method eq 'wa_sales' + && $self->district + ) { + my $cust_main_county = qsearchs( + cust_main_county => { district => $self->district } + ); + unless ( ref $cust_main_county ) { + return sprintf ( + 'WA State tax district %s does not exist in tax table', + $self->district + ); + } + } + unless ( $import or qsearch('cust_main_county', { 'country' => $self->country, 'state' => '', @@ -489,17 +545,51 @@ sub disable_if_unused { } -=item move_to +=item move_pkgs + +Returns array of cust_pkg objects that would have their location +updated by L (all packages that have this location as +their service address, and aren't canceled, and aren't supplemental +to another package, and aren't one-time charges that have already been charged.) + +=cut + +sub move_pkgs { + my $self = shift; + my @pkgs = (); + # find all packages that have the old location as their service address, + # and aren't canceled, + # and aren't supplemental to another package + # and aren't one-time charges that have already been charged + foreach my $cust_pkg ( + qsearch('cust_pkg', { + 'locationnum' => $self->locationnum, + 'cancel' => '', + 'main_pkgnum' => '', + }) + ) { + next if $cust_pkg->part_pkg->freq eq '0' + and ($cust_pkg->setup || 0) > 0; + push @pkgs, $cust_pkg; + } + return @pkgs; +} + +=item move_to NEW [ move_pkgs => \@move_pkgs ] Takes a new L object. Moves all packages that use the existing location to the new one, then sets the "disabled" flag on the old location. Returns nothing on success, an error message on error. +Use option I to override the list of packages to update +(see L.) + =cut sub move_to { my $old = shift; my $new = shift; + my %opt = @_; warn "move_to:\nFROM:".Dumper($old)."\nTO:".Dumper($new) if $DEBUG; @@ -531,19 +621,32 @@ sub move_to { return ''; } - # find all packages that have the old location as their service address, - # and aren't canceled, - # and aren't supplemental to another package. - my @pkgs = qsearch('cust_pkg', { - 'locationnum' => $old->locationnum, - 'cancel' => '', - 'main_pkgnum' => '', - }); - foreach my $cust_pkg (@pkgs) { - # don't move one-time charges that have already been charged - next if $cust_pkg->part_pkg->freq eq '0' - and ($cust_pkg->setup || 0) > 0; + my @pkgs; + if ($opt{'move_pkgs'}) { + @pkgs = @{$opt{'move_pkgs'}}; + my $pkgerr; + foreach my $pkg (@pkgs) { + my $pkgnum = $pkg->pkgnum; + $pkgerr = "cust_pkg $pkgnum has already been charged" + if $pkg->part_pkg->freq eq '0' + and ($pkg->setup || 0) > 0; + $pkgerr = "cust_pkg $pkgnum is supplemental" + if $pkg->main_pkgnum; + $pkgerr = "cust_pkg $pkgnum already cancelled" + if $pkg->cancel; + $pkgerr = "cust_pkg $pkgnum does not use this location" + unless $pkg->locationnum eq $old->locationnum; + last if $pkgerr; + } + if ($pkgerr) { + $dbh->rollback if $oldAutoCommit; + return "Cannot update package location: $pkgerr"; + } + } else { + @pkgs = $old->move_pkgs; + } + foreach my $cust_pkg (@pkgs) { $error = $cust_pkg->change( 'locationnum' => $new->locationnum, 'keep_dates' => 1 @@ -903,16 +1006,22 @@ sub _upgrade_data { # trim whitespace on records that need it local $allow_location_edit = 1; - foreach my $field (qw(city county state country district)) { + foreach my $field (@essential) { + next if $field eq 'custnum'; + next if $field eq 'disabled'; foreach my $location (qsearch({ table => 'cust_location', - extra_sql => " WHERE $field LIKE ' %' OR $field LIKE '% '" + extra_sql => " WHERE disabled IS NULL AND ($field LIKE ' %' OR $field LIKE '% ')" })) { my $error = $location->replace; die "$error (fixing whitespace in $field, locationnum ".$location->locationnum.')' if $error; - if ( $use_districts ) { + if ( + $use_districts + && !$location->district + && lc $location->state eq 'wa' + ) { my $queue = new FS::queue { 'job' => 'FS::geocode_Mixin::process_district_update' };