How to implement Payment SDK
Algorithms
There are five algorithms you must create
initialize_payment
- Create an algorithm called initialize_payment with a parameter called params
data:image/s3,"s3://crabby-images/f3232/f323259c2e7baac29de4e2274100232a8fd8cf35" alt="ss1.png 1920"
- Write your code in the code block
data:image/s3,"s3://crabby-images/aa142/aa142f3e8c01ad84b6e5809b10ca4b35542bffbb" alt="ss2.png 1519"
There are three main components you need to make the algorithm work.
- Settings object
- Payment object
- Response method
Tenant.notify(message: "Initialize Example payment", type: :info)
Tenant.notify(message: "#{params}", type: :info)
# params will contain fields
# - payment = Payment object
# get settings from the Settings datatype
setting = get_setting("EXAMPLE_SETTINGS")
# get Payment object for transaction
payment = params["payment"]
# print to the notification log
Tenant.notify(message: "Payment #{payment.to_hash}", type: :info)
# Example of how to use a template as the response
{
"action" => "render",
"template" => "Example::Example Payment Page",
"data" => {
"referenceId" => payment["referenceId"],
"transId" => SecureRandom.uuid(),
"amount" => payment["transactionData"]["amount"]
}
}
# Example of how to send data to user page
#{
# "action" => "return",
# "data" => page
#}
# Example of how to redirect to another site
#{
# "action" => "redirect",
# "redirect_to" => "http://urltohostedpayments"
# }
a. Get Settings object - To retrieve the record from the Settings datatype, call the get_setting algorithm
and pass the name of the record
setting = get_setting("EXAMPLE_SETTINGS")
b. Get Payment object - retrieve the Payment object that is sent to the algorithm through
params
payment = params["payment"]
c. Response method - There are three options for a response method:
- Render a template
{
"action" => "render",
"template" => "Example::Example Payment Page",
"data" => {
"referenceId" => payment["referenceId"],
"transId" => SecureRandom.uuid(),
"amount" => payment["transactionData"]["amount"]
}
}
- Send data to user page
{
"action" => "return",
"data" => page
}
- Redirect to another site
{
"action" => "redirect",
"redirect_to" => "http://urltohostedpayments"
}
- Save the algorithm once completed.
process_payment
- Create an algorithm called process_payment with a parameter called params
data:image/s3,"s3://crabby-images/2150b/2150b710637d13c23da85b1a60ebca7c9fdc46b7" alt="ss3.png 777"
- Write your code in the code block
data:image/s3,"s3://crabby-images/6af8d/6af8de30d4a69c955f6a907697bdcbc2671ee165" alt="ss4.png 1167"
Tenant.notify(message: "Process Example process_payment", type: :info)
Tenant.notify(message: "#{params}", type: :info)
begin
# lookup the Payment entry by referenceId
payment = get_payment(params["referenceId"])
if payment.nil?
Cenit.fail("No Payment entry found for referenceId #{params["referenceId"]}")
end
# Check if the current status is Pending Payment and apply payment if so
if payment["transactionStatus"].casecmp?("Pending Payment")
# update the transactionStatus to either Approved or Failed
status = ""
if params["status"].eql?("APR")
status = "Approved"
end
payment["transactionStatus"] = status
# update the transactionId with the response from the payment provider
payment["transactionId"] = params["transId"]
# save any parameters sent back from the payment provider for later use
payment["transactionData"] = params
# add the payment provider transaction id as the transactionData.referenceId
payment["transactionData"]["referenceId"] = params["transId"]
# set the transactionData.payee to the info returned from the payment provider
payment["transactionData"]["payee"] = "#{params["payeeFirstname"]} #{params["payeeLastname"]}"
# set the transactionData.payee to the info returned from the payment provider
payment["transactionData"]["payeeAddress"] = "#{params["payeeAddress"]}"
# optionally build a URL to the paymnet providers detail page
payment["transactionData"]["detailURL"] = "https://paymentprovider/detail"
# set the transactionData.type
payment["transactionData"]["type"] = "Credit Card"
# payment["transactionData"]["type"] = "Check"
# optionally set the convenience or service fee charged by the payment provder
#payment["transactionData"]["convFee"] = 0
# save the payment
payment = save_payment(payment)
# return a success message as required by the payment provider
#{
# "action"=> "return",
# "code"=> "200",
# "data"=> "Payment completed successfully"
#}
# redirect to transactionData.redirectURL which
# will exist on the Payment after calling save_payment with the
# transactionStatus of Approved or Failed
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
elsif payment["transactionStatus"].casecmp?("Approved") || payment["transactionStatus"].casecmp?("Failed")
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
else
{
"action"=> "return",
"code"=> "500",
"data"=> "Payment in invalid state of #{payment["transactionStatus"]} for further actions"
}
end
rescue StandardError => ex
code = ex.message =~ /^\[(\d+)\]/ ? $1.to_i : 500
message = ex.message.gsub(/^\[(\d+)\] - /, '')
{
"action"=> "return",
"code"=> code,
"data"=> message
}
end
a. Lookup Payment record
- Begin the algorithm by looking up the Payment record by the referenceId
payment = get_payment(params["referenceId"])
- Add an if statement to check if the response from get_payment is null and to end the execution if it is
if payment.nil?
Cenit.fail("No Payment entry found for referenceId #{params["referenceId"]}")
end
b. Check if current status is Pending
- Check if the current status is Pending Payment and apply payment if so
if payment["transactionStatus"].casecmp?("Pending Payment")
- Update the transactionStatus to either Approved or Failed
status = ""
if params["status"].eql?("APR")
status = "Approved"
end
payment["transactionStatus"] = status
- Update the transactionId with the response from the payment provider
payment["transactionId"] = params["transId"]
- Save any parameters sent back from the payment provider for later use
payment["transactionData"] = params
- Add the payment provider transaction id as the transactionData.referenceId
payment["transactionData"]["referenceId"] = params["transId"]
- Set the transactionData.payee to the info returned from the payment provider
payment["transactionData"]["payee"] = "#{params["payeeFirstname"]} #{params["payeeLastname"]}"
- Set the transactionData.payeeAddress to the info returned from the payment provider
payment["transactionData"]["payeeAddress"] = "#{params["payeeAddress"]}"
- Optionally build a URL to the payment providers detail page
payment["transactionData"]["detailURL"] = "https://paymentprovider/detail"
- Set the transactionData.type
payment["transactionData"]["type"] = "Credit Card"
OR
payment["transactionData"]["type"] = "Check"
- Optionally set the convenience or service fee charged by the payment provder
payment["transactionData"]["convFee"] = 0
- Save the payment
payment = save_payment(payment)
c. Redirect to transactionData.redirectURL which will exist on the Payment after calling save_payment with the transactionStatus of Approved or Failed
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
elsif payment["transactionStatus"].casecmp?("Approved") || payment["transactionStatus"].casecmp?("Failed")
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
else
{
"action"=> "return",
"code"=> "500",
"data"=> "Payment in invalid state of #{payment["transactionStatus"]} for further actions"
}
end
d. Ending code
rescue StandardError => ex
code = ex.message =~ /^\[(\d+)\]/ ? $1.to_i : 500
message = ex.message.gsub(/^\[(\d+)\] - /, '')
{
"action"=> "return",
"code"=> code,
"data"=> message
}
end
payment_confirmation
- Create an algorithm called payment_confirmation with a parameter called params
data:image/s3,"s3://crabby-images/68377/68377a8931aa92029ef5205a91c2cf3c59cfb6ac" alt="ss5.png 698"
- Write your code in the code block
data:image/s3,"s3://crabby-images/07458/0745849df9c5ce47b6bc5c8267c9f6e2c8baf315" alt="ss6.png 1181"
Tenant.notify(message: "Process Example payment_confirmation", type: :info)
Tenant.notify(message: "#{params}", type: :info)
begin
# lookup the Payment entry by referenceId
payment = get_payment(params["referenceId"])
if payment.nil?
Cenit.fail("No Payment entry found for referenceId #{params["referenceId"]}")
end
# Check if the Payment is Approved or Failed and redirect
if payment["transactionStatus"].casecmp?("Approved") || payment["transactionStatus"].casecmp?("Failed")
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
else
{
"action"=> "return",
"code"=> "500",
"data"=> "Payment in invalid state of #{payment["transactionStatus"]} for further actions"
}
end
rescue StandardError => ex
code = ex.message =~ /^\[(\d+)\]/ ? $1.to_i : 500
message = ex.message.gsub(/^\[(\d+)\] - /, '')
{
"action"=> "return",
"code"=> code,
"data"=> message
}
end
a. Similarly to the process_payment algorithm, begin payment_confirmation by looking up the Payment record by referenceId and check if it is null
payment = get_payment(params["referenceId"])
if payment.nil?
Cenit.fail("No Payment entry found for referenceId #{params["referenceId"]}")
end
b. Check if the Payment status is Approved or Failed and redirect
if payment["transactionStatus"].casecmp?("Approved") || payment["transactionStatus"].casecmp?("Failed")
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
else
{
"action"=> "return",
"code"=> "500",
"data"=> "Payment in invalid state of #{payment["transactionStatus"]} for further actions"
}
end
c. End code
rescue StandardError => ex
code = ex.message =~ /^\[(\d+)\]/ ? $1.to_i : 500
message = ex.message.gsub(/^\[(\d+)\] - /, '')
{
"action"=> "return",
"code"=> code,
"data"=> message
}
end
- Save algorithm once completed.
cancel_payment
- Create an algorithm called cancel_payment with a parameter called params
data:image/s3,"s3://crabby-images/def3b/def3b303275d1a831aec41c5b9423980cd74bd46" alt="ss7.png 1920"
- Write your code in the code block
data:image/s3,"s3://crabby-images/90a3d/90a3dee8f10b9f43b30ade7939676a20238b9e91" alt="ss8.png 1100"
Tenant.notify(message: "Process Example cancel_payment", type: :info)
Tenant.notify(message: "#{params}", type: :info)
payment = get_payment(params["referenceId"])
if payment.nil?
Cenit.fail("No Payment entry found for referenceId #{params["referenceId"]}")
end
payment["transactionStatus"] = "Cancel"
payment["transactionId"] = params["SECURETOKENID"]
payment["transactionData"] = params
payment = save_payment(payment)
if payment["transactionData"]["redirectURL"]
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
else
{
"action"=> "return",
"code"=> "200",
"data"=> "Payment cancelled"
}
end
a. Lookup Payment record by reference Id and check if it is null or not
payment = get_payment(params["referenceId"])
if payment.nil?
Cenit.fail("No Payment entry found for referenceId #{params["referenceId"]}")
end
b. Set Payment record details and save Payment
payment["transactionStatus"] = "Cancel"
payment["transactionId"] = params["SECURETOKENID"]
payment["transactionData"] = params
payment = save_payment(payment)
c. Set redirect
if payment["transactionData"]["redirectURL"]
{
"action" => "redirect",
"redirect_to" => payment["transactionData"]["redirectURL"]
}
else
{
"action"=> "return",
"code"=> "200",
"data"=> "Payment cancelled"
}
end
- Save algorithm
Using the SDK
- Go to laserfiche.velosimo.com and login
- Open your forms Process Designer Diagram and show details
- Select start in the diagram and the On Event Completion tab
data:image/s3,"s3://crabby-images/c7838/c7838d2a2ec8920c6361ba5e090d66108b90c438" alt="ss9.png 1920"
- Set redirect to website and the URL to the initialize_payment algorithm on the Velosimo site.
Example - https://dev.velosimo.io/app/payment_adapter/laserfiche/example/initialize_payment
payment_adapter = Velosimo tenant
example = Velosimo namespace
initialize_payment = Algorithm
data:image/s3,"s3://crabby-images/45cc3/45cc3b050efca7aa5dfce2546b4deee2c3da4d3b" alt="ss10.png 574"
- Submit a new form. Once submitted you will be redirected to an environment where you can test each algorithm created.
data:image/s3,"s3://crabby-images/fd283/fd283aac722a8b7984dfe8e52f6a85d5d477bdfc" alt="ss11.png 924"
a. Click Simulate Approved Postback
data:image/s3,"s3://crabby-images/d295a/d295a9742f9f4ae04b95fe32e595c059cc281245" alt="Screen Shot 2022-04-27 at 2.03.26 PM.png 1692"
Updated almost 3 years ago