Skip to content

Commit

Permalink
Delete users that have not given their consent to be transferred to M…
Browse files Browse the repository at this point in the history
…atrix
  • Loading branch information
alainvd committed Dec 13, 2024
1 parent bf6373a commit eabd49e
Show file tree
Hide file tree
Showing 5 changed files with 271 additions and 0 deletions.
57 changes: 57 additions & 0 deletions app/Console/Commands/SoftDeleteUsersWithoutConsent.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
<?php

namespace App\Console\Commands;

use App\Jobs\ProcessUserDeletion;
use App\User;
use Illuminate\Console\Command;
use Illuminate\Support\Facades\Hash;

class SoftDeleteUsersWithoutConsent extends Command
{
protected $signature = 'users:delete-without-consent';
protected $description = 'Delete users who have not given consent and transfer their data to legacy user';

public function handle()
{
$this->info('Creating or verifying legacy user...');

// Create legacy user if it doesn't exist
User::firstOrCreate(
['id' => 1000000],
[
'firstname' => 'Codeweek Legacy User',
'lastname' => 'Codeweek Legacy',
'username' => 'codeweek-legacy',
'password' => Hash::make('some-secure-password-' . time()),
'email' => '[email protected]',
'country_iso' => 'BE',
'remember_token' => '5Pb4APP0CkOmQFzDtR960OlkdreLev2tvfUiywDawyJOmKDvFb3bmOAWlUL6',
'privacy' => 1,
'email_display' => '[email protected]',
'receive_emails' => 0,
'approved' => 1,
'magic_key' => 2520090911,
'consent_given_at' => now(),
'future_consent_given_at' => now(),
'email_verified_at' => now(),
'created_at' => now(),
'updated_at' => now()
]
);

$this->info('Starting to process users without consent...');

// Get users without consent in chunks to avoid memory issues
User::whereNull('consent_given_at')
->where('id', '!=', 1000000) // Exclude the legacy user
->chunkById(100, function ($users) {
foreach ($users as $user) {
ProcessUserDeletion::dispatch($user->id);
$this->info("Dispatched deletion job for user ID: {$user->id}");
}
});

$this->info('All deletion jobs have been dispatched. Check the queue worker logs for progress.');
}
}
81 changes: 81 additions & 0 deletions app/Jobs/ProcessUserDeletion.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
<?php

namespace App\Jobs;

use App\Excellence;
use App\Participation;
use App\User;
use App\Event;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Log;

class ProcessUserDeletion implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;

protected $userId;
protected $legacyUserId = 1000000;

public function __construct($userId)
{
$this->userId = $userId;
}

/**
* @throws \Exception
*/
public function handle()
{
try {
DB::beginTransaction();

Log::info("Processing deletion for user ID: {$this->userId}");

// Update events creator_id
Event::where('creator_id', $this->userId)
->update(['creator_id' => $this->legacyUserId]);

// Update events approved_by
Event::where('approved_by', $this->userId)
->update(['approved_by' => $this->legacyUserId]);

// Update participations
Participation::where('user_id', $this->userId)
->update(['user_id' => $this->legacyUserId]);

// Update excellences
Excellence::where('user_id', $this->userId)
->update(['user_id' => $this->legacyUserId]);

// Update model_has_roles table directly
DB::table('model_has_roles')
->where('model_id', $this->userId)
->where('model_type', User::class)
->update(['model_id' => $this->legacyUserId]);

// Update leading_teacher_expertise_user table directly
DB::table('leading_teacher_expertise_user')
->where('user_id', $this->userId)
->update(['user_id' => $this->legacyUserId]);

// Hard delete the user
User::where('id', $this->userId)->delete();

DB::commit();
Log::info("Successfully deleted user ID: {$this->userId}");

} catch (\Exception $e) {
DB::rollBack();
Log::error("Error processing user ID: {$this->userId}", [
'error' => $e->getMessage(),
'trace' => $e->getTraceAsString()
]);
throw $e;
}
}
}
33 changes: 33 additions & 0 deletions database/migrations/2024_01_15_000000_create_failed_jobs_table.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
<?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
{
if (!Schema::hasTable('failed_jobs')) {
Schema::create('failed_jobs', function (Blueprint $table) {
$table->id();
$table->string('uuid')->unique();
$table->text('connection');
$table->text('queue');
$table->longText('payload');
$table->longText('exception');
$table->timestamp('failed_at')->useCurrent();
});
};
}

/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('failed_jobs');
}
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
<?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::table('events', function (Blueprint $table) {
$table->index('approved_by');
});
}

/**
* Reverse the migrations.
*/
public function down(): void
{

Schema::table('events', function (Blueprint $table) {
$table->dropIndex(['approved_by']);
});
}
};
71 changes: 71 additions & 0 deletions soft_delete_users_without_consent.sql
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
-- INSERT INTO `codeweek_prod`.`users` (`id`, `firstname`, `lastname`, `username`, `password`, `email`, `country_iso`, `remember_token`, `created_at`, `updated_at`, `privacy`, `email_display`, `receive_emails`, `approved`, `magic_key`, `consent_given_at`, `future_consent_given_at`, `email_verified_at`) VALUES (1000000, 'Codeweek Legacy User', 'Codeweek Legacy', 'codeweek-legacy', '$2y$10$zcr85hlNx0U1aKpLtpsoKeAuPeSh1U9TxDY/A045m0y.5eXARsFg6', '[email protected]', 'BE', '5Pb4APP0CkOmQFzDtR960OlkdreLev2tvfUiywDawyJOmKDvFb3bmOAWlUL6', now(), now(), 1, '[email protected]', 0, 1, 2520090911, now(), now(), now());

DROP TABLE IF EXISTS users_to_delete;
-- Create temporary table with users to process
CREATE TEMPORARY TABLE users_to_delete AS
SELECT id
FROM users
WHERE consent_given_at IS NULL
LIMIT 1000; -- Process in chunks of 1000

-- Add index to improve performance
ALTER TABLE users_to_delete ADD INDEX (id);

START TRANSACTION;
-- Update events creator_id
UPDATE events e
INNER JOIN users_to_delete u ON e.creator_id = u.id
SET e.creator_id = 1000000;
COMMIT;

START TRANSACTION;
-- Update events approved_by
UPDATE events e
INNER JOIN users_to_delete u ON e.approved_by = u.id
SET e.approved_by = 1000000;
COMMIT;

START TRANSACTION;
-- Update participations
UPDATE participations p
INNER JOIN users_to_delete u ON p.user_id = u.id
SET p.user_id = 1000000;
COMMIT;

START TRANSACTION;
-- Update excellences
UPDATE excellences e
INNER JOIN users_to_delete u ON e.user_id = u.id
SET e.user_id = 1000000;
COMMIT;

START TRANSACTION;
-- Update model_has_roles
UPDATE model_has_roles m
INNER JOIN users_to_delete u ON m.model_id = u.id
SET m.model_id = 1000000;
COMMIT;

START TRANSACTION;
-- Update leading_teacher_expertise_user
UPDATE leading_teacher_expertise_user l
INNER JOIN users_to_delete u ON l.user_id = u.id
SET l.user_id = 1000000;
COMMIT;

START TRANSACTION;
-- Soft delete the users
UPDATE users u
INNER JOIN users_to_delete d ON u.id = d.id
SET u.deleted_at = '2024-12-13 17:07:24',
u.updated_at = '2024-12-13 17:07:24';
COMMIT;

-- Delete processed users
DELETE FROM users
WHERE id IN (SELECT id FROM users_to_delete);

-- Clean up
DROP TABLE IF EXISTS users_to_delete;

-- Repeat the process for the next batch by running this script again until no more users are found

0 comments on commit eabd49e

Please sign in to comment.