336 lines
14 KiB
PHP
336 lines
14 KiB
PHP
|
<?php
|
||
|
/*********************************************************************************
|
||
|
*
|
||
|
* TimeTrex is a Workforce Management program developed by
|
||
|
* TimeTrex Software Inc. Copyright (C) 2003 - 2021 TimeTrex Software Inc.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it under
|
||
|
* the terms of the GNU Affero General Public License version 3 as published by
|
||
|
* the Free Software Foundation with the addition of the following permission
|
||
|
* added to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED
|
||
|
* WORK IN WHICH THE COPYRIGHT IS OWNED BY TIMETREX, TIMETREX DISCLAIMS THE
|
||
|
* WARRANTY OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
||
|
* FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
||
|
* details.
|
||
|
*
|
||
|
*
|
||
|
* You should have received a copy of the GNU Affero General Public License along
|
||
|
* with this program; if not, see http://www.gnu.org/licenses or write to the Free
|
||
|
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||
|
* 02110-1301 USA.
|
||
|
*
|
||
|
*
|
||
|
* You can contact TimeTrex headquarters at Unit 22 - 2475 Dobbin Rd. Suite
|
||
|
* #292 West Kelowna, BC V4T 2E9, Canada or at email address info@timetrex.com.
|
||
|
*
|
||
|
*
|
||
|
* The interactive user interfaces in modified source and object code versions
|
||
|
* of this program must display Appropriate Legal Notices, as required under
|
||
|
* Section 5 of the GNU Affero General Public License version 3.
|
||
|
*
|
||
|
*
|
||
|
* In accordance with Section 7(b) of the GNU Affero General Public License
|
||
|
* version 3, these Appropriate Legal Notices must retain the display of the
|
||
|
* "Powered by TimeTrex" logo. If the display of the logo is not reasonably
|
||
|
* feasible for technical reasons, the Appropriate Legal Notices must display
|
||
|
* the words "Powered by TimeTrex".
|
||
|
*
|
||
|
********************************************************************************/
|
||
|
|
||
|
require_once( dirname( __FILE__ ) . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'includes' . DIRECTORY_SEPARATOR . 'global.inc.php' );
|
||
|
require_once( dirname( __FILE__ ) . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'includes' . DIRECTORY_SEPARATOR . 'CLI.inc.php' );
|
||
|
require_once( dirname( __FILE__ ) . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'classes/payroll_deduction/PayrollDeduction.class.php' );
|
||
|
|
||
|
if ( $argc < 2 || in_array( $argv[1], [ '--help', '-help', '-h', '-?' ] ) ) {
|
||
|
$help_output = "Usage: generate_payroll_deduction_test_csv.php [country_code] [date]\n";
|
||
|
echo $help_output;
|
||
|
} else {
|
||
|
$country = strtoupper( $argv[1] );
|
||
|
$effective_date = strtotime( $argv[2] );
|
||
|
|
||
|
$cf = new CompanyFactory();
|
||
|
$province_arr = $cf->getOptions( 'province' );
|
||
|
|
||
|
$province_arr['US']['00'] = 'NONE';//Make an option for Federal only.
|
||
|
|
||
|
if ( !isset( $province_arr[$country] ) ) {
|
||
|
echo "Country does not have any province/states.\n";
|
||
|
}
|
||
|
ksort( $province_arr[$country] );
|
||
|
|
||
|
|
||
|
$pay_periods = 26;
|
||
|
$static_test_data = [
|
||
|
'CA' => [
|
||
|
'income' => [
|
||
|
192, //5000/year
|
||
|
384, //10000/year
|
||
|
961, //25000/year
|
||
|
1923, //50000,
|
||
|
3846, //100000,
|
||
|
6223, //180000, //Should be in the middle of the BPAF brackets.
|
||
|
9615, //250000
|
||
|
],
|
||
|
'federal_claim' => [ 0, 100 ], //Use lowest non-zero value.
|
||
|
'provincial_claim' => [ 0, 100 ], //Use lowest non-zero value.
|
||
|
],
|
||
|
'US' => [
|
||
|
'income' => [
|
||
|
192, //5000/year
|
||
|
384, //10000/year
|
||
|
961, //25000/year
|
||
|
1923, //50000,
|
||
|
3846, //100000,
|
||
|
9615, //250000
|
||
|
76923, //2000000 (For states with higher tax brackets)
|
||
|
],
|
||
|
|
||
|
//2020 Federal W2 variables.
|
||
|
'filing_status' => [ 10, 20, 40 ], //Federal filing statuses.
|
||
|
'federal_form_w4_version' => [ 2019, 2020 ],
|
||
|
'federal_claim_dependents' => [ 0, 2500, 5000 ],
|
||
|
'federal_other_income' => [ 0, 10000 ],
|
||
|
'federal_other_deductions' => [ 0, 1000 ],
|
||
|
|
||
|
'allowance' => [ 0, 1, 2, 3, 5 ],
|
||
|
],
|
||
|
];
|
||
|
|
||
|
$test_data = [];
|
||
|
|
||
|
if ( $country != '' && isset( $province_arr[$country] ) && $effective_date != '' ) {
|
||
|
foreach ( $province_arr[$country] as $province_code => $province ) {
|
||
|
//echo "Province: $province_code\n";
|
||
|
$raw_result = [];
|
||
|
|
||
|
$pd_obj = new PayrollDeduction( $country, $province_code );
|
||
|
|
||
|
//echo 'Tax Bracket Rows: '. $result->RecordCount() ."\n";
|
||
|
if ( $country == 'US' ) { //US
|
||
|
if ( isset( $pd_obj->obj->state_income_tax_rate_options ) ) {
|
||
|
$raw_result = $pd_obj->obj->getDataFromRateArray( $effective_date, $pd_obj->obj->state_income_tax_rate_options );
|
||
|
}
|
||
|
|
||
|
$result = [];
|
||
|
foreach ( $raw_result as $raw_filing_status => $row_a ) {
|
||
|
foreach ( $row_a as $row ) {
|
||
|
$row['status'] = ( $raw_filing_status == 0 ) ? 10 : $raw_filing_status;
|
||
|
$result[] = $row;
|
||
|
}
|
||
|
}
|
||
|
unset( $raw_result, $raw_filing_status, $row_a, $row );
|
||
|
|
||
|
if ( count( $result ) == 0 ) {
|
||
|
//Use static test rates.
|
||
|
|
||
|
$test_data[$country][$province_code] = $static_test_data[$country];
|
||
|
if ( $province_code != '00' ) {
|
||
|
$test_data[$country][$province_code]['filing_status'] = [ 10 ]; //No tax brackets, only use a single filing status.
|
||
|
}
|
||
|
} else {
|
||
|
//Always include the same income brackets for testing, AS WELL as one to test each individual bracket.
|
||
|
$test_data[$country][$province_code] = $static_test_data[$country];
|
||
|
|
||
|
$i = 1;
|
||
|
$prev_income = null;
|
||
|
$prev_status = null;
|
||
|
$prev_province = null;
|
||
|
foreach ( $result as $tax_row ) {
|
||
|
//Test $100 less then the first bracket, and $100 more then all other brackets for each status.
|
||
|
$income = round( ( $tax_row['income'] / $pay_periods ) );
|
||
|
$variance = round( ( 100 / $pay_periods ) );
|
||
|
|
||
|
if ( $prev_income == null || $prev_income > $income ) {
|
||
|
//echo "First bracket! $country $province ".$tax_row['income']." T: ". ($tax_row['income']-$variance) ."\n";
|
||
|
$test_data[$country][$province_code]['income'][] = ( $income - $variance );
|
||
|
$test_data[$country][$province_code]['filing_status'][] = $tax_row['status'];
|
||
|
}
|
||
|
|
||
|
$test_data[$country][$province_code]['income'][] = ( $income + $variance );
|
||
|
$test_data[$country][$province_code]['filing_status'][] = $tax_row['status'];
|
||
|
$test_data[$country][$province_code]['allowance'] = $static_test_data[$country]['allowance'];
|
||
|
|
||
|
$test_data[$country][$province_code]['income'] = array_unique( $test_data[$country][$province_code]['income'] );
|
||
|
$test_data[$country][$province_code]['filing_status'] = array_unique( $test_data[$country][$province_code]['filing_status'] );
|
||
|
|
||
|
$prev_income = $income;
|
||
|
$prev_status = $tax_row['status'];
|
||
|
$prev_province = $province_code;
|
||
|
$i++;
|
||
|
unset( $income );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $province_code != '00' ) {
|
||
|
$test_data[$country][$province_code]['federal_form_w4_version'] = [ 2020 ];
|
||
|
$test_data[$country][$province_code]['federal_claim_dependents'] = [ 0 ];
|
||
|
$test_data[$country][$province_code]['federal_other_income'] = [ 0 ];
|
||
|
$test_data[$country][$province_code]['federal_other_deductions'] = [ 0 ];
|
||
|
}
|
||
|
|
||
|
foreach ( $test_data[$country][$province_code]['filing_status'] as $filing_status ) {
|
||
|
foreach ( $test_data[$country][$province_code]['allowance'] as $allowance ) {
|
||
|
foreach ( $test_data[$country][$province_code]['federal_form_w4_version'] as $federal_form_w4_version ) {
|
||
|
foreach ( $test_data[$country][$province_code]['federal_claim_dependents'] as $federal_claim_dependents ) {
|
||
|
foreach ( $test_data[$country][$province_code]['federal_other_income'] as $federal_other_income ) {
|
||
|
foreach ( $test_data[$country][$province_code]['federal_other_deductions'] as $federal_other_deductions ) {
|
||
|
foreach ( $test_data[$country][$province_code]['income'] as $income ) {
|
||
|
$pd_obj = new PayrollDeduction( $country, ( ( $province_code == '00' ) ? 'AK' : $province_code ) ); //Valid state is needed to calculate something, even for just federal numbers.
|
||
|
$pd_obj->setDate( $effective_date );
|
||
|
$pd_obj->setAnnualPayPeriods( $pay_periods );
|
||
|
|
||
|
//Federal
|
||
|
$pd_obj->setFederalFormW4Version( $federal_form_w4_version );
|
||
|
$pd_obj->setFederalFilingStatus( $filing_status );
|
||
|
$pd_obj->setFederalAllowance( $allowance );
|
||
|
$pd_obj->setFederalMultipleJobs( false ); //2020 or newer W4 settings.
|
||
|
$pd_obj->setFederalClaimDependents( $federal_claim_dependents );
|
||
|
$pd_obj->setFederalOtherIncome( $federal_other_income );
|
||
|
$pd_obj->setFederalDeductions( $federal_other_deductions );
|
||
|
$pd_obj->setFederalAdditionalDeduction( 0 );
|
||
|
$pd_obj->setProvincialTaxExempt( false );
|
||
|
|
||
|
//State
|
||
|
$pd_obj->setStateFilingStatus( $filing_status );
|
||
|
$pd_obj->setStateAllowance( $allowance );
|
||
|
$pd_obj->setFederalTaxExempt( false );
|
||
|
|
||
|
switch ( $province_code ) {
|
||
|
case 'GA':
|
||
|
$pd_obj->setUserValue3( $allowance );
|
||
|
break;
|
||
|
case 'IN':
|
||
|
case 'IL':
|
||
|
case 'VA':
|
||
|
$pd_obj->setUserValue1( $allowance );
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
$pd_obj->setGrossPayPeriodIncome( $income );
|
||
|
|
||
|
//echo 'State: '. $province_code .' Income: '. $income .' Claim Dependents: '. $federal_claim_dependents .' Other Income: '. $federal_other_income ."\n";
|
||
|
//flush();
|
||
|
//ob_flush();
|
||
|
|
||
|
$retarr[] = [
|
||
|
'country' => $country,
|
||
|
'province' => $province_code,
|
||
|
'date' => date( 'm/d/y', $effective_date ),
|
||
|
'pay_periods' => $pay_periods,
|
||
|
'filing_status' => $filing_status,
|
||
|
'allowance' => $allowance,
|
||
|
'federal_form_w4_version' => $federal_form_w4_version,
|
||
|
'federal_claim_dependents' => $federal_claim_dependents,
|
||
|
'federal_other_income' => $federal_other_income,
|
||
|
'federal_other_deductions' => $federal_other_deductions,
|
||
|
'gross_income' => $income,
|
||
|
'federal_deduction' => Misc::MoneyRound( $pd_obj->getFederalPayPeriodDeductions() ),
|
||
|
'provincial_deduction' => Misc::MoneyRound( $pd_obj->getStatePayPeriodDeductions() ),
|
||
|
];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} else if ( $country == 'CA' ) { //Canada
|
||
|
$result = [];
|
||
|
if ( isset( $pd_obj->obj->provincial_income_tax_rate_options ) ) {
|
||
|
$result = $pd_obj->obj->getDataFromRateArray( $effective_date, $pd_obj->obj->provincial_income_tax_rate_options );
|
||
|
}
|
||
|
|
||
|
if ( count( $result ) == 0 ) {
|
||
|
//Use static test rates.
|
||
|
$test_data[$country][$province_code] = $static_test_data[$country];
|
||
|
} else {
|
||
|
$test_data[$country][$province_code] = $static_test_data[$country];
|
||
|
|
||
|
$i = 1;
|
||
|
$prev_income = null;
|
||
|
$prev_status = null;
|
||
|
$prev_province = null;
|
||
|
foreach ( $result as $tax_row ) {
|
||
|
if ( $tax_row['income'] == 0 ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//Test $100 less then the first bracket, and $100 more then all other brackets for each status.
|
||
|
$income = round( $tax_row['income'] / $pay_periods );
|
||
|
$variance = round( 100 / $pay_periods );
|
||
|
|
||
|
if ( $prev_income == null || $prev_income > $income ) {
|
||
|
//echo "First bracket! $country $province ".$tax_row['income']." T: ". ($tax_row['income']-$variance) ."\n";
|
||
|
$test_data[$country][$province_code]['income'][] = $income - $variance;
|
||
|
}
|
||
|
|
||
|
$test_data[$country][$province_code]['income'][] = $income + $variance;
|
||
|
$test_data[$country][$province_code]['federal_claim'] = $static_test_data[$country]['federal_claim'];
|
||
|
$test_data[$country][$province_code]['provincial_claim'] = $static_test_data[$country]['provincial_claim'];
|
||
|
|
||
|
$test_data[$country][$province_code]['income'] = array_unique( $test_data[$country][$province_code]['income'] );
|
||
|
|
||
|
$prev_income = $income;
|
||
|
$prev_status = ( isset( $tax_row['status'] ) ) ? $tax_row['status'] : null;
|
||
|
$prev_province = $province_code;
|
||
|
$i++;
|
||
|
unset( $income );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ( $test_data[$country][$province_code]['provincial_claim'] as $provincial_claim ) {
|
||
|
foreach ( $test_data[$country][$province_code]['federal_claim'] as $federal_claim ) {
|
||
|
foreach ( $test_data[$country][$province_code]['income'] as $income ) {
|
||
|
$pd_obj = new PayrollDeduction( $country, $province_code );
|
||
|
$pd_obj->setDate( $effective_date );
|
||
|
$pd_obj->setAnnualPayPeriods( $pay_periods );
|
||
|
$pd_obj->setEnableCPPAndEIDeduction( true ); //Deduct CPP/EI.
|
||
|
|
||
|
$pd_obj->setFederalTotalClaimAmount( $federal_claim );
|
||
|
$pd_obj->setProvincialTotalClaimAmount( $provincial_claim );
|
||
|
|
||
|
$pd_obj->setEIExempt( false );
|
||
|
$pd_obj->setCPPExempt( false );
|
||
|
|
||
|
$pd_obj->setFederalTaxExempt( false );
|
||
|
$pd_obj->setProvincialTaxExempt( false );
|
||
|
|
||
|
$pd_obj->setYearToDateCPPContribution( 0 );
|
||
|
$pd_obj->setYearToDateEIContribution( 0 );
|
||
|
|
||
|
$pd_obj->setGrossPayPeriodIncome( $income );
|
||
|
|
||
|
$retarr[] = [
|
||
|
'country' => $country,
|
||
|
'province' => $province_code,
|
||
|
'date' => date( 'm/d/y', $effective_date ),
|
||
|
'pay_periods' => $pay_periods,
|
||
|
'federal_claim' => $pd_obj->getFederalTotalClaimAmount(),
|
||
|
'provincial_claim' => $pd_obj->getProvincialTotalClaimAmount(),
|
||
|
'gross_income' => $income,
|
||
|
'federal_deduction' => Misc::MoneyRound( $pd_obj->getFederalPayPeriodDeductions() ),
|
||
|
'provincial_deduction' => Misc::MoneyRound( $pd_obj->getProvincialPayPeriodDeductions() ),
|
||
|
];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//generate column array.
|
||
|
$column_keys = array_keys( $retarr[0] );
|
||
|
foreach ( $column_keys as $column_key ) {
|
||
|
$columns[$column_key] = $column_key;
|
||
|
}
|
||
|
|
||
|
//var_dump($test_data);
|
||
|
//var_dump($retarr);
|
||
|
echo Misc::Array2CSV( $retarr, $columns, false, $include_header = true );
|
||
|
}
|
||
|
}
|
||
|
//Debug::Display();
|
||
|
?>
|