Загрузка файлов в Яндекс Облако на Joomla Cobalt
Добрый день друзья! Сегодня я хочу рассказать о доработке компонента Cobalt(конструктор контента) для Joomla, а именно добавление возможности загрузки файлов в Яндекс Облако(Yandex Object Storage) и удаления при необходимости. Загрузка файлов в облако позволяет существенно сэкономить место на хостинге и как следствие немало денежных средств, ведь место в облаке стоит гораздо дешевле. Для загрузки файлов используется JS скрипт MooUpload. При загрузке файлов обеспечивается возможность выбора типа хранилища в облаке: холодное(Cold) или горячее(Standart). Кейс также интересен тем, что по условиям задачи, требуется минимальное внесение изменений в файлы компонента Cobalt, это необходимо для дальнейшего его обновления без риска потери этих изменений.
Настройка полей в Cobalt
Для настройки, переходим в соответствующий раздел: Админпанель -> Настройки -> Кобальт -> Типы статей. Выбираем нужный тип статьи, где мы планируем добавить возможность использования функции загрузки файлов в Облако. Создаем четыре поля, два поля нам будет необходимы для загрузки изображений, другие два для всех остальных файлов.
Типы создаваемых полей:
В полях Textarea, будут храниться пути загруженных файлов(массив в формате JSON, преобразованный в строку) на Yandex Storage. При этом сами поля будут скрыты от пользователей. Рядом с ними, с помощью JS, будут выведена форма для загрузки файлов(загрузчик MooUpload).
Для загрузки изображений и выбора типа хранилища.
1) Название — Загрузка изображений (jpeg, jpg, png, gif). Тип поля Simple Form Eliments -> Textarea. Класс стилей CSS — yandexImg.
2) Название — Тип хранилища, для изображений. Тип поля Simple Form Eliments -> Select. Класс стилей CSS — yandexTypeImg. Список значений — Холодный. Первый элемент выбора — Горячий.
Для загрузки файлов и выбора типа хранилища.
3) Название — Загрузка файлов (zip, rar). Тип поля Simple Form Eliments -> Textarea. Класс стилей CSS — yandexFile.
4) Название — Тип хранилища, для файлов. Тип поля Simple Form Eliments -> Select. Класс стилей CSS — yandexTypeFile. Список значений — Холодный. Первый элемент выбора — Горячий.
Доработка MooUpload
В JS скрипт загрузчика MooUpload version: 1.1, который идет в стандартной поставке Joomla Cobalt, вносим небольшие изменения.
Расположение ./media/mint/js/mooupload/MooUpload.js
Метод populateFileList, после строки номер 544(может отличаться в зависимости от версии скрипта), добавляем код с обработкой нового статуса ответа со значением «3» при удалении файла(так как MooUpload будет слать загружаемые файлы на наш новый серверный скрипт на PHP).
Код который нужно добавить:
if(json.success == 3) { $(json.id).slide('out'); setTimeout(function() { $(json.id).destroy(); }, 500); this.filelist[j].checked = false; this.fireEvent('onFileDelete', ['0', file.filename]); } |
Листинг всего метода:
populateFileList: function(maincontainer) { var subcontainer = document.id(maincontainer.get('id') + '_listView').getElement('ul'); var maincontainer_id = maincontainer.get('id'); var options = this.options; var size = 0, key; for(key in this.options.files) { if(this.options.files.hasOwnProperty(key)) size++; } if(!size) { return; } if(this.options.maxfiles) { this.filesCount = size; } for(var i = 0, file = null; file = this.options.files[i]; i++) { this.filelist[i] = { id: String.uniqueID(), checked: true, name: file.filename, type: file.ext, size: file.size, uploaded: true, uploading: false, error: false }; this.filenum++; var liid = file.filename.toLowerCase(); liid = liid.replace('.' + file.ext.toLowerCase(), ''); var elementcontainer = new Element('li', { 'class': 'item mooupload_readonly', id: liid }).inject(subcontainer); var hiddenInput = new Element('input', { 'type': 'hidden', 'name': this.options.formname, 'value': file.filename }).inject(elementcontainer); var optionsel = new Element('div', { 'class': 'optionsel' }).inject(elementcontainer); var f = file; if(this.options.canDelete) { var optionremove = new Element('a', { 'class': 'remove' }).inject(optionsel); var func = function(file, j) { if(!confirm(this.options.texts.sure)) { return; } $$('#' + file.filename.replace('.' + file.ext, '') + ' div.result').set('html', this.options.texts.deleting).setStyle('background', 'url( "' + this.options.url_root + '/media/mint/js/mooupload/imgs/load_bg_red.gif")').setStyle('color', 'maroon'); var req = new Request.JSON({ url: this.options.action_remove_file, method: 'post', autoCancel: true, data: { filename: file.filename }, onComplete: function(json) { //console.log(json); if(json.success == 1) { $(json.id).slide('out'); setTimeout(function() { $(json.id).destroy(); }, 500); this.filelist[j].checked = false; } if(json.success == 0) { this.fireEvent('onFileDelete', ['1016', file.filename]); } if(json.success == 2) { this.fireEvent('onFileDelete', ['1017', file.filename]); } if(json.success == 3) { $(json.id).slide('out'); setTimeout(function() { $(json.id).destroy(); }, 500); this.filelist[j].checked = false; this.fireEvent('onFileDelete', ['0', file.filename]); } }.bind(this) }).send(); }; optionremove.addEvent('click', func.pass([file, i], this)); } var title = file.realname; if(this.options.allowEditTitle && file.title) { title = file.title; } var css_class = 'filename'; if(this.options.allowEditTitle) { css_class = 'filename filenameedit'; } var filename_div = new Element('div', { 'rel': f.id, 'id': maincontainer.get('id') + '_file' + i, 'class': css_class, 'title': this.options.texts.edit_title, html: title, styles: { width: '55%' // width: this.namewidth + 'px', } }).inject(elementcontainer); if(this.options.allowAddDescr) { this.addDescriptionInterface(elementcontainer, f.id, f.description); } if(this.options.allowEditTitle) { this.addTitleInterface(filename_div, filename_div.get('rel')); } new Element('div', { 'class': 'filesize', html: this.formatSize(file.size) }).inject(elementcontainer); new Element('div', { id: maincontainer_id + '_file_' + i, 'class': 'result', 'html': this.options.texts.uploaded }).inject(elementcontainer); elementcontainer.highlight('#FFF', '#E3E3E3'); } }, |
Вывод формы загрузки файлов на страницу сайта
Для того, чтобы модифицировать ранее добавленные поля Textarea и вывести на странице, для пользователя, функционал загрузки файлов с использованием MooUpload, в шаблон движка Joomla, в блок HEAD добавляем строку:
<script src="/templates/sitetheme/js/yandex.storage.js"></script> |
Содержание файла yandex.storage.js:
jQuery(document).ready(function($) { $('.yandexFile .controls').hide(); // текстовое поле, сюда сохраняется результат загрузки $('.yandexFile .controls').after('<div class="controls" id="yandexFileMooUpload"></div><p class="controls"><span class="small">Разрешенные типы файлов: zip, rar</p></p>') var filesdata = $('.yandexFile textarea').val(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; var Newfilesdata = []; $(filesdata).each(function(i, v) { var obj = {}; obj.id = i; obj.filename = v.upload_name; obj.size = v.size; obj.realname = v.name; obj.fullpath = 'files/' + v.upload_name; obj.title = null; obj.description =null; obj.width = 0; obj.height = 0; var ext = v.upload_name.split('.'); obj.ext = ext[ext.length - 1]; Newfilesdata.push(obj); }); if($('#yandexFileMooUpload').length > 0) { var myUpload = new MooUpload('yandexFileMooUpload', { action: '/storage/yandex.php?action=upload&type=files', action_remove_file: '/storage/yandex.php?action=delete', method: 'auto', tempname: 'ya.file', files: Newfilesdata, formname:'jform[fields][ya.file][]', autostart:1, field_id:16, record_id:0, maxfilesize: 1048576, exts: ['zip','rar'], maxfiles: 0, canDelete: 1, allowEditTitle: 0, allowAddDescr: 0, url_root: '', flash: { movie: '/media/mint/js//Moo.Uploader.swf' }, texts: { error : 'Ошибка', file : 'Файл', filesize : 'Размер файла', filetype : 'Тип файла', nohtml5 : 'Нет поддержки загрузки файлов HTML5!', noflash : 'Пожалуйста, установите Flash 8.5 или более новой версии (У вас отключен FlashBlock или AdBlock?)', sel : 'Действие', selectfile : 'Добавить файлы', status : 'Статус', startupload: 'Начать загрузку', uploaded : 'Загружено', sure : 'Вы уверены, что хотите удалить?', edit_descr : 'Редактировать описание', edit_title : 'Редактировать заголовок', deleting : 'Удаление' }, onFileDelete: function( error, filename){ // console.log(filename); var filesdata = $('.yandexFile textarea').val(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; var Newfilesdata = []; $(filesdata).each(function(i, v) { if(filename != v.upload_name) Newfilesdata.push(v); }); filesdata = JSON.stringify(Newfilesdata); $('.yandexFile textarea').val(filesdata); }, onFileUpload: function(error, filename){ var filesdata = $('.yandexFile textarea').val(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; filesdata.push(filename); filesdata = JSON.stringify(filesdata); $('.yandexFile textarea').val(filesdata); } }); } if($('.tab-content dd.yandexFile').length > 0) { var filesdata = $('.tab-content dd.yandexFile').text().trim(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; if(filesdata.length > 0) { $('.tab-content dd.yandexFile').html('<ul class="unstyled"></ul>'); $(filesdata).each(function(i, v) { $('.tab-content .yandexFile ul').append('<li><a target="_blank" href="https://storage.yandexcloud.net/farming-ls19/files/' + decodeURI(v.upload_name) + '" rel="noopener noreferrer">' + decodeURI(v.name) + '</a> <small>Размер: <span style="color:green">' + v.size + ' Bytes</span></small></li>'); }); } } }); jQuery(document).ready(function($) { $('.yandexImg .controls').hide(); // текстовое поле, сюда сохраняется результат загрузки $('.yandexImg .controls').after('<div class="controls" id="yandexImgMooUpload"></div><p class="controls"><span class="small">Разрешенные типы файлов: jpg, png, gif</p></p>') var filesdata = $('.yandexImg textarea').val(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; var Newfilesdata = []; $(filesdata).each(function(i, v) { var obj = {}; obj.id = i; obj.filename = v.upload_name; obj.size = v.size; obj.realname = v.name; obj.fullpath = 'files/' + v.upload_name; obj.title = null; obj.description =null; obj.width = 0; obj.height = 0; var ext = v.upload_name.split('.'); obj.ext = ext[ext.length - 1]; Newfilesdata.push(obj); }); if($('#yandexImgMooUpload').length > 0) { var myUpload = new MooUpload('yandexImgMooUpload', { action: '/storage/yandex.php?action=upload&type=images', action_remove_file: '/storage/yandex.php?action=delete', method: 'auto', tempname: 'ya.image', files: Newfilesdata, formname:'jform[fields][ya.images][]', autostart:1, field_id:16, record_id:0, maxfilesize: 1048576, exts: ['jpg','jpeg','png','gif'], maxfiles: 0, canDelete: 1, allowEditTitle: 0, allowAddDescr: 0, url_root: '', flash: { movie: '/media/mint/js//Moo.Uploader.swf' }, texts: { error : 'Ошибка', file : 'Файл', filesize : 'Размер файла', filetype : 'Тип файла', nohtml5 : 'Нет поддержки загрузки файлов HTML5!', noflash : 'Пожалуйста, установите Flash 8.5 или более новой версии (У вас отключен FlashBlock или AdBlock?)', sel : 'Действие', selectfile : 'Добавить файлы', status : 'Статус', startupload: 'Начать загрузку', uploaded : 'Загружено', sure : 'Вы уверены, что хотите удалить?', edit_descr : 'Редактировать описание', edit_title : 'Редактировать заголовок', deleting : 'Удаление' }, onFileDelete: function( error, filename){ // console.log(filename); var filesdata = $('.yandexImg textarea').val(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; var Newfilesdata = []; $(filesdata).each(function(i, v) { if(filename != v.upload_name) Newfilesdata.push(v); }); filesdata = JSON.stringify(Newfilesdata); $('.yandexImg textarea').val(filesdata); }, onFileUpload: function(error, filename){ var filesdata = $('.yandexImg textarea').val(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; filesdata.push(filename); filesdata = JSON.stringify(filesdata); $('.yandexImg textarea').val(filesdata); } }); } if($('.tab-content dd.yandexImg').length > 0) { var filesdata = $('.tab-content dd.yandexImg').text().trim(); if(filesdata) filesdata = JSON.parse(filesdata); else filesdata = []; if(filesdata.length > 0) { //$('.tab-content dd.yandexImg').html('<ul class="unstyled"></ul>'); $('.tab-content dd.yandexImg').html(''); $(filesdata).each(function(i, v) { $('.tab-content dd.yandexImg').append('<a target="_blank" href="https://storage.yandexcloud.net/farming-ls19/images/' + decodeURI(v.upload_name) + '" rel="noopener noreferrer"><img src="https://storage.yandexcloud.net/farming-ls19/images/' + decodeURI(v.upload_name) + '" width="150" alt="' + decodeURI(v.name) + ' Размер: ' + v.size + ' Bytes" /></a>'); }); } } }); |
Обработчик файлов на PHP
Для загрузки файлов с использованием MooUpload, нужна своя серверная часть, новый скрипт на PHP для приема загружаемых файлов. При этом загрузка файлов происходит частями, скрипт MooUpload делит загружаемый файл на куски и отправляет данные частями. По этому PHP скрипт должен поддерживать такую загрузку, прием данных частями и их последующую склейку. По итогу загрузки скрипт на PHP будет загружать файл целиком в Яндекс Облако(Yandex Storage). Также PHP скрипт должен поддерживать запросы от MooUpload на удаление файлов с Яндекс Облака.
Пример структуры PHP скрипта ./stotage/yandex.php (из листинга убран код работы с входящими переменными и базой данных):
use Aws\S3\S3Client; // работа с Yandex Storage if($_GET['action'] == 'upload') { // Загрузка файлов в Облако // заливаем на Storage $sharedConfig = [ 'credentials' => [ 'key' => $registry['ya.store']['key'], 'secret' => $registry['ya.store']['secret'], ], 'region' => 'us-east-1', 'endpoint' => 'https://storage.yandexcloud.net', 'version' => 'latest', ]; $sdk = new Aws\Sdk($sharedConfig); $s3Client = $sdk->createS3(); $s3Client->putObject([ 'Bucket' => $registry['ya.store']['basket'], 'Key' => $fileData['type'] . '/' . $fileData['uname'], 'Body' => $fileData['body'], 'StorageClass' => (($_GET['class1']&&$_GET['type']=='files') || ($_GET['class2']&&$_GET['type']=='images')?'COLD':'STANDARD'), ]); } if($_GET['action'] == 'delete') { // Удаление файлов из Облака // удаляем со Storage $sharedConfig = [ 'credentials' => [ 'key' => $registry['ya.store']['key'], 'secret' => $registry['ya.store']['secret'], ], 'region' => 'us-east-1', 'endpoint' => 'https://storage.yandexcloud.net', 'version' => 'latest', ]; $sdk = new Aws\Sdk($sharedConfig); $s3Client = $sdk->createS3(); try { $s3Client->deleteObject([ 'Bucket' => $registry['ya.store']['basket'], 'Key' => $testFile[0]['storage_path'] . '/' . $testFile[0]['upload_name'], ]); } catch (Aws\S3\Exception\S3Exception $e) { } } <h2>База данных</h2> Для учета загруженных файлов используется своя таблица в базе данных, данные в которую заносятся скриптом ./stotage/yandex.php (в листинге PHP скрипта, строки для работы с таблицей БД убраны). Хранение в базе данных не обязательно, эти данные мы храним для себя и не пользуемся таблицей для вывода загруженных файлов на страницах статей доступных пользователям, т.к. для этого потребовалось бы модифицировать файлы компонента Cobalt(чего мы хотим избежать), в частности для получения данных из БД. <pre lang="SQL" colla="-"> CREATE TABLE `yandex_storage` ( `id` int(11) NOT NULL, `file_id` varchar(50) NOT NULL, `file_name` varchar(200) NOT NULL, `file_size` int(11) NOT NULL, `upload_name` varchar(200) NOT NULL, `date` int(11) NOT NULL, `storage_path` varchar(1000) NOT NULL, `status` int(11) NOT NULL COMMENT '0 - процесс, 1 - ок' ) ENGINE=InnoDB DEFAULT CHARSET=cp1251 COMMENT='Файлы загруженные на Яндекс Облако'; |
Вывод загруженных файлов на странице с контентом
Так как данные о загруженных файлах, хранятся в полях Textarea(а БД для получения мы не используем, по выше описанной причине) в формате JSON в массиве преобразованном в строку, все что нужно нам сделать, это получить эти данные с помощью jQuery и преобразовать в HTML код для отображения на странице в удобном для пользователя виде(сами изображения и ссылки на скачивания файлов), работа с выводом данных происходит в файле ./templates/sitetheme/js/yandex.storage.js листинг которого преставлен выше.
Так выглядят данные о загруженных файлах в формате JSON, хранящиеся в полях Teatarea:
Так выглядят уже преобразованные данные в HTML:
Заключение
Условие запрета на изменения файлов компонента Cobalt потребовало отказаться от внесения доработок непосредственно в сам компонент, благодаря чему можно было бы расширить функционал компонента и обойтись без создания дополнительных внешних скриптов на JS, в том числе не потребовалось бы вносить изменения в файл скрипта MooUpload. Решение получилось не однозначным, но мне удалось с ним справиться и достичь поставленной задачи. Более того, такой внешний скрипт можно интегрировать в любой сайт независимо от CMS на которой он работает.
Если вам требуется помощь в доработке Cobalt или интеграция сайта в сервисами Яндекса, обращайтесь — буду рад помочь!
Похожие записи
Оставить комментарий
Senior, Architector
предложить оффер
- Исполнитель пропал, почему такое случается и понять с кем работать? спросил (а) Артем
- Можно ли WordPress считать универсальным движком? спросил (а) Андрей
- Что такое самописный скрипт или CMS? спросил (а) Антон
- Как при поиске в linux используя grep, добавить исключения? спросил (а) Алексей
- Как создать публичный ключ в RSA? спросил (а) Сергей
- Интеграция Тинькофф банк Эквайринг на сайт для приема платежей к записи
- Скрипт парсинга форума к записи
- Интеграция по API с страховым маркетплейсом INSSMART к записи
- Интеграция Тинькофф банк Эквайринг на сайт для приема платежей к записи
- Joomla не пускает в админку к записи
- Все что нужно для работы с WSDL к записи
- Интеграция по API с страховым маркетплейсом INSSMART к записи
Archive
- +2023 (2)
- Март 2023 (1)
- Февраль 2023 (1)
- +2022 (21)
- Декабрь 2022 (11)
- Ноябрь 2022 (1)
- Май 2022 (2)
- Апрель 2022 (2)
- Март 2022 (3)
- Февраль 2022 (1)
- Январь 2022 (1)
- +2021 (17)
- Декабрь 2021 (5)
- Ноябрь 2021 (2)
- Июль 2021 (1)
- Июнь 2021 (2)
- Май 2021 (5)
- Апрель 2021 (1)
- Март 2021 (1)
- +2020 (20)
- Декабрь 2020 (6)
- Сентябрь 2020 (2)
- Август 2020 (1)
- Июль 2020 (2)
- Май 2020 (2)
- Апрель 2020 (2)
- Март 2020 (2)
- Февраль 2020 (1)
- Январь 2020 (2)
- +2019 (18)
- Декабрь 2019 (3)
- Ноябрь 2019 (2)
- Октябрь 2019 (2)
- Сентябрь 2019 (1)
- Август 2019 (2)
- Июль 2019 (1)
- Июнь 2019 (1)
- Апрель 2019 (2)
- Март 2019 (1)
- Февраль 2019 (3)
- +2018 (44)
- Декабрь 2018 (4)
- Ноябрь 2018 (7)
- Октябрь 2018 (8)
- Сентябрь 2018 (1)
- Август 2018 (4)
- Июль 2018 (5)
- Май 2018 (3)
- Апрель 2018 (7)
- Март 2018 (1)
- Февраль 2018 (2)
- Январь 2018 (2)
- +2017 (19)
- Декабрь 2017 (2)
- Ноябрь 2017 (1)
- Октябрь 2017 (1)
- Сентябрь 2017 (2)
- Июль 2017 (1)
- Июнь 2017 (1)
- Май 2017 (2)
- Апрель 2017 (3)
- Март 2017 (2)
- Февраль 2017 (1)
- Январь 2017 (3)
- +2016 (37)
- Декабрь 2016 (3)
- Ноябрь 2016 (3)
- Октябрь 2016 (2)
- Сентябрь 2016 (3)
- Август 2016 (7)
- Июнь 2016 (3)
- Май 2016 (3)
- Апрель 2016 (3)
- Март 2016 (1)
- Февраль 2016 (1)
- Январь 2016 (8)
- +2015 (36)
- Ноябрь 2015 (5)
- Октябрь 2015 (4)
- Сентябрь 2015 (1)
- Август 2015 (8)
- Июнь 2015 (1)
- Май 2015 (4)
- Апрель 2015 (8)
- Март 2015 (3)
- Февраль 2015 (2)
- +2014 (26)
- Ноябрь 2014 (2)
- Октябрь 2014 (5)
- Сентябрь 2014 (6)
- Июль 2014 (1)
- Июнь 2014 (2)
- Май 2014 (3)
- Апрель 2014 (6)
- Февраль 2014 (1)
- +2013 (27)
- Декабрь 2013 (2)
- Ноябрь 2013 (1)
- Октябрь 2013 (1)
- Август 2013 (1)
- Июль 2013 (3)
- Июнь 2013 (10)
- Май 2013 (1)
- Апрель 2013 (2)
- Февраль 2013 (3)
- Январь 2013 (3)
- +2012 (42)
- Декабрь 2012 (2)
- Ноябрь 2012 (3)
- Октябрь 2012 (7)
- Сентябрь 2012 (2)
- Август 2012 (1)
- Июль 2012 (3)
- Июнь 2012 (2)
- Май 2012 (6)
- Апрель 2012 (2)
- Март 2012 (8)
- Февраль 2012 (5)
- Январь 2012 (1)
- +2011 (57)
- Декабрь 2011 (6)
- Ноябрь 2011 (2)
- Октябрь 2011 (3)
- Сентябрь 2011 (5)
- Август 2011 (4)
- Июль 2011 (3)
- Июнь 2011 (3)
- Май 2011 (3)
- Апрель 2011 (4)
- Март 2011 (10)
- Февраль 2011 (5)
- Январь 2011 (9)
- +2010 (43)
- Декабрь 2010 (7)
- Ноябрь 2010 (21)
- Октябрь 2010 (14)
- Сентябрь 2010 (1)
Свежие записи
- Почему на сайте, что-то может «само» сломаться? 16.03.2023
- Как быстро писать статьи под низкочастотные запросы 28.02.2023
- Разработка сайта для Андрея Ковалёва 30.12.2022
- Разработка плагина для WordPress — Аудиогид 12.12.2022
- Оптимизация скорости работы сайта, поиск слабых мест (от CPU до MySQL) 07.12.2022