236 lines
7.3 KiB
C#
Raw Normal View History

//
// InMemorySymmetricSecurityKey.cs
//
// Author:
// Atsushi Enomoto <atsushi@ximian.com>
//
// Copyright (C) 2006-2007 Novell, Inc. http://www.novell.com
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
using System;
using System.Collections.Generic;
using System.IO;
using System.Xml;
using System.IdentityModel.Policy;
using System.Security.Cryptography;
using System.Security.Cryptography.Xml;
using M = Mono.Security.Cryptography;
using AES = System.Security.Cryptography.RijndaelManaged;
namespace System.IdentityModel.Tokens
{
public class InMemorySymmetricSecurityKey : SymmetricSecurityKey
{
byte [] key;
public InMemorySymmetricSecurityKey (byte [] key)
: this (key, true)
{
}
public InMemorySymmetricSecurityKey (byte [] key, bool clone)
{
if (key == null)
throw new ArgumentNullException ("key");
this.key = clone ? (byte []) key.Clone() : key;
}
// SymmetricSecurityKey implementation
public override byte [] GenerateDerivedKey (
string algorithm, byte [] label, byte [] nonce,
int derivedKeyLength, int offset)
{
if (derivedKeyLength < 0)
throw new ArgumentOutOfRangeException ("derivedKeyLength");
if (offset < 0)
throw new ArgumentOutOfRangeException ("offset");
if (label == null)
throw new ArgumentNullException ("label");
if (nonce == null)
throw new ArgumentNullException ("nonce");
if (algorithm != SecurityAlgorithms.Psha1KeyDerivation)
throw new InvalidOperationException (String.Format ("Key derivation algorithm '{0}' is not supported", algorithm));
byte [] seed = new byte [label.Length + nonce.Length];
Array.Copy (label, seed, label.Length);
Array.Copy (nonce, 0, seed, label.Length, nonce.Length);
byte [] p_sha = Expand ("SHA1", key, seed, derivedKeyLength / 8);
return p_sha;
}
// from Mono.Security.Protocol.Tls.CipherSuite.Expand() with
// a bit of modification ...
byte [] Expand (string hashName, byte[] secret, byte[] seed, int length)
{
int hashLength = hashName == "MD5" ? 16 : 20;
int iterations = (int)(length / hashLength);
if ((length % hashLength) > 0)
{
iterations++;
}
M.HMAC hmac = new M.HMAC(hashName, secret);
MemoryStream resMacs = new MemoryStream ();
byte[][] hmacs = new byte[iterations + 1][];
hmacs[0] = seed;
for (int i = 1; i <= iterations; i++)
{
MemoryStream hcseed = new MemoryStream ();
hmac.TransformFinalBlock(hmacs[i-1], 0, hmacs[i-1].Length);
hmacs[i] = hmac.Hash;
hcseed.Write(hmacs[i], 0, hmacs [i].Length);
hcseed.Write(seed, 0, seed.Length);
hmac.TransformFinalBlock(hcseed.ToArray(), 0, (int)hcseed.Length);
resMacs.Write(hmac.Hash, 0, hmac.Hash.Length);
}
byte[] res = new byte[length];
Buffer.BlockCopy(resMacs.ToArray(), 0, res, 0, res.Length);
return res;
}
public override byte [] GetSymmetricKey ()
{
return (byte []) key.Clone ();
}
public override KeyedHashAlgorithm GetKeyedHashAlgorithm (
string algorithm)
{
if (algorithm == SecurityAlgorithms.HmacSha1Signature)
return new HMACSHA1 (key);
//if (algorithm == SecurityAlgorithms.HmacSha256Signature)
// return new HMACSHA256 (key);
throw new NotSupportedException ();
}
public override SymmetricAlgorithm GetSymmetricAlgorithm (string algorithm)
{
SymmetricAlgorithm s = null;
switch (algorithm) {
case SecurityAlgorithms.Aes128Encryption:
case SecurityAlgorithms.Aes192Encryption:
case SecurityAlgorithms.Aes256Encryption:
case SecurityAlgorithms.Aes128KeyWrap:
case SecurityAlgorithms.Aes192KeyWrap:
case SecurityAlgorithms.Aes256KeyWrap:
s = new AES ();
break;
case SecurityAlgorithms.TripleDesEncryption:
case SecurityAlgorithms.TripleDesKeyWrap:
if (key.Length == 24)
throw new CryptographicException ("The key size is 24 bytes, which known as vulnerable and thus not allowed.");
s = TripleDES.Create ();
break;
default:
throw new NotSupportedException (String.Format ("This symmetric security key does not support specified algorithm '{0}'", algorithm));
}
s.Mode = CipherMode.CBC;
s.GenerateIV ();
s.Key = key;
return s;
}
public override ICryptoTransform GetDecryptionTransform (string algorithm, byte [] iv)
{
if (iv == null)
throw new ArgumentNullException ("iv");
SymmetricAlgorithm alg = GetSymmetricAlgorithm (algorithm);
alg.IV = iv;
return alg.CreateDecryptor ();
}
public override ICryptoTransform GetEncryptionTransform (string algorithm, byte [] iv)
{
if (iv == null)
throw new ArgumentNullException ("iv");
SymmetricAlgorithm alg = GetSymmetricAlgorithm (algorithm);
alg.IV = iv;
return alg.CreateEncryptor ();
}
[MonoTODO]
public override int GetIVSize (string algorithm)
{
throw new NotImplementedException ();
}
// SecurityKey implementation
public override int KeySize {
get { return key.Length << 3; }
}
public override byte [] DecryptKey (string algorithm, byte [] keyData)
{
if (algorithm == null)
throw new ArgumentNullException ("algorithm");
if (keyData == null)
throw new ArgumentNullException ("keyData");
return EncryptedXml.DecryptKey (keyData, GetSymmetricAlgorithm (algorithm));
}
public override byte [] EncryptKey (string algorithm, byte [] keyData)
{
if (algorithm == null)
throw new ArgumentNullException ("algorithm");
if (keyData == null)
throw new ArgumentNullException ("keyData");
return EncryptedXml.EncryptKey (keyData, GetSymmetricAlgorithm (algorithm));
}
public override bool IsAsymmetricAlgorithm (string algorithm)
{
return GetAlgorithmSupportType (algorithm) == AlgorithmSupportType.Asymmetric;
}
public override bool IsSupportedAlgorithm (string algorithm)
{
switch (algorithm) {
case SecurityAlgorithms.HmacSha1Signature:
case SecurityAlgorithms.Psha1KeyDerivation:
case SecurityAlgorithms.Aes128Encryption:
case SecurityAlgorithms.Aes128KeyWrap:
case SecurityAlgorithms.Aes192Encryption:
case SecurityAlgorithms.Aes192KeyWrap:
case SecurityAlgorithms.Aes256Encryption:
case SecurityAlgorithms.Aes256KeyWrap:
case SecurityAlgorithms.TripleDesEncryption:
case SecurityAlgorithms.TripleDesKeyWrap:
return true;
default:
return false;
}
}
public override bool IsSymmetricAlgorithm (string algorithm)
{
return GetAlgorithmSupportType (algorithm) == AlgorithmSupportType.Symmetric;
}
}
}