From f0b8a9bdbb165f24a581a4487c2d2da688c0c469 Mon Sep 17 00:00:00 2001 From: poonam_b Date: Tue, 16 Jul 2019 19:52:15 +0530 Subject: [PATCH] Issue #2: CSV Importer in cluster --- .../com_cluster/administrator/access.xml | 19 +- .../administrator/assets/js/import.js | 187 +++++++ .../administrator/controllers/defines.php | 23 + .../administrator/controllers/import.json.php | 467 ++++++++++++++++++ .../administrator/csv/clusterDetails.csv | 4 + .../views/clusters/tmpl/clusterimport.php | 49 ++ .../views/clusters/view.html.php | 27 +- 7 files changed, 764 insertions(+), 12 deletions(-) create mode 100644 src/components/com_cluster/administrator/assets/js/import.js create mode 100644 src/components/com_cluster/administrator/controllers/defines.php create mode 100644 src/components/com_cluster/administrator/controllers/import.json.php create mode 100644 src/components/com_cluster/administrator/csv/clusterDetails.csv create mode 100644 src/components/com_cluster/administrator/views/clusters/tmpl/clusterimport.php diff --git a/src/components/com_cluster/administrator/access.xml b/src/components/com_cluster/administrator/access.xml index aa5a2bb..c40ed89 100644 --- a/src/components/com_cluster/administrator/access.xml +++ b/src/components/com_cluster/administrator/access.xml @@ -1,12 +1,13 @@ -
- - - - - - - -
+
+ + + + + + + + +
diff --git a/src/components/com_cluster/administrator/assets/js/import.js b/src/components/com_cluster/administrator/assets/js/import.js new file mode 100644 index 0000000..492b005 --- /dev/null +++ b/src/components/com_cluster/administrator/assets/js/import.js @@ -0,0 +1,187 @@ +var clusterImport = { +validateImport: function (thisfile, loaderId) { + let obj = jQuery(thisfile); + obj.closest('.controls').children('.statusbar').remove(); + + let format_lesson_form = obj.closest('form'); + + /* Hide all alerts msgs */ + let status = new clusterImport.createStatusbar(obj); //Using this we can set progress. + + /* Get uploaded file object */ + let uploadedfile = obj[0].files[0]; + + if (uploadedfile == undefined) + { + alert(Joomla.Text._('COM_CLUSTER_PLEASE_SELECT_FILE')); + return false; + } + + /* Get uploaded file name */ + let filename = uploadedfile.name; + + /* pop out extension of file*/ + let ext = filename.split('.').pop().toLowerCase(); + let fileExt = filename.split('.').pop(); + + if (fileExt != 'csv') + { + let finalMsg = new Object(); + finalMsg['errormsg'] = Joomla.Text._('COM_CLUSTER_CSV_FILE_UPLOAD_ERROR'); + status.setMsg(finalMsg); + jQuery('.fileupload-preview').empty(); + jQuery('#csv-upload').val(''); + + window.parent.SqueezeBox.close(); + return false; + } + + /* IF evrything is correct so far, popolate file name in fileupload-preview*/ + + let file_name_container = jQuery(".fileupload-preview", obj.closest('.fileupload-new')); + + jQuery(file_name_container).show(); + jQuery(file_name_container).text(filename); + + clusterImport.startImporting(uploadedfile, status, thisfile, loaderId); + }, + createStatusbar: function (obj) { + this.statusbar = jQuery("
"); + this.filename = jQuery("
").appendTo(this.statusbar); + this.size = jQuery("
").appendTo(this.statusbar); + this.success = jQuery('
').appendTo(this.statusbar); + this.error = jQuery('
').appendTo(this.statusbar); + + obj.closest('.controls').append(this.statusbar); + + this.setFileNameSize = function(name, size) + { + var sizeStr = ""; + var sizeKB = size/1024; + if(parseInt(sizeKB) > 1024) + { + var sizeMB = sizeKB/1024; + sizeStr = sizeMB.toFixed(2)+" MB"; + } + else + { + sizeStr = sizeKB.toFixed(2)+" KB"; + } + + this.filename.html(name); + this.size.html(sizeStr); + } + this.setMsg = function(msg) + { + this.statusbar.show(); + + if(msg['errormsg']) + { + Joomla.renderMessages({"error":[msg.errormsg]}); + window.parent.SqueezeBox.close(); + } + + if(msg['successmsg']) + { + Joomla.renderMessages({"success":[msg.successmsg]}); + window.parent.SqueezeBox.close(); + } + + if(msg['messages']) + { + var message = jQuery('
').addClass('import-messages'); + this.success.removeClass('msg alert'); + + jQuery.each(msg['messages'], function(i, value){ + var key = Object.keys(value)[0]; + var curMessage = jQuery('
').addClass('alert alert-' + key).html(value[key]).get(0); + message.append(curMessage); + }); + + this.success.html(message); + this.success.show(); + } + } + }, + startImporting: function (file, status, thisfile, loaderId) { + var finalMsg = new Object(); + if(file == undefined) + { + finalMsg['errormsg'] = file_not_selected_error; + status.setMsg(finalMsg); + return false; + } + + var filename = file.name; + + if(window.FormData !== undefined) // for HTML5 browsers + { + var newfilename = clusterImport.sendFileToServer(file, status, thisfile, loaderId); + } + else + { + alert(Joomla.Text._('COM_CLUSTER_PLEASE_UPGRADE_YOUR_BROWSER')); + } + }, + + sendFileToServer: function (file, status, fileinputtag, loaderId) { + var formData = new FormData(); + formData.append( 'FileInput', file ); + + var returnvar = true; + var jqXHR = jQuery.ajax({ + xhr: function() { + var xhrobj = jQuery.ajaxSettings.xhr(); + if (xhrobj.upload) { + xhrobj.upload.addEventListener('progress', function(event) { + var percent = 0; + var position = event.loaded || event.position; + var total = event.total; + if (event.lengthComputable) { + percent = Math.ceil(position / total * 100); + } + }, false); + } + return xhrobj; + }, + url:Joomla.getOptions('system.paths').base +'/index.php?option=com_cluster&task=import.csvImport&format=json', + type:'POST', + data:formData, + mimeType:"multipart/form-data", + contentType: false, + dataType:'json', + cache: false, + processData:false, + success: function(response) + { + var output = response['OUTPUT']; + var result = output['flag']; + var finalMsg = new Object(); + + /* File uploading on local is done*/ + if (result == 0) + { + finalMsg['errormsg'] = output['msg']; + status.setMsg(finalMsg); + jQuery('.fileupload-preview').empty(); + } + else + { + finalMsg['successmsg'] = output['msg']; + status.setMsg(finalMsg); + } + + jQuery('#csv-upload').val(''); + }, + error: function(jqXHR, textStatus, errorThrown) + { + finalMsg['errormsg'] = jqXHR.responseText; + status.setMsg(finalMsg); + returnvar = false; + } + }); + + return returnvar; + status.setAbort(jqXHR); + } +} diff --git a/src/components/com_cluster/administrator/controllers/defines.php b/src/components/com_cluster/administrator/controllers/defines.php new file mode 100644 index 0000000..1e907f2 --- /dev/null +++ b/src/components/com_cluster/administrator/controllers/defines.php @@ -0,0 +1,23 @@ + + * @copyright Copyright (C) 2009 - 2019 Techjoomla. All rights reserved. + * @license GNU General Public License version 2 or later; see LICENSE.txt + */ + +// No direct access +defined('_JEXEC') or die; + +define('COM_CLUSTER_GROUP_TITLE', "cluster_name"); +define('COM_CLUSTER_GROUP_EMAIL', "email"); +define('COM_CLUSTER_CLIENT', "client"); +define('COM_CLUSTER_CLIENT_ID', "client_id"); +define('COM_CLUSTER_GROUP_ID', "cluster_id"); +define('COM_CLUSTER_CLUSTER_STATE', "state_cluster"); +define('COM_CLUSTER_CLUSTERUSERS_STATE', "state_clusterusers"); +define('COM_CLUSTER_CLUSTER_CREATED_BY', "created_by_cluster"); +define('COM_CLUSTER_CLUSTERUSERS_CREATED_BY', "created_by_clusterusers"); +define('COM_CLUSTER_CLUSTER_MODIFIED_BY', "modified_by_cluster"); +define('COM_CLUSTER_CLUSTERUSERS_MODIFIED_BY', "modified_by_clusterusers"); diff --git a/src/components/com_cluster/administrator/controllers/import.json.php b/src/components/com_cluster/administrator/controllers/import.json.php new file mode 100644 index 0000000..94ee600 --- /dev/null +++ b/src/components/com_cluster/administrator/controllers/import.json.php @@ -0,0 +1,467 @@ + + * @copyright Copyright (C) 2009 - 2019 Techjoomla. All rights reserved. + * @license GNU General Public License version 2 or later; see LICENSE.txt + */ + +defined('_JEXEC') or die; + +use Joomla\CMS\MVC\Model\BaseDatabaseModel; +use Joomla\CMS\MVC\Controller\BaseController; +use Joomla\CMS\Component\ComponentHelper; +use Joomla\CMS\Factory; +use Joomla\CMS\Filesystem\File; +use Joomla\CMS\Log\Log; +use Joomla\CMS\Router\Route; +use Joomla\CMS\Session\Session; +use Joomla\CMS\Uri\Uri; +use Joomla\CMS\Table\Table; +use Joomla\CMS\Language\Text; +use Joomla\CMS\User\User; + +jimport('joomla.log.logger.formattedtext'); + +JLoader::import('components.com_users.models.users', JPATH_ADMINISTRATOR); +JLoader::import('components.com_cluster.controllers.defines', JPATH_ADMINISTRATOR); + +/** + * The cluster import controller + * + * @since 1.0.0 + */ +class ClusterControllerImport extends BaseController +{ + /** + * The main function triggered after on format upload + * + * @return object of result and message + * + * @since 1.0.0 + * */ + public function csvImport() + { + $user = Factory::getUser(); + $canImportdata = $user->authorise('core.import', 'com_cluster'); + + $app = Factory::getApplication(); + + if (!$canImportdata) + { + echo new JResponseJson(Text::_('COM_CLUSTER_NOT_AUTHORISED'), false); + jexit(); + } + + header('Cache-Control: no-cache, must-revalidate'); + header('Content-type: application/json'); + + jimport('joomla.log.log'); + $logFileName = 'com_cluster.import_' . Factory::getDate() . '.log'; + + Log::addLogger(array('text_file' => $logFileName), Log::ALL, array('com_cluster')); + + // Set log file name to session + $session = Factory::getSession(); + $session->set('com_cluster.import', $logFileName); + + $files = $app->input->files; + $file_to_upload = $files->get('FileInput', '', 'ARRAY'); + + /* Validate the uploaded file*/ + + $validate_result = $this->validateUpload($file_to_upload); + + $ret = array(); + + if ($validate_result['res'] != 1) + { + $ret['OUTPUT']['flag'] = $validate_result['res']; + $ret['OUTPUT']['msg'] = $validate_result['msg']; + echo new JResponseJson($ret); + jexit(); + } + + $return = 1; + $msg = ''; + + $file_attached = $file_to_upload['tmp_name']; + + /* Save csv content to question table */ + + $result = array(); + + $result = $this->saveCsvContent($file_to_upload); + + $filename = $file_to_upload['name']; + + $ret['OUTPUT']['flag'] = $result['returns']; + $ret['OUTPUT']['msg'] = $result['msg']; + + echo new JResponseJson($ret); + jexit(); + } + + /** + * The function to validate the uploaded format file + * + * @param MIXED $file_to_upload file object + * + * @return object of result and message + * + * @since 1.0.0 + * */ + public function validateUpload($file_to_upload) + { + $app = Factory::getApplication()->input; + $clusterParams = ComponentHelper::getParams('com_cluster'); + $filename = $file_to_upload['name']; + + $output = array(); + $return = 1; + $msg = ''; + + if ($file_to_upload["error"] == UPLOAD_ERR_OK) + { + // Check if file size in within the uploading limit of site*/ + + if ( $app->server->getString('CONTENT_LENGTH', '') > ($clusterParams->get('file_size', 0) * 1024 * 1024) + || $app->server->getString('CONTENT_LENGTH', '') > (int) (ini_get('upload_max_filesize')) * 1024 * 1024 + || $app->server->getString('CONTENT_LENGTH', '') > (int) (ini_get('post_max_size')) * 1024 * 1024 + || (($app->server->getString('CONTENT_LENGTH', '') > (int) (ini_get('memory_limit')) * 1024 * 1024) && ((int) (ini_get('memory_limit')) != -1))) + { + $return = 0; + $msg = Text::sprintf('COM_CLUSTER_UPLOAD_SIZE_ERROR', $clusterParams->get('import_size', 10, 'INT') . ' MB'); + } + + /* Check for the type/extensiom of the file*/ + if ($return == 1) + { + $fileext = File::getExt($filename); + + $valid_extensions_arr = array('csv'); + + if (!in_array($fileext, $valid_extensions_arr)) + { + $msg = Text::_('COM_CLUSTER_VALID_DOCUMENT_UPLOAD'); + $return = 0; + } + } + } + else + { + $return = 0; + $msg = Text::_('COM_CLUSTER_ERROR_UPLOADINGFILE', $filename); + } + + $output['res'] = $return; + $output['msg'] = $msg; + + return $output; + } + + /** + * Save csv content table from csv + * + * @param MIXED $file_to_upload file object + * + * @return ARRAY + * + * @since 1.0.0 + */ + public function saveCsvContent($file_to_upload) + { + $csvFileName = $file_to_upload['name']; + + $output = array(); + $success = $failed = $missingDetails = 0; + $logLink = ''; + $key = ''; + + if (($handle = fopen($file_to_upload['tmp_name'], 'r')) !== false) + { + $rowNum = ''; + $lineno = 0; + $headers = array(); + + while (($data = fgetcsv($handle)) !== false) + { + if ($rowNum == 0) + { + $lineno++; + + // Parsing the CSV header + + foreach ($data as $d) + { + $headers[] = $d; + } + } + else + { + // Parsing the data rows + $rowData = array(); + + foreach ($data as $d) + { + $rowData[] = $d; + } + + $masterData[] = array_combine($headers, $rowData); + } + + $rowNum++; + } + + fclose($handle); + + if (empty($masterData)) + { + $output['returns'] = 0; + $output['msg'] = Text::_('COM_CLUSTER_IMPORT_BLANK_FILE'); + } + + $column = array(COM_CLUSTER_GROUP_TITLE, COM_CLUSTER_GROUP_EMAIL, COM_CLUSTER_CLIENT_ID, COM_CLUSTER_CLUSTERUSERS_STATE, COM_CLUSTER_GROUP_ID); + + // Error and Info logs + $logFilepath = JRoute::_('index.php?option=com_cluster&view=clusters&task=import.downloadLog&prefix=com_cluster.import&format=json'); + + $logLink = '' . Text::_('COM_CLUSTER_LOG_FILE') . ''; + $logLink = Text::sprintf('COM_CLUSTER_LOG_FILE_PATH', $logLink); + + Table::addIncludePath(JPATH_ADMINISTRATOR . '/components/com_cluster/tables'); + + if (!empty($masterData)) + { + $user = Factory::getUser(); + $totalRecords = count($masterData); + + foreach ($masterData as $eachType) + { + $data = array(); + + foreach ($eachType as $key => $value) + { + switch ($key) + { + case COM_CLUSTER_GROUP_TITLE : + + if (!empty ($value)) + { + $data['name'] = $value; + } + + break; + + case COM_CLUSTER_GROUP_EMAIL : + + if (!empty ($value)) + { + $userModel = BaseDatabaseModel::getInstance('Users', 'UsersModel', array('ignore_request' => true)); + $userModel->setState('filter.search', $value); + $userData = $userModel->getItems(); + + $usersdata['user_id'] = $userData[0]->id; + } + + break; + + case COM_CLUSTER_CLIENT : + + if (!empty ($value)) + { + $data['client'] = $value; + } + + break; + + case COM_CLUSTER_CLIENT_ID : + + if (!empty ($value)) + { + $data['client_id'] = $value; + } + + break; + + case COM_CLUSTER_GROUP_ID : + + if (!empty ($value)) + { + $usersdata['cluster_id'] = $value; + } + + break; + + case COM_CLUSTER_CLUSTER_STATE : + + if (!empty ($value)) + { + $data['state'] = $value; + } + + break; + + case COM_CLUSTER_CLUSTERUSERS_STATE : + + if (!empty ($value)) + { + $usersdata['state'] = $value; + } + + break; + + case COM_CLUSTER_CLUSTER_CREATED_BY : + + if (!empty ($value)) + { + $data['created_by'] = $value; + } + + break; + + case COM_CLUSTER_CLUSTERUSERS_CREATED_BY : + + if (!empty ($value)) + { + $usersdata['created_by'] = $value; + } + + break; + + case COM_CLUSTER_CLUSTER_MODIFIED_BY : + + if (!empty ($value)) + { + $data['modified_by'] = $value; + } + + break; + + case COM_CLUSTER_CLUSTERUSERS_MODIFIED_BY : + + if (!empty ($value)) + { + $usersdata['modified_by'] = $value; + } + + break; + } + } + + if (!in_array($key, $column)) + { + $output['returns'] = 0; + $output['msg'] = Text::sprintf('COM_CLUSTER_INCORRECT_COLUMN_CSV_ERROR', $key); + + return $output; + } + + // If csv data missing + if (empty($data['name']) || (empty($usersdata['cluster_id']) && empty($data['client']) && empty($data['email']) && empty($data['client_id']))) + { + $missingDetails++; + } + + // Save cluster + $clusterModel = BaseDatabaseModel::getInstance('Cluster', 'ClusterModel', array('ignore_request' => true)); + $clusterModel->save($data); + + if (empty($usersdata['cluster_id'])) + { + $clusterTable = Table::getInstance('Clusters', 'ClusterTable'); + $clusterTable->load(array('name' => $data['name'])); + + $usersdata['cluster_id'] = $clusterTable->id; + } + + $clusterUserModel = BaseDatabaseModel::getInstance('ClusterUser', 'ClusterModel', array('ignore_request' => true)); + + if ($clusterUserModel->save($usersdata)) + { + $success ++; + $msg = 'COM_CLUSTER_NEW_USER_ADDED'; + Log::add(Text::sprintf($msg, $usersdata['user_id']), Log::INFO, 'com_cluster'); + } + else + { + $failed ++; + $msg = 'COM_CLUSTER_ALREADY_EXIST'; + Log::add(Text::sprintf($msg, $usersdata['user_id']), Log::ERROR, 'com_cluster'); + } + } + } + } + else + { + $output['returns'] = 0; + $output['msg'] = Text::sprintf('COM_CLUSTER_FILE_READ_ERROR'); + } + + if ($missingDetails > 0) + { + $message = ($missingDetails == 1) ? 'COM_CLUSTER_CSV_MANDATORY_FIELDS_ONE' : 'COM_CLUSTER_CSV_MANDATORY_FIELDS_MULTIPLE'; + array_push($messages, array('error' => Text::sprintf($message, $missingDetails))); + + $output['msg'] = $messages; + } + + if ($success > 0 || $failed > 0) + { + $output['returns'] = 1; + $output['msg'] = Text::sprintf('COM_CLUSTER_RECORDS_IMPORT_SUCCESSFULLY', $success) . + Text::sprintf('COM_CLUSTER_RECORDS_IMPORT_FAILED', $failed) . ' ' . $logLink; + } + + return $output; + } + + /** + * Download log on import users. + * + * @return mixed + * + * @since 1.0.0 + */ + public function downloadLog() + { + $user = Factory::getUser(); + $canImportdata = $user->authorise('core.import', 'com_cluster'); + + $app = Factory::getApplication(); + + if (!$canImportdata) + { + $app->enqueueMessage(Text::_('JERROR_ALERTNOAUTHOR'), 'error'); + $app->redirect(Route::_(Uri::base())); + + return false; + } + + jimport('joomla.filesystem.file'); + $prefix = $app->input->get('prefix', '', 'string'); + + $session = Factory::getSession(); + $config = Factory::getConfig(); + + $filename = $session->get($prefix); + + $file = $config->get('log_path') . '/' . $filename; + + if (!empty($filename) && File::exists($file)) + { + header('Content-Description: File Transfer'); + header('Content-Type: application/octet-stream'); + header('Content-Disposition: attachment; filename="' . basename($file) . '"'); + header('Expires: 0'); + header('Cache-Control: must-revalidate'); + header('Pragma: public'); + readfile($file); + } + else + { + $app->redirect(Route::_(Uri::base())); + } + } +} diff --git a/src/components/com_cluster/administrator/csv/clusterDetails.csv b/src/components/com_cluster/administrator/csv/clusterDetails.csv new file mode 100644 index 0000000..2cbbe73 --- /dev/null +++ b/src/components/com_cluster/administrator/csv/clusterDetails.csv @@ -0,0 +1,4 @@ +cluster_name,email,client,client_id,cluster_id,state_cluster,state_clusterusers,created_by_cluster,created_by_clusterusers,modified_by_cluster,modified_by_clusterusers +Avengers,poonam_b+footertest+@mailinator.com,com_tjlms.course ,1,291,1,1,3465,3465,3465,3465 +Dominators,poonam_b+footertest+@mailinator.com,com_tjlms.course ,2,292,1,1,3465,3465,3465,3465 +Soldiers,poonam_b+footertest+@mailinator.com,com_tjlms.course ,3,293,1,1,3465,3465,3465,3465 diff --git a/src/components/com_cluster/administrator/views/clusters/tmpl/clusterimport.php b/src/components/com_cluster/administrator/views/clusters/tmpl/clusterimport.php new file mode 100644 index 0000000..58a683f --- /dev/null +++ b/src/components/com_cluster/administrator/views/clusters/tmpl/clusterimport.php @@ -0,0 +1,49 @@ + +* @copyright Copyright (C) 2012-2019 Techjoomla. All rights reserved. +* @license GNU General Public License version 2 or later; see LICENSE.txt +*/ +// No direct access +defined('_JEXEC') or die; +use Joomla\CMS\Factory; + +$jinput = Factory::getApplication()->input; +$view = $jinput->get('view', '', 'string'); + +if ($view == 'clusters') +{ + $filepath = JUri::root() . 'administrator/components/com_cluster/csv/clusterDetails.csv'; +} +?> +
+
+
+
+
+
+
+ + + +
+ + + + +
+ +
+
+
+
+
+ + +
diff --git a/src/components/com_cluster/administrator/views/clusters/view.html.php b/src/components/com_cluster/administrator/views/clusters/view.html.php index d31496f..cefaa98 100644 --- a/src/components/com_cluster/administrator/views/clusters/view.html.php +++ b/src/components/com_cluster/administrator/views/clusters/view.html.php @@ -13,6 +13,10 @@ use Joomla\CMS\Factory; use Joomla\CMS\MVC\View\HtmlView; use Joomla\CMS\Object\CMSObject; +use Joomla\CMS\HTML\HTMLHelper; +use Joomla\CMS\Toolbar\ToolbarHelper; + +HTMLHelper::_('behavior.modal', 'a.modal'); /** * Clusters view @@ -139,10 +143,27 @@ protected function addToolbar() JToolbarHelper::divider(); } - if ($canDo->get('core.admin') || $canDo->get('core.options')) + if ($canDo->get('core.admin') || $canDo->get('core.options') || $canDo->get('core.import')) { - JToolbarHelper::preferences('com_cluster'); - JToolbarHelper::divider(); + $document = JFactory::getDocument(); + $document->addScript(JUri::root() . 'administrator/components/com_cluster/assets/js/import.min.js'); + + ToolbarHelper::preferences('com_cluster'); + + ToolbarHelper::divider(); + + $toolbar = JToolbar::getInstance('toolbar'); + + $url = JRoute::_('index.php?option=com_cluster&view=clusters&tmpl=component&layout=clusterimport'); + $button = ' + ' . JText::_('COM_CLUSTER_CSV_IMPORT') . ''; + $toolbar->appendButton('Link', 'export', $button); + + $filepath = JUri::root() . 'administrator/components/com_cluster/csv/clusterDetails.csv'; + + $buttonDownload = ' + ' . JText::_('COM_CLUSTER_CSV_SAMPLE') . ''; + $toolbar->appendButton('Link', 'export', $buttonDownload); } }