]> git.mxchange.org Git - quix0rs-gnu-social.git/commitdiff
Use locally cached Salmon keys for profiles
authorMikael Nordfeldth <mmn@hethane.se>
Sat, 31 May 2014 10:51:51 +0000 (12:51 +0200)
committerMikael Nordfeldth <mmn@hethane.se>
Sat, 31 May 2014 10:51:51 +0000 (12:51 +0200)
Please note that we're not yet actually caching them ourselves.

plugins/OStatus/lib/magicenvelope.php
plugins/OStatus/lib/salmonaction.php

index 9990053904c495d6b6fdc0b380c9ce90939b8b7d..1a4abb5e1bc50c6f22e923d8ccbce24f6eb04b2b 100644 (file)
@@ -59,12 +59,33 @@ class MagicEnvelope
         }
     }
 
+    /**
+     * Retrieve Salmon keypair first by checking local database, but
+     * if it's not found, attempt discovery if it has been requested.
+     *
+     * @param Profile $profile      The profile we're looking up keys for.
+     * @param boolean $discovery    Network discovery if no local cache?
+     */
+    public function getKeyPair(Profile $profile, $discovery=false) {
+        $magicsig = Magicsig::getKV('user_id', $profile->id);
+        if ($discovery && !$magicsig instanceof Magicsig) {
+            $signer_uri = $profile->getUri();
+            if (empty($signer_uri)) {
+                throw new ServerException(sprintf('Profile missing URI (id==%d)', $profile->id));
+            }
+            $magicsig = $this->discoverKeyPair($signer_uri);
+        } elseif (!$magicsig instanceof Magicsig) { // No discovery request, so we'll give up.
+            throw new ServerException(sprintf('No public key found for profile (id==%d)', $profile->id));
+        }
+        return $magicsig;
+    }
+
     /**
      * Get the Salmon keypair from a URI, uses XRD Discovery etc.
      *
      * @return Magicsig with loaded keypair
      */
-    public function getKeyPair($signer_uri)
+    public function discoverKeyPair($signer_uri)
     {
         $disco = new Discovery();
 
@@ -228,9 +249,11 @@ class MagicEnvelope
      *
      * Details of failure conditions are dumped to output log and not exposed to caller.
      *
+     * @param Profile $profile optional profile used to get locally cached public signature key.
+     *
      * @return boolean
      */
-    public function verify()
+    public function verify(Profile $profile=null)
     {
         if ($this->alg != 'RSA-SHA256') {
             common_log(LOG_DEBUG, "Salmon error: bad algorithm");
@@ -243,8 +266,12 @@ class MagicEnvelope
         }
 
         try {
-            $signer_uri = $this->getAuthorUri();
-            $magicsig = $this->getKeyPair($signer_uri);
+            if ($profile instanceof Profile) {
+                $magicsig = $this->getKeyPair($profile, true);    // Do discovery too if necessary
+            } else {
+                $signer_uri = $this->getAuthorUri();
+                $magicsig = $this->discoverKeyPair($signer_uri);
+            }
         } catch (Exception $e) {
             common_log(LOG_DEBUG, "Salmon error: ".$e->getMessage());
             return false;
index 4366822ffdb3c1f8695a44d902c76b1dc3c0a0b3..6e44ff2eb606bd219a41967b522c0673089a367f 100644 (file)
@@ -28,6 +28,8 @@ class SalmonAction extends Action
 {
     protected $needPost = true;
 
+    protected $verified = false;
+
     var $xml      = null;
     var $activity = null;
     var $target   = null;
@@ -45,15 +47,24 @@ class SalmonAction extends Action
 
         $envxml = file_get_contents('php://input');
         $magic_env = new MagicEnvelope($envxml);   // parse incoming XML as a MagicEnvelope
-        if (!$magic_env->verify()) {
+
+        $entry = $magic_env->getPayload();  // Not cryptographically verified yet!
+        $this->activity = new Activity($entry->documentElement);
+
+        try {
+            $profile = Profile::fromUri($this->activity->actor->id);
+            $this->verified = $magic_env->verify($profile);
+        } catch (UnknownUriException $e) {
+            // If we don't know the profile, perform some discovery instead
+            $this->verified = $magic_env->verify();
+        }
+
+        if (!$this->verified) {
             common_log(LOG_DEBUG, "Salmon signature verification failed.");
             // TRANS: Client error.
             $this->clientError(_m('Salmon signature verification failed.'));
         }
 
-        $entry = $magic_env->getPayload();
-
-        $this->activity = new Activity($entry->documentElement);
         return true;
     }