diff --git a/install.php b/install.php index 0effbd15fa681c15975c92deca03accbfdbe35cf..aeddb9d23608210e16de225cbf7a4900fae854b5 100644 --- a/install.php +++ b/install.php @@ -25,6 +25,11 @@ * @license http://opensource.org/licenses/gpl-2.0.php GNU General Public License * @link http://vufind.org/wiki/vufind2:installation_notes Wiki */ + +require_once __DIR__ . '/vendor/autoload.php'; + +use Zend\Console\Getopt; + define('MULTISITE_NONE', 0); define('MULTISITE_DIR_BASED', 1); define('MULTISITE_HOST_BASED', 2); @@ -35,27 +40,108 @@ $host = $module = ''; $multisiteMode = MULTISITE_NONE; $basePath = '/vufind'; +try { + $opts = new Getopt( + array( + 'use-defaults' => + 'Use VuFind Defaults to Configure (ignores any other arguments passed)', + 'overridedir=s' => + "Where would you like to store your local settings? [{$baseDir}/local]", + 'module-name=w' => + 'What module name would you like to use? Use disabled, to not use', + 'basepath=s' => + 'What base path should be used in VuFind\'s URL? [/vufind]', + 'multisite-w' => + 'Specify we are going to setup a multisite. Options: directory and host', + 'hostname=s' => + 'Specify the hostname for the VuFind Site, When multisite=host', + 'non-interactive' => + 'Use settings if provided via arguments, otherwise use defaults', + ) + ); + + $opts->parse(); +} catch (Exception $e) { + echo $e->getUsageMessage(); + exit; +} + echo "VuFind has been found in {$baseDir}.\n\n"; +// Are we allowing user interaction? +$interactive = !$opts->getOption('non-interactive'); +$userInputNeeded = array(); + // Load user settings if we are not forcing defaults: -if (!isset($argv[1]) || !in_array('--use-defaults', $argv)) { - $overrideDir = getOverrideDir($overrideDir); - $module = getModule(); - $basePath = getBasePath($basePath); +if (!$opts->getOption('use-defaults')) { + if ($opts->getOption('overridedir')) { + $overrideDir = $opts->getOption('overridedir'); + } else if ($interactive) { + $userInputNeeded['overrideDir'] = true; + } + if ($opts->getOption('module-name')) { + if ($opts->getOption('module-name') !== 'disabled') { + $module = $opts->getOption('module-name'); + if (($result = validateModule($module)) !== true) { + die($result . "\n"); + } + } + } else if ($interactive) { + $userInputNeeded['module'] = true; + } + + if ($opts->getOption('basepath')) { + $basePath = $opts->getOption('basepath'); + if (($result = validateBasePath($basePath, true)) !== true) { + die($result . "\n"); + } + } else if ($interactive) { + $userInputNeeded['basePath'] = true; + } // We assume "single site" mode unless the --multisite switch is set: - if (isset($argv[1]) && in_array('--multisite', $argv)) { + if ($opts->getOption('multisite')) { + if ($opts->getOption('multisite') === 'directory') { + $multisiteMode = MULTISITE_DIR_BASED; + } else if ($opts->getOption('multisite') === 'host') { + $multisiteMode = MULTISITE_HOST_BASED; + } else if (($bad = $opts->getOption('multisite')) && $bad !== true) { + die('Unexpected multisite mode: ' . $bad . "\n"); + } else if ($interactive) { + $userInputNeeded['multisiteMode'] = true; + } + } + + // Now that we've validated as many parameters as possible, retrieve + // user input where needed. + if (isset($userInputNeeded['overrideDir'])) { + $overrideDir = getOverrideDir($overrideDir); + } + if (isset($userInputNeeded['module'])) { + $module = getModule(); + } + if (isset($userInputNeeded['basePath'])) { + $basePath = getBasePath($basePath); + } + if (isset($userInputNeeded['multisiteMode'])) { $multisiteMode = getMultisiteMode(); } + + // Load supplemental multisite parameters: if ($multisiteMode == MULTISITE_HOST_BASED) { - $host = getHost(); + if ($opts->getOption('hostname')) { + $host = $opts->getOption('hostname'); + } else if ($interactive) { + $host = getHost(); + } } -} else { - // In interactive mode, we initialize the directory as part of the input - // process; in defaults mode, we need to do it here: - initializeOverrideDir($overrideDir, true); } +// Make sure the override directory is initialized (using defaults or CLI +// parameters will not have initialized it yet; attempt to reinitialize it +// here is harmless if it was already initialized in interactive mode): +initializeOverrideDir($overrideDir, true); + // Build the Windows start file in case we need it: buildWindowsConfig($baseDir, $overrideDir, $module); @@ -153,6 +239,24 @@ function getApacheLocation($overrideDir) } } +/** + * Validate a base path. Returns true on success, message on failure. + * + * @param string $basePath String to validate. + * @param bool $allowEmpty Are empty values acceptable? + * + * @return bool|string + */ +function validateBasePath($basePath, $allowEmpty = false) +{ + if ($allowEmpty && empty($basePath)) { + return true; + } + return preg_match('/^\/\w*$/', $basePath) + ? true + : 'Error: Base path must be alphanumeric and start with a slash.'; +} + /** * Get a base path from the user (or return a default). * @@ -168,9 +272,8 @@ function getBasePath($basePath) "What base path should be used in VuFind's URL? [{$basePath}] " ); if (!empty($basePathInput)) { - if (!preg_match('/^\/\w*$/', $basePathInput)) { - echo "Error: Base path must be alphanumeric and start with a " - . "slash.\n\n"; + if (($result = validateBasePath($basePathInput)) !== true) { + echo "$result\n\n"; } else { return $basePathInput; } @@ -226,12 +329,35 @@ function getOverrideDir($overrideDir) return str_replace('\\', '/', realpath($overrideDirInput)); } } else { - initializeOverrideDir($overrideDir, true); return $overrideDir; } } } +/** + * Validate the custom module name. Returns true on success, message on failure. + * + * @param string $module Module name to validate. + * + * @return bool|string + */ +function validateModule($module) +{ + $regex = '/^[a-zA-Z][0-9a-zA-Z_]*$/'; + $illegalModules = array( + 'VuDL', 'VuFind', 'VuFindAdmin', 'VuFindConsole', 'VuFindDevTools', + 'VuFindHttp', 'VuFindLocalTemplate', 'VuFindSearch', 'VuFindTest', + 'VuFindTheme', + ); + if (in_array($module, $illegalModules)) { + return "{$module} is a reserved module name; please try another."; + } else if (empty($module) || preg_match($regex, $module)) { + return true; + } else { + return "Illegal name: {$module}; please use alphanumeric text."; + } +} + /** * Get the custom module name from the user (or blank for none). * @@ -251,15 +377,10 @@ function getModule() "\nWhat module name would you like to use? [blank for none] " ) ); - $regex = '/^[a-zA-Z][0-9a-zA-Z_]*$/'; - $illegalModules = array('VuFind', 'VuFindConsole', 'VuFindTest'); - if (in_array($moduleInput, $illegalModules)) { - echo "\n{$moduleInput} is a reserved name; please try another.\n"; - } else if (empty($moduleInput) || preg_match($regex, $moduleInput)) { + if (($result = validateModule($moduleInput)) === true) { return $moduleInput; - } else { - echo "\nIllegal name: {$moduleInput}; please use alphanumeric text.\n"; } + echo "\n$result\n"; } } @@ -289,22 +410,36 @@ function getMultisiteMode() } /** - * Get the user's hostname preference. + * Validate the user's hostname input. Returns true on success, message on failure. * - * @return string + * @param string $host String to check + * + * @return bool|string */ -function getHost() +function validateHost($host) { // From http://stackoverflow.com/questions/106179/ // regular-expression-to-match-hostname-or-ip-address $valid = "/^(([a-zA-Z]|[a-zA-Z][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*" . "([A-Za-z]|[A-Za-z][A-Za-z0-9\-]*[A-Za-z0-9])$/"; + return preg_match($valid, $host) + ? true + : 'Invalid hostname.'; +} + +/** + * Get the user's hostname preference. + * + * @return string + */ +function getHost() +{ while (true) { $input = getInput("\nPlease enter the hostname for your site: "); - if (preg_match($valid, $input)) { + if (($result = validateHost($input)) === true) { return $input; } else { - echo "Invalid hostname.\n"; + echo "$result\n"; } } } @@ -318,19 +453,7 @@ function getHost() */ function getInput($prompt) { - // Standard function for most uses - if (function_exists('readline')) { - $in = readline($prompt); - return $in; - } else { - // Or use our own if it doesn't exist (windows) - print $prompt; - $fp = fopen("php://stdin", "r"); - $in = fgets($fp, 4094); // Maximum windows buffer size - fclose($fp); - // Seems to keep the carriage return if you don't trim - return trim($in); - } + return \Zend\Console\Prompt\Line::prompt($prompt, true); } /** @@ -375,6 +498,9 @@ function buildApacheConfig($baseDir, $overrideDir, $basePath, $module, $multi, $ ); break; case MULTISITE_HOST_BASED: + if (($result = validateHost($host)) !== true) { + die($result . "\n"); + } $config = preg_replace( '/SetEnv\s+(\w+)\s+(.*)/', 'SetEnvIfNoCase Host ' . str_replace('.', '\.', $host) . ' $1=$2',