Merge pull request #4968 from pixelfed/staging

Add Profile Migrations
This commit is contained in:
daniel 2024-03-05 00:23:35 -07:00 committed by GitHub
commit 712b6d27a9
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
27 changed files with 1125 additions and 278 deletions

View file

@ -2,18 +2,18 @@
namespace App\Http\Controllers\Admin; namespace App\Http\Controllers\Admin;
use Artisan, Cache, DB;
use Illuminate\Http\Request;
use Carbon\Carbon;
use App\{Comment, Like, Media, Page, Profile, Report, Status, User};
use App\Models\InstanceActor;
use App\Http\Controllers\Controller;
use App\Util\Lexer\PrettyNumber;
use App\Models\ConfigCache; use App\Models\ConfigCache;
use App\Models\InstanceActor;
use App\Page;
use App\Profile;
use App\Services\AccountService; use App\Services\AccountService;
use App\Services\ConfigCacheService; use App\Services\ConfigCacheService;
use App\User;
use App\Util\Site\Config; use App\Util\Site\Config;
use Illuminate\Support\Str; use Artisan;
use Cache;
use DB;
use Illuminate\Http\Request;
trait AdminSettingsController trait AdminSettingsController
{ {
@ -21,7 +21,7 @@ trait AdminSettingsController
{ {
$cloud_storage = ConfigCacheService::get('pixelfed.cloud_storage'); $cloud_storage = ConfigCacheService::get('pixelfed.cloud_storage');
$cloud_disk = config('filesystems.cloud'); $cloud_disk = config('filesystems.cloud');
$cloud_ready = !empty(config('filesystems.disks.' . $cloud_disk . '.key')) && !empty(config('filesystems.disks.' . $cloud_disk . '.secret')); $cloud_ready = ! empty(config('filesystems.disks.'.$cloud_disk.'.key')) && ! empty(config('filesystems.disks.'.$cloud_disk.'.secret'));
$types = explode(',', ConfigCacheService::get('pixelfed.media_types')); $types = explode(',', ConfigCacheService::get('pixelfed.media_types'));
$rules = ConfigCacheService::get('app.rules') ? json_decode(ConfigCacheService::get('app.rules'), true) : null; $rules = ConfigCacheService::get('app.rules') ? json_decode(ConfigCacheService::get('app.rules'), true) : null;
$jpeg = in_array('image/jpg', $types) || in_array('image/jpeg', $types); $jpeg = in_array('image/jpg', $types) || in_array('image/jpeg', $types);
@ -35,6 +35,7 @@ trait AdminSettingsController
$openReg = (bool) config_cache('pixelfed.open_registration'); $openReg = (bool) config_cache('pixelfed.open_registration');
$curOnboarding = (bool) config_cache('instance.curated_registration.enabled'); $curOnboarding = (bool) config_cache('instance.curated_registration.enabled');
$regState = $openReg ? 'open' : ($curOnboarding ? 'filtered' : 'closed'); $regState = $openReg ? 'open' : ($curOnboarding ? 'filtered' : 'closed');
$accountMigration = (bool) config_cache('federation.migration');
return view('admin.settings.home', compact( return view('admin.settings.home', compact(
'jpeg', 'jpeg',
@ -48,7 +49,8 @@ trait AdminSettingsController
'cloud_ready', 'cloud_ready',
'availableAdmins', 'availableAdmins',
'currentAdmin', 'currentAdmin',
'regState' 'regState',
'accountMigration'
)); ));
} }
@ -67,41 +69,42 @@ trait AdminSettingsController
'type_mp4' => 'nullable', 'type_mp4' => 'nullable',
'type_webp' => 'nullable', 'type_webp' => 'nullable',
'admin_account_id' => 'nullable', 'admin_account_id' => 'nullable',
'regs' => 'required|in:open,filtered,closed' 'regs' => 'required|in:open,filtered,closed',
'account_migration' => 'nullable',
]); ]);
$orb = false; $orb = false;
$cob = false; $cob = false;
switch($request->input('regs')) { switch ($request->input('regs')) {
case 'open': case 'open':
$orb = true; $orb = true;
$cob = false; $cob = false;
break; break;
case 'filtered': case 'filtered':
$orb = false; $orb = false;
$cob = true; $cob = true;
break; break;
case 'closed': case 'closed':
$orb = false; $orb = false;
$cob = false; $cob = false;
break; break;
} }
ConfigCacheService::put('pixelfed.open_registration', (bool) $orb); ConfigCacheService::put('pixelfed.open_registration', (bool) $orb);
ConfigCacheService::put('instance.curated_registration.enabled', (bool) $cob); ConfigCacheService::put('instance.curated_registration.enabled', (bool) $cob);
if($request->filled('admin_account_id')) { if ($request->filled('admin_account_id')) {
ConfigCacheService::put('instance.admin.pid', $request->admin_account_id); ConfigCacheService::put('instance.admin.pid', $request->admin_account_id);
Cache::forget('api:v1:instance-data:contact'); Cache::forget('api:v1:instance-data:contact');
Cache::forget('api:v1:instance-data-response-v1'); Cache::forget('api:v1:instance-data-response-v1');
} }
if($request->filled('rule_delete')) { if ($request->filled('rule_delete')) {
$index = (int) $request->input('rule_delete'); $index = (int) $request->input('rule_delete');
$rules = ConfigCacheService::get('app.rules'); $rules = ConfigCacheService::get('app.rules');
$json = json_decode($rules, true); $json = json_decode($rules, true);
if(!$rules || empty($json)) { if (! $rules || empty($json)) {
return; return;
} }
unset($json[$index]); unset($json[$index]);
@ -109,6 +112,7 @@ trait AdminSettingsController
ConfigCacheService::put('app.rules', $json); ConfigCacheService::put('app.rules', $json);
Cache::forget('api:v1:instance-data:rules'); Cache::forget('api:v1:instance-data:rules');
Cache::forget('api:v1:instance-data-response-v1'); Cache::forget('api:v1:instance-data-response-v1');
return 200; return 200;
} }
@ -124,8 +128,8 @@ trait AdminSettingsController
]; ];
foreach ($mimes as $key => $value) { foreach ($mimes as $key => $value) {
if($request->input($key) == 'on') { if ($request->input($key) == 'on') {
if(!in_array($value, $media_types)) { if (! in_array($value, $media_types)) {
array_push($media_types, $value); array_push($media_types, $value);
} }
} else { } else {
@ -133,7 +137,7 @@ trait AdminSettingsController
} }
} }
if($media_types !== $media_types_original) { if ($media_types !== $media_types_original) {
ConfigCacheService::put('pixelfed.media_types', implode(',', array_unique($media_types))); ConfigCacheService::put('pixelfed.media_types', implode(',', array_unique($media_types)));
} }
@ -147,15 +151,15 @@ trait AdminSettingsController
'account_limit' => 'pixelfed.max_account_size', 'account_limit' => 'pixelfed.max_account_size',
'custom_css' => 'uikit.custom.css', 'custom_css' => 'uikit.custom.css',
'custom_js' => 'uikit.custom.js', 'custom_js' => 'uikit.custom.js',
'about_title' => 'about.title' 'about_title' => 'about.title',
]; ];
foreach ($keys as $key => $value) { foreach ($keys as $key => $value) {
$cc = ConfigCache::whereK($value)->first(); $cc = ConfigCache::whereK($value)->first();
$val = $request->input($key); $val = $request->input($key);
if($cc && $cc->v != $val) { if ($cc && $cc->v != $val) {
ConfigCacheService::put($value, $val); ConfigCacheService::put($value, $val);
} else if(!empty($val)) { } elseif (! empty($val)) {
ConfigCacheService::put($value, $val); ConfigCacheService::put($value, $val);
} }
} }
@ -175,33 +179,34 @@ trait AdminSettingsController
'account_autofollow' => 'account.autofollow', 'account_autofollow' => 'account.autofollow',
'show_directory' => 'instance.landing.show_directory', 'show_directory' => 'instance.landing.show_directory',
'show_explore_feed' => 'instance.landing.show_explore', 'show_explore_feed' => 'instance.landing.show_explore',
'account_migration' => 'federation.migration',
]; ];
foreach ($bools as $key => $value) { foreach ($bools as $key => $value) {
$active = $request->input($key) == 'on'; $active = $request->input($key) == 'on';
if($key == 'activitypub' && $active && !InstanceActor::exists()) { if ($key == 'activitypub' && $active && ! InstanceActor::exists()) {
Artisan::call('instance:actor'); Artisan::call('instance:actor');
} }
if( $key == 'mobile_apis' && if ($key == 'mobile_apis' &&
$active && $active &&
!file_exists(storage_path('oauth-public.key')) && ! file_exists(storage_path('oauth-public.key')) &&
!file_exists(storage_path('oauth-private.key')) ! file_exists(storage_path('oauth-private.key'))
) { ) {
Artisan::call('passport:keys'); Artisan::call('passport:keys');
Artisan::call('route:cache'); Artisan::call('route:cache');
} }
if(config_cache($value) !== $active) { if (config_cache($value) !== $active) {
ConfigCacheService::put($value, (bool) $active); ConfigCacheService::put($value, (bool) $active);
} }
} }
if($request->filled('new_rule')) { if ($request->filled('new_rule')) {
$rules = ConfigCacheService::get('app.rules'); $rules = ConfigCacheService::get('app.rules');
$val = $request->input('new_rule'); $val = $request->input('new_rule');
if(!$rules) { if (! $rules) {
ConfigCacheService::put('app.rules', json_encode([$val])); ConfigCacheService::put('app.rules', json_encode([$val]));
} else { } else {
$json = json_decode($rules, true); $json = json_decode($rules, true);
@ -212,13 +217,13 @@ trait AdminSettingsController
Cache::forget('api:v1:instance-data-response-v1'); Cache::forget('api:v1:instance-data-response-v1');
} }
if($request->filled('account_autofollow_usernames')) { if ($request->filled('account_autofollow_usernames')) {
$usernames = explode(',', $request->input('account_autofollow_usernames')); $usernames = explode(',', $request->input('account_autofollow_usernames'));
$names = []; $names = [];
foreach($usernames as $n) { foreach ($usernames as $n) {
$p = Profile::whereUsername($n)->first(); $p = Profile::whereUsername($n)->first();
if(!$p) { if (! $p) {
continue; continue;
} }
array_push($names, $p->username); array_push($names, $p->username);
@ -236,6 +241,7 @@ trait AdminSettingsController
{ {
$path = storage_path('app/'.config('app.name')); $path = storage_path('app/'.config('app.name'));
$files = is_dir($path) ? new \DirectoryIterator($path) : []; $files = is_dir($path) ? new \DirectoryIterator($path) : [];
return view('admin.settings.backups', compact('files')); return view('admin.settings.backups', compact('files'));
} }
@ -247,6 +253,7 @@ trait AdminSettingsController
public function settingsStorage(Request $request) public function settingsStorage(Request $request)
{ {
$storage = []; $storage = [];
return view('admin.settings.storage', compact('storage')); return view('admin.settings.storage', compact('storage'));
} }
@ -258,6 +265,7 @@ trait AdminSettingsController
public function settingsPages(Request $request) public function settingsPages(Request $request)
{ {
$pages = Page::orderByDesc('updated_at')->paginate(10); $pages = Page::orderByDesc('updated_at')->paginate(10);
return view('admin.pages.home', compact('pages')); return view('admin.pages.home', compact('pages'));
} }
@ -275,30 +283,31 @@ trait AdminSettingsController
]; ];
switch (config('database.default')) { switch (config('database.default')) {
case 'pgsql': case 'pgsql':
$exp = DB::raw('select version();'); $exp = DB::raw('select version();');
$expQuery = $exp->getValue(DB::connection()->getQueryGrammar()); $expQuery = $exp->getValue(DB::connection()->getQueryGrammar());
$sys['database'] = [ $sys['database'] = [
'name' => 'Postgres', 'name' => 'Postgres',
'version' => explode(' ', DB::select($expQuery)[0]->version)[1] 'version' => explode(' ', DB::select($expQuery)[0]->version)[1],
]; ];
break; break;
case 'mysql': case 'mysql':
$exp = DB::raw('select version()'); $exp = DB::raw('select version()');
$expQuery = $exp->getValue(DB::connection()->getQueryGrammar()); $expQuery = $exp->getValue(DB::connection()->getQueryGrammar());
$sys['database'] = [ $sys['database'] = [
'name' => 'MySQL', 'name' => 'MySQL',
'version' => DB::select($expQuery)[0]->{'version()'} 'version' => DB::select($expQuery)[0]->{'version()'},
]; ];
break; break;
default: default:
$sys['database'] = [ $sys['database'] = [
'name' => 'Unknown', 'name' => 'Unknown',
'version' => '?' 'version' => '?',
]; ];
break; break;
} }
return view('admin.settings.system', compact('sys')); return view('admin.settings.system', compact('sys'));
} }
} }

View file

@ -2,11 +2,13 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use Illuminate\Http\Request;
use App\Util\Lexer\Nickname;
use App\Util\Webfinger\WebfingerUrl;
use App\Models\ProfileAlias; use App\Models\ProfileAlias;
use App\Models\ProfileMigration;
use App\Services\AccountService;
use App\Services\WebfingerService; use App\Services\WebfingerService;
use App\Util\Lexer\Nickname;
use Cache;
use Illuminate\Http\Request;
class ProfileAliasController extends Controller class ProfileAliasController extends Controller
{ {
@ -18,31 +20,47 @@ class ProfileAliasController extends Controller
public function index(Request $request) public function index(Request $request)
{ {
$aliases = $request->user()->profile->aliases; $aliases = $request->user()->profile->aliases;
return view('settings.aliases.index', compact('aliases')); return view('settings.aliases.index', compact('aliases'));
} }
public function store(Request $request) public function store(Request $request)
{ {
$this->validate($request, [ $this->validate($request, [
'acct' => 'required' 'acct' => 'required',
]); ]);
$acct = $request->input('acct'); $acct = $request->input('acct');
if($request->user()->profile->aliases->count() >= 3) { $nn = Nickname::normalizeProfileUrl($acct);
if (! $nn) {
return back()->with('error', 'Invalid account alias.');
}
if ($nn['domain'] === config('pixelfed.domain.app')) {
if (strtolower($nn['username']) == ($request->user()->username)) {
return back()->with('error', 'You cannot add an alias to your own account.');
}
}
if ($request->user()->profile->aliases->count() >= 3) {
return back()->with('error', 'You can only add 3 account aliases.'); return back()->with('error', 'You can only add 3 account aliases.');
} }
$webfingerService = WebfingerService::lookup($acct); $webfingerService = WebfingerService::lookup($acct);
if(!$webfingerService || !isset($webfingerService['url'])) { $webfingerUrl = WebfingerService::rawGet($acct);
if (! $webfingerService || ! isset($webfingerService['url']) || ! $webfingerUrl || empty($webfingerUrl)) {
return back()->with('error', 'Invalid account, cannot add alias at this time.'); return back()->with('error', 'Invalid account, cannot add alias at this time.');
} }
$alias = new ProfileAlias; $alias = new ProfileAlias;
$alias->profile_id = $request->user()->profile_id; $alias->profile_id = $request->user()->profile_id;
$alias->acct = $acct; $alias->acct = $acct;
$alias->uri = $webfingerService['url']; $alias->uri = $webfingerUrl;
$alias->save(); $alias->save();
Cache::forget('pf:activitypub:user-object:by-id:'.$request->user()->profile_id);
return back()->with('status', 'Successfully added alias!'); return back()->with('status', 'Successfully added alias!');
} }
@ -50,14 +68,25 @@ class ProfileAliasController extends Controller
{ {
$this->validate($request, [ $this->validate($request, [
'acct' => 'required', 'acct' => 'required',
'id' => 'required|exists:profile_aliases' 'id' => 'required|exists:profile_aliases',
]); ]);
$pid = $request->user()->profile_id;
$alias = ProfileAlias::where('profile_id', $request->user()->profile_id) $acct = $request->input('acct');
->where('acct', $request->input('acct')) $alias = ProfileAlias::where('profile_id', $pid)
->where('acct', $acct)
->findOrFail($request->input('id')); ->findOrFail($request->input('id'));
$migration = ProfileMigration::whereProfileId($pid)
->whereAcct($acct)
->first();
if ($migration) {
$request->user()->profile->update([
'moved_to_profile_id' => null,
]);
}
$alias->delete(); $alias->delete();
Cache::forget('pf:activitypub:user-object:by-id:'.$pid);
AccountService::del($pid);
return back()->with('status', 'Successfully deleted alias!'); return back()->with('status', 'Successfully deleted alias!');
} }

View file

@ -0,0 +1,72 @@
<?php
namespace App\Http\Controllers;
use App\Http\Requests\ProfileMigrationStoreRequest;
use App\Jobs\ProfilePipeline\ProfileMigrationDeliverMoveActivityPipeline;
use App\Jobs\ProfilePipeline\ProfileMigrationMoveFollowersPipeline;
use App\Models\ProfileAlias;
use App\Models\ProfileMigration;
use App\Services\AccountService;
use App\Services\WebfingerService;
use App\Util\ActivityPub\Helpers;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Bus;
class ProfileMigrationController extends Controller
{
public function __construct()
{
$this->middleware('auth');
}
public function index(Request $request)
{
abort_if((bool) config_cache('federation.activitypub.enabled') === false, 404);
if ((bool) config_cache('federation.migration') === false) {
return redirect(route('help.account-migration'));
}
$hasExistingMigration = ProfileMigration::whereProfileId($request->user()->profile_id)
->where('created_at', '>', now()->subDays(30))
->exists();
return view('settings.migration.index', compact('hasExistingMigration'));
}
public function store(ProfileMigrationStoreRequest $request)
{
abort_if((bool) config_cache('federation.activitypub.enabled') === false, 404);
$acct = WebfingerService::rawGet($request->safe()->acct);
if (! $acct) {
return redirect()->back()->withErrors(['acct' => 'The new account you provided is not responding to our requests.']);
}
$newAccount = Helpers::profileFetch($acct);
if (! $newAccount) {
return redirect()->back()->withErrors(['acct' => 'An error occured, please try again later. Code: res-failed-account-fetch']);
}
$user = $request->user();
ProfileAlias::updateOrCreate([
'profile_id' => $user->profile_id,
'acct' => $request->safe()->acct,
'uri' => $acct,
]);
$migration = ProfileMigration::create([
'profile_id' => $request->user()->profile_id,
'acct' => $request->safe()->acct,
'followers_count' => $request->user()->profile->followers_count,
'target_profile_id' => $newAccount['id'],
]);
$user->profile->update([
'moved_to_profile_id' => $newAccount->id,
'indexable' => false,
]);
AccountService::del($user->profile_id);
Bus::batch([
new ProfileMigrationDeliverMoveActivityPipeline($migration, $user->profile, $newAccount),
new ProfileMigrationMoveFollowersPipeline($user->profile_id, $newAccount->id),
])->onQueue('follow')->dispatch();
return redirect()->back()->with(['status' => 'Succesfully migrated account!']);
}
}

View file

@ -0,0 +1,80 @@
<?php
namespace App\Http\Requests;
use App\Models\ProfileMigration;
use App\Services\FetchCacheService;
use App\Services\WebfingerService;
use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Validation\Validator;
class ProfileMigrationStoreRequest extends FormRequest
{
/**
* Determine if the user is authorized to make this request.
*/
public function authorize(): bool
{
if ((bool) config_cache('federation.activitypub.enabled') === false ||
(bool) config_cache('federation.migration') === false) {
return false;
}
if (! $this->user() || $this->user()->status) {
return false;
}
return true;
}
/**
* Get the validation rules that apply to the request.
*
* @return array<string, \Illuminate\Contracts\Validation\ValidationRule|array<mixed>|string>
*/
public function rules(): array
{
return [
'acct' => 'required|email',
'password' => 'required|current_password',
];
}
public function after(): array
{
return [
function (Validator $validator) {
$err = $this->validateNewAccount();
if ($err !== 'noerr') {
$validator->errors()->add(
'acct',
$err
);
}
},
];
}
protected function validateNewAccount()
{
if (ProfileMigration::whereProfileId($this->user()->profile_id)->where('created_at', '>', now()->subDays(30))->exists()) {
return 'Error - You have migrated your account in the past 30 days, you can only perform a migration once per 30 days.';
}
$acct = WebfingerService::rawGet($this->acct);
if (! $acct) {
return 'The new account you provided is not responding to our requests.';
}
$pr = FetchCacheService::getJson($acct);
if (! $pr || ! isset($pr['alsoKnownAs'])) {
return 'Invalid account lookup response.';
}
if (! count($pr['alsoKnownAs']) || ! is_array($pr['alsoKnownAs'])) {
return 'The new account does not contain an alias to your current account.';
}
$curAcctUrl = $this->user()->profile->permalink();
if (! in_array($curAcctUrl, $pr['alsoKnownAs'])) {
return 'The new account does not contain an alias to your current account.';
}
return 'noerr';
}
}

View file

@ -0,0 +1,140 @@
<?php
namespace App\Jobs\ProfilePipeline;
use App\Transformer\ActivityPub\Verb\Move;
use App\Util\ActivityPub\HttpSignature;
use GuzzleHttp\Client;
use GuzzleHttp\Pool;
use Illuminate\Bus\Batchable;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldBeUniqueUntilProcessing;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\Middleware\WithoutOverlapping;
use Illuminate\Queue\SerializesModels;
use League\Fractal;
use League\Fractal\Serializer\ArraySerializer;
class ProfileMigrationDeliverMoveActivityPipeline implements ShouldBeUniqueUntilProcessing, ShouldQueue
{
use Batchable, Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
public $migration;
public $oldAccount;
public $newAccount;
public $timeout = 1400;
public $tries = 3;
public $maxExceptions = 1;
public $failOnTimeout = true;
/**
* The number of seconds after which the job's unique lock will be released.
*
* @var int
*/
public $uniqueFor = 3600;
/**
* Get the unique ID for the job.
*/
public function uniqueId(): string
{
return 'profile:migration:deliver-move-followers:id:'.$this->migration->id;
}
/**
* Get the middleware the job should pass through.
*
* @return array<int, object>
*/
public function middleware(): array
{
return [(new WithoutOverlapping('profile:migration:deliver-move-followers:id:'.$this->migration->id))->shared()->dontRelease()];
}
/**
* Create a new job instance.
*/
public function __construct($migration, $oldAccount, $newAccount)
{
$this->migration = $migration;
$this->oldAccount = $oldAccount;
$this->newAccount = $newAccount;
}
/**
* Execute the job.
*/
public function handle(): void
{
if ($this->batch()->cancelled()) {
return;
}
$migration = $this->migration;
$profile = $this->oldAccount;
$newAccount = $this->newAccount;
if ($profile->domain || ! $profile->private_key) {
return;
}
$audience = $profile->getAudienceInbox();
$activitypubObject = new Move();
$fractal = new Fractal\Manager();
$fractal->setSerializer(new ArraySerializer());
$resource = new Fractal\Resource\Item($migration, $activitypubObject);
$activity = $fractal->createData($resource)->toArray();
$payload = json_encode($activity);
$client = new Client([
'timeout' => config('federation.activitypub.delivery.timeout'),
]);
$version = config('pixelfed.version');
$appUrl = config('app.url');
$userAgent = "(Pixelfed/{$version}; +{$appUrl})";
$requests = function ($audience) use ($client, $activity, $profile, $payload, $userAgent) {
foreach ($audience as $url) {
$headers = HttpSignature::sign($profile, $url, $activity, [
'Content-Type' => 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
'User-Agent' => $userAgent,
]);
yield function () use ($client, $url, $headers, $payload) {
return $client->postAsync($url, [
'curl' => [
CURLOPT_HTTPHEADER => $headers,
CURLOPT_POSTFIELDS => $payload,
CURLOPT_HEADER => true,
CURLOPT_SSL_VERIFYPEER => false,
CURLOPT_SSL_VERIFYHOST => false,
],
]);
};
}
};
$pool = new Pool($client, $requests($audience), [
'concurrency' => config('federation.activitypub.delivery.concurrency'),
'fulfilled' => function ($response, $index) {
},
'rejected' => function ($reason, $index) {
},
]);
$promise = $pool->promise();
$promise->wait();
}
}

View file

@ -0,0 +1,95 @@
<?php
namespace App\Jobs\ProfilePipeline;
use App\Follower;
use App\Profile;
use App\Services\AccountService;
use Illuminate\Bus\Batchable;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldBeUniqueUntilProcessing;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\Middleware\WithoutOverlapping;
use Illuminate\Queue\SerializesModels;
class ProfileMigrationMoveFollowersPipeline implements ShouldBeUniqueUntilProcessing, ShouldQueue
{
use Batchable, Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
public $oldPid;
public $newPid;
public $timeout = 1400;
public $tries = 3;
public $maxExceptions = 1;
public $failOnTimeout = true;
/**
* The number of seconds after which the job's unique lock will be released.
*
* @var int
*/
public $uniqueFor = 3600;
/**
* Get the unique ID for the job.
*/
public function uniqueId(): string
{
return 'profile:migration:move-followers:oldpid-'.$this->oldPid.':newpid-'.$this->newPid;
}
/**
* Get the middleware the job should pass through.
*
* @return array<int, object>
*/
public function middleware(): array
{
return [(new WithoutOverlapping('profile:migration:move-followers:oldpid-'.$this->oldPid.':newpid-'.$this->newPid))->shared()->dontRelease()];
}
/**
* Create a new job instance.
*/
public function __construct($oldPid, $newPid)
{
$this->oldPid = $oldPid;
$this->newPid = $newPid;
}
/**
* Execute the job.
*/
public function handle(): void
{
if ($this->batch()->cancelled()) {
return;
}
$og = Profile::find($this->oldPid);
$ne = Profile::find($this->newPid);
if (! $og || ! $ne || $og == $ne) {
return;
}
$ne->followers_count = $og->followers_count;
$ne->save();
$og->followers_count = 0;
$og->save();
foreach (Follower::whereFollowingId($this->oldPid)->lazyById(200, 'id') as $follower) {
try {
$follower->following_id = $this->newPid;
$follower->save();
} catch (Exception $e) {
$follower->delete();
}
}
AccountService::del($this->oldPid);
AccountService::del($this->newPid);
}
}

View file

@ -10,6 +10,8 @@ class ProfileAlias extends Model
{ {
use HasFactory; use HasFactory;
protected $guarded = [];
public function profile() public function profile()
{ {
return $this->belongsTo(Profile::class); return $this->belongsTo(Profile::class);

View file

@ -0,0 +1,19 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use App\Profile;
class ProfileMigration extends Model
{
use HasFactory;
protected $guarded = [];
public function profile()
{
return $this->belongsTo(Profile::class, 'profile_id');
}
}

View file

@ -2,127 +2,129 @@
namespace App\Services; namespace App\Services;
use Cache;
use Config;
use App\Models\ConfigCache as ConfigCacheModel; use App\Models\ConfigCache as ConfigCacheModel;
use Cache;
class ConfigCacheService class ConfigCacheService
{ {
const CACHE_KEY = 'config_cache:_v0-key:'; const CACHE_KEY = 'config_cache:_v0-key:';
public static function get($key) public static function get($key)
{ {
$cacheKey = self::CACHE_KEY . $key; $cacheKey = self::CACHE_KEY.$key;
$ttl = now()->addHours(12); $ttl = now()->addHours(12);
if(!config('instance.enable_cc')) { if (! config('instance.enable_cc')) {
return config($key); return config($key);
} }
return Cache::remember($cacheKey, $ttl, function() use($key) { return Cache::remember($cacheKey, $ttl, function () use ($key) {
$allowed = [ $allowed = [
'app.name', 'app.name',
'app.short_description', 'app.short_description',
'app.description', 'app.description',
'app.rules', 'app.rules',
'pixelfed.max_photo_size', 'pixelfed.max_photo_size',
'pixelfed.max_album_length', 'pixelfed.max_album_length',
'pixelfed.image_quality', 'pixelfed.image_quality',
'pixelfed.media_types', 'pixelfed.media_types',
'pixelfed.open_registration', 'pixelfed.open_registration',
'federation.activitypub.enabled', 'federation.activitypub.enabled',
'instance.stories.enabled', 'instance.stories.enabled',
'pixelfed.oauth_enabled', 'pixelfed.oauth_enabled',
'pixelfed.import.instagram.enabled', 'pixelfed.import.instagram.enabled',
'pixelfed.bouncer.enabled', 'pixelfed.bouncer.enabled',
'pixelfed.enforce_email_verification', 'pixelfed.enforce_email_verification',
'pixelfed.max_account_size', 'pixelfed.max_account_size',
'pixelfed.enforce_account_limit', 'pixelfed.enforce_account_limit',
'uikit.custom.css', 'uikit.custom.css',
'uikit.custom.js', 'uikit.custom.js',
'uikit.show_custom.css', 'uikit.show_custom.css',
'uikit.show_custom.js', 'uikit.show_custom.js',
'about.title', 'about.title',
'pixelfed.cloud_storage', 'pixelfed.cloud_storage',
'account.autofollow', 'account.autofollow',
'account.autofollow_usernames', 'account.autofollow_usernames',
'config.discover.features', 'config.discover.features',
'instance.has_legal_notice', 'instance.has_legal_notice',
'instance.avatar.local_to_cloud', 'instance.avatar.local_to_cloud',
'pixelfed.directory', 'pixelfed.directory',
'app.banner_image', 'app.banner_image',
'pixelfed.directory.submission-key', 'pixelfed.directory.submission-key',
'pixelfed.directory.submission-ts', 'pixelfed.directory.submission-ts',
'pixelfed.directory.has_submitted', 'pixelfed.directory.has_submitted',
'pixelfed.directory.latest_response', 'pixelfed.directory.latest_response',
'pixelfed.directory.is_synced', 'pixelfed.directory.is_synced',
'pixelfed.directory.testimonials', 'pixelfed.directory.testimonials',
'instance.landing.show_directory', 'instance.landing.show_directory',
'instance.landing.show_explore', 'instance.landing.show_explore',
'instance.admin.pid', 'instance.admin.pid',
'instance.banner.blurhash', 'instance.banner.blurhash',
'autospam.nlp.enabled', 'autospam.nlp.enabled',
'instance.curated_registration.enabled', 'instance.curated_registration.enabled',
// 'system.user_mode'
];
if(!config('instance.enable_cc')) { 'federation.migration',
return config($key); // 'system.user_mode'
} ];
if(!in_array($key, $allowed)) { if (! config('instance.enable_cc')) {
return config($key); return config($key);
} }
$v = config($key); if (! in_array($key, $allowed)) {
$c = ConfigCacheModel::where('k', $key)->first(); return config($key);
}
if($c) { $v = config($key);
return $c->v ?? config($key); $c = ConfigCacheModel::where('k', $key)->first();
}
if(!$v) { if ($c) {
return; return $c->v ?? config($key);
} }
$cc = new ConfigCacheModel; if (! $v) {
$cc->k = $key; return;
$cc->v = $v; }
$cc->save();
return $v; $cc = new ConfigCacheModel;
}); $cc->k = $key;
} $cc->v = $v;
$cc->save();
public static function put($key, $val) return $v;
{ });
$exists = ConfigCacheModel::whereK($key)->first(); }
if($exists) { public static function put($key, $val)
$exists->v = $val; {
$exists->save(); $exists = ConfigCacheModel::whereK($key)->first();
Cache::put(self::CACHE_KEY . $key, $val, now()->addHours(12));
return self::get($key);
}
$cc = new ConfigCacheModel; if ($exists) {
$cc->k = $key; $exists->v = $val;
$cc->v = $val; $exists->save();
$cc->save(); Cache::put(self::CACHE_KEY.$key, $val, now()->addHours(12));
Cache::put(self::CACHE_KEY . $key, $val, now()->addHours(12)); return self::get($key);
}
return self::get($key); $cc = new ConfigCacheModel;
} $cc->k = $key;
$cc->v = $val;
$cc->save();
Cache::put(self::CACHE_KEY.$key, $val, now()->addHours(12));
return self::get($key);
}
} }

View file

@ -0,0 +1,79 @@
<?php
namespace App\Services;
use App\Util\ActivityPub\Helpers;
use Illuminate\Http\Client\ConnectionException;
use Illuminate\Http\Client\RequestException;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\Facades\Http;
class FetchCacheService
{
const CACHE_KEY = 'pf:fetch_cache_service:getjson:';
public static function getJson($url, $verifyCheck = true, $ttl = 3600, $allowRedirects = true)
{
$vc = $verifyCheck ? 'vc1:' : 'vc0:';
$ar = $allowRedirects ? 'ar1:' : 'ar0';
$key = self::CACHE_KEY.sha1($url).':'.$vc.$ar.$ttl;
if (Cache::has($key)) {
return false;
}
if ($verifyCheck) {
if (! Helpers::validateUrl($url)) {
Cache::put($key, 1, $ttl);
return false;
}
}
$headers = [
'User-Agent' => '(Pixelfed/'.config('pixelfed.version').'; +'.config('app.url').')',
];
if ($allowRedirects) {
$options = [
'allow_redirects' => [
'max' => 2,
'strict' => true,
],
];
} else {
$options = [
'allow_redirects' => false,
];
}
try {
$res = Http::withOptions($options)
->retry(3, function (int $attempt, $exception) {
return $attempt * 500;
})
->acceptJson()
->withHeaders($headers)
->timeout(40)
->get($url);
} catch (RequestException $e) {
Cache::put($key, 1, $ttl);
return false;
} catch (ConnectionException $e) {
Cache::put($key, 1, $ttl);
return false;
} catch (Exception $e) {
Cache::put($key, 1, $ttl);
return false;
}
if (! $res->ok()) {
Cache::put($key, 1, $ttl);
return false;
}
return $res->json();
}
}

View file

@ -2,69 +2,95 @@
namespace App\Services; namespace App\Services;
use Cache;
use App\Profile; use App\Profile;
use App\Util\ActivityPub\Helpers;
use App\Util\Webfinger\WebfingerUrl; use App\Util\Webfinger\WebfingerUrl;
use Illuminate\Support\Facades\Http; use Illuminate\Support\Facades\Http;
use App\Util\ActivityPub\Helpers;
use App\Services\AccountService;
class WebfingerService class WebfingerService
{ {
public static function lookup($query, $mastodonMode = false) public static function rawGet($url)
{ {
return (new self)->run($query, $mastodonMode); $n = WebfingerUrl::get($url);
} if (! $n) {
return false;
}
$webfinger = FetchCacheService::getJson($n);
if (! $webfinger) {
return false;
}
protected function run($query, $mastodonMode) if (! isset($webfinger['links']) || ! is_array($webfinger['links']) || empty($webfinger['links'])) {
{ return false;
if($profile = Profile::whereUsername($query)->first()) { }
return $mastodonMode ? $link = collect($webfinger['links'])
AccountService::getMastodon($profile->id, true) : ->filter(function ($link) {
AccountService::get($profile->id); return $link &&
} isset($link['rel'], $link['type'], $link['href']) &&
$url = WebfingerUrl::generateWebfingerUrl($query); $link['rel'] === 'self' &&
if(!Helpers::validateUrl($url)) { in_array($link['type'], ['application/activity+json', 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"']);
return []; })
} ->pluck('href')
->first();
try { return $link;
$res = Http::retry(3, 100) }
->acceptJson()
->withHeaders([
'User-Agent' => '(Pixelfed/' . config('pixelfed.version') . '; +' . config('app.url') . ')'
])
->timeout(20)
->get($url);
} catch (\Illuminate\Http\Client\ConnectionException $e) {
return [];
}
if(!$res->successful()) { public static function lookup($query, $mastodonMode = false)
return []; {
} return (new self)->run($query, $mastodonMode);
}
$webfinger = $res->json(); protected function run($query, $mastodonMode)
if(!isset($webfinger['links']) || !is_array($webfinger['links']) || empty($webfinger['links'])) { {
return []; if ($profile = Profile::whereUsername($query)->first()) {
} return $mastodonMode ?
AccountService::getMastodon($profile->id, true) :
AccountService::get($profile->id);
}
$url = WebfingerUrl::generateWebfingerUrl($query);
if (! Helpers::validateUrl($url)) {
return [];
}
$link = collect($webfinger['links']) try {
->filter(function($link) { $res = Http::retry(3, 100)
return $link && ->acceptJson()
isset($link['rel'], $link['type'], $link['href']) && ->withHeaders([
$link['rel'] === 'self' && 'User-Agent' => '(Pixelfed/'.config('pixelfed.version').'; +'.config('app.url').')',
in_array($link['type'], ['application/activity+json','application/ld+json; profile="https://www.w3.org/ns/activitystreams"']); ])
}) ->timeout(20)
->pluck('href') ->get($url);
->first(); } catch (\Illuminate\Http\Client\ConnectionException $e) {
return [];
}
$profile = Helpers::profileFetch($link); if (! $res->successful()) {
if(!$profile) { return [];
return; }
}
return $mastodonMode ? $webfinger = $res->json();
AccountService::getMastodon($profile->id, true) : if (! isset($webfinger['links']) || ! is_array($webfinger['links']) || empty($webfinger['links'])) {
AccountService::get($profile->id); return [];
} }
$link = collect($webfinger['links'])
->filter(function ($link) {
return $link &&
isset($link['rel'], $link['type'], $link['href']) &&
$link['rel'] === 'self' &&
in_array($link['type'], ['application/activity+json', 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"']);
})
->pluck('href')
->first();
$profile = Helpers::profileFetch($link);
if (! $profile) {
return;
}
return $mastodonMode ?
AccountService::getMastodon($profile->id, true) :
AccountService::get($profile->id);
}
} }

View file

@ -0,0 +1,26 @@
<?php
namespace App\Transformer\ActivityPub\Verb;
use App\Models\ProfileMigration;
use League\Fractal;
class Move extends Fractal\TransformerAbstract
{
public function transform(ProfileMigration $migration)
{
$objUrl = $migration->target->permalink();
$id = $migration->target->permalink('#moves/'.$migration->id);
$to = $migration->target->permalink('/followers');
return [
'@context' => 'https://www.w3.org/ns/activitystreams',
'id' => $id,
'actor' => $objUrl,
'type' => 'Move',
'object' => $objUrl,
'target' => $migration->profile->permalink(),
'to' => $to,
];
}
}

View file

@ -2,80 +2,91 @@
namespace App\Transformer\Api; namespace App\Transformer\Api;
use Auth;
use Cache;
use App\Profile; use App\Profile;
use App\Services\AccountService;
use App\Services\PronounService;
use App\User; use App\User;
use App\UserSetting; use App\UserSetting;
use Cache;
use League\Fractal; use League\Fractal;
use App\Services\PronounService;
class AccountTransformer extends Fractal\TransformerAbstract class AccountTransformer extends Fractal\TransformerAbstract
{ {
protected $defaultIncludes = [ protected $defaultIncludes = [
// 'relationship', // 'relationship',
]; ];
public function transform(Profile $profile) public function transform(Profile $profile)
{ {
if(!$profile) { if (! $profile) {
return []; return [];
} }
$adminIds = Cache::remember('pf:admin-ids', 604800, function() { $adminIds = Cache::remember('pf:admin-ids', 604800, function () {
return User::whereIsAdmin(true)->pluck('profile_id')->toArray(); return User::whereIsAdmin(true)->pluck('profile_id')->toArray();
}); });
$local = $profile->private_key != null; $local = $profile->private_key != null;
$local = $profile->user_id && $profile->private_key != null; $local = $profile->user_id && $profile->private_key != null;
$hideFollowing = false; $hideFollowing = false;
$hideFollowers = false; $hideFollowers = false;
if($local) { if ($local) {
$hideFollowing = Cache::remember('pf:acct-trans:hideFollowing:' . $profile->id, 2592000, function() use($profile) { $hideFollowing = Cache::remember('pf:acct-trans:hideFollowing:'.$profile->id, 2592000, function () use ($profile) {
$settings = UserSetting::whereUserId($profile->user_id)->first(); $settings = UserSetting::whereUserId($profile->user_id)->first();
if(!$settings) { if (! $settings) {
return false; return false;
} }
return $settings->show_profile_following_count == false;
});
$hideFollowers = Cache::remember('pf:acct-trans:hideFollowers:' . $profile->id, 2592000, function() use($profile) {
$settings = UserSetting::whereUserId($profile->user_id)->first();
if(!$settings) {
return false;
}
return $settings->show_profile_follower_count == false;
});
}
$is_admin = !$local ? false : in_array($profile->id, $adminIds);
$acct = $local ? $profile->username : substr($profile->username, 1);
$username = $local ? $profile->username : explode('@', $acct)[0];
return [
'id' => (string) $profile->id,
'username' => $username,
'acct' => $acct,
'display_name' => $profile->name,
'discoverable' => true,
'locked' => (bool) $profile->is_private,
'followers_count' => $hideFollowers ? 0 : (int) $profile->followers_count,
'following_count' => $hideFollowing ? 0 : (int) $profile->following_count,
'statuses_count' => (int) $profile->status_count,
'note' => $profile->bio ?? '',
'note_text' => $profile->bio ? strip_tags($profile->bio) : null,
'url' => $profile->url(),
'avatar' => $profile->avatarUrl(),
'website' => $profile->website,
'local' => (bool) $local,
'is_admin' => (bool) $is_admin,
'created_at' => $profile->created_at->toJSON(),
'header_bg' => $profile->header_bg,
'last_fetched_at' => optional($profile->last_fetched_at)->toJSON(),
'pronouns' => PronounService::get($profile->id),
'location' => $profile->location
];
}
protected function includeRelationship(Profile $profile) return $settings->show_profile_following_count == false;
{ });
return $this->item($profile, new RelationshipTransformer()); $hideFollowers = Cache::remember('pf:acct-trans:hideFollowers:'.$profile->id, 2592000, function () use ($profile) {
} $settings = UserSetting::whereUserId($profile->user_id)->first();
if (! $settings) {
return false;
}
return $settings->show_profile_follower_count == false;
});
}
$is_admin = ! $local ? false : in_array($profile->id, $adminIds);
$acct = $local ? $profile->username : substr($profile->username, 1);
$username = $local ? $profile->username : explode('@', $acct)[0];
$res = [
'id' => (string) $profile->id,
'username' => $username,
'acct' => $acct,
'display_name' => $profile->name,
'discoverable' => true,
'locked' => (bool) $profile->is_private,
'followers_count' => $hideFollowers ? 0 : (int) $profile->followers_count,
'following_count' => $hideFollowing ? 0 : (int) $profile->following_count,
'statuses_count' => (int) $profile->status_count,
'note' => $profile->bio ?? '',
'note_text' => $profile->bio ? strip_tags($profile->bio) : null,
'url' => $profile->url(),
'avatar' => $profile->avatarUrl(),
'website' => $profile->website,
'local' => (bool) $local,
'is_admin' => (bool) $is_admin,
'created_at' => $profile->created_at->toJSON(),
'header_bg' => $profile->header_bg,
'last_fetched_at' => optional($profile->last_fetched_at)->toJSON(),
'pronouns' => PronounService::get($profile->id),
'location' => $profile->location,
];
if ($profile->moved_to_profile_id) {
$mt = AccountService::getMastodon($profile->moved_to_profile_id, true);
if ($mt) {
$res['moved'] = $mt;
}
}
return $res;
}
protected function includeRelationship(Profile $profile)
{
return $this->item($profile, new RelationshipTransformer());
}
} }

View file

@ -3,16 +3,28 @@
namespace App\Util\Webfinger; namespace App\Util\Webfinger;
use App\Util\Lexer\Nickname; use App\Util\Lexer\Nickname;
use App\Services\InstanceService;
class WebfingerUrl class WebfingerUrl
{ {
public static function get($url)
{
$n = Nickname::normalizeProfileUrl($url);
if(!$n || !isset($n['domain'], $n['username'])) {
return false;
}
if(in_array($n['domain'], InstanceService::getBannedDomains())) {
return false;
}
return 'https://' . $n['domain'] . '/.well-known/webfinger?resource=acct:' . $n['username'] . '@' . $n['domain'];
}
public static function generateWebfingerUrl($url) public static function generateWebfingerUrl($url)
{ {
$url = Nickname::normalizeProfileUrl($url); $url = Nickname::normalizeProfileUrl($url);
$domain = $url['domain']; $domain = $url['domain'];
$username = $url['username']; $username = $url['username'];
$path = "https://{$domain}/.well-known/webfinger?resource=acct:{$username}@{$domain}"; $path = "https://{$domain}/.well-known/webfinger?resource=acct:{$username}@{$domain}";
return $path; return $path;
} }
} }

View file

@ -57,4 +57,6 @@ return [
// max size in bytes, default is 2mb // max size in bytes, default is 2mb
'max_size' => env('CUSTOM_EMOJI_MAX_SIZE', 2000000), 'max_size' => env('CUSTOM_EMOJI_MAX_SIZE', 2000000),
], ],
'migration' => env('PF_ACCT_MIGRATION_ENABLED', true),
]; ];

View file

@ -0,0 +1,31 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('profile_migrations', function (Blueprint $table) {
$table->id();
$table->unsignedBigInteger('profile_id');
$table->string('acct')->nullable();
$table->unsignedBigInteger('followers_count')->default(0);
$table->unsignedBigInteger('target_profile_id')->nullable();
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('profile_migrations');
}
};

BIN
public/js/manifest.js vendored

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View file

@ -1,7 +1,40 @@
<template> <template>
<div class="profile-timeline-component"> <div class="profile-timeline-component">
<div v-if="isLoaded" class="container-fluid mt-3"> <div v-if="isLoaded" class="container-fluid mt-3">
<div class="row"> <div v-if="profile && profile.hasOwnProperty('moved') && profile.moved.hasOwnProperty('id') && !showMoved">
<div class="row justify-content-center">
<div class="col-12 col-md-6">
<div class="card shadow-none border card-body mt-5 mb-3 ft-std" style="border-radius: 20px;">
<p class="lead font-weight-bold text-center mb-0"><i class="far fa-exclamation-triangle mr-2"></i>This account has indicated their new account is:</p>
</div>
<div class="card shadow-none border" style="border-radius: 20px;">
<div class="card-body ft-std">
<div class="d-flex justify-content-between align-items-center" style="gap: 1rem;">
<div class="d-flex align-items-center flex-shrink-1" style="gap: 1rem;">
<img :src="profile.moved.avatar" width="50" height="50" class="rounded-circle">
<p class="h3 font-weight-light mb-0 text-break">&commat;{{ profile.moved.acct }}</p>
</div>
<div class="d-flex flex-grow-1 justify-content-end" style="min-width: 200px;">
<router-link
:to="`/i/web/profile/${profile.moved.id}`"
class="btn btn-outline-primary rounded-pill font-weight-bold"
>View New Account</router-link>
</div>
</div>
</div>
</div>
<hr>
<p class="lead text-center ft-std">
<a
href="#"
class="btn btn-primary btn-lg rounded-pill font-weight-bold px-5"
@click.prevent="showMoved = true">Proceed</a>
</p>
</div>
</div>
</div>
<div v-else class="row">
<div class="col-md-3 d-md-block px-md-3 px-xl-5"> <div class="col-md-3 d-md-block px-md-3 px-xl-5">
<profile-sidebar <profile-sidebar
:profile="profile" :profile="profile"
@ -72,7 +105,8 @@
curUser: undefined, curUser: undefined,
tab: "index", tab: "index",
profile: undefined, profile: undefined,
relationship: undefined relationship: undefined,
showMoved: false,
} }
}, },

View file

@ -150,6 +150,19 @@
</a> </a>
</div> </div>
<div v-else-if="profile.hasOwnProperty('moved') && profile.moved.id" style="flex-grow: 1;">
<div class="card shadow-none rounded-lg mb-3 bg-danger">
<div class="card-body">
<div class="d-flex align-items-center ft-std text-white mb-2">
<i class="far fa-exclamation-triangle mr-2 text-white"></i>
Account has moved to:
</div>
<p class="mb-0 lead ft-std text-white text-break">
<router-link :to="`/i/web/profile/${profile.moved.id}`" class="btn btn-outline-light btn-block rounded-pill font-weight-bold">&commat;{{truncate(profile.moved.acct)}}</router-link>
</p>
</div>
</div>
</div>
<div v-else-if="profile.locked" style="flex-grow: 1;"> <div v-else-if="profile.locked" style="flex-grow: 1;">
<template v-if="!relationship.following && !relationship.requested"> <template v-if="!relationship.following && !relationship.requested">
<button <button
@ -375,6 +388,16 @@
} }
}, },
truncate(str) {
if(!str) {
return;
}
if(str.length > 15) {
return str.slice(0, 15) + '...';
}
return str;
},
formatCount(val) { formatCount(val) {
return App.util.format.count(val); return App.util.format.count(val);
}, },

View file

@ -103,6 +103,16 @@
</div> </div>
<p class="mb-4 small">ActivityPub federation, compatible with Pixelfed, Mastodon and other projects.</p> <p class="mb-4 small">ActivityPub federation, compatible with Pixelfed, Mastodon and other projects.</p>
<div class="custom-control custom-checkbox mt-2">
<input type="checkbox" name="account_migration" class="custom-control-input" id="ap_mig" {{(bool)config_cache('federation.migration') ? 'checked' : ''}} {{(bool) config_cache('federation.activitypub.enabled') ? '' : 'disabled="disabled"'}}>
<label class="custom-control-label font-weight-bold" for="ap_mig">Account Migration</label>
</div>
@if((bool) config_cache('federation.activitypub.enabled'))
<p class="mb-4 small">Allow local accounts to migrate to other local or remote accounts.</p>
@else
<p class="mb-4 small text-muted"><strong>ActivityPub Required</strong> Allow local accounts to migrate to other local or remote accounts.</p>
@endif
{{-- <div class="custom-control custom-checkbox mt-2"> {{-- <div class="custom-control custom-checkbox mt-2">
<input type="checkbox" name="open_registration" class="custom-control-input" id="openReg" {{config_cache('pixelfed.open_registration') ? 'checked' : ''}}> <input type="checkbox" name="open_registration" class="custom-control-input" id="openReg" {{config_cache('pixelfed.open_registration') ? 'checked' : ''}}>
<label class="custom-control-label font-weight-bold" for="openReg">Open Registrations</label> <label class="custom-control-label font-weight-bold" for="openReg">Open Registrations</label>

View file

@ -88,6 +88,7 @@
</div> </div>
</div> </div>
@if((bool) config_cache('federation.activitypub.enabled'))
<div class="form-group row"> <div class="form-group row">
<label for="aliases" class="col-sm-3 col-form-label font-weight-bold">Account Aliases</label> <label for="aliases" class="col-sm-3 col-form-label font-weight-bold">Account Aliases</label>
<div class="col-sm-9" id="aliases"> <div class="col-sm-9" id="aliases">
@ -95,6 +96,17 @@
<p class="help-text text-muted small">To move from another account to this one, first you need to create an alias.</p> <p class="help-text text-muted small">To move from another account to this one, first you need to create an alias.</p>
</div> </div>
</div> </div>
@if((bool) config_cache('federation.migration'))
<div class="form-group row">
<label for="aliases" class="col-sm-3 col-form-label font-weight-bold">Account Migrate</label>
<div class="col-sm-9" id="aliases">
<a class="font-weight-bold" href="/settings/account/migration/manage">Migrate to another account</a>
<p class="help-text text-muted small">To redirect this account to a different one (where supported).</p>
</div>
</div>
@endif
@endif
@if(config_cache('pixelfed.enforce_account_limit')) @if(config_cache('pixelfed.enforce_account_limit'))
<div class="pt-3"> <div class="pt-3">
<p class="font-weight-bold text-muted text-center">Storage Usage</p> <p class="font-weight-bold text-muted text-center">Storage Usage</p>

View file

@ -0,0 +1,108 @@
@extends('layouts.app')
@section('content')
@if (session('status'))
<div class="alert alert-primary px-3 h6 font-weight-bold text-center">
{{ session('status') }}
</div>
@endif
@if ($errors->any())
<div class="alert alert-danger px-3 h6 text-center">
@foreach($errors->all() as $error)
<p class="font-weight-bold mb-1">{{ $error }}</p>
@endforeach
</div>
@endif
@if (session('error'))
<div class="alert alert-danger px-3 h6 text-center">
{{ session('error') }}
</div>
@endif
<div class="container">
<div class="col-12">
<div class="card shadow-none border mt-5">
<div class="card-body">
<div class="row">
<div class="col-12 p-3 p-md-5">
<div class="title">
<div class="d-flex justify-content-between align-items-center">
<h3 class="font-weight-bold">Account Migration</h3>
<a class="font-weight-bold" href="/settings/home">
<i class="far fa-long-arrow-left"></i>
Back to Settings
</a>
</div>
<hr />
</div>
<div>
@if($hasExistingMigration)
<div class="row">
<div class="col-12 mt-5">
<p class="lead text-center">You have migrated your account already.</p>
<p>You can only migrate your account once per 30 days. If you want to migrate your followers back to this account, follow this process in reverse.</p>
</div>
</div>
@else
<div class="row">
<div class="col-12">
<p class="lead">If you want to move this account to another account, please read the following carefully.</p>
<ul class="text-danger lead">
<li class="font-weight-bold">Only followers will be transferred; no other information will be moved automatically.</li>
<li>This process will transfer all followers from your existing account to your new account.</li>
<li>A redirect notice will be added to your current account's profile, and it will be removed from search results.</li>
<li>You must set up the new account to link back to your current account before proceeding.</li>
<li>Once the transfer is initiated, there will be a waiting period during which you cannot initiate another transfer.</li>
<li>After the transfer, your current account will be limited in functionality, but you will retain the ability to export data and possibly reactivate the account.</li>
</ul>
<p class="mb-0">For more information on Aliases and Account Migration, visit the <a href="/site/kb/your-profile">Help Center</a>.</p>
<hr>
<form method="post" autocomplete="off">
@csrf
<div class="row">
<div class="col-12 col-md-6">
<div class="form-group">
<label class="font-weight-bold mb-0">New Account Handle</label>
<p class="small text-muted">Enter the username@domain of the account you want to move to</p>
<input
type="email"
class="form-control"
name="acct"
placeholder="username@domain.tld"
role="presentation"
autocomplete="new-user-email"
/>
</div>
</div>
<div class="col-12 col-md-6">
<div class="form-group">
<label class="font-weight-bold mb-0">Account Password</label>
<p class="small text-muted">For security purposes please enter the password of the current account</p>
<input
type="password"
class="form-control"
name="password"
role="presentation"
placeholder="Your account password"
autocomplete="new-password"
/>
</div>
</div>
</div>
<button class="btn btn-primary btn-block font-weight-bold btn-lg rounded-pill">Move Followers</button>
</form>
</div>
</div>
@endif
</div>
</div>
</div>
</div>
</div>
</div>
</div>
@endsection

View file

@ -0,0 +1,19 @@
@extends('site.help.partial.template', ['breadcrumb'=>'Account Migration'])
@section('section')
<div class="title">
<h3 class="font-weight-bold">Account Migration</h3>
</div>
<hr>
@if((bool) config_cache('federation.migration') === false)
<div class="alert alert-danger">
<p class="font-weight-bold mb-0">Account Migration is not available on this server.</p>
</div>
@endif
<p class="lead">Account Migration is a feature that allows users to move their account followers from one Pixelfed instance (server) to another.</p>
<p class="lead">This can be useful if a user wants to switch to a different instance due to preferences for its community, policies, or features.</p>
@endsection

View file

@ -266,6 +266,11 @@ Route::domain(config('pixelfed.domain.app'))->middleware(['validemail', 'twofact
Route::post('manage', 'ProfileAliasController@store'); Route::post('manage', 'ProfileAliasController@store');
Route::post('manage/delete', 'ProfileAliasController@delete'); Route::post('manage/delete', 'ProfileAliasController@delete');
}); });
Route::group(['prefix' => 'account/migration', 'middleware' => 'dangerzone'], function() {
Route::get('manage', 'ProfileMigrationController@index');
Route::post('manage', 'ProfileMigrationController@store');
});
}); });
Route::group(['prefix' => 'site'], function () { Route::group(['prefix' => 'site'], function () {
@ -309,6 +314,7 @@ Route::domain(config('pixelfed.domain.app'))->middleware(['validemail', 'twofact
Route::view('parental-controls', 'site.help.parental-controls'); Route::view('parental-controls', 'site.help.parental-controls');
Route::view('email-confirmation-issues', 'site.help.email-confirmation-issues')->name('help.email-confirmation-issues'); Route::view('email-confirmation-issues', 'site.help.email-confirmation-issues')->name('help.email-confirmation-issues');
Route::view('curated-onboarding', 'site.help.curated-onboarding')->name('help.curated-onboarding'); Route::view('curated-onboarding', 'site.help.curated-onboarding')->name('help.curated-onboarding');
Route::view('account-migration', 'site.help.account-migration')->name('help.account-migration');
}); });
Route::get('newsroom/{year}/{month}/{slug}', 'NewsroomController@show'); Route::get('newsroom/{year}/{month}/{slug}', 'NewsroomController@show');
Route::get('newsroom/archive', 'NewsroomController@archive'); Route::get('newsroom/archive', 'NewsroomController@archive');