diff --git a/.travis.yml b/.travis.yml index 28433656d..a9c08e055 100644 --- a/.travis.yml +++ b/.travis.yml @@ -34,7 +34,7 @@ before_script: - npm install -g browserify@11.2.0 - composer install - cp tests/travis/config.php resources/config/local.php - - bin/cm app set-deploy-version + - bin/cm app set-config deploy '{"deployVersion":'$(date +"%s")'}' script: - bin/phpunit diff --git a/client-vendor/after-body/jquery.floatbox/jquery.floatbox.js b/client-vendor/after-body/jquery.floatbox/jquery.floatbox.js index b3892f7da..9b98f6de8 100755 --- a/client-vendor/after-body/jquery.floatbox/jquery.floatbox.js +++ b/client-vendor/after-body/jquery.floatbox/jquery.floatbox.js @@ -33,6 +33,15 @@ $parent: null, $layer: null, $floatbox: null, + setOptions: function(value){ + if (_.isObject(value)) { + //`value` as object to rewrite options + _.extend(this.options, value); + this.repaint(); + } else { + throw new Error('Invalid param to floatbox.setOptions'); + } + }, show: function($element) { var $floatboxConfig = $element.find('.floatbox-config:first'); this.options.fullscreen = $floatboxConfig.data('fullscreen') || this.options.fullscreen; @@ -155,4 +164,12 @@ } }); }; + $.fn.floatbox = function(method, value) { + return this.each(function() { + var floatbox = $(this).data('floatbox'); + if (floatbox && _.isFunction(floatbox[method])) { + floatbox[method](value); + } + }); + }; })(jQuery); diff --git a/client-vendor/after-body/jquery.lazyImageSetup/jquery.lazyImageSetup.js b/client-vendor/after-body/jquery.lazyImageSetup/jquery.lazyImageSetup.js index 622a4d0c7..a706aaa1d 100644 --- a/client-vendor/after-body/jquery.lazyImageSetup/jquery.lazyImageSetup.js +++ b/client-vendor/after-body/jquery.lazyImageSetup/jquery.lazyImageSetup.js @@ -11,14 +11,14 @@ $.fn.lazyImageSetup = function(lazyLoadOptions) { return this.each(function() { var options = _.defaults(lazyLoadOptions || {}, { - threshold: 600, - failure_limit: 10 + offset: 600, + attribute: 'original' }); var $this = $(this); if ($this.closest('.scrollable').length) { options.container = $this.closest('.scrollable'); } - $this.find('img.lazy').lazyload(options); + $this.find('img.lazy').unveil(options); }); }; diff --git a/client-vendor/after-body/jquery.lazyload/jquery.lazyload.js b/client-vendor/after-body/jquery.lazyload/jquery.lazyload.js deleted file mode 100755 index 1e6843316..000000000 --- a/client-vendor/after-body/jquery.lazyload/jquery.lazyload.js +++ /dev/null @@ -1,242 +0,0 @@ -/*! - * Lazy Load - jQuery plugin for lazy loading images - * - * Copyright (c) 2007-2015 Mika Tuupola - * - * Licensed under the MIT license: - * http://www.opensource.org/licenses/mit-license.php - * - * Project home: - * http://www.appelsiini.net/projects/lazyload - * - * Version: 1.9.7 - * - */ - -(function($, window, document, undefined) { - var $window = $(window); - - $.fn.lazyload = function(options) { - var elements = this; - var $container; - var settings = { - threshold : 0, - failure_limit : 0, - event : "scroll", - effect : "show", - container : window, - data_attribute : "original", - skip_invisible : false, - appear : null, - load : null, - placeholder : "" - }; - - function update() { - var counter = 0; - - elements.each(function() { - var $this = $(this); - if (settings.skip_invisible && !$this.is(":visible")) { - return; - } - if ($.abovethetop(this, settings) || - $.leftofbegin(this, settings)) { - /* Nothing. */ - } else if (!$.belowthefold(this, settings) && - !$.rightoffold(this, settings)) { - $this.trigger("appear"); - /* if we found an image we'll load, reset the counter */ - counter = 0; - } else { - if (++counter > settings.failure_limit) { - return false; - } - } - }); - - } - - if(options) { - /* Maintain BC for a couple of versions. */ - if (undefined !== options.failurelimit) { - options.failure_limit = options.failurelimit; - delete options.failurelimit; - } - if (undefined !== options.effectspeed) { - options.effect_speed = options.effectspeed; - delete options.effectspeed; - } - - $.extend(settings, options); - } - - /* Cache container as jQuery as object. */ - $container = (settings.container === undefined || - settings.container === window) ? $window : $(settings.container); - - /* Fire one scroll event per scroll. Not one scroll event per image. */ - if (0 === settings.event.indexOf("scroll")) { - $container.bind(settings.event, function() { - return update(); - }); - } - - this.each(function() { - var self = this; - var $self = $(self); - - self.loaded = false; - - /* If no src attribute given use data:uri. */ - if ($self.attr("src") === undefined || $self.attr("src") === false) { - if ($self.is("img")) { - $self.attr("src", settings.placeholder); - } - } - - /* When appear is triggered load original image. */ - $self.one("appear", function() { - if (!this.loaded) { - if (settings.appear) { - var elements_left = elements.length; - settings.appear.call(self, elements_left, settings); - } - $("") - .bind("load", function() { - - var original = $self.attr("data-" + settings.data_attribute); - $self.hide(); - if ($self.is("img")) { - $self.attr("src", original); - } else { - $self.css("background-image", "url('" + original + "')"); - } - $self[settings.effect](settings.effect_speed); - - self.loaded = true; - - /* Remove image from array so it is not looped next time. */ - var temp = $.grep(elements, function(element) { - return !element.loaded; - }); - elements = $(temp); - - if (settings.load) { - var elements_left = elements.length; - settings.load.call(self, elements_left, settings); - } - }) - .attr("src", $self.attr("data-" + settings.data_attribute)); - } - }); - - /* When wanted event is triggered load original image */ - /* by triggering appear. */ - if (0 !== settings.event.indexOf("scroll")) { - $self.bind(settings.event, function() { - if (!self.loaded) { - $self.trigger("appear"); - } - }); - } - }); - - /* Check if something appears when window is resized. */ - $window.bind("resize", function() { - update(); - }); - - /* With IOS5 force loading images when navigating with back button. */ - /* Non optimal workaround. */ - if ((/(?:iphone|ipod|ipad).*os 5/gi).test(navigator.appVersion)) { - $window.bind("pageshow", function(event) { - if (event.originalEvent && event.originalEvent.persisted) { - elements.each(function() { - $(this).trigger("appear"); - }); - } - }); - } - - /* Force initial check if images should appear. */ - $(document).ready(function() { - update(); - }); - - return this; - }; - - /* Convenience methods in jQuery namespace. */ - /* Use as $.belowthefold(element, {threshold : 100, container : window}) */ - - $.belowthefold = function(element, settings) { - var fold; - - if (settings.container === undefined || settings.container === window) { - fold = (window.innerHeight ? window.innerHeight : $window.height()) + $window.scrollTop(); - } else { - fold = $(settings.container).offset().top + $(settings.container).height(); - } - - return fold <= $(element).offset().top - settings.threshold; - }; - - $.rightoffold = function(element, settings) { - var fold; - - if (settings.container === undefined || settings.container === window) { - fold = $window.width() + $window.scrollLeft(); - } else { - fold = $(settings.container).offset().left + $(settings.container).width(); - } - - return fold <= $(element).offset().left - settings.threshold; - }; - - $.abovethetop = function(element, settings) { - var fold; - - if (settings.container === undefined || settings.container === window) { - fold = $window.scrollTop(); - } else { - fold = $(settings.container).offset().top; - } - - return fold >= $(element).offset().top + settings.threshold + $(element).height(); - }; - - $.leftofbegin = function(element, settings) { - var fold; - - if (settings.container === undefined || settings.container === window) { - fold = $window.scrollLeft(); - } else { - fold = $(settings.container).offset().left; - } - - return fold >= $(element).offset().left + settings.threshold + $(element).width(); - }; - - $.inviewport = function(element, settings) { - return !$.rightoffold(element, settings) && !$.leftofbegin(element, settings) && - !$.belowthefold(element, settings) && !$.abovethetop(element, settings); - }; - - /* Custom selectors for your convenience. */ - /* Use as $("img:below-the-fold").something() or */ - /* $("img").filter(":below-the-fold").something() which is faster */ - - $.extend($.expr[":"], { - "below-the-fold" : function(a) { return $.belowthefold(a, {threshold : 0}); }, - "above-the-top" : function(a) { return !$.belowthefold(a, {threshold : 0}); }, - "right-of-screen": function(a) { return $.rightoffold(a, {threshold : 0}); }, - "left-of-screen" : function(a) { return !$.rightoffold(a, {threshold : 0}); }, - "in-viewport" : function(a) { return $.inviewport(a, {threshold : 0}); }, - /* Maintain BC for couple of versions. */ - "above-the-fold" : function(a) { return !$.belowthefold(a, {threshold : 0}); }, - "right-of-fold" : function(a) { return $.rightoffold(a, {threshold : 0}); }, - "left-of-fold" : function(a) { return !$.rightoffold(a, {threshold : 0}); } - }); - -})(jQuery, window, document); diff --git a/client-vendor/after-body/jquery.unveil2/README.md b/client-vendor/after-body/jquery.unveil2/README.md new file mode 100644 index 000000000..7846b480d --- /dev/null +++ b/client-vendor/after-body/jquery.unveil2/README.md @@ -0,0 +1,5 @@ +Currently we use a custom fork https://github.com/vogdb/unveil2. As soon as out changes are merged please remove this file. +Our changes are: +- https://github.com/nabble/unveil2/pull/27 +- https://github.com/nabble/unveil2/pull/28 +- https://github.com/nabble/unveil2/pull/30 \ No newline at end of file diff --git a/client-vendor/after-body/jquery.unveil2/jquery.unveil2.js b/client-vendor/after-body/jquery.unveil2/jquery.unveil2.js new file mode 100644 index 000000000..2bd7023bf --- /dev/null +++ b/client-vendor/after-body/jquery.unveil2/jquery.unveil2.js @@ -0,0 +1,378 @@ +/** + * Unveil2.js + * A very lightweight jQuery plugin to lazy load images + * Based on https://github.com/luis-almeida/unveil + * + * Licensed under the MIT license. + * Copyright 2015 Joram van den Boezem + * https://github.com/nabble/unveil2 + */ +(function(factory) { + 'use strict'; + if (typeof define === 'function' && define.amd) { + define(['jquery'], factory); + } else if (typeof exports !== 'undefined') { + module.exports = factory(require('jquery')); + } else { + factory(jQuery); + } + +}(function ($) { + + "use strict"; + + /** + * # GLOBAL VARIABLES + * --- + */ + + /** + * Store the string 'unveil' in a variable to save some bytes + */ + var unveilString = 'unveil', + + /** + * Store the string 'src' in a variable to save some bytes + */ + srcString = 'src', + + /** + * Store the string 'placeholder' in a variable to save some bytes + */ + placeholderString = 'placeholder'; + + /** + * # PLUGIN + * --- + */ + + /** + * @param {object} options An object of options, see API section in README + * @returns {$} + */ + $.fn.unveil = function (options) { + + options = options || {}; + + // Initialize variables + var $window = $(window), + height = $window.height(), + defaults = { + // Public API + placeholder: '', + offset: 0, + breakpoints: [], + throttle: 250, + debug: false, + attribute: srcString, + + // Undocumented + container: $window, + retina: window.devicePixelRatio > 1, + + // Deprecated + loading: null, + loaded: null + }, + settings = $.extend(true, {}, defaults, options); + + if (settings.debug) console.log('Called unveil on', this.length, 'elements with the following options:', settings); + + /** + * Sort sizes array, arrange highest minWidth to front of array + */ + settings.breakpoints.sort(function (a, b) { + return b.minWidth - a.minWidth; + }); + + var containerContext = settings.container.data('unveil2'); + if (!containerContext) { + containerContext = { + images: $(), + initialized: false + }; + settings.container.data('unveil2', containerContext); + } + + /** + * # UNVEIL IMAGES + * --- + */ + + /** + * This is the actual plugin logic, which determines the source attribute to use based on window width and presence of a retina screen, changes the source of the image, handles class name changes and triggers a callback if set. Once the image has been loaded, start the unveil lookup because the page layout could have changed. + */ + this.one(unveilString + '.' + unveilString, function () { + var i, $this = $(this), windowWidth = $window.width(), + attrib = settings.attribute, targetSrc, defaultSrc, retinaSrc; + + // Determine attribute to extract source from + for (i = 0; i < settings.breakpoints.length; i++) { + var dataAttrib = settings.breakpoints[i].attribute.replace(/^data-/, ''); + if (windowWidth >= settings.breakpoints[i].minWidth && $this.data(dataAttrib)) { + attrib = dataAttrib; + break; + } + } + + // Extract source + defaultSrc = retinaSrc = $this.data(attrib); + + // Do we have a retina source? + if (defaultSrc && defaultSrc.indexOf("|") !== -1) { + retinaSrc = defaultSrc.split("|")[1]; + defaultSrc = defaultSrc.split("|")[0]; + } + + // Change attribute on image + if (defaultSrc) { + targetSrc = (settings.retina && retinaSrc) ? retinaSrc : defaultSrc; + + if (settings.debug) console.log('Unveiling image', { + attribute: attrib, + retina: settings.retina, + defaultSrc: defaultSrc, + retinaSrc: retinaSrc, + targetSrc: targetSrc + }); + + // Change classes + $this.addClass(unveilString + '-loading'); + + // Fire up the callback if it's a function... + if (typeof settings.loading === 'function') { + settings.loading.call(this); + } + // ...and trigger custom event + $this.trigger('loading.unveil'); + + // When new source has loaded, do stuff + $this.one('load', function () { + + // Change classes + classLoaded($this); + + // Fire up the callback if it's a function... + if (typeof settings.loaded === 'function') { + settings.loaded.call(this); + } + // ...and trigger custom event + $this.trigger('loaded.unveil'); + + // Loading the image may have modified page layout, + // so unveil again + lookup(); + }); + + // Set new source + if (this.nodeName === 'IMG') { + $this.prop(srcString, targetSrc); + } else { + $('').attr(srcString, targetSrc).one('load', function() { + $(this).remove(); + $this.css('backgroundImage', 'url(' + targetSrc + ')').trigger('load'); + }); + } + + // If the image has instantly loaded, change classes now + if (this.complete) { + classLoaded($this); + } + } + }); + + this.one('destroy.' + unveilString, function () { + $(this).off('.unveil'); + if (containerContext.images) { + containerContext.images = containerContext.images.not(this); + if (!containerContext.images.length) { + destroyContainer(); + } + } + }); + + /** + * # HELPER FUNCTIONS + * --- + */ + + /** + * Sets the classes when an image is done loading + * + * @param {object} $elm + */ + function classLoaded($elm) { + $elm.removeClass(unveilString + '-' + placeholderString + ' ' + unveilString + '-loading'); + $elm.addClass(unveilString + '-loaded'); + } + + /** + * Filter function which returns true when a given image is in the viewport. + * + * @returns {boolean} + */ + function inview() { + // jshint validthis: true + var $this = $(this); + + if ($this.is(':hidden')) { + return; + } + + var viewport = {top: 0 - settings.offset, bottom: $window.height() + settings.offset}, + isCustomContainer = settings.container[0] !== $window[0], + elementRect = this.getBoundingClientRect(); + if (isCustomContainer) { + var containerRect = settings.container[0].getBoundingClientRect(); + if (contains(viewport, containerRect)) { + var top = containerRect.top - settings.offset; + var bottom = containerRect.bottom + settings.offset; + var containerRectWithOffset = { + top: top > viewport.top ? top : viewport.top, + bottom: bottom < viewport.bottom ? bottom : viewport.bottom + }; + return contains(containerRectWithOffset, elementRect); + } + return false; + } else { + return contains(viewport, elementRect); + } + } + + /** + * Whether `viewport` contains `rect` vertically + */ + function contains(viewport, rect) { + return rect.bottom >= viewport.top && rect.top <= viewport.bottom; + } + + /** + * Sets the window height and calls the lookup function + */ + function resize() { + height = $window.height(); + lookup(); + } + + /** + * Throttle function with function call in tail. Based on http://sampsonblog.com/749/simple-throttle-function + * + * @param {function} callback + * @returns {function} + */ + function throttle(callback) { + var wait = false; // Initially, we're not waiting + return function () { // We return a throttled function + if (!wait) { // If we're not waiting + wait = true; // Prevent future invocations + setTimeout(function () { // After a period of time + callback(); // Execute users function + wait = false; // And allow future invocations + }, settings.throttle); + } + }; + } + + /** + * # LOOKUP FUNCTION + * --- + */ + + /** + * Function which filters images which are in view and triggers the unveil event on those images. + */ + function lookup() { + if (settings.debug) console.log('Unveiling'); + + if (containerContext.images) { + var batch = containerContext.images.filter(inview); + + batch.trigger(unveilString + '.' + unveilString); + containerContext.images = containerContext.images.not(batch); + + if (batch.length) { + if (settings.debug) console.log('New images in view', batch.length, ', leaves', containerContext.length, 'in collection'); + } + } + } + + function destroyContainer() { + settings.container.off('.unveil'); + containerContext.images.off('.unveil'); + settings.container.data('unveil2', null); + containerContext.initialized = false; + containerContext.images = null; + } + + /** + * # BOOTSTRAPPING + * --- + */ + + /** + * Read and reset the src attribute, to prevent loading of original images + */ + this.each(function () { + var $this = $(this), + elmPlaceholder = $this.data(srcString + '-' + placeholderString) || settings.placeholder; + + // Add element to global array + containerContext.images = $(containerContext.images).add(this); + + // If this element has been called before, + // don't set placeholder now to prevent FOUI (Flash Of Ustyled Image) + if (!$this.data(unveilString)) { + + // Set the unveil flag + $this.data(unveilString, true); + + // Set data-src if not set + if (!$this.data(settings.attribute)) { + $this.data(settings.attribute, $this.prop(settings.attribute)); + } + + // Set placeholder + $this + .one('load', function () { + var $this = $(this); + + if ($this.hasClass(unveilString + '-loaded')) { + return; + } + + $this.addClass(unveilString + '-' + placeholderString); + lookup(); + }) + .prop(srcString, '') + .prop(srcString, elmPlaceholder); + } + }); + + if (settings.debug) console.log('Images now in collection', containerContext.images.length); + + /** + * Bind global listeners + */ + {if (!containerContext.initialized) { + settings.container.on({ + 'resize.unveil': throttle(resize), + 'scroll.unveil': throttle(lookup), + 'lookup.unveil': lookup, + 'destroy.unveil': destroyContainer + }); + containerContext.initialized = true; + }} + + /** + * Wait a little bit for the placeholder to be set, so the src attribute is not empty (which will mark the image as hidden) + */ + {setTimeout(lookup, 0);} + + /** + * That's all folks! + */ + return this; + + }; + +})); diff --git a/client-vendor/after-body/promise-utils/promise-throttler.js b/client-vendor/after-body/promise-utils/promise-throttler.js index c911defe7..b19351381 100644 --- a/client-vendor/after-body/promise-utils/promise-throttler.js +++ b/client-vendor/after-body/promise-utils/promise-throttler.js @@ -41,10 +41,26 @@ function nonameThrottler(fn, options) { var promise; + function resetClosurePromise(resetPromise) { + if (resetPromise === promise) { + promise = null; + } + } + + function isPromise(promise) { + if (!promise || !(promise instanceof Promise)) { + throw new Error('Invalid usage of promiseThrottler'); + } + return true; + } + return function() { - if (!promise || !promise.isPending()) { + if (!promise) { promise = fn.apply(this, arguments); - return promise; + if (isPromise(promise)) { + promise.finally(resetClosurePromise.bind(null, promise)); + return promise; + } } var cancelLeading = options.cancelLeading; if (cancelLeading) { @@ -57,9 +73,13 @@ var leadingPromise = promise; promise = new Promise(function(resolve) { leadingPromise.finally(function() { - resolve(fn.apply(self, args)); + var promise = fn.apply(self, args); + if (isPromise(promise)) { + resolve(promise); + } }); }); + promise.finally(resetClosurePromise.bind(null, promise)); return promise; } return promise; diff --git a/library/CM/App.js b/library/CM/App.js index 73605fa9f..335097771 100644 --- a/library/CM/App.js +++ b/library/CM/App.js @@ -499,6 +499,7 @@ var CM_App = CM_Class_Abstract.extend({ teardown: function($dom) { $dom.find('.timeago').timeago('dispose'); $dom.find('textarea.autosize, .autosize textarea').trigger('autosize.destroy'); + $dom.find('img.lazy').trigger('destroy.unveil'); }, /** @@ -1433,7 +1434,7 @@ var CM_App = CM_Class_Abstract.extend({ var urlBase = cm.options.urlBase; if (0 === url.indexOf(urlSite)) { var path = url.substr(urlBase.length); - cm.getDocument().loadPage(path); + return cm.getDocument().loadPage(path); } else { window.location.assign(url); return Promise.resolve(); diff --git a/library/CM/App/Cli.php b/library/CM/App/Cli.php index 5daf461b3..a223f0a92 100644 --- a/library/CM/App/Cli.php +++ b/library/CM/App/Cli.php @@ -69,21 +69,38 @@ public function generateConfigInternal() { } /** - * @param int|null $deployVersion + * @param string $filename + * @param string $configJson + * @param bool|null $merge */ - public function setDeployVersion($deployVersion = null) { - $deployVersion = (null !== $deployVersion) ? (int) $deployVersion : time(); - $sourceCode = join(PHP_EOL, array( + public function setConfig($filename, $configJson, $merge = null) { + $merge = (bool) $merge; + $configJson = (object) CM_Util::jsonDecode($configJson); + $configFile = new CM_File(DIR_ROOT . 'resources/config/' . $filename . '.php'); + + $config = new CM_Config_Node(); + if ($merge && $configFile->exists()) { + $config->extendWithFile($configFile); + } + $config->extendWithConfig($configJson); + $configStr = $config->exportAsString('$config'); + + $indentation = ' '; + $indent = function ($content) use ($indentation) { + return preg_replace('/(:?^|[\n])/', '$1' . $indentation, $content); + }; + + $configFile->ensureParentDirectory(); + $configFile->write(join(PHP_EOL, [ 'deployVersion = ' . $deployVersion . ';', + $indent($configStr), '};', '', - )); - $targetPath = DIR_ROOT . 'resources/config/deploy.php'; - $configFile = new CM_File($targetPath); - $configFile->ensureParentDirectory(); - $configFile->write($sourceCode); + ])); + $this->_getStreamOutput()->writeln('Created `' . $configFile->getPath() . '`'); } public static function getPackageName() { diff --git a/library/CM/Db/Db.php b/library/CM/Db/Db.php index 02fa4b248..ca0035482 100644 --- a/library/CM/Db/Db.php +++ b/library/CM/Db/Db.php @@ -150,6 +150,7 @@ public static function insertIgnore($table, $fields, $values = null) { * @param string|array|null $values Column-value OR Column-values array OR Multiple Column-values array(array) * @param array|null $onDuplicateKeyValues * @return string|null + * @deprecated Removed from MySQL 5.7 */ public static function insertDelayed($table, $fields, $values = null, array $onDuplicateKeyValues = null) { $statement = (self::_getConfig()->delayedEnabled) ? 'INSERT DELAYED' : 'INSERT'; @@ -171,6 +172,7 @@ public static function replace($table, $fields, $values = null) { * @param string|array $fields Column-name OR Column-names array OR associative field=>value pair * @param string|array|null $values Column-value OR Column-values array OR Multiple Column-values array(array) * @return string|null + * @deprecated Removed from MySQL 5.7 */ public static function replaceDelayed($table, $fields, $values = null) { $statement = (self::_getConfig()->delayedEnabled) ? 'REPLACE DELAYED' : 'REPLACE'; diff --git a/library/CM/Http/Response/Abstract.php b/library/CM/Http/Response/Abstract.php index 9cd44c129..9fdf1a449 100644 --- a/library/CM/Http/Response/Abstract.php +++ b/library/CM/Http/Response/Abstract.php @@ -294,7 +294,7 @@ protected function _runWithCatching(Closure $regularCode, Closure $errorCode) { $context = new CM_Log_Context(); $context->setUser($this->getViewer()); $context->setException($ex); - $this->getServiceManager()->getLogger()->addMessage('HTML response error', $logLevel, $context); + $this->getServiceManager()->getLogger()->addMessage('Response processing error', $logLevel, $context); } if (!$catchException && ($catchPublicExceptions && $ex->isPublic())) { $errorOptions = []; diff --git a/library/CM/Http/Response/View/Abstract.php b/library/CM/Http/Response/View/Abstract.php index cc9258658..8cd8e5d14 100644 --- a/library/CM/Http/Response/View/Abstract.php +++ b/library/CM/Http/Response/View/Abstract.php @@ -109,7 +109,7 @@ public function loadPage(CM_Params $params, CM_Http_Response_View_Ajax $response } $title = $responseEmbed->getTitle(); - $menuList = array_merge($this->getSite()->getMenus(), $responseEmbed->getRender()->getMenuList()); + $menuList = array_merge($this->getSite()->getMenus($this->getRender()->getEnvironment()), $responseEmbed->getRender()->getMenuList()); $menuEntryHashList = $this->_getMenuEntryHashList($menuList, get_class($page), $page->getParams()); $jsTracking = $responseEmbed->getRender()->getServiceManager()->getTrackings()->getJs(); diff --git a/library/CM/Jobdistribution/Cli.php b/library/CM/Jobdistribution/Cli.php index 48bf554fa..be1166f48 100644 --- a/library/CM/Jobdistribution/Cli.php +++ b/library/CM/Jobdistribution/Cli.php @@ -6,7 +6,7 @@ class CM_Jobdistribution_Cli extends CM_Cli_Runnable_Abstract { * @keepalive */ public function startWorker() { - $worker = new CM_Jobdistribution_JobWorker(); + $worker = new CM_Jobdistribution_JobWorker(1000); $worker->setServiceManager($this->getServiceManager()); foreach (CM_Jobdistribution_Job_Abstract::getClassChildren() as $jobClassName) { /** @var CM_Jobdistribution_Job_Abstract $job */ diff --git a/library/CM/Jobdistribution/Job/Abstract.php b/library/CM/Jobdistribution/Job/Abstract.php index 78aee513f..31266b4b5 100644 --- a/library/CM/Jobdistribution/Job/Abstract.php +++ b/library/CM/Jobdistribution/Job/Abstract.php @@ -9,20 +9,10 @@ abstract class CM_Jobdistribution_Job_Abstract extends CM_Class_Abstract { abstract protected function _execute(CM_Params $params); /** - * @param CM_Params $params - * @return mixed - * @throws Exception + * @return CM_Jobdistribution_Priority */ - private function _executeJob(CM_Params $params) { - CM_Service_Manager::getInstance()->getNewrelic()->startTransaction('CM Job: ' . $this->_getClassName()); - try { - $return = $this->_execute($params); - CM_Service_Manager::getInstance()->getNewrelic()->endTransaction(); - return $return; - } catch (Exception $ex) { - CM_Service_Manager::getInstance()->getNewrelic()->endTransaction(); - throw $ex; - } + public function getPriority() { + return new CM_Jobdistribution_Priority('normal'); } /** @@ -63,7 +53,7 @@ public function runMultiple(array $paramsList) { foreach ($paramsList as $params) { $workload = CM_Params::encode($params, true); - $task = $gearmanClient->addTaskHigh($this->_getJobName(), $workload); + $task = $this->_addTask($workload, $gearmanClient); if (false === $task) { throw new CM_Exception('Cannot add task', null, ['jobName' => $this->_getJobName()]); } @@ -95,7 +85,20 @@ public function queue(array $params = null) { $workload = CM_Params::encode($params, true); $gearmanClient = $this->_getGearmanClient(); - $gearmanClient->doBackground($this->_getJobName(), $workload); + $priority = $this->getPriority(); + switch ($priority) { + case CM_Jobdistribution_Priority::HIGH: + $gearmanClient->doHighBackground($this->_getJobName(), $workload); + break; + case CM_Jobdistribution_Priority::NORMAL: + $gearmanClient->doBackground($this->_getJobName(), $workload); + break; + case CM_Jobdistribution_Priority::LOW: + $gearmanClient->doLowBackground($this->_getJobName(), $workload); + break; + default: + throw new CM_Exception('Invalid priority', null, ['priority' => (string) $priority]); + } } /** @@ -116,6 +119,15 @@ public function __executeGearman(GearmanJob $job) { return CM_Params::encode($this->_executeJob($params), true); } + /** + * @param string $workload + * @param GearmanClient $gearmanClient + * @return GearmanTask + */ + protected function _addTask($workload, $gearmanClient) { + return $gearmanClient->addTaskHigh($this->_getJobName(), $workload); + } + /** * @return string */ @@ -135,6 +147,23 @@ protected function _runMultipleWithoutGearman(array $paramsList) { return $resultList; } + /** + * @param CM_Params $params + * @return mixed + * @throws Exception + */ + private function _executeJob(CM_Params $params) { + CM_Service_Manager::getInstance()->getNewrelic()->startTransaction('CM Job: ' . $this->_getClassName()); + try { + $return = $this->_execute($params); + CM_Service_Manager::getInstance()->getNewrelic()->endTransaction(); + return $return; + } catch (Exception $ex) { + CM_Service_Manager::getInstance()->getNewrelic()->endTransaction(); + throw $ex; + } + } + /** * @return boolean */ diff --git a/library/CM/Jobdistribution/JobWorker.php b/library/CM/Jobdistribution/JobWorker.php index 39a1e97bd..64432b0b0 100644 --- a/library/CM/Jobdistribution/JobWorker.php +++ b/library/CM/Jobdistribution/JobWorker.php @@ -7,11 +7,21 @@ class CM_Jobdistribution_JobWorker extends CM_Class_Abstract implements CM_Servi /** @var GearmanWorker */ private $_gearmanWorker; - public function __construct() { - $worker = $this->_getGearmanWorker(); + /** @var int */ + private $_jobLimit; + + /** + * @param int $jobLimit + */ + public function __construct($jobLimit) { + $this->_jobLimit = (int) $jobLimit; + $config = self::_getConfig(); - foreach ($config->servers as $server) { - $worker->addServer($server['host'], $server['port']); + if ($config->servers) { + $worker = $this->_getGearmanWorker(); + foreach ($config->servers as $server) { + $worker->addServer($server['host'], $server['port']); + } } } @@ -19,13 +29,18 @@ public function __construct() { * @param CM_Jobdistribution_Job_Abstract $job */ public function registerJob(CM_Jobdistribution_Job_Abstract $job) { - $this->_gearmanWorker->addFunction(get_class($job), array($job, '__executeGearman')); + $this->_gearmanWorker->addFunction(get_class($job), [$job, '__executeGearman']); } public function run() { + $jobsRun = 0; while (true) { + if ($jobsRun >= $this->_jobLimit) { + return; + } $workFailed = false; try { + $jobsRun++; CM_Cache_Storage_Runtime::getInstance()->flush(); $workFailed = !$this->_getGearmanWorker()->work(); } catch (Exception $ex) { diff --git a/library/CM/Jobdistribution/Priority.php b/library/CM/Jobdistribution/Priority.php new file mode 100644 index 000000000..ac56684b2 --- /dev/null +++ b/library/CM/Jobdistribution/Priority.php @@ -0,0 +1,9 @@ +existsCollection($collectionSource)) { + throw new CM_MongoDb_Exception('Source collection does not exist', null, [ + 'collectionSource' => $collectionSource, + 'collectionTarget' => $collectionTarget, + ]); + } + if (!$dropTarget && $this->existsCollection($collectionTarget)) { + throw new CM_MongoDb_Exception('Target collection already exists', null, [ + 'collectionSource' => $collectionSource, + 'collectionTarget' => $collectionTarget, + ]); + } + $result = $this->_getClient()->selectDB('admin')->command([ + 'renameCollection' => $this->_getDatabaseName() . '.' . $collectionSource, + 'to' => $this->_getDatabaseName() . '.' . $collectionTarget, + 'dropTarget' => $dropTarget, + ]); + $this->_checkResultForErrors($result); + return $result; + } + /** * @param string $collection * @return array diff --git a/library/CM/Process.php b/library/CM/Process.php index 7be9a0eab..20a4b4d07 100644 --- a/library/CM/Process.php +++ b/library/CM/Process.php @@ -266,13 +266,15 @@ private function _wait($keepAlive = null, $nohang = null) { } if ($pid > 0 && ($forkHandler = $this->_findForkHandlerByPid($pid))) { unset($this->_forkHandlerList[$forkHandler->getIdentifier()]); - $workloadResultList[$forkHandler->getIdentifier()] = $forkHandler->receiveWorkloadResult(); + $workloadResult = $forkHandler->receiveWorkloadResult(); + $workloadResultList[$forkHandler->getIdentifier()] = $workloadResult; $forkHandler->closeIpcStream(); if (!$this->_hasForks()) { $this->unbind('exit', [$this, 'killChildren']); } if ($keepAlive) { - CM_Service_Manager::getInstance()->getLogger()->warning('Respawning dead child.', (new CM_Log_Context())->setExtra(['pid' => $pid])); + $logLevel = $workloadResult->isSuccess() ? CM_Log_Logger::INFO : CM_Log_Logger::WARNING; + CM_Service_Manager::getInstance()->getLogger()->addMessage('Respawning dead child.', $logLevel, (new CM_Log_Context())->setExtra(['pid' => $pid])); usleep(self::RESPAWN_TIMEOUT * 1000000); $this->_fork($forkHandler->getWorkload(), $forkHandler->getIdentifier()); } diff --git a/library/CM/Site/Abstract.php b/library/CM/Site/Abstract.php index 052018481..13bc8210b 100644 --- a/library/CM/Site/Abstract.php +++ b/library/CM/Site/Abstract.php @@ -82,9 +82,10 @@ public function getDocument() { } /** + * @param CM_Frontend_Environment $environment * @return CM_Menu[] */ - public function getMenus() { + public function getMenus(CM_Frontend_Environment $environment) { return array(); } diff --git a/library/CM/SmartyPlugins/function.menu.php b/library/CM/SmartyPlugins/function.menu.php index 35b34fd92..ea153931b 100644 --- a/library/CM/SmartyPlugins/function.menu.php +++ b/library/CM/SmartyPlugins/function.menu.php @@ -35,7 +35,7 @@ function smarty_function_menu(array $params, Smarty_Internal_Template $template) $name = null; if (isset($params['name'])) { $name = $params['name']; - $menuArr = $render->getSite()->getMenus(); + $menuArr = $render->getSite()->getMenus($environment); if (isset($menuArr[$name])) { $menu = $menuArr[$name]; } diff --git a/tests/helpers/CMTest/library/CMTest/ExceptionWrapper.php b/tests/helpers/CMTest/library/CMTest/ExceptionWrapper.php new file mode 100644 index 000000000..2dd3e55d8 --- /dev/null +++ b/tests/helpers/CMTest/library/CMTest/ExceptionWrapper.php @@ -0,0 +1,11 @@ +message = get_class($e) . ': ' . $e->getMessage() . PHP_EOL . $formatter->getMetaInfo($serializedException); + } +} diff --git a/tests/helpers/CMTest/library/CMTest/TestCase.php b/tests/helpers/CMTest/library/CMTest/TestCase.php index 25dc1f9fb..277cc3fda 100644 --- a/tests/helpers/CMTest/library/CMTest/TestCase.php +++ b/tests/helpers/CMTest/library/CMTest/TestCase.php @@ -28,6 +28,13 @@ public static function tearDownAfterClass() { CMTest_TH::clearEnv(); } + protected function onNotSuccessfulTest($e) { + if ($e instanceof CM_Exception) { + $e = new CMTest_ExceptionWrapper($e); + } + parent::onNotSuccessfulTest($e); + } + /** * @param mixed $object * @param string $methodName diff --git a/tests/library/CM/Jobdistribution/Job/AbstractTest.php b/tests/library/CM/Jobdistribution/Job/AbstractTest.php index d3d61ba89..709bdbc20 100644 --- a/tests/library/CM/Jobdistribution/Job/AbstractTest.php +++ b/tests/library/CM/Jobdistribution/Job/AbstractTest.php @@ -13,9 +13,8 @@ public function testRunMultiple() { CM_Config::get()->CM_Jobdistribution_Job_Abstract->gearmanEnabled = true; $mockBuilder = $this->getMockBuilder('GearmanClient'); - $mockBuilder->setMethods(['addTaskHigh', 'runTasks', 'setCompleteCallback', 'setFailCallback']); + $mockBuilder->setMethods(['addTaskNormal', 'runTasks', 'setCompleteCallback', 'setFailCallback']); $gearmanClientMock = $mockBuilder->getMock(); - $gearmanClientMock->expects($this->exactly(2))->method('addTaskHigh')->will($this->returnValue(true)); $gearmanClientMock->expects($this->exactly(1))->method('runTasks')->will($this->returnValue(true)); $gearmanClientMock->expects($this->exactly(1))->method('setCompleteCallback')->will($this->returnCallback(function ($completeCallback) { $task1 = $this->getMockBuilder('GearmanTask')->setMethods(array('data'))->getMock(); @@ -29,8 +28,9 @@ public function testRunMultiple() { $gearmanClientMock->expects($this->exactly(1))->method('setFailCallback'); /** @var GearmanClient $gearmanClientMock */ - $job = $this->getMockBuilder('CM_Jobdistribution_Job_Abstract')->setMethods(array('_getGearmanClient'))->getMockForAbstractClass(); + $job = $this->getMockBuilder('CM_Jobdistribution_Job_Abstract')->setMethods(array('_getGearmanClient', '_addTask'))->getMockForAbstractClass(); $job->expects($this->any())->method('_getGearmanClient')->will($this->returnValue($gearmanClientMock)); + $job->expects($this->exactly(2))->method('_addTask')->will($this->returnValue(true)); /** @var CM_Jobdistribution_Job_Abstract $job */ $result = $job->runMultiple(array( @@ -44,6 +44,45 @@ public function testRunMultiple() { ), $result); } + public function testQueuePriority() { + if (!extension_loaded('gearman')) { + $this->markTestSkipped('Gearman Pecl Extension not installed.'); + } + CM_Config::get()->CM_Jobdistribution_Job_Abstract->gearmanEnabled = true; + + $gearmanClient = $this->mockClass('GearmanClient')->newInstanceWithoutConstructor(); + + $mockDoHighBackground = $gearmanClient->mockMethod('doHighBackground'); + $mockDoBackground = $gearmanClient->mockMethod('doBackground'); + $mockDoLowBackground = $gearmanClient->mockMethod('doLowBackground'); + + /** @var CM_Jobdistribution_Job_Abstract|\Mocka\AbstractClassTrait $job */ + $job = $this->mockObject('CM_Jobdistribution_Job_Abstract'); + $job->mockMethod('_getGearmanClient')->set($gearmanClient); + + // standard priority + $job->queue([['foo' => 'bar']]); + $this->assertSame(1, $mockDoBackground->getCallCount()); + + // normal priority + $priorityMock = $job->mockMethod('getPriority'); + $priorityMock->set(new CM_Jobdistribution_Priority('normal')); + $job->queue([['foo' => 'bar']]); + $this->assertSame(2, $mockDoBackground->getCallCount()); + + // high priority + $priorityMock = $job->mockMethod('getPriority'); + $priorityMock->set(new CM_Jobdistribution_Priority('high')); + $job->queue([['foo' => 'bar']]); + $this->assertSame(1, $mockDoHighBackground->getCallCount()); + + // low priority + $priorityMock = $job->mockMethod('getPriority'); + $priorityMock->set(new CM_Jobdistribution_Priority('low')); + $job->queue([['foo' => 'bar']]); + $this->assertSame(1, $mockDoLowBackground->getCallCount()); + } + public function testRunMultipleWithFailures() { if (!extension_loaded('gearman')) { $this->markTestSkipped('Gearman Pecl Extension not installed.'); diff --git a/tests/library/CM/Jobdistribution/JobWorkerTest.php b/tests/library/CM/Jobdistribution/JobWorkerTest.php index 0a1923838..dc4067b2d 100644 --- a/tests/library/CM/Jobdistribution/JobWorkerTest.php +++ b/tests/library/CM/Jobdistribution/JobWorkerTest.php @@ -3,6 +3,7 @@ class CM_JobDistribution_JobWorkerTest extends CMTest_TestCase { public function testRun() { + CM_Config::get()->CM_Jobdistribution_JobWorker->servers = []; if (!extension_loaded('gearman')) { $this->markTestSkipped('Gearman Pecl Extension not installed.'); } @@ -16,9 +17,8 @@ public function testRun() { } throw new Exception('foo-bar'); })); - $mockBuilder = $this->getMockBuilder('CM_Jobdistribution_JobWorker'); + $mockBuilder = $this->getMockBuilder('CM_Jobdistribution_JobWorker')->setConstructorArgs([1000]); $mockBuilder->setMethods(['_getGearmanWorker', '_handleException']); - $mockBuilder->disableOriginalConstructor(); $jobWorkerMock = $mockBuilder->getMock(); $jobWorkerMock->expects($this->any())->method('_getGearmanWorker')->will($this->returnValue($gearmanWorkerMock)); /** @var CM_JobDistribution_JobWorker $jobWorkerMock */ @@ -44,4 +44,24 @@ public function testRun() { $this->fail('Exception not caught.'); } } + + public function testRunJobLimit() { + $serviceManager = new CM_Service_Manager(); + $logger = $this->mockObject('CM_Log_Logger'); + $serviceManager->registerInstance('logger', $logger); + + if (!extension_loaded('gearman')) { + $this->markTestSkipped('Gearman Pecl Extension not installed.'); + } + $gearmanWorker = $this->mockClass('GearmanWorker')->newInstanceWithoutConstructor(); + $workMethod = $gearmanWorker->mockMethod('work')->set(true); + + CM_Config::get()->CM_Jobdistribution_JobWorker->servers = []; + $worker = $this->mockClass(CM_Jobdistribution_JobWorker::class)->newInstance([5]); + $worker->mockMethod('_getGearmanWorker')->set($gearmanWorker); + /** @var CM_Jobdistribution_JobWorker $worker */ + $worker->setServiceManager($serviceManager); + $worker->run(); + $this->assertSame(5, $workMethod->getCallCount()); + } } diff --git a/tests/library/CM/MongoDb/ClientTest.php b/tests/library/CM/MongoDb/ClientTest.php index d1885a62a..ef1da8b6a 100644 --- a/tests/library/CM/MongoDb/ClientTest.php +++ b/tests/library/CM/MongoDb/ClientTest.php @@ -234,6 +234,58 @@ public function testDrop() { $this->assertFalse($mongoDb->existsCollection($collectionName)); } + public function testRename() { + $mongoDb = CM_Service_Manager::getInstance()->getMongoDb(); + + $mongoDb->insert('source', array('foo' => 'origin')); + $this->assertTrue($mongoDb->existsCollection('source')); + $this->assertFalse($mongoDb->existsCollection('target')); + + $mongoDb->rename('source', 'target'); + + $this->assertFalse($mongoDb->existsCollection('source')); + $this->assertTrue($mongoDb->existsCollection('target')); + $this->assertSame(1, $mongoDb->find('target', array('foo' => 'origin'))->count()); + + $mongoDb->insert('source', array('foo' => 'origin')); + $mongoDb->insert('target', array('foo' => 'existing-value')); + $this->assertSame(2, $mongoDb->count('target')); + + $mongoDb->rename('source', 'target', true); + + $this->assertFalse($mongoDb->existsCollection('source')); + $this->assertTrue($mongoDb->existsCollection('target')); + $this->assertSame(1, $mongoDb->count('target')); + $this->assertSame(1, $mongoDb->find('target', array('foo' => 'origin'))->count()); + } + + public function testRenameThrows() { + $mongoDb = CM_Service_Manager::getInstance()->getMongoDb(); + /** @var CM_MongoDb_Exception $exception */ + $exception = $this->catchException(function () use ($mongoDb) { + $mongoDb->rename('not_defined', 'target'); + }); + $this->assertInstanceOf('CM_MongoDb_Exception', $exception); + $this->assertSame('Source collection does not exist', $exception->getMessage()); + $this->assertSame([ + 'collectionSource' => 'not_defined', + 'collectionTarget' => 'target' + ], $exception->getMetaInfo()); + + $mongoDb->insert('source', array('foo' => 'origin')); + $mongoDb->insert('target', array('foo' => 'existing-value')); + /** @var CM_MongoDb_Exception $exception */ + $exception = $this->catchException(function () use ($mongoDb) { + $mongoDb->rename('source', 'target'); + }); + $this->assertInstanceOf('CM_MongoDb_Exception', $exception); + $this->assertSame('Target collection already exists', $exception->getMessage()); + $this->assertSame([ + 'collectionSource' => 'source', + 'collectionTarget' => 'target' + ], $exception->getMetaInfo()); + } + public function testRemove() { $mongoDb = CM_Service_Manager::getInstance()->getMongoDb(); $collectionName = 'remove';