diff --git a/authority/server/db.js b/authority/server/db.js
index b59a2564d852e83641db8a7e6c6686c00b1760ba..4d362888e67f938fedc6eb608b8d12e761c89e55 100644
--- a/authority/server/db.js
+++ b/authority/server/db.js
@@ -277,3 +277,19 @@ exports.pubkeysForEmail = function(identity, cb) {
                cb(keys);
              });
 };
+
+
+// FIXME: I'm not sure I'm using this data model properly
+exports.removeEmail = function(authenticated_email, email, cb) {
+    // figure out the user, and remove Email only from addressed
+    // linked to the authenticated email address
+    emailToUserID(authenticated_email, function(user_id) {
+        executeTransaction([
+            [ "delete from emails where emails.address = ? and user = ?", [ email,user_id ] ] ,
+            [ "delete from keys where email in (select address from emails where emails.address = ? and user = ?)", [ email,user_id ] ],
+        ], function (error) {
+            if (error) cb(error);
+            else cb();
+        });
+    });
+};
\ No newline at end of file
diff --git a/authority/server/wsapi.js b/authority/server/wsapi.js
index aa234fda1fcd243b8ec580dc05445e75c24fcb0d..0bfb38b9eea745ff670d4e8766aa67a361beef3a 100644
--- a/authority/server/wsapi.js
+++ b/authority/server/wsapi.js
@@ -117,6 +117,8 @@ exports.authenticate_user = function(req, resp) {
   });
 };
 
+// need CSRF protection
+
 exports.add_email = function (req, resp) {
   var urlobj = url.parse(req.url, true);
   var getArgs = urlobj.query;
@@ -145,6 +147,27 @@ exports.add_email = function (req, resp) {
   }
 };
 
+exports.remove_email = function(req, resp) {
+    // this should really be POST, but for now I'm having trouble seeing
+    // how to get POST args properly, so it's a GET (Ben).
+    // hmmm, I really want express or some other web framework!
+    var urlobj = url.parse(req.url, true);
+    var getArgs = urlobj.query;
+    
+    if (!checkParams(getArgs, resp, [ "email"])) return;
+    if (!checkAuthed(req, resp)) return;
+    
+    logRequest("remove_email", getArgs);
+    
+    db.removeEmail(req.session.authenticatedUser, getArgs.email, function(error) {
+        if (error) {
+            console.log("error removing email " + getArgs.email);
+            httputils.badRequest(resp, error.toString());
+        } else {
+            httputils.jsonResponse(resp, true);            
+        }});
+};
+
 exports.set_key = function (req, resp) {
   var urlobj = url.parse(req.url, true);
   var getArgs = urlobj.query;
diff --git a/authority/static/dialog/index.html b/authority/static/dialog/index.html
index 91e0202e29c97784aac34245a3b4ae2b0b5eef17..62abcaf117792047a48dee8eb37b891e323e9704 100644
--- a/authority/static/dialog/index.html
+++ b/authority/static/dialog/index.html
@@ -81,6 +81,12 @@
     </div>
   </div>
 </div>
+<div id="forgot_password_dialog" class="dialog">
+  <div class="content">
+    <p class="prompt"><b>Forgot your password?</b>  No problem, enter your email, and we'll help you reset it:</p>
+    <div class="input"> <input type="text"></input></div>
+  </div>
+</div>
 <div id="sign_in_dialog" class="dialog">
   <div class="content">
     <p class="prompt">What email address would you like to use to log into <span class="sitename bad"></span>?</p>
diff --git a/authority/static/dialog/style.css b/authority/static/dialog/style.css
index 7830922fbd2f634e59a22500d188bbb76410a48c..c3d02b0e4783d909f8b71d7744a88250539cadbc 100644
--- a/authority/static/dialog/style.css
+++ b/authority/static/dialog/style.css
@@ -146,6 +146,10 @@ button.disabled {
     text-align: center;
 }
 
+#forgot_password_dialog {
+    display: inline;
+}
+
 div.input > * {
     float: left;
 }
@@ -185,6 +189,16 @@ div.input input {
     padding: .4em;
 }
 
+#forgot_password_dialog input {
+    width: 80%;
+    margin: 1em 10% 0 10%;
+    font-size: 1.5em;
+    padding: 1em .5em 1em .5em;
+    -moz-border-radius: 4px;
+    -webkit-border-radius: 4px;
+    border-radius: 4px 4px 4px 4px;
+}
+
 a {
     color: rgb(65, 126, 208);
     text-decoration: none;
@@ -209,7 +223,7 @@ div.subtle {
 div.dialog div.attention, div.dialog div.attention_awesome, div.dialog div.attention_lame  {
     -moz-border-radius: 4px;
     -webkit-border-radius: 4px;
-    border-radius: 4px;
+    border-radius: 4px 4px 4px 4px;
     width: 400px;
     padding:13px;
     margin: auto;
diff --git a/authority/static/dialog/underscore-min.js b/authority/static/dialog/underscore-min.js
new file mode 100644
index 0000000000000000000000000000000000000000..f502cf9f6a03e9fba4530d69ddb69e3be62ee87d
--- /dev/null
+++ b/authority/static/dialog/underscore-min.js
@@ -0,0 +1,26 @@
+// Underscore.js 1.1.6
+// (c) 2011 Jeremy Ashkenas, DocumentCloud Inc.
+// Underscore is freely distributable under the MIT license.
+// Portions of Underscore are inspired or borrowed from Prototype,
+// Oliver Steele's Functional, and John Resig's Micro-Templating.
+// For all details and documentation:
+// http://documentcloud.github.com/underscore
+(function(){var p=this,C=p._,m={},i=Array.prototype,n=Object.prototype,f=i.slice,D=i.unshift,E=n.toString,l=n.hasOwnProperty,s=i.forEach,t=i.map,u=i.reduce,v=i.reduceRight,w=i.filter,x=i.every,y=i.some,o=i.indexOf,z=i.lastIndexOf;n=Array.isArray;var F=Object.keys,q=Function.prototype.bind,b=function(a){return new j(a)};typeof module!=="undefined"&&module.exports?(module.exports=b,b._=b):p._=b;b.VERSION="1.1.6";var h=b.each=b.forEach=function(a,c,d){if(a!=null)if(s&&a.forEach===s)a.forEach(c,d);else if(b.isNumber(a.length))for(var e=
+0,k=a.length;e<k;e++){if(c.call(d,a[e],e,a)===m)break}else for(e in a)if(l.call(a,e)&&c.call(d,a[e],e,a)===m)break};b.map=function(a,c,b){var e=[];if(a==null)return e;if(t&&a.map===t)return a.map(c,b);h(a,function(a,g,G){e[e.length]=c.call(b,a,g,G)});return e};b.reduce=b.foldl=b.inject=function(a,c,d,e){var k=d!==void 0;a==null&&(a=[]);if(u&&a.reduce===u)return e&&(c=b.bind(c,e)),k?a.reduce(c,d):a.reduce(c);h(a,function(a,b,f){!k&&b===0?(d=a,k=!0):d=c.call(e,d,a,b,f)});if(!k)throw new TypeError("Reduce of empty array with no initial value");
+return d};b.reduceRight=b.foldr=function(a,c,d,e){a==null&&(a=[]);if(v&&a.reduceRight===v)return e&&(c=b.bind(c,e)),d!==void 0?a.reduceRight(c,d):a.reduceRight(c);a=(b.isArray(a)?a.slice():b.toArray(a)).reverse();return b.reduce(a,c,d,e)};b.find=b.detect=function(a,c,b){var e;A(a,function(a,g,f){if(c.call(b,a,g,f))return e=a,!0});return e};b.filter=b.select=function(a,c,b){var e=[];if(a==null)return e;if(w&&a.filter===w)return a.filter(c,b);h(a,function(a,g,f){c.call(b,a,g,f)&&(e[e.length]=a)});return e};
+b.reject=function(a,c,b){var e=[];if(a==null)return e;h(a,function(a,g,f){c.call(b,a,g,f)||(e[e.length]=a)});return e};b.every=b.all=function(a,c,b){var e=!0;if(a==null)return e;if(x&&a.every===x)return a.every(c,b);h(a,function(a,g,f){if(!(e=e&&c.call(b,a,g,f)))return m});return e};var A=b.some=b.any=function(a,c,d){c||(c=b.identity);var e=!1;if(a==null)return e;if(y&&a.some===y)return a.some(c,d);h(a,function(a,b,f){if(e=c.call(d,a,b,f))return m});return e};b.include=b.contains=function(a,c){var b=
+!1;if(a==null)return b;if(o&&a.indexOf===o)return a.indexOf(c)!=-1;A(a,function(a){if(b=a===c)return!0});return b};b.invoke=function(a,c){var d=f.call(arguments,2);return b.map(a,function(a){return(c.call?c||a:a[c]).apply(a,d)})};b.pluck=function(a,c){return b.map(a,function(a){return a[c]})};b.max=function(a,c,d){if(!c&&b.isArray(a))return Math.max.apply(Math,a);var e={computed:-Infinity};h(a,function(a,b,f){b=c?c.call(d,a,b,f):a;b>=e.computed&&(e={value:a,computed:b})});return e.value};b.min=function(a,
+c,d){if(!c&&b.isArray(a))return Math.min.apply(Math,a);var e={computed:Infinity};h(a,function(a,b,f){b=c?c.call(d,a,b,f):a;b<e.computed&&(e={value:a,computed:b})});return e.value};b.sortBy=function(a,c,d){return b.pluck(b.map(a,function(a,b,f){return{value:a,criteria:c.call(d,a,b,f)}}).sort(function(a,b){var c=a.criteria,d=b.criteria;return c<d?-1:c>d?1:0}),"value")};b.sortedIndex=function(a,c,d){d||(d=b.identity);for(var e=0,f=a.length;e<f;){var g=e+f>>1;d(a[g])<d(c)?e=g+1:f=g}return e};b.toArray=
+function(a){if(!a)return[];if(a.toArray)return a.toArray();if(b.isArray(a))return a;if(b.isArguments(a))return f.call(a);return b.values(a)};b.size=function(a){return b.toArray(a).length};b.first=b.head=function(a,b,d){return b!=null&&!d?f.call(a,0,b):a[0]};b.rest=b.tail=function(a,b,d){return f.call(a,b==null||d?1:b)};b.last=function(a){return a[a.length-1]};b.compact=function(a){return b.filter(a,function(a){return!!a})};b.flatten=function(a){return b.reduce(a,function(a,d){if(b.isArray(d))return a.concat(b.flatten(d));
+a[a.length]=d;return a},[])};b.without=function(a){var c=f.call(arguments,1);return b.filter(a,function(a){return!b.include(c,a)})};b.uniq=b.unique=function(a,c){return b.reduce(a,function(a,e,f){if(0==f||(c===!0?b.last(a)!=e:!b.include(a,e)))a[a.length]=e;return a},[])};b.intersect=function(a){var c=f.call(arguments,1);return b.filter(b.uniq(a),function(a){return b.every(c,function(c){return b.indexOf(c,a)>=0})})};b.zip=function(){for(var a=f.call(arguments),c=b.max(b.pluck(a,"length")),d=Array(c),
+e=0;e<c;e++)d[e]=b.pluck(a,""+e);return d};b.indexOf=function(a,c,d){if(a==null)return-1;var e;if(d)return d=b.sortedIndex(a,c),a[d]===c?d:-1;if(o&&a.indexOf===o)return a.indexOf(c);d=0;for(e=a.length;d<e;d++)if(a[d]===c)return d;return-1};b.lastIndexOf=function(a,b){if(a==null)return-1;if(z&&a.lastIndexOf===z)return a.lastIndexOf(b);for(var d=a.length;d--;)if(a[d]===b)return d;return-1};b.range=function(a,b,d){arguments.length<=1&&(b=a||0,a=0);d=arguments[2]||1;for(var e=Math.max(Math.ceil((b-a)/
+d),0),f=0,g=Array(e);f<e;)g[f++]=a,a+=d;return g};b.bind=function(a,b){if(a.bind===q&&q)return q.apply(a,f.call(arguments,1));var d=f.call(arguments,2);return function(){return a.apply(b,d.concat(f.call(arguments)))}};b.bindAll=function(a){var c=f.call(arguments,1);c.length==0&&(c=b.functions(a));h(c,function(c){a[c]=b.bind(a[c],a)});return a};b.memoize=function(a,c){var d={};c||(c=b.identity);return function(){var b=c.apply(this,arguments);return l.call(d,b)?d[b]:d[b]=a.apply(this,arguments)}};b.delay=
+function(a,b){var d=f.call(arguments,2);return setTimeout(function(){return a.apply(a,d)},b)};b.defer=function(a){return b.delay.apply(b,[a,1].concat(f.call(arguments,1)))};var B=function(a,b,d){var e;return function(){var f=this,g=arguments,h=function(){e=null;a.apply(f,g)};d&&clearTimeout(e);if(d||!e)e=setTimeout(h,b)}};b.throttle=function(a,b){return B(a,b,!1)};b.debounce=function(a,b){return B(a,b,!0)};b.once=function(a){var b=!1,d;return function(){if(b)return d;b=!0;return d=a.apply(this,arguments)}};
+b.wrap=function(a,b){return function(){var d=[a].concat(f.call(arguments));return b.apply(this,d)}};b.compose=function(){var a=f.call(arguments);return function(){for(var b=f.call(arguments),d=a.length-1;d>=0;d--)b=[a[d].apply(this,b)];return b[0]}};b.after=function(a,b){return function(){if(--a<1)return b.apply(this,arguments)}};b.keys=F||function(a){if(a!==Object(a))throw new TypeError("Invalid object");var b=[],d;for(d in a)l.call(a,d)&&(b[b.length]=d);return b};b.values=function(a){return b.map(a,
+b.identity)};b.functions=b.methods=function(a){return b.filter(b.keys(a),function(c){return b.isFunction(a[c])}).sort()};b.extend=function(a){h(f.call(arguments,1),function(b){for(var d in b)b[d]!==void 0&&(a[d]=b[d])});return a};b.defaults=function(a){h(f.call(arguments,1),function(b){for(var d in b)a[d]==null&&(a[d]=b[d])});return a};b.clone=function(a){return b.isArray(a)?a.slice():b.extend({},a)};b.tap=function(a,b){b(a);return a};b.isEqual=function(a,c){if(a===c)return!0;var d=typeof a;if(d!=
+typeof c)return!1;if(a==c)return!0;if(!a&&c||a&&!c)return!1;if(a._chain)a=a._wrapped;if(c._chain)c=c._wrapped;if(a.isEqual)return a.isEqual(c);if(b.isDate(a)&&b.isDate(c))return a.getTime()===c.getTime();if(b.isNaN(a)&&b.isNaN(c))return!1;if(b.isRegExp(a)&&b.isRegExp(c))return a.source===c.source&&a.global===c.global&&a.ignoreCase===c.ignoreCase&&a.multiline===c.multiline;if(d!=="object")return!1;if(a.length&&a.length!==c.length)return!1;d=b.keys(a);var e=b.keys(c);if(d.length!=e.length)return!1;
+for(var f in a)if(!(f in c)||!b.isEqual(a[f],c[f]))return!1;return!0};b.isEmpty=function(a){if(b.isArray(a)||b.isString(a))return a.length===0;for(var c in a)if(l.call(a,c))return!1;return!0};b.isElement=function(a){return!!(a&&a.nodeType==1)};b.isArray=n||function(a){return E.call(a)==="[object Array]"};b.isArguments=function(a){return!(!a||!l.call(a,"callee"))};b.isFunction=function(a){return!(!a||!a.constructor||!a.call||!a.apply)};b.isString=function(a){return!!(a===""||a&&a.charCodeAt&&a.substr)};
+b.isNumber=function(a){return!!(a===0||a&&a.toExponential&&a.toFixed)};b.isNaN=function(a){return a!==a};b.isBoolean=function(a){return a===!0||a===!1};b.isDate=function(a){return!(!a||!a.getTimezoneOffset||!a.setUTCFullYear)};b.isRegExp=function(a){return!(!a||!a.test||!a.exec||!(a.ignoreCase||a.ignoreCase===!1))};b.isNull=function(a){return a===null};b.isUndefined=function(a){return a===void 0};b.noConflict=function(){p._=C;return this};b.identity=function(a){return a};b.times=function(a,b,d){for(var e=
+0;e<a;e++)b.call(d,e)};b.mixin=function(a){h(b.functions(a),function(c){H(c,b[c]=a[c])})};var I=0;b.uniqueId=function(a){var b=I++;return a?a+b:b};b.templateSettings={evaluate:/<%([\s\S]+?)%>/g,interpolate:/<%=([\s\S]+?)%>/g};b.template=function(a,c){var d=b.templateSettings;d="var __p=[],print=function(){__p.push.apply(__p,arguments);};with(obj||{}){__p.push('"+a.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(d.interpolate,function(a,b){return"',"+b.replace(/\\'/g,"'")+",'"}).replace(d.evaluate||
+null,function(a,b){return"');"+b.replace(/\\'/g,"'").replace(/[\r\n\t]/g," ")+"__p.push('"}).replace(/\r/g,"\\r").replace(/\n/g,"\\n").replace(/\t/g,"\\t")+"');}return __p.join('');";d=new Function("obj",d);return c?d(c):d};var j=function(a){this._wrapped=a};b.prototype=j.prototype;var r=function(a,c){return c?b(a).chain():a},H=function(a,c){j.prototype[a]=function(){var a=f.call(arguments);D.call(a,this._wrapped);return r(c.apply(b,a),this._chain)}};b.mixin(b);h(["pop","push","reverse","shift","sort",
+"splice","unshift"],function(a){var b=i[a];j.prototype[a]=function(){b.apply(this._wrapped,arguments);return r(this._wrapped,this._chain)}});h(["concat","join","slice"],function(a){var b=i[a];j.prototype[a]=function(){return r(b.apply(this._wrapped,arguments),this._chain)}});j.prototype.chain=function(){this._chain=!0;return this};j.prototype.value=function(){return this._wrapped}})();
diff --git a/authority/static/manage.html b/authority/static/manage.html
index 78d65b3857af202cdfb462606779fb657e100ab7..0a3be84039b0d4d379a393790b711d9b96159f21 100644
--- a/authority/static/manage.html
+++ b/authority/static/manage.html
@@ -37,6 +37,7 @@
 </div>
 </body>
 <script src="dialog/jquery-min.js"></script>
+<script src="dialog/underscore-min.js"></script>
 <script>
 function display_saved_ids()
 {
@@ -46,36 +47,38 @@ function display_saved_ids()
   }
 
   $("#emailList").empty();
-  for (var e in emails)
-  {
-    if (emails.hasOwnProperty(e))
-    {
+  _(emails).each(function(data, e) {
       var block = $("<div>").addClass("emailblock");
       var label = $("<div>").addClass("email").text(e);
       var meta = $("<div>").addClass("meta");
 
-      var data = emails[e];
-      
-      var priv = $("<div class='keyblock'>").text(data.priv);
+      /* 
+        var priv = $("<div class='keyblock'>").text(data.priv);
+        priv.hide();
+       */
+
       var pub = $("<div class='keyblock'>").text(data.pub);
-      priv.hide();
       pub.hide();
       var linkblock = $("<div>");
       var puba = $("<a>").text("[show public key]");
-      var priva = $("<a>").text("[show private key]");
+      // var priva = $("<a>").text("[show private key]");
       puba.click(function() {pub.show()});
-      priva.click(function() {priv.show()});
+      // priva.click(function() {priv.show()});
       linkblock.append(puba);
-      linkblock.append(" / ");
-      linkblock.append(priva);
+      // linkblock.append(" / ");
+      // linkblock.append(priva);
       
-      var deauth = $("<button>").text("Sign Out");
+      var deauth = $("<button>").text("Forget this Email");
       meta.append(deauth);
       deauth.click(function() {
         var t = JSON.parse(window.localStorage.emails);
         delete t[e];
         window.localStorage.emails = JSON.stringify(t);
-        display_saved_ids();
+        // remove email from server
+        $.get("/wsapi/remove_email", {"email" : e}, function(response) {
+                    alert("response is : " +response);
+                    display_saved_ids();
+                    });
       });
       
       var d = new Date(data.created);
@@ -86,12 +89,11 @@ function display_saved_ids()
                   
       block.append(label);
       block.append(meta);
-      block.append(priv);
+      // block.append(priv);
       block.append(pub);
       
       $("#emailList").append(block);
-    }
-  }
+  });
 }
 </script>
 
diff --git a/verifier/server/idassertion.js b/verifier/server/idassertion.js
index ffbed8bc5b1c29a2295557db6a6feece3b9c0c00..4845f9805c59eabb1ced138cb65d64fd52061aae 100644
--- a/verifier/server/idassertion.js
+++ b/verifier/server/idassertion.js
@@ -291,7 +291,9 @@ IDAssertion.prototype  =
             if (token.verify(pubKey)) {
               // success!
               console.log("Token for " +payload.email + " verified successfully.");
-              onSuccess(true);
+
+              // send back all the verified data
+              onSuccess(payload);
               return;
             }
           } catch(e) {
diff --git a/verifier/server/run.js b/verifier/server/run.js
index 2696fd5456650c6d7c06f2ed99e257b01722d149..cc9e31fb8234ecfe307ddea152d8345573ef8f31 100644
--- a/verifier/server/run.js
+++ b/verifier/server/run.js
@@ -48,8 +48,15 @@ exports.handler = function(req, resp, serveFile) {
             var assertionObj = new idassertion.IDAssertion(assertion);
             assertionObj.verify(
                 audience,
-                function(successObj) {
-                    httputils.jsonResponse(resp, {status:"okay"});
+                function(payload) {
+                    result = {
+                        status : "okay",
+                        email : payload.email,
+                        audience : payload.audience,
+                        "valid-until" : payload["valid-until"],
+                        issuer : payload.issuer
+                    };
+                    httputils.jsonResponse(resp, result);
                 },
                 function(errorObj) {
                     httputils.jsonResponse(resp, {status:"failure", reason:errorObj});