summaryrefslogtreecommitdiffstats
path: root/src/DotNetOpenId/RelyingParty/AssociateResponse.cs
blob: c555f8c3040ec9f0ae6daf3db2c5227cfbcc4d57 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
using System;
using System.Collections.Generic;
using System.Text;
using Org.Mentalis.Security.Cryptography;
using System.Globalization;
using System.Diagnostics;

namespace DotNetOpenId.RelyingParty {
	class AssociateResponse : DirectResponse {
		public AssociateResponse(OpenIdRelyingParty relyingParty, ServiceEndpoint provider, IDictionary<string, string> args, DiffieHellman dh)
			: base(relyingParty, provider, args) {
			DH = dh;

			if (Args.ContainsKey(Protocol.openidnp.assoc_handle)) {
				initializeAssociation();
			} else {
				// Attempt to recover from an unsupported assoc_type
				if (Protocol.Version.Major >= 2) {
					if (Util.GetRequiredArg(Args, Protocol.openidnp.error_code) == Protocol.Args.ErrorCode.UnsupportedType) {
						string assoc_type = Util.GetRequiredArg(Args, Protocol.openidnp.assoc_type);
						string session_type = Util.GetRequiredArg(Args, Protocol.openidnp.session_type);
						// If the suggested options are among those we support...
						if (Array.IndexOf(Protocol.Args.SignatureAlgorithm.All, assoc_type) >= 0 &&
							Array.IndexOf(Protocol.Args.SessionType.All, session_type) >= 0 &&
							RelyingParty.Settings.IsAssociationInPermittedRange(Protocol, assoc_type)) {
							SecondAttempt = AssociateRequest.Create(RelyingParty, Provider, assoc_type, session_type, false);
						}
					}
				}
			}
		}

		void initializeAssociation() {
			string assoc_type = Util.GetRequiredArg(Args, Protocol.openidnp.assoc_type);
			if (Array.IndexOf(Protocol.Args.SignatureAlgorithm.All, assoc_type) >= 0) {
				byte[] secret;

				string session_type;
				if (!Args.TryGetValue(Protocol.openidnp.session_type, out session_type) ||
					Protocol.Args.SessionType.NoEncryption.Equals(session_type, StringComparison.Ordinal)) {
					secret = getDecoded(Protocol.openidnp.mac_key);
				} else {
					try {
						byte[] dh_server_public = getDecoded(Protocol.openidnp.dh_server_public);
						byte[] enc_mac_key = getDecoded(Protocol.openidnp.enc_mac_key);
						secret = DiffieHellmanUtil.SHAHashXorSecret(DiffieHellmanUtil.Lookup(Protocol, session_type), DH, dh_server_public, enc_mac_key);
					} catch (ArgumentException ex) {
						throw new OpenIdException(string.Format(CultureInfo.CurrentCulture,
							Strings.InvalidOpenIdQueryParameterValue,
							Protocol.openid.session_type, session_type), ex);
					}
				}

				string assocHandle = Util.GetRequiredArg(Args, Protocol.openidnp.assoc_handle);
				TimeSpan expiresIn = new TimeSpan(0, 0, Convert.ToInt32(Util.GetRequiredArg(Args, Protocol.openidnp.expires_in), CultureInfo.InvariantCulture));

				try {
					Association = HmacShaAssociation.Create(Protocol, assoc_type,
						assocHandle, secret, expiresIn);
				} catch (ArgumentException ex) {
					throw new OpenIdException(string.Format(CultureInfo.CurrentCulture,
						Strings.InvalidOpenIdQueryParameterValue,
						Protocol.openid.assoc_type, assoc_type), ex);
				}
			} else {
				throw new OpenIdException(string.Format(CultureInfo.CurrentCulture,
					Strings.InvalidOpenIdQueryParameterValue,
					Protocol.openid.assoc_type, assoc_type));
			}
		}
		public DiffieHellman DH { get; private set; }

		[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1820:TestForEmptyStringsUsingStringLength")]
		public Association Association { get; private set; }

		byte[] getDecoded(string key) {
			try {
				return Convert.FromBase64String(Util.GetRequiredArg(Args, key));
			} catch (FormatException ex) {
				throw new OpenIdException(string.Format(CultureInfo.CurrentCulture,
					Strings.ExpectedBase64OpenIdQueryParameter, key), null, ex);
			}
		}

		/// <summary>
		/// A custom-made associate request to try again when an OP
		/// doesn't support the settings we suggested, but we support
		/// the ones the OP suggested.
		/// </summary>
		public AssociateRequest SecondAttempt { get; private set; }
	}
}