changeset 26:ee97bc0e08fa

Some default OPs, along with some tweaks to keep them happy
author Adam Chlipala <adam@chlipala.net>
date Sun, 23 Jan 2011 14:57:49 -0500 (2011-01-23)
parents c560ec5bf514
children f129ddee75f3
files src/c/openid.c src/ur/openid.ur src/ur/openidUser.ur src/ur/openidUser.urs
diffstat 4 files changed, 142 insertions(+), 33 deletions(-) [+]
line wrap: on
line diff
--- a/src/c/openid.c	Thu Jan 20 12:23:44 2011 -0500
+++ b/src/c/openid.c	Sun Jan 23 14:57:49 2011 -0500
@@ -10,13 +10,17 @@
 #include <curl/curl.h>
 #include <expat.h>
 
+#include <pthread.h>
+
 #include <openid.h>
 
 #define BUF_MAX 10240
 #define BUF_INIT 1024
 
-#define PRIME_LEN 64
-#define GENERATOR DH_GENERATOR_5
+#define DEFAULT_PRIME "DCF93A0B883972EC0E19989AC5A2CE310E1D37717E8D9571BB7623731866E61EF75A2E27898B057F9891C2E27A639C3F29B60814581CD3B2CA3986D2683705577D45C2E7E52DC81C7A171876E5CEA74B1448BFDFAF18828EFD2519F14E45E3826634AF1949E5B535CC829A483B8A76223E5D490A257F05BDFF16F2FB22C583AB"
+#define DEFAULT_GENERATOR "2"
+
+static BIGNUM *default_prime, *default_generator;
 
 uw_Basis_string uw_OpenidFfi_endpoint(uw_context ctx, uw_OpenidFfi_discovery d) {
   return d.endpoint;
@@ -26,9 +30,32 @@
   return d.localId;
 }
 
+static pthread_mutex_t *locks;
+
+static void locking_function(int mode, int n, const char *file, int line) {
+  if (mode & CRYPTO_LOCK)
+    pthread_mutex_lock(&locks[n]);
+  else
+    pthread_mutex_unlock(&locks[n]);
+}
+
+static unsigned long id_function() {
+  return pthread_self();
+}
+
 uw_unit uw_OpenidFfi_init(uw_context ctx) {
+  int nl = CRYPTO_num_locks(), i;
+  locks = malloc(sizeof(pthread_mutex_t) * nl);
+  for (i = 0; i < nl; ++i)
+    pthread_mutex_init(&locks[i], NULL);
+
+  CRYPTO_set_locking_callback(locking_function);
+  CRYPTO_set_id_callback(id_function);
   curl_global_init(CURL_GLOBAL_ALL);
 
+  BN_hex2bn(&default_prime, DEFAULT_PRIME);
+  BN_dec2bn(&default_generator, DEFAULT_GENERATOR);
+
   return uw_unit_v;
 }
 
@@ -44,14 +71,20 @@
   return r;
 }
 
+typedef enum { NONE, SERVICE, TYPE, MATCHED, URI } xrds_mode;
+
 typedef struct {
   uw_context ctx;
   uw_OpenidFfi_discovery *d;
+  xrds_mode mode;
 } endpoint;
 
 static void XMLCALL startElement(void *userData, const XML_Char *name, const XML_Char **atts) {
   endpoint *ep = userData;
 
+  if (!strncmp(name, "xrd:", 4))
+    name += 4;
+
   if (!strcmp(name, "link")) {
     const XML_Char **attp;
     int found = 0;
@@ -72,6 +105,54 @@
       }
     }
   }
+  else if (!strcmp(name, "Service"))
+    ep->mode = SERVICE;
+  else if (!strcmp(name, "Type")) {
+    if (ep->mode == SERVICE)
+      ep->mode = TYPE;
+  }
+  else if (!strcmp(name, "URI")) {
+    if (ep->mode == MATCHED)
+      ep->mode = URI;
+  }
+}
+
+static char server[] = "http://specs.openid.net/auth/2.0/server";
+static char signon[] = "http://specs.openid.net/auth/2.0/signon";
+
+static void XMLCALL cdata(void *userData, const XML_Char *s, int len) {
+  endpoint *ep = userData;
+
+  switch (ep->mode) {
+  case TYPE:
+    if ((len == sizeof(server)-1 && !memcmp(server, s, sizeof(server)-1))
+        || (len == sizeof(signon)-1 && !memcmp(signon, s, sizeof(signon)-1)))
+      ep->mode = MATCHED;
+    break;
+  case URI:
+    ep->d->endpoint = uw_malloc(ep->ctx, len+1);
+    memcpy(ep->d->endpoint, s, len);
+    ep->d->endpoint[len] = 0;
+    break;
+  default:
+    break;
+  }
+}
+
+static void XMLCALL endElement(void *userData, const XML_Char *name) {
+  endpoint *ep = userData;
+
+  if (!strncmp(name, "xrd:", 4))
+    name += 4;
+
+  if (!strcmp(name, "Service"))
+    ep->mode = NONE;
+  else if (!strcmp(name, "Type")) {
+    if (ep->mode != MATCHED)
+      ep->mode = SERVICE;
+  }
+  else if (!strcmp(name, "URI"))
+    ep->mode = MATCHED;
 }
 
 typedef struct {
@@ -93,7 +174,7 @@
   CURL *c = curl(ctx);
   curl_discovery_data cd = {};
   uw_OpenidFfi_discovery *dy = uw_malloc(ctx, sizeof(uw_OpenidFfi_discovery));
-  endpoint ep = {ctx, dy};
+  endpoint ep = {ctx, dy, NONE};
   CURLcode code;
 
   dy->endpoint = dy->localId = NULL;
@@ -112,8 +193,10 @@
   cd.parser = XML_ParserCreate(NULL);
   XML_SetUserData(cd.parser, &ep);
   uw_push_cleanup(ctx, (void (*)(void *))XML_ParserFree, cd.parser);
-  XML_SetStartElementHandler(cd.parser, startElement);
+  XML_SetElementHandler(cd.parser, startElement, endElement);
+  XML_SetCharacterDataHandler(cd.parser, cdata);
 
+  curl_easy_reset(c);
   curl_easy_setopt(c, CURLOPT_URL, id);
   curl_easy_setopt(c, CURLOPT_WRITEFUNCTION, write_discovery_data);
   curl_easy_setopt(c, CURLOPT_WRITEDATA, &cd);
@@ -143,20 +226,6 @@
       sprintf(hex, "%%%02X", (unsigned char)*s);
       uw_buffer_append(buf, hex, 3);
     }
-
-    /*switch (*s) {
-    case '=':
-      uw_buffer_append(buf, "%3D", 3);
-      break;
-    case '&':
-      uw_buffer_append(buf, "%26", 3);
-      break;
-    case '%':
-      uw_buffer_append(buf, "%25", 3);
-      break;
-    default:
-      uw_buffer_append(buf, s, 1);
-    }*/
   }
 }
 
@@ -200,13 +269,13 @@
 
   uw_buffer_append(inps, "", 1);
 
+  curl_easy_reset(c);
   curl_easy_setopt(c, CURLOPT_URL, url);
   curl_easy_setopt(c, CURLOPT_POSTFIELDS, inps->start);
   curl_easy_setopt(c, CURLOPT_WRITEFUNCTION, write_buffer_data);
   curl_easy_setopt(c, CURLOPT_WRITEDATA, buf);
 
   code = curl_easy_perform(c);
-
   uw_buffer_append(buf, "", 1);
 
   if (code) {
@@ -388,16 +457,18 @@
 
 static void free_DH(void *data, int will_retry) {
   DH *dh = data;
+  dh->p = NULL;
+  dh->g = NULL;
   DH_free(dh);
 }
 
 uw_OpenidFfi_dh uw_OpenidFfi_generate(uw_context ctx) {
   DH *dh = DH_new();
 
+  dh->p = default_prime;
+  dh->g = default_generator;
   uw_register_transactional(ctx, dh, NULL, NULL, free_DH);
 
-  DH_generate_parameters_ex(dh, PRIME_LEN, GENERATOR, NULL);
-
   if (DH_generate_key(dh) != 1)
     uw_error(ctx, FATAL, "Diffie-Hellman key generation failed");
 
--- a/src/ur/openid.ur	Thu Jan 20 12:23:44 2011 -0500
+++ b/src/ur/openid.ur	Sun Jan 23 14:57:49 2011 -0500
@@ -383,18 +383,18 @@
           | Some dy =>
             case r.Association of
                 Stateless =>
-                redirect (bless (dy ^ "?openid.ns=http://specs.openid.net/auth/2.0&openid.mode=checkid_setup&openid.claimed_id="
-                                 ^ eatFragment r.Identifier
-                                 ^ "&openid.identity=http://specs.openid.net/auth/2.0/identifier_select&openid.return_to="
-                                 ^ show (effectfulUrl returnTo) ^ realmString))
+                redirect (bless (dy ^ "?openid.ns=http://specs.openid.net/auth/2.0&openid.mode=checkid_setup"
+                                 ^ "&openid.claimed_id=http://specs.openid.net/auth/2.0/identifier_select"
+                                 ^ "&openid.identity=http://specs.openid.net/auth/2.0/identifier_select&openid.assoc_handle="
+                                 ^ "&openid.return_to=" ^ show (effectfulUrl returnTo) ^ realmString))
               | Stateful ar =>
                 assoc <- association ar.AssociationType ar.AssociationSessionType dy;
                 case assoc of
                     AssError msg => return ("Association failure: " ^ msg)
                   | AssAlternate _ => return "Association failure: server didn't accept its own alternate association modes"
                   | Association assoc =>
-                    redirect (bless (dy ^ "?openid.ns=http://specs.openid.net/auth/2.0&openid.mode=checkid_setup&openid.claimed_id="
-                                     ^ eatFragment r.Identifier
+                    redirect (bless (dy ^ "?openid.ns=http://specs.openid.net/auth/2.0&openid.mode=checkid_setup"
+                                     ^ "&openid.claimed_id=http://specs.openid.net/auth/2.0/identifier_select"
                                      ^ "&openid.identity=http://specs.openid.net/auth/2.0/identifier_select&openid.assoc_handle="
                                      ^ assoc.Handle ^ "&openid.return_to=" ^ show (effectfulUrl returnTo) ^ realmString))
     end
--- a/src/ur/openidUser.ur	Thu Jan 20 12:23:44 2011 -0500
+++ b/src/ur/openidUser.ur	Sun Jan 23 14:57:49 2011 -0500
@@ -1,3 +1,10 @@
+style provider
+
+style aol
+style google
+style myspace
+style yahoo
+
 functor Make(M: sig
                  con cols :: {Type}
                  constraint [Id] ~ cols
@@ -251,12 +258,33 @@
                     error <xml>Login with your identity provider failed: {[msg]}</xml>
 
             fun signup after =
-                wrap "Account Signup" <xml>
-                  <form>
-                    OpenID Identifier: <textbox{#Identifier}/><br/>
-                    <submit value="Sign Up" action={doSignup after}/>
-                  </form>
-                </xml>
+                let
+                    fun fixed cls label url =
+                        let
+                            fun doFixedButton () =
+                                doSignup after {Identifier = url}
+                        in
+                            <xml><form class={provider}>
+                              <submit class={cls} value={label} action={doFixedButton}/>
+                            </form></xml>
+                        end
+                in
+                    wrap "Account Signup" <xml>
+                      <p>This web site uses the <b><a href="http://openid.net/">OpenID</a></b> standard, which lets you log in using your account from one of several popular web sites, without revealing your password to us.</p>
+
+                      <p>You may click one of these buttons to choose to use your account from that site:</p>
+                      {fixed aol "AOL" "https://openid.aol.com/"}
+                      {fixed google "Google" "https://www.google.com/accounts/o8/id"}
+                      {fixed myspace "Myspace" "https://www.myspace.com/openid"}
+                      {fixed yahoo "Yahoo!" "https://me.yahoo.com/"}
+
+                      <p>Visitors familiar with the details of OpenID may also enter their own identifiers:</p>
+                      <form>
+                        OpenID Identifier: <textbox{#Identifier}/><br/>
+                        <submit value="Sign Up" action={doSignup after}/>
+                      </form>
+                    </xml>
+                end
         in
             cur <- current;
             here <- currentUrl;
--- a/src/ur/openidUser.urs	Thu Jan 20 12:23:44 2011 -0500
+++ b/src/ur/openidUser.urs	Sun Jan 23 14:57:49 2011 -0500
@@ -87,3 +87,13 @@
      * which will be a log out or sign up link. *)
 
 end
+
+(* Functor outputs will contain buttons specialized to particular well-known
+ * OpenID providers.  Use these CSS classes to style those buttons. *)
+style aol
+style google
+style myspace
+style yahoo
+
+(* This style is used by forms containing the above buttons. *)
+style provider