2011-08-09 06:44:29 +02:00
< ? php
2015-12-25 23:17:34 +01:00
/**
* @ file include / diaspora . php
*
* @ todo GET / people / 9 aed8882b9f64896 / stream
*/
2015-07-11 14:37:02 +02:00
2011-08-10 03:55:46 +02:00
require_once ( 'include/crypto.php' );
2011-08-16 08:19:17 +02:00
require_once ( 'include/items.php' );
2011-08-23 12:02:31 +02:00
require_once ( 'include/bb2diaspora.php' );
2011-08-28 14:00:30 +02:00
require_once ( 'include/contact_selectors.php' );
2011-10-21 12:33:34 +02:00
require_once ( 'include/queue_fn.php' );
2012-06-15 04:58:25 +02:00
require_once ( 'include/lock.php' );
2015-03-21 20:23:47 +01:00
require_once ( 'include/threads.php' );
2015-04-05 20:43:06 +02:00
require_once ( 'mod/share.php' );
2016-01-28 22:58:05 +01:00
require_once ( 'include/enotify.php' );
2011-08-09 06:44:29 +02:00
2011-09-15 04:33:42 +02:00
function diaspora_dispatch_public ( $msg ) {
2012-04-04 23:13:34 +02:00
$enabled = intval ( get_config ( 'system' , 'diaspora_enabled' ));
2012-04-05 03:49:39 +02:00
if ( ! $enabled ) {
logger ( 'mod-diaspora: disabled' );
2012-04-04 23:13:34 +02:00
return ;
2012-04-05 03:49:39 +02:00
}
2012-04-04 23:13:34 +02:00
2015-09-05 13:26:14 +02:00
// Use a dummy importer to import the data for the public copy
$importer = array ( " uid " => 0 , " page-flags " => PAGE_FREELOVE );
$result = diaspora_dispatch ( $importer , $msg );
logger ( " Dispatcher reported " . $result , LOGGER_DEBUG );
// Now distribute it to the followers
2013-01-12 14:52:15 +01:00
$r = q ( " SELECT `user`.* FROM `user` WHERE `user`.`uid` IN
2015-08-09 20:39:11 +02:00
( SELECT `contact` . `uid` FROM `contact` WHERE `contact` . `network` = '%s' AND `contact` . `addr` = '%s' )
AND `account_expired` = 0 AND `account_removed` = 0 " ,
2011-09-15 04:33:42 +02:00
dbesc ( NETWORK_DIASPORA ),
dbesc ( $msg [ 'author' ])
);
if ( count ( $r )) {
foreach ( $r as $rr ) {
2011-09-16 23:51:25 +02:00
logger ( 'diaspora_public: delivering to: ' . $rr [ 'username' ]);
2011-09-15 04:33:42 +02:00
diaspora_dispatch ( $rr , $msg );
}
}
2015-09-05 13:26:14 +02:00
else
2015-08-09 20:39:11 +02:00
logger ( 'diaspora_public: no subscribers for ' . $msg [ " author " ] . ' ' . print_r ( $msg , true ));
2011-09-15 04:33:42 +02:00
}
2013-01-12 14:52:15 +01:00
function diaspora_dispatch ( $importer , $msg , $attempt = 1 ) {
2011-08-24 10:21:24 +02:00
2011-09-15 04:33:42 +02:00
$ret = 0 ;
2012-04-04 23:13:34 +02:00
$enabled = intval ( get_config ( 'system' , 'diaspora_enabled' ));
2012-04-05 03:49:39 +02:00
if ( ! $enabled ) {
logger ( 'mod-diaspora: disabled' );
2012-04-04 23:13:34 +02:00
return ;
2012-04-05 03:49:39 +02:00
}
2012-04-04 23:13:34 +02:00
2016-02-27 23:54:17 +01:00
$data = $msg ;
2011-10-18 02:39:25 +02:00
// php doesn't like dashes in variable names
$msg [ 'message' ] = str_replace (
array ( '<activity_streams-photo>' , '</activity_streams-photo>' ),
array ( '<asphoto>' , '</asphoto>' ),
$msg [ 'message' ]);
2011-08-24 10:21:24 +02:00
$parsed_xml = parse_xml_string ( $msg [ 'message' ], false );
$xmlbase = $parsed_xml -> post ;
2011-12-01 02:08:16 +01:00
logger ( 'diaspora_dispatch: ' . print_r ( $xmlbase , true ), LOGGER_DEBUG );
2011-08-24 10:21:24 +02:00
if ( $xmlbase -> request ) {
2016-02-27 23:54:17 +01:00
$tempfile = tempnam ( get_temppath (), " diaspora-request " );
file_put_contents ( $tempfile , json_encode ( $data ));
2011-09-15 04:33:42 +02:00
$ret = diaspora_request ( $importer , $xmlbase -> request );
2011-08-24 10:21:24 +02:00
}
elseif ( $xmlbase -> status_message ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-status_message");
//file_put_contents($tempfile, json_encode($data));
2012-07-09 07:32:04 +02:00
$ret = diaspora_post ( $importer , $xmlbase -> status_message , $msg );
2011-08-24 10:21:24 +02:00
}
2011-10-14 03:32:02 +02:00
elseif ( $xmlbase -> profile ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-profile");
//file_put_contents($tempfile, json_encode($data));
2012-07-09 07:32:04 +02:00
$ret = diaspora_profile ( $importer , $xmlbase -> profile , $msg );
2011-10-14 03:32:02 +02:00
}
2011-08-24 10:21:24 +02:00
elseif ( $xmlbase -> comment ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-comment");
//file_put_contents($tempfile, json_encode($data));
2011-09-15 04:33:42 +02:00
$ret = diaspora_comment ( $importer , $xmlbase -> comment , $msg );
2011-08-24 10:21:24 +02:00
}
elseif ( $xmlbase -> like ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-like");
//file_put_contents($tempfile, json_encode($data));
2011-09-15 04:33:42 +02:00
$ret = diaspora_like ( $importer , $xmlbase -> like , $msg );
2011-08-24 10:21:24 +02:00
}
2011-10-18 02:39:25 +02:00
elseif ( $xmlbase -> asphoto ) {
2016-02-27 23:54:17 +01:00
$tempfile = tempnam ( get_temppath (), " diaspora-asphoto " );
file_put_contents ( $tempfile , json_encode ( $data ));
2012-07-09 07:32:04 +02:00
$ret = diaspora_asphoto ( $importer , $xmlbase -> asphoto , $msg );
2011-10-18 02:39:25 +02:00
}
2011-10-19 00:56:35 +02:00
elseif ( $xmlbase -> reshare ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-reshare");
//file_put_contents($tempfile, json_encode($data));
2012-07-09 07:32:04 +02:00
$ret = diaspora_reshare ( $importer , $xmlbase -> reshare , $msg );
2011-10-19 00:56:35 +02:00
}
2011-08-24 10:21:24 +02:00
elseif ( $xmlbase -> retraction ) {
2016-02-27 23:54:17 +01:00
$tempfile = tempnam ( get_temppath (), " diaspora-retraction " );
file_put_contents ( $tempfile , json_encode ( $data ));
2011-09-15 04:33:42 +02:00
$ret = diaspora_retraction ( $importer , $xmlbase -> retraction , $msg );
2011-08-24 10:21:24 +02:00
}
2011-11-05 22:45:29 +01:00
elseif ( $xmlbase -> signed_retraction ) {
2016-02-29 08:02:50 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-signed_retraction");
//file_put_contents($tempfile, json_encode($data));
2011-12-21 23:42:12 +01:00
$ret = diaspora_signed_retraction ( $importer , $xmlbase -> signed_retraction , $msg );
2011-11-05 22:45:29 +01:00
}
2012-05-26 02:26:09 +02:00
elseif ( $xmlbase -> relayable_retraction ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-relayable_retraction");
//file_put_contents($tempfile, json_encode($data));
2012-05-26 02:26:09 +02:00
$ret = diaspora_signed_retraction ( $importer , $xmlbase -> relayable_retraction , $msg );
}
2011-08-24 10:21:24 +02:00
elseif ( $xmlbase -> photo ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-photo");
//file_put_contents($tempfile, json_encode($data));
2013-01-12 14:52:15 +01:00
$ret = diaspora_photo ( $importer , $xmlbase -> photo , $msg , $attempt );
2011-08-24 10:21:24 +02:00
}
2011-11-28 02:41:23 +01:00
elseif ( $xmlbase -> conversation ) {
2016-02-27 23:54:17 +01:00
$tempfile = tempnam ( get_temppath (), " diaspora-conversation " );
file_put_contents ( $tempfile , json_encode ( $data ));
2011-11-28 02:41:23 +01:00
$ret = diaspora_conversation ( $importer , $xmlbase -> conversation , $msg );
}
2011-12-07 04:15:42 +01:00
elseif ( $xmlbase -> message ) {
2016-02-27 23:54:17 +01:00
$tempfile = tempnam ( get_temppath (), " diaspora-message " );
file_put_contents ( $tempfile , json_encode ( $data ));
2011-12-07 04:15:42 +01:00
$ret = diaspora_message ( $importer , $xmlbase -> message , $msg );
}
2015-10-08 00:25:55 +02:00
elseif ( $xmlbase -> participation ) {
2016-02-27 23:54:17 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-participation");
//file_put_contents($tempfile, json_encode($data));
$ret = diaspora_participation ( $importer , $xmlbase -> participation );
}
elseif ( $xmlbase -> poll_participation ) {
2016-03-01 19:10:35 +01:00
//$tempfile = tempnam(get_temppath(), "diaspora-poll_participation");
//file_put_contents($tempfile, json_encode($data));
$ret = diaspora_participation ( $importer , $xmlbase -> poll_participation );
2015-10-08 00:25:55 +02:00
}
2011-08-24 10:21:24 +02:00
else {
2016-02-27 23:54:17 +01:00
$tempfile = tempnam ( get_temppath (), " diaspora-unknown " );
file_put_contents ( $tempfile , json_encode ( $data ));
2011-08-24 10:21:24 +02:00
logger ( 'diaspora_dispatch: unknown message type: ' . print_r ( $xmlbase , true ));
}
2011-09-15 04:33:42 +02:00
return $ret ;
2011-08-24 10:21:24 +02:00
}
2012-08-10 06:06:18 +02:00
function diaspora_handle_from_contact ( $contact_id ) {
$handle = False ;
logger ( " diaspora_handle_from_contact: contact id is " . $contact_id , LOGGER_DEBUG );
$r = q ( " SELECT network, addr, self, url, nick FROM contact WHERE id = %d " ,
intval ( $contact_id )
);
if ( $r ) {
$contact = $r [ 0 ];
logger ( " diaspora_handle_from_contact: contact 'self' = " . $contact [ 'self' ] . " 'url' = " . $contact [ 'url' ], LOGGER_DEBUG );
if ( $contact [ 'network' ] === NETWORK_DIASPORA ) {
$handle = $contact [ 'addr' ];
// logger("diaspora_handle_from_contact: contact id is a Diaspora person, handle = " . $handle, LOGGER_DEBUG);
}
elseif (( $contact [ 'network' ] === NETWORK_DFRN ) || ( $contact [ 'self' ] == 1 )) {
$baseurl_start = strpos ( $contact [ 'url' ], '://' ) + 3 ;
$baseurl_length = strpos ( $contact [ 'url' ], '/profile' ) - $baseurl_start ; // allows installations in a subdirectory--not sure how Diaspora will handle
$baseurl = substr ( $contact [ 'url' ], $baseurl_start , $baseurl_length );
$handle = $contact [ 'nick' ] . '@' . $baseurl ;
// logger("diaspora_handle_from_contact: contact id is a DFRN person, handle = " . $handle, LOGGER_DEBUG);
}
}
return $handle ;
}
2011-08-23 04:27:40 +02:00
function diaspora_get_contact_by_handle ( $uid , $handle ) {
$r = q ( " SELECT * FROM `contact` WHERE `network` = '%s' AND `uid` = %d AND `addr` = '%s' LIMIT 1 " ,
dbesc ( NETWORK_DIASPORA ),
intval ( $uid ),
dbesc ( $handle )
);
if ( $r && count ( $r ))
return $r [ 0 ];
2012-09-30 01:48:56 +02:00
$handle_parts = explode ( " @ " , $handle );
$nurl_sql = '%%://' . $handle_parts [ 1 ] . '%%/profile/' . $handle_parts [ 0 ];
$r = q ( " SELECT * FROM contact WHERE network = '%s' AND uid = %d AND nurl LIKE '%s' LIMIT 1 " ,
dbesc ( NETWORK_DFRN ),
intval ( $uid ),
dbesc ( $nurl_sql )
);
if ( $r && count ( $r ))
return $r [ 0 ];
2011-08-23 04:27:40 +02:00
return false ;
}
function find_diaspora_person_by_handle ( $handle ) {
2012-06-15 04:58:25 +02:00
$person = false ;
2011-09-20 10:49:08 +02:00
$update = false ;
2012-06-15 04:58:25 +02:00
$got_lock = false ;
2012-07-06 03:01:13 +02:00
$endlessloop = 0 ;
$maxloops = 10 ;
2012-06-15 04:58:25 +02:00
do {
$r = q ( " select * from fcontact where network = '%s' and addr = '%s' limit 1 " ,
dbesc ( NETWORK_DIASPORA ),
dbesc ( $handle )
);
if ( count ( $r )) {
$person = $r [ 0 ];
logger ( 'find_diaspora_person_by handle: in cache ' . print_r ( $r , true ), LOGGER_DEBUG );
// update record occasionally so it doesn't get stale
$d = strtotime ( $person [ 'updated' ] . ' +00:00' );
if ( $d < strtotime ( 'now - 14 days' ))
$update = true ;
}
// FETCHING PERSON INFORMATION FROM REMOTE SERVER
//
// If the person isn't in our 'fcontact' table, or if he/she is but
// his/her information hasn't been updated for more than 14 days, then
// we want to fetch the person's information from the remote server.
//
// Note that $person isn't changed by this block of code unless the
// person's information has been successfully fetched from the remote
// server. So if $person was 'false' to begin with (because he/she wasn't
// in the local cache), it'll stay false, and if $person held the local
// cache information to begin with, it'll keep that information. That way
// if there's a problem with the remote fetch, we can at least use our
// cached information--it's better than nothing.
if (( ! $person ) || ( $update )) {
// Lock the function to prevent race conditions if multiple items
// come in at the same time from a person who doesn't exist in
// fcontact
2012-07-06 03:01:13 +02:00
//
// Don't loop forever. On the last loop, try to create the contact
// whether the function is locked or not. Maybe the locking thread
// has died or something. At any rate, a duplicate in 'fcontact'
// is a much smaller problem than a deadlocked thread
2012-07-06 03:08:30 +02:00
$got_lock = lock_function ( 'find_diaspora_person_by_handle' , false );
if (( $endlessloop + 1 ) >= $maxloops )
$got_lock = true ;
2012-06-15 04:58:25 +02:00
if ( $got_lock ) {
logger ( 'find_diaspora_person_by_handle: create or refresh' , LOGGER_DEBUG );
require_once ( 'include/Scrape.php' );
$r = probe_url ( $handle , PROBE_DIASPORA );
// Note that Friendica contacts can return a "Diaspora person"
// if Diaspora connectivity is enabled on their server
if (( count ( $r )) && ( $r [ 'network' ] === NETWORK_DIASPORA )) {
add_fcontact ( $r , $update );
$person = ( $r );
}
unlock_function ( 'find_diaspora_person_by_handle' );
}
else {
logger ( 'find_diaspora_person_by_handle: couldn\'t lock function' , LOGGER_DEBUG );
if ( ! $person )
block_on_function_lock ( 'find_diaspora_person_by_handle' );
}
}
2012-07-06 03:01:13 +02:00
} while (( ! $person ) && ( ! $got_lock ) && ( ++ $endlessloop < $maxloops ));
2012-06-15 04:58:25 +02:00
// We need to try again if the person wasn't in 'fcontact' but the function was locked.
// The fact that the function was locked may mean that another process was creating the
// person's record. It could also mean another process was creating or updating an unrelated
// person.
//
// At any rate, we need to keep trying until we've either got the person or had a chance to
// try to fetch his/her remote information. But we don't want to block on locking the
// function, because if the other process is creating the record, then when we acquire the lock
// we'll dive right into creating another, duplicate record. We DO want to at least wait
// until the lock is released, so we don't flood the database with requests.
//
// If the person was in the 'fcontact' table, don't try again. It's not worth the time, since
// we do have some information for the person
return $person ;
2011-08-23 04:27:40 +02:00
}
2011-08-09 11:53:51 +02:00
function get_diaspora_key ( $uri ) {
logger ( 'Fetching diaspora key for: ' . $uri );
2011-08-18 13:20:30 +02:00
$r = find_diaspora_person_by_handle ( $uri );
if ( $r )
return $r [ 'pubkey' ];
2011-08-09 11:53:51 +02:00
return '' ;
}
2011-08-09 06:44:29 +02:00
2011-09-19 05:17:44 +02:00
function diaspora_pubmsg_build ( $msg , $user , $contact , $prvkey , $pubkey ) {
$a = get_app ();
logger ( 'diaspora_pubmsg_build: ' . $msg , LOGGER_DATA );
2014-04-22 00:58:17 +02:00
2011-09-19 05:17:44 +02:00
$handle = $user [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
2011-09-22 13:11:39 +02:00
// $b64_data = base64_encode($msg);
// $b64url_data = base64url_encode($b64_data);
$b64url_data = base64url_encode ( $msg );
2011-09-19 05:17:44 +02:00
$data = str_replace ( array ( " \n " , " \r " , " " , " \t " ), array ( '' , '' , '' , '' ), $b64url_data );
$type = 'application/xml' ;
$encoding = 'base64url' ;
$alg = 'RSA-SHA256' ;
2014-04-22 00:58:17 +02:00
$signable_data = $data . '.' . base64url_encode ( $type ) . '.'
2011-09-19 05:17:44 +02:00
. base64url_encode ( $encoding ) . '.' . base64url_encode ( $alg ) ;
$signature = rsa_sign ( $signable_data , $prvkey );
$sig = base64url_encode ( $signature );
$magic_env = <<< EOT
< ? xml version = '1.0' encoding = 'UTF-8' ?>
2011-09-22 13:11:39 +02:00
< diaspora xmlns = " https://joindiaspora.com/protocol " xmlns : me = " http://salmon-protocol.org/ns/magic-env " >
2011-09-19 05:17:44 +02:00
< header >
< author_id > $handle </ author_id >
</ header >
< me : env >
< me : encoding > base64url </ me : encoding >
< me : alg > RSA - SHA256 </ me : alg >
< me : data type = " application/xml " > $data </ me : data >
< me : sig > $sig </ me : sig >
</ me : env >
</ diaspora >
EOT ;
logger ( 'diaspora_pubmsg_build: magic_env: ' . $magic_env , LOGGER_DATA );
return $magic_env ;
}
2011-09-22 13:11:39 +02:00
function diaspora_msg_build ( $msg , $user , $contact , $prvkey , $pubkey , $public = false ) {
2011-08-09 06:44:29 +02:00
$a = get_app ();
2011-09-22 13:11:39 +02:00
if ( $public )
return diaspora_pubmsg_build ( $msg , $user , $contact , $prvkey , $pubkey );
2011-08-19 13:48:54 +02:00
logger ( 'diaspora_msg_build: ' . $msg , LOGGER_DATA );
2011-12-20 01:51:57 +01:00
// without a public key nothing will work
if ( ! $pubkey ) {
logger ( 'diaspora_msg_build: pubkey missing: contact id: ' . $contact [ 'id' ]);
return '' ;
}
2011-08-09 06:51:56 +02:00
$inner_aes_key = random_string ( 32 );
2011-08-09 06:44:29 +02:00
$b_inner_aes_key = base64_encode ( $inner_aes_key );
2011-08-19 23:34:28 +02:00
$inner_iv = random_string ( 16 );
2011-08-09 06:44:29 +02:00
$b_inner_iv = base64_encode ( $inner_iv );
2011-08-09 06:51:56 +02:00
$outer_aes_key = random_string ( 32 );
2011-08-09 06:44:29 +02:00
$b_outer_aes_key = base64_encode ( $outer_aes_key );
2011-08-19 23:34:28 +02:00
$outer_iv = random_string ( 16 );
2011-08-09 06:44:29 +02:00
$b_outer_iv = base64_encode ( $outer_iv );
2014-05-03 23:40:34 +02:00
2011-09-16 02:47:16 +02:00
$handle = $user [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
2011-08-09 06:44:29 +02:00
2011-08-09 11:53:51 +02:00
$padded_data = pkcs5_pad ( $msg , 16 );
2011-08-09 06:44:29 +02:00
$inner_encrypted = mcrypt_encrypt ( MCRYPT_RIJNDAEL_128 , $inner_aes_key , $padded_data , MCRYPT_MODE_CBC , $inner_iv );
$b64_data = base64_encode ( $inner_encrypted );
2011-08-09 15:40:28 +02:00
2011-08-09 11:53:51 +02:00
2011-08-09 06:44:29 +02:00
$b64url_data = base64url_encode ( $b64_data );
2011-09-15 04:33:42 +02:00
$data = str_replace ( array ( " \n " , " \r " , " " , " \t " ), array ( '' , '' , '' , '' ), $b64url_data );
2011-09-16 02:47:16 +02:00
$type = 'application/xml' ;
2011-08-09 06:44:29 +02:00
$encoding = 'base64url' ;
$alg = 'RSA-SHA256' ;
2014-04-22 00:58:17 +02:00
$signable_data = $data . '.' . base64url_encode ( $type ) . '.'
2011-09-15 04:33:42 +02:00
. base64url_encode ( $encoding ) . '.' . base64url_encode ( $alg ) ;
2011-08-09 06:44:29 +02:00
2011-08-10 03:55:46 +02:00
$signature = rsa_sign ( $signable_data , $prvkey );
2011-08-09 06:44:29 +02:00
$sig = base64url_encode ( $signature );
$decrypted_header = <<< EOT
< decrypted_header >
< iv > $b_inner_iv </ iv >
< aes_key > $b_inner_aes_key </ aes_key >
2011-09-16 02:47:16 +02:00
< author_id > $handle </ author_id >
2011-08-09 06:44:29 +02:00
</ decrypted_header >
EOT ;
2011-08-09 11:53:51 +02:00
$decrypted_header = pkcs5_pad ( $decrypted_header , 16 );
2011-08-09 06:44:29 +02:00
2011-08-09 15:40:28 +02:00
$ciphertext = mcrypt_encrypt ( MCRYPT_RIJNDAEL_128 , $outer_aes_key , $decrypted_header , MCRYPT_MODE_CBC , $outer_iv );
2011-08-09 11:53:51 +02:00
$outer_json = json_encode ( array ( 'iv' => $b_outer_iv , 'key' => $b_outer_aes_key ));
2011-08-22 13:55:09 +02:00
2011-08-09 06:44:29 +02:00
$encrypted_outer_key_bundle = '' ;
openssl_public_encrypt ( $outer_json , $encrypted_outer_key_bundle , $pubkey );
2011-08-22 13:55:09 +02:00
2011-08-09 06:44:29 +02:00
$b64_encrypted_outer_key_bundle = base64_encode ( $encrypted_outer_key_bundle );
2011-08-22 13:55:09 +02:00
2011-08-29 06:41:42 +02:00
logger ( 'outer_bundle: ' . $b64_encrypted_outer_key_bundle . ' key: ' . $pubkey , LOGGER_DATA );
2011-08-22 13:55:09 +02:00
2011-08-09 06:44:29 +02:00
$encrypted_header_json_object = json_encode ( array ( 'aes_key' => base64_encode ( $encrypted_outer_key_bundle ),
'ciphertext' => base64_encode ( $ciphertext )));
2011-08-22 02:24:50 +02:00
$cipher_json = base64_encode ( $encrypted_header_json_object );
$encrypted_header = '<encrypted_header>' . $cipher_json . '</encrypted_header>' ;
2011-08-09 06:44:29 +02:00
2011-08-09 11:53:51 +02:00
$magic_env = <<< EOT
2011-08-09 06:44:29 +02:00
< ? xml version = '1.0' encoding = 'UTF-8' ?>
2011-09-22 13:11:39 +02:00
< diaspora xmlns = " https://joindiaspora.com/protocol " xmlns : me = " http://salmon-protocol.org/ns/magic-env " >
2011-08-09 06:44:29 +02:00
$encrypted_header
2011-09-16 23:46:04 +02:00
< me : env >
2011-08-09 06:44:29 +02:00
< me : encoding > base64url </ me : encoding >
< me : alg > RSA - SHA256 </ me : alg >
2011-09-15 04:33:42 +02:00
< me : data type = " application/xml " > $data </ me : data >
2011-08-09 06:44:29 +02:00
< me : sig > $sig </ me : sig >
</ me : env >
2011-09-16 23:46:04 +02:00
</ diaspora >
2011-08-09 06:44:29 +02:00
EOT ;
2011-08-19 13:48:54 +02:00
logger ( 'diaspora_msg_build: magic_env: ' . $magic_env , LOGGER_DATA );
2011-08-09 06:44:29 +02:00
return $magic_env ;
2011-08-09 11:53:51 +02:00
}
2011-08-17 13:24:26 +02:00
/**
*
* diaspora_decode ( $importer , $xml )
* array $importer -> from user table
* string $xml -> urldecoded Diaspora salmon
*
* Returns array
* 'message' -> decoded Diaspora XML message
* 'author' -> author diaspora handle
* 'key' -> author public key ( converted to pkcs #8)
*
* Author and key are used elsewhere to save a lookup for verifying replies and likes
*/
2011-08-09 11:53:51 +02:00
2011-08-15 14:27:24 +02:00
2011-08-17 13:24:26 +02:00
function diaspora_decode ( $importer , $xml ) {
2011-08-15 14:27:24 +02:00
2011-09-15 04:33:42 +02:00
$public = false ;
2011-08-09 15:40:28 +02:00
$basedom = parse_xml_string ( $xml );
2011-08-09 11:53:51 +02:00
2011-09-16 02:47:16 +02:00
$children = $basedom -> children ( 'https://joindiaspora.com/protocol' );
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
if ( $children -> header ) {
$public = true ;
2011-09-16 23:57:41 +02:00
$author_link = str_replace ( 'acct:' , '' , $children -> header -> author_id );
2011-09-15 04:33:42 +02:00
}
else {
2011-08-17 13:24:26 +02:00
2011-09-15 04:33:42 +02:00
$encrypted_header = json_decode ( base64_decode ( $children -> encrypted_header ));
2014-05-03 23:40:34 +02:00
2011-09-15 04:33:42 +02:00
$encrypted_aes_key_bundle = base64_decode ( $encrypted_header -> aes_key );
$ciphertext = base64_decode ( $encrypted_header -> ciphertext );
$outer_key_bundle = '' ;
openssl_private_decrypt ( $encrypted_aes_key_bundle , $outer_key_bundle , $importer [ 'prvkey' ]);
$j_outer_key_bundle = json_decode ( $outer_key_bundle );
$outer_iv = base64_decode ( $j_outer_key_bundle -> iv );
$outer_key = base64_decode ( $j_outer_key_bundle -> key );
$decrypted = mcrypt_decrypt ( MCRYPT_RIJNDAEL_128 , $outer_key , $ciphertext , MCRYPT_MODE_CBC , $outer_iv );
$decrypted = pkcs5_unpad ( $decrypted );
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
/**
* $decrypted now contains something like
*
* < decrypted_header >
* < iv > 8 e + G2 + ET8l5BPuW0sVTnQw ==</ iv >
* < aes_key > UvSMb4puPeB14STkcDWq + 4 QE302Edu15oaprAQSkLKU =</ aes_key >
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
***** OBSOLETE
2011-08-15 15:27:17 +02:00
2011-09-15 04:33:42 +02:00
* < author >
* < name > Ryan Hughes </ name >
* < uri > acct : galaxor @ diaspora . pirateship . org </ uri >
* </ author >
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
***** CURRENT
2011-08-09 11:53:51 +02:00
2011-09-22 13:11:39 +02:00
* < author_id > galaxor @ diaspora . priateship . org </ author_id >
2011-08-20 13:53:11 +02:00
2011-09-15 04:33:42 +02:00
***** END DIFFS
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
* </ decrypted_header >
*/
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
logger ( 'decrypted: ' . $decrypted , LOGGER_DEBUG );
$idom = parse_xml_string ( $decrypted , false );
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
$inner_iv = base64_decode ( $idom -> iv );
$inner_aes_key = base64_decode ( $idom -> aes_key );
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
$author_link = str_replace ( 'acct:' , '' , $idom -> author_id );
}
2011-08-09 11:53:51 +02:00
$dom = $basedom -> children ( NAMESPACE_SALMON_ME );
// figure out where in the DOM tree our data is hiding
if ( $dom -> provenance -> data )
$base = $dom -> provenance ;
elseif ( $dom -> env -> data )
$base = $dom -> env ;
elseif ( $dom -> data )
$base = $dom ;
2014-04-04 10:52:53 +02:00
2011-08-09 11:53:51 +02:00
if ( ! $base ) {
logger ( 'mod-diaspora: unable to locate salmon data in xml ' );
2011-08-15 05:38:31 +02:00
http_status_exit ( 400 );
2011-08-09 11:53:51 +02:00
}
// Stash the signature away for now. We have to find their key or it won't be good for anything.
$signature = base64url_decode ( $base -> sig );
// unpack the data
// strip whitespace so our data element will return to one big base64 blob
$data = str_replace ( array ( " " , " \t " , " \r " , " \n " ), array ( " " , " " , " " , " " ), $base -> data );
2011-08-17 13:24:26 +02:00
2011-08-09 11:53:51 +02:00
// stash away some other stuff for later
$type = $base -> data [ 0 ] -> attributes () -> type [ 0 ];
$keyhash = $base -> sig [ 0 ] -> attributes () -> keyhash [ 0 ];
$encoding = $base -> encoding ;
$alg = $base -> alg ;
2011-08-17 13:24:26 +02:00
2011-09-15 04:33:42 +02:00
$signed_data = $data . '.' . base64url_encode ( $type ) . '.' . base64url_encode ( $encoding ) . '.' . base64url_encode ( $alg );
2011-08-09 11:53:51 +02:00
// decode the data
$data = base64url_decode ( $data );
2011-09-15 04:33:42 +02:00
if ( $public ) {
$inner_decrypted = $data ;
}
else {
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
// Decode the encrypted blob
2011-08-09 11:53:51 +02:00
2011-09-15 04:33:42 +02:00
$inner_encrypted = base64_decode ( $data );
$inner_decrypted = mcrypt_decrypt ( MCRYPT_RIJNDAEL_128 , $inner_aes_key , $inner_encrypted , MCRYPT_MODE_CBC , $inner_iv );
$inner_decrypted = pkcs5_unpad ( $inner_decrypted );
}
2011-08-09 11:53:51 +02:00
if ( ! $author_link ) {
logger ( 'mod-diaspora: Could not retrieve author URI.' );
2011-08-10 03:55:46 +02:00
http_status_exit ( 400 );
2011-08-09 11:53:51 +02:00
}
// Once we have the author URI, go to the web and try to find their public key
2011-08-21 00:09:09 +02:00
// (first this will look it up locally if it is in the fcontact cache)
// This will also convert diaspora public key from pkcs#1 to pkcs#8
2011-08-09 11:53:51 +02:00
logger ( 'mod-diaspora: Fetching key for ' . $author_link );
2015-08-09 20:39:11 +02:00
$key = get_diaspora_key ( $author_link );
2011-08-09 11:53:51 +02:00
if ( ! $key ) {
2011-08-09 15:40:28 +02:00
logger ( 'mod-diaspora: Could not retrieve author key.' );
2011-08-10 03:55:46 +02:00
http_status_exit ( 400 );
2011-08-09 11:53:51 +02:00
}
2011-08-10 03:55:46 +02:00
$verify = rsa_verify ( $signed_data , $signature , $key );
2011-08-09 11:53:51 +02:00
if ( ! $verify ) {
logger ( 'mod-diaspora: Message did not verify. Discarding.' );
2011-09-19 12:36:41 +02:00
http_status_exit ( 400 );
2011-08-09 11:53:51 +02:00
}
logger ( 'mod-diaspora: Message verified.' );
2011-08-17 07:31:14 +02:00
return array ( 'message' => $inner_decrypted , 'author' => $author_link , 'key' => $key );
2011-08-09 11:53:51 +02:00
}
2013-01-20 14:08:28 +01:00
2011-08-16 08:19:17 +02:00
function diaspora_request ( $importer , $xml ) {
2011-08-10 14:10:48 +02:00
2011-11-12 13:06:33 +01:00
$a = get_app ();
2011-08-16 09:52:34 +02:00
$sender_handle = unxmlify ( $xml -> sender_handle );
$recipient_handle = unxmlify ( $xml -> recipient_handle );
2011-08-12 12:01:11 +02:00
if ( ! $sender_handle || ! $recipient_handle )
return ;
2013-01-20 14:08:28 +01:00
2011-08-16 08:19:17 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $sender_handle );
if ( $contact ) {
2011-08-17 13:24:26 +02:00
// perhaps we were already sharing with this person. Now they're sharing with us.
// That makes us friends.
2015-10-10 11:06:18 +02:00
if ( $contact [ 'rel' ] == CONTACT_IS_FOLLOWER && in_array ( $importer [ 'page-flags' ], array ( PAGE_FREELOVE ))) {
2014-03-09 09:19:14 +01:00
q ( " UPDATE `contact` SET `rel` = %d, `writable` = 1 WHERE `id` = %d AND `uid` = %d " ,
2011-08-16 08:19:17 +02:00
intval ( CONTACT_IS_FRIEND ),
intval ( $contact [ 'id' ]),
intval ( $importer [ 'uid' ])
);
}
2011-11-12 13:06:33 +01:00
// send notification
$r = q ( " SELECT `hide-friends` FROM `profile` WHERE `uid` = %d AND `is-default` = 1 LIMIT 1 " ,
intval ( $importer [ 'uid' ])
);
2015-02-01 16:04:16 +01:00
if (( count ( $r )) && ( ! $r [ 0 ][ 'hide-friends' ]) && ( ! $contact [ 'hidden' ]) && intval ( get_pconfig ( $importer [ 'uid' ], 'system' , 'post_newfriend' ))) {
2011-11-12 13:06:33 +01:00
require_once ( 'include/items.php' );
$self = q ( " SELECT * FROM `contact` WHERE `self` = 1 AND `uid` = %d LIMIT 1 " ,
intval ( $importer [ 'uid' ])
);
2011-11-16 08:12:56 +01:00
// they are not CONTACT_IS_FOLLOWER anymore but that's what we have in the array
if ( count ( $self ) && $contact [ 'rel' ] == CONTACT_IS_FOLLOWER ) {
2011-11-12 13:06:33 +01:00
$arr = array ();
2015-05-03 21:25:03 +02:00
$arr [ 'uri' ] = $arr [ 'parent-uri' ] = item_new_uri ( $a -> get_hostname (), $importer [ 'uid' ]);
2011-11-12 13:06:33 +01:00
$arr [ 'uid' ] = $importer [ 'uid' ];
$arr [ 'contact-id' ] = $self [ 0 ][ 'id' ];
$arr [ 'wall' ] = 1 ;
$arr [ 'type' ] = 'wall' ;
$arr [ 'gravity' ] = 0 ;
$arr [ 'origin' ] = 1 ;
$arr [ 'author-name' ] = $arr [ 'owner-name' ] = $self [ 0 ][ 'name' ];
$arr [ 'author-link' ] = $arr [ 'owner-link' ] = $self [ 0 ][ 'url' ];
$arr [ 'author-avatar' ] = $arr [ 'owner-avatar' ] = $self [ 0 ][ 'thumb' ];
$arr [ 'verb' ] = ACTIVITY_FRIEND ;
$arr [ 'object-type' ] = ACTIVITY_OBJ_PERSON ;
2014-04-04 10:52:53 +02:00
2011-11-12 13:06:33 +01:00
$A = '[url=' . $self [ 0 ][ 'url' ] . ']' . $self [ 0 ][ 'name' ] . '[/url]' ;
$B = '[url=' . $contact [ 'url' ] . ']' . $contact [ 'name' ] . '[/url]' ;
$BPhoto = '[url=' . $contact [ 'url' ] . ']' . '[img]' . $contact [ 'thumb' ] . '[/img][/url]' ;
$arr [ 'body' ] = sprintf ( t ( '%1$s is now friends with %2$s' ), $A , $B ) . " \n \n \n " . $Bphoto ;
$arr [ 'object' ] = '<object><type>' . ACTIVITY_OBJ_PERSON . '</type><title>' . $contact [ 'name' ] . '</title>'
. '<id>' . $contact [ 'url' ] . '/' . $contact [ 'name' ] . '</id>' ;
$arr [ 'object' ] .= '<link>' . xmlify ( '<link rel="alternate" type="text/html" href="' . $contact [ 'url' ] . '" />' . " \n " );
$arr [ 'object' ] .= xmlify ( '<link rel="photo" type="image/jpeg" href="' . $contact [ 'thumb' ] . '" />' . " \n " );
$arr [ 'object' ] .= '</link></object>' . " \n " ;
$arr [ 'last-child' ] = 1 ;
$arr [ 'allow_cid' ] = $user [ 0 ][ 'allow_cid' ];
$arr [ 'allow_gid' ] = $user [ 0 ][ 'allow_gid' ];
$arr [ 'deny_cid' ] = $user [ 0 ][ 'deny_cid' ];
$arr [ 'deny_gid' ] = $user [ 0 ][ 'deny_gid' ];
$i = item_store ( $arr );
if ( $i )
2015-08-09 20:39:11 +02:00
proc_run ( 'php' , " include/notifier.php " , " activity " , " $i " );
2011-11-12 13:06:33 +01:00
}
}
2011-08-12 12:01:11 +02:00
return ;
}
2013-12-27 01:58:21 +01:00
2011-08-19 07:01:35 +02:00
$ret = find_diaspora_person_by_handle ( $sender_handle );
2011-08-13 15:52:33 +02:00
2011-08-12 12:01:11 +02:00
if (( ! count ( $ret )) || ( $ret [ 'network' ] != NETWORK_DIASPORA )) {
logger ( 'diaspora_request: Cannot resolve diaspora handle ' . $sender_handle . ' for ' . $recipient_handle );
return ;
2011-08-13 15:52:33 +02:00
}
2011-08-19 07:03:58 +02:00
2011-09-22 13:11:39 +02:00
$batch = (( $ret [ 'batch' ]) ? $ret [ 'batch' ] : implode ( '/' , array_slice ( explode ( '/' , $ret [ 'url' ]), 0 , 3 )) . '/receive/public' );
2012-01-04 23:30:25 +01:00
2011-11-02 00:08:07 +01:00
$r = q ( " INSERT INTO `contact` (`uid`, `network`,`addr`,`created`,`url`,`nurl`,`batch`,`name`,`nick`,`photo`,`pubkey`,`notify`,`poll`,`blocked`,`priority`)
VALUES ( % d , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , '%s' , % d , % d ) " ,
2011-08-13 15:52:33 +02:00
intval ( $importer [ 'uid' ]),
dbesc ( $ret [ 'network' ]),
dbesc ( $ret [ 'addr' ]),
datetime_convert (),
dbesc ( $ret [ 'url' ]),
2011-11-02 00:08:07 +01:00
dbesc ( normalise_link ( $ret [ 'url' ])),
2011-09-22 13:11:39 +02:00
dbesc ( $batch ),
2011-08-13 15:52:33 +02:00
dbesc ( $ret [ 'name' ]),
dbesc ( $ret [ 'nick' ]),
dbesc ( $ret [ 'photo' ]),
dbesc ( $ret [ 'pubkey' ]),
dbesc ( $ret [ 'notify' ]),
dbesc ( $ret [ 'poll' ]),
1 ,
2
);
2014-04-04 10:52:53 +02:00
2011-08-13 15:52:33 +02:00
// find the contact record we just created
2011-08-16 09:52:34 +02:00
$contact_record = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $sender_handle );
2011-08-12 12:01:11 +02:00
2012-01-04 23:30:25 +01:00
if ( ! $contact_record ) {
logger ( 'diaspora_request: unable to locate newly created contact record.' );
return ;
}
2012-05-18 07:44:52 +02:00
$g = q ( " select def_gid from user where uid = %d limit 1 " ,
intval ( $importer [ 'uid' ])
);
if ( $g && intval ( $g [ 0 ][ 'def_gid' ])) {
require_once ( 'include/group.php' );
group_add_member ( $importer [ 'uid' ], '' , $contact_record [ 'id' ], $g [ 0 ][ 'def_gid' ]);
}
2012-01-04 23:30:25 +01:00
if ( $importer [ 'page-flags' ] == PAGE_NORMAL ) {
$hash = random_string () . ( string ) time (); // Generate a confirm_key
2014-04-04 10:52:53 +02:00
2011-08-19 06:50:41 +02:00
$ret = q ( " INSERT INTO `intro` ( `uid`, `contact-id`, `blocked`, `knowyou`, `note`, `hash`, `datetime` )
2011-08-19 07:03:58 +02:00
VALUES ( % d , % d , % d , % d , '%s' , '%s' , '%s' ) " ,
2011-08-13 15:52:33 +02:00
intval ( $importer [ 'uid' ]),
intval ( $contact_record [ 'id' ]),
0 ,
2011-08-19 07:03:58 +02:00
0 ,
2011-08-13 15:52:33 +02:00
dbesc ( t ( 'Sharing notification from Diaspora network' )),
dbesc ( $hash ),
dbesc ( datetime_convert ())
);
}
2012-01-04 23:30:25 +01:00
else {
// automatic friend approval
require_once ( 'include/Photo.php' );
2016-01-28 11:09:08 +01:00
update_contact_avatar ( $contact_record [ 'photo' ], $importer [ 'uid' ], $contact_record [ 'id' ]);
2014-04-04 10:52:53 +02:00
2014-04-22 00:58:17 +02:00
// technically they are sharing with us (CONTACT_IS_SHARING),
2012-01-04 23:30:25 +01:00
// but if our page-type is PAGE_COMMUNITY or PAGE_SOAPBOX
// we are going to change the relationship and make them a follower.
if ( $importer [ 'page-flags' ] == PAGE_FREELOVE )
$new_relation = CONTACT_IS_FRIEND ;
else
$new_relation = CONTACT_IS_FOLLOWER ;
2016-01-28 11:09:08 +01:00
$r = q ( " UPDATE `contact` SET `rel` = %d,
2015-05-20 21:35:57 +02:00
`name-date` = '%s' ,
`uri-date` = '%s' ,
`blocked` = 0 ,
`pending` = 0 ,
`writable` = 1
2014-03-09 09:19:14 +01:00
WHERE `id` = % d
2012-01-04 23:30:25 +01:00
" ,
intval ( $new_relation ),
dbesc ( datetime_convert ()),
dbesc ( datetime_convert ()),
intval ( $contact_record [ 'id' ])
);
2012-01-05 22:25:43 +01:00
2012-02-16 09:08:38 +01:00
$u = q ( " select * from user where uid = %d limit 1 " , intval ( $importer [ 'uid' ]));
2012-01-05 22:25:43 +01:00
if ( $u )
$ret = diaspora_share ( $u [ 0 ], $contact_record );
2012-01-04 23:30:25 +01:00
}
2011-08-19 07:03:58 +02:00
2011-08-13 15:52:33 +02:00
return ;
2011-08-10 14:10:48 +02:00
}
2015-08-30 19:07:30 +02:00
function diaspora_post_allow ( $importer , $contact , $is_comment = false ) {
2013-01-20 14:08:28 +01:00
// perhaps we were already sharing with this person. Now they're sharing with us.
// That makes us friends.
// Normally this should have handled by getting a request - but this could get lost
2015-10-10 11:06:18 +02:00
if ( $contact [ 'rel' ] == CONTACT_IS_FOLLOWER && in_array ( $importer [ 'page-flags' ], array ( PAGE_FREELOVE ))) {
2014-03-09 09:19:14 +01:00
q ( " UPDATE `contact` SET `rel` = %d, `writable` = 1 WHERE `id` = %d AND `uid` = %d " ,
2013-01-20 14:08:28 +01:00
intval ( CONTACT_IS_FRIEND ),
intval ( $contact [ 'id' ]),
intval ( $importer [ 'uid' ])
);
$contact [ 'rel' ] = CONTACT_IS_FRIEND ;
logger ( 'diaspora_post_allow: defining user ' . $contact [ " nick " ] . ' as friend' );
}
2012-12-29 11:35:41 +01:00
if (( $contact [ 'blocked' ]) || ( $contact [ 'readonly' ]) || ( $contact [ 'archive' ]))
2012-02-23 04:56:28 +01:00
return false ;
if ( $contact [ 'rel' ] == CONTACT_IS_SHARING || $contact [ 'rel' ] == CONTACT_IS_FRIEND )
return true ;
if ( $contact [ 'rel' ] == CONTACT_IS_FOLLOWER )
2015-08-30 19:07:30 +02:00
if (( $importer [ 'page-flags' ] == PAGE_COMMUNITY ) OR $is_comment )
2012-02-23 04:56:28 +01:00
return true ;
2015-08-09 14:28:59 +02:00
// Messages for the global users are always accepted
if ( $importer [ 'uid' ] == 0 )
return true ;
2012-02-23 04:56:28 +01:00
return false ;
}
2015-04-25 00:50:48 +02:00
function diaspora_is_redmatrix ( $url ) {
return ( strstr ( $url , " /channel/ " ));
}
2015-04-06 12:26:31 +02:00
function diaspora_plink ( $addr , $guid ) {
2015-12-27 03:54:20 +01:00
$r = q ( " SELECT `url`, `nick`, `network` FROM `fcontact` WHERE `addr`='%s' LIMIT 1 " , dbesc ( $addr ));
2015-04-06 12:26:31 +02:00
// Fallback
if ( ! $r )
return 'https://' . substr ( $addr , strpos ( $addr , '@' ) + 1 ) . '/posts/' . $guid ;
2015-05-25 19:07:59 +02:00
// Friendica contacts are often detected as Diaspora contacts in the "fcontact" table
// So we try another way as well.
$s = q ( " SELECT `network` FROM `gcontact` WHERE `nurl`='%s' LIMIT 1 " , dbesc ( normalise_link ( $r [ 0 ][ " url " ])));
if ( $s )
$r [ 0 ][ " network " ] = $s [ 0 ][ " network " ];
if ( $r [ 0 ][ " network " ] == NETWORK_DFRN )
return ( str_replace ( " /profile/ " . $r [ 0 ][ " nick " ] . " / " , " /display/ " . $guid , $r [ 0 ][ " url " ] . " / " ));
2015-04-25 00:50:48 +02:00
if ( diaspora_is_redmatrix ( $r [ 0 ][ " url " ]))
2015-04-06 12:26:31 +02:00
return $r [ 0 ][ " url " ] . " /?f=&mid= " . $guid ;
return 'https://' . substr ( $addr , strpos ( $addr , '@' ) + 1 ) . '/posts/' . $guid ;
}
2012-02-23 04:56:28 +01:00
2016-01-21 13:21:47 +01:00
function diaspora_repair_signature ( $signature , $handle = " " , $level = 1 ) {
2016-01-21 13:57:21 +01:00
if ( $signature == " " )
return ( $signature );
2016-01-21 13:21:47 +01:00
if ( base64_encode ( base64_decode ( base64_decode ( $signature ))) == base64_decode ( $signature )) {
$signature = base64_decode ( $signature );
logger ( " Repaired double encoded signature from Diaspora/Hubzilla handle " . $handle . " - level " . $level , LOGGER_DEBUG );
// Do a recursive call to be able to fix even multiple levels
if ( $level < 10 )
$signature = diaspora_repair_signature ( $signature , $handle , ++ $level );
}
return ( $signature );
}
2012-07-09 07:32:04 +02:00
function diaspora_post ( $importer , $xml , $msg ) {
2011-08-10 14:10:48 +02:00
2011-09-03 14:23:36 +02:00
$a = get_app ();
2011-08-14 01:39:59 +02:00
$guid = notags ( unxmlify ( $xml -> guid ));
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2011-08-16 08:19:17 +02:00
2012-07-09 07:32:04 +02:00
if ( $diaspora_handle != $msg [ 'author' ]) {
logger ( 'diaspora_post: Potential forgery. Message handle is not the same as envelope sender.' );
return 202 ;
}
2011-08-16 08:19:17 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $diaspora_handle );
2015-08-09 20:39:11 +02:00
if ( ! $contact ) {
logger ( 'diaspora_post: A Contact for handle ' . $diaspora_handle . ' and user ' . $importer [ 'uid' ] . ' was not found' );
return 203 ;
}
2011-08-16 08:19:17 +02:00
2015-08-30 19:07:30 +02:00
if ( ! diaspora_post_allow ( $importer , $contact , false )) {
2011-08-16 08:19:17 +02:00
logger ( 'diaspora_post: Ignoring this author.' );
2011-09-15 04:33:42 +02:00
return 202 ;
2011-08-16 08:19:17 +02:00
}
2011-08-14 01:39:59 +02:00
$message_id = $diaspora_handle . ':' . $guid ;
2015-09-06 07:34:51 +02:00
$r = q ( " SELECT `id` FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2011-08-14 01:39:59 +02:00
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
);
2011-08-30 07:50:41 +02:00
if ( count ( $r )) {
logger ( 'diaspora_post: message exists: ' . $guid );
2015-08-09 20:39:11 +02:00
return 208 ;
2011-08-30 07:50:41 +02:00
}
2011-08-14 01:39:59 +02:00
$created = unxmlify ( $xml -> created_at );
$private = (( unxmlify ( $xml -> public ) == 'false' ) ? 1 : 0 );
2011-08-26 01:37:27 +02:00
$body = diaspora2bb ( $xml -> raw_message );
2011-08-14 01:39:59 +02:00
$datarray = array ();
2011-10-06 05:48:00 +02:00
2015-05-25 13:27:45 +02:00
$datarray [ " object " ] = json_encode ( $xml );
if ( $xml -> photo -> remote_photo_path AND $xml -> photo -> remote_photo_name )
$datarray [ " object-type " ] = ACTIVITY_OBJ_PHOTO ;
else {
$datarray [ 'object-type' ] = ACTIVITY_OBJ_NOTE ;
// Add OEmbed and other information to the body
if ( ! diaspora_is_redmatrix ( $contact [ 'url' ]))
$body = add_page_info_to_body ( $body , false , true );
}
2011-10-06 05:48:00 +02:00
$str_tags = '' ;
2011-12-07 00:24:01 +01:00
$cnt = preg_match_all ( '/@\[url=(.*?)\[\/url\]/ism' , $body , $matches , PREG_SET_ORDER );
if ( $cnt ) {
foreach ( $matches as $mtch ) {
if ( strlen ( $str_tags ))
$str_tags .= ',' ;
2014-03-02 00:45:42 +01:00
$str_tags .= '@[url=' . $mtch [ 1 ] . '[/url]' ;
2011-12-07 00:24:01 +01:00
}
}
2015-04-06 12:26:31 +02:00
$plink = diaspora_plink ( $diaspora_handle , $guid );
2014-03-02 00:45:42 +01:00
2011-08-14 01:39:59 +02:00
$datarray [ 'uid' ] = $importer [ 'uid' ];
$datarray [ 'contact-id' ] = $contact [ 'id' ];
$datarray [ 'wall' ] = 0 ;
2014-07-15 08:50:49 +02:00
$datarray [ 'network' ] = NETWORK_DIASPORA ;
$datarray [ 'verb' ] = ACTIVITY_POST ;
2011-08-14 01:39:59 +02:00
$datarray [ 'guid' ] = $guid ;
2011-08-16 04:39:49 +02:00
$datarray [ 'uri' ] = $datarray [ 'parent-uri' ] = $message_id ;
2014-03-16 17:46:05 +01:00
$datarray [ 'changed' ] = $datarray [ 'created' ] = $datarray [ 'edited' ] = datetime_convert ( 'UTC' , 'UTC' , $created );
2011-08-16 06:01:44 +02:00
$datarray [ 'private' ] = $private ;
$datarray [ 'parent' ] = 0 ;
2014-03-02 00:45:42 +01:00
$datarray [ 'plink' ] = $plink ;
2011-08-14 04:03:59 +02:00
$datarray [ 'owner-name' ] = $contact [ 'name' ];
$datarray [ 'owner-link' ] = $contact [ 'url' ];
2012-11-10 23:19:32 +01:00
//$datarray['owner-avatar'] = $contact['thumb'];
$datarray [ 'owner-avatar' ] = (( x ( $contact , 'thumb' )) ? $contact [ 'thumb' ] : $contact [ 'photo' ]);
2011-08-14 01:39:59 +02:00
$datarray [ 'author-name' ] = $contact [ 'name' ];
$datarray [ 'author-link' ] = $contact [ 'url' ];
$datarray [ 'author-avatar' ] = $contact [ 'thumb' ];
2011-08-16 04:46:47 +02:00
$datarray [ 'body' ] = $body ;
2011-10-06 05:48:00 +02:00
$datarray [ 'tag' ] = $str_tags ;
2015-05-25 13:27:45 +02:00
if ( $xml -> provider_display_name )
2015-08-09 20:39:11 +02:00
$datarray [ " app " ] = unxmlify ( $xml -> provider_display_name );
2015-05-25 13:27:45 +02:00
else
$datarray [ 'app' ] = 'Diaspora' ;
2011-08-14 01:39:59 +02:00
2011-10-20 00:18:41 +02:00
// if empty content it might be a photo that hasn't arrived yet. If a photo arrives, we'll make it visible.
$datarray [ 'visible' ] = (( strlen ( $body )) ? 1 : 0 );
2015-02-03 00:29:21 +01:00
DiasporaFetchGuid ( $datarray );
2011-09-01 06:46:37 +02:00
$message_id = item_store ( $datarray );
2015-08-09 20:39:11 +02:00
logger ( " Stored item with message id " . $message_id , LOGGER_DEBUG );
return 201 ;
2011-08-14 01:39:59 +02:00
2011-08-10 14:10:48 +02:00
}
2015-02-03 00:29:21 +01:00
function DiasporaFetchGuid ( $item ) {
preg_replace_callback ( " & \ [url=/posts/([^ \ [ \ ]]*) \ ](.*) \ [ \ /url \ ]&Usi " ,
function ( $match ) use ( $item ){
return ( DiasporaFetchGuidSub ( $match , $item ));
}, $item [ " body " ]);
}
function DiasporaFetchGuidSub ( $match , $item ) {
$a = get_app ();
2015-05-09 19:39:47 +02:00
if ( ! diaspora_store_by_guid ( $match [ 1 ], $item [ " author-link " ]))
diaspora_store_by_guid ( $match [ 1 ], $item [ " owner-link " ]);
2015-02-03 00:29:21 +01:00
}
2015-05-09 19:39:47 +02:00
function diaspora_store_by_guid ( $guid , $server , $uid = 0 ) {
2015-02-01 21:54:44 +01:00
require_once ( " include/Contact.php " );
2015-08-09 20:39:11 +02:00
$serverparts = parse_url ( $server );
$server = $serverparts [ " scheme " ] . " :// " . $serverparts [ " host " ];
2015-05-09 19:39:47 +02:00
logger ( " Trying to fetch item " . $guid . " from " . $server , LOGGER_DEBUG );
2015-02-03 00:29:21 +01:00
2015-02-01 21:54:44 +01:00
$item = diaspora_fetch_message ( $guid , $server );
if ( ! $item )
return false ;
2015-05-09 19:39:47 +02:00
logger ( " Successfully fetched item " . $guid . " from " . $server , LOGGER_DEBUG );
2015-02-01 21:54:44 +01:00
$body = $item [ " body " ];
$str_tags = $item [ " tag " ];
$app = $item [ " app " ];
$created = $item [ " created " ];
$author = $item [ " author " ];
$guid = $item [ " guid " ];
$private = $item [ " private " ];
2015-05-25 13:27:45 +02:00
$object = $item [ " object " ];
$objecttype = $item [ " object-type " ];
2015-02-01 21:54:44 +01:00
$message_id = $author . ':' . $guid ;
2015-09-06 07:34:51 +02:00
$r = q ( " SELECT `id` FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2015-05-09 19:39:47 +02:00
intval ( $uid ),
2015-02-01 21:54:44 +01:00
dbesc ( $guid )
);
if ( count ( $r ))
2015-02-03 07:00:01 +01:00
return $r [ 0 ][ " id " ];
2015-02-01 21:54:44 +01:00
$person = find_diaspora_person_by_handle ( $author );
2015-09-05 12:11:14 +02:00
$contact_id = get_contact ( $person [ 'url' ], $uid );
$contacts = q ( " SELECT * FROM `contact` WHERE `id` = %d " , intval ( $contact_id ));
$importers = q ( " SELECT * FROM `user` WHERE `uid` = %d " , intval ( $uid ));
if ( $contacts AND $importers )
if ( ! diaspora_post_allow ( $importers [ 0 ], $contacts [ 0 ], false )) {
logger ( 'Ignoring author ' . $person [ 'url' ] . ' for uid ' . $uid );
return false ;
} else
logger ( 'Author ' . $person [ 'url' ] . ' is allowed for uid ' . $uid );
2015-08-09 20:39:11 +02:00
$datarray = array ();
2015-05-09 19:39:47 +02:00
$datarray [ 'uid' ] = $uid ;
2015-09-05 12:11:14 +02:00
$datarray [ 'contact-id' ] = $contact_id ;
2015-02-01 21:54:44 +01:00
$datarray [ 'wall' ] = 0 ;
$datarray [ 'network' ] = NETWORK_DIASPORA ;
$datarray [ 'guid' ] = $guid ;
$datarray [ 'uri' ] = $datarray [ 'parent-uri' ] = $message_id ;
$datarray [ 'changed' ] = $datarray [ 'created' ] = $datarray [ 'edited' ] = datetime_convert ( 'UTC' , 'UTC' , $created );
$datarray [ 'private' ] = $private ;
$datarray [ 'parent' ] = 0 ;
2015-04-06 12:26:31 +02:00
$datarray [ 'plink' ] = diaspora_plink ( $author , $guid );
2015-02-01 21:54:44 +01:00
$datarray [ 'author-name' ] = $person [ 'name' ];
$datarray [ 'author-link' ] = $person [ 'url' ];
$datarray [ 'author-avatar' ] = (( x ( $person , 'thumb' )) ? $person [ 'thumb' ] : $person [ 'photo' ]);
$datarray [ 'owner-name' ] = $datarray [ 'author-name' ];
$datarray [ 'owner-link' ] = $datarray [ 'author-link' ];
$datarray [ 'owner-avatar' ] = $datarray [ 'author-avatar' ];
$datarray [ 'body' ] = $body ;
$datarray [ 'tag' ] = $str_tags ;
$datarray [ 'app' ] = $app ;
$datarray [ 'visible' ] = (( strlen ( $body )) ? 1 : 0 );
2015-05-25 13:27:45 +02:00
$datarray [ 'object' ] = $object ;
$datarray [ 'object-type' ] = $objecttype ;
2015-02-01 21:54:44 +01:00
2015-05-09 19:39:47 +02:00
if ( $datarray [ 'contact-id' ] == 0 )
return false ;
2015-02-03 00:29:21 +01:00
DiasporaFetchGuid ( $datarray );
2015-02-01 21:54:44 +01:00
$message_id = item_store ( $datarray );
2015-12-25 23:17:34 +01:00
/// @TODO
/// Looking if there is some subscribe mechanism in Diaspora to get all comments for this post
2015-02-02 20:24:19 +01:00
2015-02-01 21:54:44 +01:00
return $message_id ;
}
2015-01-25 00:01:58 +01:00
function diaspora_fetch_message ( $guid , $server , $level = 0 ) {
if ( $level > 5 )
return false ;
$a = get_app ();
// This will not work if the server is not a Diaspora server
$source_url = $server . '/p/' . $guid . '.xml' ;
$x = fetch_url ( $source_url );
if ( ! $x )
return false ;
$x = str_replace ( array ( '<activity_streams-photo>' , '</activity_streams-photo>' ), array ( '<asphoto>' , '</asphoto>' ), $x );
$source_xml = parse_xml_string ( $x , false );
$item = array ();
$item [ " app " ] = 'Diaspora' ;
$item [ " guid " ] = $guid ;
$body = " " ;
if ( $source_xml -> post -> status_message -> created_at )
$item [ " created " ] = unxmlify ( $source_xml -> post -> status_message -> created_at );
if ( $source_xml -> post -> status_message -> provider_display_name )
$item [ " app " ] = unxmlify ( $source_xml -> post -> status_message -> provider_display_name );
if ( $source_xml -> post -> status_message -> diaspora_handle )
$item [ " author " ] = unxmlify ( $source_xml -> post -> status_message -> diaspora_handle );
if ( $source_xml -> post -> status_message -> guid )
$item [ " guid " ] = unxmlify ( $source_xml -> post -> status_message -> guid );
$item [ " private " ] = ( unxmlify ( $source_xml -> post -> status_message -> public ) == 'false' );
2015-05-25 13:27:45 +02:00
$item [ " object " ] = json_encode ( $source_xml -> post );
2015-01-25 00:01:58 +01:00
if ( strlen ( $source_xml -> post -> asphoto -> objectId ) && ( $source_xml -> post -> asphoto -> objectId != 0 ) && ( $source_xml -> post -> asphoto -> image_url )) {
2015-05-25 13:27:45 +02:00
$item [ " object-type " ] = ACTIVITY_OBJ_PHOTO ;
2015-01-25 00:01:58 +01:00
$body = '[url=' . notags ( unxmlify ( $source_xml -> post -> asphoto -> image_url )) . '][img]' . notags ( unxmlify ( $source_xml -> post -> asphoto -> objectId )) . '[/img][/url]' . " \n " ;
$body = scale_external_images ( $body , false );
} elseif ( $source_xml -> post -> asphoto -> image_url ) {
2015-05-25 13:27:45 +02:00
$item [ " object-type " ] = ACTIVITY_OBJ_PHOTO ;
2015-01-25 00:01:58 +01:00
$body = '[img]' . notags ( unxmlify ( $source_xml -> post -> asphoto -> image_url )) . '[/img]' . " \n " ;
$body = scale_external_images ( $body );
} elseif ( $source_xml -> post -> status_message ) {
$body = diaspora2bb ( $source_xml -> post -> status_message -> raw_message );
// Checking for embedded pictures
if ( $source_xml -> post -> status_message -> photo -> remote_photo_path AND
$source_xml -> post -> status_message -> photo -> remote_photo_name ) {
2015-05-25 13:27:45 +02:00
$item [ " object-type " ] = ACTIVITY_OBJ_PHOTO ;
2015-01-25 00:01:58 +01:00
$remote_photo_path = notags ( unxmlify ( $source_xml -> post -> status_message -> photo -> remote_photo_path ));
$remote_photo_name = notags ( unxmlify ( $source_xml -> post -> status_message -> photo -> remote_photo_name ));
$body = '[img]' . $remote_photo_path . $remote_photo_name . '[/img]' . " \n " . $body ;
logger ( 'embedded picture link found: ' . $body , LOGGER_DEBUG );
2015-05-25 13:27:45 +02:00
} else
$item [ " object-type " ] = ACTIVITY_OBJ_NOTE ;
2015-01-25 00:01:58 +01:00
$body = scale_external_images ( $body );
// Add OEmbed and other information to the body
2015-12-25 23:17:34 +01:00
/// @TODO It could be a repeated redmatrix item
/// Then we shouldn't add further data to it
2015-05-25 13:27:45 +02:00
if ( $item [ " object-type " ] == ACTIVITY_OBJ_NOTE )
$body = add_page_info_to_body ( $body , false , true );
2015-01-25 00:01:58 +01:00
} elseif ( $source_xml -> post -> reshare ) {
// Reshare of a reshare
return diaspora_fetch_message ( $source_xml -> post -> reshare -> root_guid , $server , ++ $level );
} else {
// Maybe it is a reshare of a photo that will be delivered at a later time (testing)
logger ( 'no content found: ' . print_r ( $source_xml , true ));
2015-05-03 21:25:03 +02:00
return false ;
2015-01-25 00:01:58 +01:00
}
2015-05-03 21:25:03 +02:00
if ( trim ( $body ) == " " )
2015-01-25 00:01:58 +01:00
return false ;
$item [ " tag " ] = '' ;
$item [ " body " ] = $body ;
2015-03-01 20:40:38 +01:00
2015-01-25 00:01:58 +01:00
return $item ;
}
2012-07-09 07:32:04 +02:00
function diaspora_reshare ( $importer , $xml , $msg ) {
2011-10-19 00:56:35 +02:00
2011-10-20 11:07:38 +02:00
logger ( 'diaspora_reshare: init: ' . print_r ( $xml , true ));
2011-10-19 00:56:35 +02:00
$a = get_app ();
$guid = notags ( unxmlify ( $xml -> guid ));
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2011-10-20 11:07:38 +02:00
2012-07-09 07:32:04 +02:00
if ( $diaspora_handle != $msg [ 'author' ]) {
logger ( 'diaspora_post: Potential forgery. Message handle is not the same as envelope sender.' );
return 202 ;
}
2011-10-19 00:56:35 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $diaspora_handle );
if ( ! $contact )
return ;
2015-08-30 19:07:30 +02:00
if ( ! diaspora_post_allow ( $importer , $contact , false )) {
2011-10-20 06:34:48 +02:00
logger ( 'diaspora_reshare: Ignoring this author: ' . $diaspora_handle . ' ' . print_r ( $xml , true ));
2011-10-19 00:56:35 +02:00
return 202 ;
}
$message_id = $diaspora_handle . ':' . $guid ;
2015-09-06 07:34:51 +02:00
$r = q ( " SELECT `id` FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2011-10-19 00:56:35 +02:00
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
);
if ( count ( $r )) {
logger ( 'diaspora_reshare: message exists: ' . $guid );
return ;
}
$orig_author = notags ( unxmlify ( $xml -> root_diaspora_id ));
$orig_guid = notags ( unxmlify ( $xml -> root_guid ));
2015-05-25 19:07:59 +02:00
$orig_url = $a -> get_baseurl () . " /display/ " . $orig_guid ;
2011-10-19 00:56:35 +02:00
2015-01-25 00:01:58 +01:00
$create_original_post = false ;
2011-10-20 12:56:10 +02:00
2015-01-25 00:01:58 +01:00
// Do we already have this item?
2015-05-25 19:07:59 +02:00
$r = q ( " SELECT `body`, `tag`, `app`, `created`, `plink`, `object`, `object-type`, `uri` FROM `item` WHERE `guid` = '%s' AND `visible` AND NOT `deleted` AND `body` != '' LIMIT 1 " ,
2015-01-25 00:01:58 +01:00
dbesc ( $orig_guid ),
dbesc ( NETWORK_DIASPORA )
);
if ( count ( $r )) {
2015-05-25 19:07:59 +02:00
logger ( 'reshared message ' . $orig_guid . " reshared by " . $guid . ' already exists on system.' );
2011-10-19 00:56:35 +02:00
2015-01-25 00:01:58 +01:00
// Maybe it is already a reshared item?
// Then refetch the content, since there can be many side effects with reshared posts from other networks or reshares from reshares
require_once ( 'include/api.php' );
if ( api_share_as_retweet ( $r [ 0 ]))
$r = array ();
2015-05-03 22:15:26 +02:00
else {
$body = $r [ 0 ][ " body " ];
$str_tags = $r [ 0 ][ " tag " ];
$app = $r [ 0 ][ " app " ];
$orig_created = $r [ 0 ][ " created " ];
2015-05-25 19:07:59 +02:00
$orig_plink = $r [ 0 ][ " plink " ];
$orig_uri = $r [ 0 ][ " uri " ];
2015-05-25 13:27:45 +02:00
$object = $r [ 0 ][ " object " ];
$objecttype = $r [ 0 ][ " object-type " ];
2015-05-03 22:15:26 +02:00
}
2011-10-28 11:50:00 +02:00
}
2014-03-23 18:50:28 +01:00
2015-01-25 00:01:58 +01:00
if ( ! count ( $r )) {
$body = " " ;
$str_tags = " " ;
$app = " " ;
2014-03-23 18:50:28 +01:00
2015-01-25 00:01:58 +01:00
$server = 'https://' . substr ( $orig_author , strpos ( $orig_author , '@' ) + 1 );
logger ( '1st try: reshared message ' . $orig_guid . " reshared by " . $guid . ' will be fetched from original server: ' . $server );
$item = diaspora_fetch_message ( $orig_guid , $server );
2014-03-23 18:50:28 +01:00
2015-01-25 00:01:58 +01:00
if ( ! $item ) {
$server = 'https://' . substr ( $diaspora_handle , strpos ( $diaspora_handle , '@' ) + 1 );
logger ( '2nd try: reshared message ' . $orig_guid . " reshared by " . $guid . " will be fetched from sharer's server: " . $server );
$item = diaspora_fetch_message ( $orig_guid , $server );
}
if ( ! $item ) {
$server = 'http://' . substr ( $orig_author , strpos ( $orig_author , '@' ) + 1 );
logger ( '3rd try: reshared message ' . $orig_guid . " reshared by " . $guid . ' will be fetched from original server: ' . $server );
$item = diaspora_fetch_message ( $orig_guid , $server );
}
if ( ! $item ) {
$server = 'http://' . substr ( $diaspora_handle , strpos ( $diaspora_handle , '@' ) + 1 );
logger ( '4th try: reshared message ' . $orig_guid . " reshared by " . $guid . " will be fetched from sharer's server: " . $server );
$item = diaspora_fetch_message ( $orig_guid , $server );
2014-03-23 18:50:28 +01:00
}
2015-01-25 00:01:58 +01:00
if ( $item ) {
$body = $item [ " body " ];
$str_tags = $item [ " tag " ];
$app = $item [ " app " ];
$orig_created = $item [ " created " ];
$orig_author = $item [ " author " ];
$orig_guid = $item [ " guid " ];
2015-05-25 19:07:59 +02:00
$orig_plink = diaspora_plink ( $orig_author , $orig_guid );
$orig_uri = $orig_author . ':' . $orig_guid ;
2015-02-01 10:26:35 +01:00
$create_original_post = ( $body != " " );
2015-05-25 13:27:45 +02:00
$object = $item [ " object " ];
$objecttype = $item [ " object-type " ];
2015-01-25 00:01:58 +01:00
}
2011-10-19 00:56:35 +02:00
}
2015-05-25 19:07:59 +02:00
$plink = diaspora_plink ( $diaspora_handle , $guid );
2011-10-19 00:56:35 +02:00
$person = find_diaspora_person_by_handle ( $orig_author );
$created = unxmlify ( $xml -> created_at );
$private = (( unxmlify ( $xml -> public ) == 'false' ) ? 1 : 0 );
$datarray = array ();
$datarray [ 'uid' ] = $importer [ 'uid' ];
$datarray [ 'contact-id' ] = $contact [ 'id' ];
$datarray [ 'wall' ] = 0 ;
2013-12-27 01:58:21 +01:00
$datarray [ 'network' ] = NETWORK_DIASPORA ;
2011-10-19 00:56:35 +02:00
$datarray [ 'guid' ] = $guid ;
$datarray [ 'uri' ] = $datarray [ 'parent-uri' ] = $message_id ;
2014-03-16 17:46:05 +01:00
$datarray [ 'changed' ] = $datarray [ 'created' ] = $datarray [ 'edited' ] = datetime_convert ( 'UTC' , 'UTC' , $created );
2011-10-19 00:56:35 +02:00
$datarray [ 'private' ] = $private ;
$datarray [ 'parent' ] = 0 ;
2014-03-02 00:45:42 +01:00
$datarray [ 'plink' ] = $plink ;
2011-10-19 00:56:35 +02:00
$datarray [ 'owner-name' ] = $contact [ 'name' ];
$datarray [ 'owner-link' ] = $contact [ 'url' ];
2012-11-10 23:19:32 +01:00
$datarray [ 'owner-avatar' ] = (( x ( $contact , 'thumb' )) ? $contact [ 'thumb' ] : $contact [ 'photo' ]);
2013-10-15 00:49:49 +02:00
if ( ! intval ( get_config ( 'system' , 'wall-to-wall_share' ))) {
2015-04-05 20:43:06 +02:00
$prefix = share_header ( $person [ 'name' ], $person [ 'url' ], (( x ( $person , 'thumb' )) ? $person [ 'thumb' ] : $person [ 'photo' ]), $orig_guid , $orig_created , $orig_url );
2012-11-17 16:27:49 +01:00
$datarray [ 'author-name' ] = $contact [ 'name' ];
$datarray [ 'author-link' ] = $contact [ 'url' ];
$datarray [ 'author-avatar' ] = $contact [ 'thumb' ];
2012-12-19 12:18:52 +01:00
$datarray [ 'body' ] = $prefix . $body . " [/share] " ;
2012-12-21 00:08:58 +01:00
} else {
// Let reshared messages look like wall-to-wall posts
$datarray [ 'author-name' ] = $person [ 'name' ];
$datarray [ 'author-link' ] = $person [ 'url' ];
$datarray [ 'author-avatar' ] = (( x ( $person , 'thumb' )) ? $person [ 'thumb' ] : $person [ 'photo' ]);
$datarray [ 'body' ] = $body ;
2012-11-17 16:27:49 +01:00
}
2015-05-25 13:27:45 +02:00
$datarray [ " object " ] = json_encode ( $xml );
$datarray [ 'object-type' ] = $objecttype ;
2011-10-19 00:56:35 +02:00
$datarray [ 'tag' ] = $str_tags ;
2015-01-25 00:01:58 +01:00
$datarray [ 'app' ] = $app ;
2011-10-19 00:56:35 +02:00
2014-03-23 18:50:28 +01:00
// if empty content it might be a photo that hasn't arrived yet. If a photo arrives, we'll make it visible. (testing)
$datarray [ 'visible' ] = (( strlen ( $body )) ? 1 : 0 );
2015-01-25 00:01:58 +01:00
// Store the original item of a reshare
if ( $create_original_post ) {
2015-02-01 10:26:35 +01:00
require_once ( " include/Contact.php " );
2015-01-25 00:01:58 +01:00
$datarray2 = $datarray ;
$datarray2 [ 'uid' ] = 0 ;
2015-02-01 10:26:35 +01:00
$datarray2 [ 'contact-id' ] = get_contact ( $person [ 'url' ], 0 );
2015-01-25 00:01:58 +01:00
$datarray2 [ 'guid' ] = $orig_guid ;
2015-05-26 23:33:38 +02:00
$datarray2 [ 'uri' ] = $datarray2 [ 'parent-uri' ] = $orig_uri ;
2015-04-05 20:43:06 +02:00
$datarray2 [ 'changed' ] = $datarray2 [ 'created' ] = $datarray2 [ 'edited' ] = $datarray2 [ 'commented' ] = $datarray2 [ 'received' ] = datetime_convert ( 'UTC' , 'UTC' , $orig_created );
2015-05-26 23:33:38 +02:00
$datarray2 [ 'parent' ] = 0 ;
2015-05-25 19:07:59 +02:00
$datarray2 [ 'plink' ] = $orig_plink ;
2015-01-25 00:01:58 +01:00
$datarray2 [ 'author-name' ] = $person [ 'name' ];
$datarray2 [ 'author-link' ] = $person [ 'url' ];
$datarray2 [ 'author-avatar' ] = (( x ( $person , 'thumb' )) ? $person [ 'thumb' ] : $person [ 'photo' ]);
$datarray2 [ 'owner-name' ] = $datarray2 [ 'author-name' ];
$datarray2 [ 'owner-link' ] = $datarray2 [ 'author-link' ];
$datarray2 [ 'owner-avatar' ] = $datarray2 [ 'author-avatar' ];
$datarray2 [ 'body' ] = $body ;
2015-05-25 13:27:45 +02:00
$datarray2 [ " object " ] = $object ;
2015-01-25 00:01:58 +01:00
2015-02-03 00:29:21 +01:00
DiasporaFetchGuid ( $datarray2 );
2015-01-25 00:01:58 +01:00
$message_id = item_store ( $datarray2 );
logger ( " Store original item " . $orig_guid . " under message id " . $message_id );
}
2011-10-19 00:56:35 +02:00
2015-02-03 00:29:21 +01:00
DiasporaFetchGuid ( $datarray );
2015-01-25 00:01:58 +01:00
$message_id = item_store ( $datarray );
2011-10-19 00:56:35 +02:00
return ;
}
2012-07-09 07:32:04 +02:00
function diaspora_asphoto ( $importer , $xml , $msg ) {
2011-10-18 13:28:36 +02:00
logger ( 'diaspora_asphoto called' );
2011-10-18 02:39:25 +02:00
$a = get_app ();
$guid = notags ( unxmlify ( $xml -> guid ));
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2012-07-09 07:32:04 +02:00
if ( $diaspora_handle != $msg [ 'author' ]) {
logger ( 'diaspora_post: Potential forgery. Message handle is not the same as envelope sender.' );
return 202 ;
}
2011-10-18 02:39:25 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $diaspora_handle );
if ( ! $contact )
return ;
2015-08-30 19:07:30 +02:00
if ( ! diaspora_post_allow ( $importer , $contact , false )) {
2011-10-18 02:39:25 +02:00
logger ( 'diaspora_asphoto: Ignoring this author.' );
return 202 ;
}
$message_id = $diaspora_handle . ':' . $guid ;
2015-09-06 07:34:51 +02:00
$r = q ( " SELECT `id` FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2011-10-18 02:39:25 +02:00
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
);
if ( count ( $r )) {
logger ( 'diaspora_asphoto: message exists: ' . $guid );
return ;
}
$created = unxmlify ( $xml -> created_at );
$private = (( unxmlify ( $xml -> public ) == 'false' ) ? 1 : 0 );
2011-10-29 01:13:54 +02:00
if ( strlen ( $xml -> objectId ) && ( $xml -> objectId != 0 ) && ( $xml -> image_url )) {
2011-10-19 09:29:43 +02:00
$body = '[url=' . notags ( unxmlify ( $xml -> image_url )) . '][img]' . notags ( unxmlify ( $xml -> objectId )) . '[/img][/url]' . " \n " ;
2012-02-25 23:22:51 +01:00
$body = scale_external_images ( $body , false );
2011-10-29 01:13:54 +02:00
}
elseif ( $xml -> image_url ) {
2011-10-19 09:29:43 +02:00
$body = '[img]' . notags ( unxmlify ( $xml -> image_url )) . '[/img]' . " \n " ;
2012-02-25 23:22:51 +01:00
$body = scale_external_images ( $body );
2011-10-29 01:13:54 +02:00
}
2011-10-18 02:39:25 +02:00
else {
logger ( 'diaspora_asphoto: no photo url found.' );
return ;
}
2015-04-06 12:26:31 +02:00
$plink = diaspora_plink ( $diaspora_handle , $guid );
2011-10-18 02:39:25 +02:00
2014-03-02 00:45:42 +01:00
$datarray = array ();
2013-12-27 01:58:21 +01:00
2011-10-18 02:39:25 +02:00
$datarray [ 'uid' ] = $importer [ 'uid' ];
$datarray [ 'contact-id' ] = $contact [ 'id' ];
$datarray [ 'wall' ] = 0 ;
2013-12-27 01:58:21 +01:00
$datarray [ 'network' ] = NETWORK_DIASPORA ;
2011-10-18 02:39:25 +02:00
$datarray [ 'guid' ] = $guid ;
$datarray [ 'uri' ] = $datarray [ 'parent-uri' ] = $message_id ;
2014-03-16 17:46:05 +01:00
$datarray [ 'changed' ] = $datarray [ 'created' ] = $datarray [ 'edited' ] = datetime_convert ( 'UTC' , 'UTC' , $created );
2011-10-18 02:39:25 +02:00
$datarray [ 'private' ] = $private ;
$datarray [ 'parent' ] = 0 ;
2014-03-02 00:45:42 +01:00
$datarray [ 'plink' ] = $plink ;
2011-10-18 02:39:25 +02:00
$datarray [ 'owner-name' ] = $contact [ 'name' ];
$datarray [ 'owner-link' ] = $contact [ 'url' ];
2012-11-10 23:19:32 +01:00
//$datarray['owner-avatar'] = $contact['thumb'];
$datarray [ 'owner-avatar' ] = (( x ( $contact , 'thumb' )) ? $contact [ 'thumb' ] : $contact [ 'photo' ]);
2011-10-18 02:39:25 +02:00
$datarray [ 'author-name' ] = $contact [ 'name' ];
$datarray [ 'author-link' ] = $contact [ 'url' ];
$datarray [ 'author-avatar' ] = $contact [ 'thumb' ];
$datarray [ 'body' ] = $body ;
2015-05-25 13:27:45 +02:00
$datarray [ " object " ] = json_encode ( $xml );
$datarray [ 'object-type' ] = ACTIVITY_OBJ_PHOTO ;
2013-12-27 01:58:21 +01:00
2011-10-18 02:39:25 +02:00
$datarray [ 'app' ] = 'Diaspora/Cubbi.es' ;
2015-02-03 00:29:21 +01:00
DiasporaFetchGuid ( $datarray );
2011-10-18 02:39:25 +02:00
$message_id = item_store ( $datarray );
2014-03-02 00:45:42 +01:00
//if($message_id) {
2014-03-09 09:19:14 +01:00
// q("update item set plink = '%s' where id = %d",
2014-03-02 00:45:42 +01:00
// dbesc($a->get_baseurl() . '/display/' . $importer['nickname'] . '/' . $message_id),
// intval($message_id)
// );
//}
2011-10-18 02:39:25 +02:00
return ;
}
2011-08-17 07:31:14 +02:00
function diaspora_comment ( $importer , $xml , $msg ) {
2011-08-17 13:24:26 +02:00
2011-09-03 14:23:36 +02:00
$a = get_app ();
2011-08-14 04:03:59 +02:00
$guid = notags ( unxmlify ( $xml -> guid ));
2011-08-17 13:24:26 +02:00
$parent_guid = notags ( unxmlify ( $xml -> parent_guid ));
2011-08-14 04:03:59 +02:00
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2011-08-17 13:24:26 +02:00
$target_type = notags ( unxmlify ( $xml -> target_type ));
$text = unxmlify ( $xml -> text );
$author_signature = notags ( unxmlify ( $xml -> author_signature ));
2011-08-16 08:19:17 +02:00
2011-08-17 13:24:26 +02:00
$parent_author_signature = (( $xml -> parent_author_signature ) ? notags ( unxmlify ( $xml -> parent_author_signature )) : '' );
2011-08-16 08:19:17 +02:00
2011-08-17 13:24:26 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $msg [ 'author' ]);
2011-08-30 07:50:41 +02:00
if ( ! $contact ) {
logger ( 'diaspora_comment: cannot find contact: ' . $msg [ 'author' ]);
2011-08-16 08:19:17 +02:00
return ;
2011-08-30 07:50:41 +02:00
}
2011-08-16 08:19:17 +02:00
2015-08-30 19:07:30 +02:00
if ( ! diaspora_post_allow ( $importer , $contact , true )) {
2011-08-16 08:19:17 +02:00
logger ( 'diaspora_comment: Ignoring this author.' );
2011-09-15 04:33:42 +02:00
return 202 ;
2011-08-16 08:19:17 +02:00
}
2011-08-27 02:52:24 +02:00
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
);
if ( count ( $r )) {
2011-08-30 07:50:41 +02:00
logger ( 'diaspora_comment: our comment just got relayed back to us (or there was a guid collision) : ' . $guid );
2011-08-27 02:52:24 +02:00
return ;
}
2011-08-17 13:24:26 +02:00
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2011-08-14 04:03:59 +02:00
intval ( $importer [ 'uid' ]),
2011-08-17 13:24:26 +02:00
dbesc ( $parent_guid )
2011-08-14 04:03:59 +02:00
);
2015-05-09 19:39:47 +02:00
if ( ! count ( $r )) {
$result = diaspora_store_by_guid ( $parent_guid , $contact [ 'url' ], $importer [ 'uid' ]);
if ( ! $result ) {
$person = find_diaspora_person_by_handle ( $diaspora_handle );
$result = diaspora_store_by_guid ( $parent_guid , $person [ 'url' ], $importer [ 'uid' ]);
}
if ( $result ) {
logger ( " Fetched missing item " . $parent_guid . " - result: " . $result , LOGGER_DEBUG );
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $parent_guid )
);
}
}
2011-08-17 13:24:26 +02:00
if ( ! count ( $r )) {
2011-08-19 11:53:44 +02:00
logger ( 'diaspora_comment: parent item not found: parent: ' . $parent_guid . ' item: ' . $guid );
2011-08-14 04:03:59 +02:00
return ;
2011-08-17 13:24:26 +02:00
}
$parent_item = $r [ 0 ];
2011-08-14 04:03:59 +02:00
2011-08-17 13:24:26 +02:00
2012-06-09 21:54:21 +02:00
/* How Diaspora performs comment signature checking :
2011-08-17 13:24:26 +02:00
2012-06-09 21:54:21 +02:00
- If an item has been sent by the comment author to the top - level post owner to relay on
to the rest of the contacts on the top - level post , the top - level post owner should check
the author_signature , then create a parent_author_signature before relaying the comment on
- If an item has been relayed on by the top - level post owner , the contacts who receive it
check only the parent_author_signature . Basically , they trust that the top - level post
owner has already verified the authenticity of anything he / she sends out
- In either case , the signature that get checked is the signature created by the person
who sent the salmon
*/
2011-08-17 13:24:26 +02:00
2012-06-09 21:54:21 +02:00
$signed_data = $guid . ';' . $parent_guid . ';' . $text . ';' . $diaspora_handle ;
$key = $msg [ 'key' ];
if ( $parent_author_signature ) {
// If a parent_author_signature exists, then we've received the comment
// relayed from the top-level post owner. There's no need to check the
// author_signature if the parent_author_signature is valid
$parent_author_signature = base64_decode ( $parent_author_signature );
if ( ! rsa_verify ( $signed_data , $parent_author_signature , $key , 'sha256' )) {
logger ( 'diaspora_comment: top-level owner verification failed.' );
2011-08-17 13:24:26 +02:00
return ;
}
2016-01-21 13:38:30 +01:00
}
else {
2012-06-09 21:54:21 +02:00
// If there's no parent_author_signature, then we've received the comment
// from the comment creator. In that case, the person is commenting on
// our post, so he/she must be a contact of ours and his/her public key
// should be in $msg['key']
2011-08-17 13:24:26 +02:00
2012-06-09 21:54:21 +02:00
$author_signature = base64_decode ( $author_signature );
2011-08-17 13:24:26 +02:00
2012-06-09 21:54:21 +02:00
if ( ! rsa_verify ( $signed_data , $author_signature , $key , 'sha256' )) {
logger ( 'diaspora_comment: comment author verification failed.' );
2011-08-21 05:54:03 +02:00
return ;
2011-08-17 13:24:26 +02:00
}
}
// Phew! Everything checks out. Now create an item.
2012-06-16 18:41:25 +02:00
// Find the original comment author information.
// We need this to make sure we display the comment author
// information (name and avatar) correctly.
2012-06-10 16:41:23 +02:00
if ( strcasecmp ( $diaspora_handle , $msg [ 'author' ]) == 0 )
$person = $contact ;
else {
2014-03-16 17:46:05 +01:00
$person = find_diaspora_person_by_handle ( $diaspora_handle );
2012-06-10 16:41:23 +02:00
if ( ! is_array ( $person )) {
logger ( 'diaspora_comment: unable to find author details' );
return ;
}
}
2015-12-01 15:36:32 +01:00
// Fetch the contact id - if we know this contact
$r = q ( " SELECT `id`, `network` FROM `contact` WHERE `nurl` = '%s' AND `uid` = %d LIMIT 1 " ,
dbesc ( normalise_link ( $person [ 'url' ])), intval ( $importer [ 'uid' ]));
if ( $r ) {
$cid = $r [ 0 ][ 'id' ];
$network = $r [ 0 ][ 'network' ];
} else {
$cid = $contact [ 'id' ];
$network = NETWORK_DIASPORA ;
}
2011-08-26 01:37:27 +02:00
$body = diaspora2bb ( $text );
2011-08-17 13:24:26 +02:00
$message_id = $diaspora_handle . ':' . $guid ;
$datarray = array ();
2011-10-06 05:48:00 +02:00
2011-08-17 13:24:26 +02:00
$datarray [ 'uid' ] = $importer [ 'uid' ];
2015-12-01 15:36:32 +01:00
$datarray [ 'contact-id' ] = $cid ;
2012-07-13 02:53:51 +02:00
$datarray [ 'type' ] = 'remote-comment' ;
2011-08-17 13:24:26 +02:00
$datarray [ 'wall' ] = $parent_item [ 'wall' ];
2015-12-01 15:36:32 +01:00
$datarray [ 'network' ] = $network ;
2014-07-15 08:50:49 +02:00
$datarray [ 'verb' ] = ACTIVITY_POST ;
2011-08-17 13:24:26 +02:00
$datarray [ 'gravity' ] = GRAVITY_COMMENT ;
$datarray [ 'guid' ] = $guid ;
$datarray [ 'uri' ] = $message_id ;
$datarray [ 'parent-uri' ] = $parent_item [ 'uri' ];
// No timestamps for comments? OK, we'll the use current time.
2014-03-16 17:46:05 +01:00
$datarray [ 'changed' ] = $datarray [ 'created' ] = $datarray [ 'edited' ] = datetime_convert ();
2011-08-17 13:24:26 +02:00
$datarray [ 'private' ] = $parent_item [ 'private' ];
2011-10-06 04:16:05 +02:00
$datarray [ 'owner-name' ] = $parent_item [ 'owner-name' ];
$datarray [ 'owner-link' ] = $parent_item [ 'owner-link' ];
$datarray [ 'owner-avatar' ] = $parent_item [ 'owner-avatar' ];
2011-08-17 13:24:26 +02:00
$datarray [ 'author-name' ] = $person [ 'name' ];
$datarray [ 'author-link' ] = $person [ 'url' ];
$datarray [ 'author-avatar' ] = (( x ( $person , 'thumb' )) ? $person [ 'thumb' ] : $person [ 'photo' ]);
$datarray [ 'body' ] = $body ;
2015-05-25 13:27:45 +02:00
$datarray [ " object " ] = json_encode ( $xml );
$datarray [ " object-type " ] = ACTIVITY_OBJ_COMMENT ;
2011-10-26 04:59:57 +02:00
// We can't be certain what the original app is if the message is relayed.
2013-12-27 01:58:21 +01:00
if (( $parent_item [ 'origin' ]) && ( ! $parent_author_signature ))
2011-10-26 04:59:57 +02:00
$datarray [ 'app' ] = 'Diaspora' ;
2011-08-17 13:24:26 +02:00
2015-02-03 00:29:21 +01:00
DiasporaFetchGuid ( $datarray );
2011-08-18 14:08:39 +02:00
$message_id = item_store ( $datarray );
2015-04-21 10:14:20 +02:00
$datarray [ 'id' ] = $message_id ;
2014-07-22 00:36:20 +02:00
//if($message_id) {
//q("update item set plink = '%s' where id = %d",
// //dbesc($a->get_baseurl() . '/display/' . $importer['nickname'] . '/' . $message_id),
// dbesc($a->get_baseurl().'/display/'.$datarray['guid']),
// intval($message_id)
//);
//}
2011-09-01 06:46:37 +02:00
2016-01-21 13:21:47 +01:00
// If we are the origin of the parent we store the original signature and notify our followers
if ( $parent_item [ 'origin' ]) {
$author_signature_base64 = base64_encode ( $author_signature );
$author_signature_base64 = diaspora_repair_signature ( $author_signature_base64 , $diaspora_handle );
2011-08-18 14:08:39 +02:00
q ( " insert into sign (`iid`,`signed_text`,`signature`,`signer`) values (%d,'%s','%s','%s') " ,
intval ( $message_id ),
2012-07-10 04:35:59 +02:00
dbesc ( $signed_data ),
2016-01-21 13:21:47 +01:00
dbesc ( $author_signature_base64 ),
2011-08-18 14:08:39 +02:00
dbesc ( $diaspora_handle )
);
2011-08-17 13:24:26 +02:00
2016-01-21 13:21:47 +01:00
// notify others
2012-08-10 06:06:18 +02:00
proc_run ( 'php' , 'include/notifier.php' , 'comment-import' , $message_id );
2011-08-24 13:42:28 +02:00
}
2012-03-16 13:19:29 +01:00
2011-08-17 13:24:26 +02:00
return ;
2011-08-10 14:10:48 +02:00
}
2011-11-28 02:41:23 +01:00
function diaspora_conversation ( $importer , $xml , $msg ) {
$a = get_app ();
$guid = notags ( unxmlify ( $xml -> guid ));
$subject = notags ( unxmlify ( $xml -> subject ));
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
$participant_handles = notags ( unxmlify ( $xml -> participant_handles ));
$created_at = datetime_convert ( 'UTC' , 'UTC' , notags ( unxmlify ( $xml -> created_at )));
$parent_uri = $diaspora_handle . ':' . $guid ;
2015-08-09 14:28:59 +02:00
2011-11-28 02:41:23 +01:00
$messages = $xml -> message ;
if ( ! count ( $messages )) {
logger ( 'diaspora_conversation: empty conversation' );
return ;
}
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $msg [ 'author' ]);
if ( ! $contact ) {
logger ( 'diaspora_conversation: cannot find contact: ' . $msg [ 'author' ]);
return ;
}
2015-08-09 14:28:59 +02:00
if (( $contact [ 'rel' ] == CONTACT_IS_FOLLOWER ) || ( $contact [ 'blocked' ]) || ( $contact [ 'readonly' ])) {
2011-11-28 02:41:23 +01:00
logger ( 'diaspora_conversation: Ignoring this author.' );
return 202 ;
}
2011-11-29 10:19:09 +01:00
$conversation = null ;
$c = q ( " select * from conv where uid = %d and guid = '%s' limit 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
);
if ( count ( $c ))
$conversation = $c [ 0 ];
else {
2011-12-06 09:16:13 +01:00
$r = q ( " insert into conv (uid,guid,creator,created,updated,subject,recips) values(%d, '%s', '%s', '%s', '%s', '%s', '%s') " ,
2011-11-29 10:19:09 +01:00
intval ( $importer [ 'uid' ]),
dbesc ( $guid ),
2011-12-06 09:16:13 +01:00
dbesc ( $diaspora_handle ),
dbesc ( datetime_convert ( 'UTC' , 'UTC' , $created_at )),
dbesc ( datetime_convert ()),
dbesc ( $subject ),
2011-11-29 10:19:09 +01:00
dbesc ( $participant_handles )
);
if ( $r )
$c = q ( " select * from conv where uid = %d and guid = '%s' limit 1 " ,
2015-08-09 20:39:11 +02:00
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
);
2011-11-29 10:19:09 +01:00
if ( count ( $c ))
2015-08-09 20:39:11 +02:00
$conversation = $c [ 0 ];
2011-11-29 10:19:09 +01:00
}
if ( ! $conversation ) {
logger ( 'diaspora_conversation: unable to create conversation.' );
return ;
}
2011-12-05 02:26:55 +01:00
foreach ( $messages as $mesg ) {
2011-11-29 10:19:09 +01:00
$reply = 0 ;
2011-12-05 02:26:55 +01:00
$msg_guid = notags ( unxmlify ( $mesg -> guid ));
$msg_parent_guid = notags ( unxmlify ( $mesg -> parent_guid ));
$msg_parent_author_signature = notags ( unxmlify ( $mesg -> parent_author_signature ));
$msg_author_signature = notags ( unxmlify ( $mesg -> author_signature ));
$msg_text = unxmlify ( $mesg -> text );
$msg_created_at = datetime_convert ( 'UTC' , 'UTC' , notags ( unxmlify ( $mesg -> created_at )));
$msg_diaspora_handle = notags ( unxmlify ( $mesg -> diaspora_handle ));
$msg_conversation_guid = notags ( unxmlify ( $mesg -> conversation_guid ));
2011-11-28 02:41:23 +01:00
if ( $msg_conversation_guid != $guid ) {
logger ( 'diaspora_conversation: message conversation guid does not belong to the current conversation. ' . $xml );
continue ;
}
$body = diaspora2bb ( $msg_text );
$message_id = $msg_diaspora_handle . ':' . $msg_guid ;
2011-12-05 02:26:55 +01:00
$author_signed_data = $msg_guid . ';' . $msg_parent_guid . ';' . $msg_text . ';' . unxmlify ( $mesg -> created_at ) . ';' . $msg_diaspora_handle . ';' . $msg_conversation_guid ;
$author_signature = base64_decode ( $msg_author_signature );
if ( strcasecmp ( $msg_diaspora_handle , $msg [ 'author' ]) == 0 ) {
$person = $contact ;
$key = $msg [ 'key' ];
}
2011-11-28 02:41:23 +01:00
else {
2011-12-05 02:26:55 +01:00
$person = find_diaspora_person_by_handle ( $msg_diaspora_handle );
if ( is_array ( $person ) && x ( $person , 'pubkey' ))
$key = $person [ 'pubkey' ];
else {
logger ( 'diaspora_conversation: unable to find author details' );
continue ;
}
}
if ( ! rsa_verify ( $author_signed_data , $author_signature , $key , 'sha256' )) {
logger ( 'diaspora_conversation: verification failed.' );
2011-11-28 02:41:23 +01:00
continue ;
}
2011-12-05 02:26:55 +01:00
if ( $msg_parent_author_signature ) {
$owner_signed_data = $msg_guid . ';' . $msg_parent_guid . ';' . $msg_text . ';' . unxmlify ( $mesg -> created_at ) . ';' . $msg_diaspora_handle . ';' . $msg_conversation_guid ;
$parent_author_signature = base64_decode ( $msg_parent_author_signature );
$key = $msg [ 'key' ];
if ( ! rsa_verify ( $owner_signed_data , $parent_author_signature , $key , 'sha256' )) {
logger ( 'diaspora_conversation: owner verification failed.' );
continue ;
}
}
2011-11-28 02:41:23 +01:00
$r = q ( " select id from mail where `uri` = '%s' limit 1 " ,
dbesc ( $message_id )
);
if ( count ( $r )) {
logger ( 'diaspora_conversation: duplicate message already delivered.' , LOGGER_DEBUG );
continue ;
}
2011-12-07 04:15:42 +01:00
q ( " insert into mail ( `uid`, `guid`, `convid`, `from-name`,`from-photo`,`from-url`,`contact-id`,`title`,`body`,`seen`,`reply`,`uri`,`parent-uri`,`created`) values ( %d, '%s', %d, '%s', '%s', '%s', %d, '%s', '%s', %d, %d, '%s','%s','%s') " ,
2011-11-28 02:41:23 +01:00
intval ( $importer [ 'uid' ]),
2011-11-29 10:19:09 +01:00
dbesc ( $msg_guid ),
intval ( $conversation [ 'id' ]),
2011-11-28 02:41:23 +01:00
dbesc ( $person [ 'name' ]),
dbesc ( $person [ 'photo' ]),
dbesc ( $person [ 'url' ]),
2015-05-03 21:25:03 +02:00
intval ( $contact [ 'id' ]),
2011-11-28 02:41:23 +01:00
dbesc ( $subject ),
dbesc ( $body ),
0 ,
2011-12-07 04:15:42 +01:00
0 ,
2011-11-28 02:41:23 +01:00
dbesc ( $message_id ),
dbesc ( $parent_uri ),
dbesc ( $msg_created_at )
2014-03-09 09:19:14 +01:00
);
2011-11-28 02:41:23 +01:00
2014-03-09 09:19:14 +01:00
q ( " update conv set updated = '%s' where id = %d " ,
2011-12-06 09:16:13 +01:00
dbesc ( datetime_convert ()),
intval ( $conversation [ 'id' ])
2014-03-09 09:19:14 +01:00
);
2011-12-28 00:49:47 +01:00
2014-03-09 09:19:14 +01:00
notification ( array (
2011-12-28 00:49:47 +01:00
'type' => NOTIFY_MAIL ,
'notify_flags' => $importer [ 'notify-flags' ],
'language' => $importer [ 'language' ],
'to_name' => $importer [ 'username' ],
'to_email' => $importer [ 'email' ],
2015-05-03 21:25:03 +02:00
'uid' => $importer [ 'uid' ],
2011-12-28 00:49:47 +01:00
'item' => array ( 'subject' => $subject , 'body' => $body ),
'source_name' => $person [ 'name' ],
'source_link' => $person [ 'url' ],
'source_photo' => $person [ 'thumb' ],
2012-01-04 05:26:20 +01:00
'verb' => ACTIVITY_POST ,
'otype' => 'mail'
2011-12-28 00:49:47 +01:00
));
2014-04-22 00:58:17 +02:00
}
2011-11-28 02:41:23 +01:00
return ;
}
2011-12-07 04:15:42 +01:00
function diaspora_message ( $importer , $xml , $msg ) {
$a = get_app ();
$msg_guid = notags ( unxmlify ( $xml -> guid ));
$msg_parent_guid = notags ( unxmlify ( $xml -> parent_guid ));
$msg_parent_author_signature = notags ( unxmlify ( $xml -> parent_author_signature ));
$msg_author_signature = notags ( unxmlify ( $xml -> author_signature ));
$msg_text = unxmlify ( $xml -> text );
$msg_created_at = datetime_convert ( 'UTC' , 'UTC' , notags ( unxmlify ( $xml -> created_at )));
$msg_diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
$msg_conversation_guid = notags ( unxmlify ( $xml -> conversation_guid ));
2015-08-13 22:49:57 +02:00
$parent_uri = $msg_diaspora_handle . ':' . $msg_parent_guid ;
2015-08-09 14:28:59 +02:00
2011-12-07 04:15:42 +01:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $msg_diaspora_handle );
if ( ! $contact ) {
logger ( 'diaspora_message: cannot find contact: ' . $msg_diaspora_handle );
return ;
}
2015-08-09 14:28:59 +02:00
if (( $contact [ 'rel' ] == CONTACT_IS_FOLLOWER ) || ( $contact [ 'blocked' ]) || ( $contact [ 'readonly' ])) {
2011-12-07 04:15:42 +01:00
logger ( 'diaspora_message: Ignoring this author.' );
return 202 ;
}
$conversation = null ;
$c = q ( " select * from conv where uid = %d and guid = '%s' limit 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $msg_conversation_guid )
);
if ( count ( $c ))
$conversation = $c [ 0 ];
else {
logger ( 'diaspora_message: conversation not available.' );
return ;
}
$reply = 0 ;
2014-03-16 17:46:05 +01:00
2011-12-07 04:15:42 +01:00
$body = diaspora2bb ( $msg_text );
$message_id = $msg_diaspora_handle . ':' . $msg_guid ;
$author_signed_data = $msg_guid . ';' . $msg_parent_guid . ';' . $msg_text . ';' . unxmlify ( $xml -> created_at ) . ';' . $msg_diaspora_handle . ';' . $msg_conversation_guid ;
2011-11-28 02:41:23 +01:00
2011-12-07 04:15:42 +01:00
$author_signature = base64_decode ( $msg_author_signature );
2011-11-28 02:41:23 +01:00
2015-10-08 00:25:55 +02:00
$person = find_diaspora_person_by_handle ( $msg_diaspora_handle );
2011-12-07 04:15:42 +01:00
if ( is_array ( $person ) && x ( $person , 'pubkey' ))
$key = $person [ 'pubkey' ];
else {
logger ( 'diaspora_message: unable to find author details' );
return ;
}
if ( ! rsa_verify ( $author_signed_data , $author_signature , $key , 'sha256' )) {
logger ( 'diaspora_message: verification failed.' );
return ;
}
$r = q ( " select id from mail where `uri` = '%s' and uid = %d limit 1 " ,
dbesc ( $message_id ),
intval ( $importer [ 'uid' ])
);
if ( count ( $r )) {
logger ( 'diaspora_message: duplicate message already delivered.' , LOGGER_DEBUG );
return ;
}
q ( " insert into mail ( `uid`, `guid`, `convid`, `from-name`,`from-photo`,`from-url`,`contact-id`,`title`,`body`,`seen`,`reply`,`uri`,`parent-uri`,`created`) values ( %d, '%s', %d, '%s', '%s', '%s', %d, '%s', '%s', %d, %d, '%s','%s','%s') " ,
intval ( $importer [ 'uid' ]),
dbesc ( $msg_guid ),
intval ( $conversation [ 'id' ]),
dbesc ( $person [ 'name' ]),
dbesc ( $person [ 'photo' ]),
dbesc ( $person [ 'url' ]),
2015-05-25 13:27:45 +02:00
intval ( $contact [ 'id' ]),
2011-12-07 04:15:42 +01:00
dbesc ( $conversation [ 'subject' ]),
dbesc ( $body ),
0 ,
1 ,
dbesc ( $message_id ),
dbesc ( $parent_uri ),
dbesc ( $msg_created_at )
2014-03-09 09:19:14 +01:00
);
2011-12-07 04:15:42 +01:00
2014-03-09 09:19:14 +01:00
q ( " update conv set updated = '%s' where id = %d " ,
2011-12-07 04:15:42 +01:00
dbesc ( datetime_convert ()),
intval ( $conversation [ 'id' ])
2014-03-09 09:19:14 +01:00
);
2011-12-07 04:15:42 +01:00
return ;
}
2011-11-28 02:41:23 +01:00
2015-10-08 00:25:55 +02:00
function diaspora_participation ( $importer , $xml ) {
logger ( " Unsupported message type 'participation' " . print_r ( $xml , true ));
}
2011-11-28 02:41:23 +01:00
2013-01-12 14:52:15 +01:00
function diaspora_photo ( $importer , $xml , $msg , $attempt = 1 ) {
2011-08-24 03:17:35 +02:00
2011-09-03 14:23:36 +02:00
$a = get_app ();
2012-01-23 00:25:29 +01:00
logger ( 'diaspora_photo: init' , LOGGER_DEBUG );
2011-08-24 03:17:35 +02:00
$remote_photo_path = notags ( unxmlify ( $xml -> remote_photo_path ));
$remote_photo_name = notags ( unxmlify ( $xml -> remote_photo_name ));
$status_message_guid = notags ( unxmlify ( $xml -> status_message_guid ));
$guid = notags ( unxmlify ( $xml -> guid ));
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
$public = notags ( unxmlify ( $xml -> public ));
$created_at = notags ( unxmlify ( $xml_created_at ));
2012-01-23 00:25:29 +01:00
logger ( 'diaspora_photo: status_message_guid: ' . $status_message_guid , LOGGER_DEBUG );
2011-08-24 03:17:35 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $msg [ 'author' ]);
2012-01-23 00:25:29 +01:00
if ( ! $contact ) {
logger ( 'diaspora_photo: contact record not found: ' . $msg [ 'author' ] . ' handle: ' . $diaspora_handle );
2011-08-24 03:17:35 +02:00
return ;
2012-01-23 00:25:29 +01:00
}
2011-08-24 03:17:35 +02:00
2015-08-30 19:07:30 +02:00
if ( ! diaspora_post_allow ( $importer , $contact , false )) {
2011-08-24 03:17:35 +02:00
logger ( 'diaspora_photo: Ignoring this author.' );
2011-09-15 04:33:42 +02:00
return 202 ;
2011-08-24 03:17:35 +02:00
}
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $status_message_guid )
);
2015-05-09 19:39:47 +02:00
/* deactivated by now since it can lead to multiplicated pictures in posts .
if ( ! count ( $r )) {
$result = diaspora_store_by_guid ( $status_message_guid , $contact [ 'url' ], $importer [ 'uid' ]);
if ( ! $result ) {
$person = find_diaspora_person_by_handle ( $diaspora_handle );
$result = diaspora_store_by_guid ( $status_message_guid , $person [ 'url' ], $importer [ 'uid' ]);
}
if ( $result ) {
logger ( " Fetched missing item " . $status_message_guid . " - result: " . $result , LOGGER_DEBUG );
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $status_message_guid )
);
}
}
*/
2015-05-03 21:25:03 +02:00
if ( ! count ( $r )) {
2013-01-12 14:52:15 +01:00
if ( $attempt <= 3 ) {
q ( " INSERT INTO dsprphotoq (uid, msg, attempt) VALUES (%d, '%s', %d) " ,
intval ( $importer [ 'uid' ]),
dbesc ( serialize ( $msg )),
intval ( $attempt + 1 )
);
}
2015-05-03 21:25:03 +02:00
2013-01-12 14:52:15 +01:00
logger ( 'diaspora_photo: attempt = ' . $attempt . '; status message not found: ' . $status_message_guid . ' for photo: ' . $guid );
2011-08-24 03:17:35 +02:00
return ;
}
2012-01-23 00:25:29 +01:00
2011-08-24 03:17:35 +02:00
$parent_item = $r [ 0 ];
$link_text = '[img]' . $remote_photo_path . $remote_photo_name . '[/img]' . " \n " ;
2012-06-20 04:45:24 +02:00
$link_text = scale_external_images ( $link_text , true ,
2015-08-09 20:39:11 +02:00
array ( $remote_photo_name , 'scaled_full_' . $remote_photo_name ));
2011-10-29 01:13:54 +02:00
2011-10-06 06:04:53 +02:00
if ( strpos ( $parent_item [ 'body' ], $link_text ) === false ) {
2016-01-14 21:56:37 +01:00
$parent_item [ 'body' ] = $link_text . $parent_item [ 'body' ];
2015-03-21 20:23:47 +01:00
$r = q ( " UPDATE `item` SET `body` = '%s', `visible` = 1 WHERE `id` = %d AND `uid` = %d " ,
2016-01-14 21:56:37 +01:00
dbesc ( $parent_item [ 'body' ]),
2011-10-06 06:02:00 +02:00
intval ( $parent_item [ 'id' ]),
intval ( $parent_item [ 'uid' ])
);
2016-01-14 21:56:37 +01:00
put_item_in_cache ( $parent_item , true );
2015-03-21 20:23:47 +01:00
update_thread ( $parent_item [ 'id' ]);
2011-10-06 06:02:00 +02:00
}
2011-08-24 03:17:35 +02:00
return ;
}
2011-08-17 07:31:14 +02:00
function diaspora_like ( $importer , $xml , $msg ) {
2011-08-10 14:10:48 +02:00
2011-08-21 03:08:43 +02:00
$a = get_app ();
2011-08-14 04:03:59 +02:00
$guid = notags ( unxmlify ( $xml -> guid ));
2011-08-17 07:31:14 +02:00
$parent_guid = notags ( unxmlify ( $xml -> parent_guid ));
2011-08-14 04:03:59 +02:00
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2011-08-17 07:31:14 +02:00
$target_type = notags ( unxmlify ( $xml -> target_type ));
$positive = notags ( unxmlify ( $xml -> positive ));
2011-08-17 13:24:26 +02:00
$author_signature = notags ( unxmlify ( $xml -> author_signature ));
2011-08-16 08:19:17 +02:00
2011-08-17 07:31:14 +02:00
$parent_author_signature = (( $xml -> parent_author_signature ) ? notags ( unxmlify ( $xml -> parent_author_signature )) : '' );
2011-08-16 08:19:17 +02:00
2011-08-17 13:24:26 +02:00
// likes on comments not supported here and likes on photos not supported by Diaspora
2012-07-13 02:53:51 +02:00
// if($target_type !== 'Post')
// return;
2011-08-17 07:31:14 +02:00
2011-08-17 13:24:26 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $msg [ 'author' ]);
2011-08-30 07:50:41 +02:00
if ( ! $contact ) {
logger ( 'diaspora_like: cannot find contact: ' . $msg [ 'author' ]);
2011-08-16 08:19:17 +02:00
return ;
2011-08-30 07:50:41 +02:00
}
2011-08-16 08:19:17 +02:00
2015-08-30 19:07:30 +02:00
if ( ! diaspora_post_allow ( $importer , $contact , false )) {
2011-08-16 08:19:17 +02:00
logger ( 'diaspora_like: Ignoring this author.' );
2011-09-15 04:33:42 +02:00
return 202 ;
2011-08-16 08:19:17 +02:00
}
2011-08-17 07:31:14 +02:00
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2011-08-14 04:03:59 +02:00
intval ( $importer [ 'uid' ]),
2011-08-17 07:31:14 +02:00
dbesc ( $parent_guid )
2011-08-14 04:03:59 +02:00
);
2015-05-09 19:39:47 +02:00
if ( ! count ( $r )) {
$result = diaspora_store_by_guid ( $parent_guid , $contact [ 'url' ], $importer [ 'uid' ]);
if ( ! $result ) {
$person = find_diaspora_person_by_handle ( $diaspora_handle );
$result = diaspora_store_by_guid ( $parent_guid , $person [ 'url' ], $importer [ 'uid' ]);
}
if ( $result ) {
logger ( " Fetched missing item " . $parent_guid . " - result: " . $result , LOGGER_DEBUG );
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
intval ( $importer [ 'uid' ]),
dbesc ( $parent_guid )
);
}
}
2011-08-17 07:31:14 +02:00
if ( ! count ( $r )) {
logger ( 'diaspora_like: parent item not found: ' . $guid );
2011-08-14 04:03:59 +02:00
return ;
2011-08-17 07:31:14 +02:00
}
2011-08-14 04:03:59 +02:00
2011-08-17 07:31:14 +02:00
$parent_item = $r [ 0 ];
2011-08-21 03:05:05 +02:00
$r = q ( " SELECT * FROM `item` WHERE `uid` = %d AND `guid` = '%s' LIMIT 1 " ,
2011-08-17 07:31:14 +02:00
intval ( $importer [ 'uid' ]),
dbesc ( $guid )
2011-08-14 04:03:59 +02:00
);
2011-08-17 07:31:14 +02:00
if ( count ( $r )) {
if ( $positive === 'true' ) {
logger ( 'diaspora_like: duplicate like: ' . $guid );
return ;
2014-03-09 09:19:14 +01:00
}
2012-06-03 19:12:16 +02:00
// Note: I don't think "Like" objects with positive = "false" are ever actually used
// It looks like "RelayableRetractions" are used for "unlike" instead
2011-08-17 07:31:14 +02:00
if ( $positive === 'false' ) {
2012-06-03 19:12:16 +02:00
logger ( 'diaspora_like: received a like with positive set to "false"...ignoring' );
2014-03-09 09:19:14 +01:00
/* q ( " UPDATE `item` SET `deleted` = 1 WHERE `id` = %d AND `uid` = %d " ,
2011-08-17 07:31:14 +02:00
intval ( $r [ 0 ][ 'id' ]),
intval ( $importer [ 'uid' ])
2012-06-03 19:12:16 +02:00
); */
2012-06-09 21:54:21 +02:00
// FIXME--actually don't unless it turns out that Diaspora does indeed send out "false" likes
2011-08-17 07:31:14 +02:00
// send notification via proc_run()
return ;
}
}
2012-06-03 19:12:16 +02:00
// Note: I don't think "Like" objects with positive = "false" are ever actually used
// It looks like "RelayableRetractions" are used for "unlike" instead
2011-08-17 07:31:14 +02:00
if ( $positive === 'false' ) {
2012-06-03 19:12:16 +02:00
logger ( 'diaspora_like: received a like with positive set to "false"' );
logger ( 'diaspora_like: unlike received with no corresponding like...ignoring' );
2015-03-22 21:53:13 +01:00
return ;
2011-08-17 07:31:14 +02:00
}
2012-06-09 21:54:21 +02:00
/* How Diaspora performs " like " signature checking :
2011-08-17 07:31:14 +02:00
2012-06-09 21:54:21 +02:00
- If an item has been sent by the like author to the top - level post owner to relay on
to the rest of the contacts on the top - level post , the top - level post owner should check
the author_signature , then create a parent_author_signature before relaying the like on
- If an item has been relayed on by the top - level post owner , the contacts who receive it
check only the parent_author_signature . Basically , they trust that the top - level post
owner has already verified the authenticity of anything he / she sends out
- In either case , the signature that get checked is the signature created by the person
who sent the salmon
*/
2011-08-17 07:31:14 +02:00
2015-03-22 21:53:13 +01:00
// Diaspora has changed the way they are signing the likes.
// Just to make sure that we don't miss any likes we will check the old and the current way.
$old_signed_data = $guid . ';' . $target_type . ';' . $parent_guid . ';' . $positive . ';' . $diaspora_handle ;
$signed_data = $positive . ';' . $guid . ';' . $target_type . ';' . $parent_guid . ';' . $diaspora_handle ;
2012-06-09 21:54:21 +02:00
$key = $msg [ 'key' ];
2011-08-14 04:03:59 +02:00
2015-03-22 21:53:13 +01:00
if ( $parent_author_signature ) {
2012-06-09 21:54:21 +02:00
// If a parent_author_signature exists, then we've received the like
// relayed from the top-level post owner. There's no need to check the
// author_signature if the parent_author_signature is valid
2011-08-14 04:03:59 +02:00
2011-08-17 07:31:14 +02:00
$parent_author_signature = base64_decode ( $parent_author_signature );
2011-08-14 04:03:59 +02:00
2015-03-22 21:53:13 +01:00
if ( ! rsa_verify ( $signed_data , $parent_author_signature , $key , 'sha256' ) AND
! rsa_verify ( $old_signed_data , $parent_author_signature , $key , 'sha256' )) {
logger ( 'diaspora_like: top-level owner verification failed.' );
return ;
2012-06-09 21:54:21 +02:00
}
2016-01-21 13:38:30 +01:00
} else {
2012-06-09 21:54:21 +02:00
// If there's no parent_author_signature, then we've received the like
// from the like creator. In that case, the person is "like"ing
// our post, so he/she must be a contact of ours and his/her public key
// should be in $msg['key']
$author_signature = base64_decode ( $author_signature );
2015-03-22 21:53:13 +01:00
if ( ! rsa_verify ( $signed_data , $author_signature , $key , 'sha256' ) AND
! rsa_verify ( $old_signed_data , $author_signature , $key , 'sha256' )) {
logger ( 'diaspora_like: like creator verification failed.' );
return ;
2011-08-17 07:31:14 +02:00
}
}
// Phew! Everything checks out. Now create an item.
2012-06-16 18:41:25 +02:00
// Find the original comment author information.
// We need this to make sure we display the comment author
// information (name and avatar) correctly.
2012-06-10 16:41:23 +02:00
if ( strcasecmp ( $diaspora_handle , $msg [ 'author' ]) == 0 )
$person = $contact ;
else {
$person = find_diaspora_person_by_handle ( $diaspora_handle );
if ( ! is_array ( $person )) {
logger ( 'diaspora_like: unable to find author details' );
return ;
}
}
2011-08-17 07:31:14 +02:00
$uri = $diaspora_handle . ':' . $guid ;
2011-08-17 13:24:26 +02:00
$activity = ACTIVITY_LIKE ;
2011-08-17 07:31:14 +02:00
$post_type = (( $parent_item [ 'resource-id' ]) ? t ( 'photo' ) : t ( 'status' ));
2015-05-25 13:27:45 +02:00
$objtype = (( $parent_item [ 'resource-id' ]) ? ACTIVITY_OBJ_PHOTO : ACTIVITY_OBJ_NOTE );
2011-08-17 07:31:14 +02:00
$link = xmlify ( '<link rel="alternate" type="text/html" href="' . $a -> get_baseurl () . '/display/' . $importer [ 'nickname' ] . '/' . $parent_item [ 'id' ] . '" />' . " \n " ) ;
$body = $parent_item [ 'body' ];
2011-08-14 04:03:59 +02:00
$obj = <<< EOT
< object >
< type > $objtype </ type >
< local > 1 </ local >
2011-08-17 07:31:14 +02:00
< id > { $parent_item [ 'uri' ]} </ id >
2011-08-14 04:03:59 +02:00
< link > $link </ link >
< title ></ title >
< content > $body </ content >
</ object >
EOT ;
2011-08-17 07:31:14 +02:00
$bodyverb = t ( '%1$s likes %2$s\'s %3$s' );
2011-08-14 04:03:59 +02:00
2015-12-01 15:36:32 +01:00
// Fetch the contact id - if we know this contact
$r = q ( " SELECT `id`, `network` FROM `contact` WHERE `nurl` = '%s' AND `uid` = %d LIMIT 1 " ,
dbesc ( normalise_link ( $person [ 'url' ])), intval ( $importer [ 'uid' ]));
if ( $r ) {
$cid = $r [ 0 ][ 'id' ];
$network = $r [ 0 ][ 'network' ];
} else {
$cid = $contact [ 'id' ];
$network = NETWORK_DIASPORA ;
}
2011-08-14 04:03:59 +02:00
$arr = array ();
$arr [ 'uri' ] = $uri ;
2011-08-17 07:31:14 +02:00
$arr [ 'uid' ] = $importer [ 'uid' ];
2011-08-21 03:14:19 +02:00
$arr [ 'guid' ] = $guid ;
2015-12-01 15:36:32 +01:00
$arr [ 'network' ] = $network ;
$arr [ 'contact-id' ] = $cid ;
2011-08-14 04:03:59 +02:00
$arr [ 'type' ] = 'activity' ;
2011-08-17 07:31:14 +02:00
$arr [ 'wall' ] = $parent_item [ 'wall' ];
2011-08-14 04:03:59 +02:00
$arr [ 'gravity' ] = GRAVITY_LIKE ;
2011-08-17 07:31:14 +02:00
$arr [ 'parent' ] = $parent_item [ 'id' ];
$arr [ 'parent-uri' ] = $parent_item [ 'uri' ];
2011-10-21 12:33:34 +02:00
$arr [ 'owner-name' ] = $parent_item [ 'name' ];
$arr [ 'owner-link' ] = $parent_item [ 'url' ];
2012-11-10 23:19:32 +01:00
//$arr['owner-avatar'] = $parent_item['thumb'];
$arr [ 'owner-avatar' ] = (( x ( $parent_item , 'thumb' )) ? $parent_item [ 'thumb' ] : $parent_item [ 'photo' ]);
2011-08-17 07:31:14 +02:00
2011-08-21 03:14:19 +02:00
$arr [ 'author-name' ] = $person [ 'name' ];
$arr [ 'author-link' ] = $person [ 'url' ];
$arr [ 'author-avatar' ] = (( x ( $person , 'thumb' )) ? $person [ 'thumb' ] : $person [ 'photo' ]);
2013-12-27 01:58:21 +01:00
2011-08-14 04:03:59 +02:00
$ulink = '[url=' . $contact [ 'url' ] . ']' . $contact [ 'name' ] . '[/url]' ;
2011-08-17 07:31:14 +02:00
$alink = '[url=' . $parent_item [ 'author-link' ] . ']' . $parent_item [ 'author-name' ] . '[/url]' ;
2014-07-22 00:36:20 +02:00
//$plink = '[url=' . $a->get_baseurl() . '/display/' . $importer['nickname'] . '/' . $parent_item['id'] . ']' . $post_type . '[/url]';
2014-08-09 09:17:28 +02:00
$plink = '[url=' . $a -> get_baseurl () . '/display/' . urlencode ( $guid ) . ']' . $post_type . '[/url]' ;
2011-08-14 04:03:59 +02:00
$arr [ 'body' ] = sprintf ( $bodyverb , $ulink , $alink , $plink );
2011-08-27 02:52:24 +02:00
$arr [ 'app' ] = 'Diaspora' ;
2011-08-17 13:24:26 +02:00
$arr [ 'private' ] = $parent_item [ 'private' ];
2011-08-14 04:03:59 +02:00
$arr [ 'verb' ] = $activity ;
$arr [ 'object-type' ] = $objtype ;
$arr [ 'object' ] = $obj ;
$arr [ 'visible' ] = 1 ;
$arr [ 'unseen' ] = 1 ;
$arr [ 'last-child' ] = 0 ;
2011-08-18 14:08:39 +02:00
$message_id = item_store ( $arr );
2011-08-14 04:03:59 +02:00
2011-09-01 06:46:37 +02:00
2014-07-22 00:36:20 +02:00
//if($message_id) {
// q("update item set plink = '%s' where id = %d",
// //dbesc($a->get_baseurl() . '/display/' . $importer['nickname'] . '/' . $message_id),
// dbesc($a->get_baseurl().'/display/'.$guid),
// intval($message_id)
// );
//}
2011-09-01 06:46:37 +02:00
2016-01-21 13:21:47 +01:00
// If we are the origin of the parent we store the original signature and notify our followers
if ( $parent_item [ 'origin' ]) {
$author_signature_base64 = base64_encode ( $author_signature );
$author_signature_base64 = diaspora_repair_signature ( $author_signature_base64 , $diaspora_handle );
2011-08-18 14:08:39 +02:00
q ( " insert into sign (`iid`,`signed_text`,`signature`,`signer`) values (%d,'%s','%s','%s') " ,
intval ( $message_id ),
2012-07-10 04:35:59 +02:00
dbesc ( $signed_data ),
2016-01-21 13:21:47 +01:00
dbesc ( $author_signature_base64 ),
2011-08-18 14:08:39 +02:00
dbesc ( $diaspora_handle )
);
2011-08-14 04:03:59 +02:00
2016-01-21 13:21:47 +01:00
// notify others
2012-08-10 06:06:18 +02:00
proc_run ( 'php' , 'include/notifier.php' , 'comment-import' , $message_id );
2016-01-21 13:21:47 +01:00
}
2011-08-14 04:03:59 +02:00
2011-08-18 14:08:39 +02:00
return ;
2011-08-10 14:10:48 +02:00
}
2011-08-16 08:19:17 +02:00
function diaspora_retraction ( $importer , $xml ) {
2011-12-21 23:42:12 +01:00
2011-08-16 08:19:17 +02:00
$guid = notags ( unxmlify ( $xml -> guid ));
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2011-08-23 05:35:34 +02:00
$type = notags ( unxmlify ( $xml -> type ));
2011-08-16 08:19:17 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $diaspora_handle );
if ( ! $contact )
return ;
2011-08-23 05:35:34 +02:00
if ( $type === 'Person' ) {
2011-11-27 13:09:14 +01:00
require_once ( 'include/Contact.php' );
2011-08-23 05:35:34 +02:00
contact_remove ( $contact [ 'id' ]);
2015-12-03 08:35:47 +01:00
} elseif ( $type === 'StatusMessage' ) {
$guid = notags ( unxmlify ( $xml -> post_guid ));
$r = q ( " SELECT * FROM `item` WHERE `guid` = '%s' AND `uid` = %d AND NOT `file` LIKE '%%[%%' LIMIT 1 " ,
dbesc ( $guid ),
intval ( $importer [ 'uid' ])
);
if ( count ( $r )) {
if ( link_compare ( $r [ 0 ][ 'author-link' ], $contact [ 'url' ])) {
q ( " UPDATE `item` SET `deleted` = 1, `changed` = '%s' WHERE `id` = %d " ,
dbesc ( datetime_convert ()),
intval ( $r [ 0 ][ 'id' ])
);
delete_thread ( $r [ 0 ][ 'id' ], $r [ 0 ][ 'parent-uri' ]);
}
}
} elseif ( $type === 'Post' ) {
2012-03-24 12:16:27 +01:00
$r = q ( " select * from item where guid = '%s' and uid = %d and not file like '%%[%%' limit 1 " ,
2011-08-23 05:35:34 +02:00
dbesc ( 'guid' ),
intval ( $importer [ 'uid' ])
);
if ( count ( $r )) {
if ( link_compare ( $r [ 0 ][ 'author-link' ], $contact [ 'url' ])) {
2014-03-09 09:19:14 +01:00
q ( " update item set `deleted` = 1, `changed` = '%s' where `id` = %d " ,
dbesc ( datetime_convert ()),
2011-08-23 05:35:34 +02:00
intval ( $r [ 0 ][ 'id' ])
);
2015-01-07 01:46:13 +01:00
delete_thread ( $r [ 0 ][ 'id' ], $r [ 0 ][ 'parent-uri' ]);
2011-08-23 05:35:34 +02:00
}
}
}
2011-08-10 14:10:48 +02:00
2011-09-15 04:33:42 +02:00
return 202 ;
2011-08-23 05:35:34 +02:00
// NOTREACHED
2011-08-10 14:10:48 +02:00
}
2011-10-14 03:32:02 +02:00
2011-12-21 23:42:12 +01:00
function diaspora_signed_retraction ( $importer , $xml , $msg ) {
2011-11-05 22:45:29 +01:00
$guid = notags ( unxmlify ( $xml -> target_guid ));
$diaspora_handle = notags ( unxmlify ( $xml -> sender_handle ));
$type = notags ( unxmlify ( $xml -> target_type ));
$sig = notags ( unxmlify ( $xml -> target_author_signature ));
2012-05-26 02:26:09 +02:00
$parent_author_signature = (( $xml -> parent_author_signature ) ? notags ( unxmlify ( $xml -> parent_author_signature )) : '' );
2011-11-05 22:45:29 +01:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $diaspora_handle );
2011-12-21 23:42:12 +01:00
if ( ! $contact ) {
2012-09-30 01:48:56 +02:00
logger ( 'diaspora_signed_retraction: no contact ' . $diaspora_handle . ' for ' . $importer [ 'uid' ]);
2011-11-05 22:45:29 +01:00
return ;
2011-12-21 23:42:12 +01:00
}
2011-11-05 22:45:29 +01:00
$signed_data = $guid . ';' . $type ;
2012-06-09 21:54:21 +02:00
$key = $msg [ 'key' ];
2011-11-05 22:45:29 +01:00
2012-06-09 21:54:21 +02:00
/* How Diaspora performs relayable_retraction signature checking :
2011-11-05 22:45:29 +01:00
2012-06-09 21:54:21 +02:00
- If an item has been sent by the item author to the top - level post owner to relay on
to the rest of the contacts on the top - level post , the top - level post owner checks
the author_signature , then creates a parent_author_signature before relaying the item on
- If an item has been relayed on by the top - level post owner , the contacts who receive it
check only the parent_author_signature . Basically , they trust that the top - level post
owner has already verified the authenticity of anything he / she sends out
- In either case , the signature that get checked is the signature created by the person
who sent the salmon
*/
2012-05-26 02:26:09 +02:00
2012-06-09 21:54:21 +02:00
if ( $parent_author_signature ) {
$parent_author_signature = base64_decode ( $parent_author_signature );
if ( ! rsa_verify ( $signed_data , $parent_author_signature , $key , 'sha256' )) {
logger ( 'diaspora_signed_retraction: top-level post owner verification failed' );
2012-05-26 02:26:09 +02:00
return ;
}
2011-11-05 22:45:29 +01:00
2016-01-21 13:21:47 +01:00
} else {
2011-11-05 22:45:29 +01:00
2012-06-09 21:54:21 +02:00
$sig_decode = base64_decode ( $sig );
if ( ! rsa_verify ( $signed_data , $sig_decode , $key , 'sha256' )) {
logger ( 'diaspora_signed_retraction: retraction owner verification failed.' . print_r ( $msg , true ));
2012-05-26 02:26:09 +02:00
return ;
}
}
2012-06-03 00:11:31 +02:00
if ( $type === 'StatusMessage' || $type === 'Comment' || $type === 'Like' ) {
2012-03-24 12:16:27 +01:00
$r = q ( " select * from item where guid = '%s' and uid = %d and not file like '%%[%%' limit 1 " ,
2011-12-21 23:42:12 +01:00
dbesc ( $guid ),
2011-11-05 22:45:29 +01:00
intval ( $importer [ 'uid' ])
);
if ( count ( $r )) {
if ( link_compare ( $r [ 0 ][ 'author-link' ], $contact [ 'url' ])) {
2014-03-09 09:19:14 +01:00
q ( " update item set `deleted` = 1, `edited` = '%s', `changed` = '%s', `body` = '' , `title` = '' where `id` = %d " ,
dbesc ( datetime_convert ()),
dbesc ( datetime_convert ()),
2011-11-05 22:45:29 +01:00
intval ( $r [ 0 ][ 'id' ])
);
2015-01-07 01:46:13 +01:00
delete_thread ( $r [ 0 ][ 'id' ], $r [ 0 ][ 'parent-uri' ]);
2014-03-09 09:19:14 +01:00
2012-06-01 03:40:12 +02:00
// Now check if the retraction needs to be relayed by us
//
// The first item in the `item` table with the parent id is the parent. However, MySQL doesn't always
// return the items ordered by `item`.`id`, in which case the wrong item is chosen as the parent.
// The only item with `parent` and `id` as the parent id is the parent item.
2015-12-27 03:54:20 +01:00
$p = q ( " SELECT `origin` FROM `item` WHERE `parent` = %d AND `id` = %d LIMIT 1 " ,
intval ( $r [ 0 ][ 'parent' ]),
intval ( $r [ 0 ][ 'parent' ])
2012-06-01 03:40:12 +02:00
);
if ( count ( $p )) {
2016-01-21 13:21:47 +01:00
if ( $p [ 0 ][ 'origin' ]) {
2012-06-01 03:40:12 +02:00
q ( " insert into sign (`retract_iid`,`signed_text`,`signature`,`signer`) values (%d,'%s','%s','%s') " ,
$r [ 0 ][ 'id' ],
dbesc ( $signed_data ),
dbesc ( $sig ),
dbesc ( $diaspora_handle )
);
// the existence of parent_author_signature would have meant the parent_author or owner
// is already relaying.
logger ( 'diaspora_signed_retraction: relaying relayable_retraction' );
2012-08-10 06:06:18 +02:00
proc_run ( 'php' , 'include/notifier.php' , 'drop' , $r [ 0 ][ 'id' ]);
2012-06-01 03:40:12 +02:00
}
}
2011-11-05 22:45:29 +01:00
}
}
}
2011-12-21 23:42:12 +01:00
else
logger ( 'diaspora_signed_retraction: unknown type: ' . $type );
2011-11-05 22:45:29 +01:00
return 202 ;
// NOTREACHED
}
2012-07-09 07:32:04 +02:00
function diaspora_profile ( $importer , $xml , $msg ) {
2011-10-14 03:32:02 +02:00
$a = get_app ();
$diaspora_handle = notags ( unxmlify ( $xml -> diaspora_handle ));
2012-07-09 07:32:04 +02:00
if ( $diaspora_handle != $msg [ 'author' ]) {
logger ( 'diaspora_post: Potential forgery. Message handle is not the same as envelope sender.' );
return 202 ;
}
2011-10-14 03:32:02 +02:00
$contact = diaspora_get_contact_by_handle ( $importer [ 'uid' ], $diaspora_handle );
if ( ! $contact )
return ;
2016-01-06 14:13:59 +01:00
//if($contact['blocked']) {
// logger('diaspora_post: Ignoring this author.');
// return 202;
//}
2011-10-14 03:32:02 +02:00
$name = unxmlify ( $xml -> first_name ) . (( strlen ( $xml -> last_name )) ? ' ' . unxmlify ( $xml -> last_name ) : '' );
$image_url = unxmlify ( $xml -> image_url );
$birthday = unxmlify ( $xml -> birthday );
2015-01-08 00:13:43 +01:00
$location = diaspora2bb ( unxmlify ( $xml -> location ));
2015-01-07 01:46:13 +01:00
$about = diaspora2bb ( unxmlify ( $xml -> bio ));
2015-01-25 13:19:37 +01:00
$gender = unxmlify ( $xml -> gender );
2016-01-06 14:13:59 +01:00
$searchable = ( unxmlify ( $xml -> searchable ) == " true " );
$nsfw = ( unxmlify ( $xml -> nsfw ) == " true " );
2015-01-25 13:19:37 +01:00
$tags = unxmlify ( $xml -> tag_string );
2011-10-14 03:32:02 +02:00
2015-01-25 13:19:37 +01:00
$tags = explode ( " # " , $tags );
$keywords = array ();
foreach ( $tags as $tag ) {
$tag = trim ( strtolower ( $tag ));
if ( $tag != " " )
$keywords [] = $tag ;
}
$keywords = implode ( " , " , $keywords );
2012-09-20 04:35:39 +02:00
$handle_parts = explode ( " @ " , $diaspora_handle );
2016-01-06 14:13:59 +01:00
$nick = $handle_parts [ 0 ];
2012-09-20 04:35:39 +02:00
if ( $name === '' ) {
$name = $handle_parts [ 0 ];
}
2015-01-07 01:46:13 +01:00
2013-07-08 14:21:49 +02:00
if ( preg_match ( " |^https?://| " , $image_url ) === 0 ) {
2012-09-20 04:35:39 +02:00
$image_url = " http:// " . $handle_parts [ 1 ] . $image_url ;
}
/* $r = q ( " SELECT DISTINCT ( `resource-id` ) FROM `photo` WHERE `uid` = %d AND `contact-id` = %d AND `album` = 'Contact Photos' " ,
2011-10-14 03:32:02 +02:00
intval ( $importer [ 'uid' ]),
intval ( $contact [ 'id' ])
);
2012-09-20 04:35:39 +02:00
$oldphotos = (( count ( $r )) ? $r : null ); */
2011-10-14 03:32:02 +02:00
2011-10-18 10:12:51 +02:00
require_once ( 'include/Photo.php' );
2016-01-28 11:09:08 +01:00
update_contact_avatar ( $image_url , $importer [ 'uid' ], $contact [ 'id' ]);
2015-01-07 01:46:13 +01:00
// Generic birthday. We don't know the timezone. The year is irrelevant.
2011-10-14 03:32:02 +02:00
2011-10-18 10:12:51 +02:00
$birthday = str_replace ( '1000' , '1901' , $birthday );
2015-10-17 02:36:16 +02:00
if ( $birthday != " " )
$birthday = datetime_convert ( 'UTC' , 'UTC' , $birthday , 'Y-m-d' );
2011-10-14 09:20:37 +02:00
2011-12-20 04:06:25 +01:00
// this is to prevent multiple birthday notifications in a single year
// if we already have a stored birthday and the 'm-d' part hasn't changed, preserve the entry, which will preserve the notify year
if ( substr ( $birthday , 5 ) === substr ( $contact [ 'bd' ], 5 ))
$birthday = $contact [ 'bd' ];
2015-12-25 23:17:34 +01:00
/// @TODO Update name on item['author-name'] if the name changed. See consume_feed()
/// (Not doing this currently because D* protocol is scheduled for revision soon).
2012-02-25 07:47:43 +01:00
2016-01-28 11:09:08 +01:00
$r = q ( " UPDATE `contact` SET `name` = '%s', `nick` = '%s', `addr` = '%s', `name-date` = '%s', `bd` = '%s',
2016-01-28 01:26:19 +01:00
`location` = '%s' , `about` = '%s' , `keywords` = '%s' , `gender` = '%s' WHERE `id` = % d AND `uid` = % d " ,
2011-10-14 03:32:02 +02:00
dbesc ( $name ),
2016-01-06 14:13:59 +01:00
dbesc ( $nick ),
dbesc ( $diaspora_handle ),
2011-10-14 03:32:02 +02:00
dbesc ( datetime_convert ()),
2011-10-18 10:12:51 +02:00
dbesc ( $birthday ),
2015-01-07 01:46:13 +01:00
dbesc ( $location ),
dbesc ( $about ),
2015-01-25 13:19:37 +01:00
dbesc ( $keywords ),
dbesc ( $gender ),
2011-10-14 03:32:02 +02:00
intval ( $contact [ 'id' ]),
2011-10-18 10:12:51 +02:00
intval ( $importer [ 'uid' ])
2015-01-07 01:46:13 +01:00
);
2011-10-18 10:12:51 +02:00
2016-01-06 14:13:59 +01:00
if ( $searchable ) {
2015-01-25 13:19:37 +01:00
require_once ( 'include/socgraph.php' );
2016-01-06 14:13:59 +01:00
poco_check ( $contact [ 'url' ], $name , NETWORK_DIASPORA , $image_url , $about , $location , $gender , $keywords , " " ,
2015-02-15 10:52:45 +01:00
datetime_convert (), 2 , $contact [ 'id' ], $importer [ 'uid' ]);
2015-01-25 13:19:37 +01:00
}
2016-01-06 14:13:59 +01:00
update_gcontact ( array ( " url " => $contact [ 'url' ], " network " => NETWORK_DIASPORA , " generation " => 2 ,
" photo " => $image_url , " name " => $name , " location " => $location ,
" about " => $about , " birthday " => $birthday , " gender " => $gender ,
" addr " => $diaspora_handle , " nick " => $nick , " keywords " => $keywords ,
" hide " => ! $searchable , " nsfw " => $nsfw ));
2015-01-08 01:32:19 +01:00
2012-09-20 04:35:39 +02:00
/* if ( $r ) {
2011-10-14 03:32:02 +02:00
if ( $oldphotos ) {
foreach ( $oldphotos as $ph ) {
q ( " DELETE FROM `photo` WHERE `uid` = %d AND `contact-id` = %d AND `album` = 'Contact Photos' AND `resource-id` = '%s' " ,
intval ( $importer [ 'uid' ]),
intval ( $contact [ 'id' ]),
dbesc ( $ph [ 'resource-id' ])
);
}
}
2012-09-20 04:35:39 +02:00
} */
2011-10-14 03:32:02 +02:00
return ;
}
2011-08-15 05:38:31 +02:00
function diaspora_share ( $me , $contact ) {
$a = get_app ();
$myaddr = $me [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
$theiraddr = $contact [ 'addr' ];
$tpl = get_markup_template ( 'diaspora_share.tpl' );
$msg = replace_macros ( $tpl , array (
2011-08-19 14:20:30 +02:00
'$sender' => $myaddr ,
2011-08-15 05:38:31 +02:00
'$recipient' => $theiraddr
));
2011-08-19 11:24:30 +02:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $me , $contact , $me [ 'prvkey' ], $contact [ 'pubkey' ])));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$me,$contact,$me['prvkey'],$contact['pubkey']));
2011-08-15 05:38:31 +02:00
2011-10-08 00:49:41 +02:00
return ( diaspora_transmit ( $owner , $contact , $slap , false ));
2011-08-15 05:38:31 +02:00
}
2011-08-23 05:35:34 +02:00
function diaspora_unshare ( $me , $contact ) {
$a = get_app ();
$myaddr = $me [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
$tpl = get_markup_template ( 'diaspora_retract.tpl' );
$msg = replace_macros ( $tpl , array (
'$guid' => $me [ 'guid' ],
'$type' => 'Person' ,
'$handle' => $myaddr
));
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $me , $contact , $me [ 'prvkey' ], $contact [ 'pubkey' ])));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$me,$contact,$me['prvkey'],$contact['pubkey']));
2011-08-23 05:35:34 +02:00
2011-10-08 00:49:41 +02:00
return ( diaspora_transmit ( $owner , $contact , $slap , false ));
2011-08-24 10:21:24 +02:00
2011-08-23 05:35:34 +02:00
}
2011-09-22 13:11:39 +02:00
function diaspora_send_status ( $item , $owner , $contact , $public_batch = false ) {
2011-08-15 06:23:02 +02:00
$a = get_app ();
2011-08-19 06:09:44 +02:00
$myaddr = $owner [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
2011-08-15 06:23:02 +02:00
$theiraddr = $contact [ 'addr' ];
2011-08-24 03:17:35 +02:00
$images = array ();
2012-04-02 04:16:13 +02:00
$title = $item [ 'title' ];
2011-08-24 03:17:35 +02:00
$body = $item [ 'body' ];
2011-10-24 01:01:04 +02:00
/*
2011-10-24 04:05:32 +02:00
// We're trying to match Diaspora's split message/photo protocol but
// all the photos are displayed on D* as links and not img's - even
// though we're sending pretty much precisely what they send us when
// doing the same operation.
// Commented out for now, we'll use bb2diaspora to convert photos to markdown
// which seems to get through intact.
2011-08-24 03:17:35 +02:00
$cnt = preg_match_all ( '|\[img\](.*?)\[\/img\]|' , $body , $matches , PREG_SET_ORDER );
if ( $cnt ) {
foreach ( $matches as $mtch ) {
$detail = array ();
$detail [ 'str' ] = $mtch [ 0 ];
2011-08-24 03:50:18 +02:00
$detail [ 'path' ] = dirname ( $mtch [ 1 ]) . '/' ;
2011-08-24 03:17:35 +02:00
$detail [ 'file' ] = basename ( $mtch [ 1 ]);
$detail [ 'guid' ] = $item [ 'guid' ];
$detail [ 'handle' ] = $myaddr ;
$images [] = $detail ;
2011-10-20 11:21:57 +02:00
$body = str_replace ( $detail [ 'str' ], $mtch [ 1 ], $body );
2011-08-24 03:17:35 +02:00
}
2012-06-17 19:49:05 +02:00
}
2011-10-24 01:01:04 +02:00
*/
2012-06-23 12:42:01 +02:00
//if(strlen($title))
// $body = "[b]".html_entity_decode($title)."[/b]\n\n".$body;
2012-06-17 19:49:05 +02:00
2012-06-23 12:42:01 +02:00
// convert to markdown
2012-06-17 19:49:05 +02:00
$body = xmlify ( html_entity_decode ( bb2diaspora ( $body )));
2012-06-23 12:42:01 +02:00
//$body = bb2diaspora($body);
2012-04-02 04:16:13 +02:00
2012-06-23 12:42:01 +02:00
// Adding the title
if ( strlen ( $title ))
$body = " ## " . html_entity_decode ( $title ) . " \n \n " . $body ;
2012-04-02 04:16:13 +02:00
2011-11-02 01:30:52 +01:00
if ( $item [ 'attach' ]) {
$cnt = preg_match_all ( '/href=\"(.*?)\"(.*?)title=\"(.*?)\"/ism' , $item [ 'attach' ], $matches , PREG_SET_ORDER );
if ( cnt ) {
$body .= " \n " . t ( 'Attachments:' ) . " \n " ;
foreach ( $matches as $mtch ) {
$body .= '[' . $mtch [ 3 ] . '](' . $mtch [ 1 ] . ')' . " \n " ;
}
}
2014-04-22 00:58:17 +02:00
}
2011-11-02 01:30:52 +01:00
2011-08-15 06:23:02 +02:00
$public = (( $item [ 'private' ]) ? 'false' : 'true' );
require_once ( 'include/datetime.php' );
2011-08-30 03:01:59 +02:00
$created = datetime_convert ( 'UTC' , 'UTC' , $item [ 'created' ], 'Y-m-d H:i:s \U\T\C' );
2011-08-15 06:23:02 +02:00
2014-04-22 00:58:17 +02:00
// Detect a share element and do a reshare
// see: https://github.com/Raven24/diaspora-federation/blob/master/lib/diaspora-federation/entities/reshare.rb
2014-04-26 01:35:02 +02:00
if ( ! $item [ 'private' ] AND ( $ret = diaspora_is_reshare ( $item [ " body " ]))) {
$tpl = get_markup_template ( 'diaspora_reshare.tpl' );
$msg = replace_macros ( $tpl , array (
'$root_handle' => xmlify ( $ret [ 'root_handle' ]),
'$root_guid' => $ret [ 'root_guid' ],
'$guid' => $item [ 'guid' ],
'$handle' => xmlify ( $myaddr ),
'$public' => $public ,
'$created' => $created ,
'$provider' => $item [ " app " ]
));
} else {
$tpl = get_markup_template ( 'diaspora_post.tpl' );
$msg = replace_macros ( $tpl , array (
'$body' => $body ,
'$guid' => $item [ 'guid' ],
'$handle' => xmlify ( $myaddr ),
'$public' => $public ,
'$created' => $created ,
'$provider' => $item [ " app " ]
));
}
2011-08-15 06:23:02 +02:00
2014-04-26 01:35:02 +02:00
logger ( 'diaspora_send_status: ' . $owner [ 'username' ] . ' -> ' . $contact [ 'name' ] . ' base message: ' . $msg , LOGGER_DATA );
2016-01-11 14:24:54 +01:00
logger ( 'send guid ' . $item [ 'guid' ], LOGGER_DEBUG );
2011-08-15 06:23:02 +02:00
2011-09-22 13:11:39 +02:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $owner , $contact , $owner [ 'uprvkey' ], $contact [ 'pubkey' ], $public_batch )));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$owner,$contact,$owner['uprvkey'],$contact['pubkey'],$public_batch));
2011-08-15 06:23:02 +02:00
2016-01-11 14:24:54 +01:00
$return_code = diaspora_transmit ( $owner , $contact , $slap , $public_batch , false , $item [ 'guid' ]);
2011-08-24 03:17:35 +02:00
2014-04-26 02:22:30 +02:00
logger ( 'diaspora_send_status: guid: ' . $item [ 'guid' ] . ' result ' . $return_code , LOGGER_DEBUG );
2014-04-26 01:35:02 +02:00
2011-08-24 03:17:35 +02:00
if ( count ( $images )) {
2011-09-22 13:11:39 +02:00
diaspora_send_images ( $item , $owner , $contact , $images , $public_batch );
2011-08-24 03:17:35 +02:00
}
2011-08-15 06:23:02 +02:00
return $return_code ;
}
2014-04-26 01:35:02 +02:00
function diaspora_is_reshare ( $body ) {
$body = trim ( $body );
2015-08-09 20:39:11 +02:00
// Skip if it isn't a pure repeated messages
// Does it start with a share?
if ( strpos ( $body , " [share " ) > 0 )
return ( false );
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
// Does it end with a share?
if ( strlen ( $body ) > ( strrpos ( $body , " [/share] " ) + 8 ))
return ( false );
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
$attributes = preg_replace ( " / \ [share(.*?) \ ] \ s?(.*?) \ s? \ [ \ /share \ ] \ s?/ism " , " $ 1 " , $body );
// Skip if there is no shared message in there
if ( $body == $attributes )
return ( false );
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
$guid = " " ;
preg_match ( " /guid='(.*?)'/ism " , $attributes , $matches );
if ( $matches [ 1 ] != " " )
$guid = $matches [ 1 ];
2015-04-03 13:06:19 +02:00
2015-08-09 20:39:11 +02:00
preg_match ( '/guid="(.*?)"/ism' , $attributes , $matches );
if ( $matches [ 1 ] != " " )
$guid = $matches [ 1 ];
2015-04-03 13:06:19 +02:00
if ( $guid != " " ) {
$r = q ( " SELECT `contact-id` FROM `item` WHERE `guid` = '%s' AND `network` IN ('%s', '%s') LIMIT 1 " ,
dbesc ( $guid ), NETWORK_DFRN , NETWORK_DIASPORA );
if ( $r ) {
$ret = array ();
$ret [ " root_handle " ] = diaspora_handle_from_contact ( $r [ 0 ][ " contact-id " ]);
$ret [ " root_guid " ] = $guid ;
return ( $ret );
}
}
2015-08-09 20:39:11 +02:00
$profile = " " ;
preg_match ( " /profile='(.*?)'/ism " , $attributes , $matches );
if ( $matches [ 1 ] != " " )
$profile = $matches [ 1 ];
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
preg_match ( '/profile="(.*?)"/ism' , $attributes , $matches );
if ( $matches [ 1 ] != " " )
$profile = $matches [ 1 ];
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
$ret = array ();
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
$ret [ " root_handle " ] = preg_replace ( " =https?://(.*)/u/(.*)=ism " , " $ 2@ $ 1 " , $profile );
if (( $ret [ " root_handle " ] == $profile ) OR ( $ret [ " root_handle " ] == " " ))
return ( false );
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
$link = " " ;
preg_match ( " /link='(.*?)'/ism " , $attributes , $matches );
if ( $matches [ 1 ] != " " )
$link = $matches [ 1 ];
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
preg_match ( '/link="(.*?)"/ism' , $attributes , $matches );
if ( $matches [ 1 ] != " " )
$link = $matches [ 1 ];
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
$ret [ " root_guid " ] = preg_replace ( " =https?://(.*)/posts/(.*)=ism " , " $ 2 " , $link );
if (( $ret [ " root_guid " ] == $link ) OR ( $ret [ " root_guid " ] == " " ))
return ( false );
2014-04-26 01:35:02 +02:00
2015-08-09 20:39:11 +02:00
return ( $ret );
2014-04-26 01:35:02 +02:00
}
2011-08-19 06:09:44 +02:00
2011-09-22 13:11:39 +02:00
function diaspora_send_images ( $item , $owner , $contact , $images , $public_batch = false ) {
2011-08-24 03:17:35 +02:00
$a = get_app ();
if ( ! count ( $images ))
return ;
$mysite = substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 ) . '/photo' ;
$tpl = get_markup_template ( 'diaspora_photo.tpl' );
foreach ( $images as $image ) {
if ( ! stristr ( $image [ 'path' ], $mysite ))
continue ;
$resource = str_replace ( '.jpg' , '' , $image [ 'file' ]);
$resource = substr ( $resource , 0 , strpos ( $resource , '-' ));
$r = q ( " select * from photo where `resource-id` = '%s' and `uid` = %d limit 1 " ,
dbesc ( $resource ),
intval ( $owner [ 'uid' ])
);
if ( ! count ( $r ))
continue ;
$public = (( $r [ 0 ][ 'allow_cid' ] || $r [ 0 ][ 'allow_gid' ] || $r [ 0 ][ 'deny_cid' ] || $r [ 0 ][ 'deny_gid' ]) ? 'false' : 'true' );
2014-03-09 09:19:14 +01:00
$msg = replace_macros ( $tpl , array (
2011-08-24 03:17:35 +02:00
'$path' => xmlify ( $image [ 'path' ]),
'$filename' => xmlify ( $image [ 'file' ]),
'$msg_guid' => xmlify ( $image [ 'guid' ]),
'$guid' => xmlify ( $r [ 0 ][ 'guid' ]),
'$handle' => xmlify ( $image [ 'handle' ]),
'$public' => xmlify ( $public ),
2011-08-30 03:01:59 +02:00
'$created_at' => xmlify ( datetime_convert ( 'UTC' , 'UTC' , $r [ 0 ][ 'created' ], 'Y-m-d H:i:s \U\T\C' ))
2011-08-24 03:17:35 +02:00
));
2011-08-24 10:21:24 +02:00
2011-08-24 03:17:35 +02:00
logger ( 'diaspora_send_photo: base message: ' . $msg , LOGGER_DATA );
2016-01-11 14:24:54 +01:00
logger ( 'send guid ' . $r [ 0 ][ 'guid' ], LOGGER_DEBUG );
2011-09-22 13:11:39 +02:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $owner , $contact , $owner [ 'uprvkey' ], $contact [ 'pubkey' ], $public_batch )));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$owner,$contact,$owner['uprvkey'],$contact['pubkey'],$public_batch));
2011-08-24 03:17:35 +02:00
2016-01-11 14:24:54 +01:00
diaspora_transmit ( $owner , $contact , $slap , $public_batch , false , $r [ 0 ][ 'guid' ]);
2011-08-24 03:17:35 +02:00
}
}
2011-09-22 13:11:39 +02:00
function diaspora_send_followup ( $item , $owner , $contact , $public_batch = false ) {
2011-08-19 06:09:44 +02:00
$a = get_app ();
2011-08-19 11:24:30 +02:00
$myaddr = $owner [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
2012-06-17 07:58:22 +02:00
// $theiraddr = $contact['addr'];
2011-08-19 06:09:44 +02:00
2013-01-12 14:56:45 +01:00
// Diaspora doesn't support threaded comments, but some
// versions of Diaspora (i.e. Diaspora-pistos) support
// likes on comments
2013-01-12 14:52:15 +01:00
if ( $item [ 'verb' ] === ACTIVITY_LIKE && $item [ 'thr-parent' ]) {
2012-07-13 02:53:51 +02:00
$p = q ( " select guid, type, uri, `parent-uri` from item where uri = '%s' limit 1 " ,
2015-08-09 20:39:11 +02:00
dbesc ( $item [ 'thr-parent' ])
2012-07-13 02:53:51 +02:00
);
}
2013-01-12 14:52:15 +01:00
else {
2012-07-13 02:53:51 +02:00
// The first item in the `item` table with the parent id is the parent. However, MySQL doesn't always
// return the items ordered by `item`.`id`, in which case the wrong item is chosen as the parent.
// The only item with `parent` and `id` as the parent id is the parent item.
$p = q ( " select guid, type, uri, `parent-uri` from item where parent = %d and id = %d limit 1 " ,
intval ( $item [ 'parent' ]),
intval ( $item [ 'parent' ])
);
2013-01-12 14:52:15 +01:00
}
2011-08-19 06:09:44 +02:00
if ( count ( $p ))
2012-07-13 02:53:51 +02:00
$parent = $p [ 0 ];
2011-08-19 06:09:44 +02:00
else
return ;
if ( $item [ 'verb' ] === ACTIVITY_LIKE ) {
$tpl = get_markup_template ( 'diaspora_like.tpl' );
$like = true ;
2012-07-13 02:53:51 +02:00
$target_type = ( $parent [ 'uri' ] === $parent [ 'parent-uri' ] ? 'Post' : 'Comment' );
// $target_type = (strpos($parent['type'], 'comment') ? 'Comment' : 'Post');
2012-06-03 07:56:42 +02:00
// $positive = (($item['deleted']) ? 'false' : 'true');
$positive = 'true' ;
2012-06-03 19:12:16 +02:00
if (( $item [ 'deleted' ]))
logger ( 'diaspora_send_followup: received deleted "like". Those should go to diaspora_send_retraction' );
2011-08-19 06:09:44 +02:00
}
else {
$tpl = get_markup_template ( 'diaspora_comment.tpl' );
$like = false ;
}
2011-08-27 02:52:24 +02:00
$text = html_entity_decode ( bb2diaspora ( $item [ 'body' ]));
2011-08-19 06:09:44 +02:00
// sign it
if ( $like )
2016-01-21 16:01:46 +01:00
$signed_text = $positive . ';' . $item [ 'guid' ] . ';' . $target_type . ';' . $parent [ 'guid' ] . ';' . $myaddr ;
2011-08-19 06:09:44 +02:00
else
2012-07-13 02:53:51 +02:00
$signed_text = $item [ 'guid' ] . ';' . $parent [ 'guid' ] . ';' . $text . ';' . $myaddr ;
2011-08-19 06:09:44 +02:00
2011-09-15 04:33:42 +02:00
$authorsig = base64_encode ( rsa_sign ( $signed_text , $owner [ 'uprvkey' ], 'sha256' ));
2011-08-19 06:09:44 +02:00
$msg = replace_macros ( $tpl , array (
'$guid' => xmlify ( $item [ 'guid' ]),
2012-07-13 02:53:51 +02:00
'$parent_guid' => xmlify ( $parent [ 'guid' ]),
2011-08-19 06:09:44 +02:00
'$target_type' => xmlify ( $target_type ),
'$authorsig' => xmlify ( $authorsig ),
2011-08-19 11:24:30 +02:00
'$body' => xmlify ( $text ),
2011-08-19 06:09:44 +02:00
'$positive' => xmlify ( $positive ),
2011-08-19 11:24:30 +02:00
'$handle' => xmlify ( $myaddr )
2011-08-19 06:09:44 +02:00
));
logger ( 'diaspora_followup: base message: ' . $msg , LOGGER_DATA );
2016-01-11 14:24:54 +01:00
logger ( 'send guid ' . $item [ 'guid' ], LOGGER_DEBUG );
2011-08-19 06:09:44 +02:00
2011-09-22 13:11:39 +02:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $owner , $contact , $owner [ 'uprvkey' ], $contact [ 'pubkey' ], $public_batch )));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$owner,$contact,$owner['uprvkey'],$contact['pubkey'],$public_batch));
2011-08-19 06:09:44 +02:00
2016-01-11 14:24:54 +01:00
return ( diaspora_transmit ( $owner , $contact , $slap , $public_batch , false , $item [ 'guid' ]));
2011-08-19 06:09:44 +02:00
}
2011-09-22 13:11:39 +02:00
function diaspora_send_relay ( $item , $owner , $contact , $public_batch = false ) {
2011-08-19 06:09:44 +02:00
2011-08-19 11:24:30 +02:00
$a = get_app ();
2012-06-01 03:40:12 +02:00
$myaddr = $owner [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
2012-06-17 07:58:22 +02:00
// $theiraddr = $contact['addr'];
2013-01-12 14:56:45 +01:00
// Diaspora doesn't support threaded comments, but some
// versions of Diaspora (i.e. Diaspora-pistos) support
// likes on comments
2013-01-12 14:52:15 +01:00
if ( $item [ 'verb' ] === ACTIVITY_LIKE && $item [ 'thr-parent' ]) {
2012-07-13 02:53:51 +02:00
$p = q ( " select guid, type, uri, `parent-uri` from item where uri = '%s' limit 1 " ,
2015-08-09 20:39:11 +02:00
dbesc ( $item [ 'thr-parent' ])
2012-07-13 02:53:51 +02:00
);
}
2013-01-12 14:52:15 +01:00
else {
2012-07-13 02:53:51 +02:00
// The first item in the `item` table with the parent id is the parent. However, MySQL doesn't always
// return the items ordered by `item`.`id`, in which case the wrong item is chosen as the parent.
// The only item with `parent` and `id` as the parent id is the parent item.
$p = q ( " select guid, type, uri, `parent-uri` from item where parent = %d and id = %d limit 1 " ,
2012-08-10 06:06:18 +02:00
intval ( $item [ 'parent' ]),
intval ( $item [ 'parent' ])
2012-07-13 02:53:51 +02:00
);
2013-01-12 14:52:15 +01:00
}
2011-08-19 06:09:44 +02:00
if ( count ( $p ))
2012-07-13 02:53:51 +02:00
$parent = $p [ 0 ];
2011-08-19 06:09:44 +02:00
else
return ;
2012-06-01 03:40:12 +02:00
$like = false ;
$relay_retract = false ;
$sql_sign_id = 'iid' ;
2012-06-03 00:11:31 +02:00
if ( $item [ 'deleted' ]) {
$relay_retract = true ;
2012-06-17 07:58:22 +02:00
2012-06-03 00:11:31 +02:00
$target_type = ( ( $item [ 'verb' ] === ACTIVITY_LIKE ) ? 'Like' : 'Comment' );
2012-06-17 07:58:22 +02:00
$sql_sign_id = 'retract_iid' ;
$tpl = get_markup_template ( 'diaspora_relayable_retraction.tpl' );
2012-06-03 00:11:31 +02:00
}
elseif ( $item [ 'verb' ] === ACTIVITY_LIKE ) {
2011-08-19 06:09:44 +02:00
$like = true ;
2012-06-17 07:58:22 +02:00
2012-07-13 02:53:51 +02:00
$target_type = ( $parent [ 'uri' ] === $parent [ 'parent-uri' ] ? 'Post' : 'Comment' );
2012-06-03 07:56:42 +02:00
// $positive = (($item['deleted']) ? 'false' : 'true');
$positive = 'true' ;
2012-06-17 07:58:22 +02:00
$tpl = get_markup_template ( 'diaspora_like_relay.tpl' );
2011-08-19 06:09:44 +02:00
}
2012-06-17 07:58:22 +02:00
else { // item is a comment
2012-06-03 00:11:31 +02:00
$tpl = get_markup_template ( 'diaspora_comment_relay.tpl' );
2011-08-19 06:09:44 +02:00
}
2012-06-01 03:40:12 +02:00
2012-06-03 19:12:16 +02:00
// fetch the original signature if the relayable was created by a Diaspora
// or DFRN user. Relayables for other networks are not supported.
2011-08-19 06:09:44 +02:00
2016-01-20 04:22:07 +01:00
$r = q ( " SELECT `signed_text`, `signature`, `signer` FROM `sign` WHERE " . $sql_sign_id . " = %d LIMIT 1 " ,
2011-08-26 16:29:22 +02:00
intval ( $item [ 'id' ])
);
2016-01-20 04:22:07 +01:00
if ( count ( $r )) {
2011-08-26 16:29:22 +02:00
$orig_sign = $r [ 0 ];
$signed_text = $orig_sign [ 'signed_text' ];
$authorsig = $orig_sign [ 'signature' ];
2011-08-29 09:51:08 +02:00
$handle = $orig_sign [ 'signer' ];
2016-01-20 04:22:07 +01:00
// Split the signed text
$signed_parts = explode ( " ; " , $signed_text );
// Remove the parent guid
array_shift ( $signed_parts );
// Remove the comment guid
array_shift ( $signed_parts );
// Remove the handle
array_pop ( $signed_parts );
// Glue the parts together
$text = implode ( " ; " , $signed_parts );
2011-08-26 16:29:22 +02:00
}
else {
2016-01-20 04:22:07 +01:00
// This part is meant for cases where we don't have the signatur. (Which shouldn't happen with posts from Diaspora and Friendica)
// This means that the comment won't be accepted by newer Diaspora servers
2011-08-28 14:00:30 +02:00
2016-01-20 04:22:07 +01:00
$body = $item [ 'body' ];
$text = html_entity_decode ( bb2diaspora ( $body ));
2012-08-10 06:06:18 +02:00
2016-01-20 04:22:07 +01:00
$handle = diaspora_handle_from_contact ( $item [ 'contact-id' ]);
if ( ! $handle )
return ;
2011-08-26 16:29:22 +02:00
2016-01-20 04:22:07 +01:00
if ( $relay_retract )
$signed_text = $item [ 'guid' ] . ';' . $target_type ;
elseif ( $like )
$signed_text = $item [ 'guid' ] . ';' . $target_type . ';' . $parent [ 'guid' ] . ';' . $positive . ';' . $handle ;
else
$signed_text = $item [ 'guid' ] . ';' . $parent [ 'guid' ] . ';' . $text . ';' . $handle ;
$authorsig = base64_encode ( rsa_sign ( $signed_text , $owner [ 'uprvkey' ], 'sha256' ));
}
2012-06-24 06:04:20 +02:00
2012-06-17 07:58:22 +02:00
// Sign the relayable with the top-level owner's signature
2016-01-20 04:22:07 +01:00
$parentauthorsig = base64_encode ( rsa_sign ( $signed_text , $owner [ 'uprvkey' ], 'sha256' ));
2011-08-19 06:09:44 +02:00
$msg = replace_macros ( $tpl , array (
'$guid' => xmlify ( $item [ 'guid' ]),
2012-07-13 02:53:51 +02:00
'$parent_guid' => xmlify ( $parent [ 'guid' ]),
2011-08-19 06:09:44 +02:00
'$target_type' => xmlify ( $target_type ),
2012-06-01 03:40:12 +02:00
'$authorsig' => xmlify ( $authorsig ),
2011-08-19 06:09:44 +02:00
'$parentsig' => xmlify ( $parentauthorsig ),
2011-08-26 16:29:22 +02:00
'$body' => xmlify ( $text ),
2011-08-19 06:09:44 +02:00
'$positive' => xmlify ( $positive ),
2011-08-29 09:51:08 +02:00
'$handle' => xmlify ( $handle )
2011-08-19 06:09:44 +02:00
));
2016-01-21 13:57:21 +01:00
logger ( 'diaspora_send_relay: base message: ' . $msg , LOGGER_DATA );
2016-01-11 14:24:54 +01:00
logger ( 'send guid ' . $item [ 'guid' ], LOGGER_DEBUG );
2011-08-19 06:09:44 +02:00
2011-09-22 13:11:39 +02:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $owner , $contact , $owner [ 'uprvkey' ], $contact [ 'pubkey' ], $public_batch )));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$owner,$contact,$owner['uprvkey'],$contact['pubkey'],$public_batch));
2011-08-19 06:09:44 +02:00
2016-01-11 14:24:54 +01:00
return ( diaspora_transmit ( $owner , $contact , $slap , $public_batch , false , $item [ 'guid' ]));
2011-08-19 06:09:44 +02:00
}
2011-09-22 13:11:39 +02:00
function diaspora_send_retraction ( $item , $owner , $contact , $public_batch = false ) {
2011-08-19 06:09:44 +02:00
2011-08-23 05:35:34 +02:00
$a = get_app ();
$myaddr = $owner [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
2011-08-19 06:09:44 +02:00
2012-06-03 07:56:42 +02:00
// Check whether the retraction is for a top-level post or whether it's a relayable
if ( $item [ 'uri' ] !== $item [ 'parent-uri' ] ) {
2012-06-01 03:40:12 +02:00
$tpl = get_markup_template ( 'diaspora_relay_retraction.tpl' );
2012-06-03 07:56:42 +02:00
$target_type = (( $item [ 'verb' ] === ACTIVITY_LIKE ) ? 'Like' : 'Comment' );
2012-06-01 03:40:12 +02:00
}
else {
2015-05-25 13:27:45 +02:00
2012-06-01 03:40:12 +02:00
$tpl = get_markup_template ( 'diaspora_signed_retract.tpl' );
$target_type = 'StatusMessage' ;
}
$signed_text = $item [ 'guid' ] . ';' . $target_type ;
2011-11-05 22:45:29 +01:00
2011-08-23 05:35:34 +02:00
$msg = replace_macros ( $tpl , array (
2012-06-01 03:40:12 +02:00
'$guid' => xmlify ( $item [ 'guid' ]),
'$type' => xmlify ( $target_type ),
'$handle' => xmlify ( $myaddr ),
'$signature' => xmlify ( base64_encode ( rsa_sign ( $signed_text , $owner [ 'uprvkey' ], 'sha256' )))
2011-08-23 05:35:34 +02:00
));
2011-08-19 06:09:44 +02:00
2016-01-11 14:24:54 +01:00
logger ( 'send guid ' . $item [ 'guid' ], LOGGER_DEBUG );
2011-09-22 13:11:39 +02:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $msg , $owner , $contact , $owner [ 'uprvkey' ], $contact [ 'pubkey' ], $public_batch )));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($msg,$owner,$contact,$owner['uprvkey'],$contact['pubkey'],$public_batch));
2011-08-19 06:09:44 +02:00
2016-01-11 14:24:54 +01:00
return ( diaspora_transmit ( $owner , $contact , $slap , $public_batch , false , $item [ 'guid' ]));
2011-08-24 10:21:24 +02:00
}
2011-12-06 09:16:13 +01:00
function diaspora_send_mail ( $item , $owner , $contact ) {
$a = get_app ();
$myaddr = $owner [ 'nickname' ] . '@' . substr ( $a -> get_baseurl (), strpos ( $a -> get_baseurl (), '://' ) + 3 );
$r = q ( " select * from conv where id = %d and uid = %d limit 1 " ,
intval ( $item [ 'convid' ]),
intval ( $item [ 'uid' ])
);
if ( ! count ( $r )) {
logger ( 'diaspora_send_mail: conversation not found.' );
return ;
}
$cnv = $r [ 0 ];
$conv = array (
'guid' => xmlify ( $cnv [ 'guid' ]),
'subject' => xmlify ( $cnv [ 'subject' ]),
'created_at' => xmlify ( datetime_convert ( 'UTC' , 'UTC' , $cnv [ 'created' ], 'Y-m-d H:i:s \U\T\C' )),
'diaspora_handle' => xmlify ( $cnv [ 'creator' ]),
'participant_handles' => xmlify ( $cnv [ 'recips' ])
);
$body = bb2diaspora ( $item [ 'body' ]);
$created = datetime_convert ( 'UTC' , 'UTC' , $item [ 'created' ], 'Y-m-d H:i:s \U\T\C' );
2015-05-25 13:27:45 +02:00
2015-08-13 22:49:57 +02:00
$signed_text = $item [ 'guid' ] . ';' . $cnv [ 'guid' ] . ';' . $body . ';'
2011-12-06 09:16:13 +01:00
. $created . ';' . $myaddr . ';' . $cnv [ 'guid' ];
$sig = base64_encode ( rsa_sign ( $signed_text , $owner [ 'uprvkey' ], 'sha256' ));
$msg = array (
'guid' => xmlify ( $item [ 'guid' ]),
2011-12-07 04:15:42 +01:00
'parent_guid' => xmlify ( $cnv [ 'guid' ]),
2015-08-13 22:49:57 +02:00
'parent_author_signature' => xmlify ( $sig ),
2011-12-06 09:16:13 +01:00
'author_signature' => xmlify ( $sig ),
'text' => xmlify ( $body ),
'created_at' => xmlify ( $created ),
'diaspora_handle' => xmlify ( $myaddr ),
'conversation_guid' => xmlify ( $cnv [ 'guid' ])
);
2011-12-07 04:15:42 +01:00
if ( $item [ 'reply' ]) {
$tpl = get_markup_template ( 'diaspora_message.tpl' );
$xmsg = replace_macros ( $tpl , array ( '$msg' => $msg ));
}
else {
$conv [ 'messages' ] = array ( $msg );
$tpl = get_markup_template ( 'diaspora_conversation.tpl' );
$xmsg = replace_macros ( $tpl , array ( '$conv' => $conv ));
}
2011-12-06 09:16:13 +01:00
2011-12-07 04:15:42 +01:00
logger ( 'diaspora_conversation: ' . print_r ( $xmsg , true ), LOGGER_DATA );
2016-01-11 14:24:54 +01:00
logger ( 'send guid ' . $item [ 'guid' ], LOGGER_DEBUG );
2011-12-06 09:16:13 +01:00
2011-12-07 04:15:42 +01:00
$slap = 'xml=' . urlencode ( urlencode ( diaspora_msg_build ( $xmsg , $owner , $contact , $owner [ 'uprvkey' ], $contact [ 'pubkey' ], false )));
2012-06-23 12:42:01 +02:00
//$slap = 'xml=' . urlencode(diaspora_msg_build($xmsg,$owner,$contact,$owner['uprvkey'],$contact['pubkey'],false));
2011-12-06 09:16:13 +01:00
2016-01-11 14:24:54 +01:00
return ( diaspora_transmit ( $owner , $contact , $slap , false , false , $item [ 'guid' ]));
2011-12-06 09:16:13 +01:00
}
2011-08-24 10:21:24 +02:00
2016-01-11 14:24:54 +01:00
function diaspora_transmit ( $owner , $contact , $slap , $public_batch , $queue_run = false , $guid = " " ) {
2011-08-24 10:21:24 +02:00
2012-04-05 05:48:35 +02:00
$enabled = intval ( get_config ( 'system' , 'diaspora_enabled' ));
if ( ! $enabled ) {
return 200 ;
}
2011-08-24 10:21:24 +02:00
$a = get_app ();
2011-09-22 13:11:39 +02:00
$logid = random_string ( 4 );
2011-11-07 01:48:13 +01:00
$dest_url = (( $public_batch ) ? $contact [ 'batch' ] : $contact [ 'notify' ]);
if ( ! $dest_url ) {
logger ( 'diaspora_transmit: no url for contact: ' . $contact [ 'id' ] . ' batch mode =' . $public_batch );
return 0 ;
2016-01-11 14:24:54 +01:00
}
2011-11-07 01:48:13 +01:00
2016-01-11 14:24:54 +01:00
logger ( 'diaspora_transmit: ' . $logid . '-' . $guid . ' ' . $dest_url );
2011-11-07 01:48:13 +01:00
2012-07-09 17:39:09 +02:00
if ( ( ! $queue_run ) && ( was_recently_delayed ( $contact [ 'id' ])) ) {
2012-05-08 00:54:49 +02:00
$return_code = 0 ;
}
2012-01-27 05:46:42 +01:00
else {
2013-10-02 22:17:56 +02:00
if ( ! intval ( get_config ( 'system' , 'diaspora_test' ))) {
2012-05-08 00:54:49 +02:00
post_url ( $dest_url . '/' , $slap );
$return_code = $a -> get_curl_code ();
2013-10-02 22:17:56 +02:00
} else {
2012-05-08 00:54:49 +02:00
logger ( 'diaspora_transmit: test_mode' );
return 200 ;
}
2012-01-27 05:46:42 +01:00
}
2013-10-02 22:17:56 +02:00
2016-01-11 14:24:54 +01:00
logger ( 'diaspora_transmit: ' . $logid . '-' . $guid . ' returns: ' . $return_code );
2011-08-24 10:21:24 +02:00
2011-11-07 01:48:13 +01:00
if (( ! $return_code ) || (( $return_code == 503 ) && ( stristr ( $a -> get_curl_headers (), 'retry-after' )))) {
2011-08-24 10:21:24 +02:00
logger ( 'diaspora_transmit: queue message' );
2011-11-07 01:48:13 +01:00
$r = q ( " SELECT id from queue where cid = %d and network = '%s' and content = '%s' and batch = %d limit 1 " ,
intval ( $contact [ 'id' ]),
dbesc ( NETWORK_DIASPORA ),
dbesc ( $slap ),
intval ( $public_batch )
);
if ( count ( $r )) {
logger ( 'diaspora_transmit: add_to_queue ignored - identical item already in queue' );
}
else {
// queue message for redelivery
add_to_queue ( $contact [ 'id' ], NETWORK_DIASPORA , $slap , $public_batch );
}
2011-08-24 10:21:24 +02:00
}
2011-08-19 06:09:44 +02:00
2011-09-28 04:27:47 +02:00
2011-08-24 10:21:24 +02:00
return (( $return_code ) ? $return_code : ( - 1 ));
}
2015-08-09 20:39:11 +02:00
function diaspora_fetch_relay () {
$serverdata = get_config ( " system " , " relay_server " );
if ( $serverdata == " " )
return array ();
$relay = array ();
$servers = explode ( " , " , $serverdata );
foreach ( $servers AS $server ) {
$server = trim ( $server );
$batch = $server . " /receive/public " ;
$relais = q ( " SELECT `batch`, `id`, `name`,`network` FROM `contact` WHERE `uid` = 0 AND `batch` = '%s' LIMIT 1 " , dbesc ( $batch ));
if ( ! $relais ) {
$addr = " relay@ " . str_replace ( " http:// " , " " , normalise_link ( $server ));
$r = q ( " INSERT INTO `contact` (`uid`, `created`, `name`, `nick`, `addr`, `url`, `nurl`, `batch`, `network`, `rel`, `blocked`, `pending`, `writable`, `name-date`, `uri-date`, `avatar-date`)
VALUES ( 0 , '%s' , '%s' , 'relay' , '%s' , '%s' , '%s' , '%s' , '%s' , % d , 0 , 0 , 1 , '%s' , '%s' , '%s' ) " ,
datetime_convert (),
dbesc ( $addr ),
dbesc ( $addr ),
dbesc ( $server ),
dbesc ( normalise_link ( $server )),
dbesc ( $batch ),
dbesc ( NETWORK_DIASPORA ),
intval ( CONTACT_IS_FOLLOWER ),
dbesc ( datetime_convert ()),
dbesc ( datetime_convert ()),
dbesc ( datetime_convert ())
);
$relais = q ( " SELECT `batch`, `id`, `name`,`network` FROM `contact` WHERE `uid` = 0 AND `batch` = '%s' LIMIT 1 " , dbesc ( $batch ));
if ( $relais )
$relay [] = $relais [ 0 ];
} else
$relay [] = $relais [ 0 ];
}
return $relay ;
}