From 8b8a6416ac25a48a0b1240e9069417a9447704d2 Mon Sep 17 00:00:00 2001 From: Guy Sartorelli Date: Fri, 6 Sep 2024 14:30:12 +1200 Subject: [PATCH] API Update API to reflect changes to CLI interaction --- docs/en/developer.md | 28 +++++----- src/Jobs/LDAPAllSyncJob.php | 10 +++- src/Jobs/LDAPMemberSyncJob.php | 7 ++- src/Tasks/LDAPGroupSyncTask.php | 56 ++++++-------------- src/Tasks/LDAPMemberSyncOneTask.php | 56 ++++++++++---------- src/Tasks/LDAPMemberSyncTask.php | 56 ++++++-------------- src/Tasks/LDAPMigrateExistingMembersTask.php | 41 ++++---------- 7 files changed, 97 insertions(+), 157 deletions(-) diff --git a/docs/en/developer.md b/docs/en/developer.md index ab5fddb..8942516 100644 --- a/docs/en/developer.md +++ b/docs/en/developer.md @@ -340,8 +340,8 @@ See the [module docs](https://github.com/symbiote/silverstripe-queuedjobs) on ho If you don't want to run a queued job, you can set a cronjob yourself by calling: -``` -env php vendor/silverstripe/framework/cli-script.php dev/tasks/LDAPMemberSyncTask +```sh +vendor/bin/sake tasks:LDAPMemberSyncTask ``` ### Syncing AD groups and users on a schedule @@ -365,8 +365,8 @@ SilverStripe\LDAP\Jobs\LDAPAllSyncJob: If you don't want to run a queued job, you can set a cronjob yourself by calling the two sync tasks (order is important, otherwise your group memberships might not get updated): ```sh -env php vendor/silverstripe/framework/cli-script.php dev/tasks/LDAPGroupSyncTask -env php vendor/silverstripe/framework/cli-script.php dev/tasks/LDAPMemberSyncTask +vendor/bin/sake tasks:LDAPGroupSyncTask +vendor/bin/sake tasks:LDAPMemberSyncTask ``` ### Migrating existing users @@ -438,9 +438,9 @@ The fallback authenticator will be used in the following conditions: ### Extending the member and group sync tasks with custom functionality -Both `LDAPMemberSyncTask` and `LDAPGroupSyncTask` provide extension points (`onAfterLDAPMemberSyncTask` and -`onAfterLDAPGroupSyncTask` respectively) after all members/groups have been synced and before the task exits. This is a -perfect time to set values that are dependent on a full sync - for example linking a user to their manager based on DNs. +Both `LDAPMemberSyncTask` and `LDAPGroupSyncTask` provide extension points (`onAfterLDAPMemberSyncTask` and +`onAfterLDAPGroupSyncTask` respectively) after all members/groups have been synced and before the task exits. This is a +perfect time to set values that are dependent on a full sync - for example linking a user to their manager based on DNs. For example: ```yaml @@ -462,7 +462,7 @@ use SilverStripe\Security\Member; class LDAPMemberSyncExtension extends Extension { /** - * Assuming the `DN` and `ManagerDN` values are set by LDAP, this code will link a member with their manager and + * Assuming the `DN` and `ManagerDN` values are set by LDAP, this code will link a member with their manager and * store the link in the `Manager` has_one. */ protected function onAfterLDAPMemberSyncTask() @@ -505,19 +505,19 @@ This will allow users to change their AD password via the regular CMS "forgot pa ### Allow SilverStripe attributes to be reset (removed) by AD -By default if attributes are present, and then missing in subsequent requests, they are ignored (non-destructive) by +By default if attributes are present, and then missing in subsequent requests, they are ignored (non-destructive) by this module. This can cause attributes to persist when they've been deliberately removed (attribute is no longer present) -in the LDAP source data. +in the LDAP source data. -If you wish a full two way sync to occur, then set the attribute on `LDAPService` for `reset_missing_attributes` to -enable a full sync. +If you wish a full two way sync to occur, then set the attribute on `LDAPService` for `reset_missing_attributes` to +enable a full sync. *Note*: This will mean syncs are destructive, and data or attributes will be reset if missing from the master LDAP source -data. +data. ```yaml SilverStripe\LDAP\Services\LDAPService: - reset_missing_attributes: true + reset_missing_attributes: true ``` ### Writing LDAP data from SilverStripe diff --git a/src/Jobs/LDAPAllSyncJob.php b/src/Jobs/LDAPAllSyncJob.php index ad370b2..fd1fd9b 100644 --- a/src/Jobs/LDAPAllSyncJob.php +++ b/src/Jobs/LDAPAllSyncJob.php @@ -6,10 +6,12 @@ use SilverStripe\LDAP\Tasks\LDAPGroupSyncTask; use SilverStripe\Core\Config\Config; use SilverStripe\Core\Injector\Injector; +use SilverStripe\HybridExecution\HybridOutput; use SilverStripe\LDAP\Tasks\LDAPMemberSyncTask; use Symbiote\QueuedJobs\Services\AbstractQueuedJob; use Symbiote\QueuedJobs\Services\QueuedJob; use Symbiote\QueuedJobs\Services\QueuedJobService; +use Symfony\Component\Console\Input\ArrayInput; /** * Class LDAPAllSyncJob @@ -84,11 +86,15 @@ public function process() singleton(QueuedJobService::class)->queueJob($nextJob, date('Y-m-d H:i:s', time() + $regenerateTime)); } + $output = HybridOutput::create(HybridOutput::FORMAT_ANSI); + $input = new ArrayInput([]); + $input->setInteractive(false); + $task = Injector::inst()->create(LDAPGroupSyncTask::class); - $task->run(null); + $task->run($input, $output); $task = Injector::inst()->create(LDAPMemberSyncTask::class); - $task->run(null); + $task->run($input, $output); $this->isComplete = true; } diff --git a/src/Jobs/LDAPMemberSyncJob.php b/src/Jobs/LDAPMemberSyncJob.php index 503ae63..a737fcb 100644 --- a/src/Jobs/LDAPMemberSyncJob.php +++ b/src/Jobs/LDAPMemberSyncJob.php @@ -6,9 +6,11 @@ use SilverStripe\LDAP\Tasks\LDAPMemberSyncTask; use SilverStripe\Core\Config\Config; use SilverStripe\Core\Injector\Injector; +use SilverStripe\HybridExecution\HybridOutput; use Symbiote\QueuedJobs\Services\AbstractQueuedJob; use Symbiote\QueuedJobs\Services\QueuedJob; use Symbiote\QueuedJobs\Services\QueuedJobService; +use Symfony\Component\Console\Input\ArrayInput; /** * Class LDAPMemberSyncJob @@ -89,7 +91,10 @@ public function process() } $task = Injector::inst()->create(LDAPMemberSyncTask::class); - $task->run(null); + $output = HybridOutput::create(HybridOutput::FORMAT_ANSI); + $input = new ArrayInput([]); + $input->setInteractive(false); + $task->run($input, $output); $this->isComplete = true; } diff --git a/src/Tasks/LDAPGroupSyncTask.php b/src/Tasks/LDAPGroupSyncTask.php index 7f79978..6888d8a 100644 --- a/src/Tasks/LDAPGroupSyncTask.php +++ b/src/Tasks/LDAPGroupSyncTask.php @@ -4,11 +4,13 @@ use Exception; use SilverStripe\Control\Director; -use SilverStripe\Control\HTTPRequest; use SilverStripe\Dev\BuildTask; +use SilverStripe\HybridExecution\HybridOutput; use SilverStripe\LDAP\Services\LDAPService; use SilverStripe\ORM\DB; use SilverStripe\Security\Group; +use Symfony\Component\Console\Command\Command; +use Symfony\Component\Console\Input\InputInterface; /** * Class LDAPGroupSyncTask @@ -17,11 +19,7 @@ */ class LDAPGroupSyncTask extends BuildTask { - /** - * {@inheritDoc} - * @var string - */ - private static $segment = 'LDAPGroupSyncTask'; + protected static string $commandName = 'LDAPGroupSyncTask'; /** * @var array @@ -44,19 +42,12 @@ class LDAPGroupSyncTask extends BuildTask */ protected $ldapService; - /** - * @return string - */ - public function getTitle() + public function getTitle(): string { return _t(__CLASS__ . '.SYNCTITLE', 'Sync all groups from LDAP'); } - /** - * {@inheritDoc} - * @var HTTPRequest $request - */ - public function run($request) + protected function execute(InputInterface $input, HybridOutput $output): int { ini_set('max_execution_time', 900); @@ -70,8 +61,6 @@ public function run($request) 'objectguid' ); - $start = time(); - $created = 0; $updated = 0; $deleted = 0; @@ -84,14 +73,14 @@ public function run($request) $group = new Group(); $group->GUID = $data['objectguid']; - $this->log(sprintf( + $output->writeln(sprintf( 'Creating new Group (GUID: %s, sAMAccountName: %s)', $data['objectguid'], $data['samaccountname'] )); $created++; } else { - $this->log(sprintf( + $output->writeln(sprintf( 'Updating existing Group "%s" (ID: %s, GUID: %s, sAMAccountName: %s)', $group->getTitle(), $group->ID, @@ -104,7 +93,7 @@ public function run($request) try { $this->ldapService->updateGroupFromLDAP($group, $data); } catch (Exception $e) { - $this->log($e->getMessage()); + $output->writeln('' . $e->getMessage() . ''); continue; } } @@ -116,7 +105,7 @@ public function run($request) if (!isset($ldapGroups[$record['GUID']])) { $group = Group::get()->byId($record['ID']); - $this->log(sprintf( + $output->writeln(sprintf( 'Removing Group "%s" (GUID: %s) that no longer exists in LDAP.', $group->Title, $group->GUID @@ -129,7 +118,7 @@ public function run($request) } $group->delete(); } catch (Exception $e) { - $this->log($e->getMessage()); + $output->writeln('' . $e->getMessage() . ''); continue; } @@ -138,28 +127,15 @@ public function run($request) } } - $this->invokeWithExtensions('onAfterLDAPGroupSyncTask'); + $this->invokeWithExtensions('onAfterLDAPGroupSyncTask', $output); - $end = time() - $start; - - $this->log(sprintf( - 'Done. Created %s records. Updated %s records. Deleted %s records. Duration: %s seconds', + $output->writeln(sprintf( + 'Done. Created %s records. Updated %s records. Deleted %s records.', $created, $updated, - $deleted, - round($end ?? 0.0, 0) + $deleted )); - } - - /** - * Sends a message, formatted either for the CLI or browser - * - * @param string $message - */ - protected function log($message) - { - $message = sprintf('[%s] ', date('Y-m-d H:i:s')) . $message; - echo Director::is_cli() ? ($message . PHP_EOL) : ($message . '
'); + return Command::SUCCESS; } /** diff --git a/src/Tasks/LDAPMemberSyncOneTask.php b/src/Tasks/LDAPMemberSyncOneTask.php index 02ecf8f..0f13926 100644 --- a/src/Tasks/LDAPMemberSyncOneTask.php +++ b/src/Tasks/LDAPMemberSyncOneTask.php @@ -3,8 +3,11 @@ namespace SilverStripe\LDAP\Tasks; use Exception; -use SilverStripe\Control\HTTPRequest; +use SilverStripe\HybridExecution\HybridOutput; use SilverStripe\LDAP\Services\LDAPService; +use Symfony\Component\Console\Command\Command; +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\InputOption; /** * Class LDAPMemberSyncOneTask @@ -12,15 +15,11 @@ * * Debug build task that can be used to sync a single member by providing their email address registered in LDAP. * - * Usage: /dev/tasks/LDAPMemberSyncOneTask?mail=john.smith@example.com + * Usage: sake tasks:LDAPMemberSyncOneTask --email=john.smith@example.com */ class LDAPMemberSyncOneTask extends LDAPMemberSyncTask { - /** - * {@inheritDoc} - * @var string - */ - private static $segment = 'LDAPMemberSyncOneTask'; + protected static string $commandName = 'LDAPMemberSyncOneTask'; /** * @var array @@ -34,40 +33,32 @@ class LDAPMemberSyncOneTask extends LDAPMemberSyncTask */ protected $ldapService; - /** - * @return string - */ - public function getTitle() + public function getTitle(): string { return _t(__CLASS__ . '.SYNCONETITLE', 'Sync single user from LDAP'); } - /** - * Syncs a single user based on the email address passed in the URL - * - * @param HTTPRequest $request - */ - public function run($request) + protected function execute(InputInterface $input, HybridOutput $output): int { - $email = $request->getVar('email'); + $email = $input->getOption('email'); if (!$email) { - echo 'You must supply an email parameter to this method.', PHP_EOL; - exit; + $output->writeln('You must supply an email address.'); + return Command::INVALID; } $user = $this->ldapService->getUserByEmail($email); if (!$user) { - echo sprintf('No user found in LDAP for email %s', $email), PHP_EOL; - exit; + $output->writeln(sprintf('No user found in LDAP for email %s', $email)); + return Command::FAILURE; } $member = $this->findOrCreateMember($user); // If member exists already, we're updating - otherwise we're creating if ($member->exists()) { - $this->log(sprintf( + $output->writeln(sprintf( 'Updating existing Member %s: "%s" (ID: %s, SAM Account Name: %s)', $user['objectguid'], $member->getName(), @@ -75,7 +66,7 @@ public function run($request) $user['samaccountname'] )); } else { - $this->log(sprintf( + $output->writeln(sprintf( 'Creating new Member %s: "%s" (SAM Account Name: %s)', $user['objectguid'], $user['cn'], @@ -83,15 +74,17 @@ public function run($request) )); } - $this->log('User data returned from LDAP follows:'); - $this->log(var_export($user)); + $output->writeln('User data returned from LDAP follows:'); + $output->writeln(var_export($user, true)); try { $this->ldapService->updateMemberFromLDAP($member, $user); - $this->log('Done!'); } catch (Exception $e) { - $this->log($e->getMessage()); + $output->writeln('' . $e->getMessage() . ''); + return Command::FAILURE; } + + return Command::SUCCESS; } /** @@ -103,4 +96,11 @@ public function setLDAPService(LDAPService $service) $this->ldapService = $service; return $this; } + + public function getOptions(): array + { + return [ + new InputOption('email', null, InputOption::VALUE_REQUIRED, 'Email address of the member to sync (required)'), + ]; + } } diff --git a/src/Tasks/LDAPMemberSyncTask.php b/src/Tasks/LDAPMemberSyncTask.php index a1dbe47..dd307c2 100644 --- a/src/Tasks/LDAPMemberSyncTask.php +++ b/src/Tasks/LDAPMemberSyncTask.php @@ -4,12 +4,14 @@ use Exception; use SilverStripe\Control\Director; -use SilverStripe\Control\HTTPRequest; use SilverStripe\Core\Config\Config; use SilverStripe\Dev\BuildTask; +use SilverStripe\HybridExecution\HybridOutput; use SilverStripe\LDAP\Services\LDAPService; use SilverStripe\ORM\DB; use SilverStripe\Security\Member; +use Symfony\Component\Console\Command\Command; +use Symfony\Component\Console\Input\InputInterface; /** * Class LDAPMemberSyncTask @@ -18,11 +20,7 @@ */ class LDAPMemberSyncTask extends BuildTask { - /** - * {@inheritDoc} - * @var string - */ - private static $segment = 'LDAPMemberSyncTask'; + protected static string $commandName = 'LDAPMemberSyncTask'; /** * @var array @@ -45,19 +43,12 @@ class LDAPMemberSyncTask extends BuildTask */ protected $ldapService; - /** - * @return string - */ - public function getTitle() + public function getTitle(): string { return _t(__CLASS__ . '.SYNCTITLE', 'Sync all users from Active Directory'); } - /** - * {@inheritDoc} - * @param HTTPRequest $request - */ - public function run($request) + protected function execute(InputInterface $input, HybridOutput $output): int { ini_set('max_execution_time', 3600); // 3600s = 1hr ini_set('memory_limit', '1024M'); // 1GB memory limit @@ -70,8 +61,6 @@ public function run($request) array_keys(Config::inst()->get(Member::class, 'ldap_field_mappings') ?? []) )); - $start = time(); - $created = 0; $updated = 0; $deleted = 0; @@ -82,7 +71,7 @@ public function run($request) // If member exists already, we're updating - otherwise we're creating if ($member->exists()) { $updated++; - $this->log(sprintf( + $output->writeln(sprintf( 'Updating existing Member %s: "%s" (ID: %s, SAM Account Name: %s)', $data['objectguid'], $member->getName(), @@ -91,7 +80,7 @@ public function run($request) )); } else { $created++; - $this->log(sprintf( + $output->writeln(sprintf( 'Creating new Member %s: "%s" (SAM Account Name: %s)', $data['objectguid'], $data['cn'], @@ -104,7 +93,7 @@ public function run($request) try { $this->ldapService->updateMemberFromLDAP($member, $data); } catch (Exception $e) { - $this->log($e->getMessage()); + $output->writeln('' . $e->getMessage() . ''); continue; } } @@ -116,7 +105,7 @@ public function run($request) $member = Member::get()->byId($record['ID']); if (!isset($users[$record['GUID']])) { - $this->log(sprintf( + $output->writeln(sprintf( 'Removing Member "%s" (GUID: %s) that no longer exists in LDAP.', $member->getName(), $member->GUID @@ -125,7 +114,7 @@ public function run($request) try { $member->delete(); } catch (Exception $e) { - $this->log($e->getMessage()); + $output->writeln('' . $e->getMessage() . ''); continue; } @@ -134,28 +123,15 @@ public function run($request) } } - $this->invokeWithExtensions('onAfterLDAPMemberSyncTask'); + $this->invokeWithExtensions('onAfterLDAPMemberSyncTask', $output); - $end = time() - $start; - - $this->log(sprintf( - 'Done. Created %s records. Updated %s records. Deleted %s records. Duration: %s seconds', + $output->writeln(sprintf( + 'Done. Created %s records. Updated %s records. Deleted %s records.', $created, $updated, - $deleted, - round($end ?? 0.0, 0) + $deleted )); - } - - /** - * Sends a message, formatted either for the CLI or browser - * - * @param string $message - */ - protected function log($message) - { - $message = sprintf('[%s] ', date('Y-m-d H:i:s')) . $message; - echo Director::is_cli() ? ($message . PHP_EOL) : ($message . '
'); + return Command::SUCCESS; } /** diff --git a/src/Tasks/LDAPMigrateExistingMembersTask.php b/src/Tasks/LDAPMigrateExistingMembersTask.php index d6ad4ca..4dc4bbc 100644 --- a/src/Tasks/LDAPMigrateExistingMembersTask.php +++ b/src/Tasks/LDAPMigrateExistingMembersTask.php @@ -2,12 +2,13 @@ namespace SilverStripe\LDAP\Tasks; -use SilverStripe\Control\Director; -use SilverStripe\Control\HTTPRequest; use SilverStripe\Core\Convert; use SilverStripe\Dev\BuildTask; +use SilverStripe\HybridExecution\HybridOutput; use SilverStripe\LDAP\Services\LDAPService; use SilverStripe\Security\Member; +use Symfony\Component\Console\Command\Command; +use Symfony\Component\Console\Input\InputInterface; /** * Class LDAPMigrateExistingMembersTask @@ -17,11 +18,7 @@ */ class LDAPMigrateExistingMembersTask extends BuildTask { - /** - * {@inheritDoc} - * @var string - */ - private static $segment = 'LDAPMigrateExistingMembersTask'; + protected static string $commandName = 'LDAPMigrateExistingMembersTask'; /** * @var array @@ -35,22 +32,14 @@ class LDAPMigrateExistingMembersTask extends BuildTask */ public $ldapService; - /** - * @return string - */ - public function getTitle() + public function getTitle(): string { return _t(__CLASS__ . '.TITLE', 'Migrate existing members in SilverStripe into LDAP members'); } - /** - * {@inheritDoc} - * @param HTTPRequest $request - */ - public function run($request) + protected function execute(InputInterface $input, HybridOutput $output): int { $users = $this->ldapService->getUsers(['objectguid', 'mail']); - $start = time(); $count = 0; foreach ($users as $user) { @@ -73,7 +62,7 @@ public function run($request) $count++; - $this->log(sprintf( + $output->writeln(sprintf( 'Migrated Member %s (ID: %s, Email: %s)', $member->getName(), $member->ID, @@ -81,19 +70,7 @@ public function run($request) )); } - $end = time() - $start; - - $this->log(sprintf('Done. Migrated %s Member records. Duration: %s seconds', $count, round($end ?? 0.0, 0))); - } - - /** - * Sends a message, formatted either for the CLI or browser - * - * @param string $message - */ - protected function log($message) - { - $message = sprintf('[%s] ', date('Y-m-d H:i:s')) . $message; - echo Director::is_cli() ? ($message . PHP_EOL) : ($message . '
'); + $output->writeln("Done. Migrated $count Member records."); + return Command::SUCCESS; } }