333 lines
9.9 KiB
PHP
333 lines
9.9 KiB
PHP
|
<?php
|
||
|
|
||
|
/**
|
||
|
* This file is the file which all subscription gateways should call
|
||
|
* when a payment has been received - it sorts out the user status.
|
||
|
*
|
||
|
* Simple Machines Forum (SMF)
|
||
|
*
|
||
|
* @package SMF
|
||
|
* @author Simple Machines https://www.simplemachines.org
|
||
|
* @copyright 2022 Simple Machines and individual contributors
|
||
|
* @license https://www.simplemachines.org/about/smf/license.php BSD
|
||
|
*
|
||
|
* @version 2.1.2
|
||
|
*/
|
||
|
|
||
|
// Set this to true to always log $_POST info received from payment gateways.
|
||
|
$paid_debug = false;
|
||
|
|
||
|
// Start things rolling by getting SMF alive...
|
||
|
$ssi_guest_access = true;
|
||
|
if (!file_exists(dirname(__FILE__) . '/SSI.php'))
|
||
|
die('Cannot find SSI.php');
|
||
|
|
||
|
require_once(dirname(__FILE__) . '/SSI.php');
|
||
|
require_once($sourcedir . '/ManagePaid.php');
|
||
|
|
||
|
// For any admin emailing.
|
||
|
require_once($sourcedir . '/Subs-Admin.php');
|
||
|
|
||
|
// Ensure we don't trip over disabled internal functions
|
||
|
if (version_compare(PHP_VERSION, '8.0.0', '>='))
|
||
|
require_once($sourcedir . '/Subs-Compat.php');
|
||
|
|
||
|
loadLanguage('ManagePaid');
|
||
|
|
||
|
// If there's literally nothing coming in, let's take flight!
|
||
|
if (empty($_POST))
|
||
|
{
|
||
|
header('content-type: text/html; charset=' . (empty($modSettings['global_character_set']) ? (empty($txt['lang_character_set']) ? 'ISO-8859-1' : $txt['lang_character_set']) : $modSettings['global_character_set']));
|
||
|
die($txt['paid_no_data']);
|
||
|
}
|
||
|
|
||
|
// I assume we're even active?
|
||
|
if (empty($modSettings['paid_enabled']))
|
||
|
exit;
|
||
|
|
||
|
// If we have some custom people who find out about problems load them here.
|
||
|
$notify_users = array();
|
||
|
if (!empty($modSettings['paid_email_to']))
|
||
|
{
|
||
|
foreach (explode(',', $modSettings['paid_email_to']) as $email)
|
||
|
$notify_users[] = array(
|
||
|
'email' => $email,
|
||
|
'name' => $txt['who_member'],
|
||
|
'id' => 0,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// We need to see whether we can find the correct payment gateway,
|
||
|
// we'll going to go through all our gateway scripts and find out
|
||
|
// if they are happy with what we have.
|
||
|
$txnType = '';
|
||
|
$gatewayHandles = loadPaymentGateways();
|
||
|
foreach ($gatewayHandles as $gateway)
|
||
|
{
|
||
|
$gatewayClass = new $gateway['payment_class']();
|
||
|
if ($gatewayClass->isValid())
|
||
|
{
|
||
|
$txnType = $gateway['code'];
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (empty($txnType))
|
||
|
generateSubscriptionError($txt['paid_unknown_transaction_type']);
|
||
|
|
||
|
// Get the subscription and member ID, amongst others...
|
||
|
@list($subscription_id, $member_id) = $gatewayClass->precheck();
|
||
|
|
||
|
// Integer these just in case.
|
||
|
$subscription_id = (int) $subscription_id;
|
||
|
$member_id = (int) $member_id;
|
||
|
|
||
|
// This would be bad...
|
||
|
if (empty($member_id))
|
||
|
generateSubscriptionError($txt['paid_empty_member']);
|
||
|
|
||
|
// Verify the member.
|
||
|
$request = $smcFunc['db_query']('', '
|
||
|
SELECT id_member, member_name, real_name, email_address
|
||
|
FROM {db_prefix}members
|
||
|
WHERE id_member = {int:current_member}',
|
||
|
array(
|
||
|
'current_member' => $member_id,
|
||
|
)
|
||
|
);
|
||
|
// Didn't find them?
|
||
|
if ($smcFunc['db_num_rows']($request) === 0)
|
||
|
generateSubscriptionError(sprintf($txt['paid_could_not_find_member'], $member_id));
|
||
|
$member_info = $smcFunc['db_fetch_assoc']($request);
|
||
|
$smcFunc['db_free_result']($request);
|
||
|
|
||
|
// Get the subscription details.
|
||
|
$request = $smcFunc['db_query']('', '
|
||
|
SELECT cost, length, name
|
||
|
FROM {db_prefix}subscriptions
|
||
|
WHERE id_subscribe = {int:current_subscription}',
|
||
|
array(
|
||
|
'current_subscription' => $subscription_id,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
// Didn't find it?
|
||
|
if ($smcFunc['db_num_rows']($request) === 0)
|
||
|
generateSubscriptionError(sprintf($txt['paid_count_not_find_subscription'], $member_id, $subscription_id));
|
||
|
|
||
|
$subscription_info = $smcFunc['db_fetch_assoc']($request);
|
||
|
$smcFunc['db_free_result']($request);
|
||
|
|
||
|
// We wish to check the pending payments to make sure we are expecting this.
|
||
|
$request = $smcFunc['db_query']('', '
|
||
|
SELECT id_sublog, payments_pending, pending_details, end_time
|
||
|
FROM {db_prefix}log_subscribed
|
||
|
WHERE id_subscribe = {int:current_subscription}
|
||
|
AND id_member = {int:current_member}
|
||
|
LIMIT 1',
|
||
|
array(
|
||
|
'current_subscription' => $subscription_id,
|
||
|
'current_member' => $member_id,
|
||
|
)
|
||
|
);
|
||
|
if ($smcFunc['db_num_rows']($request) === 0)
|
||
|
generateSubscriptionError(sprintf($txt['paid_count_not_find_subscription_log'], $member_id, $subscription_id));
|
||
|
$subscription_info += $smcFunc['db_fetch_assoc']($request);
|
||
|
$smcFunc['db_free_result']($request);
|
||
|
|
||
|
// Is this a refund etc?
|
||
|
if ($gatewayClass->isRefund())
|
||
|
{
|
||
|
// If the end time subtracted by current time, is not greater
|
||
|
// than the duration (ie length of subscription), then we close it.
|
||
|
if ($subscription_info['end_time'] - time() < $subscription_info['length'])
|
||
|
{
|
||
|
// Delete user subscription.
|
||
|
removeSubscription($subscription_id, $member_id);
|
||
|
$subscription_act = time();
|
||
|
$status = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
loadSubscriptions();
|
||
|
$subscription_act = $subscription_info['end_time'] - $context['subscriptions'][$subscription_id]['num_length'];
|
||
|
$status = 1;
|
||
|
}
|
||
|
|
||
|
// Mark it as complete so we have a record.
|
||
|
$smcFunc['db_query']('', '
|
||
|
UPDATE {db_prefix}log_subscribed
|
||
|
SET end_time = {int:current_time}
|
||
|
WHERE id_subscribe = {int:current_subscription}
|
||
|
AND id_member = {int:current_member}
|
||
|
AND status = {int:status}',
|
||
|
array(
|
||
|
'current_time' => $subscription_act,
|
||
|
'current_subscription' => $subscription_id,
|
||
|
'current_member' => $member_id,
|
||
|
'status' => $status,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
// Receipt?
|
||
|
if (!empty($modSettings['paid_email']) && $modSettings['paid_email'] == 2)
|
||
|
{
|
||
|
$replacements = array(
|
||
|
'NAME' => $subscription_info['name'],
|
||
|
'REFUNDNAME' => $member_info['member_name'],
|
||
|
'REFUNDUSER' => $member_info['real_name'],
|
||
|
'PROFILELINK' => $scripturl . '?action=profile;u=' . $member_id,
|
||
|
'DATE' => timeformat(time(), false),
|
||
|
);
|
||
|
|
||
|
emailAdmins('paid_subscription_refund', $replacements, $notify_users);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
// Otherwise is it what we want, a purchase?
|
||
|
elseif ($gatewayClass->isPayment() || $gatewayClass->isSubscription())
|
||
|
{
|
||
|
$cost = $smcFunc['json_decode']($subscription_info['cost'], true);
|
||
|
$total_cost = $gatewayClass->getCost();
|
||
|
$notify = false;
|
||
|
|
||
|
// For one off's we want to only capture them once!
|
||
|
if (!$gatewayClass->isSubscription())
|
||
|
{
|
||
|
$real_details = $smcFunc['json_decode']($subscription_info['pending_details'], true);
|
||
|
if (empty($real_details))
|
||
|
generateSubscriptionError(sprintf($txt['paid_count_not_find_outstanding_payment'], $member_id, $subscription_id));
|
||
|
|
||
|
// Now we just try to find anything pending.
|
||
|
// We don't really care which it is as security happens later.
|
||
|
foreach ($real_details as $id => $detail)
|
||
|
{
|
||
|
unset($real_details[$id]);
|
||
|
if ($detail[3] == 'payback' && $subscription_info['payments_pending'])
|
||
|
$subscription_info['payments_pending']--;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
$subscription_info['pending_details'] = empty($real_details) ? '' : $smcFunc['json_encode']($real_details);
|
||
|
|
||
|
$smcFunc['db_query']('', '
|
||
|
UPDATE {db_prefix}log_subscribed
|
||
|
SET payments_pending = {int:payments_pending}, pending_details = {string:pending_details}
|
||
|
WHERE id_sublog = {int:current_subscription_item}',
|
||
|
array(
|
||
|
'payments_pending' => $subscription_info['payments_pending'],
|
||
|
'current_subscription_item' => $subscription_info['id_sublog'],
|
||
|
'pending_details' => $subscription_info['pending_details'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// Is this flexible?
|
||
|
if ($subscription_info['length'] == 'F')
|
||
|
{
|
||
|
$found_duration = 0;
|
||
|
|
||
|
// This is a little harder, can we find the right duration?
|
||
|
foreach ($cost as $duration => $value)
|
||
|
{
|
||
|
if ($duration == 'fixed')
|
||
|
continue;
|
||
|
elseif ((float) $value == (float) $total_cost)
|
||
|
$found_duration = strtoupper(substr($duration, 0, 1));
|
||
|
}
|
||
|
|
||
|
// If we have the duration then we're done.
|
||
|
if ($found_duration !== 0)
|
||
|
{
|
||
|
$notify = true;
|
||
|
addSubscription($subscription_id, $member_id, $found_duration);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
$actual_cost = $cost['fixed'];
|
||
|
|
||
|
// It must be at least the right amount.
|
||
|
if ($total_cost != 0 && $total_cost >= $actual_cost)
|
||
|
{
|
||
|
// Add the subscription.
|
||
|
$notify = true;
|
||
|
addSubscription($subscription_id, $member_id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Send a receipt?
|
||
|
if (!empty($modSettings['paid_email']) && $modSettings['paid_email'] == 2 && $notify)
|
||
|
{
|
||
|
$replacements = array(
|
||
|
'NAME' => $subscription_info['name'],
|
||
|
'SUBNAME' => $member_info['member_name'],
|
||
|
'SUBUSER' => $member_info['real_name'],
|
||
|
'SUBEMAIL' => $member_info['email_address'],
|
||
|
'PRICE' => sprintf($modSettings['paid_currency_symbol'], $total_cost),
|
||
|
'PROFILELINK' => $scripturl . '?action=profile;u=' . $member_id,
|
||
|
'DATE' => timeformat(time(), false),
|
||
|
);
|
||
|
|
||
|
emailAdmins('paid_subscription_new', $replacements, $notify_users);
|
||
|
}
|
||
|
}
|
||
|
// Maybe they're cancelling. Some subscriptions may require actively doing something, but PayPal doesn't, for example.
|
||
|
elseif ($gatewayClass->isCancellation())
|
||
|
{
|
||
|
if (method_exists($gatewayClass, 'performCancel'))
|
||
|
$gatewayClass->performCancel($subscription_id, $member_id, $subscription_info);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Some other "valid" transaction such as:
|
||
|
//
|
||
|
// subscr_signup: This IPN response (txn_type) is sent only the first time the user signs up for a subscription.
|
||
|
// It then does not fire in any event later. This response is received somewhere before or after the first payment of
|
||
|
// subscription is received (txn_type=subscr_payment) which is what we do process
|
||
|
//
|
||
|
// Should we log any of these ...
|
||
|
}
|
||
|
|
||
|
// In case we have anything specific to do.
|
||
|
$gatewayClass->close();
|
||
|
|
||
|
// Hidden setting to log the IPN info for debugging purposes.
|
||
|
if ($paid_debug === true)
|
||
|
generateSubscriptionError($txt['subscription'], true);
|
||
|
|
||
|
/**
|
||
|
* Log an error then exit
|
||
|
*
|
||
|
* @param string $text The error to log
|
||
|
* @param bool $debug If true, won't send an email if $modSettings['paid_email'] isn't set
|
||
|
* @return void
|
||
|
*/
|
||
|
function generateSubscriptionError($text, $debug = false)
|
||
|
{
|
||
|
global $modSettings, $notify_users, $smcFunc;
|
||
|
|
||
|
// Send an email?
|
||
|
if (!empty($modSettings['paid_email']) && !$debug)
|
||
|
{
|
||
|
$replacements = array(
|
||
|
'ERROR' => $text,
|
||
|
);
|
||
|
|
||
|
emailAdmins('paid_subscription_error', $replacements, $notify_users);
|
||
|
}
|
||
|
|
||
|
// Maybe we can try to give them the post data?
|
||
|
if (!empty($_POST))
|
||
|
{
|
||
|
foreach ($_POST as $key => $val)
|
||
|
$text .= '<br>' . $smcFunc['htmlspecialchars']($key) . ': ' . $smcFunc['htmlspecialchars']($val);
|
||
|
}
|
||
|
|
||
|
// Then just log and die.
|
||
|
log_error($text, 'paidsubs');
|
||
|
|
||
|
exit;
|
||
|
}
|
||
|
|
||
|
?>
|