From 5d5030a48be3d65df85d78e26690085c0889c6e3 Mon Sep 17 00:00:00 2001
From: Nils Adermann <naderman@naderman.de>
Date: Sun, 21 Aug 2011 02:57:01 -0400
Subject: [PATCH] [feature/extension-manager] Remove cron's dependency on the
 extension manager.

Instead a separate cron provider supplies the manager with tasks from the
extension finder.

PHPBB3-10323
---
 phpBB/common.php                              |  2 +-
 phpBB/cron.php                                |  2 +-
 phpBB/includes/cron/manager.php               | 48 +---------
 phpBB/includes/cron/provider.php              | 92 +++++++++++++++++++
 tests/cron/manager_test.php                   | 33 +++----
 tests/cron/provider_test.php                  | 45 +++++++++
 .../core => tasks}/simple_not_runnable.php    |  0
 .../cron/task/core => tasks}/simple_ready.php |  0
 .../core => tasks}/simple_should_not_run.php  |  0
 9 files changed, 156 insertions(+), 66 deletions(-)
 create mode 100644 phpBB/includes/cron/provider.php
 create mode 100644 tests/cron/provider_test.php
 rename tests/cron/{root2/includes/cron/task/core => tasks}/simple_not_runnable.php (100%)
 rename tests/cron/{root2/includes/cron/task/core => tasks}/simple_ready.php (100%)
 rename tests/cron/{root2/includes/cron/task/core => tasks}/simple_should_not_run.php (100%)

diff --git a/phpBB/common.php b/phpBB/common.php
index 61817972f9..3cc9e57e46 100644
--- a/phpBB/common.php
+++ b/phpBB/common.php
@@ -144,5 +144,5 @@ foreach ($cache->obtain_hooks() as $hook)
 
 if (!$config['use_system_cron'])
 {
-	$cron = new phpbb_cron_manager($phpbb_extension_manager, $cache->get_driver());
+	$cron = new phpbb_cron_manager(new phpbb_cron_provider($phpbb_extension_manager), $cache->get_driver());
 }
diff --git a/phpBB/cron.php b/phpBB/cron.php
index cc5964218a..6633a6c3fd 100644
--- a/phpBB/cron.php
+++ b/phpBB/cron.php
@@ -62,7 +62,7 @@ function do_cron($cron_lock, $run_tasks)
 
 if ($config['use_system_cron'])
 {
-	$cron = new phpbb_cron_manager($phpbb_root_path . 'includes/cron/task', $phpEx, $cache->get_driver());
+	$cron = new phpbb_cron_manager(new phpbb_cron_provider($phpbb_extension_manager), $cache->get_driver());
 }
 else
 {
diff --git a/phpBB/includes/cron/manager.php b/phpBB/includes/cron/manager.php
index ae48e233e0..a0bf018b33 100644
--- a/phpBB/includes/cron/manager.php
+++ b/phpBB/includes/cron/manager.php
@@ -32,65 +32,25 @@ class phpbb_cron_manager
 	*/
 	protected $tasks = array();
 
-	/**
-	* An extension manager to search for cron tasks in extensions.
-	* @var phpbb_extension_manager
-	*/
-	protected $extension_manager;
-
 	/**
 	* Constructor. Loads all available tasks.
 	*
-	* Tasks will be looked up in the core task directory located in
-	* includes/cron/task/core/ and in extensions. Task classes will be
-	* autoloaded and must be named according to autoloading naming conventions.
-	*
-	* Tasks in extensions must be located in a directory called cron or a subdir
-	* of a directory called cron. The class and filename must end in a _task
-	* suffix.
-	*
-	* @param phpbb_extension_manager $extension_manager phpBB extension manager
+	* @param array|Traversable $task_names Provides an iterable set of task names
 	*/
-	public function __construct(phpbb_extension_manager $extension_manager)
+	public function __construct($task_names)
 	{
-		$this->extension_manager = $extension_manager;
-
-		$task_names = $this->find_cron_task_names();
 		$this->load_tasks($task_names);
 	}
 
-	/**
-	* Finds cron task names using the extension manager.
-	*
-	* All PHP files in includes/cron/task/core/ are considered tasks. Tasks
-	* in extensions have to be located in a directory called cron or a subdir
-	* of a directory called cron. The class and filename must end in a _task
-	* suffix.
-	*
-	* @return array		List of task names
-	*/
-	public function find_cron_task_names()
-	{
-		$finder = $this->extension_manager->get_finder();
-
-		return $finder
-			->suffix('_task')
-			->directory('/cron')
-			->default_path('includes/cron/task/core/')
-			->default_suffix('')
-			->default_directory('')
-			->get_classes();
-	}
-
 	/**
 	* Loads tasks given by name, wraps them
 	* and puts them into $this->tasks.
 	*
-	* @param array $task_names		Array of strings
+	* @param array|Traversable $task_names		Array of strings
 	*
 	* @return void
 	*/
-	public function load_tasks(array $task_names)
+	public function load_tasks($task_names)
 	{
 		foreach ($task_names as $task_name)
 		{
diff --git a/phpBB/includes/cron/provider.php b/phpBB/includes/cron/provider.php
new file mode 100644
index 0000000000..9936da3f55
--- /dev/null
+++ b/phpBB/includes/cron/provider.php
@@ -0,0 +1,92 @@
+<?php
+/**
+*
+* @package phpBB3
+* @copyright (c) 2011 phpBB Group
+* @license http://opensource.org/licenses/gpl-license.php GNU Public License
+*
+*/
+
+/**
+* @ignore
+*/
+if (!defined('IN_PHPBB'))
+{
+	exit;
+}
+
+/**
+* Provides cron manager with tasks
+*
+* Finds installed cron tasks and makes them available to the cron manager.
+*
+* @package phpBB3
+*/
+class phpbb_cron_provider implements \IteratorAggregate
+{
+	/**
+	* Array holding all found task class names.
+	*
+	* @var array
+	*/
+	protected $task_names = array();
+
+	/**
+	* An extension manager to search for cron tasks in extensions.
+	* @var phpbb_extension_manager
+	*/
+	protected $extension_manager;
+
+	/**
+	* Constructor. Loads all available tasks.
+	*
+	* Tasks will be looked up in the core task directory located in
+	* includes/cron/task/core/ and in extensions. Task classes will be
+	* autoloaded and must be named according to autoloading naming conventions.
+	*
+	* Tasks in extensions must be located in a directory called cron or a subdir
+	* of a directory called cron. The class and filename must end in a _task
+	* suffix.
+	*
+	* @param phpbb_extension_manager $extension_manager phpBB extension manager
+	*/
+	public function __construct(phpbb_extension_manager $extension_manager)
+	{
+		$this->extension_manager = $extension_manager;
+
+		$this->task_names = $this->find_cron_task_names();
+	}
+
+	/**
+	* Finds cron task names using the extension manager.
+	*
+	* All PHP files in includes/cron/task/core/ are considered tasks. Tasks
+	* in extensions have to be located in a directory called cron or a subdir
+	* of a directory called cron. The class and filename must end in a _task
+	* suffix.
+	*
+	* @return array     List of task names
+	*/
+	public function find_cron_task_names()
+	{
+		$finder = $this->extension_manager->get_finder();
+
+		return $finder
+			->suffix('_task')
+			->directory('/cron')
+			->default_path('includes/cron/task/core/')
+			->default_suffix('')
+			->default_directory('')
+			->get_classes();
+	}
+
+	/**
+	* Retrieve an iterator over all task names
+	*
+	* @return ArrayIterator An iterator for the array of task names
+	*/
+	public function getIterator()
+	{
+		return new ArrayIterator($this->task_names);
+	}
+}
diff --git a/tests/cron/manager_test.php b/tests/cron/manager_test.php
index 80f2cd55a8..80c92e234b 100644
--- a/tests/cron/manager_test.php
+++ b/tests/cron/manager_test.php
@@ -11,33 +11,22 @@ require_once dirname(__FILE__) . '/../mock/extension_manager.php';
 require_once dirname(__FILE__) . '/includes/cron/task/core/dummy_task.php';
 require_once dirname(__FILE__) . '/includes/cron/task/core/second_dummy_task.php';
 require_once dirname(__FILE__) . '/ext/testext/cron/dummy_task.php';
-require_once dirname(__FILE__) . '/root2/includes/cron/task/core/simple_ready.php';
-require_once dirname(__FILE__) . '/root2/includes/cron/task/core/simple_not_runnable.php';
-require_once dirname(__FILE__) . '/root2/includes/cron/task/core/simple_should_not_run.php';
+require_once dirname(__FILE__) . '/tasks/simple_ready.php';
+require_once dirname(__FILE__) . '/tasks/simple_not_runnable.php';
+require_once dirname(__FILE__) . '/tasks/simple_should_not_run.php';
 
 class phpbb_cron_manager_test extends PHPUnit_Framework_TestCase
 {
 	public function setUp()
 	{
-		$this->extension_manager = new phpbb_mock_extension_manager(
-			dirname(__FILE__) . '/',
-			array(
-				'testext' => array(
-					'ext_name'		=> 'testext',
-					'ext_active'	=> true,
-					'ext_path'		=> dirname(__FILE__) . '/ext/testext/'
-				),
-			));
-		$this->manager = new phpbb_cron_manager($this->extension_manager);
+		$this->manager = new phpbb_cron_manager(array(
+			'phpbb_cron_task_core_dummy_task',
+			'phpbb_cron_task_core_second_dummy_task',
+			'phpbb_ext_testext_cron_dummy_task',
+		));
 		$this->task_name = 'phpbb_cron_task_core_dummy_task';
 	}
 
-	public function test_manager_finds_shipped_tasks()
-	{
-		$tasks = $this->manager->find_cron_task_names();
-		$this->assertEquals(3, sizeof($tasks));
-	}
-
 	public function test_manager_finds_shipped_task_by_name()
 	{
 		$task = $this->manager->find_task($this->task_name);
@@ -66,7 +55,11 @@ class phpbb_cron_manager_test extends PHPUnit_Framework_TestCase
 
 	public function test_manager_finds_only_ready_tasks()
 	{
-		$manager = new phpbb_cron_manager(new phpbb_mock_extension_manager(dirname(__FILE__) . '/root2/'));
+		$manager = new phpbb_cron_manager(array(
+			'phpbb_cron_task_core_simple_ready',
+			'phpbb_cron_task_core_simple_not_runnable',
+			'phpbb_cron_task_core_simple_should_not_run',
+		));
 		$tasks = $manager->find_all_ready_tasks();
 		$task_names = $this->tasks_to_names($tasks);
 		$this->assertEquals(array('phpbb_cron_task_core_simple_ready'), $task_names);
diff --git a/tests/cron/provider_test.php b/tests/cron/provider_test.php
new file mode 100644
index 0000000000..781425e6ab
--- /dev/null
+++ b/tests/cron/provider_test.php
@@ -0,0 +1,45 @@
+<?php
+/**
+*
+* @package testing
+* @copyright (c) 2010 phpBB Group
+* @license http://opensource.org/licenses/gpl-license.php GNU Public License
+*
+*/
+
+require_once dirname(__FILE__) . '/../mock/extension_manager.php';
+
+class phpbb_cron_provider_test extends PHPUnit_Framework_TestCase
+{
+	public function setUp()
+	{
+		$this->extension_manager = new phpbb_mock_extension_manager(
+			dirname(__FILE__) . '/',
+			array(
+				'testext' => array(
+					'ext_name'      => 'testext',
+					'ext_active'    => true,
+					'ext_path'      => dirname(__FILE__) . '/ext/testext/'
+				),
+			));
+		$this->provider = new phpbb_cron_provider($this->extension_manager);
+	}
+
+	public function test_manager_finds_shipped_tasks()
+	{
+		$task_iterator = $this->provider->find_cron_task_names();
+
+		$tasks = array();
+		foreach ($task_iterator as $task)
+		{
+			$tasks[] = $task;
+		}
+		sort($tasks);
+
+		$this->assertEquals(array(
+			'phpbb_cron_task_core_dummy_task',
+			'phpbb_cron_task_core_second_dummy_task',
+			'phpbb_ext_testext_cron_dummy_task',
+		), $tasks);
+	}
+}
diff --git a/tests/cron/root2/includes/cron/task/core/simple_not_runnable.php b/tests/cron/tasks/simple_not_runnable.php
similarity index 100%
rename from tests/cron/root2/includes/cron/task/core/simple_not_runnable.php
rename to tests/cron/tasks/simple_not_runnable.php
diff --git a/tests/cron/root2/includes/cron/task/core/simple_ready.php b/tests/cron/tasks/simple_ready.php
similarity index 100%
rename from tests/cron/root2/includes/cron/task/core/simple_ready.php
rename to tests/cron/tasks/simple_ready.php
diff --git a/tests/cron/root2/includes/cron/task/core/simple_should_not_run.php b/tests/cron/tasks/simple_should_not_run.php
similarity index 100%
rename from tests/cron/root2/includes/cron/task/core/simple_should_not_run.php
rename to tests/cron/tasks/simple_should_not_run.php