In this article, we will learn how to implement a Paykun payment gateway in ASP.NET MVC application.
Let’s begin.
Go to https://paykun.com/ and create an account.
Then go to Test Mode and login once again.
In Test Mode, go to Settings and generate API Key and Secret Key.
You can download these details by clicking Download Details button.
Then complete your registration and go to visual studio to create .NET MVC project.
C# Code Example
After create a project and then add this code in HomeController.
public ActionResult Index() { return View(); } [HttpPost] public ActionResult Index(PaykunVM paykunVM) { paykunVM = new PaykunVM() { Amount = paykunVM.Amount, Email = paykunVM.Email, Mobile = paykunVM.Mobile, Name = paykunVM.Name, ProductName = paykunVM.ProductName, }; ViewBag.Encrypt = Paykun.CreatePayment(paykunVM); return View(); }
Then add this code in Index.cshtml.
@{ ViewBag.Title = "Index"; } <html> <head> </head> <body> @{ if (ViewBag.Encrypt == "" || ViewBag.Encrypt == null) { <form action='/Demo/Index' method='post' name='server_request' target='_top'> Email <input type="text" id="Email" name="Email" /><br /> ProductName <input type="text" id="ProductName" name="ProductName" /><br /> Mobile <input type="text" id="Mobile" name="Mobile" /><br /> Amount<input type="text" id="Amount" name="Amount" /><br /> Name <input type="text" id="Name" name="Name" /><br /> <button type="submit">Pay now</button> </form> } else { <div>Processing your payment, please wait...</div> <form action='https://sandbox.paykun.com/payment' method='post' name='server_request' target='_top'> <table width='80%' align='center' border='0' cellpadding='0' cellspacing='0'> <tr> <td> <input type='text' name='encrypted_request' id='encrypted_request' value='@ViewBag.Encrypt' /> </td> </tr> <tr> <td> <input type='text' name='merchant_id' id='merchant_id' value="enter your merchant_id" /> </td> </tr> <tr> <td> <input type='text' name='access_token' id='access_token' value="enter your access_token"> </td> </tr> </table> <script type='text/javascript'>document.server_request.submit();</script> </form> } } </body> </html>
Then add an new controller Paykun.cs and add this code.
public static PaykunPayment _payment; public static void Setup(string merchantId, string accessToken, string appSecret, bool isLive) { //_payment = new PaykunPayment(merchantId, accessToken, appSecret, _isLive: isLive); _payment = new PaykunPayment("add here your merchantId", "add here your accessToken", "add here your appSecret", _isLive: false); } public static string CreatePayment(PaykunVM PaykunVM) { _payment = new PaykunPayment("add here your merchantId", "add here your accessToken", "add here your appSecret", _isLive: false); string _orderId = "ORD" + (new Random()).Next(111111111, 999999999).ToString(); string _productName = PaykunVM.ProductName; _payment.InitOrder(_orderId, PaykunVM.Amount, _productName, "", ""); _payment.AddCustomer(PaykunVM.Name, PaykunVM.Email, PaykunVM.Mobile); string _res = _payment.Submit(); return _res; }
Then add a new controller Crypto.cs and add this code.
public class Crypto { private static readonly Encoding encoding = Encoding.UTF8; public string Encrypt(string _plainText, string _encKey) { string _iv = null; string _encText = this.EncryptInternal(_plainText, _encKey, ref _iv); string _hmac = this.GetHMAC((_iv + _encText), _encKey); Dictionary<string, string> _encData = new Dictionary<string, string>(); _encData.Add("iv", _iv); _encData.Add("value", _encText); _encData.Add("mac", _hmac); string _finalString = this.DictionaryToJson(_encData); return Convert.ToBase64String(encoding.GetBytes(_finalString)); } private string EncryptInternal(string _plainText, string _encKey, ref string _iv) { try { RijndaelManaged aes = new RijndaelManaged(); aes.KeySize = 256; aes.BlockSize = 128; aes.Padding = PaddingMode.PKCS7; aes.Mode = CipherMode.CBC; aes.Key = encoding.GetBytes(_encKey); aes.GenerateIV(); var IV = aes.IV; _iv = Convert.ToBase64String(IV); ICryptoTransform AESEncrypt = aes.CreateEncryptor(aes.Key, IV); _plainText = (new Serializer()).Serialize(_plainText); byte[] buffer = encoding.GetBytes(_plainText); string encryptedText = Convert.ToBase64String(AESEncrypt.TransformFinalBlock(buffer, 0, buffer.Length)); return encryptedText; } catch (Exception e) { throw new Exception("Error encrypting: " + e.Message); } } private String GetHMAC(String text, String key) { // change according to your needs, an UTF8Encoding // could be more suitable in certain situations ASCIIEncoding encoding = new ASCIIEncoding(); Byte[] textBytes = encoding.GetBytes(text); Byte[] keyBytes = encoding.GetBytes(key); Byte[] hashBytes; using (HMACSHA256 hash = new HMACSHA256(keyBytes)) hashBytes = hash.ComputeHash(textBytes); return BitConverter.ToString(hashBytes).Replace("-", "").ToLower(); } public string DictionaryToJson(Dictionary<string, string> dict) { var entries = dict.Select(d => string.Format("\"{0}\": \"{1}\"", d.Key, string.Join(",", d.Value))); return "{" + string.Join(",", entries) + "}"; } } public class Serializer { private Dictionary<Hashtable, bool> seenHashtables; //for serialize (to infinte prevent loops) private Dictionary<ArrayList, bool> seenArrayLists; //for serialize (to infinte prevent loops) lol private int pos; //for unserialize public bool XMLSafe = true; //This member tells the serializer wether or not to strip carriage returns from strings when serializing and adding them back in when deserializing //http://www.w3.org/TR/REC-xml/#sec-line-ends public Encoding StringEncoding = new System.Text.UTF8Encoding(); private System.Globalization.NumberFormatInfo nfi; public Serializer() { this.nfi = new System.Globalization.NumberFormatInfo(); this.nfi.NumberGroupSeparator = ""; this.nfi.NumberDecimalSeparator = "."; } public string Serialize(object obj) { this.seenArrayLists = new Dictionary<ArrayList, bool>(); this.seenHashtables = new Dictionary<Hashtable, bool>(); return this.serialize(obj, new StringBuilder()).ToString(); }//Serialize(object obj) private StringBuilder serialize(object obj, StringBuilder sb) { if (obj == null) { return sb.Append("N;"); } else if (obj is string) { string str = (string)obj; if (this.XMLSafe) { str = str.Replace("\r\n", "\n");//replace \r\n with \n str = str.Replace("\r", "\n");//replace \r not followed by \n with a single \n Should we do this? } return sb.Append("s:" + this.StringEncoding.GetByteCount(str) + ":\"" + str + "\";"); } else if (obj is bool) { return sb.Append("b:" + (((bool)obj) ? "1" : "0") + ";"); } else if (obj is int) { int i = (int)obj; return sb.Append("i:" + i.ToString(this.nfi) + ";"); } else if (obj is long) { long i = (long)obj; return sb.Append("i:" + i.ToString(this.nfi) + ";"); } else if (obj is double) { double d = (double)obj; return sb.Append("d:" + d.ToString(this.nfi) + ";"); } else if (obj is ArrayList) { if (this.seenArrayLists.ContainsKey((ArrayList)obj)) return sb.Append("N;");//cycle detected else this.seenArrayLists.Add((ArrayList)obj, true); ArrayList a = (ArrayList)obj; sb.Append("a:" + a.Count + ":{"); for (int i = 0; i < a.Count; i++) { this.serialize(i, sb); this.serialize(a[i], sb); } sb.Append("}"); return sb; } else if (obj is Hashtable) { if (this.seenHashtables.ContainsKey((Hashtable)obj)) return sb.Append("N;");//cycle detected else this.seenHashtables.Add((Hashtable)obj, true); Hashtable a = (Hashtable)obj; sb.Append("a:" + a.Count + ":{"); foreach (DictionaryEntry entry in a) { this.serialize(entry.Key, sb); this.serialize(entry.Value, sb); } sb.Append("}"); return sb; } else { return sb; } }//Serialize(object obj) public object Deserialize(string str) { this.pos = 0; return deserialize(str); }//Deserialize(string str) private object deserialize(string str) { if (str == null || str.Length <= this.pos) return new Object(); int start, end, length; string stLen; switch (str[this.pos]) { case 'N': this.pos += 2; return null; case 'b': char chBool; chBool = str[pos + 2]; this.pos += 4; return chBool == '1'; case 'i': string stInt; start = str.IndexOf(":", this.pos) + 1; end = str.IndexOf(";", start); stInt = str.Substring(start, end - start); this.pos += 3 + stInt.Length; object oRet = null; try { //firt try to parse as int oRet = Int32.Parse(stInt, this.nfi); } catch { //if it failed, maybe it was too large, parse as long oRet = Int64.Parse(stInt, this.nfi); } return oRet; case 'd': string stDouble; start = str.IndexOf(":", this.pos) + 1; end = str.IndexOf(";", start); stDouble = str.Substring(start, end - start); this.pos += 3 + stDouble.Length; return Double.Parse(stDouble, this.nfi); case 's': start = str.IndexOf(":", this.pos) + 1; end = str.IndexOf(":", start); stLen = str.Substring(start, end - start); int bytelen = Int32.Parse(stLen); length = bytelen; //This is the byte length, not the character length - so we might //need to shorten it before usage. This also implies bounds checking if ((end + 2 + length) >= str.Length) length = str.Length - 2 - end; string stRet = str.Substring(end + 2, length); while (this.StringEncoding.GetByteCount(stRet) > bytelen) { length--; stRet = str.Substring(end + 2, length); } this.pos += 6 + stLen.Length + length; if (this.XMLSafe) { stRet = stRet.Replace("\n", "\r\n"); } return stRet; case 'a': //if keys are ints 0 through N, returns an ArrayList, else returns Hashtable start = str.IndexOf(":", this.pos) + 1; end = str.IndexOf(":", start); stLen = str.Substring(start, end - start); length = Int32.Parse(stLen); Hashtable htRet = new Hashtable(length); ArrayList alRet = new ArrayList(length); this.pos += 4 + stLen.Length; //a:Len:{ for (int i = 0; i < length; i++) { //read key object key = deserialize(str); //read value object val = deserialize(str); if (alRet != null) { if (key is int && (int)key == alRet.Count) alRet.Add(val); else alRet = null; } htRet[key] = val; } this.pos++; //skip the } if (this.pos < str.Length && str[this.pos] == ';')//skipping our old extra array semi-colon bug (er... php's weirdness) this.pos++; if (alRet != null) return alRet; else return htRet; default: return ""; }//switch }//unserialzie(object) }
Output:
Please give your valuable feedback and if you have any questions or issues about this article, please let me know.
Also, check Load Bootstrap Slider With MV