From bdca92773d9758c4347861ce44d602c20c3aaa9d Mon Sep 17 00:00:00 2001 From: Robin Date: Thu, 10 Jan 2019 15:50:00 +0100 Subject: [PATCH 1/7] ISSUE-1: Started implementing UI to configure replacements --- README.md | 7 +- config/install/gdpr_dumper.settings.yml | 4 +- gdpr_dumper.routing.yml | 7 ++ gdpr_dumper.services.yml | 4 + src/Event/GdprDumperEvents.php | 9 -- src/Event/GdprExpressionsEvent.php | 43 ------- src/Form/GdprDumperSettingsForm.php | 153 ++++++++++++++++++++++++ src/GdprDumperEnums.php | 31 +++++ src/Manager/DatabaseManager.php | 53 ++++++++ src/Sql/GdprSqlBase.php | 12 -- 10 files changed, 252 insertions(+), 71 deletions(-) create mode 100644 gdpr_dumper.routing.yml create mode 100644 gdpr_dumper.services.yml delete mode 100644 src/Event/GdprExpressionsEvent.php create mode 100644 src/Form/GdprDumperSettingsForm.php create mode 100644 src/GdprDumperEnums.php create mode 100644 src/Manager/DatabaseManager.php diff --git a/README.md b/README.md index f5566f2..4c50f9d 100644 --- a/README.md +++ b/README.md @@ -28,16 +28,15 @@ you are fully GDPR compliant YO! This module can be configured by editing the `gdpr_dumper.settings.yml` [file](https://github.com/robiningelbrecht/gdpr-dumper/blob/master/config/install/gdpr_dumper.settings.yml). [machbarmacher/gdpr-dump](https://github.com/machbarmacher/gdpr-dump) contains more info about -the **gdpr-expressions** and **gdpr-replacement** options. +the **gdpr-replacement** options. The provided yml file expects the same structure as explained in the readme above. ## Events -The module dispatches two events: -* `GdprDumperEvents::GDPR_EXPRESSIONS` +The module dispatches one event: * `GdprDumperEvents::GDPR_REPLACEMENTS` -This allows developers to alter the expressions and replacements through event subscribers on run-time +This allows developers to alter the replacements through event subscribers on run-time Happy GDPR'ing! diff --git a/config/install/gdpr_dumper.settings.yml b/config/install/gdpr_dumper.settings.yml index 5b0c8b5..f63898f 100644 --- a/config/install/gdpr_dumper.settings.yml +++ b/config/install/gdpr_dumper.settings.yml @@ -1,6 +1,3 @@ -gdpr_expressions: - users_field_data: - init: 'uid' gdpr_replacements: users_field_data: name: @@ -9,6 +6,7 @@ gdpr_replacements: formatter: 'email' pass: formatter: 'clear' +# TODO: move this to enum class. drivers: mysql: dump_command: 'mysqldump' diff --git a/gdpr_dumper.routing.yml b/gdpr_dumper.routing.yml new file mode 100644 index 0000000..7d90552 --- /dev/null +++ b/gdpr_dumper.routing.yml @@ -0,0 +1,7 @@ +gdpr_dumper.settings: + path: '/admin/config/development/gdpr-dumper' + defaults: + _form: '\Drupal\gdpr_dumper\Form\GdprDumperSettingsForm' + _title: 'GDPR dumper' + requirements: + _permission: 'administer gdpr dumper' diff --git a/gdpr_dumper.services.yml b/gdpr_dumper.services.yml new file mode 100644 index 0000000..80187b9 --- /dev/null +++ b/gdpr_dumper.services.yml @@ -0,0 +1,4 @@ +services: + gdpr_dumper.database_manager: + class: Drupal\gdpr_dumper\Manager\DatabaseManager + arguments: ['@database'] \ No newline at end of file diff --git a/src/Event/GdprDumperEvents.php b/src/Event/GdprDumperEvents.php index b8a05c4..fafc9f9 100644 --- a/src/Event/GdprDumperEvents.php +++ b/src/Event/GdprDumperEvents.php @@ -7,15 +7,6 @@ */ final class GdprDumperEvents { - /** - * Name of the event fired building the GDPR expressions. - * - * @Event - * - * @see \Drupal\gdpr_dumper\Event\GdprExpressionsEvent - */ - const GDPR_EXPRESSIONS = 'gdpr_dumper.expressions'; - /** * Name of the event fired building the GDPR replacements. * diff --git a/src/Event/GdprExpressionsEvent.php b/src/Event/GdprExpressionsEvent.php deleted file mode 100644 index 6c0b963..0000000 --- a/src/Event/GdprExpressionsEvent.php +++ /dev/null @@ -1,43 +0,0 @@ -expressions = $expressions; - } - - /** - * @return array - */ - public function getExpressions() { - return $this->expressions; - } - - /** - * @param array $expressions - * @return $this - */ - public function setExpressions(array $expressions) { - $this->expressions = $expressions; - return $this; - } - -} diff --git a/src/Form/GdprDumperSettingsForm.php b/src/Form/GdprDumperSettingsForm.php new file mode 100644 index 0000000..df01736 --- /dev/null +++ b/src/Form/GdprDumperSettingsForm.php @@ -0,0 +1,153 @@ +connection = $connection; + $this->databaseManager = $database_manager; + $this->settings = $this->config('gdpr_dumper.settings'); + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container) { + return new static( + $container->get('config.factory'), + $container->get('database'), + $container->get('gdpr_dumper.database_manager') + ); + } + + /** + * {@inheritdoc} + */ + public function getFormId() { + return 'gdpr_dumper_settings_form'; + } + + /** + * {@inheritdoc} + */ + protected function getEditableConfigNames() { + return ['gdpr_dumper.settings']; + } + + /** + * {@inheritdoc} + */ + public function buildForm(array $form, FormStateInterface $form_state) { + $replacements = $this->settings->get('gdpr_replacements'); + $database_tables = $this->databaseManager->getTableColumns(); + $db_schema = $this->connection->schema(); + $schema_handles_db_comments = \is_callable([$db_schema, 'getComment']); + + $form['intro'] = [ + '#type' => 'item', + '#title' => $this->t('Manage tables and columns that contain sensitive data'), + ]; + + $form['advanced'] = [ + '#type' => 'vertical_tabs', + ]; + + $form['replacements'] = [ + '#tree' => TRUE, + ]; + + foreach ($replacements as $table => $columns) { + if (isset($database_tables[$table])) { + $form['replacements'][$table] = [ + '#type' => 'details', + '#title' => $table, + '#group' => 'advanced', + ]; + + $form['replacements'][$table]['columns'] = [ + '#type' => 'table', + '#header' => [ + ['data' => $this->t('Field')], + ['data' => $this->t('Type')], + ['data' => $this->t('Description')], + ['data' => $this->t('Apply anonymization')], + ], + // @todo: attach this in JS. + //'#title' => $schema_handles_db_comments ? $db_schema->getComment($table) : NULL, + ]; + + foreach ($columns as $column => $row) { + if (isset($database_tables[$table][$column])) { + $column_info = $database_tables[$table][$column]; + + $form['replacements'][$table]['columns'][$column]['field'] = [ + '#markup' => '' . $column_info['COLUMN_NAME'] . '', + ]; + $form['replacements'][$table]['columns'][$column]['data_type'] = [ + '#markup' => '' . $column_info['DATA_TYPE'] . '', + ]; + $form['replacements'][$table]['columns'][$column]['comment'] = [ + '#markup' => '' . (empty($column_info['COLUMN_COMMENT']) ? '-' : $column_info['COLUMN_COMMENT']) . '', + ]; + $form['replacements'][$table]['columns'][$column]['anonymization'] = [ + '#type' => 'select', + '#title' => $this->t('Apply anonymization'), + '#title_display' => 'invisible', + '#options' => GdprDumperEnums::fakerFormatters(), + '#empty_value' => '', + '#empty_option' => $this->t('- No -'), + '#required' => FALSE, + '#default_value' => $row['formatter'], + ]; + } + } + + $form['replacements'][$table]['empty'] = [ + '#type' => 'checkbox', + '#title' => $this->t('Empty this table'), + '#button_type' => 'secondary', + ]; + } + + $form['actions']['add_table'] = [ + '#type' => 'submit', + '#value' => $this->t('Add table'), + ]; + } + + return parent::buildForm($form, $form_state); + } + + /** + * {@inheritdoc} + */ + public function submitForm(array &$form, FormStateInterface $form_state) { + dpm($form_state->getValues()); + parent::submitForm($form, $form_state); + } + +} \ No newline at end of file diff --git a/src/GdprDumperEnums.php b/src/GdprDumperEnums.php new file mode 100644 index 0000000..40de9fc --- /dev/null +++ b/src/GdprDumperEnums.php @@ -0,0 +1,31 @@ + 'Generate a name', + 'phoneNumber' => t('Generate a phone number'), + 'username' => t('Generate a random user name'), + 'password' => t('Generate a random password'), + 'email' => t('Generate a random email address'), + 'date' => t('Generate a date'), + 'longText' => t('Generate a sentence'), + 'number' => t('Generate a number'), + 'randomText' => t('Generate a sentence'), + 'text' => t('Generate a paragraph'), + 'uri' => t('Generate a URI'), + 'clear' => t('Generate an empty string'), + ]; + } + +} \ No newline at end of file diff --git a/src/Manager/DatabaseManager.php b/src/Manager/DatabaseManager.php new file mode 100644 index 0000000..e891612 --- /dev/null +++ b/src/Manager/DatabaseManager.php @@ -0,0 +1,53 @@ +database = $database; + } + + /** + * @return array + */ + public function getTableColumns() { + $tables = $this->database->schema()->findTables('%'); + $columns = []; + foreach ($tables as $table) { + $result = $this->getColumns($table); + if (NULL === $result) { + continue; + } + $columns[$table] = $result->fetchAllAssoc('COLUMN_NAME', \PDO::FETCH_ASSOC); + } + + return $columns; + } + + /** + * @param $table + * @return \Drupal\Core\Database\StatementInterface|null + */ + protected function getColumns($table) { + // @todo: How cross-driver is this? + $query = $this->database->select('information_schema.columns', 'columns'); + $query->fields('columns', ['COLUMN_NAME', 'DATA_TYPE', 'COLUMN_COMMENT']); + $query->condition('TABLE_SCHEMA', $this->database->getConnectionOptions()['database']); + $query->condition('TABLE_NAME', $table); + return $query->execute(); + } +} \ No newline at end of file diff --git a/src/Sql/GdprSqlBase.php b/src/Sql/GdprSqlBase.php index 6911b1f..c38a28f 100644 --- a/src/Sql/GdprSqlBase.php +++ b/src/Sql/GdprSqlBase.php @@ -5,7 +5,6 @@ use Drupal\Component\Serialization\Json; use Drupal\Core\Database\Database; use Drupal\gdpr_dumper\Event\GdprDumperEvents; -use Drupal\gdpr_dumper\Event\GdprExpressionsEvent; use Drupal\gdpr_dumper\Event\GdprReplacementsEvent; use Drush\Drush; use Drush\Sql\SqlBase; @@ -61,17 +60,6 @@ public static function getInstance($dbSpec, $options, EventDispatcherInterface $ // Fetch module settings. $config = \Drupal::config('gdpr_dumper.settings'); - if (empty($options['extra-dump']) || strpos($options['extra-dump'], '--gdpr-expressions') === FALSE) { - - // Dispatch event so the expressions can be altered. - $event = new GdprExpressionsEvent($config->get('gdpr_expressions')); - $event_dispatcher->dispatch(GdprDumperEvents::GDPR_EXPRESSIONS, $event); - // Add the configured GDPR expressions to the command. - if($expressions = Json::encode($event->getExpressions())){ - //$options['extra-dump'] .= " --gdpr-expressions='$expressions'"; - } - } - if (empty($options['extra-dump']) || strpos($options['extra-dump'], '--gdpr-replacements') === FALSE) { // Dispatch event so the replacements can be altered. $event = new GdprReplacementsEvent($config->get('gdpr_replacements')); From 63675127ba2fc227769a1d552ee318bc2e9cb24c Mon Sep 17 00:00:00 2001 From: Robin Date: Thu, 10 Jan 2019 16:30:15 +0100 Subject: [PATCH 2/7] ISSUE-1: Add tables through UI + format settings array --- config/install/gdpr_dumper.settings.yml | 1 + src/Form/GdprDumperSettingsForm.php | 111 +++++++++++++++++------- 2 files changed, 82 insertions(+), 30 deletions(-) diff --git a/config/install/gdpr_dumper.settings.yml b/config/install/gdpr_dumper.settings.yml index f63898f..140b72e 100644 --- a/config/install/gdpr_dumper.settings.yml +++ b/config/install/gdpr_dumper.settings.yml @@ -1,3 +1,4 @@ +empty_tables: {} gdpr_replacements: users_field_data: name: diff --git a/src/Form/GdprDumperSettingsForm.php b/src/Form/GdprDumperSettingsForm.php index df01736..0ed556b 100644 --- a/src/Form/GdprDumperSettingsForm.php +++ b/src/Form/GdprDumperSettingsForm.php @@ -72,6 +72,30 @@ public function buildForm(array $form, FormStateInterface $form_state) { '#title' => $this->t('Manage tables and columns that contain sensitive data'), ]; + $tables_to_add = array_diff(array_keys($database_tables), array_keys($replacements)); + + $form['table'] = [ + '#type' => 'select', + '#title' => $this->t('Select table'), + '#title_display' => 'invisible', + '#options' => array_combine($tables_to_add, $tables_to_add), + '#empty_value' => '', + '#empty_option' => $this->t('- Select -'), + ]; + + $form['add_table'] = [ + 'actions' => [ + '#type' => 'actions', + 'submit' => [ + '#type' => 'submit', + '#value' => $this->t('Add table'), + '#submit' => [ + [$this, 'submitAddTable'] + ], + ], + ], + ]; + $form['advanced'] = [ '#type' => 'vertical_tabs', ]; @@ -100,30 +124,26 @@ public function buildForm(array $form, FormStateInterface $form_state) { //'#title' => $schema_handles_db_comments ? $db_schema->getComment($table) : NULL, ]; - foreach ($columns as $column => $row) { - if (isset($database_tables[$table][$column])) { - $column_info = $database_tables[$table][$column]; - - $form['replacements'][$table]['columns'][$column]['field'] = [ - '#markup' => '' . $column_info['COLUMN_NAME'] . '', - ]; - $form['replacements'][$table]['columns'][$column]['data_type'] = [ - '#markup' => '' . $column_info['DATA_TYPE'] . '', - ]; - $form['replacements'][$table]['columns'][$column]['comment'] = [ - '#markup' => '' . (empty($column_info['COLUMN_COMMENT']) ? '-' : $column_info['COLUMN_COMMENT']) . '', - ]; - $form['replacements'][$table]['columns'][$column]['anonymization'] = [ - '#type' => 'select', - '#title' => $this->t('Apply anonymization'), - '#title_display' => 'invisible', - '#options' => GdprDumperEnums::fakerFormatters(), - '#empty_value' => '', - '#empty_option' => $this->t('- No -'), - '#required' => FALSE, - '#default_value' => $row['formatter'], - ]; - } + foreach ($database_tables[$table] as $column_name => $column_properties) { + $form['replacements'][$table]['columns'][$column_name]['field'] = [ + '#markup' => '' . $column_properties['COLUMN_NAME'] . '', + ]; + $form['replacements'][$table]['columns'][$column_name]['data_type'] = [ + '#markup' => '' . $column_properties['DATA_TYPE'] . '', + ]; + $form['replacements'][$table]['columns'][$column_name]['comment'] = [ + '#markup' => '' . (empty($column_properties['COLUMN_COMMENT']) ? '-' : $column_properties['COLUMN_COMMENT']) . '', + ]; + $form['replacements'][$table]['columns'][$column_name]['anonymization'] = [ + '#type' => 'select', + '#title' => $this->t('Apply anonymization'), + '#title_display' => 'invisible', + '#options' => GdprDumperEnums::fakerFormatters(), + '#empty_value' => '', + '#empty_option' => $this->t('- No -'), + '#required' => FALSE, + '#default_value' => isset($replacements[$table][$column_name]['formatter']) ? $replacements[$table][$column_name]['formatter'] : FALSE, + ]; } $form['replacements'][$table]['empty'] = [ @@ -132,21 +152,52 @@ public function buildForm(array $form, FormStateInterface $form_state) { '#button_type' => 'secondary', ]; } - - $form['actions']['add_table'] = [ - '#type' => 'submit', - '#value' => $this->t('Add table'), - ]; } return parent::buildForm($form, $form_state); } + /** + * Submit callback to add a table to the list. + * + * @param array $form + * @param \Drupal\Core\Form\FormStateInterface $form_state + */ + public function submitAddTable(array &$form, FormStateInterface $form_state) { + if ($table = $form_state->getValue('table')) { + $replacements = $this->settings->get('gdpr_replacements'); + + if (!isset($replacements[$table])) { + $replacements[$table] = []; + } + + // Update config. + $this->settings->set('gdpr_replacements', $replacements)->save(); + $this->messenger() + ->addStatus($this->t('The table has been added. You can configure it by selecting the corresponding tab.')); + } + } + /** * {@inheritdoc} */ public function submitForm(array &$form, FormStateInterface $form_state) { - dpm($form_state->getValues()); + $settings = [ + 'gdpr_replacements' => [], + ]; + + $replacements = $form_state->getValue('replacements'); + // Format the replacement to a suitable config array. + foreach ($replacements as $table_name => $properties) { + foreach ($properties['columns'] as $column_name => $column) { + if (!empty($column['anonymization'])) { + $settings['gdpr_replacements'][$table_name][$column_name]['formatter'] = $column['anonymization']; + } + } + } + + // @todo: save settings if driver config is moved out of config file. + parent::submitForm($form, $form_state); } From 3b05c801bc90768fdd141a828942fbf52ac0da99 Mon Sep 17 00:00:00 2001 From: Robin Date: Thu, 10 Jan 2019 16:33:44 +0100 Subject: [PATCH 3/7] ISSUE-1: Added antoher TODO --- src/Form/GdprDumperSettingsForm.php | 1 + 1 file changed, 1 insertion(+) diff --git a/src/Form/GdprDumperSettingsForm.php b/src/Form/GdprDumperSettingsForm.php index 0ed556b..d2d0cf2 100644 --- a/src/Form/GdprDumperSettingsForm.php +++ b/src/Form/GdprDumperSettingsForm.php @@ -196,6 +196,7 @@ public function submitForm(array &$form, FormStateInterface $form_state) { } } + // @todo: order tables alphabetically before saving. // @todo: save settings if driver config is moved out of config file. parent::submitForm($form, $form_state); From 15a090a4fb3df89f437261a469308c05e5c25776 Mon Sep 17 00:00:00 2001 From: Robin Date: Thu, 10 Jan 2019 19:07:45 +0100 Subject: [PATCH 4/7] ISSUE-1: Save settings. Some UI improvements --- config/install/gdpr_dumper.settings.yml | 14 +------- gdpr_dumper.libraries.yml | 8 +++++ gdpr_dumper.links.menu.yml | 5 +++ gdpr_dumper.permissions.yml | 3 ++ js/gdpr-dumper.js | 13 +++++++ src/Form/GdprDumperSettingsForm.php | 25 +++++++++++--- src/GdprDumperEnums.php | 46 +++++++++++++++++++------ src/Sql/GdprSqlBase.php | 4 +-- 8 files changed, 88 insertions(+), 30 deletions(-) create mode 100644 gdpr_dumper.libraries.yml create mode 100644 gdpr_dumper.links.menu.yml create mode 100644 gdpr_dumper.permissions.yml create mode 100644 js/gdpr-dumper.js diff --git a/config/install/gdpr_dumper.settings.yml b/config/install/gdpr_dumper.settings.yml index 140b72e..db863c7 100644 --- a/config/install/gdpr_dumper.settings.yml +++ b/config/install/gdpr_dumper.settings.yml @@ -6,16 +6,4 @@ gdpr_replacements: mail: formatter: 'email' pass: - formatter: 'clear' -# TODO: move this to enum class. -drivers: - mysql: - dump_command: 'mysqldump' - oracle: - dump_command: 'mysqldump' - pqsql: - dump_command: 'pg_dump' - sqlite: - dump_command: 'dump' - sqlsrv: - dump_command: 'mysqldump' \ No newline at end of file + formatter: 'clear' \ No newline at end of file diff --git a/gdpr_dumper.libraries.yml b/gdpr_dumper.libraries.yml new file mode 100644 index 0000000..c4a4267 --- /dev/null +++ b/gdpr_dumper.libraries.yml @@ -0,0 +1,8 @@ +settings-form: + version: 1.x + js: + js/gdpr-dumper.js: {} + dependencies: + - core/jquery + - core/drupal.form + diff --git a/gdpr_dumper.links.menu.yml b/gdpr_dumper.links.menu.yml new file mode 100644 index 0000000..d1ae0d5 --- /dev/null +++ b/gdpr_dumper.links.menu.yml @@ -0,0 +1,5 @@ +gdpr_dumper.settings: + title: 'GDPR dumper' + description: 'Configure GDPR dumper drush command' + parent: system.admin_config_development + route_name: gdpr_dumper.settings \ No newline at end of file diff --git a/gdpr_dumper.permissions.yml b/gdpr_dumper.permissions.yml new file mode 100644 index 0000000..c2e61ad --- /dev/null +++ b/gdpr_dumper.permissions.yml @@ -0,0 +1,3 @@ +administer gdpr dumper: + title: 'Administer gdpr dumper' + description: 'Allow to configure the gdpr dumper drush command' diff --git a/js/gdpr-dumper.js b/js/gdpr-dumper.js new file mode 100644 index 0000000..9b78ed8 --- /dev/null +++ b/js/gdpr-dumper.js @@ -0,0 +1,13 @@ +(function($, Drupal) { + + Drupal.behaviors.gdprDumperSummary = { + attach: function (context, settings) { + // Display the action in the vertical tab summary. + $(context).find('details[data-table-summary]').drupalSetSummary(function(context) { + return Drupal.checkPlain($(context).data('table-summary')); + }); + + } + } + +})(jQuery, Drupal); \ No newline at end of file diff --git a/src/Form/GdprDumperSettingsForm.php b/src/Form/GdprDumperSettingsForm.php index d2d0cf2..74ab0ee 100644 --- a/src/Form/GdprDumperSettingsForm.php +++ b/src/Form/GdprDumperSettingsForm.php @@ -63,6 +63,8 @@ protected function getEditableConfigNames() { */ public function buildForm(array $form, FormStateInterface $form_state) { $replacements = $this->settings->get('gdpr_replacements'); + $empty_tables = $this->settings->get('empty_tables'); + $database_tables = $this->databaseManager->getTableColumns(); $db_schema = $this->connection->schema(); $schema_handles_db_comments = \is_callable([$db_schema, 'getComment']); @@ -102,26 +104,32 @@ public function buildForm(array $form, FormStateInterface $form_state) { $form['replacements'] = [ '#tree' => TRUE, + '#attached' => [ + 'library' => ['gdpr_dumper/settings-form'] + ], ]; foreach ($replacements as $table => $columns) { if (isset($database_tables[$table])) { + $table_summary = $schema_handles_db_comments ? $db_schema->getComment($table) : '-'; $form['replacements'][$table] = [ '#type' => 'details', '#title' => $table, '#group' => 'advanced', + '#attributes' => [ + 'data-table-summary' => $table_summary, + ], ]; $form['replacements'][$table]['columns'] = [ '#type' => 'table', + '#caption' => $table_summary, '#header' => [ ['data' => $this->t('Field')], ['data' => $this->t('Type')], ['data' => $this->t('Description')], ['data' => $this->t('Apply anonymization')], ], - // @todo: attach this in JS. - //'#title' => $schema_handles_db_comments ? $db_schema->getComment($table) : NULL, ]; foreach ($database_tables[$table] as $column_name => $column_properties) { @@ -150,6 +158,7 @@ public function buildForm(array $form, FormStateInterface $form_state) { '#type' => 'checkbox', '#title' => $this->t('Empty this table'), '#button_type' => 'secondary', + '#default_value' => isset($empty_tables[$table]) ? $empty_tables[$table] : FALSE, ]; } } @@ -171,6 +180,9 @@ public function submitAddTable(array &$form, FormStateInterface $form_state) { $replacements[$table] = []; } + // Order tables alphabetically before saving. + ksort($replacements); + // Update config. $this->settings->set('gdpr_replacements', $replacements)->save(); $this->messenger() @@ -184,6 +196,7 @@ public function submitAddTable(array &$form, FormStateInterface $form_state) { public function submitForm(array &$form, FormStateInterface $form_state) { $settings = [ 'gdpr_replacements' => [], + 'empty_tables' => [], ]; $replacements = $form_state->getValue('replacements'); @@ -193,11 +206,13 @@ public function submitForm(array &$form, FormStateInterface $form_state) { if (!empty($column['anonymization'])) { $settings['gdpr_replacements'][$table_name][$column_name]['formatter'] = $column['anonymization']; } - } + }; + $settings['empty_tables'][$table_name] = $properties['empty']; } - // @todo: order tables alphabetically before saving. - // @todo: save settings if driver config is moved out of config file. + // Save settings. + $this->settings->set('gdpr_replacements', $settings['gdpr_replacements']) + ->set('empty_tables', $settings['empty_tables'])->save(); parent::submitForm($form, $form_state); } diff --git a/src/GdprDumperEnums.php b/src/GdprDumperEnums.php index 40de9fc..98068f2 100644 --- a/src/GdprDumperEnums.php +++ b/src/GdprDumperEnums.php @@ -15,17 +15,43 @@ public static function fakerFormatters() { return [ 'name' => 'Generate a name', 'phoneNumber' => t('Generate a phone number'), - 'username' => t('Generate a random user name'), - 'password' => t('Generate a random password'), - 'email' => t('Generate a random email address'), - 'date' => t('Generate a date'), - 'longText' => t('Generate a sentence'), - 'number' => t('Generate a number'), - 'randomText' => t('Generate a sentence'), - 'text' => t('Generate a paragraph'), - 'uri' => t('Generate a URI'), - 'clear' => t('Generate an empty string'), + 'username' => t('Generate a random user name'), + 'password' => t('Generate a random password'), + 'email' => t('Generate a random email address'), + 'date' => t('Generate a date'), + 'longText' => t('Generate a sentence'), + 'number' => t('Generate a number'), + 'randomText' => t('Generate a sentence'), + 'text' => t('Generate a paragraph'), + 'uri' => t('Generate a URI'), + 'clear' => t('Generate an empty string'), ]; } + /** + * @param $driver + * @return array + */ + public static function driverOptions($driver) { + $map = [ + 'mysql' => [ + 'dump_command' => 'mysqldump', + ], + 'oracle' => [ + 'dump_command' => 'mysqldump', + ], + 'pqsql' => [ + 'dump_command' => 'pg_dump', + ], + 'sqlite' => [ + 'dump_command' => 'dump', + ], + 'sqlsrv' => [ + 'dump_command' => 'mysqldump', + ], + ]; + + return isset($map[$driver]) ? $map[$driver] : []; + } + } \ No newline at end of file diff --git a/src/Sql/GdprSqlBase.php b/src/Sql/GdprSqlBase.php index 0572858..ea1d8b1 100644 --- a/src/Sql/GdprSqlBase.php +++ b/src/Sql/GdprSqlBase.php @@ -6,6 +6,7 @@ use Drupal\Core\Database\Database; use Drupal\gdpr_dumper\Event\GdprDumperEvents; use Drupal\gdpr_dumper\Event\GdprReplacementsEvent; +use Drupal\gdpr_dumper\GdprDumperEnums; use Drush\Drush; use Drush\Sql\SqlBase; use Symfony\Component\EventDispatcher\EventDispatcherInterface; @@ -71,10 +72,9 @@ public static function getInstance($dbSpec, $options, EventDispatcherInterface $ } $instance = new $className($dbSpec, $options); - $driver_options = isset($config->get('drivers')[$driver]) ? $config->get('drivers')[$driver] : []; // Inject config $instance->setConfig(Drush::config()); - $instance->setDriverOptions($driver_options); + $instance->setDriverOptions(GdprDumperEnums::driverOptions($driver)); return $instance; } From 64e0a419a3e1c31d7ee9bb77060917391f724c6c Mon Sep 17 00:00:00 2001 From: Robin Date: Thu, 10 Jan 2019 19:23:07 +0100 Subject: [PATCH 5/7] ISSUE-1: Updated README --- README.md | 17 +++++++++++------ 1 file changed, 11 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 7dcb344..b2687d1 100644 --- a/README.md +++ b/README.md @@ -25,18 +25,23 @@ be GDPR compliant YO! ## Configuration -This module can be configured by editing the `gdpr_dumper.settings.yml` [file](https://github.com/robiningelbrecht/gdpr-dumper/blob/master/config/install/gdpr_dumper.settings.yml). +This module can be configured by navigating to `admin/config/development/gdpr-dumper`. +On this page you can configure the sanitization and anonymization +of every column of every table in your database. +## Events + +The module dispatches one event: +* `GdprDumperEvents::GDPR_REPLACEMENTS` + +This allows developers to alter the replacements through event subscribers on run-time. [machbarmacher/gdpr-dump](https://github.com/machbarmacher/gdpr-dump) contains more info about the **gdpr-replacement** options. The provided yml file expects the same structure as explained in the readme above. -## Events +## TODO -The module dispatches one event: -* `GdprDumperEvents::GDPR_REPLACEMENTS` - -This allows developers to alter the replacements through event subscribers on run-time +* Provide a way to allow to only export the structure of a table without the data. Happy GDPR'ing! From 2fd9673d5a7239f80ed9b36a7fd55b53fccf66ba Mon Sep 17 00:00:00 2001 From: Robin Date: Thu, 10 Jan 2019 19:26:19 +0100 Subject: [PATCH 6/7] ISSUE-1: Updated README --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index b2687d1..eb7715f 100644 --- a/README.md +++ b/README.md @@ -42,6 +42,6 @@ The provided yml file expects the same structure as explained in the readme abov ## TODO -* Provide a way to allow to only export the structure of a table without the data. +* Provide a way to allow to export the structure of a table without the data. Happy GDPR'ing! From 782fa9591fad8695ebd8825824b4d874d9111236 Mon Sep 17 00:00:00 2001 From: Robin Date: Fri, 11 Jan 2019 08:38:26 +0100 Subject: [PATCH 7/7] ISSUE-1: Show tables that only need emptying --- src/Form/GdprDumperSettingsForm.php | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/src/Form/GdprDumperSettingsForm.php b/src/Form/GdprDumperSettingsForm.php index 74ab0ee..ef7eea7 100644 --- a/src/Form/GdprDumperSettingsForm.php +++ b/src/Form/GdprDumperSettingsForm.php @@ -64,6 +64,15 @@ protected function getEditableConfigNames() { public function buildForm(array $form, FormStateInterface $form_state) { $replacements = $this->settings->get('gdpr_replacements'); $empty_tables = $this->settings->get('empty_tables'); + // Add the empty table options to the replacement array, if they don't exist already. + // We need this because if only the "Empty table" option is checked, + // the table won't be available in the replacements array. + foreach (array_filter($empty_tables) as $empty_table => $value) { + if (!isset($replacements[$empty_table])) { + $replacements[$empty_table] = []; + } + } + $database_tables = $this->databaseManager->getTableColumns(); $db_schema = $this->connection->schema();