Accepting PayPal in games(完整的Paypal在Unity的支付)
Hello and welcome back to my blog!
In this article I’m going to talk about the process of accepting payment via PayPal in your own games. Although I focus on Flash, the techniques and web-calls can be performed in pretty much any language, as long as there is internet access and the platform is capable of opening up a browser tab/page after clicking a button in game.
Requirements
The prerequisites for the readers of this article:
- Must have access to a PayPal business account
- Must have access to a server with a LAMP stack
- Must understand php/mysql and basic server configurations
If you don’t have a server, but still want to be able to accept PayPal, or if the idea of server-side security scares you, scroll to the bottom of this article for details on how you can achieve this.
Lets set out the requirements for the design of the system as well:
- Enable users to buy individual items like levels, or the full version of their game
- Automated delivery system so users get immediate access to their purchases
- Login/registration system so users can access their purchased items from any computer
- Users, purchases and transactions stored in database
- Must be secure
- Item prices must be editable in real time
- Must have automated password-reset system
- Must have automated email sending system on transaction failure
- Must leave the IPN message notification URL free for other applications
How will it work?
PayPal has a system called IPN, which stands for Instant Payment Notification. What this means is that when someone buys something via your PayPal account, PayPal will send a notification to a location you specify somewhere on the internet.
Figure 1
Once you have been notified and have verified the details (to combat fraud), you can then give the purchaser access to the item in the database. The game will be polling your server in the background for changes in a user’s purchases, once it detects one, the game can finally give the user access to the item they purchased and the transaction is complete. This sounds like a long winded process, and technically it is, but its all over in the course of a few seconds in reality.
Setting up your PayPal business account
In order for this process to be possible IPN must be enabled on your PayPal business account. First locate the settings area in Profile->My selling preferences:
Click for enlargement
Set ‘Message Delivery’ to Enabled – you can leave the Notification URL as it was, this enables you to use other applications which require an IPN URL to be set:
Click for enlargement
Next (and this is optional), I recommend blocking eCheques. The reason is they are not an instant form of payment, and can take days to clear leading to a bad customer experience and support requests for you to deal with:
Click for enlargement
Blocking continued:
Click for enlargement
Set up a PayPal sandbox account
This is a good time to set yourself up with a PayPal sandbox account. This will let you send a lot of dummy transactions without actually spending any real money, which is good because there will no doubt be a lot of transactions sent in the course of development!
The IPN script on your server
PayPal provide some example code of what the server-side script should look like which receives notifications from PayPal:
// PHP 4.1 |
Its quite basic but it gives us a decent starting point.
Mysql set-up
In order to process transactions which arrive at the script above, we will need to be able to log them in a database, so its worth setting up a new Mysql database at this point with three tables:
- One for the users of your game
- One for the items they can buy
- One for the transactions they make
Data stored on the server
In order to facilitate the requirements I outlined at the start, the database must store the users of the game, so they can log in/out and still have access to their purchases.
It must store the individual items which are for sale, along with their prices, because we would like to be able to fine tune the prices in real time as people are playing the game.
It must also store the transactions which occur in case of disputes or support requests.
Finally, and most importantly, it must store an inventory of the actual items which each user has bought – this will get sent from the server to the game when a user logs in so the game can ‘unlock’ those items for that user.
The inventory of items
As the requirements call for an inventory of items, the most compact way this can be stored is in a uint32, with one bit per item, leaving a possible maximum of 32 items purchasable. This should be plenty for most games. This uint32 is stored directly on the user row, which makes for excellent data locality. If your game requires a virtual currency system, you can also add this to the user row in a similar fashion, but I’m not going to cover virtual currency in this article.
Note that because php doesn’t actually have a unsigned integer data type, we end up with an int32 and 31 possible purchasable items.
The full user row
Here is what the full user row looks like for this system:
Figure 2
I’ve chosen to store the user’s email address as their username (just because its much easier for them to remember), a salted sha1 hash of their password, some date information, their inventory items (item_flags) and finally one bit to indicate if there has been a transaction error which has not yet been displayed to the user.
Security note: never store a user’s password in plain-text, if your database were to be compromised you could be responsible for leaking a password which they may well use for other purposes. Its much better to hash that password. Even better is to salt and hash their password, that way you avoid possible rainbow table attacks on the hashed password, should it be leaked. To see what I mean, just type 2fd4e1c67a2d28fced849ee1bb76e7391b93eb12 into google.
The purchasable items
Because I’ve chosen a uint32 to store the users items, each individual item must be represented by a single number which must be a power of two, otherwise known as a ‘bit’.
Figure 3
Figure 3 shows the full data set for the items. An important point to note is that the price is stored decimal ‘fixed point’, to avoid any floating point rounding problems, so each actual price is multiplied by 100.
There is a gottcha associated with this decimal fixed point system; in php, if you take the following example:
$n = "19.99"; |
You end up with 1998 being echoed. Now, this wouldn’t be a problem if php was less weakly typed, but because it isn’t you need to take special care when converting between decimal and fixed point:
$n = "19.99"; |
Yields the correct result.
The transactions
Figure 4 - click to enlarge
Finally, Figure 4 shows the transactions table entries; we’re storing everything which PayPal sends us via the IPN script as well as some extra data indicating the actual user who’s purchase this is etc.
Foreign key constraint
Its also worth taking advantage of the relational nature of mysql by adding a foreign key constraint to the transactions table. This simply says that the user_uid field of the transactions table is tied to the uid field of the users table. Additionally, a transaction cannot exist without the user who made it existing first, and indeed cannot be deleted without the user being deleted first.
It will look something like this:
ALTER TABLE transactions |
Sql query to set it all up
Here is an sql query which will set up all these tables:
-- phpMyAdmin SQL Dump |
Sql injection
The phrase which fills every web developer with fear. Mysql injection is covered here, the summary is: anything which is stored in your database from user input is vulnerable to malicious attacks by users who attempt to insert SQL commands into the data being input to gain access to secret information stored in your database, or just to do harm to it.
Luckily Mysql has this covered in the form of prepared statements which handle this problem for you. The disadvantage is they are quite cumbersome compared to the traditional way of performing queries, as you can see from the following example, which attempts to find out the district which a city is located within:
/* create a prepared statement */ |
However, luckily someone has made a wrapper function to do all the hard work for you and can be found in the comments on the php documentation for mysqli::prepare(). Using this, the above becomes:
$results = mysqli_prepared_query($mysqli, "SELECT District FROM City WHERE Name=?", array("s", $city)); |
This should not be used if you need to return huge amounts of data in a single query because it packages the data up into a php array, but for simple results its more than adequate. In the example code which follows, I’ve packaged this up into a Database class for neatness.
Altering the IPN script to log the data
As I mentioned above the IPN script example which PayPal give is a good starting point, but it needs a lot of verification and logging adding to it before it will be ready for production. Here is my finished IPN script, you should take this as psudocode because it relies on a lot of the library functions I’ve built up during the course of writing the demo accompanying this article – hopefully it should be obvious enough what is happening, though:
if ( isset($_POST['item_name']) && |
The script now performs a series of operations:
- Checks for correct parameters
- Checks for valid security token (more on that soon)
- Checks the user exists
- Checks transaction is actually valid via PayPal
- Checks the merchant email (your PayPal business email) is correct
- Checks the amount paid for the given item is correct (more on that soon)
- Checks the item exists
- Checks the money actually entered your PayPal account
- Unlocks the item in the database for the correct user
- Logs the complete transaction
- Logs any failed transaction on the user row (more on that soon)
- Sends an email to your support email address and the user on failure
- Finally, it logs any special case failures in the system error_log
PAYPAL_END_POINT can either be ssl://www.sandbox.paypal.com or ssl://www.paypal.com, depending on whether you want to use sandbox or live mode.
Fraud detection
There are a couple of things going on in the script to check for fraud.
No.1 is actually confirming that the item price and number hasn’t been tampered with.
The reason for this is that the price for an item is simply a GET parameter on the HTTP request to the PayPal servers (i.e. a URL which gets opened in the browser). A fraudster* could intercept the URL the game was generating and replace the price with 0.01, which is the lowest price PayPal will allow, thereby paying far less than he should have. Instead of looking up the item number in the database and checking the price against the price paid, I check the item_hash passed in the $_GET parameters matches a hash of the given name, number and price of the item. The reason for this is that there is an edge case because the prices are transmitted to the game at log-in time. If I just checked the item price in the database, the database entry could have been updated with new price values (tweaked as per the requirements) while the user was playing the game. This would cause the IPN script would fail. Doing things with a hash mean the user always pays the price that was when he first logged in, yet you are still free to tune your prices in real time.
* I like the word ‘fraudster’. It puts be in mind of the one of the cars designed by the late Boyd Coddington, the Boydster.
No.2 is to check for a valid security token.
I will cover this in more detail later, but the idea is that this is a unique key generated on your server when a user logs in, and passed back to the client side. All future web-requests coming from the client must be accompanied with this token to be accepted.
No.3 is to check that the transaction was actually real.
If a fraudster finds your ipn script address, he could post to it with the correct parameters to fake a transaction which didn’t actually occur. In order to check for this, the script calls PayPal to confirm the transaction actually occurred.
Client side
We’ve talked exclusively about the server-side of this process up until now, but lets take a look at the client side.
Web-requests
Key to this process is the client’s ability to send web-requests to the server. The platform being used must have this ability or this process will not be possible.
I’m using POST requests almost exclusively in this article. Furthermore, I’m expecting the response to these web-requests to be in JSON format.
Here is a simple AS3 class to let you perform web-requests:
package Wildbunny.System |
And here is that class extended to handle JSON deserialisation:
package Wildbunny.System |
In order to actually perform the deserialisation, I’ve used the excellent JSON Lite, available here.
Annoyingly, Flash doesn’t let you access the content-type headers of the data returned from a web-request, so I am unable to check that the data is of type “application/json”. You should be able to get access to this in all other languages, though.
Registration
The very first thing a user will do when he is ready to commit to playing your game regularly is to register. This allows you to have a permanent record of this user and his purchases which can then be stored and accessed no matter what computer the user is using; this is in contrast to just using a flash cookie, or local store on the current machine the user is on.
Once the user has entered his details, and you’ve confirmed the email address is valid and the password is of acceptable length, he’ll press ‘Register’, and fire off a web-request to the server.
The web-request simply fills in two POST parameters, one called ‘email’, the other ‘password’ and fires off a request to a script on the server called ‘register.php’. ‘register.php’ does some validity checking (as the IPN script did), and then it creates an entry in the ‘users’ table in the database and generates the all important unique security token.
/** |
Update: thanks to user FrogsEye on reddit for pointing out that the previous version of this function was potentially vulnerable to length extension attacks.
Above is the code I use to generate a security token. As you can see the original string (which in this case is simply the users email address, since that’s unique to the user) is salted with a security secret which is known only by the server, and then passed to the HMAC method which is specifically designed to securely generate salted hashes without being vulnerable to length extension attacks which concatenated salts are subject to.
Do not be tempted to use md5 for your hashing as its extremely vulnerable to collision attacks.
The script finally sends a reply in JSON format which contains most of the data in the newly added user row, along with this security token. On the client side this security token is stored for all later web-requests. The JSON reply also contains the entire contents of the ‘items’ table in the database, along with a hash for each item. This is to allow the client side to know what items are available for purchase, how much they are and to be able to post the hash of the item required by the IPN script to verify validity.
Log-in
Returned users are allowed to log-in to the server.
Similarly to the registration process, a web-request is fired off to the server (albeit to a different script), this one attempts to look a user up in the database by username and password:
class Database |
If successful, it returns the user row just as the registration script did, along with the security token and contents of the ‘items’ table.
Password reset
In order to reduce support requests to deal with forgotten passwords, its essential to provide users with a way to reset their passwords. Again this is a web-request which only sends the user’s email address.
In order to make this process secure, never send a user his password via email – in fact this is impossible in this system because we don’t store his password, just a salted hash of it.
What the server-side script does is to look up this email address in the table of users, and send an email to it containing a URL which the user can follow in order to enter a new password. This url is also accompanied by another security token (again the salted hash of his email address). This ensures that the user can only change his own password and not anybody else’s because the token is checked again at the final stage after the user is directed to enter a new password and that has been posted to the script which actually changes the password.
- Client requests password reset
- Server confirms email exists
- Server generates token and emails user with password reset URL
- User follows URL in browser, enters new password
- Server POSTs new password with original token (and email address) to final php script
- Final script confirms token (from email address) and actually changes password
Buying an item with PayPal
Ok, now we’re starting to connect everything together, the key part is how the client actually begins the process of buying something with PayPal.
The item names and prices are sent from the server when the user registers or logs in, so they are easy to display in a list like above. When the user actually selects an item and clicks ‘Continue’ the game must open PayPal in a browser tab or page.
/// <summary> |
The above is the function which builds the PayPal GET request. There are a few important points here:
- kPayPalEndPoint can either be https://www.sandbox.paypal.com/cgi-bin/webscr or https://www.paypal.com/cgi-bin/webscr depending on whether you want the sandbox or live PayPal service
- ‘currency_code’ must be your PayPal currency code
- ‘custom’ is any custom parameter you want passed to your IPN script, in this case I’m putting the users uid in there
- ‘business’ must be your PayPal business email address
- ‘notify_url’ is the web-address of your IPN script, where PayPal will call with the result of the transaction. Some extra parameters have been added to this, including the item_hash value which the IPN script will use to verify the purchase
- ‘amount’ is the price of the item (from the items table)
- ‘item_name’ is the name of the item (from the items table)
- ‘item_number’ is the number of the item (from the items table)
- ‘return’ is the address you’d like to send the user once their purchase is complete. This should be a page in accordance with PayPal rules for transactions.
The user in then presented with a PayPal browser tab:
Click to enlarge
Once they confirm payment, PayPal will fire off a call to the script we specified above.
If all goes well in the verification of the transaction in the IPN script on the server, the user will have the item_number field bitwise or’ed into his item_flags field on the users table:
class Database |
In the meantime the client will have been polling the server at regular intervals via a web-request, the server will be returning the user row from the database each time. When the client detects a difference between its version of the user’s item_flags and the new version it knows that the user has been awarded his item on the server and can actually unlock that item in game. Additionally, the client will be checking for the ipn_error bit which is getting sent back as it polls the server; if this bit is set, the client should display a message informing the user that there was a problem with their transaction and to check their email for details.
The email was sent via the IPN script above. Once this message has been displayed to the user, its important to clear the ipn_error bit on the users row so that the user doesn’t get shown the same message multiple times. Again this is just another web-request to a different script on the server, signed with the same security token that accompanies all web-requests.
Flash security sandbox settings
In order for flash to be able to access your server while running your project locally on your machine, you will need set your security sandbox settings to allow your swf to access external domains. You do this by going to this web address in your browser Global security settings for content creators and selecting the .swf you are working on.
Click for enlargement
Crossdomain.xml
You will also need a crossdomain.xml file in the root of the domain where your php scripts are located. This allows your flash client to connect with your server wherever the client is located on the internet. It should look something like this:
<?xml version="1.0"?> |
Server security
Its worth mentioning a few points about server-side security set-up at this point.
Disallow directory browsing
You don’t want potential hackers looking at the names of all the files you’ve written.
Move all possible files outside of the web-root
Collect all library functions and globals, especially those containing sensitive information such as your mysql login details and security hash secret and move them outside of the web-server root directory. Doing this safeguards you in case you server gets hacked and suddenly starts displaying the code inside the .php files instead of executing it. You will need to add this new include path via .htaccess files like this:
php_value include_path “.:/usr/local/lib/php:/somewhere outside your web root”
Obviously you can’t do this with files which need to be visible to the client.
Stop displaying php errors to the web-browser
Have them logged in error_log instead. You don’t want an errant message to display your php code, or mysql queries.
Set up a separate sub-domain
Its advisable to set up a separate sub-domain just for your php scripts which are visible to the web-service from the client. This way, the crossdomain.xml is restricted to only allowing access to those scripts from Flash, rather than your entire website.
Back-up your data
Set up a cron job to perform a nightly back-up of your database and files.
The demo
That pretty much covers the implementation of this system, all that remains is to show the live demo:
What if I don’t have a server?
If you don’t have access to a secure server, or the thought of getting all the details right fills you with dread, don’t fear! There are services out there which will provide you with the same feature set. Indeed, here at Wildbunny I have just released such a service!
PayDirt will allow you to accept PayPal in your own Flash games, its free to sign up, come take a look!
I’d love to get some feedback
Buy the source code?
I’ve been umming and arring about allowing people to buy the source code which accompanies the demo above. The main reason I not currently going to offer it is because I don’t think I will have enough time to be able to field support requests regarding the code while also looking after the running and support of PayDirt.
If you really want to buy the code, leave a comment on this article and if I get enough responses I will make it available. Please be aware that it’s likely to be around $99 USD because of the amount of time and effort which went into creating it.
Until next time, have fun!
Cheers, Paul.
Accepting PayPal in games(完整的Paypal在Unity的支付)的更多相关文章
- 跨境网上收款 找PayPal没错(php如何实现paypal支付)
开发前准备 在我的博客中 有介绍如何获取ID 和 secret : 好了 在上一篇博客中详细介绍了也不少: 跨境网上收款 找PayPal没错(如何获取ID 和 secret) http://blog. ...
- WHMCS成功安装和使用方法及添加支付宝,PayPal收款教程
一.WHMCS安装前准备 1.WHMCS官网: 1.官方首页:http://www.whmcs.com/ 2.WHMCS需要安装在一个带MysqL数据库的PHP服务器中,一般地我们日常安装的VPS控制 ...
- 海外支付:遍布全球的Paypal
海外支付:遍布全球的Paypal 吴剑 2015-11-26 原创文章,转载必需注明出处:http://www.cnblogs.com/wu-jian 吴剑 http://www.cnblogs.co ...
- PayPal 开发详解(六):下载paypal立即付款SDK 并编译打包
1.下载PayPal REST SDKs,地址:https://developer.paypal.com/docs/api/rest-sdks/ paypal api比较混乱,有的已经不推荐使用,比如 ...
- PayPal 开发详解(一):注册PayPal帐号
1.注册paypal帐号 https://www.paypal.com 2.使用刚才注册的paypal帐号登录3.进入开发者中心 4.登录开发者中心 5.登录 查看我们paypal Sandbox测试 ...
- ecstore使用paypal支付显示不支持此支付
问题描述: ecstore使用paypal支付,下单结算时显示不支持此支付. 问题和代码分析: 1.首先必须要保证默认货币是paypal支持的货币,paypal目前支付 ["supportC ...
- 解决外贸电商难题,PayPal中国外贸电商大会圆满礼成
在全球经济一体化的背景下,越来越多的中国企业将目光转移到了海外.对中国的企业而言,要想将生意做到海外大致有两种方法可供选择,一是到海外设立分支机构或者分公司,二是通过外贸电子商务平台实现交易. ...
- 国外支付PayPal
PayPal官网https://www.paypal.com/ PayPal沙箱https://www.sandbox.paypal.com/signin?country.x=US&local ...
- Paypal 支付功能的 C# .NET / JS 实现
说明 最近用到了 Paypal 支付功能,英语一般般的我也不得不硬着头皮踩一踩这样的坑.经过近乎半个月的作,终于实现了简单的支付功能,那么首先就说说使用 Paypal 必定要知道的几点(当前日期 20 ...
随机推荐
- 20165203实验四 Andriod程序设计
20165203实验四 Andriod程序设计 实验内容 安装 Android Stuidio 学习Android Stuidio调试应用程序 实验要求 1.没有Linux基础的同学建议先学习< ...
- vue 插槽slot
本文是对官网内容的整理 https://cn.vuejs.org/v2/guide/components.html#编译作用域 在使用组件时,我们常常要像这样组合它们: <app> < ...
- SCTF 2015 pwn试题分析
Re1 是一个简单的字符串加密.程序使用了多个线程,然后进行同步.等加密线程加密好了之后才会启动验证线程.这个题比较坑的是IDA F5出来的结果不对,不知道是不是混淆机制. 刚开始看的是F5后的伪代码 ...
- mysql 闪回测试
由于前面出现过几个需求,或者误操作,或者测试,需要我把某张表恢复到操作之前的一个状态,前面在生产中有过几次经历,实在太痛苦了,下面是一张表被误删除了,我的步骤是: 1 用全备恢复整个库(恢复到其他环 ...
- Java语法知识总结
一:java概述: 1991 年Sun公司的James Gosling等人开始开发名称为 Oak 的语言,希望用于控制嵌入在有线电视交换盒.PDA等的微处理器: 1994年将Oak语言更名为Java: ...
- abtest分流随机链接方法(javascript)
¶¹¸¯¸ÉËêµÄ·¨¹úµçÊÓ¸²¸Ç --> 代码如下 <!DOCTYPE HTML> <html> <head> <script type=& ...
- 【51nod】1123 X^A Mod B (任意模数的K次剩余)
题解 K次剩余终极版!orz 写一下,WA一年,bug不花一分钱 在很久以前,我还认为,数论是一个重在思维,代码很短的东西 后来...我学了BSGS,学了EXBSGS,学了模质数的K次剩余--代码一个 ...
- 解决loadrunner 脚本和replaylog中的中文乱码问题
解决loadrunner 脚本和replaylog中的中文乱码问题 解决这个问题必须认识到一个事实就是,loadrunner和测试服务器交换数据使用的是utf8格式,但是展现在replaylog中是使 ...
- 警告: [SetPropertiesRule]{Server/Service/Engine/Host/Context} Setting property 'source' to 'org.eclipse.jst.jee.server:fhcq-oa' did not find a matching property.
当你在使用Eclipse运行web项目时,你可能会看到控制台出现: 警告: [SetPropertiesRule]{Server/Service/Engine/Host/Context} Settin ...
- 使用 Web 服务 为 ECS Linux 实例配置网站及绑定域名
Nginx 服务绑定域名 https://help.aliyun.com/knowledge_detail/41091.html?spm=a2c4e.11155515.0.0.4lvCpF 以 YUM ...