/Sherwood.SignOn/Sherwood.SignOn.Server/Models/ViewModels/ServicesViewModel.cs

http://sherwood.codeplex.com · C# · 186 lines · 119 code · 20 blank · 47 comment · 18 complexity · 82d1902ae523c34ae471c1d84ac50116 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Web;
  5. using Sherwood.Security;
  6. using System.Xml;
  7. using Sherwood.SignOn.Server.Models.Repositories;
  8. using Sherwood.SignOn.Server.Models.Services;
  9. namespace Sherwood.SignOn.Server.Models.ViewModels
  10. {
  11. /// <summary>
  12. /// View model base for Services actions.
  13. /// Provides a method for verifying a client request.
  14. /// </summary>
  15. public abstract class ServicesViewModel : ViewModelBase
  16. {
  17. public string ClientCode { get; set; }
  18. public string Timestamp { get; set; }
  19. public string Signature { get; set; }
  20. /// <summary>
  21. /// Checks the ClientCode / Timestamp parameters against the Signature parameter in order to verify the identity
  22. /// of the client.
  23. /// In order to return true:
  24. /// - The timestamp must be within a certain period of time from the current time
  25. /// - The client code must corrospond with the code of a known client for which the sytem has a public Rsa key.
  26. /// - The signature must be an SHA hash of ClientCode|Timestamp, encrypted using the clients private Rsa key.
  27. /// </summary>
  28. /// <returns>True if signature, timestamp and client code could be verified, otherwise false.</returns>
  29. public bool VerifyClient()
  30. {
  31. if (!String.IsNullOrEmpty(ClientCode) && !String.IsNullOrEmpty(Timestamp) && !String.IsNullOrEmpty(Signature))
  32. {
  33. DateTime utcTimestamp = DateTime.MinValue;
  34. utcTimestamp = DateTime.ParseExact(Timestamp, "yyyy-MM-ddTHH:mm:ss.fffZ", System.Globalization.CultureInfo.InvariantCulture);
  35. //Find the difference between the current time and the timestamp sent with the request.
  36. //If dates on both client and server are set correctly, timestamp should be before DateTime.Now
  37. //However, we assume that timestamp can also be after DateTime.Now due to incorrectly set dates.
  38. //We accept requests within a certain timeframe (as defined in web.config)
  39. int secondSinceRequest = (int)Math.Abs((DateTime.Now - utcTimestamp).TotalSeconds);
  40. if (secondSinceRequest > Config.Settings.RequestValidityInSeconds)
  41. {
  42. throw new Exception("Invalid timestamp. Seconds since request: " + secondSinceRequest);
  43. //return false;
  44. }
  45. else
  46. {
  47. bool signatureVerified = Data.Clients.VerifyMessage(ClientCode, SignatureContent, Signature);
  48. if (!signatureVerified)
  49. throw new Exception("Invalid signature.");
  50. return signatureVerified;
  51. }
  52. }
  53. else
  54. {
  55. throw new Exception("ClientCode, timestamp or signature not provided.");
  56. //return false;
  57. }
  58. }
  59. /// <summary>
  60. /// Message signatures should always sign the following content:
  61. /// ClientCode | Timestamp | <see cref="SignatureContent"/>
  62. ///
  63. /// In classes that inherit from this class, SignatureContent should return the
  64. /// SignatureContent part of the signature source.
  65. /// </summary>
  66. public abstract string SignatureContent { get; }
  67. }
  68. /// <summary>
  69. /// View model for Authentication service
  70. /// </summary>
  71. public class AuthenticateServiceViewModel : ServicesViewModel
  72. {
  73. public string UserName { get; set; }
  74. public string Password { get; set; }
  75. /// <summary>
  76. /// Authenticates a user using the username / password parameters
  77. /// </summary>
  78. /// <returns>UserAccount if user could be authenticated, otherwise null</returns>
  79. public IUserAccount AuthenticateUser()
  80. {
  81. if (VerifyClient())
  82. {
  83. if (!string.IsNullOrEmpty(UserName) && !String.IsNullOrEmpty(Password))
  84. {
  85. return Data.UserAccounts.AuthenticateUser(UserName, Password);
  86. }
  87. }
  88. return null;
  89. }
  90. public override string SignatureContent
  91. {
  92. get { return ClientCode + "|" + Timestamp + "|" + UserName + "|" + Password; }
  93. }
  94. }
  95. /// <summary>
  96. /// View model for service that returns UserProfile
  97. /// </summary>
  98. public class GetUserProfileServiceViewModel : ServicesViewModel
  99. {
  100. public string UserAccountId { get; set; }
  101. public string UserName {get; set;}
  102. public string Email { get; set; }
  103. /// <summary>
  104. /// Attempts to find a user profile that corrosponds to either UserAccountId, UserName or Email
  105. /// </summary>
  106. /// <returns>UserAccount if request has been verified and user found, otherwise null.</returns>
  107. public IUserAccount GetUserProfile()
  108. {
  109. if (VerifyClient())
  110. {
  111. if (!String.IsNullOrEmpty(UserAccountId))
  112. {
  113. return Data.UserAccounts.GetUserAccount(new Guid(UserAccountId));
  114. }
  115. else
  116. {
  117. Email = Email == null ? "" : Email.Trim().ToLower();
  118. UserName = UserName == null ? "" : UserName.Trim().ToLower();
  119. if (!String.IsNullOrEmpty(Email) || !String.IsNullOrEmpty(UserName))
  120. {
  121. return Data.UserAccounts.GetUserAccountByUsernameOrEmail(UserName, Email);
  122. }
  123. }
  124. }
  125. return null;
  126. }
  127. public override string SignatureContent
  128. {
  129. get { return ClientCode + "|" + Timestamp + "|" + UserAccountId + "|" + UserName + "|" + Email; }
  130. }
  131. }
  132. /// <summary>
  133. /// View model for service that checks if session is active.
  134. /// </summary>
  135. public class IsSessionActiveViewModel : ServicesViewModel
  136. {
  137. public string SignOnSessionId { get; set; }
  138. /// <summary>
  139. /// Checks if an active session is associated with the UserSessionId parameter.
  140. /// </summary>
  141. /// <returns>UserAccount if a session exists and request could be verified, otherwise null.</returns>
  142. public IUserSession CheckSession()
  143. {
  144. if (VerifyClient())
  145. {
  146. if (!string.IsNullOrEmpty(SignOnSessionId))
  147. {
  148. Guid guid = new Guid(SignOnSessionId);
  149. IUserSession session = Data.UserSessions.GetUserSession(guid);
  150. if (session == null)
  151. throw new Exception("SignOnSession not found in database.");
  152. return session;
  153. }
  154. else
  155. {
  156. throw new Exception("SignOnSessionId not provided.");
  157. }
  158. }
  159. return null;
  160. }
  161. public override string SignatureContent
  162. {
  163. get { return ClientCode + "|" + Timestamp + "|" + SignOnSessionId; }
  164. }
  165. }
  166. }