2010-01-07 21:19:21 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* StatusNet, the distributed open-source microblogging tool
|
|
|
|
*
|
|
|
|
* Authorize an OAuth request token
|
|
|
|
*
|
|
|
|
* PHP version 5
|
|
|
|
*
|
|
|
|
* LICENCE: This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* @category API
|
|
|
|
* @package StatusNet
|
|
|
|
* @author Zach Copley <zach@status.net>
|
|
|
|
* @copyright 2010 StatusNet, Inc.
|
|
|
|
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
|
|
|
|
* @link http://status.net/
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!defined('STATUSNET')) {
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2010-01-11 05:35:46 +00:00
|
|
|
require_once INSTALLDIR . '/lib/apioauthstore.php';
|
2010-01-07 21:19:21 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Authorize an OAuth request token
|
|
|
|
*
|
|
|
|
* @category API
|
|
|
|
* @package StatusNet
|
|
|
|
* @author Zach Copley <zach@status.net>
|
|
|
|
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
|
|
|
|
* @link http://status.net/
|
|
|
|
*/
|
|
|
|
|
|
|
|
class ApiOauthAuthorizeAction extends Action
|
|
|
|
{
|
2010-01-11 05:35:46 +00:00
|
|
|
var $oauth_token;
|
|
|
|
var $callback;
|
|
|
|
var $app;
|
|
|
|
var $nickname;
|
|
|
|
var $password;
|
|
|
|
var $store;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is this a read-only action?
|
|
|
|
*
|
|
|
|
* @return boolean false
|
|
|
|
*/
|
|
|
|
|
|
|
|
function isReadOnly($args)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function prepare($args)
|
|
|
|
{
|
|
|
|
parent::prepare($args);
|
|
|
|
|
|
|
|
common_debug(var_export($_REQUEST, true));
|
|
|
|
|
|
|
|
$this->nickname = $this->trimmed('nickname');
|
|
|
|
$this->password = $this->arg('password');
|
|
|
|
$this->oauth_token = $this->arg('oauth_token');
|
|
|
|
$this->callback = $this->arg('oauth_callback');
|
|
|
|
$this->store = new ApiStatusNetOAuthDataStore();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getApp()
|
|
|
|
{
|
|
|
|
// Look up the full req token
|
|
|
|
|
|
|
|
$req_token = $this->store->lookup_token(null,
|
|
|
|
'request',
|
|
|
|
$this->oauth_token);
|
|
|
|
|
|
|
|
if (empty($req_token)) {
|
|
|
|
|
|
|
|
common_debug("Couldn't find request token!");
|
|
|
|
|
|
|
|
$this->clientError(_('Bad request.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Look up the app
|
|
|
|
|
|
|
|
$app = new Oauth_application();
|
|
|
|
$app->consumer_key = $req_token->consumer_key;
|
|
|
|
$result = $app->find(true);
|
|
|
|
|
|
|
|
if (!empty($result)) {
|
|
|
|
$this->app = $app;
|
|
|
|
return true;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
common_debug("couldn't find the app!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle input, produce output
|
|
|
|
*
|
|
|
|
* Switches on request method; either shows the form or handles its input.
|
|
|
|
*
|
|
|
|
* @param array $args $_REQUEST data
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function handle($args)
|
|
|
|
{
|
|
|
|
parent::handle($args);
|
|
|
|
|
|
|
|
if ($_SERVER['REQUEST_METHOD'] == 'POST') {
|
|
|
|
|
|
|
|
$this->handlePost();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
2010-01-11 07:03:30 +00:00
|
|
|
// XXX: make better error messages
|
2010-01-11 05:35:46 +00:00
|
|
|
|
|
|
|
if (empty($this->oauth_token)) {
|
|
|
|
|
|
|
|
common_debug("No request token found.");
|
|
|
|
|
|
|
|
$this->clientError(_('Bad request.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$this->getApp()) {
|
|
|
|
$this->clientError(_('Bad request.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
common_debug("Requesting auth for app: $app->name.");
|
|
|
|
|
|
|
|
$this->showForm();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function handlePost()
|
|
|
|
{
|
2010-01-11 07:03:30 +00:00
|
|
|
// check session token for CSRF protection.
|
2010-01-11 05:35:46 +00:00
|
|
|
|
|
|
|
$token = $this->trimmed('token');
|
|
|
|
|
|
|
|
if (!$token || $token != common_session_token()) {
|
|
|
|
$this->showForm(_('There was a problem with your session token. '.
|
|
|
|
'Try again, please.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$this->getApp()) {
|
|
|
|
$this->clientError(_('Bad request.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check creds
|
|
|
|
|
2010-01-11 07:03:30 +00:00
|
|
|
$user = null;
|
|
|
|
|
2010-01-11 05:35:46 +00:00
|
|
|
if (!common_logged_in()) {
|
|
|
|
$user = common_check_user($this->nickname, $this->password);
|
|
|
|
if (empty($user)) {
|
|
|
|
$this->showForm(_("Invalid nickname / password!"));
|
|
|
|
return;
|
|
|
|
}
|
2010-01-11 07:03:30 +00:00
|
|
|
} else {
|
|
|
|
$user = common_current_user();
|
|
|
|
}
|
2010-01-11 05:35:46 +00:00
|
|
|
|
|
|
|
if ($this->arg('allow')) {
|
|
|
|
|
2010-01-11 07:03:30 +00:00
|
|
|
// mark the req token as authorized
|
|
|
|
|
2010-01-11 05:35:46 +00:00
|
|
|
$this->store->authorize_token($this->oauth_token);
|
|
|
|
|
2010-01-11 07:03:30 +00:00
|
|
|
// Check to see if there was a previous token associated
|
|
|
|
// with this user/app and kill it. If you're doing this you
|
|
|
|
// probably don't want any old tokens anyway.
|
|
|
|
|
|
|
|
$appUser = Oauth_application_user::getByKeys($user, $this->app);
|
|
|
|
|
|
|
|
if (!empty($appUser)) {
|
|
|
|
$result = $appUser->delete();
|
|
|
|
|
|
|
|
if (!$result) {
|
|
|
|
common_log_db_error($appUser, 'DELETE', __FILE__);
|
|
|
|
throw new ServerException(_('DB error deleting OAuth app user.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// associated the new req token with the user and the app
|
|
|
|
|
|
|
|
$appUser = new Oauth_application_user();
|
|
|
|
|
|
|
|
$appUser->profile_id = $user->id;
|
|
|
|
$appUser->application_id = $this->app->id;
|
|
|
|
$appUser->access_type = $this->app->access_type;
|
|
|
|
$appUser->token = $this->oauth_token;
|
|
|
|
$appUser->created = common_sql_now();
|
|
|
|
|
|
|
|
$result = $appUser->insert();
|
|
|
|
|
|
|
|
if (!$result) {
|
|
|
|
common_log_db_error($appUser, 'INSERT', __FILE__);
|
|
|
|
throw new ServerException(_('DB error inserting OAuth app user.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-01-11 05:35:46 +00:00
|
|
|
// if we have a callback redirect and provide the token
|
|
|
|
|
|
|
|
if (!empty($this->callback)) {
|
2010-01-11 07:03:30 +00:00
|
|
|
|
|
|
|
// XXX: Need better way to build this redirect url.
|
|
|
|
|
2010-01-11 05:35:46 +00:00
|
|
|
$target_url = $this->callback . '?oauth_token=' . $this->oauth_token;
|
|
|
|
common_redirect($target_url, 303);
|
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise inform the user that the rt was authorized
|
|
|
|
|
|
|
|
$this->elementStart('p');
|
|
|
|
|
2010-01-11 07:03:30 +00:00
|
|
|
// XXX: Do OAuth 1.0a verifier code?
|
2010-01-11 05:35:46 +00:00
|
|
|
|
|
|
|
$this->raw(sprintf(_("The request token %s has been authorized. " .
|
|
|
|
'Please exchange it for an access token.'),
|
|
|
|
$this->oauth_token));
|
|
|
|
|
|
|
|
$this->elementEnd('p');
|
|
|
|
|
|
|
|
} else if ($this->arg('deny')) {
|
|
|
|
|
|
|
|
$this->elementStart('p');
|
|
|
|
|
|
|
|
$this->raw(sprintf(_("The request token %s has been denied."),
|
|
|
|
$this->oauth_token));
|
|
|
|
|
|
|
|
$this->elementEnd('p');
|
|
|
|
} else {
|
|
|
|
$this->clientError(_('Unexpected form submission.'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function showForm($error=null)
|
|
|
|
{
|
|
|
|
$this->error = $error;
|
|
|
|
$this->showPage();
|
|
|
|
}
|
|
|
|
|
|
|
|
function showScripts()
|
|
|
|
{
|
|
|
|
parent::showScripts();
|
2010-01-11 07:03:30 +00:00
|
|
|
if (!common_logged_in()) {
|
|
|
|
$this->autofocus('nickname');
|
|
|
|
}
|
2010-01-11 05:35:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Title of the page
|
|
|
|
*
|
|
|
|
* @return string title of the page
|
|
|
|
*/
|
|
|
|
|
|
|
|
function title()
|
|
|
|
{
|
|
|
|
return _('An application would like to connect to your account');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Show page notice
|
|
|
|
*
|
|
|
|
* Display a notice for how to use the page, or the
|
|
|
|
* error if it exists.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function showPageNotice()
|
|
|
|
{
|
|
|
|
if ($this->error) {
|
|
|
|
$this->element('p', 'error', $this->error);
|
|
|
|
} else {
|
|
|
|
$instr = $this->getInstructions();
|
|
|
|
$output = common_markup_to_html($instr);
|
|
|
|
|
|
|
|
$this->raw($output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows the authorization form.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function showContent()
|
|
|
|
{
|
|
|
|
$this->elementStart('form', array('method' => 'post',
|
|
|
|
'id' => 'form_login',
|
|
|
|
'class' => 'form_settings',
|
|
|
|
'action' => common_local_url('apioauthauthorize')));
|
|
|
|
|
|
|
|
$this->hidden('token', common_session_token());
|
|
|
|
$this->hidden('oauth_token', $this->oauth_token);
|
|
|
|
$this->hidden('oauth_callback', $this->callback);
|
|
|
|
|
|
|
|
$this->elementStart('fieldset');
|
|
|
|
|
|
|
|
$this->elementStart('ul');
|
|
|
|
$this->elementStart('li');
|
|
|
|
if (!empty($this->app->icon)) {
|
|
|
|
$this->element('img', array('src' => $this->app->icon));
|
|
|
|
}
|
|
|
|
$this->elementEnd('li');
|
|
|
|
$this->elementStart('li');
|
|
|
|
|
|
|
|
$access = ($this->app->access_type & Oauth_application::$writeAccess) ?
|
|
|
|
'access and update' : 'access';
|
|
|
|
|
|
|
|
$msg = _("The application <b>%s</b> by <b>%s</b> would like " .
|
|
|
|
"the ability to <b>%s</b> your account data.");
|
|
|
|
|
|
|
|
$this->raw(sprintf($msg,
|
|
|
|
$this->app->name,
|
|
|
|
$this->app->organization,
|
|
|
|
$access));
|
|
|
|
|
|
|
|
$this->elementEnd('li');
|
|
|
|
$this->elementEnd('ul');
|
|
|
|
|
|
|
|
$this->elementEnd('fieldset');
|
|
|
|
|
|
|
|
if (!common_logged_in()) {
|
|
|
|
|
|
|
|
$this->elementStart('fieldset');
|
|
|
|
$this->element('legend', null, _('Login'));
|
|
|
|
$this->elementStart('ul', 'form_data');
|
|
|
|
$this->elementStart('li');
|
|
|
|
$this->input('nickname', _('Nickname'));
|
|
|
|
$this->elementEnd('li');
|
|
|
|
$this->elementStart('li');
|
|
|
|
$this->password('password', _('Password'));
|
|
|
|
$this->elementEnd('li');
|
|
|
|
$this->elementEnd('ul');
|
|
|
|
|
|
|
|
$this->elementEnd('fieldset');
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->element('input', array('id' => 'deny_submit',
|
|
|
|
'class' => 'submit',
|
|
|
|
'name' => 'deny',
|
|
|
|
'type' => 'submit',
|
|
|
|
'value' => _('Deny')));
|
|
|
|
|
|
|
|
$this->element('input', array('id' => 'allow_submit',
|
|
|
|
'class' => 'submit',
|
|
|
|
'name' => 'allow',
|
|
|
|
'type' => 'submit',
|
|
|
|
'value' => _('Allow')));
|
|
|
|
|
|
|
|
$this->elementEnd('form');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Instructions for using the form
|
|
|
|
*
|
|
|
|
* For "remembered" logins, we make the user re-login when they
|
|
|
|
* try to change settings. Different instructions for this case.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function getInstructions()
|
|
|
|
{
|
|
|
|
return _('Allow or deny access to your account information.');
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A local menu
|
|
|
|
*
|
|
|
|
* Shows different login/register actions.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function showLocalNav()
|
|
|
|
{
|
|
|
|
}
|
2010-01-07 21:19:21 +00:00
|
|
|
|
|
|
|
}
|