Download the PHP package remp/crm-payments-module without Composer
On this page you can find all versions of the php package remp/crm-payments-module. It is possible to download/install these versions without Composer. Possible dependencies are resolved automatically.
Download remp/crm-payments-module
More information about remp/crm-payments-module
Files in remp/crm-payments-module
Package crm-payments-module
Short Description CRM Payments Module
License MIT
Homepage https://remp2030.com
Informations about the package crm-payments-module
CRM Payments Module
Installing module
We recommend using Composer for installation and update management.
Enabling module
Add installed extension to your app/config/config.neon
file.
Add following commands to your scheduler (e.g. crontab) and change the path to match your deploy path:
Configuration
Data retention configuration
You can configure time before which application:cleanup
deletes old repository data and column which it uses by using (in your project configuration file):
Fast charge check configuration
You can configure fast charge threshold check by adding this to your configuration:
Fast charge check is done by RecurrentPaymentsChargeCommand::validateRecurrentPayment
and it prevents system
from repeated charging if error occurs while charging.
Scheduled commands
For payment module to work correctly, please add execution of following commands to your scheduler. Example displays crontab usage for execution (alter paths to your deploy paths):
Service commands
Module might provide service commands to be run in the deployed environment. Mostly to handle internal changes
and to prevent direct manipulation with the database. You can display required and optional arguments by using
--help
switch when running the command.
Payments module doesn't provide service commands.
Payment gateways
Module has a default set of supported payment gateways developed and used by us:
free
. Developed for development purposes, to be used for testing payment-related flows.bank_transfer
. Gateway generates unfinished payment and displays user bank account, amount and transaction identification so the payment can be paired and confirmed later.cardpay
(tatrabanka.sk). One-time card payment provided by Slovak bank.comfortpay
(tatrabanka.sk). Recurrent card payment provided by Slovak bank (CRM is handling charging)csob
(csob.cz). One-time card payment provided by Czech bank.csob_one_click
(csob.cz). Recurrent card payment provided by Czech bank (CRM ish handling charging)paypal
(paypal.com). One-time payment provided by major global provider.paypal_reference
(paypal.com). Recurrent payment provided by major global provider (CRM is handling charging)tatrapay
(tatrabanka.sk). One-time payment linked to Slovak bank's internet banking.
By default, only bank_transfer
as a default payment gateway is enabled by PaymentsModule. You can enable gateways you wish to use by adding following snippet to your app/config/config.neon
:
At this moment, there are several gateway implementations you can add to your CRM installation as a separate module:
stripe
(stripe.com). One-time payment provided by major global provider.stripe_recurrent
(stripe.com). Recurrent payment provided by major global provider (CRM is handling charging)slsp_sporopay
(slsp.sk). One-time payment linked to Slovak bank's internet banking.vub_eplatby
(vub.sk). One-time payment linked to Slovak bank's internet banking.
Standard (one-time) payments
Standard and initial recurrent payment have common beginning of process. Once the system generates instance of new payment, user can be redirected to payment gateway for processing. Each gateway requires different set of parameters to be provided, therefore gateway is responsible for generating the redirect URL with all required parameters.
As remp/crm-payments-module
is responsible only for actual payment processing, frontend flow can be managed by our
remp/crm-salesfunnel-module
which provides a way to create
sales funnels (payment windows), aggregates statistics and displays user success page after the payment with possibility
to extend it with widgets.
After the payment, user is directed back to the CRM. Each gateway provides its own URL where user is directed for payment completion processing.
If the payment is successful, payments module uses PaymentCompleteRedirectManager
to determine what kind of success page the user should see. If crm-salesfunnel-module
is used, user is directed
to the success page registered by the module.
The flow of payment processing can be described with following diagram:
Recurrent payments
Initiating payment
If the payment uses gateway that supports recurrent payment, the initial flow is usually the same as with the regular payments. The difference comes in during processing of successful initial payment.
PaymentsModule creates new instance of recurrent payment - a profile defining when the system should charge user again, and what subscription type will the user get when charged.
Each recurrent payment instance represent a single payment that will be charged in the future. That means, that if the charge fails, system creates new recurrent payment with charge date calculated based on retry rules and stores failing information to the original recurrent payment. Similarly, if the charge was successful, new subscription is created and new recurrent payment is defined to be charged in the next period. Thanks to that the system is able to provide information about each charge attempt for whole history of user charging including the bank approval/failure code.
This is all done on backend without system requiring any kind of user interaction. This block merely explains the flow and describes the terms so when displayed in CRM admin, the reader understands the displayed data.
Automatic charges
To charge the user, add payments:charge
command to your scheduler. Command doesn't handle concurrent runs - that means
that it's responsibility of your scheduler to prevent multiple overlapping instances of command running at the same time.
Otherwise a user could be charged twice during the same period.
We recommend using flock
or some other locking tool which will prevent command to be execute while the previous
instance is still running. Following is an example snippet for crontab to run the charging every 15 minutes:
Notifying expired cards
If gateway supports it, CRM fetches expiration date for each cid
(effectively credit card) used to execute recurring charges. Optionally, you can add command to your scheduler that automatically stops expired recurrent payments:
When recurrent payment is stopped, Crm\PaymentsModule\Events\RecurrentPaymentCardExpiredEvent
is emitted. By default, PaymentsModule
checks if there's another active recurring payment for user. If there isn't, it sends NotificationEvent
with card_expires_this_month
template code.
If you're not satisfied with the default implementation, you can remove the default handler by unregistering it in your module definition:
Implementing new gateway
You can implement and integrate new gateways to the CRM if necessary. Based on whether you're implementing standard
or recurrent gateway, you implementation should implement just Crm\PaymentsModule\Gateways\PaymentInterface
or the former and Crm\PaymentsModule\Gateways\RecurrentPaymentInterface
.
When implementing a gateway, we recommend extending Crm\PaymentsModule\Gateways\GatewayAbstract
to avoid implementing parts which are always similar and would cause code duplication.
Once you have your implementation ready, you need to seed it into the database from within seeder in your own module (see PaymentGatewaysSeeder as an example) and register it into the application's configuration:
Then, add seeder that will insert the gateway to database. See Crm\PaymentsModule\Seeders\PaymentGatewaysSeeder
as an example seeder implementation and register seeders section
of CRM skeleton documentation too see how the seeders should be registered in your module.
Adding PaymentCompletedRedirectResolver
If you want to change the success page displayed to user after the payment based on any arbitrary rule - for example
your gateway might want user to see some special offering or require him to enter some additional data - you can register
redirect resolver to process this request. When the payment is confirmed, redirect resolver will decide (based on
priority of registered resolvers) whether to redirect user to a special success page or whether a default success page
(if remp/crm-salesfunnel-module
is used) is sufficient.
The implementation of redirect resolver can look like this:
In the example the resolver first checked whether the redirection should happen for this particular payment - it should
happen if the payment was done via foo_gateway
. Then the redirectArgs
method returns array of arguments,
that will be 1:1 used in Nette's $this->redirect()
call. User will be redirected to renderSuccess
method
of SalesFunnelPresenter
that is implemented in our FooModule
and presents user our own success page.
When the implementation is ready, resolver needs to be registered in the app/config/config.neon
with the priority
specifying order of execution of resolvers - higher the number, higher the priority.
Bank Transfer landing page
When bank_transfer
is used, user isn't redirected to external payment gateway provider, but CRM displays payment information that user should use to complete the payment manually.
By default, this is handled by default resolver.
Create the resolver and register it with priority >10 in your config.neon
.
Bank email processing
Sometimes user doesn't finish the whole payment process and quits after the payment was made but before returning to the CRM for internal payment confirmation. That scenario is always unpleasant as user doesn't have money nor subscription.
To support this scenario, we've added possibility to read bank confirmation emails and try to confirm unfinished payments based on incoming emails.
The implementation is not universal yet and you'd need to create your own command for checking the mailbox. Please take a look at two implementations that we included within this package: confirmation commands for Tatra banka and for CSOB.
Mail downloaders
By default, our mail processing commands (e.g.: ImapMailDownloader. You can replace this downloader if needed by:
- creating your own mail downloader which must implement MailDownloaderInterface
- replacing default implementation with your own in your app config neon file:
Upgrades
Payments module provides a very basic way how to handle the upgrades. The upgrades are not currently configurable and work with predefined set of rules. There are 4 type of upgrades available:
- Shortening. If user's subscription doesn't end in the near time, system allows upgrade via shortening of actual subscription. The amount of days to shorten is based on number of remaining days of current subscription and the price of subscription type user's being upgraded to. Shortening is not trigerred if the shortened subscription would end sooner than in 14 days.
- Paid upgrade. If the shortening is not available, user is offered a paid upgrade. The amount to pay is calculated based
- Paid recurrent. Triggered when user's current subscription is recurrent and the subscription doesn't end within the following 5 days. Immediate charge amount is calculated based on upgraded subscription type price and the remaining number of days of current users's subscription. Next period charge is then made with the price of upgraded subscription.
- Free recurrent. If user has less than 5 days of subscription remaining, system allows the free upgrade to higher subscription. Next period charge is then made with the price of upgraded subscription.
Upgrade options for subscription types are configurable within subscription type detail in CRM admin. Upgrades don't check the content of each subscription type - therefore they cannot automatically determine that monthly web subscription can be upgraded to monthly web + print subscription. Each upgrade has to be configured manually and all upgrade options are always determined based on actual subscription, price of actual subscription type and price of upgraded subscription type.
Change of VAT rates
If the VAT rate changes in your country, there are two commands that help you update the system. Schedule them to be executed on the date of VAT change.
payments:change_vat --original-vat=X --target-vat=Y
- Command changes all existing subscription type items and each affected (
type=subscription_type
) payment item. You can test the command before the VAT change date by using--dry-run
and--verbose
options. - If you used other types of payments (e.g.
type=product
made byremp/crm-products-module
), read their READMEs or create your own command to handle the change based on the command from this module.
- Command changes all existing subscription type items and each affected (
application:change_config_value vat_default Y
- Command changes system default VAT to the new default VAT rate. Use this only when the standard VAT rate changes.
VAT modes
Before each payment is created, CRM internally assigns it one of the VAT modes - one of B2C, B2B or B2B Reverse-charge mode. Each payment is then processed according to the selected mode.
B2C mode is selected by default by payments module. If you want to set different mode for a payment, the options are:
- Include invoice module in your CRM configuration. The module provides mode selecting functionality out-of-the-box. It uses invoice address for deciding who is a valid B2B customer.
- Or implement
VatModeDataProviderInterface
data provider. It providesgetVatMode
function, which decides who is considered a valid B2B customer.
B2B
No VAT related changes are applied.
B2B Reverse-charge
When B2B reverse-charge mode is applied, VAT is not included in the payment. Practically, this is done by subtracting the VAT from the final sum (as if 0% VAT was set).
B2C
No VAT related changes are applied, unless One Stop Shop mode is enabled.
One Stop Shop
One Stop Shop (OSS) is an optional VAT feature, that can be turned on for payments made in B2C mode. For more details about its specifics, see https://vat-one-stop-shop.ec.europa.eu/index_en.
⚠ One Stop Shop works for EU countries only.
Prerequisites and setup
For turning on the OSS feature, you need:
- Populated
vat_rates
table with current VAT rates. See [Vatstack setup](Loading VAT rates using Vatstack service) for more details.
The feature can be turned on in CRM settings in /admin/config-admin/
in Payments section.
How it works
If enabled, each payment is assigned a payment country when the payment is created. Payment country is resolved according to multiple rules, sorted by priority:
- Country provided by
OneStopShopCountryResolutionDataProviderInterface
data provider.- For example,
InvoiceModule
resolves the payment country depending on invoice address country.
- For example,
- Payment address country.
- Explicitly selected payment country, for example by user in sales funnel.
- Derived from previous payment (e.g. for recurrent payments).
- Derived from user's IP address location.
After resolving, payment country is stored along with other payment data. Next, if payment is made outside the default country, OSS adjust VAT rates of the payment items.
VAT rate is adjusted accordingly:
- VAT rates for particular payment country are loaded from
vat_rates
DB table. - Table
vat_rates
contains only VAT rates of EU countries. - If no record is found, system considers that as the third country (outside of EU), 0% VAT rate is used.
- Otherwise, one of the loaded VAT rates (each country may have several VAT levels) is set.
- VAT rate level is selected either by
OneStopShopVatRateDataProviderInterface
data provider or a "standard" VAT rate is applied.- Data provider may select VAT rate level depending on arbitrary rules, for example, for "print" payment items, "reduced" VAT rate level may be applied.
European Union VAT
Loading VAT rates using Vatstack service
To load current VAT rates of EU member countries, we decided to use free VAT rate service https://vatstack.com/ which handles all issues with official EU VAT rates list for us.
After registration you'll find your API keys on page Developers->Api Keys. Add your public API key into your config.neon
:
Or you can provide API key with option --vatstack_api_key={VAT-STACK-PUBLIC-API-KEY}
when running command.
Usage
VAT rates are stored into vat_rates
table for each EU member country _(linked to countries
table with foreign key country_id
)_.
VAT rates do not change often, but we strongly recommend to add this command into your scheduler (eg. cron). Expired VAT rates are kept with column valid_to
set to date when system (this command) expired them.
Get current VAT rates for one country
To get current VAT rates for one country, call method VatRatesRepository->getByCountryAndDate()
.
API documentation
All examples use http://crm.press
as a base domain. Please change the host to the one you use
before executing the examples.
All examples use XXX
as a default value for authorization token, please replace it with the
real tokens:
- API tokens. Standard API keys for server-server communication. It identifies the calling application as a whole.
They can be generated in CRM Admin (
/api/api-tokens-admin/
) and each API key has to be whitelisted to access specific API endpoints. By default the API key has access to no endpoint. - User tokens. Generated for each user during the login process, token identify single user when communicating between
different parts of the system. The token can be read:
- From
n_token
cookie if the user was logged in via CRM. - From the response of
/api/v1/users/login
endpoint - you're free to store the response into your own cookie/local storage/session.
- From
API responses can contain following HTTP codes:
Value | Description |
---|---|
200 OK | Successful response, default value |
400 Bad Request | Invalid request (missing required parameters) |
403 Forbidden | The authorization failed (provided token was not valid) |
404 Not found | Referenced resource wasn't found |
If possible, the response includes application/json
encoded payload with message explaining
the error further.
POST /api/v1/payments/paypal-ipn
Handles the IPN notifications from PayPal. Follow this guide to enable IPN notifications for your PayPal account. Use http://crm.press/api/v1/payments/paypal-ipn
as "Notification URL" (replace http://crm.press
with your domain).
Note: when switching to "live" mode, don't forget to also change the paypal_ipn_baseurl
config to the live IPN endpoint (see here)
GET /api/v1/payments/variable-symbol
API call returns unique variable symbol (transaction ID) to be used for new payment instance.
Headers:
Name | Value | Required | Description |
---|---|---|---|
Authorization | Bearer String | yes | User token. Token must belong to user with admin flag. |
Example:
Response:
GET /api/v1/users/recurrent-payments
API call returns list of all user's recurrent payments.
Headers:
Name | Value | Required | Description |
---|---|---|---|
Authorization | Bearer String | yes | User token. |
Params:
Name | Value | Required | Description |
---|---|---|---|
states | String Array | no | Filter to return only payments with desired states. Available values: active , user_stop , admin_stop , pending , charged , charge_failed , system_stop . |
chargeable_from | String | no | ISO 8601 based date to filter only payments with charge_at parameter after the specified date. |
Example:
Response:
POST /api/v1/recurrent-payment/reactivate
API call to reactivate user's recurrent payment.
Conditions to successfully reactivate recurrent payment:
- RecurrentPayment has to be in
\Crm\PaymentsModule\Repository\RecurrentPaymentsRepository::STATE_USER_STOP
state. - Next charge of payment has to be in future (>= now).
Changes:
- State of recurrent payment is set to
\Crm\PaymentsModule\Repository\RecurrentPaymentsRepository::STATE_ACTIVE
.
Headers:
Name | Value | Required | Description |
---|---|---|---|
Authorization | Bearer String | yes | User token. |
Payload params:
Name | Value | Required | Description |
---|---|---|---|
id | Integer | yes | RecurrentPayment ID. |
Example:
Response:
On success HTTP status 200 OK is returned with recurrent payment's details.
In addition to API responses described at the beginning of API documentation section:
Value | Description |
---|---|
409 Conflict | Recurrent payment cannot be stopped by user (reason is in error message) |
POST /api/v1/recurrent-payment/stop
API call to stop user's recurrent payment.
Conditions to successfully stop recurrent payment:
- RecurrentPayment has to be in
\Crm\PaymentsModule\Repository\RecurrentPaymentsRepository::STATE_ACTIVE
state.
Changes:
- State of recurrent payment is set to
\Crm\PaymentsModule\Repository\RecurrentPaymentsRepository::STATE_USER_STOP
.
Headers:
Name | Value | Required | Description |
---|---|---|---|
Authorization | Bearer String | yes | User token. |
Payload params:
Name | Value | Required | Description |
---|---|---|---|
id | Integer | yes | RecurrentPayment ID. |
Example:
Response:
On success HTTP status 200 OK is returned with recurrent payment's details.
In addition to API responses described at the beginning of API documentation section:
Value | Description |
---|---|
409 Conflict | Recurrent payment cannot be stopped by user (reason is in error message) |
Components
ActualFreeSubscribersStatWidget
Simple admin dashboard widget showing free subscribers count.
ActualPaidSubscribersStatWidget
Simple admin dashboard widget showing paid subscribers count.
ChangePaymentStatus
Admin listing/detail change payment status modal component.
DeviceUserListingWidget
Admin user listing device component.
DonationPaymentItemListWidget
DupliciteRecurrentPayments
Admin listing of duplicit recurrent payments.
LastPayments
Admin listing of last payments in payment gateway detail.
MonthAmountStatWidget
Admin dashboard simple stat widget showing payments amount for last month.
MonthToDateAmountStatWidget
Admin dashboard simple stat widget showing payments amount for last month.
MyNextRecurrentPayment
ParsedMails
Payments admin widget showing payments with wrong amount.
PaymentItemsListWidget
Admin listing of payment items in payment detail.
SubscribersWithPaymentWidget
Admin dashboard single stat widget.
SubscriptionsWithActiveUnchargedRecurrentEndingWithinPeriodWidget
Admin listing widget.
SubscriptionsWithoutExtensionEndingWithinPeriodWidget
Admin dashboard stats widget.
SubscriptionTypeReports
Admin subscription type detail stats widget.
TodayAmountStatWidget
Admin dashboard simple single stat widget.
TotalAmountStatWidget
Admin dashboard simple single stat widget.
TotalUserPayments
Admin user detail stat widget.
UserPayments
Admin user detail listing widget.
Confirmation e-mails
Examples of confirmation mails received from banks. Every type of mail has it's own IMAP access configurable in application configuration.
TatraBanka Simple
From: b-mail (at) tatrabanka (dot) sk
Subject: e-commerce
Example e-mail file
TatraBanka
From: b-mail (at) tatrabanka (dot) sk
Subject: Kredit na ucte
Example e-mail file
TatraBanka Statement
From: vypis_obchodnik (at) tatrabanka (dot) sk
Example e-mail file
CSOB
From: notification (at) csob (dot) cz
Subject: CEB Info: Zaúčtování platby
Example e-mail file
Slovak CSOB
From: AdminTBS (at) csob (dot) sk
Subject: ČSOB Info 24 - Avízo
Example e-mail file
All versions of crm-payments-module with dependencies
ext-soap Version *
php-http/curl-client Version ^2.2.1
tomaj/bank-mails-parser Version ^2.7
tomaj/imap-mail-downloader Version ^1.2.0
rootpd/omnipay-csob Version ^4.4.0
rootpd/omnipay-paypal-reference Version ^4.0.0
singpolyma/openpgp-php Version ^0.6
tomaj/omnipay-tatra Version 4.3.0
geoip2/geoip2 Version ^v2.13.0