mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2025-07-09 02:04:30 -04:00
Added gateway fee tests
This commit is contained in:
parent
64a92bbc2e
commit
91d1a4ca4e
@ -82,6 +82,7 @@ script:
|
||||
- php ./vendor/codeception/codeception/codecept run --debug acceptance OnlinePaymentCest.php
|
||||
- php ./vendor/codeception/codeception/codecept run --debug acceptance PaymentCest.php
|
||||
- php ./vendor/codeception/codeception/codecept run --debug acceptance TaskCest.php
|
||||
- php ./vendor/codeception/codeception/codecept run --debug acceptance GatewayFeesCest.php
|
||||
- php ./vendor/codeception/codeception/codecept run --debug acceptance AllPagesCept.php
|
||||
|
||||
#- sed -i 's/NINJA_DEV=true/NINJA_PROD=true/g' .env
|
||||
|
@ -29,18 +29,6 @@ trait ChargesFees
|
||||
|
||||
if ($settings->fee_percent) {
|
||||
$amount = $this->partial > 0 ? $this->partial : $this->balance;
|
||||
|
||||
// prevent charging taxes twice on the surcharge
|
||||
if ($location != FEE_LOCATION_ITEM) {
|
||||
if ($this->$taxField) {
|
||||
$taxAmount = 0;
|
||||
foreach ($this->getTaxes() as $key => $tax) {
|
||||
$taxAmount += $tax['amount'] - $tax['paid'];
|
||||
}
|
||||
$amount -= $taxAmount;
|
||||
}
|
||||
}
|
||||
|
||||
$fee += $amount * $settings->fee_percent / 100;
|
||||
}
|
||||
|
||||
|
@ -75,7 +75,7 @@ Optional Settings
|
||||
|
||||
The following are optional query parameter settings:
|
||||
|
||||
- ``serializer``: Either array (the default) or json. If json is selected the data is returned using the `JSON API <http://jsonapi.org/>`_ format.
|
||||
- ``serializer``: Either array (the default) or `JSON <http://jsonapi.org/>`_.
|
||||
- ``include``: A comma-separated list of nested relationships to include.
|
||||
- ``client_id``: If set the results will be filtered by the client.
|
||||
- ``page``: The page number of results to return when the results are paginated.
|
||||
|
@ -33,7 +33,7 @@
|
||||
->help('gateway_fees_help')
|
||||
->label('gateway_fees')!!}
|
||||
<br/>
|
||||
{!! Former::actions( Button::success(trans('texts.save'))->submit()->appendIcon(Icon::create('floppy-disk')) ) !!}
|
||||
{!! Former::actions( Button::success(trans('texts.save'))->withAttributes(['id' => 'formSave'])->submit()->appendIcon(Icon::create('floppy-disk')) ) !!}
|
||||
</div>
|
||||
</div>
|
||||
{!! Former::close() !!}
|
||||
@ -146,14 +146,14 @@
|
||||
->onchange('onTaxRateChange(1)')
|
||||
->addOption('', '')
|
||||
->label(trans('texts.tax_rate'))
|
||||
->fromQuery($taxRates, function($model) { return $model->name . ' ' . $model->rate . '%'; }, 'public_id') !!}
|
||||
->fromQuery($taxRates, function($model) { return $model->name . ': ' . $model->rate . '%'; }, 'public_id') !!}
|
||||
|
||||
@if ($account->enable_second_tax_rate)
|
||||
{!! Former::select('tax_rate2')
|
||||
->onchange('onTaxRateChange(2)')
|
||||
->addOption('', '')
|
||||
->label(trans('texts.tax_rate'))
|
||||
->fromQuery($taxRates, function($model) { return $model->name . ' ' . $model->rate . '%'; }, 'public_id') !!}
|
||||
->fromQuery($taxRates, function($model) { return $model->name . ': ' . $model->rate . '%'; }, 'public_id') !!}
|
||||
@endif
|
||||
|
||||
@endif
|
||||
@ -187,7 +187,7 @@
|
||||
<div class="modal-footer" style="margin-top: 0px">
|
||||
<button type="button" class="btn btn-default"
|
||||
data-dismiss="modal">{{ trans('texts.cancel') }}</button>
|
||||
<button type="submit" class="btn btn-primary">{{ trans('texts.save') }}</button>
|
||||
<button type="submit" class="btn btn-primary" id="modalSave">{{ trans('texts.save') }}</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
@ -37,7 +37,7 @@
|
||||
{!! Former::select('default_tax_rate_id')
|
||||
->addOption('', '')
|
||||
->label(trans('texts.tax_rate'))
|
||||
->fromQuery($taxRates, function($model) { return $model->name . ' ' . $model->rate . '%'; }, 'id') !!}
|
||||
->fromQuery($taxRates, function($model) { return $model->name . ': ' . $model->rate . '%'; }, 'id') !!}
|
||||
@endif
|
||||
|
||||
</div>
|
||||
|
@ -54,4 +54,94 @@ class AcceptanceTester extends \Codeception\Actor
|
||||
$I->click("$dropdownSelector span.dropdown-toggle");
|
||||
$I->click("$dropdownSelector ul li:nth-child($option)");
|
||||
}
|
||||
|
||||
function createGateway(\AcceptanceTester $I)
|
||||
{
|
||||
if ( ! $I->grabFromDatabase('account_gateways', 'id', ['id' => 1])) {
|
||||
$I->wantTo('create a gateway');
|
||||
$I->amOnPage('/gateways/create?other_providers=true');
|
||||
$I->fillField(['name' =>'23_apiKey'], env('stripe_secret_key') ?: Fixtures::get('stripe_secret_key'));
|
||||
$I->fillField(['name' =>'publishable_key'], '');
|
||||
$I->click('Save');
|
||||
$I->see('Successfully created gateway');
|
||||
}
|
||||
}
|
||||
|
||||
function createClient(\AcceptanceTester $I, $email, $name = '')
|
||||
{
|
||||
$I->amOnPage('/clients/create');
|
||||
$I->fillField(['name' => 'name'], $name);
|
||||
$I->fillField(['name' => 'contacts[0][email]'], $email);
|
||||
$I->click('Save');
|
||||
$I->see($email);
|
||||
}
|
||||
|
||||
function createProduct(\AcceptanceTester $I, $productKey, $cost, $taxName = '', $taxRate = '')
|
||||
{
|
||||
$I->amOnPage('/products/create');
|
||||
$I->fillField(['name' => 'product_key'], $productKey);
|
||||
$I->fillField(['name' => 'cost'], $cost);
|
||||
|
||||
if ($taxName && $taxRate) {
|
||||
$taxOption = $taxName . ': ' . number_format($taxRate, 3) . '%';
|
||||
$I->selectOption('#default_tax_rate_id', $taxOption);
|
||||
}
|
||||
|
||||
$I->click('Save');
|
||||
$I->wait(1);
|
||||
//$I->see($productKey);
|
||||
}
|
||||
|
||||
function createTaxRate(\AcceptanceTester $I, $name, $rate)
|
||||
{
|
||||
$I->amOnPage('/tax_rates/create');
|
||||
$I->fillField(['name' => 'name'], $name);
|
||||
$I->fillField(['name' => 'rate'], $rate);
|
||||
$I->click('Save');
|
||||
$I->see($name);
|
||||
$I->see($rate);
|
||||
}
|
||||
|
||||
function fillInvoice(\AcceptanceTester $I, $clientEmail, $productKey)
|
||||
{
|
||||
$I->amOnPage('/invoices/create');
|
||||
$I->selectDropdown($I, $clientEmail, '.client_select .dropdown-toggle');
|
||||
$I->fillField('table.invoice-table tbody tr:nth-child(1) #product_key', $productKey);
|
||||
$I->click('table.invoice-table tbody tr:nth-child(1) .tt-selectable');
|
||||
}
|
||||
|
||||
function createOnlinePayment(\AcceptanceTester $I, $invoiceNumber)
|
||||
{
|
||||
$invoiceId = $I->grabFromDatabase('invoices', 'id', ['invoice_number' => $invoiceNumber]);
|
||||
$invitationKey = $I->grabFromDatabase('invitations', 'invitation_key', ['invoice_id' => $invoiceId]);
|
||||
|
||||
$clientSession = $I->haveFriend('client');
|
||||
$clientSession->does(function(AcceptanceTester $I) use ($invitationKey) {
|
||||
$I->amOnPage('/view/' . $invitationKey);
|
||||
$I->click('Pay Now');
|
||||
$I->click('Credit Card');
|
||||
$I->fillField('#card_number', '4242424242424242');
|
||||
$I->fillField('#cvv', '100');
|
||||
$I->selectOption('#expiration_month', 12);
|
||||
$I->selectOption('#expiration_year', date('Y'));
|
||||
$I->click('.btn-success');
|
||||
$I->wait(3);
|
||||
$I->see('Successfully applied payment');
|
||||
});
|
||||
}
|
||||
|
||||
function checkSettingOption(\AcceptanceTester $I, $url, $option)
|
||||
{
|
||||
$I->amOnPage('/settings/' . $url);
|
||||
$I->checkOption('#' . $option);
|
||||
$I->click('Save');
|
||||
}
|
||||
|
||||
function uncheckSettingOption(\AcceptanceTester $I, $url, $option)
|
||||
{
|
||||
$I->amOnPage('/settings/' . $url);
|
||||
$I->uncheckOption('#' . $option);
|
||||
$I->click('Save');
|
||||
}
|
||||
|
||||
}
|
||||
|
191
tests/acceptance/GatewayFeesCest.php
Normal file
191
tests/acceptance/GatewayFeesCest.php
Normal file
@ -0,0 +1,191 @@
|
||||
<?php
|
||||
|
||||
use Faker\Factory;
|
||||
|
||||
class GatewayFeesCest
|
||||
{
|
||||
/**
|
||||
* @var \Faker\Generator
|
||||
*/
|
||||
private $faker;
|
||||
|
||||
public function _before(AcceptanceTester $I)
|
||||
{
|
||||
$I->checkIfLogin($I);
|
||||
|
||||
$this->faker = Factory::create();
|
||||
}
|
||||
|
||||
public function checkLineItemFee(AcceptanceTester $I)
|
||||
{
|
||||
$clientName = $this->faker->word();
|
||||
$clientEmail = $this->faker->safeEmail;
|
||||
$productKey = $this->faker->word();
|
||||
$taxName = $this->faker->word();
|
||||
|
||||
//$cost = $this->faker->randomFloat(2, 50, 100);
|
||||
//$feeAmount = $this->faker->randomFloat(2, 1, 5);
|
||||
//$taxRate = $this->faker->randomFloat(3, 5, 15);
|
||||
//$feePercent = $this->faker->randomFloat(3, 1, 5);
|
||||
|
||||
$cost = 100;
|
||||
$feeAmount = 10;
|
||||
$taxRate = 10;
|
||||
$feePercent = 10;
|
||||
|
||||
$total = $cost + ($cost * $taxRate / 100);
|
||||
$fee = $feeAmount + ($total * $feeAmount / 100);
|
||||
$feeWithTax = $fee + ($fee * $taxRate / 100);
|
||||
$partialFee = $feeAmount + ($total / 2 * $feeAmount / 100);
|
||||
$partialFeeWithTax = $partialFee + ($partialFee * $taxRate / 100);
|
||||
|
||||
// surcharge gateway fee
|
||||
$this->configureSurchargeTaxRates($I, $taxName, $taxRate);
|
||||
$this->configureFees($I, $feeAmount, $feePercent);
|
||||
|
||||
$I->createProduct($I, $productKey, $cost);
|
||||
$I->createClient($I, $clientEmail, $clientName);
|
||||
|
||||
// without taxing the fee
|
||||
$I->uncheckSettingOption($I, 'invoice_settings#invoice_surcharges', 'custom_invoice_taxes1');
|
||||
$this->createInvoice($I, $clientName, $productKey, $total, $fee);
|
||||
|
||||
// with taxing the fee
|
||||
$I->checkSettingOption($I, 'invoice_settings#invoice_surcharges', 'custom_invoice_taxes1');
|
||||
$this->createInvoice($I, $clientName, $productKey, $total, $feeWithTax);
|
||||
|
||||
|
||||
// line item gateway fee
|
||||
$this->configureLineItemTaxRates($I, $taxName, $taxRate);
|
||||
$productKey = $this->faker->word();
|
||||
$I->createProduct($I, $productKey, $cost, $taxName, $taxRate);
|
||||
|
||||
// without taxing the fee
|
||||
$this->configureGatewayFeeTax($I);
|
||||
$this->createInvoice($I, $clientName, $productKey, $total, $fee);
|
||||
|
||||
// with taxing the fee
|
||||
$this->configureGatewayFeeTax($I, $taxName, $taxRate);
|
||||
$this->createInvoice($I, $clientName, $productKey, $total, $feeWithTax);
|
||||
|
||||
|
||||
// partial invoice
|
||||
$invoiceNumber = $this->createInvoice($I, $clientName, $productKey, $total, $partialFeeWithTax, $total / 2);
|
||||
$this->createPayment($I, $invoiceNumber, $total + $partialFeeWithTax, 0, $partialFeeWithTax);
|
||||
}
|
||||
|
||||
private function configureGatewayFeeTax($I, $taxName = '', $taxRate = '')
|
||||
{
|
||||
if ($taxName && $taxRate) {
|
||||
$taxOption = $taxName . ': ' . number_format($taxRate, 3) . '%';
|
||||
} else {
|
||||
$taxOption = '';
|
||||
}
|
||||
|
||||
$I->amOnPage('/settings/online_payments#fees');
|
||||
$I->executeJS('javascript:showLimitsModal(\'Credit Card\', 1)');
|
||||
$I->click('Fees');
|
||||
$I->selectOption('tax_rate1', $taxOption);
|
||||
$I->click('#modalSave');
|
||||
|
||||
$I->executeJS('javascript:showLimitsModal(\'Bank Transfer\', 2)');
|
||||
$I->click('Fees');
|
||||
$I->selectOption('tax_rate1', $taxOption);
|
||||
$I->click('#modalSave');
|
||||
}
|
||||
|
||||
private function configureSurchargeTaxRates($I, $taxName, $taxRate)
|
||||
{
|
||||
$taxOption = $taxName . ': ' . number_format($taxRate, 3) . '%';
|
||||
|
||||
$I->createTaxRate($I, $taxName, $taxRate);
|
||||
$I->amOnPage('/settings/tax_rates');
|
||||
$I->checkOption('#invoice_item_taxes');
|
||||
$I->selectOption('default_tax_rate_id', $taxOption);
|
||||
$I->click('Save');
|
||||
}
|
||||
|
||||
private function configureLineItemTaxRates($I, $taxName, $taxRate)
|
||||
{
|
||||
$taxOption = $taxName . ': ' . number_format($taxRate, 3) . '%';
|
||||
|
||||
// set the gateway fee to use line items
|
||||
$I->amOnPage('/settings/online_payments#fees');
|
||||
$I->selectOption('gateway_fee_location', 'invoice_item');
|
||||
$I->click('#formSave');
|
||||
|
||||
// disable the default invoice tax
|
||||
$I->amOnPage('/settings/tax_rates');
|
||||
$I->selectOption('#default_tax_rate_id', '');
|
||||
$I->click('Save');
|
||||
}
|
||||
|
||||
private function configureFees($I, $feeAmount, $feePercent)
|
||||
{
|
||||
$I->createGateway($I);
|
||||
|
||||
// enable gateway fees
|
||||
$I->amOnPage('/settings/online_payments#fees');
|
||||
$I->selectOption('gateway_fee_location', 'custom_value1');
|
||||
$I->click('#formSave');
|
||||
|
||||
$I->executeJS('javascript:showLimitsModal(\'Credit Card\', 1)');
|
||||
$I->click('Fees');
|
||||
$I->fillField(['name' => 'fee_amount'], $feeAmount);
|
||||
$I->fillField(['name' => 'fee_percent'], $feePercent);
|
||||
$I->click('#modalSave');
|
||||
|
||||
$I->executeJS('javascript:showLimitsModal(\'Bank Transfer\', 2)');
|
||||
$I->click('Fees');
|
||||
$I->fillField(['name' => 'fee_amount'], $feeAmount * 2);
|
||||
$I->fillField(['name' => 'fee_percent'], $feePercent * 2);
|
||||
$I->click('#modalSave');
|
||||
}
|
||||
|
||||
private function createInvoice($I, $clientEmail, $productKey, $amount, $fee, $partial = false)
|
||||
{
|
||||
$I->fillInvoice($I, $clientEmail, $productKey);
|
||||
$invoiceNumber = $I->grabAttributeFrom('#invoice_number', 'value');
|
||||
|
||||
if ($partial) {
|
||||
$amount = ($partial * 2);
|
||||
$I->fillField('#partial', $partial);
|
||||
}
|
||||
|
||||
$I->click('Mark Sent');
|
||||
$I->see($clientEmail);
|
||||
|
||||
$balance = $partial ? ($amount - $partial) : 0;
|
||||
$this->createPayment($I, $invoiceNumber, $amount, $balance, $fee);
|
||||
|
||||
return $invoiceNumber;
|
||||
}
|
||||
|
||||
private function createPayment($I, $invoiceNumber, $amount, $balance, $fee)
|
||||
{
|
||||
$invoiceId = $I->grabFromDatabase('invoices', 'id', ['invoice_number' => $invoiceNumber]);
|
||||
$invitationKey = $I->grabFromDatabase('invitations', 'invitation_key', ['invoice_id' => $invoiceId]);
|
||||
|
||||
// check we correctly remove/add back the gateway fee
|
||||
$I->amOnPage('/view/' . $invitationKey);
|
||||
$I->click('Pay Now');
|
||||
$I->see('$' . number_format($fee, 2) . ' Fee');
|
||||
$I->see('$' . number_format($fee * 2, 2) . ' Fee');
|
||||
|
||||
$I->amOnPage('/payment/' . $invitationKey . '/credit_card');
|
||||
$I->amOnPage('/payment/' . $invitationKey . '/bank_transfer');
|
||||
$I->amOnPage('/payment/' . $invitationKey . '/credit_card');
|
||||
|
||||
$I->amOnPage('/view/' . $invitationKey);
|
||||
$I->click('Pay Now');
|
||||
$I->see('$' . number_format($fee, 2) . ' Fee');
|
||||
$I->see('$' . number_format($fee * 2, 2) . ' Fee');
|
||||
|
||||
$I->createOnlinePayment($I, $invoiceNumber);
|
||||
$I->seeInDatabase('invoices', [
|
||||
'invoice_number' => $invoiceNumber,
|
||||
'amount' => ($amount + $fee),
|
||||
'balance' => $balance
|
||||
]);
|
||||
}
|
||||
}
|
@ -21,23 +21,8 @@ class OnlinePaymentCest
|
||||
$clientEmail = $this->faker->safeEmail;
|
||||
$productKey = $this->faker->text(10);
|
||||
|
||||
// set gateway info
|
||||
if ( ! $I->grabFromDatabase('account_gateways', 'id', ['id' => 1])) {
|
||||
$I->wantTo('create a gateway');
|
||||
$I->amOnPage('/gateways/create?other_providers=true');
|
||||
|
||||
$I->fillField(['name' =>'23_apiKey'], env('stripe_secret_key') ?: Fixtures::get('stripe_secret_key'));
|
||||
// Fails to load StripeJS causing "ReferenceError: Can't find variable: Stripe"
|
||||
//$I->fillField(['name' =>'stripe_publishable_key'], env('stripe_secret_key') ?: Fixtures::get('stripe_publishable_key'));
|
||||
$I->click('Save');
|
||||
$I->see('Successfully created gateway');
|
||||
}
|
||||
|
||||
// create client
|
||||
$I->amOnPage('/clients/create');
|
||||
$I->fillField(['name' => 'contacts[0][email]'], $clientEmail);
|
||||
$I->click('Save');
|
||||
$I->see($clientEmail);
|
||||
$I->createGateway($I);
|
||||
$I->createClient($I, $clientEmail);
|
||||
|
||||
// create product
|
||||
$I->amOnPage('/products/create');
|
||||
@ -58,6 +43,10 @@ class OnlinePaymentCest
|
||||
|
||||
// enter payment
|
||||
$clientId = $I->grabFromDatabase('contacts', 'client_id', ['email' => $clientEmail]);
|
||||
$invoiceNumber = $I->grabFromDatabase('invoices', 'invoice_number', ['client_id' => $clientId]);
|
||||
$I->createOnlinePayment($I, $invoiceNumber);
|
||||
|
||||
/*
|
||||
$invoiceId = $I->grabFromDatabase('invoices', 'id', ['client_id' => $clientId]);
|
||||
$invitationKey = $I->grabFromDatabase('invitations', 'invitation_key', ['invoice_id' => $invoiceId]);
|
||||
|
||||
@ -66,18 +55,6 @@ class OnlinePaymentCest
|
||||
$I->amOnPage('/view/' . $invitationKey);
|
||||
$I->click('Pay Now');
|
||||
$I->click('Credit Card');
|
||||
|
||||
/*
|
||||
$I->fillField(['name' => 'first_name'], $this->faker->firstName);
|
||||
$I->fillField(['name' => 'last_name'], $this->faker->lastName);
|
||||
$I->fillField(['name' => 'address1'], $this->faker->streetAddress);
|
||||
$I->fillField(['name' => 'address2'], $this->faker->streetAddress);
|
||||
$I->fillField(['name' => 'city'], $this->faker->city);
|
||||
$I->fillField(['name' => 'state'], $this->faker->state);
|
||||
$I->fillField(['name' => 'postal_code'], $this->faker->postcode);
|
||||
$I->selectDropdown($I, 'United States', '.country-select .dropdown-toggle');
|
||||
*/
|
||||
|
||||
$I->fillField('#card_number', '4242424242424242');
|
||||
$I->fillField('#cvv', '100');
|
||||
$I->selectOption('#expiration_month', 12);
|
||||
@ -86,12 +63,11 @@ class OnlinePaymentCest
|
||||
$I->wait(3);
|
||||
$I->see('Successfully applied payment');
|
||||
});
|
||||
|
||||
$I->wait(1);
|
||||
*/
|
||||
|
||||
// create recurring invoice and auto-bill
|
||||
$I->amOnPage('/recurring_invoices/create');
|
||||
//$I->selectDropdown($I, $clientEmail, '.client_select .dropdown-toggle');
|
||||
$I->selectDropdown($I, 'Test Test', '.client_select .dropdown-toggle');
|
||||
$I->fillField('table.invoice-table tbody tr:nth-child(1) #product_key', $productKey);
|
||||
$I->click('table.invoice-table tbody tr:nth-child(1) .tt-selectable');
|
||||
@ -99,6 +75,5 @@ class OnlinePaymentCest
|
||||
$I->executeJS('onConfirmEmailClick()');
|
||||
$I->wait(4);
|
||||
$I->see("$0.00");
|
||||
|
||||
}
|
||||
}
|
||||
|
@ -34,17 +34,8 @@ class TaxRatesCest
|
||||
$invoiceTaxRate = number_format($invoiceTaxRate, 3);
|
||||
|
||||
// create tax rates
|
||||
$I->amOnPage('/tax_rates/create');
|
||||
$I->fillField(['name' => 'name'], $itemTaxName);
|
||||
$I->fillField(['name' => 'rate'], $itemTaxRate);
|
||||
$I->click('Save');
|
||||
$I->see($itemTaxName);
|
||||
|
||||
$I->amOnPage('/tax_rates/create');
|
||||
$I->fillField(['name' => 'name'], $invoiceTaxName);
|
||||
$I->fillField(['name' => 'rate'], $invoiceTaxRate);
|
||||
$I->click('Save');
|
||||
$I->see($invoiceTaxName);
|
||||
$I->createTaxRate($I, $itemTaxName, $itemTaxRate);
|
||||
$I->createTaxRate($I, $invoiceTaxName, $invoiceTaxRate);
|
||||
|
||||
// enable line item taxes
|
||||
$I->amOnPage('/settings/tax_rates');
|
||||
|
Loading…
x
Reference in New Issue
Block a user