Skip to content

Commit 99f6e2f

Browse files
author
Eugene Gusarov
committed
Allow to load certificate from memory
1 parent 5d2f3ce commit 99f6e2f

File tree

5 files changed

+66
-14
lines changed

5 files changed

+66
-14
lines changed

CyberSource/Client/BaseClient.cs

+34
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
using CyberSource.Base;
22
using System;
33
using System.Net;
4+
using System.Security.Cryptography.X509Certificates;
45
using System.ServiceModel;
56
using System.Xml.Serialization;
67
using System.ServiceModel.Channels;
@@ -359,5 +360,38 @@ protected static CustomBinding getWCFCustomBinding(Configuration config)
359360
currentBinding.Elements.Add(httpsTransport);
360361
return currentBinding;
361362
}
363+
364+
/// <summary>
365+
/// Creates a new instance of X509Certificate2
366+
/// </summary>
367+
/// <param name="config">
368+
/// Configuration object containing the key content or file path
369+
/// </param>
370+
/// <returns>New instance of X509Certificate2</returns>
371+
protected static X509Certificate2 GetCertificate(Configuration config)
372+
{
373+
var flags = X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet;
374+
return config.Key != null
375+
? new X509Certificate2(config.Key, config.EffectivePassword, flags)
376+
: new X509Certificate2(config.EffectiveKeyFilePath, config.EffectivePassword, flags);
377+
}
378+
379+
/// <summary>
380+
/// Creates a certificate collection with an imported certificate
381+
/// </summary>
382+
/// <param name="config">
383+
/// Configuration object containing the key content or file path
384+
/// </param>
385+
/// <returns>New instance of X509Certificate2Collection with an imported certificate</returns>
386+
protected static X509Certificate2Collection GetCertificateCollection(Configuration config)
387+
{
388+
var collection = new X509Certificate2Collection();
389+
var flags = X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet;
390+
if (config.Key != null)
391+
collection.Import(config.Key, config.EffectivePassword, flags);
392+
else
393+
collection.Import(config.EffectiveKeyFilePath, config.EffectivePassword, flags);
394+
return collection;
395+
}
362396
}
363397
}

CyberSource/Client/Configuration.cs

+23
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
using System;
2+
using System.IO;
23

34
namespace CyberSource.Clients
45
{
@@ -62,6 +63,7 @@ public class Configuration
6263
private string logDirectory = null;
6364
private string serverURL = null;
6465
private string keyFilename = null;
66+
private byte[] key = null;
6567
private string password = null;
6668
private string logFilename = null;
6769
private int logMaximumSize = -1;
@@ -161,6 +163,15 @@ public string ServerURL
161163
set { serverURL = value; }
162164
}
163165

166+
/// <summary>
167+
/// This is optional. When set, it reads key from memory rather than from file system
168+
/// </summary>
169+
public byte[] Key
170+
{
171+
get { return key; }
172+
set { key = value; }
173+
}
174+
164175
/// <summary>
165176
/// Corresponds to [cybs.][merchantID].keyFilename.
166177
///
@@ -380,6 +391,18 @@ internal string EffectiveKeyFilename
380391
}
381392
}
382393

394+
/// <summary>
395+
/// Return the key file path that will take effect given
396+
/// the current state of this Configuration object.
397+
/// </summary>
398+
internal string EffectiveKeyFilePath
399+
{
400+
get
401+
{
402+
return Path.Combine(KeysDirectory, EffectiveKeyFilename);
403+
}
404+
}
405+
383406
/// <summary>
384407
/// Returns the password that will take effect given
385408
/// the current state of this Configuration object.

CyberSource/Client/NVPClient.cs

+2-3
Original file line numberDiff line numberDiff line change
@@ -80,13 +80,12 @@ public static Hashtable RunTransaction(
8080

8181

8282
string keyFilePath = Path.Combine(config.KeysDirectory, config.EffectiveKeyFilename);
83-
proc.ClientCredentials.ClientCertificate.Certificate = new X509Certificate2(keyFilePath, config.EffectivePassword, X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet);
83+
proc.ClientCredentials.ClientCertificate.Certificate = GetCertificate(config);
8484

8585
proc.ClientCredentials.ServiceCertificate.Authentication.CertificateValidationMode = System.ServiceModel.Security.X509CertificateValidationMode.None;
8686

8787
// Changes for SHA2 certificates support
88-
X509Certificate2Collection collection = new X509Certificate2Collection();
89-
collection.Import(keyFilePath, config.EffectivePassword, X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet);
88+
X509Certificate2Collection collection = GetCertificateCollection(config);
9089

9190
foreach (X509Certificate2 cert1 in collection)
9291
{

CyberSource/Client/SoapClient.cs

+6-8
Original file line numberDiff line numberDiff line change
@@ -79,15 +79,13 @@ public static ReplyMessage RunTransaction(
7979
currentBinding.SendTimeout = timeOut;
8080

8181
//add certificate credentials
82-
string keyFilePath = Path.Combine(config.KeysDirectory,config.EffectiveKeyFilename);
83-
proc.ClientCredentials.ClientCertificate.Certificate = new X509Certificate2(keyFilePath,config.EffectivePassword, X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet);
82+
proc.ClientCredentials.ClientCertificate.Certificate = GetCertificate(config);
83+
84+
proc.ClientCredentials.ServiceCertificate.Authentication.CertificateValidationMode = System.ServiceModel.Security.X509CertificateValidationMode.None;
85+
86+
// Changes for SHA2 certificates support
87+
X509Certificate2Collection collection = GetCertificateCollection(config);
8488

85-
proc.ClientCredentials.ServiceCertificate.Authentication.CertificateValidationMode = System.ServiceModel.Security.X509CertificateValidationMode.None;
86-
87-
// Changes for SHA2 certificates support
88-
X509Certificate2Collection collection = new X509Certificate2Collection();
89-
collection.Import(keyFilePath, config.EffectivePassword, X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet);
90-
9189
foreach (X509Certificate2 cert1 in collection)
9290
{
9391
if (cert1.Subject.Contains(config.MerchantID))

CyberSource/Client/XmlClient.cs

+1-3
Original file line numberDiff line numberDiff line change
@@ -84,13 +84,11 @@ public static XmlDocument RunTransaction(
8484
XmlDocument doc = SoapWrap(request, nspace);
8585

8686
//Get the X509 cert and sign the SOAP Body
87-
string keyFilePath = Path.Combine(config.KeysDirectory, config.EffectiveKeyFilename);
8887

8988
X509Certificate2 cert = null;
9089
X509Certificate2 cybsCert = null;
9190

92-
X509Certificate2Collection collection = new X509Certificate2Collection();
93-
collection.Import(keyFilePath, config.EffectivePassword, X509KeyStorageFlags.Exportable | X509KeyStorageFlags.PersistKeySet);
91+
X509Certificate2Collection collection = GetCertificateCollection(config);
9492

9593
foreach (X509Certificate2 cert1 in collection)
9694
{

0 commit comments

Comments
 (0)