mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
Merge branch '3.1'
Conflicts: .travis.yml docs/en/misc/contributing/code.md javascript/HtmlEditorField.js
This commit is contained in:
commit
fbce9fd7cd
20
.travis.yml
20
.travis.yml
@ -2,26 +2,20 @@ language: php
|
||||
|
||||
php:
|
||||
- 5.3
|
||||
- 5.4
|
||||
|
||||
env:
|
||||
- DB=MYSQL CORE_RELEASE=master
|
||||
- DB=PGSQL CORE_RELEASE=master
|
||||
- DB=SQLITE3 CORE_RELEASE=master
|
||||
- PHPCS=1 CORE_RELEASE=master
|
||||
|
||||
matrix:
|
||||
exclude:
|
||||
- php: 5.4
|
||||
include:
|
||||
- php: 5.3
|
||||
env: DB=PGSQL CORE_RELEASE=master
|
||||
- php: 5.3
|
||||
env: DB=SQLITE CORE_RELEASE=master
|
||||
- php: 5.4
|
||||
env: DB=SQLITE3 CORE_RELEASE=master
|
||||
- php: 5.4
|
||||
env: PHPCS=1 CORE_RELEASE=master
|
||||
allow_failures:
|
||||
- env: DB=PGSQL CORE_RELEASE=master
|
||||
- env: DB=SQLITE3 CORE_RELEASE=master
|
||||
- env: PHPCS=1 CORE_RELEASE=master
|
||||
env: DB=MYSQL CORE_RELEASE=master
|
||||
- php: 5.5
|
||||
env: DB=MYSQL CORE_RELEASE=master
|
||||
|
||||
before_script:
|
||||
- phpenv rehash
|
||||
|
@ -12,7 +12,7 @@ and [installation from source](http://doc.silverstripe.org/framework/en/installa
|
||||
|
||||
## Bugtracker ##
|
||||
|
||||
Bugs are tracked on [github.com](https://github.com/silverstripe/framework/issues).
|
||||
Bugs are tracked on [github.com](https://github.com/silverstripe/silverstripe-framework/issues).
|
||||
Please read our [issue reporting guidelines](http://doc.silverstripe.org/framework/en/misc/contributing/issues).
|
||||
|
||||
## Development and Contribution ##
|
||||
|
@ -17,6 +17,12 @@ class CMSBatchActionHandler extends RequestHandler {
|
||||
'$BatchAction' => 'handleBatchAction',
|
||||
);
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'handleBatchAction',
|
||||
'handleApplicablePages',
|
||||
'handleConfirmation',
|
||||
);
|
||||
|
||||
protected $parentController;
|
||||
|
||||
/**
|
||||
|
@ -6,6 +6,7 @@ class CMSProfileController extends LeftAndMain {
|
||||
private static $menu_title = 'My Profile';
|
||||
|
||||
private static $required_permission_codes = false;
|
||||
|
||||
private static $tree_class = 'Member';
|
||||
|
||||
public function getResponseNegotiator() {
|
||||
|
@ -76,7 +76,7 @@ class LeftAndMain extends Controller implements PermissionProvider {
|
||||
* @config
|
||||
* @var string
|
||||
*/
|
||||
private static $help_link = 'http://3.0.userhelp.silverstripe.org';
|
||||
private static $help_link = 'http://userhelp.silverstripe.org/framework/en/3.1';
|
||||
|
||||
/**
|
||||
* @var array
|
||||
|
@ -460,7 +460,8 @@ body.cms { overflow: hidden; }
|
||||
.cms-add-form .step-label { opacity: 0.9; }
|
||||
.cms-add-form .step-label .flyout { height: 17px; padding-top: 5px; }
|
||||
.cms-add-form .step-label .title { padding-top: 5px; font-weight: bold; text-shadow: 1px 1px 0 white; }
|
||||
.cms-add-form ul.SelectionGroup { padding-left: 28px; }
|
||||
.cms-add-form ul.SelectionGroup { padding-left: 28px; overflow: visible; *zoom: 1; }
|
||||
.cms-add-form ul.SelectionGroup:after { content: "\0020"; display: block; height: 0; clear: both; overflow: hidden; visibility: hidden; }
|
||||
.cms-add-form .parent-mode { padding: 8px; overflow: auto; }
|
||||
|
||||
#Form_AddForm_PageType_Holder ul { padding-left: 20px; }
|
||||
@ -667,7 +668,7 @@ body.cms-dialog { overflow: auto; background: url("../images/textures/bg_cms_mai
|
||||
.htmleditorfield-mediaform .htmleditorfield-from-cms .ss-uploadfield h4 { float: left; margin-top: 4px; margin-bottom: 0; }
|
||||
.htmleditorfield-mediaform .htmleditorfield-from-cms .ss-uploadfield .middleColumn { margin-top: 16px; margin-left: 184px; }
|
||||
.htmleditorfield-mediaform .htmleditorfield-from-cms .ss-uploadfield .field.treedropdown { border-bottom: 0; padding: 0; }
|
||||
.htmleditorfield-mediaform .ss-uploadfield-editandorganize { display: none; }
|
||||
.htmleditorfield-mediaform .ss-assetuploadfield .ss-uploadfield-editandorganize .ss-uploadfield-files .ss-uploadfield-item-info { background-color: #9e9e9e; background-image: url(''); background-size: 100%; background-image: -webkit-gradient(linear, 50% 0%, 50% 100%, color-stop(0%, #9e9e9e), color-stop(8%, #9d9d9d), color-stop(50%, #878787), color-stop(54%, #868686), color-stop(96%, #6b6b6b), color-stop(100%, #6c6c6c)); background-image: -webkit-linear-gradient(top, #9e9e9e 0%, #9d9d9d 8%, #878787 50%, #868686 54%, #6b6b6b 96%, #6c6c6c 100%); background-image: -moz-linear-gradient(top, #9e9e9e 0%, #9d9d9d 8%, #878787 50%, #868686 54%, #6b6b6b 96%, #6c6c6c 100%); background-image: -o-linear-gradient(top, #9e9e9e 0%, #9d9d9d 8%, #878787 50%, #868686 54%, #6b6b6b 96%, #6c6c6c 100%); background-image: linear-gradient(top, #9e9e9e 0%, #9d9d9d 8%, #878787 50%, #868686 54%, #6b6b6b 96%, #6c6c6c 100%); }
|
||||
|
||||
/** -------------------------------------------- Search forms (used in AssetAdmin, ModelAdmin, etc) -------------------------------------------- */
|
||||
.cms-search-form { margin-bottom: 16px; }
|
||||
@ -751,7 +752,7 @@ form.import-form label.left { width: 250px; }
|
||||
.cms .jstree .jstree-wholerow-real, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow-real { position: relative; z-index: 1; }
|
||||
.cms .jstree .jstree-wholerow-real li, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow-real li { cursor: pointer; }
|
||||
.cms .jstree .jstree-wholerow-real a, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow-real a { border-left-color: transparent !important; border-right-color: transparent !important; }
|
||||
.cms .jstree .jstree-wholerow, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow, .cms .jstree .jstree-wholerow ul, .cms .jstree .jstree-wholerow li, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow ul, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow ul, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow li, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow li, .cms .jstree .jstree-wholerow a, .cms .jstree .jstree-wholerow a:hover, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow a, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a:hover, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow a:hover, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow ul, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow ul, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow li, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow li, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow ul, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow li, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow a, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow a:hover, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a:hover, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a:hover { margin: 0 !important; padding: 0 !important; }
|
||||
.cms .jstree .jstree-wholerow, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow, .cms .jstree .jstree-wholerow ul, .cms .jstree .jstree-wholerow li, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow ul, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow ul, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow li, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow li, .cms .jstree .jstree-wholerow a, .cms .jstree .jstree-wholerow a:hover, .cms .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .cms .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow ul, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow li, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a:hover { margin: 0 !important; padding: 0 !important; }
|
||||
.cms .jstree .jstree-wholerow, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow { position: relative; z-index: 0; height: 0; background: transparent !important; }
|
||||
.cms .jstree .jstree-wholerow ul, .cms .jstree .jstree-wholerow li, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow ul, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow li { background: transparent !important; width: 100%; }
|
||||
.cms .jstree .jstree-wholerow a, .cms .jstree .jstree-wholerow a:hover, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a, .TreeDropdownField .treedropdownfield-panel .jstree .jstree-wholerow a:hover { text-indent: -9999px !important; width: 100%; border-right-width: 0px !important; border-left-width: 0px !important; }
|
||||
@ -770,7 +771,7 @@ form.import-form label.left { width: 250px; }
|
||||
.cms .jstree-themeroller a, .TreeDropdownField .treedropdownfield-panel .jstree-themeroller a { padding: 0 2px; }
|
||||
.cms .jstree-themeroller .ui-icon, .TreeDropdownField .treedropdownfield-panel .jstree-themeroller .ui-icon { overflow: visible; }
|
||||
.cms .jstree-themeroller .jstree-no-icon, .TreeDropdownField .treedropdownfield-panel .jstree-themeroller .jstree-no-icon { display: none; }
|
||||
.cms #jstree-marker, .cms .TreeDropdownField .treedropdownfield-panel #jstree-marker, .TreeDropdownField .treedropdownfield-panel .cms #jstree-marker, .cms #jstree-marker-line, .cms .TreeDropdownField .treedropdownfield-panel #jstree-marker-line, .TreeDropdownField .treedropdownfield-panel .cms #jstree-marker-line, .TreeDropdownField .treedropdownfield-panel .cms #jstree-marker, .cms .TreeDropdownField .treedropdownfield-panel #jstree-marker, .TreeDropdownField .treedropdownfield-panel #jstree-marker, .TreeDropdownField .treedropdownfield-panel .cms #jstree-marker-line, .cms .TreeDropdownField .treedropdownfield-panel #jstree-marker-line, .TreeDropdownField .treedropdownfield-panel #jstree-marker-line { padding: 0; margin: 0; overflow: hidden; position: absolute; top: -30px; background-repeat: no-repeat; display: none; }
|
||||
.cms #jstree-marker, .cms .TreeDropdownField .treedropdownfield-panel #jstree-marker, .TreeDropdownField .treedropdownfield-panel .cms #jstree-marker, .cms #jstree-marker-line, .cms .TreeDropdownField .treedropdownfield-panel #jstree-marker-line, .TreeDropdownField .treedropdownfield-panel .cms #jstree-marker-line, .TreeDropdownField .treedropdownfield-panel #jstree-marker, .TreeDropdownField .treedropdownfield-panel #jstree-marker-line { padding: 0; margin: 0; overflow: hidden; position: absolute; top: -30px; background-repeat: no-repeat; display: none; }
|
||||
.cms #jstree-marker, .TreeDropdownField .treedropdownfield-panel #jstree-marker { line-height: 10px; font-size: 12px; height: 12px; width: 8px; z-index: 10001; background-color: transparent; text-shadow: 1px 1px 1px white; color: black; }
|
||||
.cms #jstree-marker-line, .TreeDropdownField .treedropdownfield-panel #jstree-marker-line { line-height: 0%; font-size: 1px; height: 1px; width: 100px; z-index: 10000; background-color: #456c43; cursor: pointer; border: 1px solid #eeeeee; border-left: 0; -moz-box-shadow: 0px 0px 2px #666; -webkit-box-shadow: 0px 0px 2px #666; box-shadow: 0px 0px 2px #666; -moz-border-radius: 1px; border-radius: 1px; -webkit-border-radius: 1px; }
|
||||
.cms #vakata-contextmenu, .TreeDropdownField .treedropdownfield-panel #vakata-contextmenu { display: block; visibility: hidden; left: 0; top: -200px; position: absolute; margin: 0; padding: 0; min-width: 180px; background: #FFF; border: 1px solid silver; z-index: 10000; *width: 180px; -webkit-box-shadow: 0 0 10px #cccccc; -moz-box-shadow: 0 0 10px #cccccc; box-shadow: 0 0 10px #cccccc; }
|
||||
@ -813,7 +814,7 @@ form.import-form label.left { width: 250px; }
|
||||
.tree-holder.jstree-apple span.badge.status-deletedonlive, .tree-holder.jstree-apple span.badge.status-removedfromdraft, .cms-tree.jstree-apple span.badge.status-deletedonlive, .cms-tree.jstree-apple span.badge.status-removedfromdraft { color: #636363; border: 1px solid #E49393; background-color: #F2DADB; }
|
||||
.tree-holder.jstree-apple span.badge.status-workflow-approval, .cms-tree.jstree-apple span.badge.status-workflow-approval { color: #56660C; border: 1px solid #7C8816; background-color: #DAE79A; }
|
||||
.tree-holder.jstree-apple span.comment-count, .cms-tree.jstree-apple span.comment-count { clear: both; position: relative; text-transform: uppercase; display: inline-block; overflow: visible; padding: 0px 3px; font-size: 0.75em; line-height: 1em; margin-left: 3px; margin-right: 6px; -webkit-border-radius: 2px 2px; -moz-border-radius: 2px / 2px; border-radius: 2px / 2px; color: #7E7470; border: 1px solid #C9B800; background-color: #FFF0BC; }
|
||||
.tree-holder.jstree-apple span.comment-count span.comment-count:before, .tree-holder.jstree-apple span.comment-count .cms-tree.jstree-apple span.comment-count:before, .cms-tree.jstree-apple .tree-holder.jstree-apple span.comment-count span.comment-count:before, .tree-holder.jstree-apple span.comment-count span.comment-count:after, .tree-holder.jstree-apple span.comment-count .cms-tree.jstree-apple span.comment-count:after, .cms-tree.jstree-apple .tree-holder.jstree-apple span.comment-count span.comment-count:after, .cms-tree.jstree-apple span.comment-count .tree-holder.jstree-apple span.comment-count:before, .tree-holder.jstree-apple .cms-tree.jstree-apple span.comment-count span.comment-count:before, .cms-tree.jstree-apple span.comment-count span.comment-count:before, .cms-tree.jstree-apple span.comment-count .tree-holder.jstree-apple span.comment-count:after, .tree-holder.jstree-apple .cms-tree.jstree-apple span.comment-count span.comment-count:after, .cms-tree.jstree-apple span.comment-count span.comment-count:after { content: ""; position: absolute; border-style: solid; /* reduce the damage in FF3.0 */ display: block; width: 0; }
|
||||
.tree-holder.jstree-apple span.comment-count span.comment-count:before, .tree-holder.jstree-apple span.comment-count span.comment-count:after, .cms-tree.jstree-apple span.comment-count span.comment-count:before, .cms-tree.jstree-apple span.comment-count span.comment-count:after { content: ""; position: absolute; border-style: solid; /* reduce the damage in FF3.0 */ display: block; width: 0; }
|
||||
.tree-holder.jstree-apple span.comment-count:before, .cms-tree.jstree-apple span.comment-count:before { bottom: -4px; /* value = - border-top-width - border-bottom-width */ left: 3px; /* controls horizontal position */ border-width: 4px 4px 0; border-color: #C9B800 transparent; }
|
||||
.tree-holder.jstree-apple span.comment-count:after, .cms-tree.jstree-apple span.comment-count:after { bottom: -3px; /* value = - border-top-width - border-bottom-width */ left: 4px; /* value = (:before left) + (:before border-left) - (:after border-left) */ border-width: 3px 3px 0; border-color: #FFF0BC transparent; }
|
||||
.tree-holder.jstree-apple .jstree-hovered, .cms-tree.jstree-apple .jstree-hovered { text-shadow: none; text-decoration: none; }
|
||||
|
@ -5,10 +5,23 @@ jQuery.noConflict();
|
||||
*/
|
||||
(function($) {
|
||||
|
||||
window.onresize = function(e) {
|
||||
var windowWidth, windowHeight;
|
||||
$(window).bind('resize.leftandmain', function(e) {
|
||||
// Entwine's 'fromWindow::onresize' does not trigger on IE8. Use synthetic event.
|
||||
$('.cms-container').trigger('windowresize');
|
||||
};
|
||||
var cb = function() {$('.cms-container').trigger('windowresize');};
|
||||
|
||||
// Workaround to avoid IE8 infinite loops when elements are resized as a result of this event
|
||||
if($.browser.msie && parseInt($.browser.version, 10) < 9) {
|
||||
var newWindowWidth = $(window).width(), newWindowHeight = $(window).height();
|
||||
if(newWindowWidth != windowWidth || newWindowHeight != windowHeight) {
|
||||
windowWidth = newWindowWidth;
|
||||
windowHeight = newWindowHeight;
|
||||
cb();
|
||||
}
|
||||
} else {
|
||||
cb();
|
||||
}
|
||||
});
|
||||
|
||||
// setup jquery.entwine
|
||||
$.entwine.warningLevel = $.entwine.WARN_LEVEL_BESTPRACTISE;
|
||||
|
@ -153,7 +153,7 @@
|
||||
iframe.bind('load', function(e) {
|
||||
if($(this).attr('src') == 'about:blank') return;
|
||||
|
||||
$(this).show();
|
||||
iframe.addClass('loaded').show(); // more reliable than 'src' attr check (in IE)
|
||||
self._resizeIframe();
|
||||
self.uiDialog.removeClass('loading');
|
||||
}).hide();
|
||||
@ -170,7 +170,7 @@
|
||||
var self = this, iframe = this.element.children('iframe');
|
||||
|
||||
// Load iframe
|
||||
if(!iframe.attr('src') || this.options.reloadOnOpen) {
|
||||
if(this.options.iframeUrl && (!iframe.hasClass('loaded') || this.options.reloadOnOpen)) {
|
||||
iframe.hide();
|
||||
iframe.attr('src', this.options.iframeUrl);
|
||||
this.uiDialog.addClass('loading');
|
||||
@ -186,7 +186,7 @@
|
||||
$(window).unbind('resize.ssdialog');
|
||||
},
|
||||
_resizeIframe: function() {
|
||||
var opts = {}, newWidth, newHeight;
|
||||
var opts = {}, newWidth, newHeight, iframe = this.element.children('iframe');;
|
||||
if(this.options.widthRatio) {
|
||||
newWidth = $(window).width() * this.options.widthRatio;
|
||||
if(this.options.minWidth && newWidth < this.options.minWidth) {
|
||||
@ -207,11 +207,26 @@
|
||||
opts.height = newHeight;
|
||||
}
|
||||
}
|
||||
if(this.options.autoPosition) {
|
||||
opts.position = this.options.position;
|
||||
}
|
||||
|
||||
if(!jQuery.isEmptyObject(opts)) {
|
||||
this._setOptions(opts);
|
||||
|
||||
// Resize iframe within dialog
|
||||
iframe.attr('width',
|
||||
opts.width
|
||||
- parseFloat(this.element.css('paddingLeft'))
|
||||
- parseFloat(this.element.css('paddingRight'))
|
||||
);
|
||||
iframe.attr('height',
|
||||
opts.height
|
||||
- parseFloat(this.element.css('paddingTop'))
|
||||
- parseFloat(this.element.css('paddingBottom'))
|
||||
);
|
||||
|
||||
// Enforce new position
|
||||
if(this.options.autoPosition) {
|
||||
this._setOption("position", this.options.position);
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
|
@ -529,6 +529,8 @@ body.cms {
|
||||
}
|
||||
ul.SelectionGroup {
|
||||
padding-left:28px;
|
||||
overflow: visible;
|
||||
@include legacy-pie-clearfix;
|
||||
}
|
||||
.parent-mode {
|
||||
padding: $grid-x;
|
||||
@ -1533,8 +1535,13 @@ body.cms-dialog {
|
||||
}
|
||||
}
|
||||
|
||||
.ss-uploadfield-editandorganize {
|
||||
display: none;
|
||||
.ss-assetuploadfield .ss-uploadfield-editandorganize {
|
||||
.ss-uploadfield-files {
|
||||
.ss-uploadfield-item-info {
|
||||
background-color: grayscale(#5db4df);
|
||||
@include background-image(linear-gradient(top, grayscale(#5db4df) 0%, grayscale(#5db1dd) 8%, grayscale(#439bcb) 50%, grayscale(#3f99cd) 54%, grayscale(#207db6) 96%, grayscale(#1e7cba) 100%));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -75,7 +75,7 @@ require_once("model/DB.php");
|
||||
if(!isset($databaseConfig) || !isset($databaseConfig['database']) || !$databaseConfig['database']) {
|
||||
echo "\nPlease configure your database connection details. You can do this by creating a file
|
||||
called _ss_environment.php in either of the following locations:\n\n";
|
||||
echo " - " . BASE_PATH ."_ss_environment.php\n - " . dirname(BASE_PATH) . "_ss_environment.php\n\n";
|
||||
echo " - " . BASE_PATH . DIRECTORY_SEPARATOR . "_ss_environment.php\n - " . dirname(BASE_PATH) . DIRECTORY_SEPARATOR . "_ss_environment.php\n\n";
|
||||
echo <<<ENVCONTENT
|
||||
|
||||
Put the following content into this file:
|
||||
|
@ -17,6 +17,8 @@
|
||||
* - SS_DATABASE_SUFFIX: A suffix to add to the database name.
|
||||
* - SS_DATABASE_PREFIX: A prefix to add to the database name.
|
||||
* - SS_DATABASE_TIMEZONE: Set the database timezone to something other than the system timezone.
|
||||
* - SS_DATABASE_MEMORY: Use in-memory state if possible. Useful for testing, currently only
|
||||
* supported by the SQLite database adapter.
|
||||
*
|
||||
* There is one more setting that is intended to be used by people who work on SilverStripe.
|
||||
* - SS_DATABASE_CHOOSE_NAME: Boolean/Int. If set, then the system will choose a default database name for you if
|
||||
@ -110,6 +112,10 @@ if(defined('SS_DATABASE_USERNAME') && defined('SS_DATABASE_PASSWORD')) {
|
||||
// For schema enabled drivers:
|
||||
if(defined('SS_DATABASE_SCHEMA'))
|
||||
$databaseConfig["schema"] = SS_DATABASE_SCHEMA;
|
||||
|
||||
// For SQlite3 memory databases (mainly for testing purposes)
|
||||
if(defined('SS_DATABASE_MEMORY'))
|
||||
$databaseConfig["memory"] = SS_DATABASE_MEMORY;
|
||||
}
|
||||
|
||||
if(defined('SS_SEND_ALL_EMAILS_TO')) {
|
||||
|
@ -458,6 +458,8 @@ class Controller extends RequestHandler implements TemplateGlobalProvider {
|
||||
|
||||
/**
|
||||
* Redirect to the given URL.
|
||||
*
|
||||
* @return SS_HTTPResponse
|
||||
*/
|
||||
public function redirect($url, $code=302) {
|
||||
if(!$this->response) $this->response = new SS_HTTPResponse();
|
||||
@ -473,7 +475,7 @@ class Controller extends RequestHandler implements TemplateGlobalProvider {
|
||||
$url = Director::baseURL() . $url;
|
||||
}
|
||||
|
||||
$this->response->redirect($url, $code);
|
||||
return $this->response->redirect($url, $code);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -97,6 +97,16 @@ class RequestHandler extends ViewableData {
|
||||
*/
|
||||
private static $allowed_actions = null;
|
||||
|
||||
/**
|
||||
* @config
|
||||
* @var boolean Enforce presence of $allowed_actions when checking acccess.
|
||||
* Defaults to TRUE, meaning all URL actions will be denied.
|
||||
* When set to FALSE, the controller will allow *all* public methods to be called.
|
||||
* In most cases this isn't desireable, and in fact a security risk,
|
||||
* since some helper methods can cause side effects which shouldn't be exposed through URLs.
|
||||
*/
|
||||
private static $require_allowed_actions = true;
|
||||
|
||||
public function __construct() {
|
||||
$this->brokenOnConstruct = false;
|
||||
|
||||
@ -430,12 +440,12 @@ class RequestHandler extends ViewableData {
|
||||
// If defined as empty array, deny action
|
||||
$isAllowed = false;
|
||||
} elseif($allowedActions === null) {
|
||||
// If undefined, allow action
|
||||
$isAllowed = true;
|
||||
// If undefined, allow action based on configuration
|
||||
$isAllowed = !Config::inst()->get('RequestHandler', 'require_allowed_actions');
|
||||
}
|
||||
|
||||
// If we don't have a match in allowed_actions,
|
||||
// whitelist the 'index' action as well as undefined actions.
|
||||
// whitelist the 'index' action as well as undefined actions based on configuration.
|
||||
if(!$isDefined && ($action == 'index' || empty($action))) {
|
||||
$isAllowed = true;
|
||||
}
|
||||
|
@ -86,7 +86,7 @@
|
||||
class Session {
|
||||
|
||||
/**
|
||||
* @var $timeout Set session timeout
|
||||
* @var $timeout Set session timeout in seconds.
|
||||
* @config
|
||||
*/
|
||||
private static $timeout = 0;
|
||||
@ -523,9 +523,11 @@ class Session {
|
||||
|
||||
if(!session_id() && !headers_sent()) {
|
||||
if($domain) {
|
||||
session_set_cookie_params($timeout, $path, $domain, $secure /* secure */, true /* httponly */);
|
||||
session_set_cookie_params($timeout, $path, $domain,
|
||||
$secure /* secure */, true /* httponly */);
|
||||
} else {
|
||||
session_set_cookie_params($timeout, $path, null, $secure /* secure */, true /* httponly */);
|
||||
session_set_cookie_params($timeout, $path, null,
|
||||
$secure /* secure */, true /* httponly */);
|
||||
}
|
||||
|
||||
// Allow storing the session in a non standard location
|
||||
@ -541,7 +543,8 @@ class Session {
|
||||
// Modify the timeout behaviour so it's the *inactive* time before the session expires.
|
||||
// By default it's the total session lifetime
|
||||
if($timeout && !headers_sent()) {
|
||||
Cookie::set(session_name(), session_id(), time()+$timeout, $path, $domain ? $domain : null, $secure, true);
|
||||
Cookie::set(session_name(), session_id(), $timeout/86400, $path, $domain ? $domain
|
||||
: null, $secure, true);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -295,8 +295,12 @@ class Config {
|
||||
* instead, the last manifest to be added always wins
|
||||
*/
|
||||
public function pushConfigYamlManifest(SS_ConfigManifest $manifest) {
|
||||
array_unshift($this->manifests, $manifest->yamlConfig);
|
||||
array_unshift($this->manifests, $manifest);
|
||||
|
||||
// Now that we've got another yaml config manifest we need to clean the cache
|
||||
$this->cache->clean();
|
||||
// We also need to clean the cache if the manifest's calculated config values change
|
||||
$manifest->registerChangeCallback(array($this->cache, 'clean'));
|
||||
|
||||
// @todo: Do anything with these. They're for caching after config.php has executed
|
||||
$this->collectConfigPHPSettings = true;
|
||||
@ -479,10 +483,13 @@ class Config {
|
||||
}
|
||||
}
|
||||
|
||||
$value = $nothing = null;
|
||||
|
||||
// Then the manifest values
|
||||
foreach($this->manifests as $manifest) {
|
||||
if (isset($manifest[$class][$name])) {
|
||||
self::merge_low_into_high($result, $manifest[$class][$name], $suppress);
|
||||
$value = $manifest->get($class, $name, $nothing);
|
||||
if ($value !== $nothing) {
|
||||
self::merge_low_into_high($result, $value, $suppress);
|
||||
if ($result !== null && !is_array($result)) return $result;
|
||||
}
|
||||
}
|
||||
|
@ -80,7 +80,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
* @param int $page
|
||||
*/
|
||||
public function setCurrentPage($page) {
|
||||
$this->pageStart = ($page - 1) * $this->pageLength;
|
||||
$this->pageStart = ($page - 1) * $this->getPageLength();
|
||||
return $this;
|
||||
}
|
||||
|
||||
@ -91,8 +91,8 @@ class PaginatedList extends SS_ListDecorator {
|
||||
*/
|
||||
public function getPageStart() {
|
||||
if ($this->pageStart === null) {
|
||||
if ($this->request && isset($this->request[$this->getVar])) {
|
||||
$this->pageStart = (int) $this->request[$this->getVar];
|
||||
if ($this->request && isset($this->request[$this->getPaginationGetVar()])) {
|
||||
$this->pageStart = (int) $this->request[$this->getPaginationGetVar()];
|
||||
} else {
|
||||
$this->pageStart = 0;
|
||||
}
|
||||
@ -181,7 +181,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
if($this->limitItems) {
|
||||
$tmptList = clone $this->list;
|
||||
return new IteratorIterator(
|
||||
$tmptList->limit($this->pageLength, $this->getPageStart())
|
||||
$tmptList->limit($this->getPageLength(), $this->getPageStart())
|
||||
);
|
||||
} else {
|
||||
return new IteratorIterator($this->list);
|
||||
@ -223,7 +223,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
for ($i = $start; $i < $end; $i++) {
|
||||
$result->push(new ArrayData(array(
|
||||
'PageNum' => $i + 1,
|
||||
'Link' => HTTP::setGetVar($this->getVar, $i * $this->pageLength),
|
||||
'Link' => HTTP::setGetVar($this->getPaginationGetVar(), $i * $this->getPageLength()),
|
||||
'CurrentBool' => $this->CurrentPage() == ($i + 1)
|
||||
)));
|
||||
}
|
||||
@ -292,7 +292,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
}
|
||||
|
||||
for ($i = 0; $i < $total; $i++) {
|
||||
$link = HTTP::setGetVar($this->getVar, $i * $this->pageLength);
|
||||
$link = HTTP::setGetVar($this->getPaginationGetVar(), $i * $this->getPageLength());
|
||||
$num = $i + 1;
|
||||
|
||||
$emptyRange = $num != 1 && $num != $total && (
|
||||
@ -321,14 +321,14 @@ class PaginatedList extends SS_ListDecorator {
|
||||
* @return int
|
||||
*/
|
||||
public function CurrentPage() {
|
||||
return floor($this->getPageStart() / $this->pageLength) + 1;
|
||||
return floor($this->getPageStart() / $this->getPageLength()) + 1;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return int
|
||||
*/
|
||||
public function TotalPages() {
|
||||
return ceil($this->getTotalItems() / $this->pageLength);
|
||||
return ceil($this->getTotalItems() / $this->getPageLength());
|
||||
}
|
||||
|
||||
/**
|
||||
@ -369,9 +369,9 @@ class PaginatedList extends SS_ListDecorator {
|
||||
*/
|
||||
public function LastItem() {
|
||||
if ($start = $this->getPageStart()) {
|
||||
return min($start + $this->pageLength, $this->getTotalItems());
|
||||
return min($start + $this->getPageLength(), $this->getTotalItems());
|
||||
} else {
|
||||
return min($this->pageLength, $this->getTotalItems());
|
||||
return min($this->getPageLength(), $this->getTotalItems());
|
||||
}
|
||||
}
|
||||
|
||||
@ -381,7 +381,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
* @return string
|
||||
*/
|
||||
public function FirstLink() {
|
||||
return HTTP::setGetVar($this->getVar, 0);
|
||||
return HTTP::setGetVar($this->getPaginationGetVar(), 0);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -390,7 +390,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
* @return string
|
||||
*/
|
||||
public function LastLink() {
|
||||
return HTTP::setGetVar($this->getVar, ($this->TotalPages() - 1) * $this->pageLength);
|
||||
return HTTP::setGetVar($this->getPaginationGetVar(), ($this->TotalPages() - 1) * $this->getPageLength());
|
||||
}
|
||||
|
||||
/**
|
||||
@ -401,7 +401,7 @@ class PaginatedList extends SS_ListDecorator {
|
||||
*/
|
||||
public function NextLink() {
|
||||
if ($this->NotLastPage()) {
|
||||
return HTTP::setGetVar($this->getVar, $this->getPageStart() + $this->pageLength);
|
||||
return HTTP::setGetVar($this->getPaginationGetVar(), $this->getPageStart() + $this->getPageLength());
|
||||
}
|
||||
}
|
||||
|
||||
@ -413,8 +413,8 @@ class PaginatedList extends SS_ListDecorator {
|
||||
*/
|
||||
public function PrevLink() {
|
||||
if ($this->NotFirstPage()) {
|
||||
return HTTP::setGetVar($this->getVar, $this->getPageStart() - $this->pageLength);
|
||||
return HTTP::setGetVar($this->getPaginationGetVar(), $this->getPageStart() - $this->getPageLength());
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
@ -9,6 +9,15 @@
|
||||
*/
|
||||
class SS_ConfigManifest {
|
||||
|
||||
/** @var string - The base path used when building the manifest */
|
||||
protected $base;
|
||||
|
||||
/** @var string - A string to prepend to all cache keys to ensure all keys are unique to just this $base */
|
||||
protected $key;
|
||||
|
||||
/** @var bool - Whether `test` directories should be searched when searching for configuration */
|
||||
protected $includeTests;
|
||||
|
||||
/**
|
||||
* All the values needed to be collected to determine the correct combination of fragements for
|
||||
* the current environment.
|
||||
@ -34,6 +43,17 @@ class SS_ConfigManifest {
|
||||
*/
|
||||
public $yamlConfig = array();
|
||||
|
||||
/**
|
||||
* The variant key state as when yamlConfig was loaded
|
||||
* @var string
|
||||
*/
|
||||
protected $yamlConfigVariantKey = null;
|
||||
|
||||
/**
|
||||
* @var [callback] A list of callbacks to be called whenever the content of yamlConfig changes
|
||||
*/
|
||||
protected $configChangeCallbacks = array();
|
||||
|
||||
/**
|
||||
* A side-effect of collecting the _config fragments is the calculation of all module directories, since
|
||||
* the definition of a module is "a directory that contains either a _config.php file or a _config directory
|
||||
@ -64,6 +84,8 @@ class SS_ConfigManifest {
|
||||
*/
|
||||
public function __construct($base, $includeTests = false, $forceRegen = false ) {
|
||||
$this->base = $base;
|
||||
$this->key = sha1($base).'_';
|
||||
$this->includeTests = $includeTests;
|
||||
|
||||
// Get the Zend Cache to load/store cache into
|
||||
$this->cache = SS_Cache::factory('SS_Configuration', 'Core', array(
|
||||
@ -74,24 +96,31 @@ class SS_ConfigManifest {
|
||||
// Unless we're forcing regen, try loading from cache
|
||||
if (!$forceRegen) {
|
||||
// The PHP config sources are always needed
|
||||
$this->phpConfigSources = $this->cache->load('php_config_sources');
|
||||
$this->phpConfigSources = $this->cache->load($this->key.'php_config_sources');
|
||||
// Get the variant key spec
|
||||
$this->variantKeySpec = $this->cache->load('variant_key_spec');
|
||||
// Try getting the pre-filtered & merged config for this variant
|
||||
if (!($this->yamlConfig = $this->cache->load('yaml_config_'.$this->variantKey()))) {
|
||||
// Otherwise, if we do have the yaml config fragments (and we should since we have a variant key spec)
|
||||
// work out the config for this variant
|
||||
if ($this->yamlConfigFragments = $this->cache->load('yaml_config_fragments')) {
|
||||
$this->buildYamlConfigVariant();
|
||||
}
|
||||
}
|
||||
$this->variantKeySpec = $this->cache->load($this->key.'variant_key_spec');
|
||||
}
|
||||
|
||||
// If we don't have a config yet, we need to do a full regen to get it
|
||||
if (!$this->yamlConfig) {
|
||||
// If we don't have a variantKeySpec (because we're forcing regen, or it just wasn't in the cache), generate it
|
||||
if (!$this->variantKeySpec) {
|
||||
$this->regenerate($includeTests);
|
||||
}
|
||||
|
||||
// At this point $this->variantKeySpec will always contain something valid, so we can build the variant
|
||||
$this->buildYamlConfigVariant();
|
||||
}
|
||||
|
||||
/**
|
||||
* Register a callback to be called whenever the calculated merged config changes
|
||||
*
|
||||
* In some situations the merged config can change - for instance, code in _config.php can cause which Only
|
||||
* and Except fragments match. Registering a callback with this function allows code to be called when
|
||||
* this happens.
|
||||
*
|
||||
* @param callback $callback
|
||||
*/
|
||||
public function registerChangeCallback($callback) {
|
||||
$this->configChangeCallbacks[] = $callback;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -102,6 +131,22 @@ class SS_ConfigManifest {
|
||||
foreach ($this->phpConfigSources as $config) {
|
||||
require_once $config;
|
||||
}
|
||||
|
||||
if ($this->variantKey() != $this->yamlConfigVariantKey) $this->buildYamlConfigVariant();
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the (merged) config value for the given class and config property name
|
||||
*
|
||||
* @param string $class - The class to get the config property value for
|
||||
* @param string $name - The config property to get the value for
|
||||
* @param any $default - What to return if no value was contained in any YAML file for the passed $class and $name
|
||||
* @return any - The merged set of all values contained in all the YAML configuration files for the passed
|
||||
* $class and $name, or $default if there are none
|
||||
*/
|
||||
public function get($class, $name, $default=null) {
|
||||
if (isset($this->yamlConfig[$class][$name])) return $this->yamlConfig[$class][$name];
|
||||
else return $default;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -165,9 +210,9 @@ class SS_ConfigManifest {
|
||||
$this->buildVariantKeySpec();
|
||||
|
||||
if ($cache) {
|
||||
$this->cache->save($this->phpConfigSources, 'php_config_sources');
|
||||
$this->cache->save($this->yamlConfigFragments, 'yaml_config_fragments');
|
||||
$this->cache->save($this->variantKeySpec, 'variant_key_spec');
|
||||
$this->cache->save($this->phpConfigSources, $this->key.'php_config_sources');
|
||||
$this->cache->save($this->yamlConfigFragments, $this->key.'yaml_config_fragments');
|
||||
$this->cache->save($this->variantKeySpec, $this->key.'variant_key_spec');
|
||||
}
|
||||
}
|
||||
|
||||
@ -395,10 +440,17 @@ class SS_ConfigManifest {
|
||||
$matchingFragments = array();
|
||||
|
||||
foreach ($this->yamlConfigFragments as $i => $fragment) {
|
||||
$failsonly = isset($fragment['only']) && !$this->matchesPrefilterVariantRules($fragment['only']);
|
||||
$matchesexcept = isset($fragment['except']) && $this->matchesPrefilterVariantRules($fragment['except']);
|
||||
$matches = true;
|
||||
|
||||
if (!$failsonly && !$matchesexcept) $matchingFragments[] = $fragment;
|
||||
if (isset($fragment['only'])) {
|
||||
$matches = $matches && ($this->matchesPrefilterVariantRules($fragment['only']) !== false);
|
||||
}
|
||||
|
||||
if (isset($fragment['except'])) {
|
||||
$matches = $matches && ($this->matchesPrefilterVariantRules($fragment['except']) !== true);
|
||||
}
|
||||
|
||||
if ($matches) $matchingFragments[] = $fragment;
|
||||
}
|
||||
|
||||
$this->yamlConfigFragments = $matchingFragments;
|
||||
@ -413,22 +465,26 @@ class SS_ConfigManifest {
|
||||
* which values means accept or reject a fragment
|
||||
*/
|
||||
public function matchesPrefilterVariantRules($rules) {
|
||||
$matches = "undefined"; // Needs to be truthy, but not true
|
||||
|
||||
foreach ($rules as $k => $v) {
|
||||
switch (strtolower($k)) {
|
||||
case 'classexists':
|
||||
if (!ClassInfo::exists($v)) return false;
|
||||
$matches = $matches && ClassInfo::exists($v);
|
||||
break;
|
||||
|
||||
case 'moduleexists':
|
||||
if (!$this->moduleExists($v)) return false;
|
||||
$matches = $matches && $this->moduleExists($v);
|
||||
break;
|
||||
|
||||
default:
|
||||
// NOP
|
||||
}
|
||||
|
||||
if ($matches === false) return $matches;
|
||||
}
|
||||
|
||||
return true;
|
||||
return $matches;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -481,26 +537,61 @@ class SS_ConfigManifest {
|
||||
/**
|
||||
* Calculates which yaml config fragments are applicable in this variant, and merge those all together into
|
||||
* the $this->yamlConfig propperty
|
||||
*
|
||||
* Checks cache and takes care of loading yamlConfigFragments if they aren't already present, but expects
|
||||
* $variantKeySpec to already be set
|
||||
*/
|
||||
public function buildYamlConfigVariant($cache = true) {
|
||||
// Only try loading from cache if we don't have the fragments already loaded, as there's no way to know if a
|
||||
// given variant is stale compared to the complete set of fragments
|
||||
if (!$this->yamlConfigFragments) {
|
||||
// First try and just load the exact variant
|
||||
if ($this->yamlConfig = $this->cache->load($this->key.'yaml_config_'.$this->variantKey())) {
|
||||
$this->yamlConfigVariantKey = $this->variantKey();
|
||||
return;
|
||||
}
|
||||
// Otherwise try and load the fragments so we can build the variant
|
||||
else {
|
||||
$this->yamlConfigFragments = $this->cache->load($this->key.'yaml_config_fragments');
|
||||
}
|
||||
}
|
||||
|
||||
// If we still don't have any fragments we have to build them
|
||||
if (!$this->yamlConfigFragments) {
|
||||
$this->regenerate($this->includeTests, $cache);
|
||||
}
|
||||
|
||||
$this->yamlConfig = array();
|
||||
$this->yamlConfigVariantKey = $this->variantKey();
|
||||
|
||||
foreach ($this->yamlConfigFragments as $i => $fragment) {
|
||||
$failsonly = isset($fragment['only']) && !$this->matchesVariantRules($fragment['only']);
|
||||
$matchesexcept = isset($fragment['except']) && $this->matchesVariantRules($fragment['except']);
|
||||
$matches = true;
|
||||
|
||||
if (!$failsonly && !$matchesexcept) $this->mergeInYamlFragment($this->yamlConfig, $fragment['fragment']);
|
||||
if (isset($fragment['only'])) {
|
||||
$matches = $matches && ($this->matchesVariantRules($fragment['only']) !== false);
|
||||
}
|
||||
|
||||
if (isset($fragment['except'])) {
|
||||
$matches = $matches && ($this->matchesVariantRules($fragment['except']) !== true);
|
||||
}
|
||||
|
||||
if ($matches) $this->mergeInYamlFragment($this->yamlConfig, $fragment['fragment']);
|
||||
}
|
||||
|
||||
if ($cache) {
|
||||
$this->cache->save($this->yamlConfig, 'yaml_config_'.$this->variantKey());
|
||||
$this->cache->save($this->yamlConfig, $this->key.'yaml_config_'.$this->variantKey());
|
||||
}
|
||||
|
||||
// Since yamlConfig has changed, call any callbacks that are interested
|
||||
foreach ($this->configChangeCallbacks as $callback) call_user_func($callback);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns false if the non-prefilterable parts of the rule aren't met, and true if they are
|
||||
*/
|
||||
public function matchesVariantRules($rules) {
|
||||
$matches = "undefined"; // Needs to be truthy, but not true
|
||||
|
||||
foreach ($rules as $k => $v) {
|
||||
switch (strtolower($k)) {
|
||||
case 'classexists':
|
||||
@ -510,13 +601,13 @@ class SS_ConfigManifest {
|
||||
case 'environment':
|
||||
switch (strtolower($v)) {
|
||||
case 'live':
|
||||
if (!Director::isLive()) return false;
|
||||
$matches = $matches && Director::isLive();
|
||||
break;
|
||||
case 'test':
|
||||
if (!Director::isTest()) return false;
|
||||
$matches = $matches && Director::isTest();
|
||||
break;
|
||||
case 'dev':
|
||||
if (!Director::isDev()) return false;
|
||||
$matches = $matches && Director::isDev();
|
||||
break;
|
||||
default:
|
||||
user_error('Unknown environment '.$v.' in config fragment', E_USER_ERROR);
|
||||
@ -524,21 +615,25 @@ class SS_ConfigManifest {
|
||||
break;
|
||||
|
||||
case 'envvarset':
|
||||
if (isset($_ENV[$k])) break;
|
||||
return false;
|
||||
$matches = $matches && isset($_ENV[$v]);
|
||||
break;
|
||||
|
||||
case 'constantdefined':
|
||||
if (defined($k)) break;
|
||||
return false;
|
||||
$matches = $matches && defined($v);
|
||||
break;
|
||||
|
||||
default:
|
||||
if (isset($_ENV[$k]) && $_ENV[$k] == $v) break;
|
||||
if (defined($k) && constant($k) == $v) break;
|
||||
return false;
|
||||
}
|
||||
$matches = $matches && (
|
||||
(isset($_ENV[$k]) && $_ENV[$k] == $v) ||
|
||||
(defined($k) && constant($k) == $v)
|
||||
);
|
||||
break;
|
||||
}
|
||||
|
||||
return true;
|
||||
if ($matches === false) return $matches;
|
||||
}
|
||||
|
||||
return $matches;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -78,7 +78,8 @@ class SS_ConfigStaticManifest {
|
||||
$static = $this->statics[$class][$name];
|
||||
|
||||
if ($static['access'] != T_PRIVATE) {
|
||||
Deprecation::notice('3.2.0', "Config static $class::\$$name must be marked as private", Deprecation::SCOPE_GLOBAL);
|
||||
Deprecation::notice('3.2.0', "Config static $class::\$$name must be marked as private",
|
||||
Deprecation::SCOPE_GLOBAL);
|
||||
// Don't warn more than once per static
|
||||
$this->statics[$class][$name]['access'] = T_PRIVATE;
|
||||
}
|
||||
@ -211,13 +212,23 @@ class SS_ConfigStaticManifest_Parser {
|
||||
$class = $next[1];
|
||||
}
|
||||
else if($type == T_NAMESPACE) {
|
||||
$namespace = '';
|
||||
while(true) {
|
||||
$next = $this->next();
|
||||
|
||||
if($next == ';') {
|
||||
break;
|
||||
} elseif($next[0] == T_NS_SEPARATOR) {
|
||||
$namespace .= $next[1];
|
||||
$next = $this->next();
|
||||
}
|
||||
|
||||
if($next[0] != T_STRING) {
|
||||
user_error("Couldn\'t parse {$this->path} when building config static manifest", E_USER_ERROR);
|
||||
}
|
||||
|
||||
$namespace = $next[1];
|
||||
$namespace .= $next[1];
|
||||
}
|
||||
}
|
||||
else if($type == '{' || $type == T_CURLY_OPEN || $type == T_DOLLAR_OPEN_CURLY_BRACES){
|
||||
$depth += 1;
|
||||
@ -288,7 +299,8 @@ class SS_ConfigStaticManifest_Parser {
|
||||
$depth -= 1;
|
||||
}
|
||||
|
||||
// Parse out the assignment side of a static declaration, ending on either a ';' or a ',' outside an array
|
||||
// Parse out the assignment side of a static declaration,
|
||||
// ending on either a ';' or a ',' outside an array
|
||||
if($type == T_WHITESPACE) {
|
||||
$value .= ' ';
|
||||
}
|
||||
|
@ -43,4 +43,4 @@ Used in side panels and action tabs
|
||||
.ss-uploadfield .ss-uploadfield-addfile.borderTop { border-top: 1px solid #b3b3b3; }
|
||||
|
||||
.ss-upload .clear { clear: both; }
|
||||
.ss-upload .ss-uploadfield-fromcomputer input { /* since we can't really style the file input, we use this hack to make it as big as the button and hide it */ position: absolute; top: 0; right: 0; margin: 0; border: solid #000; border-width: 0 0 100px 200px; opacity: 0; filter: alpha(opacity=0); -o-transform: translate(250px, -50px) scale(1); -moz-transform: translate(-300px, 0) scale(4); direction: ltr; cursor: pointer; }
|
||||
.ss-upload .ss-uploadfield-fromcomputer input { /* since we can't really style the file input, we use this hack to make it as big as the button and hide it */ position: absolute; top: 0; right: 0; margin: 0; opacity: 0; filter: alpha(opacity=0); transform: translate(-300px, 0) scale(4); font-size: 23px; direction: ltr; cursor: pointer; height: 30px; line-height: 30px; }
|
||||
|
6
dev/TestRunner.php
Normal file → Executable file
6
dev/TestRunner.php
Normal file → Executable file
@ -325,6 +325,12 @@ class TestRunner extends Controller {
|
||||
$phpunitwrapper->setSuite($suite);
|
||||
$phpunitwrapper->setCoverageStatus($coverage);
|
||||
|
||||
// Make sure TearDown is called (even in the case of a fatal error)
|
||||
$self = $this;
|
||||
register_shutdown_function(function() use ($self) {
|
||||
$self->tearDown();
|
||||
});
|
||||
|
||||
$phpunitwrapper->runTests();
|
||||
|
||||
// get results of the PhpUnitWrapper class
|
||||
|
@ -35,6 +35,7 @@
|
||||
* Optional integration with ImageMagick as a new image manipulation backend
|
||||
* Support for PHP 5.4's built-in webserver
|
||||
* Support for [Composer](http://getcomposer.org) dependency manager (also works with 3.0)
|
||||
* Added support for filtering incoming HTML from TinyMCE (disabled by default, see [security](/topics/security))
|
||||
|
||||
## Upgrading
|
||||
|
||||
@ -219,8 +220,7 @@ For more information about how to use the config system, see the ["Configuration
|
||||
|
||||
In order to make controller access checks more consistent and easier to
|
||||
understand, the routing will require definition of `$allowed_actions`
|
||||
on your own `Controller` subclasses if they contain any actions
|
||||
accessible through URLs, or any forms.
|
||||
on your own `Controller` subclasses if they contain any actions accessible through URLs.
|
||||
|
||||
:::php
|
||||
class MyController extends Controller {
|
||||
@ -228,8 +228,13 @@ accessible through URLs, or any forms.
|
||||
public function myaction($request) {}
|
||||
}
|
||||
|
||||
Please review all rules governing allowed actions in the
|
||||
["controller" topic](/topics/controller).
|
||||
You can overwrite the default behaviour on undefined `$allowed_actions` to allow all actions,
|
||||
by setting the `RequestHandler.require_allowed_actions` config value to `false` (not recommended).
|
||||
|
||||
This applies to anything extending `RequestHandler`, so please check your `Form` and `FormField`
|
||||
subclasses as well. Keep in mind, action methods as denoted through `FormAction` names should NOT
|
||||
be mentioned in `$allowed_actions` to avoid CSRF issues.
|
||||
Please review all rules governing allowed actions in the ["controller" topic](/topics/controller).
|
||||
|
||||
### Removed support for "*" rules in `Controller::$allowed_actions`
|
||||
|
||||
|
@ -44,7 +44,7 @@ First we create all the fields we want in the contact form, and put them inside
|
||||
We then create a `[api:FieldList]` of the form actions, or the buttons that submit the form. Here we add a single form action, with the name 'submit', and the label 'Submit'. We'll use the name of the form action later.
|
||||
|
||||
:::php
|
||||
return new Form('Form', $this, $fields, $actions);
|
||||
return new Form($this, 'Form', $fields, $actions);
|
||||
|
||||
Finally we create the `Form` object and return it. The first argument is the name of the form – this has to be the same as the name of the function that creates the form, so we've used 'Form'. The second argument is the controller that the form is on – this is almost always $this. The third and fourth arguments are the fields and actions we created earlier.
|
||||
|
||||
|
@ -85,6 +85,25 @@ there are any problems they will follow up with you, so please ensure they have
|
||||
|
||||
![Workflow diagram](http://www.silverstripe.org/assets/doc-silverstripe-org/collaboration-on-github.png)
|
||||
|
||||
### Quickfire Do's and Don't's
|
||||
|
||||
If you aren't familiar with git and GitHub, try reading the ["GitHub bootcamp documentation"](http://help.github.com/).
|
||||
We also found the [free online git book](http://progit.org/book/) and the [git crash course](http://gitref.org/) useful.
|
||||
If you're familiar with it, here's the short version of what you need to know. Once you fork and download the code:
|
||||
|
||||
* **Don't develop on the master branch.** Always create a development branch specific to "the issue" you're working on (mostly on our [bugtracker](/misc/contributing/issues)). Name it by issue number and description. For example, if you're working on Issue #100, a `DataObject::get_one()` bugfix, your development branch should be called 100-dataobject-get-one. If you decide to work on another issue mid-stream, create a new branch for that issue--don't work on both in one branch.
|
||||
|
||||
* **Do not merge the upstream master** with your development branch; *rebase* your branch on top of the upstream master.
|
||||
|
||||
* **A single development branch should represent changes related to a single issue.** If you decide to work on another issue, create another branch.
|
||||
|
||||
* **Squash your commits, so that each commit addresses a single issue.** After you rebase your work on top of the upstream master, you can squash multiple commits into one. Say, for instance, you've got three commits in related to Issue #100. Squash all three into one with the message "Issue #100 Description of the issue here." We won't accept pull requests for multiple commits related to a single issue; it's up to you to squash and clean your commit tree. (Remember, if you squash commits you've already pushed to GitHub, you won't be able to push that same branch again. Create a new local branch, squash, and push the new squashed branch.)
|
||||
|
||||
* **Choose the correct branch**: Assume the current release is 3.0.3, and 3.1.0 is in beta state.
|
||||
Most pull requests should go against the `3.1.x-dev` *pre-release branch*, only critical bugfixes
|
||||
against the `3.0.x-dev` *release branch*. If you're changing an API or introducing a major feature,
|
||||
the pull request should go against `master` (read more about our [release process](/misc/release-process)). Branches are periodically merged "upwards" (3.0 into 3.1, 3.1 into master).
|
||||
|
||||
### Editing files directly on GitHub.com
|
||||
|
||||
If you see a typo or another small fix that needs to be made, and you don't have an installation set up for contributions, you can edit files directly in the github.com web interface. Every file view has an "edit this file" link.
|
||||
|
@ -1,7 +1,7 @@
|
||||
# GridField
|
||||
|
||||
GridField is SilverStripe's implementation of data grids. Its main purpose is to display tabular data
|
||||
in a format that is easy to view and modify. It's a can be thought of as a HTML table with some tricks.
|
||||
Gridfield is SilverStripe's implementation of data grids. Its main purpose is to display tabular data
|
||||
in a format that is easy to view and modify. It can be thought of as a HTML table with some tricks.
|
||||
|
||||
It's built in a way that provides developers with an extensible way to display tabular data in a
|
||||
table and minimise the amount of code that needs to be written.
|
||||
|
@ -173,7 +173,7 @@ the markup in the `else` clause is used, if that clause is present.
|
||||
:::ss
|
||||
<% if $MyDinner=="quiche" %>
|
||||
Real men don't eat quiche
|
||||
<% else_if $MyDinner=$YourDinner %>
|
||||
<% else_if $MyDinner==$YourDinner %>
|
||||
We both have good taste
|
||||
<% else %>
|
||||
Can I have some of your chips?
|
||||
|
@ -227,8 +227,8 @@ To accommodate this, value sections can be filtered to only be used when either
|
||||
current environment.
|
||||
|
||||
To achieve this you add a key to the related header section, either "Only" when the value section should be included
|
||||
only when the rules contained match, or "Except" when the value section should be included except when the rules
|
||||
contained match.
|
||||
only when all the rules contained match, or "Except" when the value section should be included except when all of the
|
||||
rules contained match.
|
||||
|
||||
You then list any of the following rules as sub-keys, with informational values as either a single value or a list.
|
||||
|
||||
@ -256,6 +256,15 @@ For instance, to add a property to "foo" when a module exists, and "bar" otherwi
|
||||
property: 'bar'
|
||||
---
|
||||
|
||||
Note than when you have more than one rule for a nested fragment, they're joined like
|
||||
|
||||
FRAGMENT_INCLUDED = (ONLY && ONLY) && !(EXCEPT && EXCEPT)
|
||||
|
||||
That is, the fragment will be included if all Only rules match, except if all Except rules match.
|
||||
|
||||
Also, due to YAML limitations, having multiple conditions of the same kind (say, two `EnvVarSet` in one "Only" block)
|
||||
will result in only the latter coming through.
|
||||
|
||||
### The values
|
||||
|
||||
The values section of YAML configuration files is quite simple - it is simply a nested key / value pair structure
|
||||
|
@ -87,9 +87,7 @@ way to perform checks against permission codes or custom logic.
|
||||
There's a couple of rules guiding these checks:
|
||||
|
||||
* Each class is only responsible for access control on the methods it defines
|
||||
* If `$allowed_actions` is defined as an empty array, no actions are allowed
|
||||
* If `$allowed_actions` is undefined, all public methods on the specific class are allowed
|
||||
(not recommended)
|
||||
* If `$allowed_actions` is an empty array or undefined, only the `index` action is allowed
|
||||
* Access checks on parent classes need to be overwritten via the Config API
|
||||
* Only public methods can be made accessible
|
||||
* If a method on a parent class is overwritten, access control for it has to be redefined as well
|
||||
@ -102,6 +100,8 @@ There's a couple of rules guiding these checks:
|
||||
* `$allowed_actions` can be defined on `Extension` classes applying to the controller.
|
||||
|
||||
If the permission check fails, SilverStripe will return a "403 Forbidden" HTTP status.
|
||||
You can overwrite the default behaviour on undefined `$allowed_actions` to allow all actions,
|
||||
by setting the `RequestHandler.require_allowed_actions` config value to `false` (not recommended).
|
||||
|
||||
### Through the action
|
||||
|
||||
@ -173,21 +173,6 @@ through `/fastfood/drivethrough/` to use the same order function.
|
||||
'drivethrough/$Action/$ID/$Name' => 'order'
|
||||
);
|
||||
|
||||
## Access Control
|
||||
|
||||
### Through $allowed_actions
|
||||
|
||||
* If `$allowed_actions` is undefined, `null` or `array()`, no actions are accessible
|
||||
* Each class is only responsible for access control on the methods it defines
|
||||
* Access checks on parent classes need to be overwritten via the Config API
|
||||
* Only public methods can be made accessible
|
||||
* If a method on a parent class is overwritten, access control for it has to be redefined as well
|
||||
* An action named "index" is whitelisted by default
|
||||
* Methods returning forms also count as actions which need to be defined
|
||||
* Form action methods (targets of `FormAction`) should NOT be included in `$allowed_actions`,
|
||||
they're handled separately through the form routing (see the ["forms" topic](/topics/forms))
|
||||
* `$allowed_actions` can be defined on `Extension` classes applying to the controller.
|
||||
|
||||
## URL Patterns
|
||||
|
||||
The `[api:RequestHandler]` class will parse all rules you specify against the
|
||||
|
@ -272,6 +272,15 @@ Some rules of thumb:
|
||||
* Don't concatenate URLs in a template. It only works in extremely simple cases that usually contain bugs.
|
||||
* Use *Controller::join_links()* to concatenate URLs. It deals with query strings and other such edge cases.
|
||||
|
||||
### Filtering incoming HTML from TinyMCE
|
||||
|
||||
In some cases you may be particularly concerned about which HTML elements are addable to Content via the CMS.
|
||||
By default, although TinyMCE is configured to restrict some dangerous tags (such as `script` tags), this restriction
|
||||
is not enforced server-side. A malicious user with write access to the CMS might create a specific request to avoid
|
||||
these restrictions.
|
||||
|
||||
To enable server side filtering using the same whitelisting controls as TinyMCE, set the
|
||||
HtmlEditorField::$sanitise_server_side config property to true.
|
||||
|
||||
## Cross-Site Request Forgery (CSRF)
|
||||
|
||||
|
@ -247,7 +247,10 @@ class ConfirmedPasswordField extends FormField {
|
||||
*
|
||||
* @return ConfirmedPasswordField
|
||||
*/
|
||||
public function setValue($value) {
|
||||
public function setValue($value, $data = null) {
|
||||
// If $data is a DataObject, don't use the value, since it's a hashed value
|
||||
if ($data && $data instanceof DataObject) $value = '';
|
||||
|
||||
if(is_array($value)) {
|
||||
if($value['_Password'] || (!$value['_Password'] && !$this->canBeEmpty)) {
|
||||
$this->value = $value['_Password'];
|
||||
|
@ -149,6 +149,12 @@ class Form extends RequestHandler {
|
||||
*/
|
||||
protected $attributes = array();
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'handleField',
|
||||
'httpSubmission',
|
||||
'forTemplate',
|
||||
);
|
||||
|
||||
/**
|
||||
* @var FormTemplateHelper
|
||||
*/
|
||||
@ -375,6 +381,19 @@ class Form extends RequestHandler {
|
||||
return $this->httpError(404);
|
||||
}
|
||||
|
||||
public function checkAccessAction($action) {
|
||||
return (
|
||||
parent::checkAccessAction($action)
|
||||
// Always allow actions which map to buttons. See httpSubmission() for further access checks.
|
||||
|| $this->actions->dataFieldByName('action_' . $action)
|
||||
// Always allow actions on fields
|
||||
|| (
|
||||
$field = $this->checkFieldsForAction($this->Fields(), $action)
|
||||
&& $field->checkAccessAction($action)
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the appropriate response up the controller chain
|
||||
* if {@link validate()} fails (which is checked prior to executing any form actions).
|
||||
@ -427,7 +446,7 @@ class Form extends RequestHandler {
|
||||
if($field = $this->checkFieldsForAction($field->FieldList(), $funcName)) {
|
||||
return $field;
|
||||
}
|
||||
} elseif ($field->hasMethod($funcName)) {
|
||||
} elseif ($field->hasMethod($funcName) && $field->checkAccessAction($funcName)) {
|
||||
return $field;
|
||||
}
|
||||
}
|
||||
|
@ -14,6 +14,18 @@ class HtmlEditorField extends TextareaField {
|
||||
*/
|
||||
private static $use_gzip = true;
|
||||
|
||||
/**
|
||||
* @config
|
||||
* @var Integer Default insertion width for Images and Media
|
||||
*/
|
||||
private static $insert_width = 600;
|
||||
|
||||
/**
|
||||
* @config
|
||||
* @var bool Should we check the valid_elements (& extended_valid_elements) rules from HtmlEditorConfig server side?
|
||||
*/
|
||||
private static $sanitise_server_side = false;
|
||||
|
||||
protected $rows = 30;
|
||||
|
||||
/**
|
||||
@ -106,6 +118,11 @@ class HtmlEditorField extends TextareaField {
|
||||
|
||||
$htmlValue = Injector::inst()->create('HTMLValue', $this->value);
|
||||
|
||||
if($this->config()->sanitise_server_side) {
|
||||
$santiser = Injector::inst()->create('HtmlEditorSanitiser', HtmlEditorConfig::get_active());
|
||||
$santiser->sanitise($htmlValue);
|
||||
}
|
||||
|
||||
if(class_exists('SiteTree')) {
|
||||
// Populate link tracking for internal links & links to asset files.
|
||||
if($links = $htmlValue->getElementsByTagName('a')) foreach($links as $link) {
|
||||
@ -428,8 +445,6 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
||||
$computerUploadField->removeExtraClass('ss-uploadfield');
|
||||
$computerUploadField->setTemplate('HtmlEditorField_UploadField');
|
||||
$computerUploadField->setFolderName(Config::inst()->get('Upload', 'uploads_folder'));
|
||||
// @todo - Remove this once this field supports display and recovery of file upload validation errors
|
||||
$computerUploadField->setOverwriteWarning(false);
|
||||
|
||||
$tabSet = new TabSet(
|
||||
"MediaFormInsertMediaTabs",
|
||||
@ -622,10 +637,10 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
||||
'CSSClass',
|
||||
_t('HtmlEditorField.CSSCLASS', 'Alignment / style'),
|
||||
array(
|
||||
'left' => _t('HtmlEditorField.CSSCLASSLEFT', 'On the left, with text wrapping around.'),
|
||||
'leftAlone' => _t('HtmlEditorField.CSSCLASSLEFTALONE', 'On the left, on its own.'),
|
||||
'right' => _t('HtmlEditorField.CSSCLASSRIGHT', 'On the right, with text wrapping around.'),
|
||||
'center' => _t('HtmlEditorField.CSSCLASSCENTER', 'Centered, on its own.'),
|
||||
'left' => _t('HtmlEditorField.CSSCLASSLEFT', 'On the left, with text wrapping around.'),
|
||||
'right' => _t('HtmlEditorField.CSSCLASSRIGHT', 'On the right, with text wrapping around.')
|
||||
)
|
||||
)->addExtraClass('last')
|
||||
);
|
||||
@ -636,12 +651,12 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
||||
TextField::create(
|
||||
'Width',
|
||||
_t('HtmlEditorField.IMAGEWIDTHPX', 'Width'),
|
||||
$file->Width
|
||||
$file->InsertWidth
|
||||
)->setMaxLength(5),
|
||||
TextField::create(
|
||||
'Height',
|
||||
_t('HtmlEditorField.IMAGEHEIGHTPX', 'Height'),
|
||||
$file->Height
|
||||
$file->InsertHeight
|
||||
)->setMaxLength(5)
|
||||
)->addExtraClass('dimensions last')
|
||||
);
|
||||
@ -661,7 +676,7 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
||||
), 'CaptionText');
|
||||
}
|
||||
|
||||
$this->extend('updateFieldsForImage', $fields, $url, $file);
|
||||
$this->extend('updateFieldsForOembed', $fields, $url, $file);
|
||||
|
||||
return $fields;
|
||||
}
|
||||
@ -746,10 +761,10 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
||||
'CSSClass',
|
||||
_t('HtmlEditorField.CSSCLASS', 'Alignment / style'),
|
||||
array(
|
||||
'left' => _t('HtmlEditorField.CSSCLASSLEFT', 'On the left, with text wrapping around.'),
|
||||
'leftAlone' => _t('HtmlEditorField.CSSCLASSLEFTALONE', 'On the left, on its own.'),
|
||||
'right' => _t('HtmlEditorField.CSSCLASSRIGHT', 'On the right, with text wrapping around.'),
|
||||
'center' => _t('HtmlEditorField.CSSCLASSCENTER', 'Centered, on its own.'),
|
||||
'left' => _t('HtmlEditorField.CSSCLASSLEFT', 'On the left, with text wrapping around.'),
|
||||
'right' => _t('HtmlEditorField.CSSCLASSRIGHT', 'On the right, with text wrapping around.')
|
||||
)
|
||||
)->addExtraClass('last')
|
||||
);
|
||||
@ -759,12 +774,12 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
||||
TextField::create(
|
||||
'Width',
|
||||
_t('HtmlEditorField.IMAGEWIDTHPX', 'Width'),
|
||||
$file->Width
|
||||
$file->InsertWidth
|
||||
)->setMaxLength(5),
|
||||
TextField::create(
|
||||
'Height',
|
||||
" x " . _t('HtmlEditorField.IMAGEHEIGHTPX', 'Height'),
|
||||
$file->Height
|
||||
$file->InsertHeight
|
||||
)->setMaxLength(5)
|
||||
)->addExtraClass('dimensions last')
|
||||
);
|
||||
@ -910,6 +925,29 @@ class HtmlEditorField_Embed extends HtmlEditorField_File {
|
||||
return $this->oembed->Height ?: 100;
|
||||
}
|
||||
|
||||
/**
|
||||
* Provide an initial width for inserted media, restricted based on $embed_width
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getInsertWidth() {
|
||||
$width = $this->getWidth();
|
||||
$maxWidth = Config::inst()->get('HtmlEditorField', 'insert_width');
|
||||
return ($width <= $maxWidth) ? $width : $maxWidth;
|
||||
}
|
||||
|
||||
/**
|
||||
* Provide an initial height for inserted media, scaled proportionally to the initial width
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getInsertHeight() {
|
||||
$width = $this->getWidth();
|
||||
$height = $this->getHeight();
|
||||
$maxWidth = Config::inst()->get('HtmlEditorField', 'insert_width');
|
||||
return ($width <= $maxWidth) ? $height : round($height*($maxWidth/$width));
|
||||
}
|
||||
|
||||
public function getPreview() {
|
||||
if(isset($this->oembed->thumbnail_url)) {
|
||||
return sprintf('<img src="%s" />', $this->oembed->thumbnail_url);
|
||||
@ -966,6 +1004,29 @@ class HtmlEditorField_Image extends HtmlEditorField_File {
|
||||
return ($this->file) ? $this->file->Height : $this->height;
|
||||
}
|
||||
|
||||
/**
|
||||
* Provide an initial width for inserted image, restricted based on $embed_width
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getInsertWidth() {
|
||||
$width = $this->getWidth();
|
||||
$maxWidth = Config::inst()->get('HtmlEditorField', 'insert_width');
|
||||
return ($width <= $maxWidth) ? $width : $maxWidth;
|
||||
}
|
||||
|
||||
/**
|
||||
* Provide an initial height for inserted image, scaled proportionally to the initial width
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getInsertHeight() {
|
||||
$width = $this->getWidth();
|
||||
$height = $this->getHeight();
|
||||
$maxWidth = Config::inst()->get('HtmlEditorField', 'insert_width');
|
||||
return ($width <= $maxWidth) ? $height : round($height*($maxWidth/$width));
|
||||
}
|
||||
|
||||
public function getPreview() {
|
||||
return ($this->file) ? $this->file->CMSThumbnail() : sprintf('<img src="%s" />', $this->url);
|
||||
}
|
||||
|
304
forms/HtmlEditorSanitiser.php
Normal file
304
forms/HtmlEditorSanitiser.php
Normal file
@ -0,0 +1,304 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* Sanitises an HTMLValue so it's contents are the elements and attributes that are whitelisted
|
||||
* using the same configuration as TinyMCE
|
||||
*
|
||||
* See www.tinymce.com/wiki.php/configuration:valid_elements for details on the spec of TinyMCE's
|
||||
* whitelist configuration
|
||||
*
|
||||
* Class HtmlEditorSanitiser
|
||||
*/
|
||||
class HtmlEditorSanitiser {
|
||||
|
||||
/** @var [stdClass] - $element => $rule hash for whitelist element rules where the element name isn't a pattern */
|
||||
protected $elements = array();
|
||||
/** @var [stdClass] - Sequential list of whitelist element rules where the element name is a pattern */
|
||||
protected $elementPatterns = array();
|
||||
|
||||
/** @var [stdClass] - The list of attributes that apply to all further whitelisted elements added */
|
||||
protected $globalAttributes = array();
|
||||
|
||||
/**
|
||||
* Construct a sanitiser from a given HtmlEditorConfig
|
||||
*
|
||||
* Note that we build data structures from the current state of HtmlEditorConfig - later changes to
|
||||
* the passed instance won't cause this instance to update it's whitelist
|
||||
*
|
||||
* @param HtmlEditorConfig $config
|
||||
*/
|
||||
public function __construct(HtmlEditorConfig $config) {
|
||||
$valid = $config->getOption('valid_elements');
|
||||
if ($valid) $this->addValidElements($valid);
|
||||
|
||||
$valid = $config->getOption('extended_valid_elements');
|
||||
if ($valid) $this->addValidElements($valid);
|
||||
}
|
||||
|
||||
/**
|
||||
* Given a TinyMCE pattern (close to unix glob style), create a regex that does the match
|
||||
*
|
||||
* @param $str - The TinyMCE pattern
|
||||
* @return string - The equivalent regex
|
||||
*/
|
||||
protected function patternToRegex($str) {
|
||||
return '/^' . preg_replace('/([?+*])/', '.$1', $str) . '$/';
|
||||
}
|
||||
|
||||
/**
|
||||
* Given a valid_elements string, parse out the actual element and attribute rules and add to the
|
||||
* internal whitelist
|
||||
*
|
||||
* Logic based heavily on javascript version from tiny_mce_src.js
|
||||
*
|
||||
* @param string $validElements - The valid_elements or extended_valid_elements string to add to the whitelist
|
||||
*/
|
||||
protected function addValidElements($validElements) {
|
||||
$elementRuleRegExp = '/^([#+\-])?([^\[\/]+)(?:\/([^\[]+))?(?:\[([^\]]+)\])?$/';
|
||||
$attrRuleRegExp = '/^([!\-])?(\w+::\w+|[^=:<]+)?(?:([=:<])(.*))?$/';
|
||||
$hasPatternsRegExp = '/[*?+]/';
|
||||
|
||||
foreach(explode(',', $validElements) as $validElement) {
|
||||
if(preg_match($elementRuleRegExp, $validElement, $matches)) {
|
||||
|
||||
$prefix = @$matches[1];
|
||||
$elementName = @$matches[2];
|
||||
$outputName = @$matches[3];
|
||||
$attrData = @$matches[4];
|
||||
|
||||
// Create the new element
|
||||
$element = new stdClass();
|
||||
$element->attributes = array();
|
||||
$element->attributePatterns = array();
|
||||
|
||||
$element->attributesRequired = array();
|
||||
$element->attributesDefault = array();
|
||||
$element->attributesForced = array();
|
||||
|
||||
foreach(array('#' => 'paddEmpty', '-' => 'removeEmpty') as $match => $means) {
|
||||
$element->$means = ($prefix === $match);
|
||||
}
|
||||
|
||||
// Copy attributes from global rule into current rule
|
||||
if($this->globalAttributes) {
|
||||
$element->attributes = array_merge($element->attributes, $this->globalAttributes);
|
||||
}
|
||||
|
||||
// Attributes defined
|
||||
if($attrData) {
|
||||
foreach(explode('|', $attrData) as $attr) {
|
||||
if(preg_match($attrRuleRegExp, $attr, $matches)) {
|
||||
$attr = new stdClass();
|
||||
|
||||
$attrType = @$matches[1];
|
||||
$attrName = str_replace('::', ':', @$matches[2]);
|
||||
$prefix = @$matches[3];
|
||||
$value = @$matches[4];
|
||||
|
||||
// Required
|
||||
if($attrType === '!') {
|
||||
$element->attributesRequired[] = $attrName;
|
||||
$attr->required = true;
|
||||
}
|
||||
|
||||
// Denied from global
|
||||
else if($attrType === '-') {
|
||||
unset($element->attributes[$attrName]);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Default value
|
||||
if($prefix) {
|
||||
// Default value
|
||||
if($prefix === '=') {
|
||||
$element->attributesDefault[$attrName] = $value;
|
||||
$attr->defaultValue = $value;
|
||||
}
|
||||
|
||||
// Forced value
|
||||
else if($prefix === ':') {
|
||||
$element->attributesForced[$attrName] = $value;
|
||||
$attr->forcedValue = $value;
|
||||
}
|
||||
|
||||
// Required values
|
||||
else if($prefix === '<') {
|
||||
$attr->validValues = explode('?', $value);
|
||||
}
|
||||
}
|
||||
|
||||
// Check for attribute patterns
|
||||
if(preg_match($hasPatternsRegExp, $attrName)) {
|
||||
$attr->pattern = $this->patternToRegex($attrName);
|
||||
$element->attributePatterns[] = $attr;
|
||||
}
|
||||
else {
|
||||
$element->attributes[$attrName] = $attr;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Global rule, store away these for later usage
|
||||
if(!$this->globalAttributes && $elementName == '@') {
|
||||
$this->globalAttributes = $element->attributes;
|
||||
}
|
||||
|
||||
// Handle substitute elements such as b/strong
|
||||
if($outputName) {
|
||||
$element->outputName = $elementName;
|
||||
$this->elements[$outputName] = $element;
|
||||
}
|
||||
|
||||
// Add pattern or exact element
|
||||
if(preg_match($hasPatternsRegExp, $elementName)) {
|
||||
$element->pattern = $this->patternToRegex($elementName);
|
||||
$this->elementPatterns[] = $element;
|
||||
}
|
||||
else {
|
||||
$this->elements[$elementName] = $element;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Given an element tag, return the rule structure for that element
|
||||
* @param string $tag - The element tag
|
||||
* @return stdClass - The element rule
|
||||
*/
|
||||
protected function getRuleForElement($tag) {
|
||||
if(isset($this->elements[$tag])) {
|
||||
return $this->elements[$tag];
|
||||
}
|
||||
else foreach($this->elementPatterns as $element) {
|
||||
if(preg_match($element->pattern, $tag)) return $element;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Given an attribute name, return the rule structure for that attribute
|
||||
* @param string $name - The attribute name
|
||||
* @return stdClass - The attribute rule
|
||||
*/
|
||||
protected function getRuleForAttribute($elementRule, $name) {
|
||||
if(isset($elementRule->attributes[$name])) {
|
||||
return $elementRule->attributes[$name];
|
||||
}
|
||||
else foreach($elementRule->attributePatterns as $attribute) {
|
||||
if(preg_match($attribute->pattern, $name)) return $attribute;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Given a DOMElement and an element rule, check if that element passes the rule
|
||||
* @param DOMElement $element - the element to check
|
||||
* @param stdClass $rule - the rule to check against
|
||||
* @return bool - true if the element passes (and so can be kept), false if it fails (and so needs stripping)
|
||||
*/
|
||||
protected function elementMatchesRule($element, $rule = null) {
|
||||
// If the rule doesn't exist at all, the element isn't allowed
|
||||
if(!$rule) return false;
|
||||
|
||||
// If the rule has attributes required, check them to see if this element has at least one
|
||||
if($rule->attributesRequired) {
|
||||
$hasMatch = false;
|
||||
|
||||
foreach($rule->attributesRequired as $attr) {
|
||||
if($element->getAttribute($attr)) {
|
||||
$hasMatch = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if(!$hasMatch) return false;
|
||||
}
|
||||
|
||||
// If the rule says to remove empty elements, and this element is empty, remove it
|
||||
if($rule->removeEmpty && !$element->firstChild) return false;
|
||||
|
||||
// No further tests required, element passes
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Given a DOMAttr and an attribute rule, check if that attribute passes the rule
|
||||
* @param DOMAttr $attr - the attribute to check
|
||||
* @param stdClass $rule - the rule to check against
|
||||
* @return bool - true if the attribute passes (and so can be kept), false if it fails (and so needs stripping)
|
||||
*/
|
||||
protected function attributeMatchesRule($attr, $rule = null) {
|
||||
// If the rule doesn't exist at all, the attribute isn't allowed
|
||||
if(!$rule) return false;
|
||||
|
||||
// If the rule has a set of valid values, check them to see if this attribute is one
|
||||
if(isset($rule->validValues) && !in_array($attr->value, $rule->validValues)) return false;
|
||||
|
||||
// No further tests required, attribute passes
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Given an SS_HTMLValue instance, will remove and elements and attributes that are
|
||||
* not explicitly included in the whitelist passed to __construct on instance creation
|
||||
*
|
||||
* @param SS_HTMLValue $html - The HTMLValue to remove any non-whitelisted elements & attributes from
|
||||
*/
|
||||
public function sanitise (SS_HTMLValue $html) {
|
||||
if(!$this->elements && !$this->elementPatterns) return;
|
||||
|
||||
$doc = $html->getDocument();
|
||||
|
||||
foreach($html->query('//body//*') as $el) {
|
||||
$elementRule = $this->getRuleForElement($el->tagName);
|
||||
|
||||
// If this element isn't allowed, strip it
|
||||
if(!$this->elementMatchesRule($el, $elementRule)) {
|
||||
// If it's a script or style, we don't keep contents
|
||||
if($el->tagName === 'script' || $el->tagName === 'style') {
|
||||
$el->parentNode->removeChild($el);
|
||||
}
|
||||
// Otherwise we replace this node with all it's children
|
||||
else {
|
||||
// First, create a new fragment with all of $el's children moved into it
|
||||
$frag = $doc->createDocumentFragment();
|
||||
while($el->firstChild) $frag->appendChild($el->firstChild);
|
||||
|
||||
// Then replace $el with the frags contents (which used to be it's children)
|
||||
$el->parentNode->replaceChild($frag, $el);
|
||||
}
|
||||
}
|
||||
// Otherwise tidy the element
|
||||
else {
|
||||
// First, if we're supposed to pad & this element is empty, fix that
|
||||
if($elementRule->paddEmpty && !$el->firstChild) {
|
||||
$el->nodeValue = ' ';
|
||||
}
|
||||
|
||||
// Then filter out any non-whitelisted attributes
|
||||
$children = $el->attributes;
|
||||
$i = $children->length;
|
||||
while($i--) {
|
||||
$attr = $children->item($i);
|
||||
$attributeRule = $this->getRuleForAttribute($elementRule, $attr->name);
|
||||
|
||||
// If this attribute isn't allowed, strip it
|
||||
if(!$this->attributeMatchesRule($attr, $attributeRule)) {
|
||||
$el->removeAttributeNode($attr);
|
||||
}
|
||||
}
|
||||
|
||||
// Then enforce any default attributes
|
||||
foreach($elementRule->attributesDefault as $attr => $default) {
|
||||
if(!$el->getAttribute($attr)) $el->setAttribute($attr, $default);
|
||||
}
|
||||
|
||||
// And any forced attributes
|
||||
foreach($elementRule->attributesForced as $attr => $forced) {
|
||||
$el->setAttribute($attr, $forced);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -1331,6 +1331,12 @@ class UploadField_ItemHandler extends RequestHandler {
|
||||
'' => 'index',
|
||||
);
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'delete',
|
||||
'edit',
|
||||
'EditForm',
|
||||
);
|
||||
|
||||
/**
|
||||
* @param UploadFIeld $parent
|
||||
* @param int $item
|
||||
@ -1484,6 +1490,10 @@ class UploadField_SelectHandler extends RequestHandler {
|
||||
'' => 'index',
|
||||
);
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'Form'
|
||||
);
|
||||
|
||||
public function __construct($parent, $folderName = null) {
|
||||
$this->parent = $parent;
|
||||
$this->folderName = $folderName;
|
||||
|
@ -195,6 +195,12 @@ class GridFieldDetailForm implements GridField_URLHandler {
|
||||
*/
|
||||
class GridFieldDetailForm_ItemRequest extends RequestHandler {
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'edit',
|
||||
'view',
|
||||
'ItemEditForm'
|
||||
);
|
||||
|
||||
/**
|
||||
*
|
||||
* @var GridField
|
||||
|
@ -382,7 +382,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
success: function(html) {
|
||||
dialog.html(html);
|
||||
dialog.getForm().setElement(self);
|
||||
dialog.trigger('dialogopen');
|
||||
dialog.trigger('ssdialogopen');
|
||||
}
|
||||
});
|
||||
}
|
||||
@ -452,7 +452,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
},
|
||||
|
||||
fromDialog: {
|
||||
ondialogopen: function(){
|
||||
onssdialogopen: function(){
|
||||
var ed = this.getEditor();
|
||||
ed.onopen();
|
||||
|
||||
@ -467,7 +467,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
this.redraw();
|
||||
},
|
||||
|
||||
ondialogclose: function(){
|
||||
onssdialogclose: function(){
|
||||
var ed = this.getEditor();
|
||||
ed.onclose();
|
||||
|
||||
@ -826,7 +826,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
});
|
||||
|
||||
ed.repaint();
|
||||
})
|
||||
});
|
||||
|
||||
this.getDialog().close();
|
||||
return false;
|
||||
@ -944,8 +944,9 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
var uploadedFiles = $('.ss-uploadfield-files', this).children('.ss-uploadfield-item');
|
||||
uploadedFiles.each(function(){
|
||||
var uploadedID = $(this).data('fileid');
|
||||
if ($.inArray(uploadedID, editFieldIDs) == -1) {
|
||||
if (uploadedID && $.inArray(uploadedID, editFieldIDs) == -1) {
|
||||
//trigger the detail view for filling out details about the file we are about to insert into TinyMCE
|
||||
$(this).remove(); // Remove successfully added item from the queue
|
||||
form.showFileView(uploadedID);
|
||||
}
|
||||
});
|
||||
@ -1271,12 +1272,6 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
this._super();
|
||||
|
||||
this.setOrigVal(parseInt(this.val(), 10));
|
||||
|
||||
// Default to a managable size for the HTML view. Can be overwritten by user after initialization
|
||||
if(this.attr('name') == 'Width') {
|
||||
this.closest('.ss-htmleditorfield-file').updateDimensions('Width', 600);
|
||||
}
|
||||
|
||||
},
|
||||
onunmatch: function() {
|
||||
this._super();
|
||||
|
@ -4,8 +4,22 @@
|
||||
* On resize of any close the open treedropdownfields
|
||||
* as we'll need to redo with widths
|
||||
*/
|
||||
$(window).resize(function() {
|
||||
$('.TreeDropdownField').closePanel();
|
||||
var windowWidth, windowHeight;
|
||||
$(window).bind('resize.treedropdownfield', function() {
|
||||
// Entwine's 'fromWindow::onresize' does not trigger on IE8. Use synthetic event.
|
||||
var cb = function() {$('.TreeDropdownField').closePanel();};
|
||||
|
||||
// Workaround to avoid IE8 infinite loops when elements are resized as a result of this event
|
||||
if($.browser.msie && parseInt($.browser.version, 10) < 9) {
|
||||
var newWindowWidth = $(window).width(), newWindowHeight = $(window).height();
|
||||
if(newWindowWidth != windowWidth || newWindowHeight != windowHeight) {
|
||||
windowWidth = newWindowWidth;
|
||||
windowHeight = newWindowHeight;
|
||||
cb();
|
||||
}
|
||||
} else {
|
||||
cb();
|
||||
}
|
||||
});
|
||||
|
||||
var strings = {
|
||||
|
@ -64,13 +64,16 @@
|
||||
.addClass('ui-state-warning-text');
|
||||
data.context.find('.ss-uploadfield-item-progress').hide();
|
||||
data.context.find('.ss-uploadfield-item-overwrite').show();
|
||||
data.context.find('.ss-uploadfield-item-overwrite-warning').on('click', function(){
|
||||
data.context.find('.ss-uploadfield-item-overwrite-warning').on('click', function(e){
|
||||
data.context.find('.ss-uploadfield-item-progress').show();
|
||||
data.context.find('.ss-uploadfield-item-overwrite').hide();
|
||||
data.context.find('.ss-uploadfield-item-status')
|
||||
.removeClass('ui-state-warning-text');
|
||||
//upload only if the "overwrite" button is clicked
|
||||
$.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
||||
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
});
|
||||
} else { //regular file upload
|
||||
return $.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
||||
@ -319,12 +322,14 @@
|
||||
$('div.ss-upload .ss-uploadfield-startall').entwine({
|
||||
onclick: function(e) {
|
||||
this.closest('.ss-upload').find('.ss-uploadfield-item-start button').click();
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
}
|
||||
});
|
||||
$('div.ss-upload .ss-uploadfield-item-cancelfailed').entwine({
|
||||
onclick: function(e) {
|
||||
this.closest('.ss-uploadfield-item').remove();
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
}
|
||||
});
|
||||
@ -349,6 +354,7 @@
|
||||
fileupload._trigger('destroy', e, {context: item});
|
||||
}
|
||||
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
}
|
||||
});
|
||||
@ -371,6 +377,7 @@
|
||||
}
|
||||
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
}
|
||||
});
|
||||
$( 'div.ss-upload:not(.disabled):not(.readonly) .ss-uploadfield-item-edit').entwine({
|
||||
@ -403,6 +410,7 @@
|
||||
editform.toggleEditForm();
|
||||
}
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
@ -489,8 +497,9 @@
|
||||
});
|
||||
$('div.ss-upload .ss-uploadfield-fromfiles').entwine({
|
||||
onclick: function(e) {
|
||||
e.preventDefault();
|
||||
this.getUploadField().openSelectDialog(this.closest('.ss-uploadfield-item'));
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
return false;
|
||||
}
|
||||
});
|
||||
});
|
||||
|
@ -156,6 +156,14 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
protected static $_cache_custom_database_fields = array();
|
||||
protected static $_cache_field_labels = array();
|
||||
|
||||
// base fields which are not defined in static $db
|
||||
private static $fixed_fields = array(
|
||||
'ID' => 'Int',
|
||||
'ClassName' => 'Enum',
|
||||
'LastEdited' => 'SS_Datetime',
|
||||
'Created' => 'SS_Datetime',
|
||||
);
|
||||
|
||||
/**
|
||||
* Non-static relationship cache, indexed by component name.
|
||||
*/
|
||||
@ -223,6 +231,8 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
}
|
||||
|
||||
return array_merge (
|
||||
// TODO: should this be using self::$fixed_fields? only difference is ID field
|
||||
// and ClassName creates an Enum with all values
|
||||
array (
|
||||
'ClassName' => self::$classname_spec_cache[$class],
|
||||
'Created' => 'SS_Datetime',
|
||||
@ -1651,11 +1661,16 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
/**
|
||||
* Return all of the database fields defined in self::$db and all the parent classes.
|
||||
* Doesn't include any fields specified by self::$has_one. Use $this->has_one() to get these fields
|
||||
* Also returns "base" fields like "Created", "LastEdited", et cetera.
|
||||
*
|
||||
* @param string $fieldName Limit the output to a specific field name
|
||||
* @return array The database fields
|
||||
*/
|
||||
public function db($fieldName = null) {
|
||||
if ($fieldName && array_key_exists($fieldName, self::$fixed_fields)) {
|
||||
return self::$fixed_fields[$fieldName];
|
||||
}
|
||||
|
||||
$classes = ClassInfo::ancestry($this);
|
||||
$good = false;
|
||||
$items = array();
|
||||
@ -1694,6 +1709,10 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
}
|
||||
}
|
||||
|
||||
if (!$fieldName) {
|
||||
// trying to get all fields, so add the fixed fields to return value
|
||||
$items = array_merge(self::$fixed_fields, $items);
|
||||
}
|
||||
return $items;
|
||||
}
|
||||
|
||||
@ -2387,15 +2406,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
* @return boolean
|
||||
*/
|
||||
public function hasDatabaseField($field) {
|
||||
// Add base fields which are not defined in static $db
|
||||
static $fixedFields = array(
|
||||
'ID' => 'Int',
|
||||
'ClassName' => 'Enum',
|
||||
'LastEdited' => 'SS_Datetime',
|
||||
'Created' => 'SS_Datetime',
|
||||
);
|
||||
|
||||
if(isset($fixedFields[$field])) return true;
|
||||
if(isset(self::$fixed_fields[$field])) return true;
|
||||
|
||||
return array_key_exists($field, $this->inheritedDatabaseFields());
|
||||
}
|
||||
@ -2659,8 +2670,13 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
} else if($fieldName == 'ID') {
|
||||
return new PrimaryKey($fieldName, $this);
|
||||
|
||||
// General casting information for items in $db or $casting
|
||||
} else if($helper = $this->castingHelper($fieldName)) {
|
||||
// Special case for ClassName
|
||||
} else if($fieldName == 'ClassName') {
|
||||
$val = get_class($this);
|
||||
return DBField::create_field('Varchar', $val, $fieldName, $this);
|
||||
|
||||
// General casting information for items in $db
|
||||
} else if($helper = $this->db($fieldName)) {
|
||||
$obj = Object::create_from_string($helper, $fieldName);
|
||||
$obj->setValue($this->$fieldName, $this->record, false);
|
||||
return $obj;
|
||||
@ -2669,11 +2685,6 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
||||
} else if(preg_match('/ID$/', $fieldName) && $this->has_one(substr($fieldName,0,-2))) {
|
||||
$val = $this->$fieldName;
|
||||
return DBField::create_field('ForeignKey', $val, $fieldName, $this);
|
||||
|
||||
// Special case for ClassName
|
||||
} else if($fieldName == 'ClassName') {
|
||||
$val = get_class($this);
|
||||
return DBField::create_field('Varchar', $val, $fieldName, $this);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -217,14 +217,14 @@
|
||||
top: 0;
|
||||
right: 0;
|
||||
margin: 0;
|
||||
border: solid #000;
|
||||
border-width: 0 0 100px 200px;
|
||||
opacity: 0;
|
||||
filter: alpha(opacity=0);
|
||||
-o-transform: translate(250px, -50px) scale(1);
|
||||
-moz-transform: translate(-300px, 0) scale(4);
|
||||
transform: translate(-300px, 0) scale(4);
|
||||
font-size: 23px;
|
||||
direction: ltr;
|
||||
cursor: pointer;
|
||||
height: 30px;
|
||||
line-height: 30px;
|
||||
}
|
||||
}
|
||||
}
|
@ -346,7 +346,9 @@ class Security extends Controller {
|
||||
$member = Member::currentUser();
|
||||
if($member) $member->logOut();
|
||||
|
||||
if($redirect) $this->redirectBack();
|
||||
if($redirect && (!$this->response || !$this->response->isFinished())) {
|
||||
$this->redirectBack();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
@ -422,7 +422,7 @@ class CmsUiContext extends BehatContext
|
||||
) {
|
||||
if($container->isVisible() && in_array($class, explode(' ', $container->getAttribute('class')))) {
|
||||
return $container;
|
||||
}
|
||||
}
|
||||
$container = $container->getParent();
|
||||
}
|
||||
|
||||
|
@ -1,3 +1,4 @@
|
||||
@assets
|
||||
Feature: Insert an image into a page
|
||||
As a cms author
|
||||
I want to insert an image into a page
|
||||
|
@ -53,15 +53,31 @@ class ControllerTest extends FunctionalTest {
|
||||
|
||||
$response = $this->get("ControllerTest_UnsecuredController/index");
|
||||
$this->assertEquals(200, $response->getStatusCode(),
|
||||
'Access granted on index action without $allowed_actions on defining controller, ' .
|
||||
'Access denied on index action without $allowed_actions on defining controller, ' .
|
||||
'when called with an action in the URL'
|
||||
);
|
||||
|
||||
Config::inst()->update('RequestHandler', 'require_allowed_actions', false);
|
||||
$response = $this->get("ControllerTest_UnsecuredController/index");
|
||||
$this->assertEquals(200, $response->getStatusCode(),
|
||||
'Access granted on index action without $allowed_actions on defining controller, ' .
|
||||
'when called with an action in the URL, and explicitly allowed through config'
|
||||
);
|
||||
Config::inst()->update('RequestHandler', 'require_allowed_actions', true);
|
||||
|
||||
$response = $this->get("ControllerTest_UnsecuredController/method1");
|
||||
$this->assertEquals(403, $response->getStatusCode(),
|
||||
'Access denied on action without $allowed_actions on defining controller, ' .
|
||||
'when called without an action in the URL'
|
||||
);
|
||||
|
||||
Config::inst()->update('RequestHandler', 'require_allowed_actions', false);
|
||||
$response = $this->get("ControllerTest_UnsecuredController/method1");
|
||||
$this->assertEquals(200, $response->getStatusCode(),
|
||||
'Access granted on action without $allowed_actions on defining controller, ' .
|
||||
'when called without an action in the URL'
|
||||
'when called without an action in the URL, and explicitly allowed through config'
|
||||
);
|
||||
Config::inst()->update('RequestHandler', 'require_allowed_actions', true);
|
||||
|
||||
$response = $this->get("ControllerTest_AccessBaseController/");
|
||||
$this->assertEquals(200, $response->getStatusCode(),
|
||||
@ -110,6 +126,12 @@ class ControllerTest extends FunctionalTest {
|
||||
'if action is not a method but rather a template discovered by naming convention'
|
||||
);
|
||||
|
||||
$response = $this->get("ControllerTest_AccessSecuredController/templateaction");
|
||||
$this->assertEquals(403, $response->getStatusCode(),
|
||||
'Access denied on action with $allowed_actions on defining controller, ' .
|
||||
'if action is not a method but rather a template discovered by naming convention'
|
||||
);
|
||||
|
||||
$this->session()->inst_set('loggedInAs', $adminUser->ID);
|
||||
$response = $this->get("ControllerTest_AccessSecuredController/templateaction");
|
||||
$this->assertEquals(200, $response->getStatusCode(),
|
||||
@ -391,7 +413,7 @@ class ControllerTest_UnsecuredController extends Controller implements TestOnly
|
||||
|
||||
// Granted for all
|
||||
public function method2() {}
|
||||
}
|
||||
}
|
||||
|
||||
class ControllerTest_AccessBaseController extends Controller implements TestOnly {
|
||||
|
||||
@ -402,7 +424,7 @@ class ControllerTest_AccessBaseController extends Controller implements TestOnly
|
||||
|
||||
// Denied for all
|
||||
public function method2() {}
|
||||
}
|
||||
}
|
||||
|
||||
class ControllerTest_AccessSecuredController extends ControllerTest_AccessBaseController implements TestOnly {
|
||||
|
||||
@ -427,7 +449,7 @@ class ControllerTest_AccessWildcardSecuredController extends ControllerTest_Acce
|
||||
"*" => "ADMIN", // should throw exception
|
||||
);
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
class ControllerTest_IndexSecuredController extends ControllerTest_AccessBaseController implements TestOnly {
|
||||
|
||||
@ -435,7 +457,7 @@ class ControllerTest_IndexSecuredController extends ControllerTest_AccessBaseCon
|
||||
"index" => "ADMIN",
|
||||
);
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
class ControllerTest_AccessBaseControllerExtension extends Extension implements TestOnly {
|
||||
|
||||
@ -457,7 +479,7 @@ class ControllerTest_AccessBaseControllerExtension extends Extension implements
|
||||
|
||||
public function internalextensionmethod() {}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
class ControllerTest_HasAction extends Controller {
|
||||
|
||||
|
@ -348,6 +348,13 @@ class DirectorTest extends SapphireTest {
|
||||
|
||||
class DirectorTestRequest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'returnGetValue',
|
||||
'returnPostValue',
|
||||
'returnRequestValue',
|
||||
'returnCookieValue',
|
||||
);
|
||||
|
||||
public function returnGetValue($request) { return $_GET['somekey']; }
|
||||
|
||||
public function returnPostValue($request) { return $_POST['somekey']; }
|
||||
|
@ -130,10 +130,6 @@ class RequestHandlingTest extends FunctionalTest {
|
||||
}
|
||||
|
||||
public function testDisallowedExtendedActions() {
|
||||
/* Actions on magic methods are only accessible if explicitly allowed on the controller. */
|
||||
$response = Director::test("testGoodBase1/extendedMethod");
|
||||
$this->assertEquals(404, $response->getStatusCode());
|
||||
|
||||
/* Actions on an extension are allowed because they specifically provided appropriate allowed_actions items */
|
||||
$response = Director::test("testGoodBase1/otherExtendedMethod");
|
||||
$this->assertEquals("otherExtendedMethod", $response->getBody());
|
||||
@ -146,9 +142,10 @@ class RequestHandlingTest extends FunctionalTest {
|
||||
$response = Director::test("RequestHandlingTest_AllowedController/failoverMethod");
|
||||
$this->assertEquals("failoverMethod", $response->getBody());
|
||||
|
||||
/* The action on the extension has also been explicitly allowed even though it wasn't on the extension */
|
||||
/* The action on the extension is allowed when explicitly allowed on extension,
|
||||
even if its not mentioned in controller */
|
||||
$response = Director::test("RequestHandlingTest_AllowedController/extendedMethod");
|
||||
$this->assertEquals("extendedMethod", $response->getBody());
|
||||
$this->assertEquals(200, $response->getStatusCode());
|
||||
|
||||
/* This action has been blocked by an argument to a method */
|
||||
$response = Director::test('RequestHandlingTest_AllowedController/blockMethod');
|
||||
@ -421,9 +418,13 @@ class RequestHandlingTest_FormActionController extends Controller {
|
||||
* Simple extension for the test controller
|
||||
*/
|
||||
class RequestHandlingTest_ControllerExtension extends Extension {
|
||||
|
||||
public static $called_error = false;
|
||||
|
||||
public static $called_404_error = false;
|
||||
|
||||
private static $allowed_actions = array('extendedMethod');
|
||||
|
||||
public function extendedMethod() {
|
||||
return "extendedMethod";
|
||||
}
|
||||
@ -455,7 +456,6 @@ class RequestHandlingTest_AllowedController extends Controller implements TestOn
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'failoverMethod', // part of the failover object
|
||||
'extendedMethod', // part of the RequestHandlingTest_ControllerExtension object
|
||||
'blockMethod' => '->provideAccess(false)',
|
||||
'allowMethod' => '->provideAccess',
|
||||
);
|
||||
@ -538,14 +538,15 @@ class RequestHandlingTest_Form extends Form {
|
||||
|
||||
class RequestHandlingTest_ControllerFormWithAllowedActions extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
public function Form() {
|
||||
return new RequestHandlingTest_FormWithAllowedActions(
|
||||
$this,
|
||||
'Form',
|
||||
new FieldList(),
|
||||
new FieldList(
|
||||
new FormAction('allowedformaction'),
|
||||
new FormAction('disallowedformaction') // disallowed through $allowed_actions in form
|
||||
new FormAction('allowedformaction')
|
||||
)
|
||||
);
|
||||
}
|
||||
@ -555,7 +556,6 @@ class RequestHandlingTest_FormWithAllowedActions extends Form {
|
||||
|
||||
private static $allowed_actions = array(
|
||||
'allowedformaction' => 1,
|
||||
'httpSubmission' => 1, // TODO This should be an exception on the parent class
|
||||
);
|
||||
|
||||
public function allowedformaction() {
|
||||
@ -603,6 +603,9 @@ class RequestHandlingTest_FormField extends FormField {
|
||||
* Form field for the test
|
||||
*/
|
||||
class RequestHandlingTest_SubclassedFormField extends RequestHandlingTest_FormField {
|
||||
|
||||
private static $allowed_actions = array('customSomething');
|
||||
|
||||
// We have some url_handlers defined that override RequestHandlingTest_FormField handlers.
|
||||
// We will confirm that the url_handlers inherit.
|
||||
private static $url_handlers = array(
|
||||
@ -621,6 +624,8 @@ class RequestHandlingTest_SubclassedFormField extends RequestHandlingTest_FormFi
|
||||
*/
|
||||
class RequestHandlingFieldTest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('TestForm');
|
||||
|
||||
public function TestForm() {
|
||||
return new Form($this, "TestForm", new FieldList(
|
||||
new RequestHandlingTest_HandlingField("MyField")
|
||||
@ -642,4 +647,8 @@ class RequestHandlingTest_HandlingField extends FormField {
|
||||
public function actionOnField() {
|
||||
return "Test method on $this->name";
|
||||
}
|
||||
|
||||
public function actionNotAllowedOnField() {
|
||||
return "actionNotAllowedOnField on $this->name";
|
||||
}
|
||||
}
|
||||
|
@ -8,6 +8,174 @@ class ConfigManifestTest_ConfigManifestAccess extends SS_ConfigManifest {
|
||||
|
||||
class ConfigManifestTest extends SapphireTest {
|
||||
|
||||
protected function getConfigFixtureValue($name) {
|
||||
$manifest = new SS_ConfigManifest(dirname(__FILE__).'/fixtures/configmanifest', true, true);
|
||||
return $manifest->get('ConfigManifestTest', $name);
|
||||
}
|
||||
|
||||
public function testClassRules() {
|
||||
$config = $this->getConfigFixtureValue('Class');
|
||||
|
||||
$this->assertEquals(
|
||||
'Yes', @$config['DirectorExists'],
|
||||
'Only rule correctly detects existing class'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['NoSuchClassExists'],
|
||||
'Except rule correctly detects missing class'
|
||||
);
|
||||
}
|
||||
|
||||
public function testModuleRules() {
|
||||
$config = $this->getConfigFixtureValue('Module');
|
||||
|
||||
$this->assertEquals(
|
||||
'Yes', @$config['MysiteExists'],
|
||||
'Only rule correctly detects existing module'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['NoSuchModuleExists'],
|
||||
'Except rule correctly detects missing module'
|
||||
);
|
||||
}
|
||||
|
||||
public function testEnvVarSetRules() {
|
||||
$_ENV['EnvVarSet_Foo'] = 1;
|
||||
$config = $this->getConfigFixtureValue('EnvVarSet');
|
||||
|
||||
$this->assertEquals(
|
||||
'Yes', @$config['FooSet'],
|
||||
'Only rule correctly detects set environment variable'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['BarSet'],
|
||||
'Except rule correctly detects unset environment variable'
|
||||
);
|
||||
}
|
||||
|
||||
public function testConstantDefinedRules() {
|
||||
define('ConstantDefined_Foo', 1);
|
||||
$config = $this->getConfigFixtureValue('ConstantDefined');
|
||||
|
||||
$this->assertEquals(
|
||||
'Yes', @$config['FooDefined'],
|
||||
'Only rule correctly detects defined constant'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['BarDefined'],
|
||||
'Except rule correctly detects undefined constant'
|
||||
);
|
||||
}
|
||||
|
||||
public function testEnvOrConstantMatchesValueRules() {
|
||||
$_ENV['EnvOrConstantMatchesValue_Foo'] = 'Foo';
|
||||
define('EnvOrConstantMatchesValue_Bar', 'Bar');
|
||||
$config = $this->getConfigFixtureValue('EnvOrConstantMatchesValue');
|
||||
|
||||
$this->assertEquals(
|
||||
'Yes', @$config['FooIsFoo'],
|
||||
'Only rule correctly detects environment variable matches specified value'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'Yes', @$config['BarIsBar'],
|
||||
'Only rule correctly detects constant matches specified value'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['FooIsQux'],
|
||||
'Except rule correctly detects environment variable that doesn\'t match specified value'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['BarIsQux'],
|
||||
'Except rule correctly detects environment variable that doesn\'t match specified value'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
'No', @$config['BazIsBaz'],
|
||||
'Except rule correctly detects undefined variable'
|
||||
);
|
||||
}
|
||||
|
||||
public function testEnvironmentRules() {
|
||||
foreach (array('dev', 'test', 'live') as $env) {
|
||||
Config::inst()->nest();
|
||||
|
||||
Config::inst()->update('Director', 'environment_type', $env);
|
||||
$config = $this->getConfigFixtureValue('Environment');
|
||||
|
||||
foreach (array('dev', 'test', 'live') as $check) {
|
||||
$this->assertEquals(
|
||||
$env == $check ? $check : 'not'.$check, @$config[ucfirst($check).'Environment'],
|
||||
'Only & except rules correctly detect environment'
|
||||
);
|
||||
}
|
||||
|
||||
Config::inst()->unnest();
|
||||
}
|
||||
}
|
||||
|
||||
public function testDynamicEnvironmentRules() {
|
||||
Config::inst()->nest();
|
||||
|
||||
// First, make sure environment_type is live
|
||||
Config::inst()->update('Director', 'environment_type', 'live');
|
||||
$this->assertEquals('live', Config::inst()->get('Director', 'environment_type'));
|
||||
|
||||
// Then, load in a new manifest, which includes a _config.php that sets environment_type to dev
|
||||
$manifest = new SS_ConfigManifest(dirname(__FILE__).'/fixtures/configmanifest_dynamicenv', true, true);
|
||||
Config::inst()->pushConfigYamlManifest($manifest);
|
||||
|
||||
// Make sure that stuck
|
||||
$this->assertEquals('dev', Config::inst()->get('Director', 'environment_type'));
|
||||
|
||||
// And that the dynamic rule was calculated correctly
|
||||
$this->assertEquals('dev', Config::inst()->get('ConfigManifestTest', 'DynamicEnvironment'));
|
||||
|
||||
Config::inst()->unnest();
|
||||
}
|
||||
|
||||
public function testMultipleRules() {
|
||||
$_ENV['MultilpleRules_EnvVariableSet'] = 1;
|
||||
define('MultilpleRules_DefinedConstant', 'defined');
|
||||
$config = $this->getConfigFixtureValue('MultipleRules');
|
||||
|
||||
$this->assertFalse(
|
||||
isset($config['TwoOnlyFail']),
|
||||
'Fragment is not included if one of the Only rules fails.'
|
||||
);
|
||||
|
||||
$this->assertTrue(
|
||||
isset($config['TwoOnlySucceed']),
|
||||
'Fragment is included if both Only rules succeed.'
|
||||
);
|
||||
|
||||
$this->assertTrue(
|
||||
isset($config['TwoExceptSucceed']),
|
||||
'Fragment is included if one of the Except rules matches.'
|
||||
);
|
||||
|
||||
$this->assertFalse(
|
||||
isset($config['TwoExceptFail']),
|
||||
'Fragment is not included if both of the Except rules fail.'
|
||||
);
|
||||
|
||||
$this->assertFalse(
|
||||
isset($config['TwoBlocksFail']),
|
||||
'Fragment is not included if one block fails.'
|
||||
);
|
||||
|
||||
$this->assertTrue(
|
||||
isset($config['TwoBlocksSucceed']),
|
||||
'Fragment is included if both blocks succeed.'
|
||||
);
|
||||
}
|
||||
|
||||
public function testRelativeOrder() {
|
||||
$accessor = new ConfigManifestTest_ConfigManifestAccess(BASE_PATH, true, false);
|
||||
|
||||
|
@ -170,7 +170,8 @@ DOC;
|
||||
return;
|
||||
}
|
||||
|
||||
$parser = new SS_ConfigStaticManifest_Parser(__DIR__ . '/ConfigStaticManifestTest/ConfigStaticManifestTestMyObject.php');
|
||||
$parser = new SS_ConfigStaticManifest_Parser(__DIR__ .
|
||||
'/ConfigStaticManifestTest/ConfigStaticManifestTestMyObject.php');
|
||||
$parser->parse();
|
||||
|
||||
$statics = $parser->getStatics();
|
||||
@ -182,4 +183,19 @@ DOC;
|
||||
|
||||
$this->assertEquals($expectedValue, $statics['ConfigStaticManifestTestMyObject']['db']['value']);
|
||||
}
|
||||
|
||||
public function testParsingNamespacesclass() {
|
||||
$parser = new SS_ConfigStaticManifest_Parser(__DIR__ .
|
||||
'/ConfigStaticManifestTest/ConfigStaticManifestTestNamespace.php');
|
||||
$parser->parse();
|
||||
|
||||
$statics = $parser->getStatics();
|
||||
|
||||
$expectedValue = array(
|
||||
'Name' => 'Varchar',
|
||||
'Description' => 'Text',
|
||||
);
|
||||
|
||||
$this->assertEquals($expectedValue, $statics['config\staticmanifest\NamespaceTest']['db']['value']);
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,10 @@
|
||||
<?php
|
||||
|
||||
namespace config\staticmanifest;
|
||||
|
||||
class NamespaceTest implements \TestOnly {
|
||||
static private $db = array(
|
||||
'Name' => 'Varchar',
|
||||
'Description' => 'Text',
|
||||
);
|
||||
}
|
@ -0,0 +1,28 @@
|
||||
---
|
||||
Only:
|
||||
ClassExists: Director
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Class:
|
||||
DirectorExists: Yes
|
||||
---
|
||||
Only:
|
||||
ClassExists: NoSuchClass
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Class:
|
||||
NoSuchClassExists: Yes
|
||||
---
|
||||
Except:
|
||||
ClassExists: Director
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Class:
|
||||
DirectorExists: No
|
||||
---
|
||||
Except:
|
||||
ClassExists: NoSuchClass
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Class:
|
||||
NoSuchClassExists: No
|
@ -0,0 +1,28 @@
|
||||
---
|
||||
Only:
|
||||
ConstantDefined: ConstantDefined_Foo
|
||||
---
|
||||
ConfigManifestTest:
|
||||
ConstantDefined:
|
||||
FooDefined: Yes
|
||||
---
|
||||
Only:
|
||||
ConstantDefined: ConstantDefined_Bar
|
||||
---
|
||||
ConfigManifestTest:
|
||||
ConstantDefined:
|
||||
BarDefined: Yes
|
||||
---
|
||||
Except:
|
||||
ConstantDefined: ConstantDefined_Foo
|
||||
---
|
||||
ConfigManifestTest:
|
||||
ConstantDefined:
|
||||
FooDefined: No
|
||||
---
|
||||
Except:
|
||||
ConstantDefined: ConstantDefined_Bar
|
||||
---
|
||||
ConfigManifestTest:
|
||||
ConstantDefined:
|
||||
BarDefined: No
|
@ -0,0 +1,42 @@
|
||||
---
|
||||
Only:
|
||||
Environment: live
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Environment:
|
||||
LiveEnvironment: live
|
||||
---
|
||||
Only:
|
||||
Environment: dev
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Environment:
|
||||
DevEnvironment: dev
|
||||
---
|
||||
Only:
|
||||
Environment: test
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Environment:
|
||||
TestEnvironment: test
|
||||
---
|
||||
Except:
|
||||
Environment: live
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Environment:
|
||||
LiveEnvironment: notlive
|
||||
---
|
||||
Except:
|
||||
Environment: dev
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Environment:
|
||||
DevEnvironment: notdev
|
||||
---
|
||||
Except:
|
||||
Environment: test
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Environment:
|
||||
TestEnvironment: nottest
|
@ -0,0 +1,28 @@
|
||||
---
|
||||
Only:
|
||||
EnvVarSet: EnvVarSet_Foo
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvVarSet:
|
||||
FooSet: Yes
|
||||
---
|
||||
Only:
|
||||
EnvVarSet: EnvVarSet_Bar
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvVarSet:
|
||||
BarSet: Yes
|
||||
---
|
||||
Except:
|
||||
EnvVarSet: EnvVarSet_Foo
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvVarSet:
|
||||
FooSet: No
|
||||
---
|
||||
Except:
|
||||
EnvVarSet: EnvVarSet_Bar
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvVarSet:
|
||||
BarSet: No
|
@ -0,0 +1,28 @@
|
||||
---
|
||||
Only:
|
||||
ModuleExists: mysite
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Module:
|
||||
MysiteExists: Yes
|
||||
---
|
||||
Only:
|
||||
ModuleExists: nosuchmodule
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Module:
|
||||
NoSuchModuleExists: Yes
|
||||
---
|
||||
Except:
|
||||
ModuleExists: mysite
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Module:
|
||||
MysiteExists: No
|
||||
---
|
||||
Except:
|
||||
ModuleExists: nosuchmodule
|
||||
---
|
||||
ConfigManifestTest:
|
||||
Module:
|
||||
NoSuchModuleExists: No
|
@ -0,0 +1,50 @@
|
||||
---
|
||||
Only:
|
||||
ConstantDefined: MultilpleRules_UndefinedConstant
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
---
|
||||
ConfigManifestTest:
|
||||
MultipleRules:
|
||||
TwoOnlyFail: "not included - one of the onlies fails"
|
||||
---
|
||||
Only:
|
||||
ConstantDefined: MultilpleRules_DefinedConstant
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
---
|
||||
ConfigManifestTest:
|
||||
MultipleRules:
|
||||
TwoOnlySucceed: "included - both onlies succeed"
|
||||
---
|
||||
Except:
|
||||
ConstantDefined: MultilpleRules_UndefinedConstant
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
---
|
||||
ConfigManifestTest:
|
||||
MultipleRules:
|
||||
TwoExceptSucceed: "included - one of the excepts succeeds"
|
||||
---
|
||||
Except:
|
||||
ConstantDefined: MultilpleRules_DefinedConstant
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
---
|
||||
ConfigManifestTest:
|
||||
MultipleRules:
|
||||
TwoExceptFail: "not included - both excepts fail"
|
||||
---
|
||||
Except:
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
Only:
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
---
|
||||
ConfigManifestTest:
|
||||
MultipleRules:
|
||||
TwoBlocksFail: "not included - one block fails"
|
||||
---
|
||||
Except:
|
||||
ConstantDefined: MultilpleRules_UndefinedConstant
|
||||
Only:
|
||||
EnvVarSet: MultilpleRules_EnvVariableSet
|
||||
---
|
||||
ConfigManifestTest:
|
||||
MultipleRules:
|
||||
TwoBlocksSucceed: "included - both blocks succeed"
|
@ -0,0 +1,70 @@
|
||||
---
|
||||
Only:
|
||||
EnvOrConstantMatchesValue_Foo: Foo
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
FooIsFoo: Yes
|
||||
---
|
||||
Only:
|
||||
EnvOrConstantMatchesValue_Foo: Qux
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
FooIsQux: Yes
|
||||
---
|
||||
Only:
|
||||
EnvOrConstantMatchesValue_Bar: Bar
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
BarIsBar: Yes
|
||||
---
|
||||
Only:
|
||||
EnvOrConstantMatchesValue_Bar: Qux
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
BarIsQux: Yes
|
||||
---
|
||||
Only:
|
||||
EnvOrConstantMatchesValue_Baz: Baz
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
BazIsBaz: Yes
|
||||
---
|
||||
Except:
|
||||
EnvOrConstantMatchesValue_Foo: Foo
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
FooIsFoo: No
|
||||
---
|
||||
Except:
|
||||
EnvOrConstantMatchesValue_Foo: Qux
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
FooIsQux: No
|
||||
---
|
||||
Except:
|
||||
EnvOrConstantMatchesValue_Bar: Bar
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
BarIsBar: No
|
||||
---
|
||||
Except:
|
||||
EnvOrConstantMatchesValue_Bar: Qux
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
BarIsQux: No
|
||||
---
|
||||
Except:
|
||||
EnvOrConstantMatchesValue_Baz: Baz
|
||||
---
|
||||
ConfigManifestTest:
|
||||
EnvOrConstantMatchesValue:
|
||||
BazIsBaz: No
|
@ -0,0 +1,4 @@
|
||||
<?php
|
||||
|
||||
// Dynamically change environment
|
||||
Config::inst()->update('Director', 'environment_type', 'dev');
|
@ -0,0 +1,18 @@
|
||||
---
|
||||
Only:
|
||||
Environment: live
|
||||
---
|
||||
ConfigManifestTest:
|
||||
DynamicEnvironment: live
|
||||
---
|
||||
Only:
|
||||
Environment: dev
|
||||
---
|
||||
ConfigManifestTest:
|
||||
DynamicEnvironment: dev
|
||||
---
|
||||
Only:
|
||||
Environment: test
|
||||
---
|
||||
ConfigManifestTest:
|
||||
DynamicEnvironment: test
|
@ -15,6 +15,26 @@ class ConfirmedPasswordFieldTest extends SapphireTest {
|
||||
$this->assertEquals('valueB', $field->children->fieldByName($field->getName() . '[_ConfirmPassword]')->Value());
|
||||
}
|
||||
|
||||
public function testHashHidden() {
|
||||
$field = new ConfirmedPasswordField('Password', 'Password', 'valueA');
|
||||
$field->setCanBeEmpty(true);
|
||||
|
||||
$this->assertEquals('valueA', $field->Value());
|
||||
$this->assertEquals('valueA', $field->children->fieldByName($field->getName() . '[_Password]')->Value());
|
||||
$this->assertEquals('valueA', $field->children->fieldByName($field->getName() . '[_ConfirmPassword]')->Value());
|
||||
|
||||
$member = new Member();
|
||||
$member->Password = "valueB";
|
||||
$member->write();
|
||||
|
||||
$form = new Form($this, 'Form', new FieldList($field), new FieldList());
|
||||
$form->loadDataFrom($member);
|
||||
|
||||
$this->assertEquals('', $field->Value());
|
||||
$this->assertEquals('', $field->children->fieldByName($field->getName() . '[_Password]')->Value());
|
||||
$this->assertEquals('', $field->children->fieldByName($field->getName() . '[_ConfirmPassword]')->Value());
|
||||
}
|
||||
|
||||
public function testSetShowOnClick() {
|
||||
//hide by default and display show/hide toggle button
|
||||
$field = new ConfirmedPasswordField('Test', 'Testing', 'valueA', null, true);
|
||||
|
@ -73,6 +73,8 @@ class EmailFieldTest_Validator extends Validator {
|
||||
|
||||
class EmailFieldTest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
private static $url_handlers = array(
|
||||
'$Action//$ID/$OtherID' => "handleAction",
|
||||
);
|
||||
|
@ -479,6 +479,9 @@ class FormTest_Team extends DataObject implements TestOnly {
|
||||
* @subpackage tests
|
||||
*/
|
||||
class FormTest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
private static $url_handlers = array(
|
||||
'$Action//$ID/$OtherID' => "handleAction",
|
||||
);
|
||||
@ -528,6 +531,9 @@ class FormTest_Controller extends Controller implements TestOnly {
|
||||
* @subpackage tests
|
||||
*/
|
||||
class FormTest_ControllerWithSecurityToken extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
private static $url_handlers = array(
|
||||
'$Action//$ID/$OtherID' => "handleAction",
|
||||
);
|
||||
@ -562,6 +568,9 @@ class FormTest_ControllerWithSecurityToken extends Controller implements TestOnl
|
||||
}
|
||||
|
||||
class FormTest_ControllerWithStrictPostCheck extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
protected $template = 'BlankPage';
|
||||
|
||||
public function Link($action = null) {
|
||||
|
57
tests/forms/HtmlEditorSanitiserTest.php
Normal file
57
tests/forms/HtmlEditorSanitiserTest.php
Normal file
@ -0,0 +1,57 @@
|
||||
<?php
|
||||
/**
|
||||
* @package framework
|
||||
* @subpackage tests
|
||||
*/
|
||||
class HtmlEditorSanitiserTest extends FunctionalTest {
|
||||
|
||||
public function testSanitisation() {
|
||||
$tests = array(
|
||||
array(
|
||||
'p,strong',
|
||||
'<p>Leave Alone</p><div>Strip parent<strong>But keep children</strong> in order</div>',
|
||||
'<p>Leave Alone</p>Strip parent<strong>But keep children</strong> in order',
|
||||
'Non-whitelisted elements are stripped, but children are kept'
|
||||
),
|
||||
array(
|
||||
'p,strong',
|
||||
'<div>A <strong>B <div>Nested elements are still filtered</div> C</strong> D</div>',
|
||||
'A <strong>B Nested elements are still filtered C</strong> D',
|
||||
'Non-whitelisted elements are stripped even when children of non-whitelisted elements'
|
||||
),
|
||||
array(
|
||||
'p',
|
||||
'<p>Keep</p><script>Strip <strong>including children</strong></script>',
|
||||
'<p>Keep</p>',
|
||||
'Non-whitelisted script elements are totally stripped, including any children'
|
||||
),
|
||||
array(
|
||||
'p[id]',
|
||||
'<p id="keep" bad="strip">Test</p>',
|
||||
'<p id="keep">Test</p>',
|
||||
'Non-whitelisted attributes are stripped'
|
||||
),
|
||||
array(
|
||||
'p[default1=default1|default2=default2|force1:force1|force2:force2]',
|
||||
'<p default1="specific1" force1="specific1">Test</p>',
|
||||
'<p default1="specific1" force1="force1" default2="default2" force2="force2">Test</p>',
|
||||
'Default attributes are set when not present in input, forced attributes are always set'
|
||||
)
|
||||
);
|
||||
|
||||
$config = HtmlEditorConfig::get('htmleditorsanitisertest');
|
||||
|
||||
foreach($tests as $test) {
|
||||
list($validElements, $input, $output, $desc) = $test;
|
||||
|
||||
$config->setOptions(array('valid_elements' => $validElements));
|
||||
$sanitiser = new HtmlEditorSanitiser($config);
|
||||
|
||||
$htmlValue = Injector::inst()->create('HTMLValue', $input);
|
||||
$sanitiser->sanitise($htmlValue);
|
||||
|
||||
$this->assertEquals($output, $htmlValue->getContent(), $desc);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -97,6 +97,8 @@ class GridFieldAddExistingAutocompleterTest extends FunctionalTest {
|
||||
|
||||
class GridFieldAddExistingAutocompleterTest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
protected $template = 'BlankPage';
|
||||
|
||||
public function Form() {
|
||||
|
@ -282,6 +282,7 @@ class GridFieldDetailFormTest_PeopleGroup extends DataObject implements TestOnly
|
||||
}
|
||||
|
||||
class GridFieldDetailFormTest_Category extends DataObject implements TestOnly {
|
||||
|
||||
private static $db = array(
|
||||
'Name' => 'Varchar'
|
||||
);
|
||||
@ -306,6 +307,9 @@ class GridFieldDetailFormTest_Category extends DataObject implements TestOnly {
|
||||
}
|
||||
|
||||
class GridFieldDetailFormTest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
protected $template = 'BlankPage';
|
||||
|
||||
public function Form() {
|
||||
@ -326,6 +330,9 @@ class GridFieldDetailFormTest_Controller extends Controller implements TestOnly
|
||||
}
|
||||
|
||||
class GridFieldDetailFormTest_GroupController extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
protected $template = 'BlankPage';
|
||||
|
||||
public function Form() {
|
||||
@ -339,6 +346,9 @@ class GridFieldDetailFormTest_GroupController extends Controller implements Test
|
||||
}
|
||||
|
||||
class GridFieldDetailFormTest_CategoryController extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
protected $template = 'BlankPage';
|
||||
|
||||
public function Form() {
|
||||
|
@ -30,6 +30,9 @@ class GridField_URLHandlerTest extends FunctionalTest {
|
||||
}
|
||||
|
||||
class GridField_URLHandlerTest_Controller extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('Form');
|
||||
|
||||
public function Link() {
|
||||
return get_class($this) ."/";
|
||||
}
|
||||
@ -51,6 +54,9 @@ class GridField_URLHandlerTest_Controller extends Controller implements TestOnly
|
||||
* Test URLHandler with a nested request handler
|
||||
*/
|
||||
class GridField_URLHandlerTest_Component extends RequestHandler implements GridField_URLHandler {
|
||||
|
||||
private static $allowed_actions = array('Form', 'showform', 'testpage', 'handleItem');
|
||||
|
||||
protected $gridField;
|
||||
|
||||
public function getURLHandlers($gridField) {
|
||||
@ -96,8 +102,13 @@ class GridField_URLHandlerTest_Component extends RequestHandler implements GridF
|
||||
}
|
||||
|
||||
class GridField_URLHandlerTest_Component_ItemRequest extends RequestHandler {
|
||||
|
||||
private static $allowed_actions = array('Form', 'showform', 'testpage');
|
||||
|
||||
protected $gridField;
|
||||
|
||||
protected $link;
|
||||
|
||||
protected $id;
|
||||
|
||||
public function __construct($gridField, $id, $link) {
|
||||
|
@ -22,6 +22,46 @@ class DataListTest extends SapphireTest {
|
||||
'DataObjectTest\NamespacedClass',
|
||||
);
|
||||
|
||||
public function testFilterDataObjectByCreatedDate() {
|
||||
// create an object to test with
|
||||
$obj1 = new DataObjectTest_ValidatedObject();
|
||||
$obj1->Name = 'test obj 1';
|
||||
$obj1->write();
|
||||
$this->assertTrue($obj1->isInDB());
|
||||
|
||||
// reload the object from the database and reset its Created timestamp to a known value
|
||||
$obj1 = DataObjectTest_ValidatedObject::get()->filter(array('Name' => 'test obj 1'))->first();
|
||||
$this->assertTrue(is_object($obj1));
|
||||
$this->assertEquals('test obj 1', $obj1->Name);
|
||||
$obj1->Created = '2013-01-01 00:00:00';
|
||||
$obj1->write();
|
||||
|
||||
// reload the object again and make sure that our Created date was properly persisted
|
||||
$obj1 = DataObjectTest_ValidatedObject::get()->filter(array('Name' => 'test obj 1'))->first();
|
||||
$this->assertTrue(is_object($obj1));
|
||||
$this->assertEquals('test obj 1', $obj1->Name);
|
||||
$this->assertEquals('2013-01-01 00:00:00', $obj1->Created);
|
||||
|
||||
// now save a second object to the DB with an automatically-set Created value
|
||||
$obj2 = new DataObjectTest_ValidatedObject();
|
||||
$obj2->Name = 'test obj 2';
|
||||
$obj2->write();
|
||||
$this->assertTrue($obj2->isInDB());
|
||||
|
||||
// and a third object
|
||||
$obj3 = new DataObjectTest_ValidatedObject();
|
||||
$obj3->Name = 'test obj 3';
|
||||
$obj3->write();
|
||||
$this->assertTrue($obj3->isInDB());
|
||||
|
||||
// now test the filtering based on Created timestamp
|
||||
$list = DataObjectTest_ValidatedObject::get()
|
||||
->filter(array('Created:GreaterThan' => '2013-02-01 00:00:00'))
|
||||
->toArray();
|
||||
$this->assertEquals(2, count($list));
|
||||
|
||||
}
|
||||
|
||||
public function testSubtract(){
|
||||
$comment1 = $this->objFromFixture('DataObjectTest_TeamComment', 'comment1');
|
||||
$subtractList = DataObjectTest_TeamComment::get()->filter('ID', $comment1->ID);
|
||||
|
@ -19,6 +19,19 @@ class DataObjectTest extends SapphireTest {
|
||||
'DataObjectTest_TeamComment'
|
||||
);
|
||||
|
||||
public function testValidObjectsForBaseFields() {
|
||||
$obj = new DataObjectTest_ValidatedObject();
|
||||
|
||||
foreach (array('Created', 'LastEdited', 'ClassName', 'ID') as $field) {
|
||||
$helper = $obj->dbObject($field);
|
||||
$this->assertTrue(
|
||||
($helper instanceof DBField),
|
||||
"for {$field} expected helper to be DBField, but was " .
|
||||
(is_object($helper) ? get_class($helper) : "null")
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
public function testDataIntegrityWhenTwoSubclassesHaveSameField() {
|
||||
// Save data into DataObjectTest_SubTeam.SubclassDatabaseField
|
||||
$obj = new DataObjectTest_SubTeam();
|
||||
|
@ -44,11 +44,11 @@ class GroupTest extends FunctionalTest {
|
||||
$form->saveInto($member);
|
||||
$updatedGroups = $member->Groups();
|
||||
|
||||
$this->assertEquals(
|
||||
array($adminGroup->ID, $parentGroup->ID),
|
||||
$updatedGroups->column(),
|
||||
$this->assertEquals(2, count($updatedGroups->column()),
|
||||
"Adding a toplevel group works"
|
||||
);
|
||||
$this->assertContains($adminGroup->ID, $updatedGroups->column('ID'));
|
||||
$this->assertContains($parentGroup->ID, $updatedGroups->column('ID'));
|
||||
|
||||
// Test unsetting relationship
|
||||
$form->loadDataFrom($member);
|
||||
@ -60,11 +60,10 @@ class GroupTest extends FunctionalTest {
|
||||
$form->saveInto($member);
|
||||
$member->flushCache();
|
||||
$updatedGroups = $member->Groups();
|
||||
$this->assertEquals(
|
||||
array($adminGroup->ID),
|
||||
$updatedGroups->column(),
|
||||
$this->assertEquals(1, count($updatedGroups->column()),
|
||||
"Removing a previously added toplevel group works"
|
||||
);
|
||||
$this->assertContains($adminGroup->ID, $updatedGroups->column('ID'));
|
||||
|
||||
// Test adding child group
|
||||
|
||||
@ -77,23 +76,21 @@ class GroupTest extends FunctionalTest {
|
||||
$orphanGroup->ParentID = 99999;
|
||||
$orphanGroup->write();
|
||||
|
||||
$this->assertEquals(
|
||||
array($parentGroup->ID),
|
||||
$parentGroup->collateAncestorIDs(),
|
||||
$this->assertEquals(1, count($parentGroup->collateAncestorIDs()),
|
||||
'Root node only contains itself'
|
||||
);
|
||||
$this->assertContains($parentGroup->ID, $parentGroup->collateAncestorIDs());
|
||||
|
||||
$this->assertEquals(
|
||||
array($childGroup->ID, $parentGroup->ID),
|
||||
$childGroup->collateAncestorIDs(),
|
||||
$this->assertEquals(2, count($childGroup->collateAncestorIDs()),
|
||||
'Contains parent nodes, with child node first'
|
||||
);
|
||||
$this->assertContains($parentGroup->ID, $childGroup->collateAncestorIDs());
|
||||
$this->assertContains($childGroup->ID, $childGroup->collateAncestorIDs());
|
||||
|
||||
$this->assertEquals(
|
||||
array($orphanGroup->ID),
|
||||
$orphanGroup->collateAncestorIDs(),
|
||||
$this->assertEquals(1, count($orphanGroup->collateAncestorIDs()),
|
||||
'Orphaned nodes dont contain invalid parent IDs'
|
||||
);
|
||||
$this->assertContains($orphanGroup->ID, $orphanGroup->collateAncestorIDs());
|
||||
}
|
||||
|
||||
public function testDelete() {
|
||||
|
@ -41,8 +41,11 @@ class MemberCsvBulkLoaderTest extends SapphireTest {
|
||||
$results = $loader->load($this->getCurrentRelativePath() . '/MemberCsvBulkLoaderTest.csv');
|
||||
|
||||
$created = $results->Created()->toArray();
|
||||
$this->assertEquals($created[0]->Groups()->column('ID'), array($existinggroup->ID));
|
||||
$this->assertEquals($created[1]->Groups()->column('ID'), array($existinggroup->ID));
|
||||
$this->assertEquals(1, count($created[0]->Groups()->column('ID')));
|
||||
$this->assertContains($existinggroup->ID, $created[0]->Groups()->column('ID'));
|
||||
|
||||
$this->assertEquals(1, count($created[1]->Groups()->column('ID')));
|
||||
$this->assertContains($existinggroup->ID, $created[1]->Groups()->column('ID'));
|
||||
}
|
||||
|
||||
public function testAddToCsvColumnGroupsByCode() {
|
||||
@ -55,8 +58,12 @@ class MemberCsvBulkLoaderTest extends SapphireTest {
|
||||
$this->assertEquals($newgroup->Title, 'newgroup');
|
||||
|
||||
$created = $results->Created()->toArray();
|
||||
$this->assertEquals($created[0]->Groups()->column('ID'), array($existinggroup->ID));
|
||||
$this->assertEquals($created[1]->Groups()->column('ID'), array($existinggroup->ID, $newgroup->ID));
|
||||
$this->assertEquals(1, count($created[0]->Groups()->column('ID')));
|
||||
$this->assertContains($existinggroup->ID, $created[0]->Groups()->column('ID'));
|
||||
|
||||
$this->assertEquals(2, count($created[1]->Groups()->column('ID')));
|
||||
$this->assertContains($existinggroup->ID, $created[1]->Groups()->column('ID'));
|
||||
$this->assertContains($newgroup->ID, $created[1]->Groups()->column('ID'));
|
||||
}
|
||||
|
||||
public function testCleartextPasswordsAreHashedWithDefaultAlgo() {
|
||||
|
@ -439,6 +439,9 @@ class SecurityTest extends FunctionalTest {
|
||||
}
|
||||
|
||||
class SecurityTest_SecuredController extends Controller implements TestOnly {
|
||||
|
||||
private static $allowed_actions = array('index');
|
||||
|
||||
public function index() {
|
||||
if(!Permission::check('ADMIN')) return Security::permissionFailure($this);
|
||||
|
||||
|
@ -165,6 +165,18 @@ SS;
|
||||
'Permissions template functions result correct result');
|
||||
}
|
||||
|
||||
public function testNonFieldCastingHelpersNotUsedInHasValue() {
|
||||
// check if Link without $ in front of variable
|
||||
$result = $this->render(
|
||||
'A<% if Link %>$Link<% end_if %>B', new SSViewerTest_Object());
|
||||
$this->assertEquals('Asome/url.htmlB', $result, 'casting helper not used for <% if Link %>');
|
||||
|
||||
// check if Link with $ in front of variable
|
||||
$result = $this->render(
|
||||
'A<% if $Link %>$Link<% end_if %>B', new SSViewerTest_Object());
|
||||
$this->assertEquals('Asome/url.htmlB', $result, 'casting helper not used for <% if $Link %>');
|
||||
}
|
||||
|
||||
public function testLocalFunctionsTakePriorityOverGlobals() {
|
||||
$data = new ArrayData(array(
|
||||
'Page' => new SSViewerTest_Object()
|
||||
@ -1078,7 +1090,8 @@ after')
|
||||
array(
|
||||
'name' => 'SSViewerTestCommentsFullSourceHTML4Doctype',
|
||||
'expected' => ""
|
||||
. "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01//EN\"\t\t\"http://www.w3.org/TR/html4/strict.dtd\">"
|
||||
. "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML "
|
||||
. "4.01//EN\"\t\t\"http://www.w3.org/TR/html4/strict.dtd\">"
|
||||
. "<!-- template $f/SSViewerTestCommentsFullSourceHTML4Doctype.ss -->"
|
||||
. "<html>"
|
||||
. "\t<head></head>"
|
||||
@ -1197,6 +1210,45 @@ after')
|
||||
"tests/forms/RequirementsTest_a.js"
|
||||
));
|
||||
}
|
||||
|
||||
public function testCallsWithArguments() {
|
||||
$data = new ArrayData(array(
|
||||
'Set' => new ArrayList(array(
|
||||
new SSViewerTest_Object("1"),
|
||||
new SSViewerTest_Object("2"),
|
||||
new SSViewerTest_Object("3"),
|
||||
new SSViewerTest_Object("4"),
|
||||
new SSViewerTest_Object("5"),
|
||||
)),
|
||||
'Level' => new SSViewerTest_LevelTest(1),
|
||||
'Nest' => array(
|
||||
'Level' => new SSViewerTest_LevelTest(2),
|
||||
),
|
||||
));
|
||||
|
||||
$tests = array(
|
||||
'$Level.output(1)' => '1-1',
|
||||
'$Nest.Level.output($Set.First.Number)' => '2-1',
|
||||
'<% with $Set %>$Up.Level.output($First.Number)<% end_with %>' => '1-1',
|
||||
'<% with $Set %>$Top.Nest.Level.output($First.Number)<% end_with %>' => '2-1',
|
||||
'<% loop $Set %>$Up.Nest.Level.output($Number)<% end_loop %>' => '2-12-22-32-42-5',
|
||||
'<% loop $Set %>$Top.Level.output($Number)<% end_loop %>' => '1-11-21-31-41-5',
|
||||
'<% with $Nest %>$Level.output($Top.Set.First.Number)<% end_with %>' => '2-1',
|
||||
'<% with $Level %>$output($Up.Set.Last.Number)<% end_with %>' => '1-5',
|
||||
'<% with $Level.forWith($Set.Last.Number) %>$output("hi")<% end_with %>' => '5-hi',
|
||||
'<% loop $Level.forLoop($Set.First.Number) %>$Number<% end_loop %>' => '!0',
|
||||
'<% with $Nest %>
|
||||
<% with $Level.forWith($Up.Set.First.Number) %>$output("hi")<% end_with %>
|
||||
<% end_with %>' => '1-hi',
|
||||
'<% with $Nest %>
|
||||
<% loop $Level.forLoop($Top.Set.Last.Number) %>$Number<% end_loop %>
|
||||
<% end_with %>' => '!0!1!2!3!4',
|
||||
);
|
||||
|
||||
foreach($tests as $template => $expected) {
|
||||
$this->assertEquals($expected, trim($this->render($template, $data)));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@ -1274,6 +1326,11 @@ class SSViewerTest_Object extends DataObject {
|
||||
|
||||
public $number = null;
|
||||
|
||||
private static $casting = array(
|
||||
'Link' => 'Text',
|
||||
);
|
||||
|
||||
|
||||
public function __construct($number = null) {
|
||||
parent::__construct();
|
||||
$this->number = $number;
|
||||
@ -1290,6 +1347,10 @@ class SSViewerTest_Object extends DataObject {
|
||||
public function lotsOfArguments11($a, $b, $c, $d, $e, $f, $g, $h, $i, $j, $k) {
|
||||
return $a. $b. $c. $d. $e. $f. $g. $h. $i. $j. $k;
|
||||
}
|
||||
|
||||
public function Link() {
|
||||
return 'some/url.html';
|
||||
}
|
||||
}
|
||||
|
||||
class SSViewerTest_GlobalProvider implements TemplateGlobalProvider, TestOnly {
|
||||
@ -1326,3 +1387,28 @@ class SSViewerTest_GlobalProvider implements TemplateGlobalProvider, TestOnly {
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
class SSViewerTest_LevelTest extends ViewableData implements TestOnly {
|
||||
protected $depth;
|
||||
|
||||
public function __construct($depth = 1) {
|
||||
$this->depth = $depth;
|
||||
}
|
||||
|
||||
public function output($val) {
|
||||
return "$this->depth-$val";
|
||||
}
|
||||
|
||||
public function forLoop($number) {
|
||||
$ret = array();
|
||||
for($i = 0; $i < (int)$number; ++$i) {
|
||||
$ret[] = new SSViewerTest_Object("!$i");
|
||||
}
|
||||
return new ArrayList($ret);
|
||||
}
|
||||
|
||||
public function forWith($number) {
|
||||
return new self($number);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -81,6 +81,19 @@ class ViewableDataTest extends SapphireTest {
|
||||
$this->assertEquals('casted', $newViewableData->forTemplate());
|
||||
}
|
||||
|
||||
public function testDefaultValueWrapping() {
|
||||
$data = new ArrayData(array('Title' => 'SomeTitleValue'));
|
||||
// this results in a cached raw string in ViewableData:
|
||||
$this->assertTrue($data->hasValue('Title'));
|
||||
$this->assertFalse($data->hasValue('SomethingElse'));
|
||||
// this should cast the raw string to a StringField since we are
|
||||
// passing true as the third argument:
|
||||
$obj = $data->obj('Title', null, true);
|
||||
$this->assertTrue(is_object($obj));
|
||||
// and the string field should have the value of the raw string:
|
||||
$this->assertEquals('SomeTitleValue', $obj->forTemplate());
|
||||
}
|
||||
|
||||
public function testRAWVal() {
|
||||
$data = new ViewableDataTest_Castable();
|
||||
$data->test = 'This & This';
|
||||
@ -121,6 +134,28 @@ class ViewableDataTest extends SapphireTest {
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
public function testObjWithCachedStringValueReturnsValidObject() {
|
||||
$obj = new ViewableDataTest_NoCastingInformation();
|
||||
|
||||
// Save a literal string into cache
|
||||
$cache = true;
|
||||
$uncastedData = $obj->obj('noCastingInformation', null, false, $cache);
|
||||
|
||||
// Fetch the cached string as an object
|
||||
$forceReturnedObject = true;
|
||||
$castedData = $obj->obj('noCastingInformation', null, $forceReturnedObject);
|
||||
|
||||
// Uncasted data should always be the nonempty string
|
||||
$this->assertNotEmpty($uncastedData, 'Uncasted data was empty.');
|
||||
$this->assertTrue(is_string($uncastedData), 'Uncasted data should be a string.');
|
||||
|
||||
// Casted data should be the string wrapped in a DBField-object.
|
||||
$this->assertNotEmpty($castedData, 'Casted data was empty.');
|
||||
$this->assertInstanceOf('DBField', $castedData, 'Casted data should be instance of DBField.');
|
||||
|
||||
$this->assertEquals($uncastedData, $castedData->getValue(), 'Casted and uncasted strings are not equal.');
|
||||
}
|
||||
}
|
||||
|
||||
/**#@+
|
||||
@ -212,4 +247,10 @@ class ViewableDataTest_CastingClass extends ViewableData {
|
||||
);
|
||||
}
|
||||
|
||||
class ViewableDataTest_NoCastingInformation extends ViewableData {
|
||||
public function noCastingInformation() {
|
||||
return "No casting information";
|
||||
}
|
||||
}
|
||||
|
||||
/**#@-*/
|
||||
|
@ -615,7 +615,7 @@ class SSTemplateParser extends Parser {
|
||||
|
||||
|
||||
function Lookup__construct(&$res) {
|
||||
$res['php'] = '$scope';
|
||||
$res['php'] = '$scope->locally()';
|
||||
$res['LookupSteps'] = array();
|
||||
}
|
||||
|
||||
|
@ -157,7 +157,7 @@ class SSTemplateParser extends Parser {
|
||||
*/
|
||||
|
||||
function Lookup__construct(&$res) {
|
||||
$res['php'] = '$scope';
|
||||
$res['php'] = '$scope->locally()';
|
||||
$res['LookupSteps'] = array();
|
||||
}
|
||||
|
||||
|
@ -49,7 +49,8 @@ class SSViewer_Scope {
|
||||
|
||||
public function __construct($item){
|
||||
$this->item = $item;
|
||||
$this->localIndex=0;
|
||||
$this->localIndex = 0;
|
||||
$this->localStack = array();
|
||||
$this->itemStack[] = array($this->item, null, 0, null, null, 0);
|
||||
}
|
||||
|
||||
@ -57,10 +58,25 @@ class SSViewer_Scope {
|
||||
return $this->itemIterator ? $this->itemIterator->current() : $this->item;
|
||||
}
|
||||
|
||||
public function resetLocalScope(){
|
||||
/** Called at the start of every lookup chain by SSTemplateParser to indicate a new lookup from local scope */
|
||||
public function locally() {
|
||||
list($this->item, $this->itemIterator, $this->itemIteratorTotal, $this->popIndex, $this->upIndex,
|
||||
$this->currentIndex) = $this->itemStack[$this->localIndex];
|
||||
array_splice($this->itemStack, $this->localIndex+1);
|
||||
|
||||
// Remember any un-completed (resetLocalScope hasn't been called) lookup chain. Even if there isn't an
|
||||
// un-completed chain we need to store an empty item, as resetLocalScope doesn't know the difference later
|
||||
$this->localStack[] = array_splice($this->itemStack, $this->localIndex+1);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function resetLocalScope(){
|
||||
$previousLocalState = $this->localStack ? array_pop($this->localStack) : null;
|
||||
|
||||
array_splice($this->itemStack, $this->localIndex+1, count($this->itemStack), $previousLocalState);
|
||||
|
||||
list($this->item, $this->itemIterator, $this->itemIteratorTotal, $this->popIndex, $this->upIndex,
|
||||
$this->currentIndex) = end($this->itemStack);
|
||||
}
|
||||
|
||||
public function getObj($name, $arguments = null, $forceReturnedObject = true, $cache = false, $cacheName = null) {
|
||||
|
@ -383,7 +383,9 @@ class ViewableData extends Object implements IteratorAggregate {
|
||||
|
||||
if(!is_object($value) && $forceReturnedObject) {
|
||||
$default = Config::inst()->get('ViewableData', 'default_cast', Config::FIRST_SET);
|
||||
$value = new $default($fieldName);
|
||||
$castedValue = new $default($fieldName);
|
||||
$castedValue->setValue($value);
|
||||
$value = $castedValue;
|
||||
}
|
||||
|
||||
return $value;
|
||||
|
Loading…
Reference in New Issue
Block a user