180 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			180 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| // This file is part of GNU social - https://www.gnu.org/software/social
 | |
| //
 | |
| // GNU social is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU Affero General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // GNU social is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| // GNU Affero General Public License for more details.
 | |
| //
 | |
| // You should have received a copy of the GNU Affero General Public License
 | |
| // along with GNU social.  If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| /**
 | |
|  * TheFreeNetwork, "automagic" migration of internal remote profiles
 | |
|  * between different federation protocols.
 | |
|  *
 | |
|  * @package   GNUsocial
 | |
|  * @author    Bruno Casteleiro <brunoccast@fc.up.pt>
 | |
|  * @copyright 2019 Free Software Foundation, Inc http://www.fsf.org
 | |
|  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
 | |
|  */
 | |
| 
 | |
| defined('GNUSOCIAL') || die();
 | |
| 
 | |
| /**
 | |
|  * Class TheFreeNetworkModule
 | |
|  * This module ensures that multiple protocols serving the same purpose won't result in duplicated data.
 | |
|  * This class is not to be extended but a developer implementing a new protocol should be aware of it and notify the
 | |
|  * StartTFNCensus event.
 | |
|  *
 | |
|  * @category  Module
 | |
|  * @package   GNUsocial
 | |
|  * @author    Bruno Casteleiro <brunoccast@fc.up.pt>
 | |
|  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
 | |
|  */
 | |
| class TheFreeNetworkModule extends Module
 | |
| {
 | |
|     const MODULE_VERSION = '0.1.0alpha0';
 | |
| 
 | |
|     public $protocols = null; // protocols TFN should handle
 | |
| 
 | |
|     private $lrdd = false; // whether LRDD plugin is active or not
 | |
| 
 | |
|     /**
 | |
|      * Initialize TFN
 | |
|      *
 | |
|      * @return bool hook value
 | |
|      */
 | |
|     public function onInitializePlugin(): bool
 | |
|     {
 | |
|         // some protocol plugins can be unactivated,
 | |
|         // require needed classes
 | |
|         $plugin_dir = dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'plugins';
 | |
| 
 | |
|         foreach ($this->protocols as $protocol => $class) {
 | |
|             require_once $plugin_dir . DIRECTORY_SEPARATOR . $protocol . DIRECTORY_SEPARATOR . 'classes' . DIRECTORY_SEPARATOR . $class . '.php';
 | |
|         }
 | |
| 
 | |
|         // $lrdd flag
 | |
|         $this->lrdd = PluginList::isPluginActive("LRDD");
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * A new remote profile is being added, check if we
 | |
|      * already have someone with the same URI.
 | |
|      *
 | |
|      * @param string $uri
 | |
|      * @param string $class profile class that triggered this event
 | |
|      * @param int|null &$profile_id Profile:id associated with the remote entity found
 | |
|      * @return bool hook flag
 | |
|      */
 | |
|     public function onStartTFNLookup(string $uri, string $class, int &$profile_id = null): bool
 | |
|     {
 | |
|         $profile_id = $this->lookup($uri, $class);
 | |
| 
 | |
|         $perf = common_config('performance', 'high');
 | |
| 
 | |
|         if (is_null($profile_id) && !$perf && $this->lrdd) {
 | |
|             // Force lookup with online resources
 | |
|             $profile_id = $this->lookup($uri, $class, true);
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * A new remote profile was sucessfully added, delete
 | |
|      * other remotes associated with the same Profile entity.
 | |
|      *
 | |
|      * @param string $class profile class that triggered this event
 | |
|      * @param int $profile_id Profile:id associated with the new remote profile
 | |
|      * @return bool hook flag
 | |
|      */
 | |
|     public function onEndTFNLookup(string $class, int $profile_id): bool
 | |
|     {
 | |
|         foreach ($this->protocols as $p => $cls) {
 | |
|             if ($cls != $class) {
 | |
|                 $profile = $cls::getKV('profile_id', $profile_id);
 | |
|                 if ($profile instanceof $cls) {
 | |
|                     $this->log(LOG_INFO, 'Deleting remote ' . $cls . ' associated with Profile:' . $profile_id);
 | |
|                     $i = new $cls();
 | |
|                     $i->profile_id = $profile_id;
 | |
|                     $i->delete();
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Plugin version information
 | |
|      *
 | |
|      * @param array $versions
 | |
|      * @return bool hook true
 | |
|      */
 | |
|     public function onPluginVersion(array &$versions): bool
 | |
|     {
 | |
|         $versions[] = [
 | |
|             'name' => 'TheFreeNetwork',
 | |
|             'version' => self::MODULE_VERSION,
 | |
|             'author' => 'Bruno Casteleiro',
 | |
|             'homepage' => 'https://notabug.org/diogo/gnu-social/src/nightly/plugins/TheFreeNetwork',
 | |
|             // TRANS: Module description.
 | |
|             'rawdescription' => '"Automagically" migrate internal remote profiles between different federation protocols'
 | |
|         ];
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Search remote profile tables to find someone by URI.
 | |
|      * When set to search online, it will grab the remote
 | |
|      * entity's aliases and search with each one.
 | |
|      * The table associated with the class that triggered
 | |
|      * this lookup process will be discarded in the search.
 | |
|      *
 | |
|      * @param string $uri
 | |
|      * @param string $class
 | |
|      * @param bool $online
 | |
|      * @return int|null Profile:id associated with the remote entity found
 | |
|      */
 | |
|     private function lookup(string $uri, string $class, bool $online = false): ?int
 | |
|     {
 | |
|         if ($online) {
 | |
|             $this->log(LOG_INFO, 'Searching with online resources for a remote profile with URI: ' . $uri);
 | |
|             $all_ids = LRDDPlugin::grab_profile_aliases($uri);
 | |
|         } else {
 | |
|             $this->log(LOG_INFO, 'Searching for a remote profile with URI: ' . $uri);
 | |
|             $all_ids = [$uri];
 | |
|         }
 | |
| 
 | |
|         if ($all_ids == null) {
 | |
|             $this->log(LOG_INFO, 'Unable to find a remote profile with URI: ' . $uri);
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         foreach ($this->protocols as $p => $cls) {
 | |
|             if ($cls != $class) {
 | |
|                 foreach ($all_ids as $alias) {
 | |
|                     $profile = $cls::getKV('uri', $alias);
 | |
|                     if ($profile instanceof $cls) {
 | |
|                         $this->log(LOG_INFO, 'Found a remote ' . $cls . ' associated with Profile:' . $profile->getID());
 | |
|                         return $profile->getID();
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $this->log(LOG_INFO, 'Unable to find a remote profile with URI: ' . $uri);
 | |
|         return null;
 | |
|     }
 | |
| }
 |