forked from DotNetOpenAuth/DotNetOpenAuth
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRelyingPartyApplicationDbStore.cs
94 lines (84 loc) · 3.53 KB
/
RelyingPartyApplicationDbStore.cs
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
93
94
//-----------------------------------------------------------------------
// <copyright file="RelyingPartyApplicationDbStore.cs" company="Outercurve Foundation">
// Copyright (c) Outercurve Foundation. All rights reserved.
// </copyright>
//-----------------------------------------------------------------------
namespace RelyingPartyLogic {
using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using DotNetOpenAuth;
using DotNetOpenAuth.Messaging.Bindings;
using DotNetOpenAuth.OpenId;
/// <summary>
/// A database-backed state store for OpenID relying parties.
/// </summary>
public class RelyingPartyApplicationDbStore : NonceDbStore, IOpenIdApplicationStore {
/// <summary>
/// Initializes a new instance of the <see cref="RelyingPartyApplicationDbStore"/> class.
/// </summary>
public RelyingPartyApplicationDbStore() {
}
#region ICryptoStore Members
public CryptoKey GetKey(string bucket, string handle) {
using (var dataContext = new TransactedDatabaseEntities(System.Data.IsolationLevel.ReadCommitted)) {
var associations = from assoc in dataContext.SymmetricCryptoKeys
where assoc.Bucket == bucket
where assoc.Handle == handle
where assoc.ExpirationUtc > DateTime.UtcNow
select assoc;
return associations.AsEnumerable()
.Select(assoc => new CryptoKey(assoc.Secret, assoc.ExpirationUtc.AsUtc()))
.FirstOrDefault();
}
}
public IEnumerable<KeyValuePair<string, CryptoKey>> GetKeys(string bucket) {
using (var dataContext = new TransactedDatabaseEntities(System.Data.IsolationLevel.ReadCommitted)) {
var relevantAssociations = from assoc in dataContext.SymmetricCryptoKeys
where assoc.Bucket == bucket
where assoc.ExpirationUtc > DateTime.UtcNow
orderby assoc.ExpirationUtc descending
select assoc;
var qualifyingAssociations = relevantAssociations.AsEnumerable()
.Select(assoc => new KeyValuePair<string, CryptoKey>(assoc.Handle, new CryptoKey(assoc.Secret, assoc.ExpirationUtc.AsUtc())));
return qualifyingAssociations.ToList(); // the data context is closing, so we must cache the result.
}
}
public void StoreKey(string bucket, string handle, CryptoKey key) {
using (var dataContext = new TransactedDatabaseEntities(System.Data.IsolationLevel.ReadCommitted)) {
var sharedAssociation = new SymmetricCryptoKey {
Bucket = bucket,
Handle = handle,
ExpirationUtc = key.ExpiresUtc,
Secret = key.Key,
};
dataContext.AddToSymmetricCryptoKeys(sharedAssociation);
}
}
public void RemoveKey(string bucket, string handle) {
using (var dataContext = new TransactedDatabaseEntities(System.Data.IsolationLevel.ReadCommitted)) {
var association = dataContext.SymmetricCryptoKeys.FirstOrDefault(a => a.Bucket == bucket && a.Handle == handle);
if (association != null) {
dataContext.DeleteObject(association);
} else {
}
}
}
#endregion
/// <summary>
/// Clears all expired associations from the store.
/// </summary>
/// <remarks>
/// If another algorithm is in place to periodically clear out expired associations,
/// this method call may be ignored.
/// This should be done frequently enough to avoid a memory leak, but sparingly enough
/// to not be a performance drain.
/// </remarks>
internal void ClearExpiredCryptoKeys() {
using (var dataContext = new TransactedDatabaseEntities(IsolationLevel.ReadCommitted)) {
dataContext.ClearExpiredCryptoKeys(dataContext.Transaction);
}
}
}
}