123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371 |
- <?php
- /* For licensing terms, see /license.txt */
- use Cocur\Slugify\Slugify;
- use Symfony\Component\Finder\Finder;
- use Symfony\Component\Console\Input\ArrayInput;
- use Symfony\Component\Console\Output\StreamOutput;
- use Symfony\Component\Console\Application;
- use Doctrine\ORM\EntityManager;
- /**
- * Class Virtual
- */
- class Virtual
- {
- /**
- * @param array $_configuration
- */
- public static function hookConfiguration(& $_configuration)
- {
- global $virtualChamilo;
- if (defined('CLI_SCRIPT') && !defined('CLI_VCHAMILO_OVERRIDE')) {
- return;
- }
- // provides an effective value for the virtual root_web based on domain analysis
- self::getHostName($_configuration);
- // We are on physical chamilo. Let original config play
- $virtualChamiloWebRoot = rtrim($_configuration['vchamilo_web_root'], '/').'/';
- $virtualChamilo = [];
- if ($_configuration['root_web'] == $virtualChamiloWebRoot) {
- return;
- }
- // pre hook to chamilo main table and get alternate configuration.
- // sure Database object is not set up. Soo use bootstrap connection
- /** @var \Doctrine\DBAL\Connection $connection */
- $connection = self::bootConnection($_configuration);
- $query = "SELECT * FROM vchamilo WHERE root_web = '$virtualChamiloWebRoot'";
- $result = $connection->executeQuery($query);
- if ($result->rowCount()) {
- $data = $result->fetch();
- $excludes = array('id', 'name');
- $query = "SELECT * FROM settings_current WHERE subkey = 'vchamilo'";
- $virtualSettings = $connection->executeQuery($query);
- $virtualSettings = $virtualSettings->fetchAll();
- $homePath = '';
- $coursePath = '';
- $archivePath = '';
- $uploadPath = '';
- $passwordEncryption = '';
- foreach ($virtualSettings as $setting) {
- switch ($setting['variable']) {
- case 'vchamilo_upload_real_root':
- $uploadPath = $setting['selected_value'];
- break;
- case 'vchamilo_home_real_root':
- $homePath = $setting['selected_value'];
- break;
- case 'vchamilo_course_real_root':
- $coursePath = $setting['selected_value'];
- break;
- case 'vchamilo_archive_real_root':
- $archivePath = $setting['selected_value'];
- break;
- case 'vchamilo_password_encryption':
- $passwordEncryption = $setting['selected_value'];
- break;
- }
- }
- if (empty($homePath) || empty($coursePath) || empty($archivePath) || empty($uploadPath)) {
- echo 'Configure correctly the vchamilo plugin';
- exit;
- }
- // Only load if is visible
- if ($data && $data['visible'] === '1') {
- foreach ($data as $key => $value) {
- if (!in_array($key, $excludes)) {
- // Avoid empty password_encryption
- if ($key == 'password_encryption' && empty($value)) {
- continue;
- }
- $_configuration[$key] = $value;
- }
- $_configuration['virtual'] = $data['root_web'].'/';
- }
- $data['SYS_ARCHIVE_PATH'] = self::addTrailingSlash($archivePath).$data['slug'];
- $data['SYS_HOME_PATH'] = self::addTrailingSlash($homePath).$data['slug'];
- $data['SYS_COURSE_PATH'] = self::addTrailingSlash($coursePath).$data['slug'];
- $data['SYS_UPLOAD_PATH'] = self::addTrailingSlash($uploadPath).$data['slug'];
- $data['WEB_HOME_PATH'] = self::addTrailingSlash($data['home_url']);
- $data['WEB_UPLOAD_PATH'] = self::addTrailingSlash($data['upload_url']);
- $data['WEB_ARCHIVE_PATH'] = self::addTrailingSlash($data['archive_url']);
- //$data['WEB_COURSE_PATH'] = self::addTrailingSlash($data['course_url']);
- if (!empty($passwordEncryption)) {
- $_configuration['password_encryption'] = $passwordEncryption;
- }
- // Instance cannot have multiple urls
- $_configuration['multiple_access_urls'] = false;
- $_configuration['virtual_css_theme_folder'] = '';
- if (isset($data['css_theme_folder']) && !empty($data['css_theme_folder'])) {
- $_configuration['virtual_css_theme_folder'] = $data['css_theme_folder'];
- }
- $virtualChamilo = $data;
- } else {
- exit("This portal is disabled. Please contact your administrator");
- }
- } // otherwise it means the plugin was not configured yet
- }
- /**
- * @param array $_configuration
- */
- public static function getHostName(&$_configuration)
- {
- if (!empty($_SERVER['HTTP_X_FORWARDED_PROTO']) &&
- $_SERVER['HTTP_X_FORWARDED_PROTO'] == 'https' || !empty($_configuration['force_https_forwarded_proto'])
- ) {
- $protocol = 'https';
- } else {
- if (!empty($_SERVER['HTTPS'])) {
- $protocol = 'https';
- } else {
- $protocol = 'http';
- }
- }
- if (defined('CLI_VCHAMILO_OVERRIDE')) {
- $_configuration['vchamilo_web_root'] = CLI_VCHAMILO_OVERRIDE;
- $_configuration['vchamilo_name'] = preg_replace('#https?://#', '', CLI_VCHAMILO_OVERRIDE);
- // remove radical from override for name
- // fake the server signature
- global $_SERVER;
- $_SERVER['SERVER_NAME'] = $_configuration['vchamilo_name'];
- $_SERVER['HTTP_HOST'] = $_configuration['vchamilo_name'];
- $_SERVER['QUERY_STRING'] = '';
- $_SERVER['REQUEST_URI'] = CLI_VCHAMILO_OVERRIDE;
- return;
- }
- $contentPrefix = '/';
- if (isset($_SERVER['CONTEXT_PREFIX']) && !empty($_SERVER['CONTEXT_PREFIX'])) {
- $contentPrefix = $_SERVER['CONTEXT_PREFIX'];
- } else {
- // Getting url_append from URL
- if (isset($_SERVER['REQUEST_URI'])) {
- $requestUri = $_SERVER['REQUEST_URI'];
- if (strpos($requestUri, '/courses/') !== false) {
- $result = substr($requestUri, 0, strpos($requestUri, '/courses/'));
- if (!empty($result) && $result != '/') {
- $contentPrefix = $result;
- }
- }
- }
- }
- $_configuration['vchamilo_web_root'] = "{$protocol}://".@$_SERVER['HTTP_HOST'].$contentPrefix;
- $_configuration['vchamilo_name'] = @$_SERVER['HTTP_HOST'];
- if (empty($_configuration['vchamilo_name'])) { // try again with another source if has failed
- $_configuration['vchamilo_name'] = "{$protocol}://".$_SERVER['SERVER_NAME'];
- if ($_SERVER['SERVER_PORT'] != 80) {
- $_configuration['vchamilo_name'] .= ':'.$_SERVER['SERVER_PORT'];
- }
- $_configuration['vchamilo_name'] = $_SERVER['SERVER_NAME'];
- }
- }
- /**
- * @param string $path
- * @return string
- */
- public static function addTrailingSlash($path)
- {
- return substr($path, -1) == '/' ? $path : $path.'/';
- }
- /**
- * provides a side connection to a vchamilo database
- * @param array $_configuration
- *
- * @return \Doctrine\DBAL\Driver\Connection
- */
- public static function bootConnection(&$_configuration)
- {
- $dbParams = array(
- 'driver' => 'pdo_mysql',
- 'host' => $_configuration['db_host'],
- 'user' => $_configuration['db_user'],
- 'password' => $_configuration['db_password'],
- 'dbname' => isset($_configuration['main_database']) ? $_configuration['main_database'] : '',
- // Only relevant for pdo_sqlite, specifies the path to the SQLite database.
- 'path' => isset($_configuration['db_path']) ? $_configuration['db_path'] : '',
- // Only relevant for pdo_mysql, pdo_pgsql, and pdo_oci/oci8,
- 'port' => isset($_configuration['db_port']) ? $_configuration['db_port'] : '',
- );
- try {
- $database = new \Database();
- $connection = $database->connect(
- $dbParams,
- $_configuration['root_sys'],
- $_configuration['root_sys'],
- true
- );
- } catch (Exception $e) {
- echo('Side connection failure with '.$_configuration['db_host'].', '.$_configuration['db_user'].', ******** ');
- die();
- }
- return $connection;
- }
- /**
- * @param string $url
- */
- public static function redirect($url)
- {
- if (preg_match('#https?://#', $url)) {
- header('location: '.$url);
- } else {
- header('location: '.api_get_path(WEB_PATH).$url);
- }
- exit;
- }
- /**
- * @param string $course_folder
- * @return string
- */
- public static function getHtaccessFragment($course_folder)
- {
- $str = "
- # Change this file to fit your configuration and save it as .htaccess in the courses folder #
- # Chamilo mod rewrite
- # Comment lines start with # and are not processed
-
- <IfModule mod_rewrite.c>
- RewriteEngine On
-
- # Rewrite base is the dir chamilo is installed in with trailing slash
- RewriteBase /{$course_folder}/
-
- # Do not rewrite on the main dir
- # Change this path to the path of your main folder
- RewriteCond %{REQUEST_URI} !^/main/
-
- #replace nasty ampersands by 3 slashes, we change these back in download.php
- RewriteRule ([^/]+)/document/(.*)&(.*)$ $1/document/$2///$3 [N]
-
- # Rewrite everything in the scorm folder of a course to the download script
- RewriteRule ([^/]+)/scorm/(.*)$ /main/document/download_scorm.php?doc_url=/$2&cDir=$1 [QSA,L]
-
- # Rewrite everything in the document folder of a course to the download script
- RewriteRule ([^/]+)/document/(.*)$ /main/document/download.php?doc_url=/$2&cDir=$1 [QSA,L]
-
- # Rewrite everything in the work folder
- RewriteRule ([^/]+)/work/(.*)$ /main/work/download.php?file=work/$2&cDir=$1 [QSA,L]
- </IfModule>
- ";
- return $str;
- }
- /**
- * @return string
- */
- public static function getDefaultCourseIndexFragment()
- {
- return "<html><head></head><body></body></html>";
- }
- /**
- * @param string $template
- * @return bool
- */
- public static function templateExists($template)
- {
- global $_configuration;
- // Find and checktemplate directory (files and SQL).
- $separator = DIRECTORY_SEPARATOR;
- $templatefoldername = 'plugin'.$separator.'vchamilo'.$separator.'templates';
- $relative_datadir = $templatefoldername.$separator.$template.'_sql';
- $absolute_datadir = $_configuration['root_sys'].$relative_datadir;
- return is_dir($absolute_datadir);
- }
- /**
- * drop a vchamilo instance databases using the physical connection
- * @param stdClass $params
- * return an array of errors or false if ok
- */
- public static function dropDatabase($params)
- {
- $params = clone $params;
- if (empty($params->main_database)) {
- Display::addFlash(Display::return_message('No database found'));
- return;
- }
- $databaseToDelete = $params->main_database;
- unset($params->main_database);
- $connection = self::getConnectionFromInstance($params);
- if ($connection) {
- $databases = $connection->getSchemaManager()->listDatabases();
- if (in_array($databaseToDelete, $databases)) {
- $connection->getSchemaManager()->dropDatabase(
- $databaseToDelete
- );
- Display::addFlash(
- Display::return_message(
- 'Database deleted: '.$databaseToDelete
- )
- );
- } else {
- Display::addFlash(
- Display::return_message(
- 'Database does not exist: '.$databaseToDelete
- )
- );
- }
- } else {
- Display::addFlash(
- Display::return_message(
- "Cannot connect DB: $databaseToDelete"
- )
- );
- }
- return false;
- }
- /**
- * @param stdClass $params
- * @return bool
- */
- public static function createDatabase($params)
- {
- $params = clone $params;
- $databaseName = $params->main_database;
- unset($params->main_database);
- $connection = self::getConnectionFromInstance($params);
- if ($connection) {
- $databaseList = $connection->getSchemaManager()->listDatabases();
- if (!in_array($databaseName, $databaseList)) {
- $connection->getSchemaManager()->createDatabase($databaseName);
- Display::addFlash(
- Display::return_message("Creating DB ".$databaseName)
- );
- } else {
- Display::addFlash(
- Display::return_message("DB already exists: ".$databaseName)
- );
- }
- return true;
- }
- return false;
- }
- /**
- * get a proper SQLdump command
- * @param object $vchamilodata the complete new host information
- * @return string the shell command
- */
- public static function getDatabaseDumpCmd($vchamilodata)
- {
- $pgm = self::getConfig('vchamilo', 'mysql_cmd');
- if (!$pgm) {
- $pgm = '/usr/bin/mysql';
- }
- $phppgm = str_replace("\\", '/', $pgm);
- $phppgm = str_replace("\"", '', $phppgm);
- $pgm = str_replace("/", DIRECTORY_SEPARATOR, $pgm);
- if (!is_executable($phppgm)) {
- throw new Exception('databasecommanddoesnotmatchanexecutablefile');
- }
- // Retrieves the host configuration (more secure).
- $vchamilodata = empty($vchamilodata) ? self::makeThis() : $vchamilodata;
- if (strstr($vchamilodata->db_host, ':') !== false) {
- list($vchamilodata->db_host, $vchamilodata->db_port) = explode(
- ':',
- $vchamilodata->db_host
- );
- }
- // Password.
- $databasePassword = '';
- if (!empty($vchamilodata->db_password)) {
- $databasePassword = '-p'.escapeshellarg($vchamilodata->db_password).' ';
- }
- // Making the command line (see 'vconfig.php' file for defining the right paths).
- $sqlcmd = $pgm.' -h'.$vchamilodata->db_host.(isset($vchamilodata->db_port) ? ' -P'.$vchamilodata->db_port.' ' : ' ');
- $sqlcmd .= '-u'.$vchamilodata->db_user.' '.$databasePassword;
- $sqlcmd .= '%DATABASE% < ';
- return $sqlcmd;
- }
- /**
- * @param stdClass $vchamilo
- * @param string $template
- * @return bool
- */
- public static function loadDbTemplate($vchamilo, $template)
- {
- global $_configuration;
- // Make template directory (files and SQL).
- $separator = DIRECTORY_SEPARATOR;
- $templatefoldername = 'plugin'.$separator.'vchamilo'.$separator.'templates';
- $absolute_datadir = $_configuration['root_sys'].$templatefoldername.$separator.$template.$separator.'dump.sql';
- if (!$sqlcmd = self::getDatabaseDumpCmd($vchamilo)) {
- return false;
- }
- $sqlcmd = str_replace('%DATABASE%', $vchamilo->main_database, $sqlcmd);
- // Make final commands to execute, depending on the database type.
- $import = $sqlcmd.$absolute_datadir;
- // Execute the command.
- Display::addFlash(Display::return_message("Load database from template dump: \n $import "));
- if (!defined('CLI_SCRIPT')) {
- putenv('LANG=en_US.utf-8');
- }
- // ensure utf8 is correctly handled by php exec()
- // @see http://stackoverflow.com/questions/10028925/call-a-program-via-shell-exec-with-utf-8-text-input
- exec($import, $output, $return);
- if (!empty($output)) {
- Display::addFlash(Display::return_message(implode("\n", $output)."\n"));
- }
- return true;
- }
- /**
- * Backups a database for having a snapshot.
- * @param $vchamilo object The Vchamilo object.
- * @param $outputfilerad string The output SQL file radical.
- * @return bool If TRUE, dumping database was a success, otherwise FALSE.
- */
- public static function backupDatabase($vchamilo, $outputfilerad)
- {
- // Separating host and port, if sticked.
- if (strstr($vchamilo->db_host, ':') !== false) {
- list($host, $port) = explode(':', $vchamilo->db_host);
- } else {
- $host = $vchamilo->db_host;
- }
- // By default, empty password.
- $pass = '';
- $pgm = null;
- if (empty($port)) {
- $port = 3306;
- }
- // Password.
- if (!empty($vchamilo->db_password)) {
- $pass = "-p".escapeshellarg($vchamilo->db_password);
- }
- // Making the commands for each database.
- $cmds = array();
- // Windows environments are not supported for this plugin at this time
- //if ($CFG->ostype == 'WINDOWS') {
- // $cmd_main = "-h{$host} -P{$port} -u{$vchamilo->db_user} {$pass} {$vchamilo->main_database}";
- // $cmds[] = $cmd_main . ' > ' . $outputfilerad;
- //} else {
- $cmd_main = "-h{$host} -P{$port} -u{$vchamilo->db_user} {$pass} {$vchamilo->main_database}";
- $cmds[] = $cmd_main.' > '.escapeshellarg($outputfilerad);
- //}
- $mysqldumpcmd = self::getConfig('vchamilo', 'cmd_mysqldump', true);
- $pgm = !empty($mysqldumpcmd) ? stripslashes($mysqldumpcmd) : false;
- if (!$pgm) {
- $message = "Database dump command not available check here: ";
- $url = api_get_path(WEB_CODE_PATH).'admin/configure_plugin.php?name=vchamilo';
- $message .= Display::url($url, $url);
- Display::addFlash(Display::return_message($message));
- return false;
- } else {
- $phppgm = str_replace("\\", '/', $pgm);
- $phppgm = str_replace("\"", '', $phppgm);
- $pgm = str_replace('/', DIRECTORY_SEPARATOR, $pgm);
- if (!is_executable($phppgm)) {
- $message = "Database dump command $phppgm does not match any executable";
- Display::addFlash(Display::return_message($message));
- return false;
- }
- // executing all commands
- foreach ($cmds as $cmd) {
- // Final command.
- $cmd = $pgm.' '.$cmd;
- // Executes the SQL command.
- exec($cmd, $execoutput, $returnvalue);
- }
- }
- // End with success.
- return 1;
- }
- /**
- * read manifest values in vchamilo template.
- */
- public static function getVmanifest($version)
- {
- $templatewwwroot = '';
- // Define the $templatewwwroot content, found in manifest.php for this template
- $file = api_get_path(SYS_PATH).'/plugin/vchamilo/templates/'.$version.'/manifest.php';
- if (file_exists($file)) {
- include $file;
- $manifest = new stdClass();
- $manifest->templatewwwroot = $templatewwwroot;
- // $manifest->templatevdbprefix = $templatevdbprefix;
- // $manifest->coursefolder = $coursefolder;
- return $manifest;
- }
- return false;
- }
- /**
- * make a fake vchamilo that represents the current host
- */
- public static function makeThis()
- {
- global $_configuration;
- $thisPortal = new stdClass();
- $thisPortal->root_web = $_configuration['root_web'];
- $thisPortal->db_host = $_configuration['db_host'];
- $thisPortal->db_user = $_configuration['db_user'];
- $thisPortal->db_password = $_configuration['db_password'];
- $thisPortal->main_database = $_configuration['main_database'];
- return $thisPortal;
- }
- /**
- * Get available templates for defining a new virtual host.
- * @return array The available templates, or EMPTY array.
- */
- public static function getAvailableTemplates()
- {
- global $_configuration;
- $separator = DIRECTORY_SEPARATOR;
- $templatefoldername = 'plugin'.$separator.'vchamilo'.$separator.'templates';
- $tempDir = $_configuration['root_sys'].$templatefoldername;
- // Scans the templates.
- if (!is_dir($tempDir)) {
- mkdir($tempDir, 0777, true);
- }
- $finder = new \Symfony\Component\Finder\Finder();
- $dirs = $finder->in($tempDir)->depth('== 0');
- // Retrieves template(s) name(s). Should be hostnames.
- $templates = [];
- /*if ($addEmptyTemplate) {
- $templates = array('' => $plugin->get_lang('emptysite'));
- }*/
- $template = self::getConfig('vchamilo', 'default_template');
- if ($dirs) {
- /** @var Symfony\Component\Finder\SplFileInfo $dir */
- foreach ($dirs as $dir) {
- if (is_dir($dir->getPathname())) {
- // A template is considered when a dump.sql exists.
- if (file_exists($dir->getPathname().'/dump.sql')) {
- $templateName = $dir->getRelativePathname();
- if ($templateName == $template) {
- $templateName .= ' (default)';
- }
- $templates[$dir->getRelativePathname()] = $templateName;
- }
- }
- }
- }
- return $templates;
- }
- /**
- * this function set will map standard moodle API calls to chamilo
- * internal primitives. This avoids too many changes to do in imported
- * code
- */
- public static function getConfig($module, $key, $isplugin = true)
- {
- if ($isplugin) {
- $key = $module.'_'.$key;
- }
- $params = array('variable = ? AND subkey = ?' => [$key, $module]);
- $result = api_get_settings_params_simple($params);
- if ($result) {
- return $result['selected_value'];
- }
- return false;
- }
- /**
- * @param stdClass $vchamilo
- * @param string $template
- */
- public static function loadFilesFromTemplate($vchamilo, $template)
- {
- global $_configuration;
- // Make template directory (files and SQL).
- $separator = DIRECTORY_SEPARATOR;
- $templateDir = $_configuration['root_sys'].'plugin'.$separator.'vchamilo'.$separator.'templates'.$separator.$template;
- $vchamilo->virtual = true;
- $coursePath = self::getConfig('vchamilo', 'course_real_root').$separator.$vchamilo->slug;
- $homePath = self::getConfig('vchamilo', 'home_real_root').$separator.$vchamilo->slug;
- $archivePath = self::getConfig('vchamilo', 'archive_real_root').$separator.$vchamilo->slug;
- $uploadPath = self::getConfig('vchamilo', 'upload_real_root').$separator.$vchamilo->slug;
- // get the protocol free hostname
- Display::addFlash(
- Display::return_message("Copying {$templateDir}/data/courses => $coursePath")
- );
- copyDirTo(
- self::chopLastSlash($templateDir.'/data/courses'),
- self::chopLastSlash($coursePath),
- false
- );
- Display::addFlash(
- Display::return_message("Copying {$templateDir}/data/archive => $archivePath")
- );
- copyDirTo(
- self::chopLastSlash($templateDir.'/data/archive'),
- self::chopLastSlash($archivePath),
- false
- );
- Display::addFlash(
- Display::return_message("Copying {$templateDir}/data/home => $homePath")
- );
- copyDirTo(
- self::chopLastSlash($templateDir.'/data/home'),
- self::chopLastSlash($homePath),
- false
- );
- // Upload
- Display::addFlash(
- Display::return_message("Copying {$templateDir}/data/upload => $uploadPath")
- );
- copyDirTo(
- self::chopLastSlash($templateDir.'/data/upload/'),
- self::chopLastSlash($uploadPath),
- false
- );
- }
- /**
- * @param string $path
- *
- * @return mixed
- */
- public static function chopLastSlash($path)
- {
- return preg_replace('/\/$/', '', $path);
- }
- /**
- * @param string $str
- */
- public static function ctrace($str)
- {
- error_log($str);
- Display::addFlash(Display::return_message($str, 'normal', false));
- }
- /**
- * @param $file
- * @param $component
- * @param bool $return
- * @return string
- */
- public static function requireJs($file, $component, $return = false)
- {
- global $_configuration, $htmlHeadXtra;
- if (preg_match('/^local_/', $component)) {
- $component = str_replace('local_', '', $component);
- $path = 'local/';
- } else {
- $path = 'plugin/';
- }
- // Secure the postslashing of the roots.
- $root_web = $_configuration['root_web'].'/';
- $root_web = preg_replace('#//$#', '/', $root_web);
- $str = '<script type="text/javascript" src="'.$root_web.$path.$component.'/js/'.$file.'"></script>'."\n";
- if ($return === 'head') {
- $htmlHeadXtra[] = $str;
- }
- if ($return) {
- return $str;
- }
- echo $str;
- }
- /**
- * @param $file
- * @param $component
- * @param bool $return
- * @return string
- */
- public static function requireCss($file, $component, $return = false)
- {
- global $_configuration, $htmlHeadXtra;
- if (preg_match('/^local_/', $component)) {
- $component = str_replace('local_', '', $component);
- $path = 'local/';
- } else {
- $path = 'plugin/';
- }
- // Secure the postslashing of the roots.
- $root_web = $_configuration['root_web'].'/';
- $root_web = preg_replace('#//$#', '/', $root_web);
- $str = '<link rel="stylesheet" type="text/css" href="'.$root_web.$path.$component.'/'.$file.'.css" />'."\n";
- if ($return === 'head') {
- $htmlHeadXtra[] = $str;
- }
- if ($return) {
- return $str;
- }
- echo $str;
- }
- /**
- * @param string $url
- * @return string
- */
- public static function getSlugFromUrl($url)
- {
- $slugify = new Slugify();
- $urlInfo = parse_url($url);
- if (isset($urlInfo['host'])) {
- $path = $urlInfo['path'] != '/' ? '_'.$urlInfo['path'] : '';
- return $slugify->slugify($urlInfo['host'].$path);
- }
- return false;
- }
- /**
- * Check if all settings are complete
- */
- public static function checkSettings()
- {
- $enabled = self::getConfig('vchamilo', 'enable_virtualisation');
- if (empty($enabled)) {
- api_not_allowed(true, 'Plugin is not enabled');
- }
- global $virtualChamilo;
- if (!isset($virtualChamilo)) {
- api_not_allowed(
- true,
- 'You have to edit the configuration.php. Please check the readme file.'
- );
- }
- $coursePath = self::getConfig('vchamilo', 'course_real_root');
- $homePath = self::getConfig('vchamilo', 'home_real_root');
- $archivePath = self::getConfig('vchamilo', 'archive_real_root');
- $uploadPath = self::getConfig('vchamilo', 'upload_real_root');
- $cmdSql = self::getConfig('vchamilo', 'cmd_mysql');
- $cmdMySql = self::getConfig('vchamilo', 'cmd_mysqldump');
- if (empty($coursePath) || empty($homePath) || empty($uploadPath) || empty($archivePath) || empty($cmdSql) || empty($cmdMySql)) {
- api_not_allowed(true, 'You have to complete all plugin settings.');
- }
- $separator = DIRECTORY_SEPARATOR;
- $templatePath = api_get_path(SYS_PATH).'plugin'.$separator.'vchamilo'.$separator.'templates';
- $paths = [
- $coursePath,
- $homePath,
- $archivePath,
- $uploadPath,
- $templatePath
- ];
- foreach ($paths as $path) {
- $path = trim($path);
- if (is_dir($path)) {
- if (!is_writable($path)) {
- Display::addFlash(
- Display::return_message("Directory must have writable permissions: '$path'", 'warning')
- );
- };
- } else {
- Display::addFlash(
- Display::return_message("Directory doesn't exist: '$path'", 'warning')
- );
- }
- }
- }
- /**
- * @param object $instance
- * @return bool|\Doctrine\DBAL\Connection
- */
- public static function getConnectionFromInstance($instance, $getManager = false)
- {
- $dbParams = array(
- 'driver' => 'pdo_mysql',
- 'host' => $instance->db_host,
- 'user' => $instance->db_user,
- 'password' => $instance->db_password,
- //'dbname' => $instance->main_database,
- // Only relevant for pdo_sqlite, specifies the path to the SQLite database.
- //'path' => isset($_configuration['db_path']) ? $_configuration['db_path'] : '',
- // Only relevant for pdo_mysql, pdo_pgsql, and pdo_oci/oci8,
- //'port' => isset($_configuration['db_port']) ? $_configuration['db_port'] : '',
- );
- if (!empty($instance->main_database)) {
- $dbParams['dbname'] = $instance->main_database;
- }
- try {
- $database = new \Database();
- $manager = $database->connect(
- $dbParams,
- api_get_configuration_value('root_sys'),
- api_get_configuration_value('root_sys'),
- false,
- true
- );
- if ($getManager) {
- return $manager;
- }
- return $manager->getConnection();
- } catch (Exception $e) {
- error_log($e->getMessage());
- }
- return false;
- }
- /**
- * @param $data
- */
- public static function addInstance($data)
- {
- if (isset($data->what)) {
- unset($data->what);
- }
- if (isset($data->submitbutton)) {
- unset($data->submitbutton);
- }
- if (isset($data->id)) {
- unset($data->id);
- }
- if (isset($data->vid)) {
- unset($data->vid);
- }
- if (isset($data->testconnection)) {
- unset($data->testconnection);
- }
- if (isset($data->testdatapath)) {
- unset($data->testdatapath);
- }
- $registeronly = $data->registeronly;
- unset($data->registeronly);
- $data->lastcron = 0;
- $data->lastcrongap = 0;
- $data->croncount = 0;
- if (isset($data->template) && !empty($data->template)) {
- $template = $data->template;
- } else {
- $template = '';
- }
- $mainDatabase = api_get_configuration_value('main_database');
- if ($mainDatabase == $data->main_database) {
- Display::addFlash(
- Display::return_message('You cannot use the same database as the chamilo master', 'error')
- );
- return;
- }
- $databaseName = $data->main_database;
- $data->main_database = '';
- $connection = self::getConnectionFromInstance($data);
- $data->main_database = $databaseName;
- if (!$connection) {
- Display::addFlash(
- Display::return_message(
- 'Cannot connect to database with params: '.print_r($data, 1),
- 'error'
- )
- );
- return;
- }
- $data->root_web = api_add_trailing_slash($data->root_web);
- $data->archive_url = api_add_trailing_slash($data->archive_url);
- $data->home_url = api_add_trailing_slash($data->home_url);
- $data->upload_url = api_add_trailing_slash($data->upload_url);
- $data->course_url = api_add_trailing_slash($data->course_url);
- if (substr($data->root_web, 0, 4) != 'http') {
- $data->root_web = api_get_protocol().'://'.$data->root_web;
- }
- self::ctrace('Registering: '.$data->root_web);
- $tablename = Database::get_main_table('vchamilo');
- $sql = "SELECT * FROM $tablename
- WHERE root_web = '".Database::escape_string($data->root_web)."'";
- $result = Database::query($sql);
- if (Database::num_rows($result)) {
- Database::update($tablename, $data, ['root_web = ?' => $data->root_web]);
- $virtualInfo = Database::fetch_array($result);
- $slug = $virtualInfo['slug'];
- } else {
- $slug = $data->slug = self::getSlugFromUrl($data->root_web);
- if (empty($slug)) {
- Display::addFlash(
- Display::return_message('Cannot create slug from url: '.$data->root_web, 'error')
- );
- return;
- }
- Database::insert($tablename, (array) $data);
- }
- if ($registeronly) {
- // Stop it now.
- self::ctrace('Registering only. out.');
- self::redirect(api_get_path(WEB_PLUGIN_PATH).'vchamilo/views/manage.php');
- }
- // or we continue with physical creation
- self::createDirsFromSlug($slug);
- if (!$template) {
- // Create empty database for install
- self::ctrace("Creating database");
- self::createDatabase($data);
- } else {
- // Deploy template database
- self::ctrace("Creating databases from template '$template'");
- self::createDatabase($data);
- self::ctrace("Loading data template '$template'");
- self::loadDbTemplate($data, $template);
- self::ctrace("Coying files from template '$template'");
- self::loadFilesFromTemplate($data, $template);
- }
- // pluging in site name institution
- $settingstable = $data->main_database.'.settings_current';
- $accessurltable = $data->main_database.'.access_url';
- $sitename = Database::escape_string($data->sitename);
- $institution = Database::escape_string($data->institution);
- $sqls[] = "UPDATE {$settingstable} SET selected_value = '{$sitename}'
- WHERE variable = 'siteName' AND category = 'Platform' ";
- $sqls[] = "UPDATE {$settingstable} SET selected_value = '{$institution}'
- WHERE variable = 'institution' AND category = 'Platform' ";
- $sqls[] = "UPDATE {$accessurltable} SET url = '{$data->root_web}' WHERE id = '1' ";
- foreach ($sqls as $sql) {
- Database::query($sql);
- }
- self::ctrace("Finished");
- }
- /**
- * @param stdClass $data
- * @param string $fromVersion
- */
- public static function importInstance($data, $fromVersion)
- {
- if (isset($data->what)) {
- unset($data->what);
- }
- if (isset($data->submitbutton)) {
- unset($data->submitbutton);
- }
- if (isset($data->id)) {
- unset($data->id);
- }
- if (isset($data->vid)) {
- unset($data->vid);
- }
- if (isset($data->testconnection)) {
- unset($data->testconnection);
- }
- if (isset($data->testdatapath)) {
- unset($data->testdatapath);
- }
- $fromCoursePath = $data->course_path;
- $fromHomePath = $data->home_path;
- $fromUploadPath = $data->upload_path;
- unset($data->course_path);
- unset($data->home_path);
- unset($data->upload_path);
- $newDatabase = clone $data;
- $newDatabase->main_database = $newDatabase->import_to_main_database;
- $newDatabase->db_user = $newDatabase->import_to_db_user;
- $newDatabase->db_password = $newDatabase->import_to_db_password;
- $newDatabase->db_host = $newDatabase->import_to_db_host;
- unset($newDatabase->import_to_main_database);
- unset($newDatabase->import_to_db_user);
- unset($newDatabase->import_to_db_password);
- unset($newDatabase->import_to_db_host);
- unset($data->import_to_main_database);
- unset($data->import_to_db_user);
- unset($data->import_to_db_password);
- unset($data->import_to_db_host);
- $data->lastcron = 0;
- $data->lastcrongap = 0;
- $data->croncount = 0;
- $mainDatabase = api_get_configuration_value('main_database');
- if ($mainDatabase == $data->main_database) {
- Display::addFlash(
- Display::return_message('You cannot use the same database as the chamilo master', 'error')
- );
- return false;
- }
- self::ctrace('Registering: '.$data->root_web);
- $table = Database::get_main_table('vchamilo');
- $sql = "SELECT * FROM $table
- WHERE root_web = '".Database::escape_string($data->root_web)."'";
- $result = Database::query($sql);
- $id = null;
- if (Database::num_rows($result)) {
- Display::addFlash(
- Display::return_message('Instance was already added: '.$data->root_web, 'error')
- );
- return false;
- } else {
- /** @var EntityManager $em */
- $em = self::getConnectionFromInstance($data, true);
- if ($em) {
- $connection = $em->getConnection();
- $statement = $connection->query('SELECT * FROM settings_current');
- $settings = $statement->fetchAll();
- $settings = array_column(
- $settings,
- 'selected_value',
- 'variable'
- );
- $institution = $settings['Institution'];
- $siteName = $settings['siteName'];
- $newDatabase->sitename = $siteName;
- $newDatabase->institution = $institution;
- $slug = $newDatabase->slug = $data->slug = self::getSlugFromUrl($data->root_web);
- $id = Database::insert($table, (array) $newDatabase);
- }
- }
- if (!$id) {
- // Show data detail to help debug
- //var_dump($data);
- throw new Exception('New/Imported instance was not registered - edit '.__FILE__.' on line '.__LINE__.'to var_dump');
- }
- if (empty($slug)) {
- throw new Exception('Slug is empty');
- }
- self::createDirsFromSlug($slug);
- $databaseCreated = self::createDatabase($newDatabase);
- if (!$databaseCreated) {
- Display::addFlash(
- Display::return_message('Error while creating a DB', 'error')
- );
- return false;
- }
- $coursePath = self::getConfig('vchamilo', 'course_real_root').'/'.$slug;
- $homePath = self::getConfig('vchamilo', 'home_real_root').'/'.$slug;
- $uploadPath = self::getConfig('vchamilo', 'upload_real_root').'/'.$slug;
- $dumpFile = api_get_path(SYS_ARCHIVE_PATH).uniqid($data->main_database.'_dump_', true).'.sql';
- self::ctrace('Create backup from "'.$data->main_database.'" here: '.$dumpFile.' ');
- self::backupDatabase($data, $dumpFile);
- $sqlcmd = self::getDatabaseDumpCmd($newDatabase);
- $sqlcmd = str_replace('%DATABASE%', $newDatabase->main_database, $sqlcmd);
- $import = $sqlcmd.$dumpFile;
- // Execute the command.
- if (!defined('CLI_SCRIPT')) {
- putenv('LANG=en_US.utf-8');
- }
- // ensure utf8 is correctly handled by php exec()
- // @see http://stackoverflow.com/questions/10028925/call-a-program-via-shell-exec-with-utf-8-text-input
- $result = exec($import, $output, $return);
- self::ctrace('Restore backup here "'.$newDatabase->main_database.'" : <br />'.$import.' ');
- self::ctrace($result);
- $command = new \Chash\Command\Installation\UpgradeDatabaseCommand();
- // Creates the helper set
- $helperSet = \Doctrine\ORM\Tools\Console\ConsoleRunner::createHelperSet($em);
- $helpers = array(
- 'configuration' => new Chash\Helpers\ConfigurationHelper(),
- 'dialog' => new \Symfony\Component\Console\Helper\QuestionHelper(),
- );
- foreach ($helpers as $name => $helper) {
- $helperSet->set($helper, $name);
- }
- $command->setHelperSet($helperSet);
- $tmpFile = tmpfile();
- $outputStream = new \Symfony\Component\Console\Output\BufferedOutput($tmpFile);
- $arguments = array(
- 'from-version' => $fromVersion, // @todo change value
- 'to-version' => '1.11.x',
- 'host' => $newDatabase->db_host,
- 'username' => $newDatabase->db_user,
- 'password' => $newDatabase->db_password,
- 'db_name' => $newDatabase->main_database,
- 'root_sys' => api_get_configuration_value('root_sys')
- );
- $input = new ArrayInput($arguments);
- $command->run($input, $outputStream);
- error_log($outputStream->fetch());
- if (file_exists($dumpFile)) {
- unlink($dumpFile);
- }
- // Course
- self::ctrace("Copy from '$fromCoursePath' to backup '$coursePath' ");
- copyDirTo(
- self::chopLastSlash($fromCoursePath),
- self::chopLastSlash($coursePath),
- false
- );
- // Home
- self::ctrace("Copy from '$fromHomePath' to backup '$homePath' ");
- copyDirTo(
- self::chopLastSlash($fromHomePath),
- self::chopLastSlash($homePath),
- false
- );
- // Upload
- self::ctrace("Copy from '$fromUploadPath' to backup '$uploadPath' ");
- copyDirTo(
- self::chopLastSlash($fromUploadPath),
- self::chopLastSlash($uploadPath),
- false
- );
- self::ctrace("Finished");
- }
- /**
- * @param string $slug
- *
- * @return string
- */
- public static function createDirsFromSlug($slug)
- {
- // We continue with physical creation
- // Create course directory for operations.
- // this is very important here (DO NOT USE api_get_path() !!) because storage may be remotely located
- $absAlternateCourse = self::getConfig('vchamilo', 'course_real_root');
- $courseDir = $absAlternateCourse.'/'.$slug;
- if (!is_dir($courseDir)) {
- self::ctrace("Creating physical course dir in $courseDir");
- mkdir($courseDir, 0777, true);
- // initiate default index
- $indexFile = $courseDir.'/index.html';
- if ($indexFile) {
- file_put_contents($indexFile, self::getDefaultCourseIndexFragment());
- }
- $htaccessFile = $courseDir.'/.htaccess';
- if ($htaccessFile) {
- file_put_contents($htaccessFile, self::getHtaccessFragment($slug));
- }
- }
- $absAlternateHome = self::getConfig('vchamilo', 'home_real_root');
- $absAlternateArchive = self::getConfig('vchamilo', 'archive_real_root');
- $absAlternateUpload = self::getConfig('vchamilo', 'upload_real_root');
- // absalternatehome is a vchamilo config setting that tells where the
- // real physical storage for home pages are.
- $homeDir = $absAlternateHome.'/'.$slug;
- $archiveDir = $absAlternateArchive.'/'.$slug;
- $uploadDir = $absAlternateUpload.'/'.$slug;
- $dirs = [
- $homeDir,
- $archiveDir,
- $uploadDir
- ];
- foreach ($dirs as $dir) {
- self::ctrace("Making dir as $dir");
- if (!is_dir($dir)) {
- if (!mkdir($dir, 0777, true)) {
- self::ctrace("Error creating dir $dir \n");
- }
- }
- }
- }
- /**
- * @param $id
- * @return array|mixed
- */
- public static function getInstance($id)
- {
- $vhost = new stdClass();
- if ($id) {
- $id = (int) $id;
- $sql = "SELECT * FROM vchamilo WHERE id = $id";
- $result = Database::query($sql);
- $vhost = (object) Database::fetch_array($result, 'ASSOC');
- }
- return $vhost;
- }
- /**
- * @param stdClass $instance
- *
- * @return bool|string returns the original version of the app
- */
- public static function canBeUpgraded($instance)
- {
- $connection = self::getConnectionFromInstance($instance);
- if ($connection) {
- $sql = 'SELECT * FROM settings_current WHERE variable = "chamilo_database_version"';
- $statement = $connection->query($sql);
- $settings = $statement->fetchAll();
- $settings = array_column($settings, 'selected_value', 'variable');
- $version = $settings['chamilo_database_version'];
- $versionParts = explode('.', $version);
- $version = implode('.', [$versionParts[0], $versionParts[1], '0']);
- $currentVersion = api_get_setting('chamilo_database_version');
- $versionParts = explode('.', $currentVersion);
- $currentVersion = implode(
- '.',
- [$versionParts[0], $versionParts[1], '0']
- );
- if (version_compare($version, $currentVersion, '<')) {
- return $version;
- }
- }
- return false;
- }
- /**
- * @return array
- */
- public static function getEncryptList()
- {
- $encryptList = [
- 'bcrypt',
- 'sha1',
- 'md5',
- 'none'
- ];
- return array_combine($encryptList, $encryptList);
- }
- }
|