Completed
Push — main ( b6e7c9...8e589f )
by Paolo
33s queued 25s
created

AnonymizeUsersCommand   A

Complexity

Total Complexity 8

Size/Duplication

Total Lines 127
Duplicated Lines 0 %

Importance

Changes 4
Bugs 3 Features 0
Metric Value
wmc 8
eloc 52
c 4
b 3
f 0
dl 0
loc 127
rs 10

4 Methods

Rating   Name   Duplication   Size   Complexity  
A buildOptionParser() 0 10 1
A objectsGenerator() 0 8 2
A anonymize() 0 18 2
A execute() 0 29 3
1
<?php
2
declare(strict_types=1);
3
4
/**
5
 * BEdita, API-first content management framework
6
 * Copyright 2024 Atlas Srl, Chialab Srl
7
 *
8
 * This file is part of BEdita: you can redistribute it and/or modify
9
 * it under the terms of the GNU Lesser General Public License as published
10
 * by the Free Software Foundation, either version 3 of the License, or
11
 * (at your option) any later version.
12
 *
13
 * See LICENSE.LGPL or <http://gnu.org/licenses/lgpl-3.0.html> for more details.
14
 */
15
16
namespace BEdita\ImportTools\Command;
17
18
use BEdita\Core\Model\Entity\User;
19
use BEdita\Core\Model\Table\UsersTable;
20
use BEdita\Core\Utility\LoggedUser;
21
use Cake\Command\Command;
22
use Cake\Console\Arguments;
23
use Cake\Console\ConsoleIo;
24
use Cake\Console\ConsoleOptionParser;
25
use Cake\ORM\Query;
26
use Cake\Utility\Text;
27
use Faker\Factory;
28
use Faker\Generator;
29
30
/**
31
 * Anonymize users command.
32
 */
33
class AnonymizeUsersCommand extends Command
34
{
35
    /**
36
     * Users to preserve by id.
37
     *
38
     * @var array
39
     */
40
    protected array $preserveUsers = [
41
        1, // admin
42
    ];
43
44
    /**
45
     * @inheritDoc
46
     */
47
    public function buildOptionParser(ConsoleOptionParser $parser): ConsoleOptionParser
48
    {
49
        return parent::buildOptionParser($parser)
50
            ->addOptions([
51
                'id' => [
52
                    'help' => 'User id',
53
                ],
54
                'preserve' => [
55
                    'help' => 'Users to preserve by id',
56
                    'default' => '1',
57
                ],
58
            ]);
59
    }
60
61
    /**
62
     * {@inheritDoc}
63
     *
64
     * Anonymize Users command.
65
     *
66
     * $ bin/cake anonymize_users --help
67
     *
68
     * Usage:
69
     * cake anonymize_users [options]
70
     *
71
     * Options:
72
     *
73
     * --id           User id
74
     * --preserve     Users to preserve by id
75
     * --help, -h     Display this help.
76
     * --verbose, -v  Enable verbose output.
77
     *
78
     * # basic
79
     * $ bin/cake anonymize_users --id 2
80
     * $ bin/cake anonymize_users --preserve 1,2,3
81
     */
82
    public function execute(Arguments $args, ConsoleIo $io): int
83
    {
84
        $io->success('Start.');
85
        LoggedUser::setUserAdmin();
86
        /** @var \BEdita\Core\Model\Table\UsersTable $table */
87
        $table = $this->fetchTable('Users');
88
        $this->preserveUsers = array_map('intval', explode(',', $args->getOption('preserve')));
89
        $query = $table->find()
90
            ->where([
91
                $table->aliasField('locked') => 0,
92
                $table->aliasField('deleted') => 0,
93
                $table->aliasField('id') . ' NOT IN' => $this->preserveUsers,
94
            ]);
95
        $id = $args->getOption('id');
96
        if ($id) {
97
            $query = $query->where([$table->aliasField('id') => $id]);
98
        }
99
        $faker = Factory::create('it_IT');
100
        $processed = $saved = $errors = 0;
101
        /** @var \BEdita\Core\Model\Entity\User $user */
102
        foreach ($this->objectsGenerator($query) as $user) {
103
            $this->anonymize($faker, $user, $table, $io, $processed, $saved, $errors);
104
        }
105
        $io->out(sprintf('Users processed: %s', $processed));
106
        $io->out(sprintf('Users saved: %s', $saved));
107
        $io->out(sprintf('Users not saved: %s', $errors));
108
        $io->success('Done.');
109
110
        return Command::CODE_SUCCESS;
111
    }
112
113
    /**
114
     * Update user.
115
     *
116
     * @param \Faker\Generator $faker Faker generator
117
     * @param \BEdita\Core\Model\Entity\User $user User entity
118
     * @param \BEdita\Core\Model\Table\UsersTable $table Users table
119
     * @param \Cake\Console\ConsoleIo $io Console IO
120
     * @param int $processed Processed users
121
     * @param int $saved Saved users
122
     * @param int $errors Errors
123
     * @return void
124
     */
125
    public function anonymize(Generator $faker, User $user, UsersTable $table, ConsoleIo $io, int &$processed, int &$saved, int &$errors): void
126
    {
127
        $io->verbose(sprintf('Processing user %s [username: %s, email: %s]', $user->id, $user->username, $user->email));
128
        $user->name = $faker->firstName();
129
        $user->surname = $faker->lastName();
130
        $user->email = sprintf('%s.%s.%d@%s', Text::slug($user->name), Text::slug($user->surname), $user->id, $faker->safeEmailDomain());
131
        $user->uname = sprintf('user-%s', Text::uuid());
132
        $user->username = $user->email;
133
        $processed++;
134
        try {
135
            $table->saveOrFail($user);
136
            $this->log(sprintf('[OK] User %s updated', $user->id), 'debug');
137
            $saved++;
138
            $io->verbose(sprintf('Saved %s as [username: %s, email: %s]', $user->id, $user->username, $user->email));
139
        } catch (\Exception $e) {
140
            $this->log(sprintf('[KO] User %s not updated', $user->id), 'error');
141
            $errors++;
142
            $io->verbose(sprintf('Error %s as [username: %s, email: %s]', $user->id, $user->username, $user->email));
143
        }
144
    }
145
146
    /**
147
     * Objects generator.
148
     *
149
     * @param \Cake\ORM\Query $query Query object
150
     * @return \Generator
151
     */
152
    protected function objectsGenerator(Query $query): \Generator
153
    {
154
        $pageSize = 1000;
155
        $pages = ceil($query->count() / $pageSize);
156
        for ($page = 1; $page <= $pages; $page++) {
157
            yield from $query
158
                ->page($page, $pageSize)
159
                ->toArray();
160
        }
161
    }
162
}
163