2010-02-09 06:37:45 +00:00
< ? php
/*
* StatusNet - the distributed open - source microblogging tool
2010-03-03 21:40:26 +00:00
* Copyright ( C ) 2009 - 2010 , StatusNet , Inc .
2010-02-09 06:37:45 +00:00
*
* 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 />.
*/
/**
* @ package OStatusPlugin
2010-02-20 20:46:48 +00:00
* @ maintainer Brion Vibber < brion @ status . net >
2010-02-09 06:37:45 +00:00
*/
if ( ! defined ( 'STATUSNET' ) && ! defined ( 'LACONICA' )) { exit ( 1 ); }
2010-02-23 01:58:05 +00:00
/**
* Key UI methods :
*
* showInputForm () - form asking for a remote profile account or URL
* We end up back here on errors
*
* showPreviewForm () - surrounding form for preview - and - confirm
2010-03-03 21:40:26 +00:00
* preview () - display profile for a remote user
2010-02-23 01:58:05 +00:00
*
2010-03-03 21:40:26 +00:00
* success () - redirects to subscriptions page on subscribe
2010-02-23 01:58:05 +00:00
*/
2010-02-09 06:37:45 +00:00
class OStatusSubAction extends Action
{
2010-02-23 01:58:05 +00:00
protected $profile_uri ; // provided acct: or URI of remote entity
protected $oprofile ; // Ostatus_profile of remote entity, if valid
2010-02-20 20:46:48 +00:00
/**
2010-02-23 01:58:05 +00:00
* Show the initial form , when we haven ' t yet been given a valid
* remote profile .
2010-02-20 20:46:48 +00:00
*/
2010-02-23 01:58:05 +00:00
function showInputForm ()
2010-02-09 06:37:45 +00:00
{
$user = common_current_user ();
$profile = $user -> getProfile ();
$this -> elementStart ( 'form' , array ( 'method' => 'post' ,
2010-02-22 16:07:48 +00:00
'id' => 'form_ostatus_sub' ,
2010-02-09 06:37:45 +00:00
'class' => 'form_settings' ,
2010-03-03 21:40:26 +00:00
'action' => $this -> selfLink ()));
2010-02-09 06:37:45 +00:00
$this -> hidden ( 'token' , common_session_token ());
$this -> elementStart ( 'fieldset' , array ( 'id' => 'settings_feeds' ));
$this -> elementStart ( 'ul' , 'form_data' );
2010-02-13 17:46:10 +00:00
$this -> elementStart ( 'li' );
2010-02-20 20:46:48 +00:00
$this -> input ( 'profile' ,
2010-03-04 01:08:55 +00:00
_m ( 'Subscribe to' ),
2010-02-20 20:46:48 +00:00
$this -> profile_uri ,
2010-03-04 03:56:50 +00:00
_m ( " OStatus user's address, like nickname@example.com or http://example.net/nickname " ));
2010-02-09 06:37:45 +00:00
$this -> elementEnd ( 'li' );
$this -> elementEnd ( 'ul' );
2010-02-23 01:58:05 +00:00
$this -> submit ( 'validate' , _m ( 'Continue' ));
$this -> elementEnd ( 'fieldset' );
$this -> elementEnd ( 'form' );
}
/**
* Show the preview - and - confirm form . We ' ve got a valid remote
* profile and are ready to poke it !
*
* This controls the wrapper form ; actual profile display will
* be in previewUser () or previewGroup () depending on the type .
*/
function showPreviewForm ()
{
2010-03-03 21:40:26 +00:00
$ok = $this -> preview ();
2010-02-23 19:56:17 +00:00
if ( ! $ok ) {
// @fixme maybe provide a cancel button or link back?
return ;
2010-02-23 03:59:34 +00:00
}
$this -> elementStart ( 'div' , 'entity_actions' );
$this -> elementStart ( 'ul' );
$this -> elementStart ( 'li' , 'entity_subscribe' );
2010-02-23 01:58:05 +00:00
$this -> elementStart ( 'form' , array ( 'method' => 'post' ,
'id' => 'form_ostatus_sub' ,
2010-02-23 03:59:34 +00:00
'class' => 'form_remote_authorize' ,
2010-02-23 01:58:05 +00:00
'action' =>
2010-03-03 21:40:26 +00:00
$this -> selfLink ()));
2010-02-23 03:59:34 +00:00
$this -> elementStart ( 'fieldset' );
2010-02-23 01:58:05 +00:00
$this -> hidden ( 'token' , common_session_token ());
$this -> hidden ( 'profile' , $this -> profile_uri );
if ( $this -> oprofile -> isGroup ()) {
2010-02-23 03:59:34 +00:00
$this -> submit ( 'submit' , _m ( 'Join' ), 'submit' , null ,
_m ( 'Join this group' ));
2010-02-20 20:46:48 +00:00
} else {
2010-03-02 14:57:28 +00:00
$this -> submit ( 'submit' , _m ( 'Confirm' ), 'submit' , null ,
2010-02-23 03:59:34 +00:00
_m ( 'Subscribe to this user' ));
2010-02-20 20:46:48 +00:00
}
2010-02-09 06:37:45 +00:00
$this -> elementEnd ( 'fieldset' );
$this -> elementEnd ( 'form' );
2010-02-23 03:59:34 +00:00
$this -> elementEnd ( 'li' );
$this -> elementEnd ( 'ul' );
$this -> elementEnd ( 'div' );
2010-02-23 01:58:05 +00:00
}
/**
* Show a preview for a remote user ' s profile
2010-02-23 19:56:17 +00:00
* @ return boolean true if we ' re ok to try subscribing
2010-02-23 01:58:05 +00:00
*/
2010-03-03 21:40:26 +00:00
function preview ()
2010-02-23 01:58:05 +00:00
{
$oprofile = $this -> oprofile ;
$profile = $oprofile -> localProfile ();
2010-02-23 19:56:17 +00:00
$cur = common_current_user ();
if ( $cur -> isSubscribed ( $profile )) {
$this -> element ( 'div' , array ( 'class' => 'error' ),
_m ( " You are already subscribed to this user. " ));
$ok = false ;
} else {
$ok = true ;
}
2010-02-23 19:37:49 +00:00
$avatar = $profile -> getAvatar ( AVATAR_PROFILE_SIZE );
$avatarUrl = $avatar ? $avatar -> displayUrl () : false ;
$this -> showEntity ( $profile ,
$profile -> profileurl ,
$avatarUrl ,
$profile -> bio );
2010-02-23 19:56:17 +00:00
return $ok ;
2010-02-23 01:58:05 +00:00
}
2010-02-23 19:37:49 +00:00
function showEntity ( $entity , $profile , $avatar , $note )
2010-02-23 03:59:34 +00:00
{
$nickname = $entity -> nickname ;
$fullname = $entity -> fullname ;
$homepage = $entity -> homepage ;
$location = $entity -> location ;
2010-02-23 19:37:49 +00:00
if ( ! $avatar ) {
$avatar = Avatar :: defaultImage ( AVATAR_PROFILE_SIZE );
}
2010-02-23 03:59:34 +00:00
$this -> elementStart ( 'div' , 'entity_profile vcard' );
$this -> elementStart ( 'dl' , 'entity_depiction' );
$this -> element ( 'dt' , null , _ ( 'Photo' ));
$this -> elementStart ( 'dd' );
2010-02-23 19:37:49 +00:00
$this -> element ( 'img' , array ( 'src' => $avatar ,
'class' => 'photo avatar' ,
'width' => AVATAR_PROFILE_SIZE ,
'height' => AVATAR_PROFILE_SIZE ,
'alt' => $nickname ));
2010-02-23 03:59:34 +00:00
$this -> elementEnd ( 'dd' );
$this -> elementEnd ( 'dl' );
$this -> elementStart ( 'dl' , 'entity_nickname' );
$this -> element ( 'dt' , null , _ ( 'Nickname' ));
$this -> elementStart ( 'dd' );
$hasFN = ( $fullname !== '' ) ? 'nickname' : 'fn nickname' ;
$this -> elementStart ( 'a' , array ( 'href' => $profile ,
'class' => 'url ' . $hasFN ));
$this -> raw ( $nickname );
$this -> elementEnd ( 'a' );
$this -> elementEnd ( 'dd' );
$this -> elementEnd ( 'dl' );
if ( ! is_null ( $fullname )) {
$this -> elementStart ( 'dl' , 'entity_fn' );
$this -> elementStart ( 'dd' );
$this -> elementStart ( 'span' , 'fn' );
$this -> raw ( $fullname );
$this -> elementEnd ( 'span' );
$this -> elementEnd ( 'dd' );
$this -> elementEnd ( 'dl' );
}
if ( ! is_null ( $location )) {
$this -> elementStart ( 'dl' , 'entity_location' );
$this -> element ( 'dt' , null , _ ( 'Location' ));
$this -> elementStart ( 'dd' , 'label' );
$this -> raw ( $location );
$this -> elementEnd ( 'dd' );
$this -> elementEnd ( 'dl' );
}
if ( ! is_null ( $homepage )) {
$this -> elementStart ( 'dl' , 'entity_url' );
$this -> element ( 'dt' , null , _ ( 'URL' ));
$this -> elementStart ( 'dd' );
$this -> elementStart ( 'a' , array ( 'href' => $homepage ,
'class' => 'url' ));
$this -> raw ( $homepage );
$this -> elementEnd ( 'a' );
$this -> elementEnd ( 'dd' );
$this -> elementEnd ( 'dl' );
}
2010-02-23 19:37:49 +00:00
if ( ! is_null ( $note )) {
2010-02-23 03:59:34 +00:00
$this -> elementStart ( 'dl' , 'entity_note' );
$this -> element ( 'dt' , null , _ ( 'Note' ));
$this -> elementStart ( 'dd' , 'note' );
2010-02-23 19:37:49 +00:00
$this -> raw ( $note );
2010-02-23 03:59:34 +00:00
$this -> elementEnd ( 'dd' );
$this -> elementEnd ( 'dl' );
}
$this -> elementEnd ( 'div' );
2010-02-23 01:58:05 +00:00
}
/**
* Redirect on successful remote user subscription
*/
2010-03-03 21:40:26 +00:00
function success ()
2010-02-23 01:58:05 +00:00
{
$cur = common_current_user ();
$url = common_local_url ( 'subscriptions' , array ( 'nickname' => $cur -> nickname ));
common_redirect ( $url , 303 );
}
/**
* Pull data for a remote profile and check if it ' s valid .
* Fills out error UI string in $this -> error
* Fills out $this -> oprofile on success .
*
* @ return boolean
*/
2010-03-03 21:40:26 +00:00
function pullRemoteProfile ()
2010-02-23 01:58:05 +00:00
{
2010-03-03 21:40:26 +00:00
$this -> profile_uri = $this -> trimmed ( 'profile' );
2010-02-23 01:58:05 +00:00
try {
2010-02-23 21:11:44 +00:00
if ( Validate :: email ( $this -> profile_uri )) {
$this -> oprofile = Ostatus_profile :: ensureWebfinger ( $this -> profile_uri );
} else if ( Validate :: uri ( $this -> profile_uri )) {
$this -> oprofile = Ostatus_profile :: ensureProfile ( $this -> profile_uri );
} else {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that address. Please make sure that the OStatus address is like nickname@example.com or http://example.net/nickname " );
common_debug ( 'Invalid address format.' , __FILE__ );
2010-02-23 21:11:44 +00:00
return false ;
}
2010-02-23 01:58:05 +00:00
return true ;
} catch ( FeedSubBadURLException $e ) {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that address. Please make sure that the OStatus address is like nickname@example.com or http://example.net/nickname " );
common_debug ( 'Invalid URL or could not reach server.' , __FILE__ );
2010-02-23 01:58:05 +00:00
} catch ( FeedSubBadResponseException $e ) {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that feed. Please try that OStatus address again later. " );
common_debug ( 'Cannot read feed; server returned error.' , __FILE__ );
2010-02-23 01:58:05 +00:00
} catch ( FeedSubEmptyException $e ) {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that feed. Please try that OStatus address again later. " );
common_debug ( 'Cannot read feed; server returned an empty page.' , __FILE__ );
2010-02-23 01:58:05 +00:00
} catch ( FeedSubBadHTMLException $e ) {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that feed. Please try that OStatus address again later. " );
common_debug ( 'Bad HTML, could not find feed link.' , __FILE__ );
2010-02-23 01:58:05 +00:00
} catch ( FeedSubNoFeedException $e ) {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that feed. Please try that OStatus address again later. " );
common_debug ( 'Could not find a feed linked from this URL.' , __FILE__ );
2010-02-23 01:58:05 +00:00
} catch ( FeedSubUnrecognizedTypeException $e ) {
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that feed. Please try that OStatus address again later. " );
common_debug ( 'Not a recognized feed type.' , __FILE__ );
2010-03-04 02:23:28 +00:00
} catch ( Exception $e ) {
2010-02-23 01:58:05 +00:00
// Any new ones we forgot about
2010-03-04 03:56:50 +00:00
$this -> error = _m ( " Sorry, we could not reach that address. Please make sure that the OStatus address is like nickname@example.com or http://example.net/nickname " );
common_debug ( sprintf ( 'Bad feed URL: %s %s' , get_class ( $e ), $e -> getMessage ()), __FILE__ );
2010-02-23 01:58:05 +00:00
}
return false ;
}
2010-03-03 21:40:26 +00:00
function validateRemoteProfile ()
{
if ( $this -> oprofile -> isGroup ()) {
// Send us to the group subscription form for conf
$target = common_local_url ( 'ostatusgroup' , array (), array ( 'profile' => $this -> profile_uri ));
common_redirect ( $target , 303 );
}
}
2010-02-23 01:58:05 +00:00
/**
* Attempt to finalize subscription .
* validateFeed must have been run first .
*
2010-03-03 21:40:26 +00:00
* Calls showForm on failure or success on success .
2010-02-23 01:58:05 +00:00
*/
function saveFeed ()
{
// And subscribe the current user to the local profile
$user = common_current_user ();
2010-03-03 21:40:26 +00:00
$local = $this -> oprofile -> localProfile ();
if ( $user -> isSubscribed ( $local )) {
// TRANS: OStatus remote subscription dialog error.
$this -> showForm ( _m ( 'Already subscribed!' ));
} elseif ( $this -> oprofile -> subscribeLocalToRemote ( $user )) {
$this -> success ();
2010-02-23 01:58:05 +00:00
} else {
2010-03-03 21:40:26 +00:00
// TRANS: OStatus remote subscription dialog error.
$this -> showForm ( _m ( 'Remote subscription failed!' ));
2010-02-20 20:46:48 +00:00
}
}
function prepare ( $args )
{
parent :: prepare ( $args );
2010-02-21 16:24:04 +00:00
if ( ! common_logged_in ()) {
// XXX: selfURL() didn't work. :<
common_set_returnto ( $_SERVER [ 'REQUEST_URI' ]);
if ( Event :: handle ( 'RedirectToLogin' , array ( $this , null ))) {
common_redirect ( common_local_url ( 'login' ), 303 );
}
return false ;
}
2010-03-03 21:40:26 +00:00
if ( $this -> pullRemoteProfile ()) {
$this -> validateRemoteProfile ();
}
2010-02-20 20:46:48 +00:00
return true ;
}
2010-02-23 01:58:05 +00:00
/**
* Handle the submission .
*/
2010-02-20 20:46:48 +00:00
function handle ( $args )
{
parent :: handle ( $args );
if ( $_SERVER [ 'REQUEST_METHOD' ] == 'POST' ) {
$this -> handlePost ();
} else {
2010-02-23 01:58:05 +00:00
$this -> showForm ();
2010-02-20 20:46:48 +00:00
}
2010-02-09 06:37:45 +00:00
}
2010-02-23 01:58:05 +00:00
2010-02-09 06:37:45 +00:00
/**
* Handle posts to this form
*
* @ return void
*/
function handlePost ()
{
// CSRF protection
$token = $this -> trimmed ( 'token' );
if ( ! $token || $token != common_session_token ()) {
$this -> showForm ( _ ( 'There was a problem with your session token. ' .
'Try again, please.' ));
return ;
}
2010-03-03 21:40:26 +00:00
if ( $this -> oprofile ) {
2010-02-23 19:37:49 +00:00
if ( $this -> arg ( 'submit' )) {
2010-02-23 01:58:05 +00:00
$this -> saveFeed ();
return ;
}
2010-02-09 06:37:45 +00:00
}
2010-02-23 01:58:05 +00:00
$this -> showForm ();
2010-02-09 06:37:45 +00:00
}
/**
2010-02-23 01:58:05 +00:00
* Show the appropriate form based on our input state .
2010-02-09 06:37:45 +00:00
*/
2010-02-23 01:58:05 +00:00
function showForm ( $err = null )
2010-02-09 06:37:45 +00:00
{
2010-02-23 01:58:05 +00:00
if ( $err ) {
$this -> error = $err ;
2010-02-09 06:37:45 +00:00
}
2010-02-23 01:58:05 +00:00
if ( $this -> boolean ( 'ajax' )) {
header ( 'Content-Type: text/xml;charset=utf-8' );
$this -> xw -> startDocument ( '1.0' , 'UTF-8' );
$this -> elementStart ( 'html' );
$this -> elementStart ( 'head' );
$this -> element ( 'title' , null , _m ( 'Subscribe to user' ));
$this -> elementEnd ( 'head' );
$this -> elementStart ( 'body' );
$this -> showContent ();
$this -> elementEnd ( 'body' );
$this -> elementEnd ( 'html' );
} else {
$this -> showPage ();
2010-02-09 06:37:45 +00:00
}
}
2010-02-23 01:58:05 +00:00
/**
* Title of the page
*
* @ return string Title of the page
*/
2010-02-09 06:37:45 +00:00
2010-02-23 01:58:05 +00:00
function title ()
{
2010-03-01 22:10:13 +00:00
// TRANS: Page title for OStatus remote subscription form
2010-03-04 04:47:27 +00:00
return _m ( 'Confirm' );
2010-02-23 01:58:05 +00:00
}
2010-02-20 20:46:48 +00:00
2010-02-23 01:58:05 +00:00
/**
* Instructions for use
*
* @ return instructions for use
*/
2010-02-20 20:46:48 +00:00
2010-02-23 01:58:05 +00:00
function getInstructions ()
{
return _m ( 'You can subscribe to users from other supported sites. Paste their address or profile URI below:' );
2010-02-09 06:37:45 +00:00
}
2010-02-23 01:58:05 +00:00
function showPageNotice ()
2010-02-09 06:37:45 +00:00
{
2010-02-23 19:37:49 +00:00
if ( ! empty ( $this -> error )) {
2010-02-23 01:58:05 +00:00
$this -> element ( 'p' , 'error' , $this -> error );
2010-02-09 06:37:45 +00:00
}
}
2010-02-23 01:58:05 +00:00
/**
* Content area of the page
*
* Shows a form for associating a remote OStatus account with this
* StatusNet account .
*
* @ return void
*/
function showContent ()
2010-02-20 20:46:48 +00:00
{
2010-02-23 01:58:05 +00:00
if ( $this -> oprofile ) {
$this -> showPreviewForm ();
} else {
$this -> showInputForm ();
}
2010-02-20 20:46:48 +00:00
}
2010-02-09 06:37:45 +00:00
2010-02-20 20:46:48 +00:00
function showScripts ()
{
parent :: showScripts ();
$this -> autofocus ( 'feedurl' );
}
2010-03-03 21:40:26 +00:00
function selfLink ()
{
return common_local_url ( 'ostatussub' );
}
2010-02-13 17:46:10 +00:00
}