Skip to content

Instantly share code, notes, and snippets.

@andrewconnell
Created December 17, 2014 15:38
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save andrewconnell/d56396036c6d6033c976 to your computer and use it in GitHub Desktop.
Save andrewconnell/d56396036c6d6033c976 to your computer and use it in GitHub Desktop.
Azure AD & ASP.NET MVC - Walk-Through Implementing ADAL & OWIN - EfAdalTokenCache.cs
using System;
using System.Collections.Generic;
using System.Data.Entity;
using System.Linq;
using System.Web;
using System.Web.Caching;
using Microsoft.IdentityModel.Clients.ActiveDirectory;
using SampleMvcAzAuth.Data;
using SampleMvcAzAuth.Models;
namespace SampleMvcAzAuth.Utils {
public class EfAdalTokenCache : TokenCache {
private TokenCacheDataContext db = new TokenCacheDataContext();
string User;
private PerUserWebCache Cache;
// constructor
public EfAdalTokenCache(string user) {
// associate the cache to the current user of the web app
User = user;
this.AfterAccess = AfterAccessNotification;
this.BeforeAccess = BeforeAccessNotification;
this.BeforeWrite = BeforeWriteNotification;
// look up the entry in the DB
Cache = db.PerUserCacheList.FirstOrDefault(c => c.WebUserUniqueId== User);
// place the entry in memory
this.Deserialize((Cache == null) ? null : Cache.CacheBits);
}
// clean up the DB
public override void Clear() {
base.Clear();
foreach (var cacheEntry in db.PerUserCacheList)
db.PerUserCacheList.Remove(cacheEntry);
db.SaveChanges();
}
// Notification raised before ADAL accesses the cache.
// This is your chance to update the in-memory copy from the DB, if the in-memory version is stale
void BeforeAccessNotification(TokenCacheNotificationArgs args) {
if (Cache == null) {
// first time access
Cache = db.PerUserCacheList.FirstOrDefault(c => c.WebUserUniqueId == User);
} else { // retrieve last write from the DB
var status = from e in db.PerUserCacheList
where (e.WebUserUniqueId == User)
select new {
LastWrite = e.LastWrite
};
// if the in-memory copy is older than the persistent copy
if (status.First().LastWrite > Cache.LastWrite)
//// read from from storage, update in-memory copy
{
Cache = db.PerUserCacheList.FirstOrDefault(c => c.WebUserUniqueId == User);
}
}
this.Deserialize((Cache == null) ? null : Cache.CacheBits);
}
// Notification raised after ADAL accessed the cache.
// If the HasStateChanged flag is set, ADAL changed the content of the cache
void AfterAccessNotification(TokenCacheNotificationArgs args) {
// if state changed
if (this.HasStateChanged) {
Cache = new PerUserWebCache {
WebUserUniqueId = User,
CacheBits = this.Serialize(),
LastWrite = DateTime.Now
};
//// update the DB and the lastwrite
db.Entry(Cache).State = Cache.EntryId == 0 ? EntityState.Added : EntityState.Modified;
db.SaveChanges();
this.HasStateChanged = false;
}
}
void BeforeWriteNotification(TokenCacheNotificationArgs args) {
// if you want to ensure that no concurrent write take place, use this notification to place a lock on the entry
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment