. * * @category Plugin * @package GNUsocial * @author Diogo Cordeiro * @copyright 2018 Free Software Foundation http://fsf.org * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 * @link https://www.gnu.org/software/social/ */ if (!defined('GNUSOCIAL')) { exit(1); } /** * Inbox Request Handler * * @category Plugin * @package GNUsocial * @author Diogo Cordeiro * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 * @link http://www.gnu.org/software/social/ */ use GuzzleHttp\Psr7; use HttpSignatures\Context; class apInboxAction extends ManagedAction { protected $needLogin = false; protected $canPost = true; /** * Handle the Inbox request * * @author Diogo Cordeiro * @return void */ protected function handle() { if ($_SERVER['REQUEST_METHOD'] !== 'POST') { ActivityPubReturn::error('Only POST requests allowed.'); } common_debug('ActivityPub Inbox: Received a POST request.'); $data = file_get_contents('php://input'); common_debug('ActivityPub Inbox: Request contents: '.$data); $data = json_decode(file_get_contents('php://input'), true); if (!isset($data['actor'])) { ActivityPubReturn::error('Actor not found in the request.'); } $actor = ActivityPub_explorer::get_profile_from_url($data['actor']); $actor_public_key = new Activitypub_rsa(); $actor_public_key = $actor_public_key->ensure_public_key($actor); common_debug('ActivityPub Inbox: HTTP Signature: Validation will now start!'); $headers = $this->get_all_headers(); common_debug('ActivityPub Inbox: Request Headers: '.print_r($headers, true)); try { $res = HTTPSignature::parse($headers); common_debug('ActivityPub Inbox: Request Res: '.print_r($res, true)); } catch (HttpSignatureError $e) { common_debug('ActivityPub Inbox: HTTP Signature Error: '. $e->getMessage()); ActivityPubReturn::error('HTTP Signature Error: '. $e->getMessage()); } catch (Exception $e) { ActivityPubReturn::error($e->getMessage()); } /*if (HTTPSignature::verify($res, $actor_public_key, 'rsa') == FALSE) { common_debug('ActivityPub Inbox: Could not authorize request.'); ActivityPubReturn::error('Unauthorized.', 403); }*/ $context = new Context([ 'keys' => [$res['params']['keyId'] => $actor_public_key], 'algorithm' => $res['params']['algorithm'], 'headers' => $res['headers'], ]); $request = new Psr7\Request($_SERVER['REQUEST_METHOD'], (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] === 'on' ? "https" : "http") . "://$_SERVER[HTTP_HOST]$_SERVER[REQUEST_URI]", $headers); if ($context->verifier()->isValid($request) == false) { common_debug('ActivityPub Inbox: HTTP Signature: Unauthorized request.'); ActivityPubReturn::error('Unauthorized.', 403); } common_debug('ActivityPub Inbox: HTTP Signature: Authorized request. Will now start the inbox handler.'); try { new Activitypub_inbox_handler($data, $actor); ActivityPubReturn::answer(); } catch (Exception $e) { ActivityPubReturn::error($e->getMessage()); } } /** * Get all HTTP header key/values as an associative array for the current request. * * @author PHP Manual Contributed Notes * @return string[string] The HTTP header key/value pairs. */ private function get_all_headers() { $headers = []; foreach ($_SERVER as $name => $value) { if (substr($name, 0, 5) == 'HTTP_') { $headers[strtolower(str_replace(' ', '-', ucwords(strtolower(str_replace('_', ' ', substr($name, 5))))))] = $value; } } return $headers; } }