mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-15 07:33:04 +01:00
481 lines
14 KiB
PHP
481 lines
14 KiB
PHP
<?php
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace App\Models;
|
|
|
|
use App\Utils\Number;
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
|
|
|
/**
|
|
* App\Models\CompanyGateway
|
|
*
|
|
* @property int $id
|
|
* @property int $company_id
|
|
* @property int $user_id
|
|
* @property string $gateway_key
|
|
* @property bool $accepted_credit_cards
|
|
* @property bool $require_cvv
|
|
* @property bool $require_billing_address
|
|
* @property bool $require_shipping_address
|
|
* @property bool $update_details
|
|
* @property bool $is_deleted
|
|
* @property string $config
|
|
* @property mixed $fees_and_limits
|
|
* @property string|null $custom_value1
|
|
* @property string|null $custom_value2
|
|
* @property string|null $custom_value3
|
|
* @property string|null $custom_value4
|
|
* @property int|null $created_at
|
|
* @property int|null $updated_at
|
|
* @property int|null $deleted_at
|
|
* @property string $token_billing
|
|
* @property string|null $label
|
|
* @property bool $require_client_name
|
|
* @property bool $require_postal_code
|
|
* @property bool $require_client_phone
|
|
* @property bool $require_contact_name
|
|
* @property bool $require_contact_email
|
|
* @property bool $require_custom_value1
|
|
* @property bool $require_custom_value2
|
|
* @property bool $require_custom_value3
|
|
* @property bool $require_custom_value4
|
|
* @property-read int|null $client_gateway_tokens_count
|
|
* @property-read \App\Models\Company $company
|
|
* @property-read \App\Models\Gateway $gateway
|
|
* @property-read mixed $hashed_id
|
|
* @method getConfigField(string $field)
|
|
* @method static \Illuminate\Database\Eloquent\Builder|BaseModel company()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway filter(\App\Filters\QueryFilters $filters)
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway newModelQuery()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway newQuery()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway onlyTrashed()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway query()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|BaseModel scope()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway withTrashed()
|
|
* @method static \Illuminate\Database\Eloquent\Builder|CompanyGateway withoutTrashed()
|
|
* @property-read \Illuminate\Database\Eloquent\Collection<int, \App\Models\ClientGatewayToken> $client_gateway_tokens
|
|
* @method static CompanyGateway find($value)
|
|
* @mixin \Eloquent
|
|
*/
|
|
class CompanyGateway extends BaseModel
|
|
{
|
|
use SoftDeletes;
|
|
use Filterable;
|
|
|
|
public const GATEWAY_CREDIT = 10000000;
|
|
|
|
protected $casts = [
|
|
'fees_and_limits' => 'object',
|
|
'updated_at' => 'timestamp',
|
|
'created_at' => 'timestamp',
|
|
'deleted_at' => 'timestamp',
|
|
];
|
|
|
|
protected $with = [
|
|
'gateway',
|
|
];
|
|
|
|
protected $fillable = [
|
|
'gateway_key',
|
|
'accepted_credit_cards',
|
|
'require_cvv',
|
|
'require_billing_address',
|
|
'require_shipping_address',
|
|
'require_client_name',
|
|
'require_postal_code',
|
|
'require_client_phone',
|
|
'require_contact_name',
|
|
'require_contact_email',
|
|
'require_custom_value1',
|
|
'require_custom_value2',
|
|
'require_custom_value3',
|
|
'require_custom_value4',
|
|
'update_details',
|
|
'config',
|
|
'fees_and_limits',
|
|
'custom_value1',
|
|
'custom_value2',
|
|
'custom_value3',
|
|
'custom_value4',
|
|
'token_billing',
|
|
'label',
|
|
];
|
|
|
|
public static $credit_cards = [
|
|
1 => ['card' => 'images/credit_cards/Test-Visa-Icon.png', 'text' => 'Visa'],
|
|
2 => ['card' => 'images/credit_cards/Test-MasterCard-Icon.png', 'text' => 'Master Card'],
|
|
4 => ['card' => 'images/credit_cards/Test-AmericanExpress-Icon.png', 'text' => 'American Express'],
|
|
8 => ['card' => 'images/credit_cards/Test-Diners-Icon.png', 'text' => 'Diners'],
|
|
16 => ['card' => 'images/credit_cards/Test-Discover-Icon.png', 'text' => 'Discover'],
|
|
];
|
|
|
|
// const TYPE_PAYPAL = 300;
|
|
// const TYPE_STRIPE = 301;
|
|
// const TYPE_LEDGER = 302;
|
|
// const TYPE_FAILURE = 303;
|
|
// const TYPE_CHECKOUT = 304;
|
|
// const TYPE_AUTHORIZE = 305;
|
|
// const TYPE_CUSTOM = 306;
|
|
// const TYPE_BRAINTREE = 307;
|
|
// const TYPE_WEPAY = 309;
|
|
// const TYPE_PAYFAST = 310;
|
|
// const TYPE_PAYTRACE = 311;
|
|
// const TYPE_MOLLIE = 312;
|
|
// const TYPE_EWAY = 313;
|
|
// const TYPE_FORTE = 314;
|
|
// const PAYPAL_PPCP = 323;
|
|
|
|
public $gateway_consts = [
|
|
'38f2c48af60c7dd69e04248cbb24c36e' => 300,
|
|
'd14dd26a37cecc30fdd65700bfb55b23' => 301,
|
|
'd14dd26a47cecc30fdd65700bfb67b34' => 301,
|
|
'3758e7f7c6f4cecf0f4f348b9a00f456' => 304,
|
|
'3b6621f970ab18887c4f6dca78d3f8bb' => 305,
|
|
'54faab2ab6e3223dbe848b1686490baa' => 306,
|
|
'f7ec488676d310683fb51802d076d713' => 307,
|
|
'8fdeed552015b3c7b44ed6c8ebd9e992' => 309,
|
|
'd6814fc83f45d2935e7777071e629ef9' => 310,
|
|
'bbd736b3254b0aabed6ad7fda1298c88' => 311,
|
|
'1bd651fb213ca0c9d66ae3c336dc77e7' => 312,
|
|
'944c20175bbe6b9972c05bcfe294c2c7' => 313,
|
|
'kivcvjexxvdiyqtj3mju5d6yhpeht2xs' => 314,
|
|
'65faab2ab6e3223dbe848b1686490baz' => 320,
|
|
'b9886f9257f0c6ee7c302f1c74475f6c' => 321,
|
|
'hxd6gwg3ekb9tb3v9lptgx1mqyg69zu9' => 322,
|
|
'80af24a6a691230bbec33e930ab40666' => 323,
|
|
];
|
|
|
|
protected $touches = [];
|
|
|
|
public function getEntityType()
|
|
{
|
|
return self::class;
|
|
}
|
|
|
|
public function system_logs()
|
|
{
|
|
return $this->company
|
|
->system_log_relation
|
|
->where('type_id', $this->gateway_consts[$this->gateway->key])
|
|
->take(50);
|
|
}
|
|
|
|
public function company(): \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
{
|
|
return $this->belongsTo(Company::class);
|
|
}
|
|
|
|
public function client_gateway_tokens(): \Illuminate\Database\Eloquent\Relations\HasMany
|
|
{
|
|
return $this->hasMany(ClientGatewayToken::class);
|
|
}
|
|
|
|
public function gateway(): \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
{
|
|
return $this->belongsTo(Gateway::class, 'gateway_key', 'key');
|
|
}
|
|
|
|
public function getTypeAlias($gateway_type_id)
|
|
{
|
|
return GatewayType::getAlias($gateway_type_id);
|
|
}
|
|
|
|
/* This is the public entry point into the payment superclass */
|
|
public function driver(Client $client = null)
|
|
{
|
|
// $class = static::driver_class();
|
|
$class = self::driver_class();
|
|
|
|
if (!$class) {
|
|
return false;
|
|
}
|
|
|
|
return new $class($this, $client);
|
|
}
|
|
|
|
private function driver_class()
|
|
{
|
|
$class = 'App\\PaymentDrivers\\'.$this->gateway->provider.'PaymentDriver';
|
|
$class = str_replace('_', '', $class);
|
|
|
|
if (class_exists($class)) {
|
|
return $class;
|
|
}
|
|
|
|
return false;
|
|
|
|
// throw new \Exception("Payment Driver does not exist");
|
|
}
|
|
|
|
/**
|
|
* @param $config
|
|
*/
|
|
public function setConfig($config)
|
|
{
|
|
$this->config = encrypt(json_encode($config));
|
|
}
|
|
|
|
/**
|
|
* setConfigField
|
|
*
|
|
* @param mixed $field
|
|
* @param mixed $value
|
|
* @return void
|
|
*/
|
|
public function setConfigField($field, $value): void
|
|
{
|
|
$config = $this->getConfig();
|
|
$config->{$field} = $value;
|
|
|
|
$this->setConfig($config);
|
|
$this->save();
|
|
}
|
|
|
|
/**
|
|
* @return mixed
|
|
*/
|
|
public function getConfig()
|
|
{
|
|
//return decrypt($this->config);
|
|
return json_decode(decrypt($this->config));
|
|
}
|
|
|
|
public function getConfigTransformed()
|
|
{
|
|
return $this->config ? decrypt($this->config) : '';
|
|
}
|
|
|
|
/**
|
|
* @param $field
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public function getConfigField($field)
|
|
{
|
|
return object_get($this->getConfig(), $field, false);
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getAchEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_ach'));
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getApplePayEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_apple_pay'));
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getAlipayEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_alipay'));
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getSofortEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_sofort'));
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getSepaEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_sepa'));
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getBitcoinEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_bitcoin'));
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function getPayPalEnabled()
|
|
{
|
|
return ! empty($this->getConfigField('enable_pay_pal'));
|
|
}
|
|
|
|
// public function feesEnabled()
|
|
// {
|
|
// return floatval($this->fee_amount) || floatval($this->fee_percent);
|
|
// }
|
|
|
|
/**
|
|
* Returns the current test mode of the gateway.
|
|
*
|
|
* @return bool whether the gateway is in testmode or not.
|
|
*/
|
|
public function isTestMode() :bool
|
|
{
|
|
$config = $this->getConfig();
|
|
|
|
if ($this->gateway && $this->gateway->provider == 'Stripe' && property_exists($config, 'publishableKey') && strpos($config->publishableKey, 'test')) {
|
|
return true;
|
|
}
|
|
|
|
if ($config && property_exists($config, 'testMode') && $config->testMode) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Get Publishable Key
|
|
* Only works for STRIPE and PAYMILL.
|
|
* @return string The Publishable key
|
|
*/
|
|
public function getPublishableKey() :string
|
|
{
|
|
return $this->getConfigField('publishableKey');
|
|
}
|
|
|
|
public function getFeesAndLimits($gateway_type_id)
|
|
{
|
|
if (is_null($this->fees_and_limits) || empty($this->fees_and_limits) || !property_exists($this->fees_and_limits, $gateway_type_id)) {
|
|
return false;
|
|
}
|
|
|
|
if ($gateway_type_id == GatewayType::CUSTOM) {
|
|
$gateway_type_id = GatewayType::CREDIT_CARD;
|
|
}
|
|
|
|
return $this->fees_and_limits->{$gateway_type_id};
|
|
}
|
|
|
|
/**
|
|
* Returns the formatted fee amount for the gateway.
|
|
*
|
|
* @param float $amount The payment amount
|
|
* @param Client $client The client object
|
|
* @param int $gateway_type_id
|
|
* @return string The fee amount formatted in the client currency
|
|
*/
|
|
public function calcGatewayFeeLabel($amount, Client $client, $gateway_type_id = GatewayType::CREDIT_CARD) :string
|
|
{
|
|
$label = ' ';
|
|
|
|
$fee = $this->calcGatewayFee($amount, $gateway_type_id);
|
|
|
|
if ($fee > 0) {
|
|
$fees_and_limits = $this->fees_and_limits->{$gateway_type_id};
|
|
|
|
if (strlen($fees_and_limits->fee_percent) >=1) {
|
|
$label .= $fees_and_limits->fee_percent . '%';
|
|
}
|
|
|
|
if (strlen($fees_and_limits->fee_amount) >=1 && $fees_and_limits->fee_amount > 0) {
|
|
if (strlen($label) > 1) {
|
|
$label .= ' + ' . Number::formatMoney($fees_and_limits->fee_amount, $client);
|
|
} else {
|
|
$label .= Number::formatMoney($fees_and_limits->fee_amount, $client);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
return $label;
|
|
}
|
|
|
|
public function calcGatewayFee($amount, $gateway_type_id, $include_taxes = false)
|
|
{
|
|
$fees_and_limits = $this->getFeesAndLimits($gateway_type_id);
|
|
|
|
if (! $fees_and_limits) {
|
|
return false;
|
|
}
|
|
|
|
$fee = 0;
|
|
|
|
|
|
if ($fees_and_limits->adjust_fee_percent ?? false) {
|
|
$adjusted_fee = 0;
|
|
|
|
if ($fees_and_limits->fee_amount) {
|
|
$adjusted_fee += $fees_and_limits->fee_amount + $amount;
|
|
} else {
|
|
$adjusted_fee = $amount;
|
|
}
|
|
|
|
if ($fees_and_limits->fee_percent) {
|
|
$divisor = 1 - ($fees_and_limits->fee_percent/100);
|
|
|
|
$gross_amount = round($adjusted_fee/$divisor, 2);
|
|
$fee = $gross_amount - $amount;
|
|
}
|
|
} else {
|
|
if ($fees_and_limits->fee_amount) {
|
|
$fee += $fees_and_limits->fee_amount;
|
|
}
|
|
|
|
if ($fees_and_limits->fee_percent) {
|
|
if ($fees_and_limits->fee_percent == 100) { //unusual edge case if the user wishes to charge a fee of 100% 09/01/2022
|
|
$fee += $amount;
|
|
} else {
|
|
$fee += round(($amount * $fees_and_limits->fee_percent / 100), 2);
|
|
}
|
|
//elseif ($fees_and_limits->adjust_fee_percent) {
|
|
// $fee += round(($amount / (1 - $fees_and_limits->fee_percent / 100) - $amount), 2);
|
|
//} else {
|
|
|
|
//}
|
|
}
|
|
}
|
|
/* Cap fee if we have to here. */
|
|
if ($fees_and_limits->fee_cap > 0 && ($fee > $fees_and_limits->fee_cap)) {
|
|
$fee = $fees_and_limits->fee_cap;
|
|
}
|
|
|
|
$pre_tax_fee = $fee;
|
|
|
|
/**/
|
|
if ($include_taxes) {
|
|
if ($fees_and_limits->fee_tax_rate1) {
|
|
$fee += round(($pre_tax_fee * $fees_and_limits->fee_tax_rate1 / 100), 2);
|
|
// info("fee after adding fee tax 1 = {$fee}");
|
|
}
|
|
|
|
if ($fees_and_limits->fee_tax_rate2) {
|
|
$fee += round(($pre_tax_fee * $fees_and_limits->fee_tax_rate2 / 100), 2);
|
|
// info("fee after adding fee tax 2 = {$fee}");
|
|
}
|
|
|
|
if ($fees_and_limits->fee_tax_rate3) {
|
|
$fee += round(($pre_tax_fee * $fees_and_limits->fee_tax_rate3 / 100), 2);
|
|
// info("fee after adding fee tax 3 = {$fee}");
|
|
}
|
|
}
|
|
|
|
return $fee;
|
|
}
|
|
|
|
public function webhookUrl()
|
|
{
|
|
return route('payment_webhook', ['company_key' => $this->company->company_key, 'company_gateway_id' => $this->hashed_id]);
|
|
}
|
|
}
|